Trace across Multiple Services
Logfire builds on OpenTelemetry, which keeps track of context to determine the parent trace/span of a new span/log and whether it should be included by sampling. Context propagation is when this context is serialized and sent to another process, so that tracing can be distributed across services while allowing the full tree of spans to be cleanly reconstructed and viewed together.
Manual Context Propagation¶
Logfire provides a thin wrapper around the OpenTelemetry context propagation API to make manual distributed tracing easier. You shouldn't usually need to do this yourself, but it demonstrates the concept nicely. Here's an example:
import logfire
logfire.configure()
with logfire.span('parent'):
    ctx = logfire.get_context()
print(ctx)
# Attach the context in another execution environment
with logfire.attach_context(ctx):
    logfire.info('child')  # This log will be a child of the parent span.
ctx will look something like this:
{'traceparent': '00-d1b9e555b056907ee20b0daebf62282c-7dcd821387246e1c-01'}
This contains 4 fields:
- 00is a version number which you can ignore.
- d1b9e555b056907ee20b0daebf62282cis the- trace_id.
- 7dcd821387246e1cis the- span_idof the parent span, i.e. the- parent_span_idof the child log.
- 01is the- trace_flagsfield and indicates that the trace should be included by sampling.
See the API reference for more details about these functions.
Integrations¶
OpenTelemetry instrumentation libraries (which Logfire uses for its integrations) handle context propagation automatically, even across different programming languages. For example:
- Instrumented HTTP clients such as requestsandhttpxwill automatically set thetraceparentheader when making requests.
- Instrumented web servers such as flaskandfastapiwill automatically extract thetraceparentheader and use it to set the context for server spans.
- The celeryintegration will automatically propagate the context to child tasks.
Thread and Pool executors¶
Logfire automatically patches ThreadPoolExecutor and ProcessPoolExecutor to propagate context to child threads and processes. This means that logs and spans created in child threads and processes will be correctly associated with the parent span. Here's an example to demonstrate:
import logfire
from concurrent.futures import ThreadPoolExecutor
logfire.configure()
@logfire.instrument("Doubling {x}")
def double(x: int):
    return x * 2
with logfire.span("Doubling everything") as span:
    executor = ThreadPoolExecutor()
    results = list(executor.map(double, range(3)))
    span.set_attribute("results", results)
Unintentional Distributed Tracing¶
Because instrumented web servers automatically extract the traceparent header by default, your spans can accidentally pick up the wrong context from an externally instrumented client, or from your cloud provider such as Google Cloud Run. This can lead to:
- Spans missing their parent.
- Spans being mysteriously grouped together.
- Spans missing entirely because the original trace was excluded by sampling.
By default, Logfire warns you when trace context is extracted, e.g. when server instrumentation finds a traceparent header. You can deal with this by setting the distributed_tracing argument of logfire.configure() or by setting the LOGFIRE_DISTRIBUTED_TRACING environment variable:
- Setting to Falsewill prevent trace context from being extracted. This is recommended for web services exposed to the public internet. You can still attach/inject context to propagate to other services and create distributed traces with the web service as the root.
- Setting to Trueimplies that the context propagation is intentional and will silence the warning.
The distributed_tracing configuration (including the warning by default) only applies when the raw OpenTelemetry API is used to extract context, as this is typically done by third-party libraries. By default, logfire.attach_context assumes that context propagation is intended by the application. If you are writing a library, use attach_context(context, third_party=True) to respect the distributed_tracing configuration.