3-3、Processor 处理器

Camel中另一个重要的元素是Processor处理器,它用于接收从控制端点、路由选择条件又或者另一个处理器的Exchange中传来的消息信息,并进行处理。Camel核心包和各个Plugin组件都提供了很多Processor的实现,开发人员也可以通过实现org.apache.camel.Processor接口自定义处理器(后者是通常做法)。

既然是做编码,那么我们自然可以在自定义的Processor处理器中做很多事情。这些事情可能包括处理业务逻辑、建立数据库连接去做业务数据存储、建立和某个第三方业务系统的RPC连接,但是我们一般不会那样做——那是Endpoint的工作。Processor处理器中最主要的工作是进行业务数据格式的转换和中间数据的临时存储。这样做是因为Processor处理器是Camel编排的路由中,主要进行Exchange输入输出消息交换的地方。

不过开发人员当然可以在Processor处理器中连接数据库。例如开发人员需要根据上一个Endpoint中携带的“订单编号前缀”信息,在Processor处理器中连接到一个独立的数据库中(或者缓存服务中)查找其对应的路由信息,以便动态决定下一个路由路径。由于Camel支持和JAVA语言的Spring框架无缝集成,所以要在Processor处理器中操作数据库只需要进行非常简单的配置。

以下代码片段是自定义的Processor处理器实现,其中的process(Exchange exchange)方法是必须进行实现的:

// 一个自定义处理器的实现
 // 就是我们上文看到过的处理器实现了
 public class OtherProcessor implements Processor {
 ......
 @Override
 public void process(Exchange exchange) throws Exception {
 Message message = exchange.getIn();
 String body = message.getBody().toString();
 //===============
 // 您可以在这里进行数据格式转换
 // 并且将结果存储到out message中
 //=============== 
 // 存入到exchange的out区域
 if(exchange.getPattern() == ExchangePattern.InOut) {
 Message outMessage = exchange.getOut();
 outMessage.setBody(body + " || other out");
 }
 }
 ......
 }

注意,处理器Processor是和控制端点平级的概念。要看一个URI对应的实现是否是一个控制端点,最根本的就是看这个实现类是否实现了org.apache.camel.Endpoint接口;而要看一个路由中的元素是否是Processor处理器,最根本的就是看这个类是否实现了org.apache.camel.Processor接口。

3-5、Routing路由条件

在控制端点和处理器之间、处理器和处理器之间,Camel允许开发人员进行路由条件设置。例如开发人员可以拥有当Exchange In Message的内容为A的情况下将消息送入下一个处理器A,当Exchange In Message的内容为B时将消息送入下一个处理器B的处理能力。又例如,无论编排的路由中上一个元素的处理消息如何,都将携带消息的Exchange对象复制 多份,分别送入下一处理器X、Y、Z。开发人员甚至还可以通过路由规则完成Exchange到多个Endpoint的负载传输。

Camel中支持的路由规则非常丰富,包括:Message Filter、Based Router、Dynamic Router、Splitter、Aggregator、Resequencer等等。在Camel的官方文档中使用了非常形象化的图形来表示这些路由功能.

ProcessBuilder中文文档 processor._开发人员

实际上EIP规则中所描述的大部分业务集成模式,在以上页面都能找到对应的图形化表达。但由于篇幅和本专题的中心思想限制,恕笔者不能对Camel中的路由规则逐一讲解。这里我们选取两个比较有代表性的路由规则进行讲解:Content Based Router和Recipient List。希望对各位读者理解Camel中的路由规则有所帮助:

3-5-1、Content Based Router 基于内容的路由

把Content Based Router译为基于内容的路由,笔者觉得更为贴切(并不能译作基本路由,实际上你无法定义什么是基本路由)。它并不是一种单一的路由方式,而是多种基于条件和判断表达式的路由方式。其中可能包括choice语句/方法、when语句/方法、otherwise语句/方法。请看以下示例:

