Viewing file: boto3.py (3.64 KB) -rw-r--r-- Select action/file-type: (+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
from __future__ import absolute_import
from sentry_sdk import Hub from sentry_sdk.integrations import Integration, DidNotEnable from sentry_sdk.tracing import Span
from sentry_sdk._functools import partial from sentry_sdk._types import MYPY
if MYPY: from typing import Any from typing import Dict from typing import Optional from typing import Type
try: from botocore.client import BaseClient # type: ignore from botocore.response import StreamingBody # type: ignore from botocore.awsrequest import AWSRequest # type: ignore except ImportError: raise DidNotEnable("botocore is not installed")
class Boto3Integration(Integration): identifier = "boto3"
@staticmethod def setup_once(): # type: () -> None orig_init = BaseClient.__init__
def sentry_patched_init(self, *args, **kwargs): # type: (Type[BaseClient], *Any, **Any) -> None orig_init(self, *args, **kwargs) meta = self.meta service_id = meta.service_model.service_id.hyphenize() meta.events.register( "request-created", partial(_sentry_request_created, service_id=service_id), ) meta.events.register("after-call", _sentry_after_call) meta.events.register("after-call-error", _sentry_after_call_error)
BaseClient.__init__ = sentry_patched_init
def _sentry_request_created(service_id, request, operation_name, **kwargs): # type: (str, AWSRequest, str, **Any) -> None hub = Hub.current if hub.get_integration(Boto3Integration) is None: return
description = "aws.%s.%s" % (service_id, operation_name) span = hub.start_span( hub=hub, op="aws.request", description=description, ) span.set_tag("aws.service_id", service_id) span.set_tag("aws.operation_name", operation_name) span.set_data("aws.request.url", request.url)
# We do it in order for subsequent http calls/retries be # attached to this span. span.__enter__()
# request.context is an open-ended data-structure # where we can add anything useful in request life cycle. request.context["_sentrysdk_span"] = span
def _sentry_after_call(context, parsed, **kwargs): # type: (Dict[str, Any], Dict[str, Any], **Any) -> None span = context.pop("_sentrysdk_span", None) # type: Optional[Span]
# Span could be absent if the integration is disabled. if span is None: return span.__exit__(None, None, None)
body = parsed.get("Body") if not isinstance(body, StreamingBody): return
streaming_span = span.start_child( op="aws.request.stream", description=span.description, )
orig_read = body.read orig_close = body.close
def sentry_streaming_body_read(*args, **kwargs): # type: (*Any, **Any) -> bytes try: ret = orig_read(*args, **kwargs) if not ret: streaming_span.finish() return ret except Exception: streaming_span.finish() raise
body.read = sentry_streaming_body_read
def sentry_streaming_body_close(*args, **kwargs): # type: (*Any, **Any) -> None streaming_span.finish() orig_close(*args, **kwargs)
body.close = sentry_streaming_body_close
def _sentry_after_call_error(context, exception, **kwargs): # type: (Dict[str, Any], Type[BaseException], **Any) -> None span = context.pop("_sentrysdk_span", None) # type: Optional[Span]
# Span could be absent if the integration is disabled. if span is None: return span.__exit__(type(exception), exception, None)
|