This is a proof-of-concept to demonstrate how the Artemis services can be auto-discovered and traffic being routed based on the active Spring Profiles.
- Build the API-Gateway and the Artemis-Service:
cd api-gateway/ && ./gradlew build && cd ../artemis && ./gradlew build
- Run the following command:
docker compose up -d
- Access the JHipster registry at http://localhost:8761 and verify that it looks like this:
See that the two Artemis services have different Spring Profiles enabled.
- Wait for a few seconds.
- Access the functionality of the QuizController by calling /api/quiz in your browser. This is provided by both artemis instances.
- Verify that there is a response and by spamming the requests, the response changes (different instance name).
- Repeat the same for /api/quiz. However, here, only
artemis-1
should be part of the response string as only this service has enabled this profile. - Open /profiles to see the active Artemis profiles of the API-Gateway (cluster-aggregated).
Note: If you see a 503 response, you might just need to wait a little bit more until the API gateway has fetched the available service instances from Jhipster Registry.
Since all artemis instances have the same service name ("ARTEMIS"), we need to decide for each request where to route.
This is what the CustomLoadBalancer.java does. It uses the Spring Profiles to decide where to route the request based on the path prefix.
This will override the default behavior of round-robin load-balancing.
To overcome this, ServiceInstance#pickBasedOnPathOrAttribute()
performs simple round-robin based on a general (1) and a per-module (2) round-robin counter.
When receiving a request, the load balancer will match the path prefix to the respective Spring profile.
To do so, it uses the ProfilePathStore.java which reads a mapping from the application.yml
file.
When running the Artemis services as Java processes, these will use the IP address of their VM to register at the JHipster registry. This address is routable by other VMs. The port is also determined automatically.
When using Docker to run the Artemis nodes, we need to make some smaller adjustments. When the service registers at the registry, it does so with the docker containers ip address. While this is routable within the same docker network (i.e. compose setup), this does not work for a distributed docker environment where the services are running on different machines.
To overcome this, we can pass the EUREKA_INSTANCE_IP_ADDRESS
environment variable to the service. This will then be used to register the service at the registry.
For instance, artemis-1
has EUREKA_INSTANCE_IP_ADDRESS=host.docker.internal
which causes the service to register with the host's "IP address" (it's rather a hostname but that's irrelevant here).
The port is determined by the server.port
property in the application.yml
file (or default 8080).
Since this might differ from the host port the docker container is attached to, one has to pass the EUREKA_INSTANCE_NON_SECURE_PORT
environment variable to the service.
This is only required if the host port differs from the container port, but could be set in any case.
Note that the term unsecure
refers to the fact that it's not an HTTPS port, but rather HTTP - the proxy already does TLS termination.
Then, you can call /api/text and /api/quiz to see that artemis-1 also responds.
- The CustomLoadBalancerConfig.java is just used to enable the
CustomLoadBalancer
for theARTEMIS
-services. - Each Artemis instance could have their own endpoint /profiles to show their active profiles. With this, the client is unaware of whether calling the gateway or the service directly.
- Performance should be non-critical. In case it is, we could consider caching the service instances (to module mapping) for a certain amount of time.