ν’€μŠ€νƒ μ›ΉπŸŒ 개발자 지망생 πŸ§‘πŸ½β€πŸ’»
βž• 인곡지λŠ₯ 관심 πŸ€–


Categories


Recent views

  • 1
  • 2
  • 3
  • 4
  • 5

Spring5-톡합

  1. μŠ€ν”„λ§ 톡합(Spring Integration)
    • μ—¬λŸ¬ 톡합 ν”Œλ‘œμš° ꡬ성 방법
      • 톡합 ν”Œλ‘œμš° μ»΄ν¬λ„ŒνŠΈμ™€ μ˜ˆμ‹œ

      Spring5-톡합

      πŸ—£οΈ 좜처

      _ 초보 μ›Ή 개발자λ₯Ό μœ„ν•œ μŠ€ν”„λ§ 5 ν”„λ‘œκ·Έλž˜λ° μž…λ¬Έ _와 μŠ€ν”„λ§ 인 μ•‘μ…˜ 의 λ‚΄μš©μ„ λ°”νƒ•μœΌλ‘œ μ •λ¦¬ν•œ λ‚΄μš©μž…λ‹ˆλ‹€.

      ⚠️ μŠ€ν”„λ§ λΆ€νŠΈλ₯Ό μ‚¬μš©ν•œλ‹€λŠ” κ°€μ • ν•˜μ— μ§„ν–‰λ©λ‹ˆλ‹€.

      μŠ€ν”„λ§ 톡합(Spring Integration)

      μ–΄ν”Œλ¦¬μΌ€μ΄μ…˜μ΄ 이메일, 파일 μ‹œμŠ€ν…œ, RSS, Twitter λ“±μ˜ μ™ΈλΆ€ μ„œλΉ„μŠ€μ™€ μƒν˜Έμž‘μš© μ‹œ, 데이터λ₯Ό 읽고 μ“°κ³ , ν˜•νƒœλ₯Ό λ³€κ²½ν•  λ•Œ μ‚¬μš©ν•˜λŠ” νŒ¨ν„΄μ΄λ‹€.

      각 톡합 νŒ¨ν„΄μ€ ν•˜λ‚˜μ˜ μ»΄ν¬λ„ŒνŠΈλ‘œ μ •μ˜λ˜λ©°, 톡합 νŒ¨ν„΄μ˜ νŒŒμ΄ν”„λΌμΈμ„ λ”°λ₯΄λŠ” λ©”μ‹œμ§€λ₯Ό 톡해 데이터λ₯Ό μš΄λ°˜ν•˜λ©° κ°€κ³΅ν•œλ‹€.

      μŠ€ν”„λ§ λΆ€νŠΈλ‘œ 톡합 ν”Œλ‘œμš°λ₯Ό μ„ μ–Έν•˜κΈ° μœ„ν•΄ λ‹€μŒκ³Ό 같은 μ˜μ‘΄μ„±μ΄ ν•„μš”ν•˜λ‹€.
      org.springframework.boot.spring-boot-starter-integration : 톡합에 ν•„μš”ν•œ ν•„μˆ˜ λͺ¨λ“ˆ
      org.springframework.integration.spring-integration-{μ›ν•˜λŠ” μ—”λ“œν¬μΈνŠΈ λͺ¨λ“ˆ} : κ΅¬ν˜„ν•˜λ €λŠ” μ‹œμŠ€ν…œμ„ μœ„ν•œ λͺ¨λ“ˆ

      μ—¬λŸ¬ 톡합 ν”Œλ‘œμš° ꡬ성 방법

      톡합 ν”Œλ‘œμš°λŠ” μ—¬λŸ¬ λ°©λ²•μœΌλ‘œ ꡬ성할 수 있으며, μ˜ˆμ‹œλ₯Ό μ•Œμ•„λ³΄μž.
      λ¨Όμ €, 톡합 ν”Œλ‘œμš°λ‘œ 데이터λ₯Ό μ „μ†‘ν•˜λŠ” 파일 μ“°κΈ° κ²Œμ΄νŠΈμ›¨μ΄λ₯Ό μž‘μ„±ν•œλ‹€.

      🧾️ FileWriteGateway.java

      이제 톡합 ν”Œλ‘œμš° ꡬ성을 μ„ μ–Έν•˜λ©΄ μ•„λž˜ κ²Œμ΄νŠΈμ›¨μ΄λ₯Ό μ‚¬μš©ν•  수 μžˆλ‹€.

      package sia5;
      import org.springframework.integration.annotation.MessagingGateway;
      import org.springframework.integration.file.FileHeaders;
      import org.springframework.messaging.handler.annotation.Header;
      
      @MessagingGateway(defaultRequestChannel="textInChannel")
      public interface FileWriterGateway {
        void writeToFile(
            @Header(FileHeaders.FILENAME) String filename,
            String data);
      }
      

      μžμ„Έν•œ μ„€λͺ…은 λ‚˜μ€‘μ— DSL둜 μ•Œμ•„ λ³Ό 것이닀.

      • XML ꡬ성
      🧾️ filewriter.xml

      사싀, XML ꡬ성은 과거에 λΉ„ν•΄ 잘 μ‚¬μš©λ˜μ§€ μ•ŠλŠ”λ‹€.

      <?xml version="1.0" encoding="UTF-8"?>
      <beans xmlns="http://www.springframework.org/schema/beans"
        xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
        xmlns:int="http://www.springframework.org/schema/integration"
        xmlns:int-file="http://www.springframework.org/schema/integration/file"
        xsi:schemaLocation="http://www.springframework.org/schema/beans
          http://www.springframework.org/schema/beans/spring-beans.xsd
          http://www.springframework.org/schema/integration
          http://www.springframework.org/schema/integration/spring-integration.xsd
          http://www.springframework.org/schema/integration/file
          http://www.springframework.org/schema/integration/file/spring-integration-file.xsd">
          <int:channel id="textInChannel" />
          <int:transformer id="upperCase"
              input-channel="textInChannel"
              output-channel="fileWriterChannel"
              expression="payload.toUpperCase()" />
          <int:channel id="fileWriterChannel" />
          <int-file:outbound-channel-adapter id="writer"
              channel="fileWriterChannel"
              directory="/tmp/sia5/files"
              mode="APPEND"
              append-new-line="true" />
      </beans>
      

      이후, 아무 ꡬ성 ν΄λž˜μŠ€μ— @ImportResource μ–΄λ…Έν…Œμ΄μ…˜μœΌλ‘œ 뢈러였기 섀정을 ν•˜μž.

      @Configuration
      @ImportResource(&#34;classpath:/filewriter-config.xml&#34;)
      public class FileWriterIntegrationConfig&#123;...&#125;
      
      
      • μžλ°” 파일 ꡬ성
      🧾️ FileWriterIntegrationConfig.java

      μžμ„Έν•œ μ„€λͺ…은 μ»΄ν¬λ„ŒνŠΈμ—μ„œ μ„€λͺ…

      package sia5;
      import java.io.File;
      //...
      import org.springframework.integration.annotation.ServiceActivator;
      import org.springframework.integration.annotation.Transformer;
      import org.springframework.integration.dsl.IntegrationFlow;
      import org.springframework.integration.dsl.IntegrationFlows;
      import org.springframework.integration.dsl.channel.MessageChannels;
      import org.springframework.integration.file.FileWritingMessageHandler;
      import org.springframework.integration.file.dsl.Files;
      import org.springframework.integration.file.support.FileExistsMode;
      import org.springframework.integration.transformer.GenericTransformer;
      
      @Configuration
      public class FileWriterIntegrationConfig {  
        /*
        // μ•žμ„  xml κ΅¬μ„±μš© κ΅¬μ„±νŒŒμΌ
        @Profile("xmlconfig")
        @Configuration
        @ImportResource("classpath:/filewriter-config.xml")
        public static class XmlConfiguration {}
        */
        // 채널 이동간에 λ³€ν™˜
        @Profile("javaconfig")
        @Bean
        @Transformer(inputChannel="textInChannel",
                     outputChannel="fileWriterChannel")
        public GenericTransformer<String, String> upperCaseTransformer() {
          return text -> text.toUpperCase();
        }
        
        // fileWriterChannel둜 λΆ€ν„° λ©”μ‹œμ§€λ₯Ό 받은 λ’€μ˜ μ„œλΉ„μŠ€
        @Profile("javaconfig")
        @Bean
        @ServiceActivator(inputChannel="fileWriterChannel")
        public FileWritingMessageHandler fileWriter() {
          FileWritingMessageHandler handler =
              new FileWritingMessageHandler(new File("/tmp/sia5/files"));
          handler.setExpectReply(false); //λ‹¨λ°˜ν–₯ κ²Œμ΄νŠΈμ›¨μ΄λ‘œ μ„€μ •
          handler.setFileExistsMode(FileExistsMode.APPEND);
          handler.setAppendNewLine(true);
          return handler; 
        }
        /*
        // νŠΉμ • 채널이름이 μ‘΄μž¬ν•˜μ§€ μ•ŠμœΌλ©΄ μžλ™μœΌλ‘œ κ΅¬μ„±ν•˜μ§€λ§Œ, 직접 μƒμ„±ν•˜κ³  μ„€μ •ν•˜κ³  μ‹Άλ‹€λ©΄ μ•„λž˜μ™€ 같이 κ°€λŠ₯
        @Bean
        public MessageChannel textInChannel() {
          return new DirectChannel();
        }
        
        @Bean
        public MessageChannel fileWriterChannel() {
          return new DirectChannel();
        }  
        */
      }
      
      • DSL(Domain Specific Language)
      🧾️ μ•žμ„  μžλ°” ꡬ성 νŒŒμΌμ— μΆ”κ°€

      λͺ¨λ‘ 같은 κΈ°λŠ₯, 더 κ°„μ†Œν™” 되고 μ•Œμ•„λ³΄κΈ° 쉽닀. 전체 ν”Œλ‘œμš°κ°€ ν•˜λ‚˜μ˜ 빈으둜 ꡬ성됨

      import java.io.File;  
      
      import org.springframework.context.annotation.Bean;
      import org.springframework.context.annotation.Configuration;
      import org.springframework.integration.dsl.IntegrationFlow;
      import org.springframework.integration.dsl.IntegrationFlows;
      import org.springframework.integration.dsl.channel.MessageChannels;
      import org.springframework.integration.file.dsl.Files;
      import org.springframework.integration.file.support.FileExistsMode;
      
      @Configuration
      public class FileWriterIntegrationConfig {
      
        @Profile("javadsl")
        @Bean
        public IntegrationFlow fileWriterFlow() {
          return IntegrationFlows
              .from(MessageChannels.direct("textInChannel"))
              .<String, String>transform(t -> t.toUpperCase())
              //.channel(MessageChannels.direct("fileWriterChannel")) // 채널 μˆ˜λ™ ꡬ성
              .handle(Files
                  .outboundAdapter(new File("/tmp/sia5/files"))
                  .fileExistsMode(FileExistsMode.APPEND)
                  .appendNewLine(true))
              .get();
        }
      }
      

      톡합 ν”Œλ‘œμš° μ»΄ν¬λ„ŒνŠΈμ™€ μ˜ˆμ‹œ

      보톡, κ²Œμ΄νŠΈμ›¨μ΄-채널-λ³€ν™˜κΈ°-채널-μ–΄λŒ‘ν„° 순으둜 μ—°κ²°ν•œλ‹€.

      채널(Channel)

      ν•œ μš”μ†Œλ‘œλΆ€ν„° λ‹€λ₯Έ μš”μ†Œλ‘œ λ©”μ‹œμ§€ 전달
      κΈ°λŠ₯에 따라 λ‹€μŒκ³Ό 같이 μ—¬λŸ¬ 채널이 μ‚¬μš© κ°€λŠ₯

      • PublishSubscribeChannel : ν•˜λ‚˜ μ΄μƒμ˜ 컨슈머 λͺ¨λ‘μ—κ²Œ 전달
      • QueueChannel : FIFO λ°©μ‹μœΌλ‘œ μ»¨μŠˆλ¨Έκ°€ ν•˜λ‚˜μ”© 가져감
      • PriorityChannel : priority 헀더 기반으둜 μ»¨μŠˆλ¨Έκ°€ 가져감
      • RendezvousChannel: μ»¨μŠˆλ¨Έκ°€ λ©”μ‹œμ§€ μˆ˜μ‹ μ‹œκΉŒμ§€ μ „μ†‘μžκ°€ 채널 μ‚¬μš© λΆˆκ°€(동기화)
      • DirectChannel: 단일 μ‚¬μš©μžλ§Œ μ‚¬μš©κ°€λŠ₯ν•œ PublishSubscribeChannel, νŠΈλžœμž­μ…˜ 지원
      • ExecutorChannel: TaskExecutor둜 λ©”μ‹œμ§€ 전솑, νŠΈλžœμž­μ…˜ 지원 μ•ˆν•¨
      • FluxMessageChannel: 리앑터 ν”ŒλŸ­μŠ€ 기반 채널

      기본적으둜 νŠΉμ • μ΄λ¦„μ˜ 채널이 μ‘΄μž¬ν•˜μ§€ μ•ŠμœΌλ©΄ DirectChannel둜 μƒμ„±λ˜λ©°, λ‹€λ₯Έ 채널을 κ΅¬ν˜„ν•˜κ³  μ‹Άλ‹€λ©΄ 직접 채널을 κ΅¬ν˜„ν•΄μ•Ό ν•œλ‹€.

      🧾️ QueueChannel μ„ μ–Έκ³Ό μ‚¬μš©
      @Bean
      public MessageChannel orderChannel() {
      	return new QueueChannel();
      }
      
      @ServiceActivator(inputChannel="orderChannel" poller=@Poller(fixedRate="1000"))
      // 채널λͺ… 지λͺ… + queueμ—μ„œ κ°€μ Έμ˜€λŠ” μ£ΌκΈ° μ„€μ •
      

      ν•„ν„°(Filter)

      쑰건에 λ§žλŠ” λ©”μ‹œμ§€λ§Œ ν”Œλ‘œμš°λ₯Ό ν†΅κ³Όν•˜κ²Œ 함
      톡합 νŒŒμ΄ν”„λΌμΈ 채널 사이 μ€‘κ°„μ—λ§Œ μœ„μΉ˜ κ°€λŠ₯

      🧾️ ν•„ν„°μ˜ μ˜ˆμ‹œ
      @Filter(inputChannel="numberChannel",
      	   outputChannel="evenNumberChannel")
      public boolean evenNumberFilter(Integer number) {
      	return number % 2 == 0;
      }
      
      //DSL 버전
      @Bean
      public IntegrationFlow evenNumberFlow(AtomicInteger integerSource) {
      	return IntegrationFlows
      	//...
      	.<Integer>filter((p)->p%2==0) // GenericSelector ν˜Ήμ€ λžŒλ‹€λ‘œ ν•„ν„° κ΅¬ν˜„
      	//...
      	.get();
      }
      

      λ³€ν™˜κΈ°(Transformer)

      λ©”μ‹œμ§€ 값을 λ³€κ²½ν•˜κ±°λ‚˜ λ©”μ‹œμ§€ νŽ˜μ΄λ‘œλ“œμ˜ νƒ€μž…μ„ λ‹€λ₯Έ νƒ€μž…μœΌλ‘œ λ³€κ²½

      🧾️ GenericTransformer κ΅¬ν˜„ μ˜ˆμ‹œ
      @Bean
      @Transformer(inputChannel="numberChannel", outputChannel="romanNumberChannel")
      public GenericTransformer<Integer, String> romanNumTransformer() {
      	return RomanNumbers::toRoman;
      }
      
      @Bean
      public IntegrationFlow transformerFlow() {
      	return IntegrationFlows
      	//...
      	.transform(RomanNumbers::toRoman)
      	//...
      	.get();
      }
      

      λΌμš°ν„°(Router)

      μ—¬λŸ¬ 채널 쀑 ν•˜λ‚˜λ‘œ λ©”μ‹œμ§€λ₯Ό μ „λ‹¬ν•˜λ©°, λŒ€κ²Œ λ©”μ‹œμ§€ 헀더λ₯Ό 기반으둜 함

      μ΄λ•Œ 전달 쑰건을 직접 κ΅¬ν˜„ν•΄μ£Όμ–΄μ•Ό ν•œλ‹€.

      🧾️ λΌμš°ν„° κ΅¬ν˜„ μ˜ˆμ‹œ
      @Bean
      @Router(inputChannel="numberChannel")
      public AbstractMessageRouter evenOddRouter() {
      	return new AbstractMessageRouter() {
      		@Override
      		protected Collection<MessageChannel> determineTargetChannels(Message<?> message) {
      			Integer number = (Integer) message.getPayload();
      			if (number % 2 == 0) {
      				// channel을 μˆ˜λ™ μ„€μ •ν•΄μ€˜μ•Ό 함
      				return Collections.singleton(evenChannel());
      			}
      			return Collections.singleton(oddChannel());
      		}
      	};
      }
      // DSL ꡬ성
      @Bean
      public IntegrationFlow numberRoutingFlow(AtomicInteger source) {
      	return IntegrationFlows
      		//...
      		.<Integer, String>route(n -> n%2==0 ? "EVEN":"ODD", mapping -> mapping)
      		.subFlowMapping("Even", sf -> sf.<Integer, Integer>transform(n -> n * 10).handle((i, h) -> {...}))
      		.subFlowMapping("ODD", sf -> sf.transform(RomanNumbers::toRoman).handle((i,h)-> {...}))
      		.get();
      }
      

      λΆ„λ°°κΈ°(Splitter)

      λ“€μ–΄μ˜€λŠ” λ©”μ‹œμ§€λ₯Ό 두 개 μ΄μƒμ˜ λ©”μ‹œμ§€λ‘œ λΆ„ν• ν•˜λ©°, λΆ„ν• λœ 각 λ©”μ‹œμ§€λŠ” λ‹€λ₯Έ μ±„λ„λ‘œ μ „μ†‘λœλ‹€.

      크게 두가지 λ°©λ²•μœΌλ‘œ λΆ„ν• ν•œλ‹€.

      • 같은 νƒ€μž…μ˜ μ»¬λ ‰μ…˜ ν•­λͺ©λ“€μ„ μ—¬λŸ¬ λ©”μ‹œμ§€ νŽ˜μ΄λ‘œλ“œλ‘œ λ‚˜λˆ”
      • ν•˜λ‚˜μ˜ λ©”μ‹œμ§€ νŽ˜μ΄λ‘œλ“œλ₯Ό μ„œλ‘œ λ‹€λ₯Έ νƒ€μž… λ‘κ°œ 이상 λ©”μ‹œμ§€λ‘œ λ‚˜λˆ”
      🧾️ λΆ„λ°°κΈ° μ˜ˆμ‹œ
      public class OrderSplitter {
      	// splitOrderIntoParts λ©”μ„œλ“œ κ΅¬ν˜„
      	public Collection<Object> splitOrderIntoParts(PurchaseOrder po) {
      		ArrayList<Object> parts = new ArrayList<>();
      		parts.add(po.getBillingInfo());
      		parts.add(po.getLineItems());
      		return parts
      	}
      }
      
      @Bean
      @Splitter(inputChannel="poChannel", outputChannel="splitOrderChannel")
      public OrderSplitter orderSplitter() {
      	return new OrderSpliter();
      }
      
      //μ•„λž˜μ™€ 같은 λΌμš°ν„°λ‘œ λ”°λ‘œ λ³΄λ‚΄μ€˜μ•Ό ν•œλ‹€.
      @Bean
      @Router(inputChannel="splitOrderChannel")
      public MessageRouter splitOrderRouter() {
      	PayloadTypeRouter router = new PayloadTypeRouter();
      	router.setChannelMapping(
      		BillingInfo.class.getName(), "billingInfoChannel");
      	router.setChannelMapping(
      		List.class.getName(), "lineItemsChannel");
      	return router;
      }
      
      // λ§Œμ•½ Splitterλ₯Ό μ—°μ†μœΌλ‘œ μ μš©ν•˜κ³  μ‹Άλ‹€λ©΄ λ‹€μŒκ³Ό 같이 빈 객체가 μ•„λ‹Œ μŠ€ν”Œλ¦¬ν„°λ₯Ό μ—°μ†μœΌλ‘œ μ§€μ •ν•˜λ©΄ λœλ‹€.
      @Splitter(inputChannel="lineItemsChannel", outputChannel="lineItemChannel")
      public List<LineItem> lineItemSplitter(List<LineItem> lineItems) {
      	return lineItems; // 각 μ•„μ΄ν…œλ“€μ΄ λ‚˜λ‰˜μ–΄ ν•œ 채널에 μ—¬λŸ¬κ°œλ‘œ λ“€μ–΄κ°„λ‹€.
      }
      
      // DSL 버전
      return IntegrationFlows
      	//...
      	.split(orderSpliter())
      	.<Object, String> route(
      		p -> {
      			if (p.getClass().isAssignableFrom(BillingInfo.class)){
      				return "BILLING_INFO";
      			} else {
      				return "LINE_ITEMS";
      			}
      		}, mapping -> mapping
      			.subFlowMapping("BILLING_INFO", sf -> sf.<BilingInfo> handle((billingInfo, h) -> {
      				//...
      			}))
      			.subFlowMapping("LINE_ITEMS",
      				sf -> sf.split()
      					.<LineItem> handle((lineItem, h) -> {
      						//...
      					}))
      			)
      		.get();
      
      

      집적기(Aggregator)

      뢄배기와 μƒλ°˜λœ κ²ƒμœΌλ‘œ λ³„κ°œμ˜ μ±„λ„λ‘œλΆ€ν„° μ „λ‹¬λ˜λŠ” λ‹€μˆ˜μ˜ λ©”μ‹œμ§€λ₯Ό ν•˜λ‚˜μ˜ λ©”μ‹œμ§€λ‘œ κ²°ν•©ν•œλ‹€.

      μ„œλΉ„μŠ€ 앑티베이터(Service activator)

      λ©”μ‹œμ§€λ₯Ό μ²˜λ¦¬ν•˜λ„λ‘ Handler κ΅¬ν˜„ λ©”μ„œλ“œμ— λ©”μ‹œμ§€λ₯Ό λ„˜κ²¨μ€€ ν›„ λ©”μ„œλ“œμ˜ λ°˜ν™˜κ°’μ„ 좜λ ₯ μ±„λ„λ‘œ μ „μ†‘ν•œλ‹€.

      MessageHandlerλ‚˜ GenericHandler μΈν„°νŽ˜μ΄μŠ€λ₯Ό κ΅¬ν˜„ν•œ λ©”μ„œλ“œλ₯Ό μ΄μš©ν•  수 μžˆλ‹€.

      • λ§Œμ•½, λ©”μ„œλ“œ λ°˜ν™˜κ°’μ΄ μ‘΄μž¬ν•˜μ§€ μ•Šλ‹€λ©΄ MessageHandler
      • λ©”μ„œλ“œ λ°˜ν™˜κ°’μœΌλ‘œ μƒˆλ‘œμš΄ λ©”μ‹œμ§€κ°€ μ‘΄μž¬ν•œλ‹€λ©΄ GenericHandlerλ₯Ό μ΄μš©ν•΄μ•Ό ν•œλ‹€.
        • 단, ν”Œλ‘œμš°μ˜ 끝에 두면 ν•΄λ‹Ή λ©”μ‹œμ§€λ₯Ό 받을 채널이 μ—†μ–΄ μ—λŸ¬ λ°œμƒ
        • ꡳ이 끝에 둬야 ν•œλ‹€λ©΄ return 값을 null둜 ν•΄μ•Ό ν•œλ‹€.
      🧾️ μ„œλΉ„μŠ€ 엑티베이터와 ν•Έλ“€λŸ¬ λ©”μ„œλ“œ μ˜ˆμ‹œ
      @Bean
      @ServiceActivator(inputChannel="someChannel")
      public MessageHandler sysoutHandler() {
      	return message -> {
      		System.out.println("Message payload: " + message.getPayload());
      	};
      }
      
      @Bean
      @ServiceActivator(inputChannel="orderChannel", outputChannel="completeChannel")
      public GenericHandler<Order> orderHandler(OrderRepository orderRepo) {
      	return (payload, headers) -> {
      		return orderRepo.save(payload);	
      	};
      }
      
      //DSL 버전
      public IntegrationFlow someFlow() {
      	return IntegrationFlows
      		//...
      			.handle(msg -> {
      				System.out.println("Message payload: " + msg.getPayload());
      			})
      			.get();
      }
      
      
      

      주둜 μ•„μ›ƒλ°”μš΄λ“œ 채널 μ–΄λŒ‘ν„°λ‘œ μ΄μš©λœλ‹€.

      채널 μ–΄λŒ‘ν„°(Channel adapter)

      μ™ΈλΆ€ μ‹œμŠ€ν…œμ— 채널을 μ—°κ²°ν•˜μ—¬ μž…μΆœλ ₯ κ°€λŠ₯μΌ€ 함, ν”Œλ‘œμš°μ˜ μž…κ΅¬μ™€ 좜ꡬ

      • μž…κ΅¬ 역할을 ν•˜λŠ” μΈλ°”μš΄λ“œ 채널 μ–΄λŒ‘ν„°
      🧾️ μΈλ°”μš΄λ“œ μ–΄λŒ‘ν„° μ˜ˆμ‹œ

      AtomicInteger둜 λΆ€ν„°μ˜ 값을 1μ΄ˆλ§ˆλ‹€ ν”Œλ‘œμš°λ‘œ μ „λ‹¬ν•˜λŠ” μ–΄λŒ‘ν„°

      @Bean
      @InboundChannelAdapter(poller=@Poller(fixedRate="1000"), channel="numberChannel")
      public MessageSource<Integer> numberSource(AtomicInteger source) {
      	return () -> {
      		return new GenericMessage<>(source.getAndIncrement());
      	};
      }
      
      @Bean
      public IntegrationFlow someFlow(AtomicInteger integerSource) {
      	return IntegrationFlows
      		.from(integerSource, "getAndIncrement",
      			c -> c.poller(Pollers.fixedRate(1000)))
      		//...
      		.get();
      }
      
      • 좜ꡬ 역할을 ν•˜λŠ” μ•„μ›ƒλ°”μš΄λ“œ 채널 μ–΄λŒ‘ν„° : κ²°κ³Όλ₯Ό λ°˜ν™˜ν•˜λŠ” μ„œλΉ„μŠ€ μ•‘ν‹°λ² μ΄ν„°λ‘œ 자주 κ΅¬ν˜„ν•œλ‹€.

      채널 μ–΄λŒ‘ν„°λŠ” μ—”λ“œ λͺ¨λ“ˆμ—μ„œ 직접 μ œκ³΅ν•˜κΈ°λ„ ν•œλ‹€.

      κ²Œμ΄νŠΈμ›¨μ΄(Gateway)

      μΈν„°νŽ˜μ΄μŠ€λ₯Ό 톡해 톡합 ν”Œλ‘œμš°λ‘œ 데이터 전달, μ„ νƒμ μœΌλ‘œ 응닡도 λ°›μŒ(μ–‘λ°©ν–₯ κ²Œμ΄νŠΈμ›¨μ΄)

      즉, μ–΄ν”Œλ¦¬μΌ€μ΄μ…˜μ΄ 톡합 ν”Œλ‘œμš°λ‘œ λ©”μ‹œμ§€λ₯Ό μ „μ†‘ν•˜κ³  λ°›μ•„λ‚Ό 수 μžˆλŠ” λ©”μ„œλ“œμ΄λ©°, μΈν„°νŽ˜μ΄μŠ€λ‘œ κ΅¬ν˜„λ˜μ–΄ μžˆλ‹€.

      🧾️ κ²Œμ΄νŠΈμ›¨μ΄ κ΅¬ν˜„ μ˜ˆμ‹œ
      package com.example.demo;
      
      import org.springframework.integration.annotation.MessagingGateway;
      import org.springframework.stereotype.Component;
      
      //μΈν„°νŽ˜μ΄μŠ€λ§Œ λ§Œλ“€λ©΄ μŠ€ν”„λ§μ΄ μžλ™μœΌλ‘œ κ΅¬ν˜„ν•˜λ―€λ‘œ uppercase λ©”μ„œλ“œλ₯Ό μ‚¬μš©ν•˜λ©΄ 됨.
      @Component
      @MessagingGateway(defaultRequestChannel="inChannel", defaultReplyChannel="outChannel")
      public interface UpperCaseGateway {
      	String uppercase(String in); // in λ¬Έμžμ—΄μ΄ ν”Œλ‘œμš°λ‘œ 전달됨, 
      	// 아웃 μ±„λ„λ‘œ λ„μ°©ν•˜λ©΄ 같은 λ¬Έμžμ—΄μ΄ λ°˜ν™˜λ¨
      	// uppercase ν•˜λŠ” λ©”μ„œλ“œλŠ” λ³€ν™˜κΈ°λ‘œ κ΅¬ν˜„
      }
      
      //dsl 버전
      @Bean
      public IntegrationFlow uppercaseFlow() {
      	return IntegrationFlows
      		.from("inChannel")
      		.<String, String> transform(s -> s.toUpperCase())
      		.channel("outChannel")
      		.get();
      }
      

      μ—”λ“œν¬μΈνŠΈ λͺ¨λ“ˆ(Endpoint module)

      λ‹€μ–‘ν•œ μ™ΈλΆ€ μ‹œμŠ€ν…œκ³Ό 톡합을 μœ„ν•΄ λ‹€μŒκ³Ό 같은 μ˜ˆμ‹œ μ—”λ“œν¬μΈνŠΈ λͺ¨λ“ˆμ΄ μ‘΄μž¬ν•œλ‹€.

      • AMQP(spring-integration-amqp)
      • RSS(spring-integration-feed)
      • 파일 μ‹œμŠ€ν…œ(spring-integration-file)
      • FTP(spring-integration-ftp)
      • 이메일(spring-integration-mail)
      • 슀트림(spring-integration-stream)
      • νŠΈμœ„ν„°(spring-integration-twitter)
      • WebSocket(spring-integration-websocket)
      • 기타 λ“±λ“±