From ff1e38bfa954dd747a5ff185dfe51c120d5ab5e7 Mon Sep 17 00:00:00 2001 From: Christof Rabensteiner Date: Fri, 6 Dec 2019 13:18:12 +0100 Subject: Change msg service: Acknowledge Response iff Backend Succeeds - Before: Upon receipt of a message via the msg/ endpoint, MOA ZS would immediately acknowledge the receipt without verifying that the message was successfully processed by the backend. - Now: MOA ZS receives a message via the msg/ endpoint, forwards it to the sinks, and acknowledges the receipt if and only if the processing succeeded. --- .../java/at/gv/egiz/moazs/service/MsgService.java | 47 +++++++++++++++++----- 1 file changed, 37 insertions(+), 10 deletions(-) (limited to 'src/main/java/at/gv/egiz/moazs/service/MsgService.java') diff --git a/src/main/java/at/gv/egiz/moazs/service/MsgService.java b/src/main/java/at/gv/egiz/moazs/service/MsgService.java index 6b260e7..6fce6ae 100644 --- a/src/main/java/at/gv/egiz/moazs/service/MsgService.java +++ b/src/main/java/at/gv/egiz/moazs/service/MsgService.java @@ -20,6 +20,7 @@ * that you distribute must include a readable copy of the "NOTICE" text file. *******************************************************************************/ package at.gv.egiz.moazs.service; +import at.gv.egiz.moazs.backend.MsgResponseBackend; import at.gv.egiz.moazs.repository.DeliveryRepository; import at.gv.egiz.moazs.scheme.MsgResponse; import at.gv.egiz.moazs.scheme.NameSpace; @@ -27,15 +28,21 @@ import at.gv.egiz.moazs.scheme.NotificationResponse; import at.gv.egiz.moazs.scheme.RequestStatusResponse; import at.gv.zustellung.msg.xsd.*; import org.apache.cxf.annotations.SchemaValidation; +import org.apache.cxf.binding.soap.SoapFault; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.core.task.TaskExecutor; import org.springframework.stereotype.Service; -import java.util.function.Consumer; +import javax.xml.namespace.QName; +import java.util.concurrent.CompletableFuture; +import java.util.concurrent.ExecutionException; import static at.gv.zustellung.msg.xsd.DeliveryNotificationACKType.deliveryNotificationACKTypeBuilder; import static at.gv.zustellung.msg.xsd.DeliveryRequestStatusACKType.deliveryRequestStatusACKTypeBuilder; import static at.gv.zustellung.msg.xsd.GetVersionResponse.getVersionResponseBuilder; +import static java.lang.String.format; import static java.util.concurrent.CompletableFuture.runAsync; /** @@ -47,12 +54,15 @@ import static java.util.concurrent.CompletableFuture.runAsync; @org.apache.cxf.feature.Features (features = "org.apache.cxf.ext.logging.LoggingFeature") public class MsgService implements Zuse2AppPort { + private static final Logger log = LoggerFactory.getLogger(MsgService.class); + private static final String BACKEND_ERROR_MSG = "Could not process %s."; + private final DeliveryRepository repository; - private final Consumer backend; + private final MsgResponseBackend backend; private final TaskExecutor taskExecutor; @Autowired - public MsgService(DeliveryRepository repository, Consumer msgResponseBackend, TaskExecutor taskExecutor) { + public MsgService(DeliveryRepository repository, MsgResponseBackend msgResponseBackend, TaskExecutor taskExecutor) { this.repository = repository; this.backend = msgResponseBackend; this.taskExecutor = taskExecutor; @@ -68,8 +78,14 @@ public class MsgService implements Zuse2AppPort { @Override public DeliveryRequestStatusACKType status(DeliveryRequestStatusType status) { var response = new RequestStatusResponse(status); - sendToWork(response); - return statusAck(response.getAppDeliveryID(), response.getZSDeliveryID()); + try { + sendToBackend(response).get(); + return statusAck(response.getAppDeliveryID(), response.getZSDeliveryID()); + } catch (InterruptedException | ExecutionException e) { + var message = format(BACKEND_ERROR_MSG, "request status"); + log.error(message, e); + throw toFault(e, message); + } } private DeliveryRequestStatusACKType statusAck(String appDeliveryID, String zsDeliveryID) { @@ -83,13 +99,24 @@ public class MsgService implements Zuse2AppPort { @Override public DeliveryNotificationACKType notification(DeliveryNotificationType notification) { var response = new NotificationResponse(notification); - sendToWork(response); - return notificationAck(response.getAppDeliveryID(), response.getZSDeliveryID()); + try { + sendToBackend(response).get(); + return notificationAck(response.getAppDeliveryID(), response.getZSDeliveryID()); + } catch (InterruptedException | ExecutionException e) { + var message = format(BACKEND_ERROR_MSG, "delivery notification"); + log.error(message, e); + throw toFault(e, message); + } + } + + private RuntimeException toFault(Exception e, String fallbackMessage) { + Throwable cause = (e.getCause() != null) ? e.getCause() : e; + return new SoapFault(fallbackMessage, cause, new QName("faultcode")); } - private void sendToWork(MsgResponse response) { - runAsync(() -> repository.store(response), taskExecutor) - .thenRunAsync(() -> backend.accept(response.getResponseID()), taskExecutor); + private CompletableFuture sendToBackend(MsgResponse response) { + return runAsync(() -> repository.store(response), taskExecutor) + .thenCompose(ignore -> backend.accept(response.getResponseID())); } private DeliveryNotificationACKType notificationAck(String appDeliveryID, String zsDeliveryID) { -- cgit v1.2.3