为链中的每个处理程序分配不同的错误处理程序

assigning a different error handler to each handler in a chain

我有一个指定简单流程的应用程序,其中需要以特定顺序调用服务链以执行某些功能。 Spring 集成及其 DSL 似乎适合这种情况。

允许这些单个服务中的任何一个失败(这些服务是对外部应用程序的调用),此时链应该停止。不仅如此,还需要根据失败的服务将控制流重定向到其他地方。换句话说,我需要为每个服务使用不同的错误处理程序。

问题是我很难找到如何做到这一点,如果有人能指出正确的方向,我将不胜感激。到目前为止,这就是我解决问题的方式:

@Configuration
@EnableAutoConfiguration
@IntegrationComponentScan
public class Application {

public static void main(String[] args) throws InterruptedException {
    ConfigurableApplicationContext ctx =
            SpringApplication.run(Application.class, args);

    System.out.println(ctx.getBean(MySource.class).send("foo"));

    ctx.close();
}

@MessagingGateway
public interface MySource {

    @Gateway(requestChannel = "flow.input")
    String send(String string);

}

@Bean
public IntegrationFlow flow() {
    return f -> f
            .handle(String.class, (i, map) -> {
 //             if (true){
 //                 throw new RuntimeException();
 //             }
                System.out.println(i);
                return i + "a";
            })
            .handle(String.class, (i, map) -> {
                System.out.println(i);
                return i + "b";
            })
            .handle(String.class, (i, map) -> {
                System.out.println(i);
                return i + "c";
            });
   }
}

我一直在寻找如何为每个处理程序设置错误 handler/channel,但我没有找到任何有用的东西。

差点放弃搜索开始自己写代码(),大概是这样的:

public class FlowHarness {

    public static void main(String[] args) {
        Flow flow = FlowBuilder.flow()
            .nextStep(i -> ((Integer)i) + 1, e -> System.out.println("failed at step 1"))
            .nextStep(i -> ((Integer)i) + 1, e -> System.out.println("failed at step 2"))
            .nextStep(i -> ((Integer)i) + 1, e -> System.out.println("failed at step 3"))
            .build();

        flow.execute(new Integer(1));
    }
}

.nextStep() 的左侧参数是处理程序,而右侧参数是错误处理程序。非常简单,但我希望它能说明这一点。

尽管所有现实生活中的流程都可以写成一个大 class,但我认为将每个子流程分解成单独的私有 methods/classes 会更方便。非常感谢。

由于您希望链在 failure.I 上停止执行,因此建议您可以使用具有适当处理程序的自定义异常

.handle()(以及任何其他消费者端点)可以与 ExpressionEvaluatingRequestHandlerAdvice 一起提供:http://docs.spring.io/spring-integration/docs/4.3.9.RELEASE/reference/html/messaging-endpoints-chapter.html#expression-advice

@Bean
public IntegrationFlow advised() {
    return f -> f.handle((GenericHandler<String>) (payload, headers) -> {
        if (payload.equals("good")) {
            return null;
        }
        else {
            throw new RuntimeException("some failure");
        }
    }, c -> c.advice(expressionAdvice()));
}

@Bean
public Advice expressionAdvice() {
    ExpressionEvaluatingRequestHandlerAdvice advice = new ExpressionEvaluatingRequestHandlerAdvice();
    advice.setSuccessChannelName("success.input");
    advice.setOnSuccessExpressionString("payload + ' was successful'");
    advice.setFailureChannelName("failure.input");
    advice.setOnFailureExpressionString(
            "payload + ' was bad, with reason: ' + #exception.cause.message");
    advice.setTrapException(true);
    return advice;
}

@Bean
public IntegrationFlow success() {
    return f -> f.handle(System.out::println);
}

@Bean
public IntegrationFlow failure() {
    return f -> f.handle(System.out::println);
}