Skip to content

Latest commit

 

History

History

google-cloud-functions

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 
 
 
 
 
 
 

Instrumenting a Java Google Cloud Run Function with OpenTelemetry

This example demonstrates how to instrument an serverless Google Cloud Run function written in Java using OpenTelemetry, and then export the data to Splunk Observability Cloud. We'll use Java v21 for this example, but the steps for other Java versions are similar.

Prerequisites

The following tools are required to deploy Java Google Cloud Run functions:

  • An Google Cloud Platform account with permissions to create and execute Google Cloud Run functions
  • An OpenTelemetry collector that's accessible to the Google Cloud Run function
  • gCloud CLI

Splunk Distribution of the OpenTelemetry Collector

For this example, we deployed the Splunk Distribution of the OpenTelemetry Collector onto a virtual machine in GCP using Gateway mode, and ensured it's accessible to our Google Cloud Run function.

We configured it with the SPLUNK_HEC_TOKEN and SPLUNK_HEC_URL environment variables, so that it exports logs to our Splunk Cloud instance.

Please refer to Install the Collector using packages and deployment tools for collector installation instructions.

Application Overview

If you just want to build and deploy the example, feel free to skip this section.

The application used for this example is a simple Hello World application.

We added a helper class named SplunkTelemetryConfiguration, and included code to assist with initializing the OpenTelemetry SDK:

    public static OpenTelemetrySdk configureOpenTelemetry() {

        String serviceName = System.getenv("OTEL_SERVICE_NAME");
        String otelResourceAttributes = System.getenv("OTEL_RESOURCE_ATTRIBUTES");
        String otelExporterEndpoint = System.getenv("OTEL_EXPORTER_OTLP_ENDPOINT");

        if (serviceName == null)
            throw new IllegalArgumentException("The OTEL_SERVICE_NAME environment variable must be populated");
        if (otelResourceAttributes == null)
            throw new IllegalArgumentException("The OTEL_RESOURCE_ATTRIBUTES environment variable must be populated");
        if (otelExporterEndpoint == null)
            throw new IllegalArgumentException("The OTEL_EXPORTER_OTLP_ENDPOINT environment variable must be populated");

        OpenTelemetrySdk openTelemetrySdk =
            AutoConfiguredOpenTelemetrySdk.initialize().getOpenTelemetrySdk();

        Resource autoResource = ResourceConfiguration.createEnvironmentResource();

        GCPResourceProvider resourceProvider = new GCPResourceProvider();

        return openTelemetrySdk;
    }

It requires the following dependencies to be added to the pom.xml file:

   <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>io.opentelemetry</groupId>
                <artifactId>opentelemetry-bom</artifactId>
                <version>1.44.1</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

    <dependencies>
        ...
        <dependency>
            <groupId>io.opentelemetry</groupId>
            <artifactId>opentelemetry-api</artifactId>
        </dependency>
        <dependency>
            <groupId>io.opentelemetry</groupId>
            <artifactId>opentelemetry-sdk</artifactId>
        </dependency>
        <dependency>
            <!-- Not managed by opentelemetry-bom -->
            <groupId>io.opentelemetry.semconv</groupId>
            <artifactId>opentelemetry-semconv</artifactId>
            <version>1.28.0-alpha</version>
        </dependency>
        <dependency>
            <groupId>io.opentelemetry</groupId>
            <artifactId>opentelemetry-exporter-otlp</artifactId>
        </dependency>
        <dependency>
            <groupId>io.opentelemetry.instrumentation</groupId>
            <artifactId>opentelemetry-log4j-context-data-2.17-autoconfigure</artifactId>
            <version>2.8.0-alpha</version>
            <scope>runtime</scope>
        </dependency>
        <dependency>
            <groupId>io.opentelemetry.contrib</groupId>
            <artifactId>opentelemetry-gcp-resources</artifactId>
            <version>1.44.0-alpha</version>
        </dependency>
        <dependency>
            <groupId>io.opentelemetry</groupId>
            <artifactId>opentelemetry-sdk-extension-autoconfigure</artifactId>
        </dependency>
    </dependencies>

Note that we've added opentelemetry-log4j-context-data-2.17-autoconfigure as a dependency as well, which injects the trace ID and span ID from an active span into Log4j's context data. Refer to ContextData Instrumentation for Log4j2 for further details.

