Skip to content

Latest commit

 

History

History

Instrumenting a Python Azure Function with OpenTelemetry

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

Prerequisites

The following tools are required to deploy Python Azure functions:

  • An Azure account with permissions to create and execute Azure functions
  • Visual Studio Code
  • An OpenTelemetry collector that's accessible to the Azure function
  • Azure Functions extension for Visual Studio Code
  • Python extension for Visual Studio Code

Splunk Distribution of the OpenTelemetry Collector

For this example, we deployed the Splunk Distribution of the OpenTelemetry Collector onto a virtual machine in Azure using Gateway mode, and ensured it's accessible to our Azure 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 file named splunk_opentelemetry.py to include a helper function that initializes OpenTelemetry for traces, logs, and metrics:

def init_opentelemetry(): 

    if 'OTEL_SERVICE_NAME' not in os.environ: 
        raise Exception('The OTEL_SERVICE_NAME environment variable must be set')

    if 'OTEL_EXPORTER_OTLP_ENDPOINT' not in os.environ: 
        raise Exception('The OTEL_EXPORTER_OTLP_ENDPOINT environment variable must be set')
    
    if 'OTEL_RESOURCE_ATTRIBUTES' not in os.environ: 
        raise Exception('The OTEL_RESOURCE_ATTRIBUTES environment variable must be set')

    traceProvider = TracerProvider()
    processor = BatchSpanProcessor(OTLPSpanExporter())
    traceProvider.add_span_processor(processor)
    trace.set_tracer_provider(traceProvider)

    reader = PeriodicExportingMetricReader(OTLPMetricExporter())
    meterProvider = MeterProvider(metric_readers=[reader])
    metrics.set_meter_provider(meterProvider)

    _logs.set_logger_provider(
        LoggerProvider()
    )
    logging.getLogger().addHandler(
        LoggingHandler(
            logger_provider=_logs.get_logger_provider().add_log_record_processor(
                BatchLogRecordProcessor(OTLPLogExporter())
            )
        )
    )

We also updated the function_app.py file to call our helper function to initialize OpenTelemetry instrumentation, start a new span when the HTTP trigger function is invoked, and add an attribute to the span:

import azure.functions as func
import logging
from splunk_opentelemetry import init_opentelemetry
from opentelemetry import trace

logger = logging.getLogger(__name__)
logger.setLevel(logging.INFO)

init_opentelemetry()
tracer = trace.get_tracer("azure_function_python_opentelemetry_example")

app = func.FunctionApp(http_auth_level=func.AuthLevel.ANONYMOUS)

@app.route(route="azure_function_python_opentelemetry_example")
def azure_function_python_opentelemetry_example(req: func.HttpRequest) -> func.HttpResponse:

    with tracer.start_as_current_span("azure_function_python_opentelemetry_example") as span:
        
        logging.info('Python HTTP trigger function processed a request.')

        name = req.params.get('name')
        if not name:
            try:
                req_body = req.get_json()
            except ValueError:
                pass
            else:
                name = req_body.get('name')

        if name:
            span.set_attribute("app.name", name)
            return func.HttpResponse(f"Hello, {name}!")
        else:
            return func.HttpResponse(
                "Hello, World!",
                status_code=200
            )

These code changes required several OpenTelemetry packages to be installed, which we can see in the requirements.txt file:

opentelemetry-api
opentelemetry-sdk
opentelemetry-exporter-otlp-proto-http

# add this package to ensure trace context is added to logs
opentelemetry-instrumentation-logging

For this example, we'll send metrics, traces, and logs to a collector running on another virtual machine in Azure. The local.settings.json file was updated as follows:

{
  "IsEncrypted": false,
  "Values": {
    "AzureWebJobsStorage": "",
    "FUNCTIONS_WORKER_RUNTIME": "python", 
    "OTEL_SERVICE_NAME": "azure_function_python_opentelemetry_example",
    "OTEL_EXPORTER_OTLP_ENDPOINT": "http://<Collector IP Address>:4318",
    "OTEL_RESOURCE_ATTRIBUTES": "deployment.environment=test",
    "OTEL_PYTHON_LOG_CORRELATION": "true"
  }
}

If you don't have a collector available and want to send traces directly to Splunk Observability Cloud, use the following configuration instead for local.settings.json:

  "Values": {
    "AzureWebJobsStorage": "UseDevelopmentStorage=true",
    "FUNCTIONS_WORKER_RUNTIME": "node",
    "OTEL_EXPORTER_OTLP_ENDPOINT": "https://ingest.<Realm>.signalfx.com/v2/trace/otlp", 
    "OTEL_EXPORTER_OTLP_HEADERS": "X-SF-TOKEN=<Access Token>", 
    "OTEL_SERVICE_NAME": "azure-function-python-opentelemetry-example", 
    "OTEL_RESOURCE_ATTRIBUTES": "deployment.environment=test" 
  }

Build and Deploy

Open the following project using Visual Studio Code:

splunk-opentelemetry-examples/instrumentation/python/azure-functions

Create a Function App in Azure

Create a Function App in Azure if you don't already have one. For my example, I used opentelemetry-python-examples as the function name, and used the region of “West US 2” with Python 3.11 as the runtime.

Azure Function App

Create a Deployment Slot (Optional)

By default, Azure will use a deployment slot named "Production" for an Azure Function App.
In my example, I created a deployment slot named "test".

Deployment Slot

Set Environment Variables

To allow OpenTelemetry to send trace data to Splunk Observability Cloud, we need to set the OTEL_SERVICE_NAME, OTEL_EXPORTER_OTLP_ENDPOINT, OTEL_RESOURCE_ATTRIBUTES, and OTEL_PYTHON_LOG_CORRELATION environment variables for our Azure Function App:

Environment Variables

Build and Deploy the Azure Function

In the Azure section of Visual Studio Code, right click on the deployment slot of interest and select Deploy to Slot.

Deploy

It will ask you to confirm:

Confirm Deploy

Test the Azure Function

Copy the function URL from the Azure function:

Function URL

Then point your browser to that URL, 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

Note that the bottom-right of the trace includes a button that links to the related log entries.

Add Trace Context to Logs

In general, logs generated by an Azure function get sent to Application Insights. Various methods exist for ingesting logs into Splunk platform from Application Insights, such as the Splunk Add-on for Microsoft Cloud Services.

In this example, the Splunk distribution of OpenTelemetry Python also exports logs to our collector using OTLP.

By following the link from the trace show above, we can see all of the log entries associated with this trace:

Related Logs

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