package com.yinwenjie.test.cameltest.helloworld; 
......
 /**
 * 使用条件选择进行路由编排
 * @author yinwenjie
 */
 public class ChoiceCamel extends RouteBuilder { public static void main(String[] args) throws Exception {
 // 这是camel上下文对象,整个路由的驱动全靠它了。
 ModelCamelContext camelContext = new DefaultCamelContext();
 // 启动route
 camelContext.start();
 // 将我们编排的一个完整消息路由过程,加入到上下文中
 camelContext.addRoutes(new ChoiceCamel()); // 通用没有具体业务意义的代码,只是为了保证主线程不退出
 synchronized (ChoiceCamel.class) {
 ChoiceCamel.class.wait();
 }
 } @Override
 public void configure() throws Exception {
 // 这是一个JsonPath表达式,用于从http携带的json信息中,提取orgId属性的值
 JsonPathExpression jsonPathExpression = new JsonPathExpression("$.data.orgId");
 jsonPathExpression.setResultType(String.class); // 通用使用http协议接受消息
 from("jetty:http://0.0.0.0:8282/choiceCamel")
 // 首先送入HttpProcessor,
 // 负责将exchange in Message Body之中的stream转成字符串
 // 当然,不转的话,下面主要的choice操作也可以运行
 // HttpProcessor中的实现和上文代码片段中的一致,这里就不再重复贴出
 .process(new HttpProcessor())
 // 将orgId属性的值存储 exchange in Message的header中,以便后续进行判断
 .setHeader("orgId", jsonPathExpression)
 .choice()
 // 当orgId == yuanbao,执行OtherProcessor
 // 当orgId == yinwenjie,执行OtherProcessor2
 // 其它情况执行OtherProcessor3
 .when(header("orgId").isEqualTo("yuanbao"))
 .process(new OtherProcessor())
 .when(header("orgId").isEqualTo("yinwenjie"))
 .process(new OtherProcessor2())
 .otherwise()
 .process(new OtherProcessor3())
 // 结束
 .endChoice();
 } /**
 * 这个处理器用来完成输入的json格式的转换
 * 和上一篇文章出现的HttpProcessor 内容基本一致。就不再贴出了
 * @author yinwenjie
 */
 public class HttpProcessor implements Processor {
 ......
 } /**
 * 另一个处理器OtherProcessor
 * @author yinwenjie
 */
 public class OtherProcessor implements Processor { @Override
 public void process(Exchange exchange) throws Exception {
 Message message = exchange.getIn();
 String body = message.getBody().toString(); // 存入到exchange的out区域
 if(exchange.getPattern() == ExchangePattern.InOut) {
 Message outMessage = exchange.getOut();
 outMessage.setBody(body + " || 被OtherProcessor处理");
 }
 }
 } /**
 * 很简单的处理器OtherProcessor2
 * 和OtherProcessor基本相同,就不再重复贴出
 * @author yinwenjie
 */
 public class OtherProcessor2 implements Processor {
 ......
 outMessage.setBody(body + " || 被OtherProcessor2处理");
 } /**
 * 很简单的处理器OtherProcessor3
 * 和OtherProcessor基本相同,就不再重复贴出
 * @author yinwenjie
 */
 public class OtherProcessor3 implements Processor {
 ......
 outMessage.setBody(body + " || 被OtherProcessor3处理");
 }
 }

以上代码片段中,开发人员首先使用JsonPath表达式,从Http中携带的json信息中寻找到orgId这个属性的值,然后将这个值存储在Exchange的header区域(这样做只是为了后续方便判断,您也可以将值存储在Exchange的properties区域,还可以直接使用JsonPath表达式进行判断) 。接下来,通过判断存储在header区域的值,让消息路由进入不同的Processor处理器。由于我们设置的from-jetty-endpoint中默认的Exchange Pattern值为InOut,所以在各个Processor处理器中完成处理后 Out Message的Body内容会以Http响应结果的形式返回到from-jetty-endPoint中。最后我们将在测试页面上看到Processor处理器中的消息值。