aboutsummaryrefslogtreecommitdiff
path: root/.venv/lib/python3.12/site-packages/sentry_sdk/integrations/boto3.py
diff options
context:
space:
mode:
authorS. Solomon Darnell2025-03-28 21:52:21 -0500
committerS. Solomon Darnell2025-03-28 21:52:21 -0500
commit4a52a71956a8d46fcb7294ac71734504bb09bcc2 (patch)
treeee3dc5af3b6313e921cd920906356f5d4febc4ed /.venv/lib/python3.12/site-packages/sentry_sdk/integrations/boto3.py
parentcc961e04ba734dd72309fb548a2f97d67d578813 (diff)
downloadgn-ai-master.tar.gz
two version of R2R are hereHEADmaster
Diffstat (limited to '.venv/lib/python3.12/site-packages/sentry_sdk/integrations/boto3.py')
-rw-r--r--.venv/lib/python3.12/site-packages/sentry_sdk/integrations/boto3.py137
1 files changed, 137 insertions, 0 deletions
diff --git a/.venv/lib/python3.12/site-packages/sentry_sdk/integrations/boto3.py b/.venv/lib/python3.12/site-packages/sentry_sdk/integrations/boto3.py
new file mode 100644
index 00000000..0207341f
--- /dev/null
+++ b/.venv/lib/python3.12/site-packages/sentry_sdk/integrations/boto3.py
@@ -0,0 +1,137 @@
+from functools import partial
+
+import sentry_sdk
+from sentry_sdk.consts import OP, SPANDATA
+from sentry_sdk.integrations import _check_minimum_version, Integration, DidNotEnable
+from sentry_sdk.tracing import Span
+from sentry_sdk.utils import (
+ capture_internal_exceptions,
+ ensure_integration_enabled,
+ parse_url,
+ parse_version,
+)
+
+from typing import TYPE_CHECKING
+
+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"
+ origin = f"auto.http.{identifier}"
+
+ @staticmethod
+ def setup_once():
+ # type: () -> None
+ version = parse_version(BOTOCORE_VERSION)
+ _check_minimum_version(Boto3Integration, version, "botocore")
+
+ 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
+
+
+@ensure_integration_enabled(Boto3Integration)
+def _sentry_request_created(service_id, request, operation_name, **kwargs):
+ # type: (str, AWSRequest, str, **Any) -> None
+ description = "aws.%s.%s" % (service_id, operation_name)
+ span = sentry_sdk.start_span(
+ op=OP.HTTP_CLIENT,
+ name=description,
+ origin=Boto3Integration.origin,
+ )
+
+ 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,
+ name=span.description,
+ origin=Boto3Integration.origin,
+ )
+
+ 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)