How can @MessagingGateway be configured with Spring Cloud Stream MessageChannels?

前端 未结 3 962
野趣味
野趣味 2020-12-19 12:50

I have developed asynchronous Spring Cloud Stream services, and I am trying to develop an edge service that uses @MessagingGateway to provide synchronous access to services

相关标签:
3条回答
  • 2020-12-19 13:10

    With Artem's help, I've found the solution I was looking for. I have taken the code Artem posted and split it into two services, a Gateway service and a CloudStream service. I also added a @RestController for testing purposes. This essentially mimics what I was wanting to do with durable queues. Thanks Artem for your assistance! I really appreciate your time! I hope this helps others who want to do the same thing.

    Gateway Code

    package com.example.demo;
    
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    import org.springframework.cloud.stream.annotation.EnableBinding;
    import org.springframework.cloud.stream.annotation.Input;
    import org.springframework.cloud.stream.annotation.Output;
    
    import org.springframework.context.annotation.Bean;
    import org.springframework.http.HttpStatus;
    import org.springframework.http.MediaType;
    import org.springframework.http.ResponseEntity;
    import org.springframework.integration.annotation.Gateway;
    import org.springframework.integration.annotation.MessagingGateway;
    import org.springframework.integration.dsl.HeaderEnricherSpec;
    import org.springframework.integration.dsl.IntegrationFlow;
    import org.springframework.integration.dsl.IntegrationFlows;
    import org.springframework.messaging.MessageChannel;
    import org.springframework.messaging.SubscribableChannel;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.PathVariable;
    import org.springframework.web.bind.annotation.RestController;
    
    @EnableBinding({GatewayApplication.GatewayChannels.class})
    @SpringBootApplication
    public class GatewayApplication {
    
      interface GatewayChannels {
    
        String TO_UPPERCASE_REPLY = "to-uppercase-reply";
        String TO_UPPERCASE_REQUEST = "to-uppercase-request";
    
        @Input(TO_UPPERCASE_REPLY)
        SubscribableChannel toUppercaseReply();
    
        @Output(TO_UPPERCASE_REQUEST)
        MessageChannel toUppercaseRequest();
      }
    
      @MessagingGateway
      public interface StreamGateway {
        @Gateway(requestChannel = ENRICH, replyChannel = GatewayChannels.TO_UPPERCASE_REPLY)
        String process(String payload);
      }
    
      private static final String ENRICH = "enrich";
    
      public static void main(String[] args) {
        SpringApplication.run(GatewayApplication.class, args);
      }
    
      @Bean
      public IntegrationFlow headerEnricherFlow() {
        return IntegrationFlows.from(ENRICH).enrichHeaders(HeaderEnricherSpec::headerChannelsToString)
            .channel(GatewayChannels.TO_UPPERCASE_REQUEST).get();
      }
    
      @RestController
      public class UppercaseController {
        @Autowired
        StreamGateway gateway;
    
        @GetMapping(value = "/string/{string}",
            produces = {MediaType.APPLICATION_JSON_VALUE, MediaType.APPLICATION_XML_VALUE})
        public ResponseEntity<String> getUser(@PathVariable("string") String string) {
          return new ResponseEntity<String>(gateway.process(string), HttpStatus.OK);
        }
      }
    
    }
    

    Gateway Config (application.yml)

    spring:
      cloud:
        stream:
          bindings:
            to-uppercase-request:
              destination: to-uppercase-request
              producer:
                required-groups: stream-to-uppercase-request
            to-uppercase-reply:
              destination: to-uppercase-reply
              group: gateway-to-uppercase-reply
    server:
      port: 8080
    

    CloudStream Code

    package com.example.demo;
    
    import org.springframework.boot.SpringApplication;
    import org.springframework.boot.autoconfigure.SpringBootApplication;
    import org.springframework.cloud.stream.annotation.EnableBinding;
    import org.springframework.cloud.stream.annotation.Input;
    import org.springframework.cloud.stream.annotation.Output;
    import org.springframework.cloud.stream.annotation.StreamListener;
    import org.springframework.messaging.Message;
    import org.springframework.messaging.MessageChannel;
    import org.springframework.messaging.SubscribableChannel;
    import org.springframework.messaging.handler.annotation.SendTo;
    import org.springframework.messaging.support.MessageBuilder;
    
    @EnableBinding({CloudStreamApplication.CloudStreamChannels.class})
    @SpringBootApplication
    public class CloudStreamApplication {
    
      interface CloudStreamChannels {
    
        String TO_UPPERCASE_REPLY = "to-uppercase-reply";
        String TO_UPPERCASE_REQUEST = "to-uppercase-request";
    
        @Output(TO_UPPERCASE_REPLY)
        SubscribableChannel toUppercaseReply();
    
        @Input(TO_UPPERCASE_REQUEST)
        MessageChannel toUppercaseRequest();
      }
    
      public static void main(String[] args) {
        SpringApplication.run(CloudStreamApplication.class, args);
      }
    
      @StreamListener(CloudStreamChannels.TO_UPPERCASE_REQUEST)
      @SendTo(CloudStreamChannels.TO_UPPERCASE_REPLY)
      public Message<?> process(Message<String> request) {
        return MessageBuilder.withPayload(request.getPayload().toUpperCase())
            .copyHeaders(request.getHeaders()).build();
      }
    
    }
    

    CloudStream Config (application.yml)

    spring:
      cloud:
        stream:
          bindings:
            to-uppercase-request:
              destination: to-uppercase-request
              group: stream-to-uppercase-request
            to-uppercase-reply:
              destination: to-uppercase-reply
              producer:
                required-groups: gateway-to-uppercase-reply
    server:
      port: 8081
    
    0 讨论(0)
  • 2020-12-19 13:16

    It's good question and really good idea. But it isn't going to work so easy.

    First of all we have to determine for ourselves that gateway means request/reply, therefore correlation. And this available in @MessagingGateway via replyChannel header in face of TemporaryReplyChannel instance. Even if you have an explicit replyChannel = AccountChannels.ACCOUNT_CREATED, the correlation is done only via the mentioned header and its value. The fact that this TemporaryReplyChannel is not serializable and can't be transferred over the network to the consumer on another side.

    Luckily Spring Integration provide some solution for us. It is a part of the HeaderEnricher and its headerChannelsToString option behind HeaderChannelRegistry:

    Starting with Spring Integration 3.0, a new sub-element <int:header-channels-to-string/> is available; it has no attributes. This converts existing replyChannel and errorChannel headers (when they are a MessageChannel) to a String and stores the channel(s) in a registry for later resolution when it is time to send a reply, or handle an error. This is useful for cases where the headers might be lost; for example when serializing a message into a message store or when transporting the message over JMS. If the header does not already exist, or it is not a MessageChannel, no changes are made.

    https://docs.spring.io/spring-integration/docs/5.0.0.RELEASE/reference/html/messaging-transformation-chapter.html#header-enricher

    But in this case you have to introduce an internal channel from the gateway to the HeaderEnricher and only the last one will send the message to the AccountChannels.CREATE_ACCOUNT_REQUEST. So, the replyChannel header will be converted to a string representation and be able to travel over the network. On the consumer side when you send a reply you should ensure that you transfer that replyChannel header as well, as it is. So, when the message will arrive to the AccountChannels.ACCOUNT_CREATED on the producer side, where we have that @MessagingGateway, the correlation mechanism is able to convert a channel identificator to the proper TemporaryReplyChannel and correlate the reply to the waiting gateway call.

    Only the problem here that your producer application must be as single consumer in the group for the AccountChannels.ACCOUNT_CREATED - we have to ensure that only one instance in the cloud is operating at a time. Just because only one instance has that TemporaryReplyChannel in its memory.

    More info about gateway: https://docs.spring.io/spring-integration/docs/5.0.0.RELEASE/reference/html/messaging-endpoints-chapter.html#gateway

    UPDATE

    Some code for help:

    @EnableBinding(AccountChannels.class)
    @MessagingGateway
    
    public interface AccountService {
      @Gateway(requestChannel = AccountChannels.INTERNAL_CREATE_ACCOUNT_REQUEST, replyChannel = AccountChannels.ACCOUNT_CREATED, replyTimeout = 60000, requestTimeout = 60000)
      Account createAccount(@Payload Account account, @Header("Authorization") String authorization);
    }
    
    @Bean
    public IntegrationFlow headerEnricherFlow() {
       return IntegrationFlows.from(AccountChannels.INTERNAL_CREATE_ACCOUNT_REQUEST)
                .enrichHeaders(headerEnricher -> headerEnricher.headerChannelsToString())
                .channel(AccountChannels.CREATE_ACCOUNT_REQUEST)
                .get();
    
    }
    

    UPDATE

    Some simple application to demonstrate the PoC:

    @EnableBinding({ Processor.class, CloudStreamGatewayApplication.GatewayChannels.class })
    @SpringBootApplication
    public class CloudStreamGatewayApplication {
    
        interface GatewayChannels {
    
            String REQUEST = "request";
    
            @Output(REQUEST)
            MessageChannel request();
    
    
            String REPLY = "reply";
    
            @Input(REPLY)
            SubscribableChannel reply();
        }
    
        private static final String ENRICH = "enrich";
    
    
        @MessagingGateway
        public interface StreamGateway {
    
            @Gateway(requestChannel = ENRICH, replyChannel = GatewayChannels.REPLY)
            String process(String payload);
    
        }
    
        @Bean
        public IntegrationFlow headerEnricherFlow() {
            return IntegrationFlows.from(ENRICH)
                    .enrichHeaders(HeaderEnricherSpec::headerChannelsToString)
                    .channel(GatewayChannels.REQUEST)
                    .get();
        }
    
        @StreamListener(Processor.INPUT)
        @SendTo(Processor.OUTPUT)
        public Message<?> process(Message<String> request) {
            return MessageBuilder.withPayload(request.getPayload().toUpperCase())
                    .copyHeaders(request.getHeaders())
                    .build();
        }
    
    
        public static void main(String[] args) {
            ConfigurableApplicationContext applicationContext =
                    SpringApplication.run(CloudStreamGatewayApplication.class, args);
    
            StreamGateway gateway = applicationContext.getBean(StreamGateway.class);
    
            String result = gateway.process("foo");
    
            System.out.println(result);
        }
    
    }
    

    The application.yml:

    spring:
      cloud:
        stream:
          bindings:
            input:
              destination: requests
            output:
              destination: replies
            request:
              destination: requests
            reply:
              destination: replies
    

    I use spring-cloud-starter-stream-rabbit.

    The

    MessageBuilder.withPayload(request.getPayload().toUpperCase())
                .copyHeaders(request.getHeaders())
                .build()
    

    Does the trick copying request headers to the reply message. So, the gateway is able on the reply side to convert channel identifier in the headers to the appropriate TemporaryReplyChannel to convey the reply properly to the caller of gateway.

    The SCSt issue on the matter: https://github.com/spring-cloud/spring-cloud-stream/issues/815

    0 讨论(0)
  • 2020-12-19 13:29

    Hmm, I am a bit confused as well as to what you are trying to accomplish, but let's se if we can figure this out. Mixing SI and SCSt is definitely natural as one builds on another so all should work: Here is an example code snippet I just dug up from an old sample that exposes REST endpoint yet delegates (via Gateway) to Source's output channel. See if that helps:

    @EnableBinding(Source.class)
    @SpringBootApplication
    @RestController
    public class FooApplication {
        . . . 
    
        @Autowired
        private Source channels;
    
        @Autowired
        private CompletionService completionService;
    
        @RequestMapping("/complete")
        public String completeRequest(@RequestParam int id) {
            this.completionService.complete("foo");
            return "OK";
        }
    
        @MessagingGateway
        interface CompletionService {
            @Gateway(requestChannel = Source.OUTPUT)
            void complete(String message);
        }
    }
    
    0 讨论(0)
提交回复
热议问题