This hands-on tutorial gives you an introduction to the process for integrating Liferay with OpenTelemetry to extract logs, traces, and metrics from Liferay. We’ll spin-up Liferay using the pre-packaged Liferay DXP container from LIferay, Inc. and integrate it with a locally-run Jaeger container using the OpenTelemetry connector.

Instrumenting Liferay with OpenTelemetry: Prerequisites

To run through the tutorial steps, you’ll need a recent version of Docker installed. Make sure you have the latest version of Docker installed and available from a terminal prompt.

You’ll want the latest version of the OpenTelemetry Java Agent downloaded as well. As of the time this document was authored, you can download the newest version of the agent JAR file here.

To make communications between the Liferay container and the Jaeger container simpler, we create a Docker network to be used by those two containers; to do so, issue the following command in a terminal window:

docker network create --subnet 172.18.0.0/16 liferay-net

Start Jaeger Container

Jaeger is an engine geared towards distributed tracing and monitoring for application operations; out of the box, Jaeger can run in multiple modes. We’ll be using the all-in-one container for this exercise, as we’re not looking to scale out the collection infrastructure.

To pull the latest Jaeger container from Docker Hub, issue the following command:

docker pull jaegertracing/all-in-one:latest

To start up the Jaeger container, issue the following command:

docker run -d --name jaeger --net liferay-net --ip 172.18.0.2 \
    -e COLLECTOR_ZIPKIN_HTTP_PORT=9411 \
    -p 5775:5775/udp -p 6831:6831/udp -p 6832:6832/udp \
    -p 5778:5778 -p 16686:16686 -p 14268:14268 \
    -p 14250:14250 -p 9411:9411 jaegertracing/all-in-one:latest

This will start up a Jaeger instance with local ports mapped to localhost; to access the Jaeger UI, open a web browser and visit http://localhost:16686/search. Once opened, you should see the following screen:

Instrumenting Liferay with OpenTelemetry Jaeger UI

Start Liferay Container

Running Liferay in a container is fairly straightforward. However, there are a few considerations to take into account when spinning up a Liferay container. Liferay provides a custom Docker container image that bundles a Liferay instance up with the OpenJDK Java JVM and Apache Tomcat to make a convenient, flexible deployment of Liferay possible via containers. The container allows you to mount a separate volume containing configuration and library field, which is copied into the $liferay.home location during container initialization. For this exercise, we set up a liferay-home folder containing the following structure:

.
├── files
│   ├── portal-ext.properties
│   └── tomcat
│       ├── bin
│       │   └── setenv.sh
│       └── lib
│           └── opentelemetry-javaagent-all.jar
└── scripts
    └── remove_unneeded_modules.sh

The file structure in the files directory will be copied to the container’s $liferay.home directory as-is, and the files in the scripts directory will be executed before bringing up the Liferay process inside the container. To connect the Liferay instance to the Jaeger instance, the following files or file contents must be present:

  • The Tomcat setenv.sh file needs to have a line that looks like the following:
CATALINA_OPTS="${CATALINA_OPTS} -Dapplication.name=liferay -Dapplication.home=/opt/liferay/tomcat -Dotel.exporter=jaeger -Dotel.exporter.jaeger.endpoint=172.18.0.2:14250 -Dotel.exporter.jaeger.service.name=liferay-dxp -javaagent:/opt/liferay/tomcat/lib/opentelemetry-javaagent-all.jar"

This line loads the OpenTelemetry agent into the Liferay JVM and instructs the agent to connect to the Jaeger collector via the IP address we specified for the Jaeger container instance.

  • The OpenTelemetry Java Agent needs to be present in the Tomcat lib directory.

Next, we’ll pull down the liferay/dxp container using the following command:

docker pull liferay/dxp:7.3.10-ga1

This retrieves the current GA1 release container for Liferay DXP 7.3. You may wish to check the Liferay Docker Hub tags page to see if there’s a more recent container available. Next, we start up an instance of the Liferay container, mounting the above directory to the /mnt/liferay mount point:

docker run -m 4g -p 8080:8080 --net liferay-net \
    -v $(pwd)/liferay-home:/mnt/liferay \
    -e JAVA_VERSION=zulu11 liferay/dxp:7.3.10-ga1

Note that the command above will map port 8080 on the container to http://localhost:8080; this is how you’ll access the Liferay instance.

You can view a proof-of-concept implementation of this on XTIVIA’s Github organization.

Instrumenting Liferay with OpenTelemetry: Validate Connectivity

Once both the Jaeger and Liferay containers have initialized, you should start seeing trace data from your Liferay instance appear in your Jaeger UI. This example POC tags all Liferay data with the liferay-dxp service name in Jaeger. In the Jaeger UI, you’ll see the Service dropdown populated with that service name and be able to query the trace data coming in.

Instrumenting Liferay with OpenTelemetry Validate Connectivity

Instrumenting Liferay with OpenTelemetry: Next Steps

Now that you’re ingesting data from your Liferay DXP instance into your Jaeger instance, you can proceed with tagging specific spans and instrumenting your custom code to gather performance or business metrics. Note that this proof-of-concept is not intended to be a production-level OpenTelemetry/Liferay integration. For that, we need to set up a highly-available collector & storage backend infrastructure using Elasticsearch, and we would likely also integrate Kafka as an intermediary to smooth out telemetry data collection spikes. For that type of scenario, XTIVIA recommends using the Jaeger Operator if your Liferay infrastructure runs in Kubernetes. The best way to attach the collector to the Liferay container for other containerized Liferay architectures depends largely on the container orchestration platform used.

If you have questions on how you can best implement observability best practices into your custom application and/or need help with your Liferay DXP implementation or overall Cloud Services, please engage with us via comments on this blog post or reach out to us here.