Skip to content

Instantly share code, notes, and snippets.

@DiTo97
Last active October 18, 2025 15:26
Show Gist options
  • Save DiTo97/77f9fe95a6a42729ee07f98a2b0689b8 to your computer and use it in GitHub Desktop.
Save DiTo97/77f9fe95a6a42729ee07f98a2b0689b8 to your computer and use it in GitHub Desktop.
OpenTelemetry tracing utilities for any library.
"""OpenTelemetry tracing utilities for any library.
This module provides decorators to simplify the integration of OpenTelemetry
tracing into Python applications. It offers `trace_function` for instrumenting
individual functions (both synchronous and asynchronous) and `trace_class`
for instrumenting multiple methods within a class.
The tracer is initialized with the module name and version defined by
`INSTRUMENTING_MODULE` ('<module>') and
`INSTRUMENTING_MODULE_VERSION` ('<version>').
Features:
- Automatic span creation for decorated functions/methods.
- Support for both synchronous and asynchronous functions.
- Default span naming based on module and function/class/method name.
- Customizable span names, kinds, and static attributes.
- Dynamic attribute setting via an `attribute_extractor` callback.
- Automatic recording of exceptions and setting of span status.
- Selective method tracing in classes using include/exclude lists.
Usage:
For a single function:
```python
from your_module import trace_function
@trace_function
def my_function():
# ...
pass
@trace_function(span_name='custom.op', kind=SpanKind.CLIENT)
async def my_async_function():
# ...
pass
```
For a class:
```python
from your_module import trace_class
@trace_class(exclude_list=['internal_method'])
class MyService:
def public_api(self, user_id):
# This method will be traced
pass
def internal_method(self):
# This method will not be traced
pass
```
The implementation is inspired by:
https://github.com/a2aproject/a2a-python/blob/d5856359034f4d3d1e4578804727f47a3cd7c322/src/a2a/utils/telemetry.py
"""
import asyncio
import functools
import inspect
import logging
from collections.abc import Callable
from typing import TYPE_CHECKING, Any
if TYPE_CHECKING:
from opentelemetry.trace import SpanKind as SpanKindType
else:
SpanKindType = object
logger = logging.getLogger(__name__)
try:
from opentelemetry import trace
from opentelemetry.trace import SpanKind as _SpanKind
from opentelemetry.trace import StatusCode
except ImportError:
logger.debug(
'OpenTelemetry not found, tracing will be disabled.'
)
class _NoOp:
"""A no-op object that absorbs all tracing calls when OpenTelemetry is not installed."""
def __call__(self, *args: Any, **kwargs: Any) -> Any:
return self
def __enter__(self) -> '_NoOp':
return self
def __exit__(self, *args: object, **kwargs: Any) -> None:
pass
def __getattr__(self, name: str) -> Any:
return self
trace = _NoOp() # type: ignore
_SpanKind = _NoOp() # type: ignore
StatusCode = _NoOp() # type: ignore
SpanKind = _SpanKind
__all__ = ['SpanKind']
INSTRUMENTING_MODULE = '<module>'
INSTRUMENTING_MODULE_VERSION = '<version>'
def trace_function( # noqa: PLR0915
func: Callable | None = None,
*,
span_name: str | None = None,
kind: SpanKindType = SpanKind.INTERNAL,
attributes: dict[str, Any] | None = None,
attribute_extractor: Callable | None = None,
) -> Callable:
"""A decorator to automatically trace a function call with OpenTelemetry.
This decorator can be used to wrap both sync and async functions.
When applied, it creates a new span for each call to the decorated function.
The span will record the execution time, status (OK or ERROR), and any
exceptions that occur.
It can be used in two ways:
1. As a direct decorator: `@trace_function`
2. As a decorator factory to provide arguments: `@trace_function(span_name="custom.name")`
Args:
func (callable, optional): The function to be decorated. If None,
the decorator returns a partial function, allowing it to be called
with arguments. Defaults to None.
span_name (str, optional): Custom name for the span. If None,
it defaults to ``f'{func.__module__}.{func.__name__}'``.
Defaults to None.
kind (SpanKind, optional): The ``opentelemetry.trace.SpanKind`` for the
created span. Defaults to ``SpanKind.INTERNAL``.
attributes (dict, optional): A dictionary of static attributes to be
set on the span. Keys are attribute names (str) and values are
the corresponding attribute values. Defaults to None.
attribute_extractor (callable, optional): A function that can be used
to dynamically extract and set attributes on the span.
It is called within a ``finally`` block, ensuring it runs even if
the decorated function raises an exception.
The function signature should be:
``attribute_extractor(span, args, kwargs, result, exception)``
where:
- ``span`` : the OpenTelemetry ``Span`` object.
- ``args`` : a tuple of positional arguments passed
- ``kwargs`` : a dictionary of keyword arguments passed
- ``result`` : return value (None if an exception occurred)
- ``exception`` : exception object if raised (None otherwise).
Any exception raised by the ``attribute_extractor`` itself will be
caught and logged. Defaults to None.
Returns:
callable: The wrapped function that includes tracing, or a partial
decorator if ``func`` is None.
"""
if func is None:
return functools.partial(
trace_function,
span_name=span_name,
kind=kind,
attributes=attributes,
attribute_extractor=attribute_extractor,
)
actual_span_name = span_name or f'{func.__module__}.{func.__name__}'
is_async_func = inspect.iscoroutinefunction(func)
logger.debug(
'Start tracing for %s, is_async_func %s',
actual_span_name,
is_async_func,
)
@functools.wraps(func)
async def async_wrapper(*args, **kwargs) -> Any:
"""Async Wrapper for the decorator."""
logger.debug('Start async tracer')
tracer = trace.get_tracer(
INSTRUMENTING_MODULE, INSTRUMENTING_MODULE_VERSION
)
with tracer.start_as_current_span(actual_span_name, kind=kind) as span:
if attributes:
for k, v in attributes.items():
span.set_attribute(k, v)
result = None
exception = None
try:
# Async wrapper, await for the function call to complete.
result = await func(*args, **kwargs)
span.set_status(StatusCode.OK)
# asyncio.CancelledError extends from BaseException
except asyncio.CancelledError as ce:
exception = None
logger.debug('CancelledError in span %s', actual_span_name)
span.record_exception(ce)
raise
except Exception as e:
exception = e
span.record_exception(e)
span.set_status(StatusCode.ERROR, description=str(e))
raise
finally:
if attribute_extractor:
try:
attribute_extractor(
span, args, kwargs, result, exception
)
except Exception:
logger.exception(
'attribute_extractor error in span %s',
actual_span_name,
)
return result
@functools.wraps(func)
def sync_wrapper(*args, **kwargs) -> Any:
"""Sync Wrapper for the decorator."""
tracer = trace.get_tracer(INSTRUMENTING_MODULE)
with tracer.start_as_current_span(actual_span_name, kind=kind) as span:
if attributes:
for k, v in attributes.items():
span.set_attribute(k, v)
result = None
exception = None
try:
# Sync wrapper, execute the function call.
result = func(*args, **kwargs)
span.set_status(StatusCode.OK)
except Exception as e:
exception = e
span.record_exception(e)
span.set_status(StatusCode.ERROR, description=str(e))
raise
finally:
if attribute_extractor:
try:
attribute_extractor(
span, args, kwargs, result, exception
)
except Exception:
logger.exception(
'attribute_extractor error in span %s',
actual_span_name,
)
return result
return async_wrapper if is_async_func else sync_wrapper
def trace_class(
include_list: list[str] | None = None,
exclude_list: list[str] | None = None,
kind: SpanKindType = SpanKind.INTERNAL,
) -> Callable:
"""A class decorator to automatically trace specified methods of a class.
This decorator iterates over the methods of a class and applies the
`trace_function` decorator to them, based on the `include_list` and
`exclude_list` criteria. Methods starting or ending with double underscores
(dunder methods, e.g., `__init__`, `__call__`) are always excluded by default.
Args:
include_list (list[str], optional): A list of method names to
explicitly include for tracing. If provided, only methods in this
list (that are not dunder methods) will be traced.
Defaults to None (trace all non-dunder methods).
exclude_list (list[str], optional): A list of method names to exclude
from tracing. This is only considered if `include_list` is not
provided. Dunder methods are implicitly excluded.
Defaults to an empty list.
kind (SpanKind, optional): The `opentelemetry.trace.SpanKind` for the
created spans on the methods. Defaults to `SpanKind.INTERNAL`.
Returns:
callable: A decorator function that, when applied to a class,
modifies the class to wrap its specified methods with tracing.
Example:
To trace all methods except 'internal_method':
```python
@trace_class(exclude_list=['internal_method'])
class MyService:
def public_api(self):
pass
def internal_method(self):
pass
```
To trace only 'method_one' and 'method_two':
```python
@trace_class(include_list=['method_one', 'method_two'])
class AnotherService:
def method_one(self):
pass
def method_two(self):
pass
def not_traced_method(self):
pass
```
"""
logger.debug('Trace all class %s, %s', include_list, exclude_list)
exclude_list = exclude_list or []
def decorator(cls: Any) -> Any:
for name, method in inspect.getmembers(cls, inspect.isfunction):
if name.startswith('__') and name.endswith('__'):
continue
if include_list and name not in include_list:
continue
if not include_list and name in exclude_list:
continue
span_name = f'{cls.__module__}.{cls.__name__}.{name}'
setattr(
cls,
name,
trace_function(span_name=span_name, kind=kind)(method),
)
return cls
return decorator
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment