45 MINUTE EXERCISE
In this lab you will learn about Eclipse Vert.x and how you can build microservices using reactive principles. During this lab you will create a scalable API Gateway that aggregates Catalog and Inventory APIs.
Eclipse Vert.x is a toolkit for building reactive applications on the Java Virtual Machine (JVM). Vert.x does not impose a specific framework or packaging model and can be used within your existing applications and frameworks in order to add reactive functionality by just adding the Vert.x jar files to the application classpath.
Eclipse Vert.x enables building reactive systems as defined by The Reactive Manifesto and build services that are:
-
Responsive: to handle requests in a reasonable time
-
Resilient: to stay responsive in the face of failures
-
Elastic: to stay responsive under various loads and be able to scale up and down
-
Message driven: components interact using asynchronous message-passing
Eclipse Vert.x is designed to be event-driven and non-blocking. Events are delivered in an event loop that must never be blocked. Unlike traditional applications, Vert.x uses a very small number of threads responsible for dispatching the events to event handlers. If the event loop is blocked, the events won’t be delivered anymore and therefore the code needs to be mindful of this execution model.
The gateway-vertx project has the following structure which shows the components of the Vert.x project laid out in different subdirectories according to Maven best practices:
This is a minimal Vert.x project with support for RESTful services. This project currently contains no code other than the main class, GatewayVerticle.java which is there to bootstrap the Vert.x application. Verticles are encapsulated parts of the application that can run completely independently and communicate with each other using HTTP. Verticles get deployed and run by Vert.x in an event loop and therefore it is important that the code in a Verticle does not block. This asynchronous architecture allows Vert.x applications to easily scale and handle large amounts of throughput with few threads.All API calls in Vert.x by default are non-blocking and support this concurrency model.
Although you can have multiple, there is currently only one Verticle created in the gateway-vertx project.
Examine 'GatewayVerticle' class
in the com.redhat.cloudnative.gateway package in the /projects/labs/gateway-vertx/src directory.
package com.redhat.cloudnative.gateway;
import io.vertx.core.Future;
import io.vertx.reactivex.core.AbstractVerticle;
import io.vertx.reactivex.ext.web.Router;
import io.vertx.reactivex.ext.web.handler.StaticHandler;
public class GatewayVerticle extends AbstractVerticle { // (1)
@Override
public void start(Future<Void> future) { // (2)
Router router = Router.router(vertx); // (3)
router.get("/*").handler(StaticHandler.create("assets")); // (4)
vertx.createHttpServer().requestHandler(router)
.listen(Integer.getInteger("http.port", 8080)); // (5)
}
}
-
A Verticle is created by extending from AbstractVerticle class
-
The start() method creates an HTTP server
-
Router is retrieved for mapping the REST endpoints
-
A REST endpoint is created for / to return a static HTML page assets/index.html
-
An HTTP Server is created which listens on port 8080
You can use Maven to make sure the skeleton project builds successfully. You should get a BUILD SUCCESS message in the build logs, otherwise the build has failed.
In CodeReady Workspaces, right click on 'gateway-vertx'
project in the project explorer then, click on 'Commands > Build > build'
Once successfully built, the resulting jar is located in the target/ directory:
$ ls /projects/labs/gateway-vertx/target/*.jar /projects/labs/gateway-vertx/target/gateway-1.0-SNAPSHOT.jar
This is an uber-jar with all the dependencies required packaged in the jar to enable running the
application with java -jar
.
In the previous labs, you have created two RESTful services: Catalog and Inventory. Instead of the web frontend contacting each of these backend services, you can create an API Gateway which is an entry point for the web frontend to access all backend services from a single place. This pattern is expectedly called API Gateway and is a common practice in Microservices architecture.
Replace the content of '/projects/labs/gateway-vertx/src/main/java/com.redhat.cloudnative.gateway/GatewayVerticle.java'
class with the following:
package com.redhat.cloudnative.gateway;
import io.vertx.core.http.HttpMethod;
import io.vertx.core.json.JsonArray;
import io.vertx.core.json.JsonObject;
import io.vertx.ext.web.client.WebClientOptions;
import io.vertx.reactivex.core.AbstractVerticle;
import io.vertx.reactivex.ext.web.Router;
import io.vertx.reactivex.ext.web.RoutingContext;
import io.vertx.reactivex.ext.web.client.WebClient;
import io.vertx.reactivex.ext.web.client.predicate.ResponsePredicate;
import io.vertx.reactivex.ext.web.codec.BodyCodec;
import io.vertx.reactivex.ext.web.handler.CorsHandler;
import io.vertx.reactivex.ext.web.handler.StaticHandler;
import io.vertx.reactivex.servicediscovery.ServiceDiscovery;
import io.vertx.reactivex.servicediscovery.types.HttpEndpoint;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import io.reactivex.Observable;
import io.reactivex.Single;
import java.util.ArrayList;
import java.util.List;
public class GatewayVerticle extends AbstractVerticle {
private static final Logger LOG = LoggerFactory.getLogger(GatewayVerticle.class);
private WebClient catalog;
private WebClient inventory;
@Override
public void start() {
Router router = Router.router(vertx);
router.route().handler(CorsHandler.create("*").allowedMethod(HttpMethod.GET));
router.get("/*").handler(StaticHandler.create("assets"));
router.get("/health").handler(ctx -> ctx.response().end(new JsonObject().put("status", "UP").toString()));
router.get("/api/products").handler(this::products); // (1)
ServiceDiscovery.create(vertx, discovery -> { // (2)
// Catalog lookup
Single<WebClient> catalogDiscoveryRequest = HttpEndpoint.rxGetWebClient(discovery,
rec -> rec.getName().equals("catalog"))
.onErrorReturn(t -> WebClient.create(vertx, new WebClientOptions()
.setDefaultHost(System.getProperty("catalog.api.host", "localhost"))
.setDefaultPort(Integer.getInteger("catalog.api.port", 9000))));
// Inventory lookup
Single<WebClient> inventoryDiscoveryRequest = HttpEndpoint.rxGetWebClient(discovery,
rec -> rec.getName().equals("inventory"))
.onErrorReturn(t -> WebClient.create(vertx, new WebClientOptions()
.setDefaultHost(System.getProperty("inventory.api.host", "localhost"))
.setDefaultPort(Integer.getInteger("inventory.api.port", 9001))));
// Zip all 3 requests
Single.zip(catalogDiscoveryRequest, inventoryDiscoveryRequest, (c, i) -> {
// When everything is done
catalog = c;
inventory = i;
return vertx.createHttpServer()
.requestHandler(router)
.listen(Integer.getInteger("http.port", 8080));
}).subscribe();
});
}
private void products(RoutingContext rc) { // (3)
// Retrieve catalog
catalog
.get("/api/catalog")
.expect(ResponsePredicate.SC_OK)
.as(BodyCodec.jsonArray())
.rxSend()
.map(resp -> {
// Map the response to a list of JSON object
List<JsonObject> listOfProducts = new ArrayList<>();
for (Object product : resp.body()) {
listOfProducts.add((JsonObject)product);
}
return listOfProducts;
})
.flatMap(products -> {
// For each item from the catalog, invoke the inventory service
// and create a JsonArray containing all the results
return Observable.fromIterable(products)
.flatMapSingle(this::getAvailabilityFromInventory)
.collect(JsonArray::new, JsonArray::add);
}
)
.subscribe(
list -> rc.response().end(list.encodePrettily()),
error -> rc.response().setStatusCode(500).end(new JsonObject().put("error", error.getMessage()).toString())
);
}
private Single<JsonObject> getAvailabilityFromInventory(JsonObject product) { // (4)
// Retrieve the inventory for a given product
return inventory
.get("/api/inventory/" + product.getString("itemId"))
.as(BodyCodec.jsonObject())
.rxSend()
.map(resp -> {
if (resp.statusCode() != 200) {
LOG.warn("Inventory error for {}: status code {}",
product.getString("itemId"), resp.statusCode());
return product.copy();
}
return product.copy().put("availability",
new JsonObject().put("quantity", resp.body().getInteger("quantity")));
});
}
}
-
A REST mapping to map /api/products to the products() method
-
Vert.x provides built-in service discovery for finding where dependent services are deployed and accessing their endpoints. Vert.x service discovery can be seamlessly integrated with external service discovery mechanisms provided by OpenShift, Kubernetes, Consul, Redis, etc. In this lab, since you will deploy the API Gateway on OpenShift, the OpenShift service discovery bridge is used to automatically import OpenShift services into the Vert.x application as they get deployed and undeployed. Since you also want to test the API Gateway locally, there is an onErrorReturn() method clause in the service lookup to fallback on a local service for Inventory and Catalog REST APIs.
-
The products() method invokes the Catalog REST endpoint and retrieves the products. It then iterates over the retrieved products and for each product invokes the Inventory REST endpoint to get the inventory status and enrich the product data with availability info using the getAvailabilityFromInventory() method. Note that instead of making blocking calls to the Catalog and Inventory REST APIs, all calls are non-blocking and handled using RxJava. Due to its non-blocking nature, the product() method can immediately return without waiting for the Catalog and Inventory REST invocations to complete and whenever the result of the REST calls is ready, the result will be acted upon and update the response which is then sent back to the client.
-
The getAvailabilityFromInventory() method is similar to the product() method, it invokes the Inventory REST endpoint and retrieves the inventory.
Build and package the Gateway Service using Maven by right-clicking on gateway-vertx
project in the project explorer then, click on 'Commands > Build > build'
It’s time to build and deploy our service on OpenShift.
OpenShift {{OPENSHIFT_DOCS_BASE}}/architecture/core_concepts/builds_and_image_streams.html#source-build[Source-to-Image (S2I)^] feature can be used to build a container image from your project. OpenShift S2I uses the supported OpenJDK container image to build the final container image of the API Gateway service by uploading the Vert.x uber-jar from the target folder to the OpenShift platform.
Maven projects can use the Fabric8 Maven Plugin to access the OpenShift S2I for building the application container image. This maven plugin is a Kubernetes/OpenShift client and uses the REST API to communicate with OpenShift and issue commands to build, deploy and launch the application as a pod.
To build and deploy the Gateway Service on OpenShift using the fabric8 maven plugin,
which is already configured in CodeReady Workspaces, right click on 'gateway-vertx'
project in the project explorer then, click on 'Commands > Deploy > fabric8:deploy'
Tip
|
fabric8:deploy
It will cause the following to happen:
|
Once this completes, your project should be up and running. OpenShift runs the different components of the project in one or more pods which are the unit of runtime deployment and consists of the running containers for the project.
Let’s take a moment and review the OpenShift resources that are created for the API Gateway:
-
Build Config: gateway-s2i build config is the configuration for building the Gateway container image from the gateway source code or JAR archive
-
Image Stream: gateway image stream is the virtual view of all gateway container images built and pushed to the OpenShift integrated registry.
-
Deployment Config: gateway deployment config deploys and redeploys the Gateway container image whenever a new Gateway container image becomes available
-
Service: gateway service is an internal load balancer which identifies a set of pods (containers) in order to proxy the connections it receives to them. Backing pods can be added to or removed from a service arbitrarily while the service remains consistently available, enabling anything that depends on the service to refer to it at a consistent address (service name or IP).
-
Route: gateway route registers the service on the built-in external load-balancer and assigns a public DNS name to it so that it can be reached from outside OpenShift cluster.
You can review the above resources in the OpenShift Web Console or using oc describe
command:
Tip
|
bc is the short-form of buildconfig and can be interchangeably used instead of it with the OpenShift CLI. The same goes for is instead of imagestream, dc instead of deploymentconfig and svc instead of service. |
$ oc describe bc gateway-s2i $ oc describe is gateway $ oc describe dc gateway $ oc describe svc gateway $ oc describe route gateway
You can see the expose DNS url for the Gateway Service in the {{OPENSHIFT_CONSOLE_URL}}[OpenShift Web Console^]or using OpenShift CLI.
$ oc get routes NAME HOST/PORT PATH SERVICES PORT TERMINATION catalog catalog-{{COOLSTORE_PROJECT}}.{{APPS_HOSTNAME_SUFFIX}} catalog 8080 None gateway gateway-{{COOLSTORE_PROJECT}}.{{APPS_HOSTNAME_SUFFIX}} gateway 8080 None inventory inventory-{{COOLSTORE_PROJECT}}.{{APPS_HOSTNAME_SUFFIX}} inventory 8080 None
Click on the OpenShift Route of 'Gateway Service'
from the {{OPENSHIFT_CONSOLE_URL}}[OpenShift Web Console^].
Then click on 'Test it'
. You should have the following output:
[ {
"itemId" : "329299",
"name" : "Red Fedora",
"desc" : "Official Red Hat Fedora",
"price" : 34.99,
"availability" : {
"quantity" : 35
}
},
...
]
As mentioned earlier, Vert.x built-in service discovery is integrated with OpenShift service discovery to lookup the Catalog and Inventory APIs.
Well done! You are ready to move on to the next lab.