Skip to content
This repository has been archived by the owner on Oct 8, 2024. It is now read-only.

Commit

Permalink
fixed wrapping for not functools-wrapped methods (#156)
Browse files Browse the repository at this point in the history
  • Loading branch information
PietroPasotti authored Aug 1, 2024
1 parent 44e715f commit f1a6048
Show file tree
Hide file tree
Showing 2 changed files with 87 additions and 10 deletions.
40 changes: 30 additions & 10 deletions lib/charms/tempo_k8s/v1/charm_tracing.py
Original file line number Diff line number Diff line change
Expand Up @@ -219,7 +219,7 @@ def my_tracing_endpoint(self) -> Optional[str]:
# Increment this PATCH version before using `charmcraft publish-lib` or reset
# to 0 if you are raising the major API version

LIBPATCH = 12
LIBPATCH = 13

PYDEPS = ["opentelemetry-exporter-otlp-proto-http==1.21.0"]

Expand Down Expand Up @@ -642,38 +642,58 @@ def trace_type(cls: _T) -> _T:
dev_logger.info(f"skipping {method} (dunder)")
continue

new_method = trace_method(method)
if isinstance(inspect.getattr_static(cls, method.__name__), staticmethod):
# the span title in the general case should be:
# method call: MyCharmWrappedMethods.b
# if the method has a name (functools.wrapped or regular method), let
# _trace_callable use its default algorithm to determine what name to give the span.
trace_method_name = None
try:
qualname_c0 = method.__qualname__.split(".")[0]
if not hasattr(cls, method.__name__):
# if the callable doesn't have a __name__ (probably a decorated method),
# it probably has a bad qualname too (such as my_decorator.<locals>.wrapper) which is not
# great for finding out what the trace is about. So we use the method name instead and
# add a reference to the decorator name. Result:
# method call: @my_decorator(MyCharmWrappedMethods.b)
trace_method_name = f"@{qualname_c0}({cls.__name__}.{name})"
except Exception: # noqa: failsafe
pass

new_method = trace_method(method, name=trace_method_name)

if isinstance(inspect.getattr_static(cls, name), staticmethod):
new_method = staticmethod(new_method)
setattr(cls, name, new_method)

return cls


def trace_method(method: _F) -> _F:
def trace_method(method: _F, name: Optional[str] = None) -> _F:
"""Trace this method.
A span will be opened when this method is called and closed when it returns.
"""
return _trace_callable(method, "method")
return _trace_callable(method, "method", name=name)


def trace_function(function: _F) -> _F:
def trace_function(function: _F, name: Optional[str] = None) -> _F:
"""Trace this function.
A span will be opened when this function is called and closed when it returns.
"""
return _trace_callable(function, "function")
return _trace_callable(function, "function", name=name)


def _trace_callable(callable: _F, qualifier: str) -> _F:
def _trace_callable(callable: _F, qualifier: str, name: Optional[str] = None) -> _F:
dev_logger.info(f"instrumenting {callable}")

# sig = inspect.signature(callable)
@functools.wraps(callable)
def wrapped_function(*args, **kwargs): # type: ignore
name = getattr(callable, "__qualname__", getattr(callable, "__name__", str(callable)))
with _span(f"{qualifier} call: {name}"): # type: ignore
name_ = name or getattr(
callable, "__qualname__", getattr(callable, "__name__", str(callable))
)
with _span(f"{qualifier} call: {name_}"): # type: ignore
return callable(*args, **kwargs) # type: ignore

# wrapped_function.__signature__ = sig
Expand Down
57 changes: 57 additions & 0 deletions tests/scenario/test_charm_tracing.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
import functools
import logging
import os
from unittest.mock import patch
Expand Down Expand Up @@ -592,3 +593,59 @@ def test_inheritance_tracing(caplog):
ctx.run("start", State())
spans = f.call_args_list[0].args[0]
assert spans[0].name == "method call: SuperCharm.foo"


def bad_wrapper(func):
def wrapper(*args, **kwargs):
return func(*args, **kwargs)

return wrapper


def good_wrapper(func):
@functools.wraps(func)
def wrapper(*args, **kwargs):
return func(*args, **kwargs)

return wrapper


class MyCharmWrappedMethods(CharmBase):
META = {"name": "catgod"}

def __init__(self, fw):
super().__init__(fw)
fw.observe(self.on.start, self._on_start)

@good_wrapper
def a(self):
pass

@bad_wrapper
def b(self):
pass

def _on_start(self, _):
self.a()
self.b()

@property
def tempo(self):
return "foo.bar:80"


autoinstrument(MyCharmWrappedMethods, "tempo")


def test_wrapped_method_wrapping(caplog):
import opentelemetry

with patch(
"opentelemetry.exporter.otlp.proto.http.trace_exporter.OTLPSpanExporter.export"
) as f:
f.return_value = opentelemetry.sdk.trace.export.SpanExportResult.SUCCESS
ctx = Context(MyCharmWrappedMethods, meta=MyCharmWrappedMethods.META)
ctx.run("start", State())
spans = f.call_args_list[0].args[0]
assert spans[0].name == "method call: MyCharmWrappedMethods.a"
assert spans[1].name == "method call: @bad_wrapper(MyCharmWrappedMethods.b)"

0 comments on commit f1a6048

Please sign in to comment.