| Viewing file:  boto3.py (4.44 KB)      -rw-r--r-- Select action/file-type:
 
  (+) |  (+) |  (+) | Code (+) | Session (+) |  (+) | SDB (+) |  (+) |  (+) |  (+) |  (+) |  (+) | 
 
from __future__ import absolute_import
 from sentry_sdk import Hub
 from sentry_sdk.consts import OP, SPANDATA
 from sentry_sdk.integrations import Integration, DidNotEnable
 from sentry_sdk.tracing import Span
 
 from sentry_sdk._functools import partial
 from sentry_sdk._types import TYPE_CHECKING
 from sentry_sdk.utils import capture_internal_exceptions, parse_url, parse_version
 
 if TYPE_CHECKING:
 from typing import Any
 from typing import Dict
 from typing import Optional
 from typing import Type
 
 try:
 from botocore import __version__ as BOTOCORE_VERSION  # type: ignore
 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
 
 version = parse_version(BOTOCORE_VERSION)
 
 if version is None:
 raise DidNotEnable(
 "Unparsable botocore version: {}".format(BOTOCORE_VERSION)
 )
 
 if version < (1, 12):
 raise DidNotEnable("Botocore 1.12 or newer is required.")
 
 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=OP.HTTP_CLIENT,
 description=description,
 )
 
 with capture_internal_exceptions():
 parsed_url = parse_url(request.url, sanitize=False)
 span.set_data("aws.request.url", parsed_url.url)
 span.set_data(SPANDATA.HTTP_QUERY, parsed_url.query)
 span.set_data(SPANDATA.HTTP_FRAGMENT, parsed_url.fragment)
 
 span.set_tag("aws.service_id", service_id)
 span.set_tag("aws.operation_name", operation_name)
 span.set_data(SPANDATA.HTTP_METHOD, request.method)
 
 # 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=OP.HTTP_CLIENT_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)
 
 |