boto3.py 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. from functools import partial
  2. import sentry_sdk
  3. from sentry_sdk.consts import OP, SPANDATA
  4. from sentry_sdk.integrations import _check_minimum_version, Integration, DidNotEnable
  5. from sentry_sdk.tracing import Span
  6. from sentry_sdk.utils import (
  7. capture_internal_exceptions,
  8. ensure_integration_enabled,
  9. parse_url,
  10. parse_version,
  11. )
  12. from typing import TYPE_CHECKING
  13. if TYPE_CHECKING:
  14. from typing import Any
  15. from typing import Dict
  16. from typing import Optional
  17. from typing import Type
  18. try:
  19. from botocore import __version__ as BOTOCORE_VERSION # type: ignore
  20. from botocore.client import BaseClient # type: ignore
  21. from botocore.response import StreamingBody # type: ignore
  22. from botocore.awsrequest import AWSRequest # type: ignore
  23. except ImportError:
  24. raise DidNotEnable("botocore is not installed")
  25. class Boto3Integration(Integration):
  26. identifier = "boto3"
  27. origin = f"auto.http.{identifier}"
  28. @staticmethod
  29. def setup_once() -> None:
  30. version = parse_version(BOTOCORE_VERSION)
  31. _check_minimum_version(Boto3Integration, version, "botocore")
  32. orig_init = BaseClient.__init__
  33. def sentry_patched_init(
  34. self: "Type[BaseClient]", *args: "Any", **kwargs: "Any"
  35. ) -> None:
  36. orig_init(self, *args, **kwargs)
  37. meta = self.meta
  38. service_id = meta.service_model.service_id.hyphenize()
  39. meta.events.register(
  40. "request-created",
  41. partial(_sentry_request_created, service_id=service_id),
  42. )
  43. meta.events.register("after-call", _sentry_after_call)
  44. meta.events.register("after-call-error", _sentry_after_call_error)
  45. BaseClient.__init__ = sentry_patched_init
  46. @ensure_integration_enabled(Boto3Integration)
  47. def _sentry_request_created(
  48. service_id: str, request: "AWSRequest", operation_name: str, **kwargs: "Any"
  49. ) -> None:
  50. description = "aws.%s.%s" % (service_id, operation_name)
  51. span = sentry_sdk.start_span(
  52. op=OP.HTTP_CLIENT,
  53. name=description,
  54. origin=Boto3Integration.origin,
  55. )
  56. with capture_internal_exceptions():
  57. parsed_url = parse_url(request.url, sanitize=False)
  58. span.set_data("aws.request.url", parsed_url.url)
  59. span.set_data(SPANDATA.HTTP_QUERY, parsed_url.query)
  60. span.set_data(SPANDATA.HTTP_FRAGMENT, parsed_url.fragment)
  61. span.set_tag("aws.service_id", service_id)
  62. span.set_tag("aws.operation_name", operation_name)
  63. span.set_data(SPANDATA.HTTP_METHOD, request.method)
  64. # We do it in order for subsequent http calls/retries be
  65. # attached to this span.
  66. span.__enter__()
  67. # request.context is an open-ended data-structure
  68. # where we can add anything useful in request life cycle.
  69. request.context["_sentrysdk_span"] = span
  70. def _sentry_after_call(
  71. context: "Dict[str, Any]", parsed: "Dict[str, Any]", **kwargs: "Any"
  72. ) -> None:
  73. span: "Optional[Span]" = context.pop("_sentrysdk_span", None)
  74. # Span could be absent if the integration is disabled.
  75. if span is None:
  76. return
  77. span.__exit__(None, None, None)
  78. body = parsed.get("Body")
  79. if not isinstance(body, StreamingBody):
  80. return
  81. streaming_span = span.start_child(
  82. op=OP.HTTP_CLIENT_STREAM,
  83. name=span.description,
  84. origin=Boto3Integration.origin,
  85. )
  86. orig_read = body.read
  87. orig_close = body.close
  88. def sentry_streaming_body_read(*args: "Any", **kwargs: "Any") -> bytes:
  89. try:
  90. ret = orig_read(*args, **kwargs)
  91. if not ret:
  92. streaming_span.finish()
  93. return ret
  94. except Exception:
  95. streaming_span.finish()
  96. raise
  97. body.read = sentry_streaming_body_read
  98. def sentry_streaming_body_close(*args: "Any", **kwargs: "Any") -> None:
  99. streaming_span.finish()
  100. orig_close(*args, **kwargs)
  101. body.close = sentry_streaming_body_close
  102. def _sentry_after_call_error(
  103. context: "Dict[str, Any]", exception: "Type[BaseException]", **kwargs: "Any"
  104. ) -> None:
  105. span: "Optional[Span]" = context.pop("_sentrysdk_span", None)
  106. # Span could be absent if the integration is disabled.
  107. if span is None:
  108. return
  109. span.__exit__(type(exception), exception, None)