在 Java Spring 集成中,transformer 元素是否可以包含路由器功能?
In Java Spring Integration, can a transformer element include router functionality?
我需要一个组件:
接收消息,对消息的有效负载和 header 进行转换(到目前为止它就像一个转换器)。
然后,根据 header 路由中传递的值到适当的通道(类似于 header-router)。
我想纯粹使用 java 中的注释而不是 XML 来配置它,但我绝对会在这方面尽我所能。如果有人知道 XML 解决方案,请传递。
如果有帮助,我的用例是我希望转换器对消息负载执行的转换依赖于自定义加载的 header 值。我还希望用于从转换器发送消息的通道取决于 header 值。
我知道涉及多个转换器的解决方案,一个在 header 值中定义的每个转换类型,然后是一个路由器。我试图避免这种解决方案,并且最多只使用一个路由器和一个变压器。
感谢您的帮助。
在 Spring 中,集成通道充当任何其他 bean。您可以使用服务激活器来调用任何 bean 上的方法。该 bean 可以注入所需的通道。您可以对 select 使用 @Qualifier
注释,应该注入哪个通道,或者只是自动装配一个 Map<String, MessageChannel>
将被通道的 bean 名称索引。它可以将转换后的消息传递到这些频道。
A Spring 引导应用程序:
package demo;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.ImportResource;
import org.springframework.integration.config.EnableIntegration;
@SpringBootApplication
@EnableIntegration
@ImportResource("classpath:int.xml")
public class DemoApplication {
public static void main(String[] args) {
SpringApplication.run(DemoApplication.class, args);
}
}
网关接口:
package demo;
public interface MyGateway {
public void send(Object o);
}
要调用的服务:
package demo;
import java.util.Map;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.integration.support.MessageBuilder;
import org.springframework.messaging.Message;
import org.springframework.messaging.MessageChannel;
import org.springframework.stereotype.Service;
@Service
public class MyService {
private final Map<String, MessageChannel> channels;
@Autowired
public MyService(Map<String, MessageChannel> channels) {
super();
this.channels = channels;
}
public void transformAndRoute(Message<?> in) {
// do your business logic here
Message<?> out = MessageBuilder.fromMessage(in).build();
// if(something)...
channels.get("fooChannel").send(out);
}
}
集成配置:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:int="http://www.springframework.org/schema/integration"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
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">
<int:gateway id="myGateway" service-interface="demo.MyGateway"
default-request-channel="inChannel" />
<int:service-activator input-channel="inChannel"
ref="myService" method="transformAndRoute" />
<int:channel id="inChannel" />
<int:logging-channel-adapter id="fooChannel" level="INFO" log-full-message="true"/>
</beans>
最后是一个简单的集成测试:
package demo;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes=DemoApplication.class)
public class MyGatewayIT {
@Autowired
MyGateway myGateway;
@Test
public void test() {
myGateway.send(new Object());
// do your assertions here
}
}
输出:
14:17:19.733 [main] DEBUG o.s.i.handler.LoggingHandler - org.springframework.integration.handler.LoggingHandler#0 received message: GenericMessage [payload=java.lang.Object@6f2cfcc2, headers={id=6791344c-07b4-d420-0d17-e2344f4bf15b, timestamp=1437826639733}]
但是
开发基于消息的系统的主要好处是,您可以使用易于测试、重用和更改的小型、松散耦合的组件来创建应用程序。创建一个在流程中扮演两个角色的组件打破了这个规则。此外,如果您按上述方式创建代码,则它确实与 Spring 集成相关联。您可以做的是创建几个组件,每个组件都有一个单一的职责,然后将它们配置为在链中运行。
您可以做的是创建一个转换器来修改消息的负载和 headers。该转换器将封装您的业务逻辑,并设置一个 header(例如,myRoutingHeader
),稍后可以在路由中使用。如果有一个用于业务逻辑的转换器和一个用于添加 header 的 header enricher 可能会更好。但是让我们假设您是在单个 class 中完成的。将 class 定义为一个 bean(比如 myTransformer
)。然后在你的配置中:
<int:channel id="inChannel/>
<!-- if performance is important consider using a SpEL expression to
invoke your method instead as they can be configured to be compiled -->
<int:transformer ref="myTransformer" input-channel="inChannel"
method="transform" output-channel="routingChannel"/>
<int:channel id="routingChannel/>
<int:header-value-router input-channel="routingChannel" header-name="myRoutingHeader">
<int:mapping value="foo" channel="fooChannel" />
<int:mapping value="bar" channel="barChannel" />
</int:header-value-router>
听起来链接可以为您做到这一点
<chain input-chanel="source" output-channel="routing-channel">
<transform/>
<!-- any enrichment process or intermediate process can go here -->
</chain>
我需要一个组件:
接收消息,对消息的有效负载和 header 进行转换(到目前为止它就像一个转换器)。
然后,根据 header 路由中传递的值到适当的通道(类似于 header-router)。
我想纯粹使用 java 中的注释而不是 XML 来配置它,但我绝对会在这方面尽我所能。如果有人知道 XML 解决方案,请传递。
如果有帮助,我的用例是我希望转换器对消息负载执行的转换依赖于自定义加载的 header 值。我还希望用于从转换器发送消息的通道取决于 header 值。
我知道涉及多个转换器的解决方案,一个在 header 值中定义的每个转换类型,然后是一个路由器。我试图避免这种解决方案,并且最多只使用一个路由器和一个变压器。
感谢您的帮助。
在 Spring 中,集成通道充当任何其他 bean。您可以使用服务激活器来调用任何 bean 上的方法。该 bean 可以注入所需的通道。您可以对 select 使用 @Qualifier
注释,应该注入哪个通道,或者只是自动装配一个 Map<String, MessageChannel>
将被通道的 bean 名称索引。它可以将转换后的消息传递到这些频道。
A Spring 引导应用程序:
package demo;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.annotation.ImportResource;
import org.springframework.integration.config.EnableIntegration;
@SpringBootApplication
@EnableIntegration
@ImportResource("classpath:int.xml")
public class DemoApplication {
public static void main(String[] args) {
SpringApplication.run(DemoApplication.class, args);
}
}
网关接口:
package demo;
public interface MyGateway {
public void send(Object o);
}
要调用的服务:
package demo;
import java.util.Map;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.integration.support.MessageBuilder;
import org.springframework.messaging.Message;
import org.springframework.messaging.MessageChannel;
import org.springframework.stereotype.Service;
@Service
public class MyService {
private final Map<String, MessageChannel> channels;
@Autowired
public MyService(Map<String, MessageChannel> channels) {
super();
this.channels = channels;
}
public void transformAndRoute(Message<?> in) {
// do your business logic here
Message<?> out = MessageBuilder.fromMessage(in).build();
// if(something)...
channels.get("fooChannel").send(out);
}
}
集成配置:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:int="http://www.springframework.org/schema/integration"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
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">
<int:gateway id="myGateway" service-interface="demo.MyGateway"
default-request-channel="inChannel" />
<int:service-activator input-channel="inChannel"
ref="myService" method="transformAndRoute" />
<int:channel id="inChannel" />
<int:logging-channel-adapter id="fooChannel" level="INFO" log-full-message="true"/>
</beans>
最后是一个简单的集成测试:
package demo;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes=DemoApplication.class)
public class MyGatewayIT {
@Autowired
MyGateway myGateway;
@Test
public void test() {
myGateway.send(new Object());
// do your assertions here
}
}
输出:
14:17:19.733 [main] DEBUG o.s.i.handler.LoggingHandler - org.springframework.integration.handler.LoggingHandler#0 received message: GenericMessage [payload=java.lang.Object@6f2cfcc2, headers={id=6791344c-07b4-d420-0d17-e2344f4bf15b, timestamp=1437826639733}]
但是
开发基于消息的系统的主要好处是,您可以使用易于测试、重用和更改的小型、松散耦合的组件来创建应用程序。创建一个在流程中扮演两个角色的组件打破了这个规则。此外,如果您按上述方式创建代码,则它确实与 Spring 集成相关联。您可以做的是创建几个组件,每个组件都有一个单一的职责,然后将它们配置为在链中运行。
您可以做的是创建一个转换器来修改消息的负载和 headers。该转换器将封装您的业务逻辑,并设置一个 header(例如,myRoutingHeader
),稍后可以在路由中使用。如果有一个用于业务逻辑的转换器和一个用于添加 header 的 header enricher 可能会更好。但是让我们假设您是在单个 class 中完成的。将 class 定义为一个 bean(比如 myTransformer
)。然后在你的配置中:
<int:channel id="inChannel/>
<!-- if performance is important consider using a SpEL expression to
invoke your method instead as they can be configured to be compiled -->
<int:transformer ref="myTransformer" input-channel="inChannel"
method="transform" output-channel="routingChannel"/>
<int:channel id="routingChannel/>
<int:header-value-router input-channel="routingChannel" header-name="myRoutingHeader">
<int:mapping value="foo" channel="fooChannel" />
<int:mapping value="bar" channel="barChannel" />
</int:header-value-router>
听起来链接可以为您做到这一点
<chain input-chanel="source" output-channel="routing-channel">
<transform/>
<!-- any enrichment process or intermediate process can go here -->
</chain>