春の統合を使用してルーティング スリップを実装しようとしています。構成は次のとおりです。
@Bean
@Transformer(inputChannel = "inputChannel", outputChannel = "replyChannel")
public HeaderEnricher headerEnricher() {
return new HeaderEnricher(Collections.singletonMap(IntegrationMessageHeaderAccessor.ROUTING_SLIP,
new RoutingSlipHeaderValueMessageProcessor("channel1", "channel2", "channel3"
, "channel4"
)));
}
と
@Bean
public MessageChannel inputChannel() {
return new DirectChannel();
}
@Bean
@BridgeTo
public MessageChannel replyChannel() {
return new DirectChannel();
}
追加したヘッダーエンリッチャーの後にルーティングスリップを参照するために
@BridgeTo
この実装は機能しますが、2 回に 1 回は機能します。Spring コードをデバッグした後、このコードに出くわし、問題を解決しました。
UnicastingDispatcher クラス内
private boolean doDispatch(Message<?> message) {
if (this.tryOptimizedDispatch(message)) {
return true;
}
boolean success = false;
Iterator<MessageHandler> handlerIterator = this.getHandlerIterator(message);
if (!handlerIterator.hasNext()) {
throw new MessageDispatchingException(message, "Dispatcher has no subscribers");
}
List<RuntimeException> exceptions = new ArrayList<RuntimeException>();
while (!success && handlerIterator.hasNext()) {
MessageHandler handler = handlerIterator.next();
try {
handler.handleMessage(message);
success = true; // we have a winner.
}
catch (Exception e) {
RuntimeException runtimeException = this.wrapExceptionIfNecessary(message, e);
exceptions.add(runtimeException);
this.handleExceptions(exceptions, message, !handlerIterator.hasNext());
}
}
return success;
}
Iterator<MessageHandler> handlerIterator = this.getHandlerIterator(message);
[org.springframework.integration.jms.JmsSendingMessageHandler#11, routingSlip.replyChannel.bridgeTo.handler]を与える
そしてラウンドロビン方式での抽選です。そのため、次に BridgeHandler と JmsSendingMessageHandler を選択すると、次のようになります。
BridgeHandler を常に参照するために見逃した構成はありますか?