-
Notifications
You must be signed in to change notification settings - Fork 416
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(aiohttp): add client integration (#3362)
This PR adds support for aiohttp client. Work was long ago started on this in (#294 (thank you @thehesiod!!!). There were a number of issues in the library that led to not being able to merge in the work such as async context management, service naming, integration configuration and more which have all since been addressed. #294 and later #1372 included additional support for ClientResponse and StreamReader which are omitted here with the intention of introducing them as follow ups. Co-authored-by: Alexander Mohr <[email protected]>
- Loading branch information
1 parent
9dab78a
commit 2ac1fc9
Showing
18 changed files
with
803 additions
and
13 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,25 +1,139 @@ | ||
import os | ||
|
||
from yarl import URL | ||
|
||
from ddtrace import config | ||
from ddtrace.internal.logger import get_logger | ||
from ddtrace.internal.utils import get_argument_value | ||
from ddtrace.internal.utils.formats import asbool | ||
from ddtrace.vendor import wrapt | ||
|
||
from ...ext import SpanTypes | ||
from ...internal.compat import parse | ||
from ...pin import Pin | ||
from ...propagation.http import HTTPPropagator | ||
from ..trace_utils import ext_service | ||
from ..trace_utils import set_http_meta | ||
from ..trace_utils import unwrap | ||
from ..trace_utils import with_traced_module as with_traced_module_sync | ||
from ..trace_utils import wrap | ||
from ..trace_utils_async import with_traced_module | ||
|
||
|
||
log = get_logger(__name__) | ||
|
||
|
||
# Server config | ||
config._add( | ||
"aiohttp", | ||
dict(distributed_tracing=True), | ||
) | ||
|
||
config._add( | ||
"aiohttp_client", | ||
dict( | ||
distributed_tracing=True, | ||
distributed_tracing=asbool(os.getenv("DD_AIOHTTP_CLIENT_DISTRIBUTED_TRACING", True)), | ||
), | ||
) | ||
|
||
|
||
class _WrappedConnectorClass(wrapt.ObjectProxy): | ||
def __init__(self, obj, pin): | ||
super().__init__(obj) | ||
pin.onto(self) | ||
|
||
async def connect(self, req, *args, **kwargs): | ||
pin = Pin.get_from(self) | ||
with pin.tracer.trace("%s.connect" % self.__class__.__name__): | ||
result = await self.__wrapped__.connect(req, *args, **kwargs) | ||
return result | ||
|
||
async def _create_connection(self, req, *args, **kwargs): | ||
pin = Pin.get_from(self) | ||
with pin.tracer.trace("%s._create_connection" % self.__class__.__name__): | ||
result = await self.__wrapped__._create_connection(req, *args, **kwargs) | ||
return result | ||
|
||
|
||
@with_traced_module | ||
async def _traced_clientsession_request(aiohttp, pin, func, instance, args, kwargs): | ||
method = get_argument_value(args, kwargs, 0, "method") # type: str | ||
url = URL(get_argument_value(args, kwargs, 1, "url")) # type: URL | ||
params = kwargs.get("params") | ||
headers = kwargs.get("headers") or {} | ||
|
||
with pin.tracer.trace( | ||
"aiohttp.request", span_type=SpanTypes.HTTP, service=ext_service(pin, config.aiohttp_client) | ||
) as span: | ||
if pin._config["distributed_tracing"]: | ||
HTTPPropagator.inject(span.context, headers) | ||
kwargs["headers"] = headers | ||
|
||
# Params can be included separate of the URL so the URL has to be constructed | ||
# with the passed params. | ||
url_str = str(url.update_query(params) if params else url) | ||
parsed_url = parse.urlparse(url_str) | ||
set_http_meta( | ||
span, | ||
config.aiohttp_client, | ||
method=method, | ||
url=url_str, | ||
query=parsed_url.query, | ||
request_headers=headers, | ||
) | ||
resp = await func(*args, **kwargs) # type: aiohttp.ClientResponse | ||
set_http_meta( | ||
span, config.aiohttp_client, response_headers=resp.headers, status_code=resp.status, status_msg=resp.reason | ||
) | ||
return resp | ||
|
||
|
||
@with_traced_module_sync | ||
def _traced_clientsession_init(aiohttp, pin, func, instance, args, kwargs): | ||
func(*args, **kwargs) | ||
instance._connector = _WrappedConnectorClass(instance._connector, pin) | ||
|
||
|
||
def _patch_client(aiohttp): | ||
Pin().onto(aiohttp) | ||
pin = Pin(_config=config.aiohttp_client.copy()) | ||
pin.onto(aiohttp.ClientSession) | ||
|
||
wrap("aiohttp", "ClientSession.__init__", _traced_clientsession_init(aiohttp)) | ||
wrap("aiohttp", "ClientSession._request", _traced_clientsession_request(aiohttp)) | ||
|
||
|
||
def patch(): | ||
# Legacy patch aiohttp_jinja2 | ||
from ddtrace.contrib.aiohttp_jinja2 import patch as aiohttp_jinja2_patch | ||
|
||
aiohttp_jinja2_patch() | ||
|
||
import aiohttp | ||
|
||
if getattr(aiohttp, "_datadog_patch", False): | ||
return | ||
|
||
_patch_client(aiohttp) | ||
|
||
setattr(aiohttp, "_datadog_patch", True) | ||
|
||
|
||
def _unpatch_client(aiohttp): | ||
unwrap(aiohttp.ClientSession, "__init__") | ||
unwrap(aiohttp.ClientSession, "_request") | ||
|
||
|
||
def unpatch(): | ||
from ddtrace.contrib.aiohttp_jinja2 import unpatch as aiohttp_jinja2_unpatch | ||
|
||
aiohttp_jinja2_unpatch() | ||
|
||
import aiohttp | ||
|
||
if not getattr(aiohttp, "_datadog_patch", False): | ||
return | ||
|
||
_unpatch_client(aiohttp) | ||
|
||
setattr(aiohttp, "_datadog_patch", False) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,39 @@ | ||
""" | ||
async tracing utils | ||
Note that this module should only be imported in Python 3.5+. | ||
""" | ||
from ddtrace import Pin | ||
from ddtrace.internal.logger import get_logger | ||
|
||
|
||
log = get_logger(__name__) | ||
|
||
|
||
def with_traced_module(func): | ||
"""Async version of trace_utils.with_traced_module. | ||
Usage:: | ||
@with_traced_module | ||
async def my_traced_wrapper(django, pin, func, instance, args, kwargs): | ||
# Do tracing stuff | ||
pass | ||
def patch(): | ||
import django | ||
wrap(django.somefunc, my_traced_wrapper(django)) | ||
""" | ||
|
||
def with_mod(mod): | ||
async def wrapper(wrapped, instance, args, kwargs): | ||
pin = Pin._find(instance, mod) | ||
if pin and not pin.enabled(): | ||
return await wrapped(*args, **kwargs) | ||
elif not pin: | ||
log.debug("Pin not found for traced method %r", wrapped) | ||
return await wrapped(*args, **kwargs) | ||
return await func(mod, pin, wrapped, instance, args, kwargs) | ||
|
||
return wrapper | ||
|
||
return with_mod |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
--- | ||
features: | ||
- | | ||
aiohttp: add client integration. This integration traces requests made using the aiohttp client and includes support | ||
for distributed tracing. See `the documentation <https://ddtrace.readthedocs.io/en/stable/integrations.html#aiohttp>`_ | ||
for more information. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.