Я определенно должен чего-то упустить, поэтому ищу помощи здесь.
Я создаю простое приложение REST с использованием Spring Integration со следующим входящим шлюзом HTTP:
<!-- Gateway -->
<int-http:inbound-gateway id="fruitQuotePOSTGateway"
request-channel="fruitQuotePOSTRequests"
supported-methods="POST"
path="/api/v1/fruit-quote"
request-payload-type="java.lang.String"
reply-timeout="10000"
reply-channel="fruitQuotePOSTResponses"
error-channel="applicationErrors">
<int-http:request-mapping consumes="application/xml" produces="application/xml"/>
</int-http:inbound-gateway>
Как только XML попадает в этот шлюз, он выполняет следующие простые шаги:
- Преобразование для создания объекта JAXB, соответствующего входящему запросу
- Расширение заголовка сообщения, которое считывает "uuid" из объекта JAXB и устанавливает его в заголовок сообщения SI (Spring Integration).
- Преобразование для генерации XML-ответа вызывающему клиенту.
Для начала вот XML-конфигурация всего приложения (для краткости опущены пространства имен HTTP):
<!-- Gateway -->
<int-http:inbound-gateway id="fruitQuotePOSTGateway"
request-channel="fruitQuotePOSTRequests"
supported-methods="POST"
path="/api/v1/fruit-quote"
request-payload-type="java.lang.String"
reply-timeout="10000"
reply-channel="fruitQuotePOSTResponses"
error-channel="applicationErrors">
<int-http:request-mapping consumes="application/xml" produces="application/xml"/>
</int-http:inbound-gateway>
<!--
- Generate fruit quote request JAXB from the incoming request
- Create a header "requestUUID" by reading it from fruit quote request JAXB
- Generate fruit quote acknowledgement response for the calling client
-->
<int:transformer input-channel="fruitQuotePOSTRequests"
ref="fruitQuoteTransformation"
method="generateFruitQuoteRequestJAXB"/>
<int:header-enricher input-channel="requestUUIDEnrichment" output-channel="orderIDGeneration">
<int:header name="requestUUID" expression="payload.getFruitQuoteRequestJAXB().getFRUITQUOTEREQUESTDATA().getUuid()"/>
</int:header-enricher>
<int:transformer input-channel="fruitQuoteAcknowledgementGeneration"
ref="fruitQuoteTransformation"
method="generateFruitQuoteAcknowledgement"
output-channel="fruitQuotePOSTResponses"/>
<!-- Error handling -->
<int:transformer input-channel="applicationErrors"
ref="fruitQuoteTransformation"
method="generateFruitQuoteAcknowledgementWithError"
output-channel="fruitQuotePOSTResponses"/>
<!-- Channels -->
<int:channel id="fruitQuotePOSTRequests"/>
<int:channel id="requestUUIDEnrichment"/>
<int:channel id="fruitQuotePOSTResponses"/>
<int:channel id="fruitQuoteAcknowledgementGeneration"/>
<int:channel id="applicationErrors"/>
Полезная нагрузка, передаваемая от одного шага к другому в приложении, представляет собой настраиваемый объект Builder, как показано ниже (без имени пакета):
import static java.util.Objects.nonNull;
public class FruiteQuoteComposite {
private final FRUITQUOTEREQUEST fruitQuoteRequestJAXB;
private final FruitQuoteApplicationException fruitQuoteApplicationException;
private final Integer orderID;
private final ErrorInformation errorInformation;
private FruiteQuoteComposite(FruiteQuoteCompositeBuilder fruiteQuoteCompositeBuilder) {
this.fruitQuoteRequestJAXB = fruiteQuoteCompositeBuilder.fruitQuoteRequestJAXB;
this.fruitQuoteApplicationException = fruiteQuoteCompositeBuilder.fruitQuoteApplicationException;
this.orderID = fruiteQuoteCompositeBuilder.orderID;
this.errorInformation = fruiteQuoteCompositeBuilder.errorInformation;
}
public FruitQuoteApplicationException getFruitQuoteApplicationException() {
return fruitQuoteApplicationException;
}
public FRUITQUOTEREQUEST getFruitQuoteRequestJAXB() {
return fruitQuoteRequestJAXB;
}
public Integer getOrderID() {
return orderID;
}
public ErrorInformation getErrorInformation() {
return errorInformation;
}
public static class FruiteQuoteCompositeBuilder {
private FRUITQUOTEREQUEST fruitQuoteRequestJAXB;
private FruitQuoteApplicationException fruitQuoteApplicationException;
private Integer orderID;
private ErrorInformation errorInformation;
public FruiteQuoteCompositeBuilder() {
}
public FruiteQuoteCompositeBuilder setFruitQuoteRequestJAXB(FRUITQUOTEREQUEST fruitQuoteRequestJAXB) {
if (nonNull(fruitQuoteRequestJAXB)) {
this.fruitQuoteRequestJAXB = fruitQuoteRequestJAXB;
}
return this;
}
public FruiteQuoteCompositeBuilder setFruitQuoteApplicationException(FruitQuoteApplicationException fruitQuoteApplicationException) {
if (nonNull(fruitQuoteApplicationException)) {
this.fruitQuoteApplicationException = fruitQuoteApplicationException;
}
return this;
}
public FruiteQuoteCompositeBuilder setOrderID(Integer orderID) {
if(nonNull(orderID)) {
this.orderID = orderID;
}
return this;
}
public FruiteQuoteCompositeBuilder setErrorInformation(ErrorInformation errorInformation) {
if (nonNull( errorInformation )) {
this.errorInformation = errorInformation;
}
return this;
}
public FruiteQuoteComposite build() {
return new FruiteQuoteComposite(this);
}
}
}
Причина, по которой я не использовал «выходной канал» на трансформаторах, заключается в том, что я хотел явно выбрать маршрут replyChannel / outgoing внутри логики Java, выполняющей преобразование.
Например, внутри метода FruitQuoteTransformation.generateFruitQuoteRequestJAXB я установил один маршрут для успеха и другой маршрут для исключений / ошибок следующим образом:
public Message<FruiteQuoteComposite> generateFruitQuoteRequestJAXB(Message<String> fruitQuoteRequestMessage) {
String fruitQuoteRequest = fruitQuoteRequestMessage.getPayload();
Unmarshaller unmarshaller;
FRUITQUOTEREQUEST fruitQuoteRequestJAXB;
try {
unmarshaller = requireNonNull(fruitQuoteRequestJaxbContext).createUnmarshaller();
fruitQuoteRequestJAXB = (FRUITQUOTEREQUEST) requireNonNull(unmarshaller)
.unmarshal(new StringReader(fruitQuoteRequest));
} catch (JAXBException jaxbException) {
logger.error("JAXB Unmarshalling exception occurred with error code :: " + ERR_FRUIT_QUOTE_REQUEST_JAXB_TRANSFORMATION, jaxbException);
FruitQuoteApplicationException fruitQuoteApplicationException = generateFruitQuoteApplicationException(ERR_FRUIT_QUOTE_REQUEST_JAXB_TRANSFORMATION, MESSAGE_FRUIT_QUOTE_INTERNAL_SYSTEM_ERROR);
FruiteQuoteComposite outboundFruitQuoteComposite = new FruiteQuoteComposite.FruiteQuoteCompositeBuilder()
.setFruitQuoteApplicationException(fruitQuoteApplicationException)
.build();
return withPayload(requireNonNull(outboundFruitQuoteComposite))
.setHeader(MessageHeaders.REPLY_CHANNEL, "applicationErrors")
.build();
}
FruiteQuoteComposite outboundFruitQuoteComposite = new FruiteQuoteComposite.FruiteQuoteCompositeBuilder()
.setFruitQuoteRequestJAXB(fruitQuoteRequestJAXB)
.build();
return withPayload(requireNonNull(outboundFruitQuoteComposite))
.setHeader(MessageHeaders.REPLY_CHANNEL, "requestUUIDEnrichment")
.build();
}
- Мой первый вопрос. По какой-то причине вызов .setHeader работает не так, как ожидалось, и сообщение не переходит на следующий канал. Что-то мне не хватает? Результат тот же, даже когда я использую .setReplyChannelName.
- Мой второй вопрос. Если есть решение вопроса 1), сохраняя общую конфигурацию SI на основе XML, есть ли альтернативный подход к настройке индивидуальных каналов ответа? Единственный вариант, который пришел мне в голову, - использовать маршрутизатор после каждого трансформатора, но это казалось слишком многословным.
Не могли бы вы помочь?