- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个服务器和客户端应用程序,它们都使用 Apache Camel 来配置和管理其路由。我正在尝试使用 REST 组件从客户端向服务器发出 POST 请求,并在请求正文中发送消息。似乎正在调用 REST 端点,但主体未通过。我认为我要么遗漏了一些步骤,要么做得不正确。
另一件事要提到的是,我使用的是一个通用消息类,它基本上可以包含任何对象(包括字节或字符数组)。也许这是我遇到的问题背后的部分(或全部)原因。
GenericMessage.java:
public class GenericMessage<T extends Object> {
private T payload;
public GenericMessage(T payload) {
this.payload = payload;
}
public T getPayload() {
return payload;
}
public void setPayload(T payload) {
this.payload = payload;
}
@Override
public String toString() {
return "GenericMessage [payload=" + payload + "]";
}
}
服务器
camel-context.xml:
<camelContext id="camel-context" xmlns="http://camel.apache.org/schema/spring">
<restConfiguration component="jetty" bindingMode="json"
contextPath="/" host="localhost" port="9100">
<dataFormatProperty key="prettyPrint" value="true" />
</restConfiguration>
<rest>
<get uri="/data?size={size}" produces="JSON">
<route id="get-data-by-size">
<log loggingLevel="INFO" message="start - get-data-by-size" />
<process ref="get-request-processor" />
<log loggingLevel="INFO" message="end - get-data-by-size" />
</route>
</get>
<post uri="post-data" consumes="json">
<route id="post-data">
<log loggingLevel="INFO" message="start - post-data" />
<process ref="post-request-processor" />
<log loggingLevel="INFO" message="end - post-data" />
</route>
</post>
</rest>
</camelContext>
GetRequestProcessor.java:
@Component("get-request-processor")
public class GetRequestProcessor implements Processor {
@Autowired
private DataProvider provider;
public void process(Exchange exchange) throws Exception {
int size = exchange.getIn().getHeader("size", Integer.class);
// TODO currently only sends char[] data
GenericMessage<?> data = provider.getCharData(size);
exchange.getMessage().setBody(data, GenericMessage.class);
}
}
PostRequestProcessor.java:
@Component("post-request-processor")
public class PostRequestProcessor implements Processor {
private final Logger logger = LoggerFactory.getLogger(PostRequestProcessor.class);
public void process(Exchange exchange) throws Exception {
GenericMessage<?> data = exchange.getIn().getBody(GenericMessage.class);
if (data != null) {
logger.info("process: object received: {}", data);
} else {
logger.warn("process: null object");
}
}
}
DataProvider.java:
@Component
public class DataProvider {
private final Logger logger = LoggerFactory.getLogger(DataProvider.class);
public GenericMessage<?> getByteData(final int size) {
logger.info("getByteData: size={}", size);
return new GenericMessage<byte[]>(DataGenerator.generateByteArray(size));
}
public GenericMessage<?> getCharData(final int size) {
logger.info("getCharData: size={}", size);
return new GenericMessage<char[]>(DataGenerator.generateCharArray(size));
}
}
客户端
camel-context.xml:
<camelContext id="camel-context" xmlns="http://camel.apache.org/schema/spring">
<restConfiguration component="jetty" bindingMode="json"
contextPath="/" host="{{http-client.server.host}}"
port="{{http-client.server.port}}">
<dataFormatProperty key="prettyPrint" value="true" />
</restConfiguration>
<route id="rest-get">
<from
uri="timer:{{http-client.timers.http-get.name}}?delay={{http-client.timers.http-get.start-delay}}&fixedRate=true&period={{http-client.timers.http-get.period}}&repeatCount={{http-client.timers.http-get.repeat-count}}" />
<log loggingLevel="INFO" message="start - rest-get" />
<to uri="rest:get:{{http-client.endpoints.http-get}}" />
<process ref="process-get-response" />
<log loggingLevel="INFO" message="end - rest-get" />
</route>
<route id="rest-post">
<from
uri="timer:{{http-client.timers.http-post.name}}?delay={{http-client.timers.http-post.start-delay}}&fixedRate=true&period={{http-client.timers.http-post.period}}&repeatCount={{http-client.timers.http-post.repeat-count}}" />
<log loggingLevel="INFO" message="start - rest-post" />
<process ref="add-post-body" />
<to uri="rest:post:{{http-client.endpoints.http-post}}" />
<log loggingLevel="INFO" message="end - rest-post" />
</route>
</camelContext>
AddPostRequestBody.java:
@Component("add-post-body")
public class AddPostRequestBody implements Processor {
private final Logger logger = LoggerFactory.getLogger(AddPostRequestBody.class);
@Autowired
private DataProvider provider;
@Override
public void process(Exchange exchange) throws Exception {
GenericMessage<?> data = null;
int intValue = RandomUtil.generateInt(1);
switch (intValue) {
case 0:
data = provider.produceByteArrayData();
break;
case 1:
data = provider.produceCharArrayData();
break;
default:
break;
}
logger.info("adding POST request body:\n{}", data);
exchange.getMessage().setHeader(Exchange.HTTP_METHOD, "POST");
exchange.getMessage().setHeader(Exchange.CONTENT_TYPE, "application/json");
exchange.getMessage().setBody(data, GenericMessage.class);
}
}
GetResponseProcessor.java:
@Component("process-get-response")
public class GetResponseProcessor implements Processor {
private final Logger logger = LoggerFactory.getLogger(GetResponseProcessor.class);
@Override
public void process(Exchange exchange) throws Exception {
GenericMessage<?> body = (GenericMessage<?>) exchange.getIn().getBody();
logger.info("body: {}", body);
}
}
DataProvider.java:
@Component
public class DataProvider {
private final Logger logger = LoggerFactory.getLogger(DataProvider.class);
public GenericMessage<byte[]> produceByteArrayData() {
int size = RandomUtil.generateInt(1000, 20000000);
byte[] bytes = DataGenerator.generateByteArray(size);
logger.info("produceByteArrayData: generated {} bytes", size);
GenericMessage<byte[]> data = new GenericMessage<>(bytes);
return data;
}
public GenericMessage<char[]> produceCharArrayData() {
int size = RandomUtil.generateInt(1000, 20000000);
char[] chars = DataGenerator.generateCharArray(size);
logger.info("produceCharArrayData: generated {} characters", size);
GenericMessage<char[]> data = new GenericMessage<>(chars);
return data;
}
}
我已将其配置为执行来自客户端的一个 POST 请求。分散的日志语句可以让您了解正在发生的情况。
客户端控制台:
2020-04-13 13:23:39.805 INFO 17464 --- [ main] o.a.c.i.e.AbstractCamelContext : Route: rest-get started and consuming from: timer://http-get
2020-04-13 13:23:39.806 INFO 17464 --- [ main] o.a.c.i.e.AbstractCamelContext : Route: rest-post started and consuming from: timer://http-post
2020-04-13 13:23:39.812 INFO 17464 --- [ main] o.a.c.i.e.AbstractCamelContext : Total 2 routes, of which 2 are started
2020-04-13 13:23:39.812 INFO 17464 --- [ main] o.a.c.i.e.AbstractCamelContext : Apache Camel 3.1.0 (CamelContext: RESTClient) started in 1.679 seconds
2020-04-13 13:23:42.308 INFO 17464 --- [mer://http-post] rest-post : start - rest-post
2020-04-13 13:23:42.312 INFO 17464 --- [mer://http-post] e.m.l.m.s.DataProvider : produceCharArrayData: generated 1039 characters
2020-04-13 13:23:42.313 INFO 17464 --- [mer://http-post] e.m.l.m.p.AddPostRequestBody : CLIENT: Adding POST request body:
GenericMessage [payload=[C@2124c204]
2020-04-13 13:23:42.408 INFO 17464 --- [mer://http-post] rest-post : end - rest-post
服务器控制台:
2020-04-13 13:23:10.846 INFO 23260 --- [ main] o.a.c.i.e.AbstractCamelContext : Route: get-data-by-size started and consuming from: jetty:http://localhost:9100/data
2020-04-13 13:23:10.847 INFO 23260 --- [ main] o.a.c.i.e.AbstractCamelContext : Route: post-data started and consuming from: jetty:http://localhost:9100/post-data
2020-04-13 13:23:10.852 INFO 23260 --- [ main] o.a.c.i.e.AbstractCamelContext : Total 2 routes, of which 2 are started
2020-04-13 13:23:10.853 INFO 23260 --- [ main] o.a.c.i.e.AbstractCamelContext : Apache Camel 3.1.0 (CamelContext: RESTServer) started in 0.314 seconds
2020-04-13 13:23:42.395 INFO 23260 --- [tp1276761134-37] post-data : start - post-data
2020-04-13 13:23:42.396 INFO 23260 --- [tp1276761134-37] e.m.l.m.p.PostRequestProcessor : SERVER: Processing POST request
2020-04-13 13:23:42.400 WARN 23260 --- [tp1276761134-37] e.m.l.m.p.PostRequestProcessor : null object
2020-04-13 13:23:42.400 INFO 23260 --- [tp1276761134-37] post-data : end - post-data
最佳答案
Camel 不知道你的GenericMessage
类,因此它不知道如何与它进行转换。
Camel使用可插拔type converters告诉它如何在不同类型的对象之间转换消息(Body)。
内置类型转换器涵盖了许多常见情况(例如,字符串到整数、迭代器到ArrayList等),但它们不知道如何与您编写的任何自定义类进行转换。
这就是服务器处理器中的这一行返回 null 的原因:
GenericMessage<?> data = exchange.getIn().getBody(GenericMessage.class);
因为当时Body是一个Stream,但是Camel不知道如何将它转换为GenericMessage
你已经要求了。
( getBody(Class<T>)
的默认行为是,如果无法转换,则返回 null。如果您希望抛出异常,请尝试使用 getMandatoryBody(Class<T>)
。)
在客户端:如何获得 GenericMessage
的响应?您需要使用解码。首先将 JSON 解码为 Java 对象。此时,您需要向 Camel 提示要解码到哪个类。
<dataFormats>
<json id="myGeneric" .... unmarshalTypeName="....GenericMessage"/>
</dataFormats>
...
<unmarshal><custom ref="myGeneric"/></unmarshal>
那么 body 就是你的GenericMessage
类型。
请注意,在 Camel 中,类型转换和编码/反编码是两个不同的事情。
在服务器端:Camel 将尝试自动将 JSON 请求解码为简单的 Java 对象(例如 HashMap 等)。要告诉它解码到特定的 POJO,请设置 type
关于 REST 动词定义,例如:
<post uri="post-data" consumes="json" type="com.example.demo.GenericMessage">
您还可以使用相同的方法来定义 outType
(用于将响应编码回 JSON)
关于java - Apache Camel : Problem POSTing request body,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61146802/
我创建了一个 spring-boot 应用程序,我在其中使用 camel-reSTLet 组件将我的 camel 路由公开为 rest 端点。 我的 camel 路由很简单:它们接受来自北向休息端点的
我有一条路由 (route1),它将数据发送到 HTTP 端点。为此,它必须设置授权 header 。 header 值每小时超时一次,必须更新。 为此,我创建了另一个路由 (route2),它使用提
我正在使用 camel-cdi,它正在注入(inject) CamelContext,检测项目中的所有路由。但是我想要一个带有注册表的 CamelContext,因为我有一些在 Camel route
我想使用来自网络服务的数据并将其放入 Camel eh-cache 中。后来我想通过 CacheManager 在 Camel 上下文之外使用这个缓存。我没有找到任何方法。 在下面的代码中,我跳过了
问题描述: 我无法从我的 Camel servlet 路由到 cxfbean。路由初始化失败并显示以下错误消息: "Failed to create route route1 at: >>> To[c
我想了解 Camel 中的工作单元概念。我有一个简单的问题,希望这里有人可以提供帮助。 例如,如果路由 Exchange 涉及多个路由 from("aws-sqs:Q1").to("direct:pr
首先是我正在尝试做的事情的基本轮廓 我有一个 MQ,我想从 读取消息 预处理 XML,并在 Exchange 上设置属性 发出 HTTP 请求 处理来自 http 请求和初始交换中的属性的数据 将其放
我有一个 SFTP 路由(在 Spring XML 中),它的 from 路径以每日更改的目录(即/yyyyMMdd)结尾,并且在 autoCreate=true 时一切正常或者路径开始时目录存在。但
如何用 Camel 实现这样的过程: 拆分 处理每个拆分的项目 聚合结果 如果发生异常: 停止 split 返回异常前所有item的聚合结果及异常信息 split时定义.stopOnException
我在 Camel 中有一条路线,我想在发生异常时重试,但我想设置一个属性,以便路线第二次可以做一些稍微不同的事情,以尝试阻止错误在重试时再次发生。这是说明我目前正在尝试的想法的路线。 from("di
这两个有何不同 from(endpoint).to(endpoint:a, endpoint:b) from(endpoint).multicast().to(endpoint:a, endpoint
我的 Camel 路线如下(示例) from (activemq:xyz) --- 从 QUEUE 接收消息 to(smpp:abc) --- 提交短信至短信中心 to(cxf:hij) --- 基于
我的 Camel 路线如下(示例) from (activemq:xyz) --- 从队列接收消息 to(smpp:abc) --- 将消息提交给 SMSC to(cxf:hij) --- 基于 SM
当捕获异常时,有什么方法可以停止路由执行(显示日志消息后)? java.lang.IllegalA
我正在使用 Camel 进行集成。我有一个用例,其中 Camel 应该将 1 条消息从一个队列传输到另一个队列,但它不断向队列发送相同的消息。请查看我的以下路线: ProducerTemplate正在
当异常在多播内部抛出时,Camel 不会传播异常。 考虑到以下设置,其中 direct:route 从其 beanRef 抛出异常: rest("/...") .pos
有没有办法使用生产者模板设置 Camel 交换属性? 想象一个接收客户订单的休息端点(尚未在 Camel route )。使用生产者模板,我想 在交易所上设置客户 ID 属性。 稍后在路由 需要时使用
再见,我的基本要求是有一个可以发送消息的路由,并将其放在 JMS 队列中。 camel 上下文在 JavaEE 6 容器中运行,即 JBoss AS 7.1.1,因此它是 HornetQ for JM
Camel 2.23.1 Karaf 4.2.4 白羊座蓝图(用于注册所有内容的外部容器) Camel 蓝图(用于 Camel 路线) Camel CXF(用于 rsServer) CXF 核心(用于
现在我在 Java EE 7 应用程序上使用 JMS 2.0 和 Artemis 1.2.0,我想用 Camel 做一些集成任务。 现在查看 camel-jms 文档,没有提及如何使用通用的 came
我是一名优秀的程序员,十分优秀!