diff --git a/.github/workflows/quickstart_microprofile-lra_ci.yml b/.github/workflows/quickstart_microprofile-lra_ci.yml new file mode 100644 index 0000000000..ba62366703 --- /dev/null +++ b/.github/workflows/quickstart_microprofile-lra_ci.yml @@ -0,0 +1,14 @@ +name: WildFly microprofile-lra Quickstart CI + +on: + pull_request: + types: [opened, synchronize, reopened, ready_for_review] + paths: + - 'microprofile-lra/**' + - '.github/workflows/quickstart_ci.yml' +jobs: + call-quickstart_ci: + uses: ./.github/workflows/quickstart_ci.yml + with: + QUICKSTART_PATH: microprofile-lra + TEST_BOOTABLE_JAR: true diff --git a/microprofile-lra/README.adoc b/microprofile-lra/README.adoc new file mode 100644 index 0000000000..9c1c7b398e --- /dev/null +++ b/microprofile-lra/README.adoc @@ -0,0 +1,851 @@ +include::../shared-doc/attributes.adoc[] + += microprofile-lra: MicroProfile LRA QuickStart +:author: Martin Stefanko +:level: Beginner +:technologies: MicroProfile LRA + +[abstract] +The `microprofile-lra` quickstart demonstrates the use of the MicroProfile LRA specification in {productName}. + +:standalone-server-type: microprofile +:configFileName: standalone-microprofile.xml +:archiveType: war +:archiveName: {artifactId} +:microprofile-lra: +:restoreScriptName: restore-configuration.cli +:openshift: true +:custom-bootable-jar-layers: +:custom-openshift-layers: +:extra-openshift-test-arguments: + +== What is it? + +link:https://github.com/eclipse/microprofile-lra[MicroProfile LRA specification] aims to provide an API that the +applications utilize to cooperate actions in +distributed +transactions based on the saga pattern. The user applications enlist within the LRA which in turn notifies all enlisted +participants about the LRA (transaction) outcome. The saga pattern provides different transactional guarantees than ACID +transactions. Saga allows individual operations to execute right when they are invoked. Meaning together with the +enlistment in the LRA. It also requires each participant to define a compensating action which is a semantic undo of the +original operation. Note that this doesn't need to be opposite action. The compensation is required to put the state of +the system into the semantically same state as before the action invocation, not exactly same. If your action is for +instance sending an email, your compensation might be another email cancelling previous email. + +If all actions execute successfully, the LRA is closed and the optional Complete callbacks are invoked on enlisted +participants. If any action fails, then the LRA is cancelled and all compensation actions (Compensate callbacks) of all +enlisted participants are invoked. The state of the system is said to be eventually consistent, since if we don't start +any new LRAs, the state is bound to become consistent eventually. + +The implementation used in the {productName} is provided by the +link:https://github.com/jbosstm/narayana/tree/main/rts/lra[Narayana project]. + +== Architecture + +In this quickstart, we have a simple REST application that exposes several REST endpoints that enlist the application as +different LRA participants and provide callbacks for completions and compensations respectively. It's REST API consists +of the following +endpoints: + +- `GET /participant1/work` - work action of Participant 1 +- `GET /participant2/work` - work action of Participant 2 +- `PUT /participant1/compensate` - compensating action of Participant 1 +- `PUT /participant2/compensate` - compensating action of Participant 2 +- `PUT /participant1/complete` - complete action of Participant 1 +- `PUT /participant2/complete` - complete action of Participant 2 + + +// System Requirements +include::../shared-doc/system-requirements.adoc[leveloffset=+1] + +// Use of {jbossHomeName} +include::../shared-doc/use-of-jboss-home-name.adoc[leveloffset=+1] + +// Back Up the {productName} Standalone Server Configuration +include::../shared-doc/back-up-server-standalone-configuration.adoc[leveloffset=+1] + +// Start the {productName} Standalone Server +include::../shared-doc/start-the-standalone-server.adoc[leveloffset=+1] + +[[configure_the_server]] +== Configure the Server + +You can configure the LRA extensions and subsystems (both for LRA coordinator and LRA participant respectively) by running CLI commands. +For your convenience, this quickstart batches the commands into a `enable-microprofile-lra.cli` script provided in the root directory +of this quickstart. + +. Before you begin, make sure you do the following: + +* xref:back_up_standalone_server_configuration[Back up the {productName} standalone server configuration] as described above. +* xref:start_the_eap_standalone_server[Start the {productName} server with the standalone default profile] as described above. + +. Review the `enable-microprofile-lra.cli` file in the root of this quickstart directory. It enables two extensions and adds +two subsystems, one for LRA coordinator and one for LRA participant respectively. +. Open a new terminal, navigate to the root directory of this quickstart, and run the following command, replacing `__{jbossHomeName}__` +with the path to your server: ++ +[source,subs="+quotes,attributes+",options="nowrap"] +---- +$ __{jbossHomeName}__/bin/jboss-cli.sh --connect --file=enable-microprofile-lra.cli +---- ++ +NOTE: For Windows, use the `__{jbossHomeName}__\bin\jboss-cli.bat` script. ++ + +You should see the following result when you run the script: ++ +[source,options="nowrap"] +---- +The batch executed successfully +---- + +. Stop the {productName} server. + +== Review the Modified Server Configuration + +After stopping the server, open the `__{jbossHomeName}__/standalone/configuration/{configFileName}` file and review the changes. + +. The script added the following two extensions: ++ +[source,xml,options="nowrap"] +---- + + +---- ++ + +. And also the following two subsystems: ++ +[source,xml,options="nowrap"] +---- + + +---- + + +[[solution]] +== Solution + +We recommend that you follow the instructions that +<>. However, you can +also go right to the completed example which is available in this directory. + +// Build and Deploy the Quickstart +include::../shared-doc/build-and-deploy-the-quickstart.adoc[leveloffset=+1] + +// Server Distribution Testing +include::../shared-doc/run-integration-tests-with-server-distribution.adoc[leveloffset=+2] + +// Undeploy the Quickstart +include::../shared-doc/undeploy-the-quickstart.adoc[leveloffset=+1] + +// Restore the {productName} Standalone Server Configuration +:restoreScriptName: restore-configuration.cli +include::../shared-doc/restore-standalone-server-configuration.adoc[leveloffset=+1] + +// Additional information about this script +This script removes the added extensions and subsystems for the LRA participant and the LRA coordinator. + +[source,options="nowrap"] +---- +The batch executed successfully +process-state: reload-required +---- + +// Restore the {productName} Standalone Server Configuration Manually +include::../shared-doc/restore-standalone-server-configuration-manual.adoc[leveloffset=+2] + +//Bootable JAR +include::../shared-doc/build-and-run-the-quickstart-with-bootable-jar.adoc[leveloffset=+1] + +// OpenShift +include::../shared-doc/build-and-run-the-quickstart-with-openshift.adoc[leveloffset=+1] + +[[creating-new-project]] +== Creating the Maven Project + +[source,options="nowrap"] +---- +mvn archetype:generate \ + -DgroupId=org.wildfly.quickstarts \ + -DartifactId=microprofile-lra \ + -DinteractiveMode=false \ + -DarchetypeGroupId=org.apache.maven.archetypes \ + -DarchetypeArtifactId=maven-archetype-webapp +cd microprofile-lra +---- + +Open the project in your favourite IDE. + +Open the generated `pom.xml`. + +The first thing to do is to change the minimum JDK to Java 11 and set the other relevant version properties: + +[source,xml] +---- +11 +11 + + +30.0.0.Final + +${version.server} +${version.server} +5.0.0.Final +4.2.0.Final +10.0.0.Final +---- + +Next we need to setup our dependencies. Add the following section to your +`pom.xml`: + +[source,xml,subs="attributes+"] +---- + + + + org.wildfly.bom + wildfly-ee-with-tools + ${version.bom.ee} + pom + import + + + org.wildfly.bom + wildfly-microprofile + ${version.bom.microprofile} + pom + import + + + +---- + +Now we need to add the following dependencies: + +[source,xml] +---- + + org.eclipse.microprofile.lra + microprofile-lra-api + provided + + + jakarta.ws.rs + jakarta.ws.rs-api + provided + + + jakarta.enterprise + jakarta.enterprise.cdi-api + provided + + + org.jboss.logging + jboss-logging + provided + +---- + +NOTE: We need Jakarta REST (JAX-RS) since LRA exposes functionality over JAX-RS resources and uses HTTP as its +communication protocol. + +All dependencies can have provided scope. The versions are taken from the above +defined BOM. + +As we are going to be deploying this application to the {productName} server, let's +also add a maven plugin that will simplify the deployment operations (you can replace +the generated build section): + +[source,xml] +---- + + + ${project.artifactId} + + + + org.wildfly.plugins + wildfly-maven-plugin + ${version.plugin.wildfly} + + + org.wildfly.plugins + wildfly-jar-maven-plugin + ${version.plugin.wildfly-jar} + + + + +---- + +// Setup required repositories +include::../shared-doc/setup-repositories.adoc[leveloffset=+1] + +Now we are ready to start working with MicroProfile LRA. + +== Set up JAX-RS server and result wrapper + +LRA works on top of JAX-RS. To set up JAX-RS server in our service, we need to create a new application class +`org.wildfly.quickstarts.microprofile.lra.JaxRsApplication` in the file +`microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/JaxRsApplication.java` that looks like this: + +[source,java] +---- +package org.wildfly.quickstarts.microprofile.lra; + +import jakarta.ws.rs.ApplicationPath; +import jakarta.ws.rs.core.Application; + +@ApplicationPath("/") +public class JaxRsApplication extends Application { +} +---- + +Now we can declare our LRA JAX-RS resources. + +The LRAs we're going to create also accumulate results in a wrapper called `ParticipantResult` which we can create in `org.wildfly.quickstarts.microprofile.lra.ParticipantResult` class: + +[source,java] +---- +package org.wildfly.quickstarts.microprofile.lra; + +public class ParticipantResult { + + private String workLRAId; + private String workRecoveryId; + private String completeLRAId; + private String completeRecoveryId; + private String compensateLRAId; + private String compensateRecoveryId; + + public ParticipantResult() {} + + public ParticipantResult(String workLRAId, String workRecoveryId, + String completeLRAId, String completeRecoveryId, + String compensateLRAId, String compensateRecoveryId) { + this.workLRAId = workLRAId; + this.workRecoveryId = workRecoveryId; + this.completeLRAId = completeLRAId; + this.completeRecoveryId = completeRecoveryId; + this.compensateLRAId = compensateLRAId; + this.compensateRecoveryId = compensateRecoveryId; + } + + public String getWorkLRAId() { + return workLRAId; + } + + public void setWorkLRAId(String workLRAId) { + this.workLRAId = workLRAId; + } + + public String getWorkRecoveryId() { + return workRecoveryId; + } + + public void setWorkRecoveryId(String workRecoveryId) { + this.workRecoveryId = workRecoveryId; + } + + public String getCompleteLRAId() { + return completeLRAId; + } + + public void setCompleteLRAId(String completeLRAId) { + this.completeLRAId = completeLRAId; + } + + public String getCompleteRecoveryId() { + return completeRecoveryId; + } + + public void setCompleteRecoveryId(String completeRecoveryId) { + this.completeRecoveryId = completeRecoveryId; + } + + public String getCompensateLRAId() { + return compensateLRAId; + } + + public void setCompensateLRAId(String compensateLRAId) { + this.compensateLRAId = compensateLRAId; + } + + public String getCompensateRecoveryId() { + return compensateRecoveryId; + } + + public void setCompensateRecoveryId(String compensateRecoveryId) { + this.compensateRecoveryId = compensateRecoveryId; + } + + @Override + public String toString() { + return "ParticipantResult{" + + "workLRAId='" + workLRAId + '\'' + + ", workRecoveryId='" + workRecoveryId + '\'' + + ", completeLRAId='" + completeLRAId + '\'' + + ", completeRecoveryId='" + completeRecoveryId + '\'' + + ", compensateLRAId='" + compensateLRAId + '\'' + + ", compensateRecoveryId='" + compensateRecoveryId + '\'' + + '}'; + } +} +---- + +== Creating LRA participants + +In LRA, we define LRA execution and participation with the same `@LRA` annotation. If placed on a method, it acts +similarly to `@Transactional` annotation from JTA. By default, it uses the `REQUIRED` LRA type meaning new LRA is +started or existing LRA (if passed to the invocation) is joined before the method is started. The LRA is also closed +(success) or cancelled (failure/exception) at the end of the method. + +LRA currently works on top of the JAX-RS resources. We can place `@LRA` annotation on any JAX-RS method and the LRA +is already managed for us by {productName}. Let's create a simple JAX-RS resource that uses lra in `org.wildfly .quickstarts.microprofile.lra.LRAParticipant1`: + +[source,java] +---- +/* + * JBoss, Home of Professional Open Source. + * Copyright 2023, Red Hat, Inc., and individual contributors + * as indicated by the @author tags. See the copyright.txt file in the + * distribution for a full listing of individual contributors. + * + * This is free software; you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1 of + * the License, or (at your option) any later version. + * + * This software is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this software; if not, write to the Free + * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA + * 02110-1301 USA, or see the FSF site: http://www.fsf.org. + */ + +package org.wildfly.quickstarts.microprofile.lra; + +import jakarta.enterprise.context.ApplicationScoped; +import jakarta.ws.rs.GET; +import jakarta.ws.rs.HeaderParam; +import jakarta.ws.rs.PUT; +import jakarta.ws.rs.Path; +import jakarta.ws.rs.Produces; +import jakarta.ws.rs.QueryParam; +import jakarta.ws.rs.client.Client; +import jakarta.ws.rs.client.ClientBuilder; +import jakarta.ws.rs.core.Context; +import jakarta.ws.rs.core.MediaType; +import jakarta.ws.rs.core.Response; +import jakarta.ws.rs.core.UriInfo; +import org.eclipse.microprofile.lra.annotation.Compensate; +import org.eclipse.microprofile.lra.annotation.Complete; +import org.eclipse.microprofile.lra.annotation.ws.rs.LRA; +import org.jboss.logging.Logger; + +import java.net.URI; + +@Path("/participant1") +@ApplicationScoped +public class LRAParticipant1 { + + private static final Logger LOGGER = Logger.getLogger(LRAParticipant1.class); + + private String workLRAId; + private String workRecoveryId; + private String completeLRAId; + private String completeRecoveryId; + private String compensateLRAId; + private String compensateRecoveryId; + + @Context + UriInfo uriInfo; + + @LRA + @GET + @Path("/work") + public Response work(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId, + @QueryParam("failLRA") boolean failLRA) { + LOGGER.infof("Executing action of Participant 1 enlisted in LRA %s " + + "that was assigned %s participant Id.", lraId, participantId); + + workLRAId = lraId.toASCIIString(); + workRecoveryId = participantId.toASCIIString(); + compensateLRAId = null; + compensateRecoveryId = null; + completeLRAId = null; + completeRecoveryId = null; + + return failLRA ? Response.status(Response.Status.INTERNAL_SERVER_ERROR).entity(lraId.toASCIIString()).build() : + Response.ok(lraId.toASCIIString()).build(); + } + + @Compensate + @PUT + @Path("/compensate") + public Response compensateWork(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId) { + LOGGER.infof("Compensating action for Participant 1 (%s) in LRA %s.", participantId, lraId); + + compensateLRAId = lraId.toASCIIString(); + compensateRecoveryId = participantId.toASCIIString(); + + return Response.ok().build(); + } + + @Complete + @PUT + @Path("/complete") + public Response completeWork(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId) { + LOGGER.infof("Complete action for Participant 1 (%s) in LRA %s.", participantId, lraId); + + completeLRAId = lraId.toASCIIString(); + completeRecoveryId = participantId.toASCIIString(); + + return Response.ok().build(); + } + + @GET + @Path("/result") + @Produces(MediaType.APPLICATION_JSON) + public ParticipantResult getParticipantResult() { + return new ParticipantResult(workLRAId, workRecoveryId, + completeLRAId, completeRecoveryId, + compensateLRAId, compensateRecoveryId); + } +} +---- + +Let's look at it part by part. + +The most important method is called `work` and it looks like this: + +[source,java] +---- +@LRA +@GET +@Path("/work") +public Response work(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId, + @QueryParam("failLRA") boolean failLRA) { + LOGGER.infof("Executing action of Participant 1 enlisted in LRA %s " + + "that was assigned %s participant Id.", lraId, participantId); + + workLRAId = lraId.toASCIIString(); + workRecoveryId = participantId.toASCIIString(); + compensateLRAId = null; + compensateRecoveryId = null; + completeLRAId = null; + completeRecoveryId = null; + + return failLRA ? Response.status(Response.Status.INTERNAL_SERVER_ERROR).entity(lraId.toASCIIString()).build() : + Response.ok(lraId.toASCIIString()).build(); +} +---- + +In this `GET` JAX-RS method, we also use the `@LRA` annotation that either starts a new LRA or joins an existing one +which is defined by the default LRA type `REQUIRED`. This is decided based on the `LRA.LRA_HTTP_CONTEXT_HEADER` header +we called `lraId`. If the framework starts a new LRA, +this header is automatically populated with its ID. If the caller specifies this `LRA.LRA_HTTP_CONTEXT_HEADER` +manually in the request, the received LRA is joined. As you can see, the LRA context or ID is propagated in HTTP +headers. + +The second header parameter `LRA.LRA_HTTP_RECOVERY_HEADER` is considered a unique participant ID for a particular +enlistment within LRA. If we would like to enlist `LRAParticipant1` in the same LRA (`LRA.LRA_HTTP_CONTEXT_HEADER`) +multiple times, this recovery ID would be different so we can associate the invocations of compensate and complete +methods. + +Each LRA participant needs to define the `@Compensate` method that defines the compensating action. + +[source,java] +---- +@Compensate +@PUT +@Path("/compensate") +public Response compensateWork(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId) { + LOGGER.infof("Compensating action for Participant 1 (%s) in LRA %s.", participantId, lraId); + + compensateLRAId = lraId.toASCIIString(); + compensateRecoveryId = participantId.toASCIIString(); + + return Response.ok().build(); +} +---- + +The compensation is defined by the `@Compensate` annotation which needs to be placed on the JAX-RS PUT method so the LRA +coordinator knows how to call it. For simplicity, we are just printing the messages to the console. The participant can +control how it finishes its participation in LRA via the returned status code. Please see the +link:https://github.com/eclipse/microprofile-lra/blob/main/spec/src/main/asciidoc/microprofile-lra-spec.asciidoc[specification] +for more details. + +The complete method looks similarly. It uses the `@Complete` annotation and it also needs to be the JAX-RS PUT method. + +[source,java] +---- +@Complete +@PUT +@Path("/complete") +public Response completeWork(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId) { + LOGGER.infof("Complete action for Participant 1 (%s) in LRA %s.", participantId, lraId); + + completeLRAId = lraId.toASCIIString(); + completeRecoveryId = participantId.toASCIIString(); + + return Response.ok().build(); +} +---- + +The LRA coordinator invokes the `@Compensate` method when the LRA cancels on failure and it invokes the `@Complete` +method when the LRA closes successfully. + +NOTE: The `@Complete` and `@Compensate` methods don't need to be JAX-RS methods. See the specification for details. + +Now we are already able to start our first LRA. You can deploy the application to the {productName} as demonstrated in +the <> section. Remember that you need to enable the LRA extensions and subsystems with the +`enable-microprofile-lra.cli` script. + +Then you can invoke the `LRAParticipant1` JAX-RS resource as: + +[source,bash] +---- +$ curl http://localhost:8080/microprofile-lra/participant1/work +---- + +or if you want to simulate LRA failure as: + +[source,bash] +---- +$ curl "http://localhost:8080/microprofile-lra/participant1/work?failLRA=true" +---- + +In either case, you will see the LRA execution message printed in the {productName} console: + +[source,bash] +---- +INFO [org.wildfly.quickstarts.microprofile.lra.LRAParticipant1] (default task-1) Executing action of Participant 1 enlisted in LRA http://localhost:8080/lra-coordinator/lra-coordinator/0_ffff0aca949a_-4998614b_64e74427_48 that was assigned http://localhost:8080/lra-coordinator/lra-coordinator/recoveryhttp%3A%2F%2Flocalhost%3A8080%2Flra-coordinator%2Flra-coordinator%2F0_ffff0aca949a_-4998614b_64e74427_48/0_ffff0aca949a_-4998614b_64e74427_4a participant Id. +---- + +And either the complete or compensate message depending on the `failLRA` paramater that can fail the LRA causing it +to cancel: + +[source,bash] +---- +INFO [org.wildfly.quickstarts.microprofile.lra.LRAParticipant1] (default task-4) Complete action for Participant 1 (http://localhost:8080/lra-coordinator/lra-coordinator/recoveryhttp%3A%2F%2Flocalhost%3A8080%2Flra-coordinator%2Flra-coordinator%2F0_ffff0aca949a_-4998614b_64e74427_37/0_ffff0aca949a_-4998614b_64e74427_39) in LRA http://localhost:8080/lra-coordinator/lra-coordinator/0_ffff0aca949a_-4998614b_64e74427_37. + + +INFO [org.wildfly.quickstarts.microprofile.lra.LRAParticipant1] (default task-4) Compensating action for Participant 1 (http://localhost:8080/lra-coordinator/lra-coordinator/recoveryhttp%3A%2F%2Flocalhost%3A8080%2Flra-coordinator%2Flra-coordinator%2F0_ffff0aca949a_-4998614b_64e74427_48/0_ffff0aca949a_-4998614b_64e74427_4a) in LRA http://localhost:8080/lra-coordinator/lra-coordinator/0_ffff0aca949a_-4998614b_64e74427_48. +---- + +== Multiple participants in the LRA + +One participant that starts and ends the LRA is probably enough to demonstrate the functionality, but it rarely makes +sense in distributed microservices architecture to only have one service that participates in a distributed +transaction. So let's add another participant into the LRA started in the `LRAParticipant1`. + +Copy the `LRAParticipant1` into a new class `LRAParticipant2` and change all references to `participant1` to +`participant2`. +The +full class is provided for convenience also here: + +[source,java] +---- +/* + * JBoss, Home of Professional Open Source. + * Copyright 2023, Red Hat, Inc., and individual contributors + * as indicated by the @author tags. See the copyright.txt file in the + * distribution for a full listing of individual contributors. + * + * This is free software; you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1 of + * the License, or (at your option) any later version. + * + * This software is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this software; if not, write to the Free + * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA + * 02110-1301 USA, or see the FSF site: http://www.fsf.org. + */ + +package org.wildfly.quickstarts.microprofile.lra; + +import jakarta.enterprise.context.ApplicationScoped; +import jakarta.ws.rs.GET; +import jakarta.ws.rs.HeaderParam; +import jakarta.ws.rs.PUT; +import jakarta.ws.rs.Path; +import jakarta.ws.rs.Produces; +import jakarta.ws.rs.core.MediaType; +import jakarta.ws.rs.core.Response; +import org.eclipse.microprofile.lra.annotation.Compensate; +import org.eclipse.microprofile.lra.annotation.Complete; +import org.eclipse.microprofile.lra.annotation.ws.rs.LRA; +import org.jboss.logging.Logger; + +import java.net.URI; + +@Path("/participant2") +@ApplicationScoped +public class LRAParticipant2 { + + private static final Logger LOGGER = Logger.getLogger(LRAParticipant2.class); + + private String workLRAId; + private String workRecoveryId; + private String completeLRAId; + private String completeRecoveryId; + private String compensateLRAId; + private String compensateRecoveryId; + + @LRA(end = false) + @GET + @Path("/work") + public Response work(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId) { + LOGGER.infof("Executing action of Participant 2 enlisted in LRA %s " + + "that was assigned %s participant Id.", lraId, participantId); + + workLRAId = lraId.toASCIIString(); + workRecoveryId = participantId.toASCIIString(); + compensateLRAId = null; + compensateRecoveryId = null; + completeLRAId = null; + completeRecoveryId = null; + + return Response.ok().build(); + } + + @Compensate + @PUT + @Path("/compensate") + public Response compensateWork(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId) { + LOGGER.infof("Compensating action for Participant 2 (%s) in LRA %s.", participantId, lraId); + + compensateLRAId = lraId.toASCIIString(); + compensateRecoveryId = participantId.toASCIIString(); + + return Response.ok().build(); + } + + @Complete + @PUT + @Path("/complete") + public Response completeWork(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId) { + LOGGER.infof("Complete action for Participant 2 (%s) in LRA %s.", participantId, lraId); + + completeLRAId = lraId.toASCIIString(); + completeRecoveryId = participantId.toASCIIString(); + + return Response.ok().build(); + } + + @GET + @Path("/result") + @Produces(MediaType.APPLICATION_JSON) + public ParticipantResult getParticipantResult() { + return new ParticipantResult(workLRAId, workRecoveryId, + completeLRAId, completeRecoveryId, + compensateLRAId, compensateRecoveryId); + } + +} +---- + +The only notable change is the `LRA` annotation that now contains the `@LRA(end = false)`. This parameter states that +the LRA should not be ended when this business method ends. If we ended the LRA here, it would still invoke +compensate or complete callbacks on all enlisted participants (including `LRAParticipant1` which will propagate the +LRA into this class soon). However, it would also try to close/cancel LRA at the end of the `LRAParticipant1#work` +method and by this time the LRA would already be ended. This would be reported by the coordinator. + +We also need to add the call to the newly created JAX-RS resource to the `LRAParticipant1#work` method as showed in +this +snipped: + +[source,java] +---- +@LRA +@GET +@Path("/work") +public Response work(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId, + @QueryParam("failLRA") boolean failLRA) { + LOGGER.infof("Executing action of Participant 1 enlisted in LRA %s " + + "that was assigned %s participant Id.", lraId, participantId); + + workLRAId = lraId.toASCIIString(); + workRecoveryId = participantId.toASCIIString(); + compensateLRAId = null; + compensateRecoveryId = null; + completeLRAId = null; + completeRecoveryId = null; + + // call Participant 2 to propagate the LRA + try (Client client = ClientBuilder.newClient()) { + client.target(uriInfo.getBaseUri() + "/participant2/work") + .request().get(); + } + + return failLRA ? Response.status(Response.Status.INTERNAL_SERVER_ERROR).entity(lraId.toASCIIString()).build() : + Response.ok(lraId.toASCIIString()).build(); +} +---- + +You might remember that we need to propagate the LRA id (LRA context) in the `LRA.LRA_HTTP_CONTEXT_HEADER`. However, if +we make the outgoing JAX-RS call in the JAX-RS method that already carries an active LRA context, the context is +automatically added to the outgoing call. So we don't need to pass it manually to each outgoing call. + +Now we are ready to propagate LRA started in Participant 1 to the Participant 2, enlist both in the newly started +LRA, and finish the LRA when the Participant 1 ends its `work` method. + +Redeploy the application into the {productName} as showed in <>. Then you can repeat the calls to the +`LRAParticipant1` JAX-RS resource as we used them previously: + +[source,bash] +---- +$ curl http://localhost:8080/microprofile-lra/participant1/work +---- + +or if you want to simulate LRA failure as: + +[source,bash] +---- +$ curl "http://localhost:8080/microprofile-lra/participant1/work?failLRA=true" +---- + +But this time, you will see the LRA is propagated to the `LRAParticipant2` and its (complete or compensate) callbacks +are invoked by the LRA coordinator in the same way as for `LRAParticipant1`: + +[source,bash] +---- +INFO [org.wildfly.quickstarts.microprofile.lra.LRAParticipant1] (default task-1) Executing action of Participant 1 enlisted in LRA http://localhost:8080/lra-coordinator/lra-coordinator/0_ffff0aca949a_-4998614b_64e74427_38b that was assigned http://localhost:8080/lra-coordinator/lra-coordinator/recoveryhttp%3A%2F%2Flocalhost%3A8080%2Flra-coordinator%2Flra-coordinator%2F0_ffff0aca949a_-4998614b_64e74427_38b/0_ffff0aca949a_-4998614b_64e74427_38d participant Id. + +INFO [org.wildfly.quickstarts.microprofile.lra.LRAParticipant2] (default task-2) Executing action of Participant 2 enlisted in LRA http://localhost:8080/lra-coordinator/lra-coordinator/0_ffff0aca949a_-4998614b_64e74427_38b that was assigned http://localhost:8080/lra-coordinator/lra-coordinator/recoveryhttp%3A%2F%2Flocalhost%3A8080%2Flra-coordinator%2Flra-coordinator%2F0_ffff0aca949a_-4998614b_64e74427_38b/0_ffff0aca949a_-4998614b_64e74427_38f participant Id. + +INFO [org.wildfly.quickstarts.microprofile.lra.LRAParticipant1] (default task-5) Compensating action for Participant 1 (http://localhost:8080/lra-coordinator/lra-coordinator/recoveryhttp%3A%2F%2Flocalhost%3A8080%2Flra-coordinator%2Flra-coordinator%2F0_ffff0aca949a_-4998614b_64e74427_38b/0_ffff0aca949a_-4998614b_64e74427_38d) in LRA http://localhost:8080/lra-coordinator/lra-coordinator/0_ffff0aca949a_-4998614b_64e74427_38b. + +INFO [org.wildfly.quickstarts.microprofile.lra.LRAParticipant2] (default task-5) Compensating action for Participant 2 (http://localhost:8080/lra-coordinator/lra-coordinator/recoveryhttp%3A%2F%2Flocalhost%3A8080%2Flra-coordinator%2Flra-coordinator%2F0_ffff0aca949a_-4998614b_64e74427_38b/0_ffff0aca949a_-4998614b_64e74427_38f) in LRA http://localhost:8080/lra-coordinator/lra-coordinator/0_ffff0aca949a_-4998614b_64e74427_38b. +---- + +== Conclusion + +MicroProfile LRA provides a simple API for the distributed transactions based on the saga pattern. To use it on {productName} we need to enable the appropriate extensions and subsystems for the LRA Coordinator (a service that +manages LRAs) and the LRA participant (client API). The LRAs are controlled through annotations provided by the +specification. + +Congratulations! You have reached the end of this tutorial. You can find more information +about the MicroProfile LRA in the specification https://github.com/eclipse/microprofile-lra[github repository]. diff --git a/microprofile-lra/charts/helm.yaml b/microprofile-lra/charts/helm.yaml new file mode 100644 index 0000000000..61fba52af3 --- /dev/null +++ b/microprofile-lra/charts/helm.yaml @@ -0,0 +1,6 @@ +build: + uri: https://github.com/wildfly/quickstart.git + ref: main + contextDir: microprofile-lra +deploy: + replicas: 1 \ No newline at end of file diff --git a/microprofile-lra/enable-microprofile-lra.cli b/microprofile-lra/enable-microprofile-lra.cli new file mode 100644 index 0000000000..c563814a27 --- /dev/null +++ b/microprofile-lra/enable-microprofile-lra.cli @@ -0,0 +1,8 @@ +batch + +/extension=org.wildfly.extension.microprofile.lra-coordinator:add +/extension=org.wildfly.extension.microprofile.lra-participant:add +/subsystem=microprofile-lra-coordinator:add +/subsystem=microprofile-lra-participant:add + +run-batch diff --git a/microprofile-lra/pom.xml b/microprofile-lra/pom.xml new file mode 100644 index 0000000000..382db54304 --- /dev/null +++ b/microprofile-lra/pom.xml @@ -0,0 +1,255 @@ + + 4.0.0 + + org.wildfly.quickstarts + wildfly-quickstart-parent + + 5 + + + + microprofile-lra + 31.0.0.Beta1-SNAPSHOT + war + Quickstart: microprofile-lra + + + + 30.0.0.Final + + ${version.server} + ${version.server} + 5.0.0.Final + 4.2.0.Final + 10.0.0.Final + + + + + jboss-public-maven-repository + JBoss Public Maven Repository + https://repository.jboss.org/nexus/content/groups/public/ + + true + never + + + true + never + + default + + + redhat-ga-maven-repository + Red Hat GA Maven Repository + https://maven.repository.redhat.com/ga/ + + true + never + + + true + never + + default + + + + + jboss-public-maven-repository + JBoss Public Maven Repository + https://repository.jboss.org/nexus/content/groups/public/ + + true + + + true + + + + redhat-ga-maven-repository + Red Hat GA Maven Repository + https://maven.repository.redhat.com/ga/ + + true + + + true + + + + + + + + org.wildfly.bom + wildfly-ee-with-tools + ${version.bom.ee} + pom + import + + + org.wildfly.bom + wildfly-microprofile + ${version.bom.microprofile} + pom + import + + + + + + + org.eclipse.microprofile.lra + microprofile-lra-api + provided + + + jakarta.ws.rs + jakarta.ws.rs-api + provided + + + jakarta.enterprise + jakarta.enterprise.cdi-api + provided + + + org.jboss.logging + jboss-logging + provided + + + + + junit + junit + test + + + org.jboss.resteasy + resteasy-client + test + + + org.jboss.resteasy + resteasy-jackson2-provider + test + + + + + + ${project.artifactId} + + + + org.wildfly.plugins + wildfly-maven-plugin + ${version.plugin.wildfly} + + + org.wildfly.plugins + wildfly-jar-maven-plugin + ${version.plugin.wildfly-jar} + + + + + + + + bootable-jar + + + + org.wildfly.plugins + wildfly-jar-maven-plugin + + wildfly@maven(org.jboss.universe:community-universe)#${version.server} + + jaxrs-server + microprofile-lra-coordinator + microprofile-lra-participant + + + true + + + + + + package + + + + + + + + + openshift + + + + org.wildfly.plugins + wildfly-maven-plugin + + + + org.wildfly:wildfly-galleon-pack:${version.server} + + + org.wildfly.cloud:wildfly-cloud-galleon-pack:${version.pack.cloud} + + + + jaxrs-server + microprofile-lra-coordinator + microprofile-lra-participant + + ROOT.war + + + + + package + + + + + + + + + integration-testing + + + + org.apache.maven.plugins + maven-failsafe-plugin + + + **/BasicRuntimeIT + **/MicroProfileLRAIT + + + + + + integration-test + verify + + + + + + + + + + diff --git a/microprofile-lra/restore-configuration.cli b/microprofile-lra/restore-configuration.cli new file mode 100644 index 0000000000..2205990669 --- /dev/null +++ b/microprofile-lra/restore-configuration.cli @@ -0,0 +1,12 @@ +# This script restores the configuration with the enabled MicroProfile LRA extensions and subsystems. + +batch + +/subsystem=microprofile-lra-participant:remove +/subsystem=microprofile-lra-coordinator:remove +/extension=org.wildfly.extension.microprofile.lra-participant:remove +/extension=org.wildfly.extension.microprofile.lra-coordinator:remove + +run-batch + +reload diff --git a/microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/JaxRsApplication.java b/microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/JaxRsApplication.java new file mode 100644 index 0000000000..49b9f96676 --- /dev/null +++ b/microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/JaxRsApplication.java @@ -0,0 +1,30 @@ +/* + * JBoss, Home of Professional Open Source. + * Copyright 2023, Red Hat, Inc., and individual contributors + * as indicated by the @author tags. See the copyright.txt file in the + * distribution for a full listing of individual contributors. + * + * This is free software; you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1 of + * the License, or (at your option) any later version. + * + * This software is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this software; if not, write to the Free + * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA + * 02110-1301 USA, or see the FSF site: http://www.fsf.org. + */ + +package org.wildfly.quickstarts.microprofile.lra; + +import jakarta.ws.rs.ApplicationPath; +import jakarta.ws.rs.core.Application; + +@ApplicationPath("/") +public class JaxRsApplication extends Application { +} diff --git a/microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/LRAParticipant1.java b/microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/LRAParticipant1.java new file mode 100644 index 0000000000..7b66902856 --- /dev/null +++ b/microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/LRAParticipant1.java @@ -0,0 +1,121 @@ +/* + * JBoss, Home of Professional Open Source. + * Copyright 2023, Red Hat, Inc., and individual contributors + * as indicated by the @author tags. See the copyright.txt file in the + * distribution for a full listing of individual contributors. + * + * This is free software; you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1 of + * the License, or (at your option) any later version. + * + * This software is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this software; if not, write to the Free + * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA + * 02110-1301 USA, or see the FSF site: http://www.fsf.org. + */ + +package org.wildfly.quickstarts.microprofile.lra; + +import jakarta.enterprise.context.ApplicationScoped; +import jakarta.ws.rs.GET; +import jakarta.ws.rs.HeaderParam; +import jakarta.ws.rs.PUT; +import jakarta.ws.rs.Path; +import jakarta.ws.rs.Produces; +import jakarta.ws.rs.QueryParam; +import jakarta.ws.rs.client.Client; +import jakarta.ws.rs.client.ClientBuilder; +import jakarta.ws.rs.core.Context; +import jakarta.ws.rs.core.MediaType; +import jakarta.ws.rs.core.Response; +import jakarta.ws.rs.core.UriInfo; +import org.eclipse.microprofile.lra.annotation.Compensate; +import org.eclipse.microprofile.lra.annotation.Complete; +import org.eclipse.microprofile.lra.annotation.ws.rs.LRA; +import org.jboss.logging.Logger; + +import java.net.URI; + +@Path("/participant1") +@ApplicationScoped +public class LRAParticipant1 { + + private static final Logger LOGGER = Logger.getLogger(LRAParticipant1.class); + + private String workLRAId; + private String workRecoveryId; + private String completeLRAId; + private String completeRecoveryId; + private String compensateLRAId; + private String compensateRecoveryId; + + @Context + UriInfo uriInfo; + + @LRA + @GET + @Path("/work") + public Response work(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId, + @QueryParam("failLRA") boolean failLRA) { + LOGGER.infof("Executing action of Participant 1 enlisted in LRA %s " + + "that was assigned %s participant Id.", lraId, participantId); + + workLRAId = lraId.toASCIIString(); + workRecoveryId = participantId.toASCIIString(); + compensateLRAId = null; + compensateRecoveryId = null; + completeLRAId = null; + completeRecoveryId = null; + + // call Participant 2 to propagate the LRA + try (Client client = ClientBuilder.newClient()) { + client.target(uriInfo.getBaseUri() + "/participant2/work") + .request().get(); + } + + return failLRA ? Response.status(Response.Status.INTERNAL_SERVER_ERROR).entity(lraId.toASCIIString()).build() : + Response.ok(lraId.toASCIIString()).build(); + } + + @Compensate + @PUT + @Path("/compensate") + public Response compensateWork(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId) { + LOGGER.infof("Compensating action for Participant 1 (%s) in LRA %s.", participantId, lraId); + + compensateLRAId = lraId.toASCIIString(); + compensateRecoveryId = participantId.toASCIIString(); + + return Response.ok().build(); + } + + @Complete + @PUT + @Path("/complete") + public Response completeWork(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId) { + LOGGER.infof("Complete action for Participant 1 (%s) in LRA %s.", participantId, lraId); + + completeLRAId = lraId.toASCIIString(); + completeRecoveryId = participantId.toASCIIString(); + + return Response.ok().build(); + } + + @GET + @Path("/result") + @Produces(MediaType.APPLICATION_JSON) + public ParticipantResult getParticipantResult() { + return new ParticipantResult(workLRAId, workRecoveryId, + completeLRAId, completeRecoveryId, + compensateLRAId, compensateRecoveryId); + } +} diff --git a/microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/LRAParticipant2.java b/microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/LRAParticipant2.java new file mode 100644 index 0000000000..60e67c421e --- /dev/null +++ b/microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/LRAParticipant2.java @@ -0,0 +1,106 @@ +/* + * JBoss, Home of Professional Open Source. + * Copyright 2023, Red Hat, Inc., and individual contributors + * as indicated by the @author tags. See the copyright.txt file in the + * distribution for a full listing of individual contributors. + * + * This is free software; you can redistribute it and/or modify it + * under the terms of the GNU Lesser General Public License as + * published by the Free Software Foundation; either version 2.1 of + * the License, or (at your option) any later version. + * + * This software is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + * Lesser General Public License for more details. + * + * You should have received a copy of the GNU Lesser General Public + * License along with this software; if not, write to the Free + * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA + * 02110-1301 USA, or see the FSF site: http://www.fsf.org. + */ + +package org.wildfly.quickstarts.microprofile.lra; + +import jakarta.enterprise.context.ApplicationScoped; +import jakarta.ws.rs.GET; +import jakarta.ws.rs.HeaderParam; +import jakarta.ws.rs.PUT; +import jakarta.ws.rs.Path; +import jakarta.ws.rs.Produces; +import jakarta.ws.rs.core.MediaType; +import jakarta.ws.rs.core.Response; +import org.eclipse.microprofile.lra.annotation.Compensate; +import org.eclipse.microprofile.lra.annotation.Complete; +import org.eclipse.microprofile.lra.annotation.ws.rs.LRA; +import org.jboss.logging.Logger; + +import java.net.URI; + +@Path("/participant2") +@ApplicationScoped +public class LRAParticipant2 { + + private static final Logger LOGGER = Logger.getLogger(LRAParticipant2.class); + + private String workLRAId; + private String workRecoveryId; + private String completeLRAId; + private String completeRecoveryId; + private String compensateLRAId; + private String compensateRecoveryId; + + @LRA(end = false) + @GET + @Path("/work") + public Response work(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId) { + LOGGER.infof("Executing action of Participant 2 enlisted in LRA %s " + + "that was assigned %s participant Id.", lraId, participantId); + + workLRAId = lraId.toASCIIString(); + workRecoveryId = participantId.toASCIIString(); + compensateLRAId = null; + compensateRecoveryId = null; + completeLRAId = null; + completeRecoveryId = null; + + return Response.ok().build(); + } + + @Compensate + @PUT + @Path("/compensate") + public Response compensateWork(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId) { + LOGGER.infof("Compensating action for Participant 2 (%s) in LRA %s.", participantId, lraId); + + compensateLRAId = lraId.toASCIIString(); + compensateRecoveryId = participantId.toASCIIString(); + + return Response.ok().build(); + } + + @Complete + @PUT + @Path("/complete") + public Response completeWork(@HeaderParam(LRA.LRA_HTTP_CONTEXT_HEADER) URI lraId, + @HeaderParam(LRA.LRA_HTTP_RECOVERY_HEADER) URI participantId) { + LOGGER.infof("Complete action for Participant 2 (%s) in LRA %s.", participantId, lraId); + + completeLRAId = lraId.toASCIIString(); + completeRecoveryId = participantId.toASCIIString(); + + return Response.ok().build(); + } + + @GET + @Path("/result") + @Produces(MediaType.APPLICATION_JSON) + public ParticipantResult getParticipantResult() { + return new ParticipantResult(workLRAId, workRecoveryId, + completeLRAId, completeRecoveryId, + compensateLRAId, compensateRecoveryId); + } + +} diff --git a/microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/ParticipantResult.java b/microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/ParticipantResult.java new file mode 100644 index 0000000000..720f429233 --- /dev/null +++ b/microprofile-lra/src/main/java/org/wildfly/quickstarts/microprofile/lra/ParticipantResult.java @@ -0,0 +1,84 @@ +package org.wildfly.quickstarts.microprofile.lra; + +public class ParticipantResult { + + private String workLRAId; + private String workRecoveryId; + private String completeLRAId; + private String completeRecoveryId; + private String compensateLRAId; + private String compensateRecoveryId; + + public ParticipantResult() {} + + public ParticipantResult(String workLRAId, String workRecoveryId, + String completeLRAId, String completeRecoveryId, + String compensateLRAId, String compensateRecoveryId) { + this.workLRAId = workLRAId; + this.workRecoveryId = workRecoveryId; + this.completeLRAId = completeLRAId; + this.completeRecoveryId = completeRecoveryId; + this.compensateLRAId = compensateLRAId; + this.compensateRecoveryId = compensateRecoveryId; + } + + public String getWorkLRAId() { + return workLRAId; + } + + public void setWorkLRAId(String workLRAId) { + this.workLRAId = workLRAId; + } + + public String getWorkRecoveryId() { + return workRecoveryId; + } + + public void setWorkRecoveryId(String workRecoveryId) { + this.workRecoveryId = workRecoveryId; + } + + public String getCompleteLRAId() { + return completeLRAId; + } + + public void setCompleteLRAId(String completeLRAId) { + this.completeLRAId = completeLRAId; + } + + public String getCompleteRecoveryId() { + return completeRecoveryId; + } + + public void setCompleteRecoveryId(String completeRecoveryId) { + this.completeRecoveryId = completeRecoveryId; + } + + public String getCompensateLRAId() { + return compensateLRAId; + } + + public void setCompensateLRAId(String compensateLRAId) { + this.compensateLRAId = compensateLRAId; + } + + public String getCompensateRecoveryId() { + return compensateRecoveryId; + } + + public void setCompensateRecoveryId(String compensateRecoveryId) { + this.compensateRecoveryId = compensateRecoveryId; + } + + @Override + public String toString() { + return "ParticipantResult{" + + "workLRAId='" + workLRAId + '\'' + + ", workRecoveryId='" + workRecoveryId + '\'' + + ", completeLRAId='" + completeLRAId + '\'' + + ", completeRecoveryId='" + completeRecoveryId + '\'' + + ", compensateLRAId='" + compensateLRAId + '\'' + + ", compensateRecoveryId='" + compensateRecoveryId + '\'' + + '}'; + } +} diff --git a/microprofile-lra/src/test/java/org/wildfly/quickstarts/microprofile/lra/BasicRuntimeIT.java b/microprofile-lra/src/test/java/org/wildfly/quickstarts/microprofile/lra/BasicRuntimeIT.java new file mode 100644 index 0000000000..4c0124cd32 --- /dev/null +++ b/microprofile-lra/src/test/java/org/wildfly/quickstarts/microprofile/lra/BasicRuntimeIT.java @@ -0,0 +1,50 @@ +/* + * Copyright 2023 JBoss by Red Hat. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.wildfly.quickstarts.microprofile.lra; + +import org.junit.Test; + +import java.io.IOException; +import java.net.URI; +import java.net.URISyntaxException; +import java.net.http.HttpClient; +import java.net.http.HttpRequest; +import java.net.http.HttpResponse; +import java.time.Duration; + +import static org.junit.Assert.assertEquals; + +/** + * The very basic runtime integration testing. + * @author emartins + */ +public class BasicRuntimeIT { + + @Test + public void testHTTPEndpointIsAvailable() throws IOException, InterruptedException, URISyntaxException { + String serverHost = TestUtils.getServerHost() + "/participant1/work"; + final HttpRequest request = HttpRequest.newBuilder() + .uri(new URI(serverHost)) + .GET() + .build(); + final HttpClient client = HttpClient.newBuilder() + .followRedirects(HttpClient.Redirect.ALWAYS) + .connectTimeout(Duration.ofMinutes(1)) + .build(); + final HttpResponse response = client.send(request, HttpResponse.BodyHandlers.ofString()); + assertEquals(200, response.statusCode()); + } +} \ No newline at end of file diff --git a/microprofile-lra/src/test/java/org/wildfly/quickstarts/microprofile/lra/MicroProfileLRAIT.java b/microprofile-lra/src/test/java/org/wildfly/quickstarts/microprofile/lra/MicroProfileLRAIT.java new file mode 100644 index 0000000000..e85873aa8d --- /dev/null +++ b/microprofile-lra/src/test/java/org/wildfly/quickstarts/microprofile/lra/MicroProfileLRAIT.java @@ -0,0 +1,128 @@ +/* + * JBoss, Home of Professional Open Source + * Copyright 2023, Red Hat, Inc. and/or its affiliates, and individual + * contributors by the @authors tag. See the copyright.txt in the + * distribution for a full listing of individual contributors. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * http://www.apache.org/licenses/LICENSE-2.0 + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.wildfly.quickstarts.microprofile.lra; + +import jakarta.ws.rs.client.Client; +import jakarta.ws.rs.client.ClientBuilder; +import jakarta.ws.rs.client.WebTarget; +import jakarta.ws.rs.core.Response; +import org.junit.After; +import org.junit.Before; +import org.junit.Test; + +import java.util.function.Function; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertNull; +import static org.wildfly.quickstarts.microprofile.lra.TestUtils.getServerHost; + +public class MicroProfileLRAIT { + + private Client client; + + @Before + public void before() { + client = ClientBuilder.newClient(); + } + + @After + public void after() { + client.close(); + } + + @Test + public void testLRAExecutionSuccess() { + Response response = getResponse("/participant1/work"); + assertEquals(200, response.getStatus()); + String lraId = response.readEntity(String.class); + + response = getResponse("/participant1/result"); + assertEquals(200, response.getStatus()); + ParticipantResult participantResult1 = response.readEntity(ParticipantResult.class); + + response = getResponse("/participant2/result"); + assertEquals(200, response.getStatus()); + ParticipantResult participantResult2 = response.readEntity(ParticipantResult.class); + + assertEquals(lraId, participantResult1.getWorkLRAId()); + String recoveryId1 = participantResult1.getWorkRecoveryId(); + + assertEquals(lraId, participantResult2.getWorkLRAId()); + String recoveryId2 = participantResult2.getWorkRecoveryId(); + + // LRA closed successfully, Complete callbacks called + assertEquals(lraId, participantResult1.getCompleteLRAId()); + assertEquals(recoveryId1, participantResult1.getCompleteRecoveryId()); + assertEquals(lraId, participantResult2.getCompleteLRAId()); + assertEquals(recoveryId2, participantResult2.getCompleteRecoveryId()); + + // Compensate callbacks should not be called + assertNull(participantResult1.getCompensateLRAId()); + assertNull(participantResult1.getCompensateRecoveryId()); + assertNull(participantResult2.getCompensateLRAId()); + assertNull(participantResult2.getCompensateRecoveryId()); + } + + @Test + public void testLRAExecutionFailure() { + Response response = getResponse("/participant1/work", + webTarget -> webTarget.queryParam("failLRA", "true")); + assertEquals(500, response.getStatus()); + String lraId = response.readEntity(String.class); + + response = getResponse("/participant1/result"); + assertEquals(200, response.getStatus()); + ParticipantResult participantResult1 = response.readEntity(ParticipantResult.class); + + response = getResponse("/participant2/result"); + assertEquals(200, response.getStatus()); + ParticipantResult participantResult2 = response.readEntity(ParticipantResult.class); + + assertEquals(lraId, participantResult1.getWorkLRAId()); + String recoveryId1 = participantResult1.getWorkRecoveryId(); + + assertEquals(lraId, participantResult2.getWorkLRAId()); + String recoveryId2 = participantResult2.getWorkRecoveryId(); + + // LRA canceled on failure, Compensate callbacks called + assertEquals(lraId, participantResult1.getCompensateLRAId()); + assertEquals(recoveryId1, participantResult1.getCompensateRecoveryId()); + assertEquals(lraId, participantResult2.getCompensateLRAId()); + assertEquals(recoveryId2, participantResult2.getCompensateRecoveryId()); + + // Complete callbacks should not be called + assertNull(participantResult1.getCompleteLRAId()); + assertNull(participantResult1.getCompleteRecoveryId()); + assertNull(participantResult2.getCompleteLRAId()); + assertNull(participantResult2.getCompleteRecoveryId()); + } + + private Response getResponse(String path) { + return getResponse(path, null); + } + + private Response getResponse(String path, Function weTargetProcessor) { + WebTarget target = client.target(getServerHost()) + .path(path); + + if (weTargetProcessor != null) { + target = weTargetProcessor.apply(target); + } + + return target.request().get(); + } +} diff --git a/microprofile-lra/src/test/java/org/wildfly/quickstarts/microprofile/lra/TestUtils.java b/microprofile-lra/src/test/java/org/wildfly/quickstarts/microprofile/lra/TestUtils.java new file mode 100644 index 0000000000..a20614b286 --- /dev/null +++ b/microprofile-lra/src/test/java/org/wildfly/quickstarts/microprofile/lra/TestUtils.java @@ -0,0 +1,16 @@ +package org.wildfly.quickstarts.microprofile.lra; + +public class TestUtils { + static final String DEFAULT_SERVER_HOST = "http://localhost:8080/microprofile-lra"; + + static String getServerHost() { + String serverHost = System.getenv("SERVER_HOST"); + if (serverHost == null) { + serverHost = System.getProperty("server.host"); + } + if (serverHost == null) { + serverHost = DEFAULT_SERVER_HOST; + } + return serverHost; + } +} diff --git a/pom.xml b/pom.xml index a8866cf8fa..7a3701bfb7 100644 --- a/pom.xml +++ b/pom.xml @@ -334,6 +334,7 @@ microprofile-fault-tolerance microprofile-health microprofile-jwt + microprofile-lra microprofile-openapi microprofile-reactive-messaging-kafka microprofile-rest-client