νμ€ν μΉπ κ°λ°μ μ§λ§μ π§π½βπ»
β μΈκ³΅μ§λ₯ κ΄μ¬ π€
Categories
-
β£
βΆ COMPUTER_SCIENCE
π: 7 -
β£
βΆ WEB
π: 3 -
β£
βΆ ETC
π: 3-
β
β£
ETCS
π: 10 -
β
β£
SUBBRAIN κ°λ°κΈ°
π: 5 -
β
β
YOS κ°λ°κΈ°
π: 1
-
β
β£
-
β
βΆ AI
π: 9-
β£
AITOOLS
π: 3 -
β£
CV
π: 2 -
β£
DEEP_LEARNING
π: 1 -
β£
DATA_VIS
π: 2 -
β£
GRAPH
π: 1 -
β£
LIGHTWEIGHT
π: 1 -
β£
MATH
π: 1 -
β£
NLP
π: 3 -
β
STRUCTURED_DATA
π: 2
-
β£
Spring5-ν΅ν©
- μ€νλ§ ν΅ν©(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("classpath:/filewriter-config.xml")
public class FileWriterIntegrationConfig{...}
- μλ° νμΌ κ΅¬μ±
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)
- κΈ°ν λ±λ±
_articles/web/backend/Spring/Spring5-ν΅ν©.md