The log4j2.xml configuration file was modified to utilize this trace context and add it to the log output:

    <Appenders>
        <Console name="console" target="SYSTEM_OUT">
            <PatternLayout>
                <pattern>%d %5p [%t] %c{3} - trace_id=%X{trace_id} span_id=%X{span_id} trace_flags=%X{trace_flags} service.name=${env:OTEL_SERVICE_NAME} %m%n</pattern>
            </PatternLayout>
        </Console>
    </Appenders>

The HelloHttpFunction.java file was then modified to configure OpenTelemetry using the helper class as follows:

public class HelloHttpFunction implements HttpFunction {

    private final OpenTelemetrySdk openTelemetrySdk = SplunkTelemetryConfigurator.configureOpenTelemetry();
    private final Tracer tracer = openTelemetrySdk.getTracer(HelloHttpFunction.class.getName(), "0.1.0");
    private static final Logger logger = LogManager.getLogger(HelloHttpFunction.class);

    public void service(final HttpRequest request, final HttpResponse response) throws Exception {
        Span span = tracer.spanBuilder("HelloHttpFunction").startSpan();

        try (Scope scope = span.makeCurrent()) {
            logger.info("Handling the HelloHttpFunction call");

            final BufferedWriter writer = response.getWriter();
            writer.write("Hello world!");
        }
        catch (Throwable t) {
            span.recordException(t);
        }
        finally {
            span.end();
        }
    }
}

Build and Deploy

Open a terminal and navigate to the following directory:

splunk-opentelemetry-examples/instrumentation/java/google-cloud-functions

Initialize the gCloud CLI

If you haven't already done so, install and initialize the gcloud CLI.

Build and Deploy the Google Cloud Run Function

Use the following command to deploy the Google Cloud Run function, substituting the region that's best for you. To allow OpenTelemetry to send trace data to Splunk Observability Cloud, we also need to set the OTEL_EXPORTER_OTLP_ENDPOINT, OTEL_SERVICE_NAME, and DEPLOYMENT_ENVIRONMENT environment variables as part of the gcloud deploy command:

gcloud functions deploy java-gcloud-function-example \
    --gen2 \
    --region=us-central1 \
    --runtime=java21 \
    --source=. \
    --entry-point=gcfv2.HelloHttpFunction \
    --trigger-http \
    --set-env-vars OTEL_SERVICE_NAME=java-gcloud-function-example,OTEL_EXPORTER_OTLP_ENDPOINT=http://<collector IP address>:4317,OTEL_RESOURCE_ATTRIBUTES='deployment.environment=test'

Answer "y" to the following question when asked:

Allow unauthenticated invocations of new function [java-gcloud-function-example]? (y/N)? 

If the function is created successfully, it should provide you with a URL such as the following:

https://us-central1-gcp-<account name>.cloudfunctions.net/java-gcloud-function-example

Test the Google Cloud Run Function

Take the URL provided by the gcloud CLI above and enter it into your browser. It should return:

Hello World! 

View Traces in Splunk Observability Cloud

After a minute or so, you should start to see traces for the serverless function appearing in Splunk Observability Cloud:

Trace

Add Trace Context to Logs

Logs generated by a Google Cloud Run function get sent to Google Cloud Logging. Various methods exist for streaming logs into Splunk platform from Google Cloud Logging, as described in Stream logs from Google Cloud to Splunk.

Once the logs are in Splunk platform, they can be made available to Splunk Observability Cloud using Log Observer Connect.

Here's an example log entry, which includes the trace_id and span_id:

{
    insertId: "67c9e019000d8d25bdcb7fcf"
    labels: {3}
    logName: "projects/gcp-splunko11ystrat-nprd-11062/logs/run.googleapis.com%2Fstdout"
    payload: "textPayload"
    receiveLocation: "us-central1"
    receiveTimestamp: "2025-03-06T17:49:13.893130330Z"
    resource: {2}
    severity: "DEFAULT"
    textPayload: "2025-03-06 17:49:13,887  INFO [qtp959447386-15] gcfv2.HelloHttpFunction - trace_id=21cce75d5a25204f34662f3d72cf49fb span_id=30d5eed8929fcd04 trace_flags=01 service.name=java-gcloud-function-example Handling the HelloHttpFunction call"
    timestamp: "2025-03-06T17:49:13.888101Z"
    traceSampled: false
}

We can see that the log entry includes a trace_id and span_id, which allows us to correlate logs with traces.