pyreqwest.py 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. import sentry_sdk
  2. from sentry_sdk import start_span
  3. from sentry_sdk.consts import OP, SPANDATA
  4. from sentry_sdk.integrations import Integration, DidNotEnable
  5. from sentry_sdk.tracing import BAGGAGE_HEADER_NAME
  6. from sentry_sdk.tracing_utils import (
  7. should_propagate_trace,
  8. add_http_request_source,
  9. add_sentry_baggage_to_headers,
  10. )
  11. from sentry_sdk.utils import (
  12. SENSITIVE_DATA_SUBSTITUTE,
  13. capture_internal_exceptions,
  14. logger,
  15. parse_url,
  16. )
  17. from contextlib import contextmanager
  18. from typing import Any, Generator
  19. try:
  20. from pyreqwest.client import ClientBuilder, SyncClientBuilder # type: ignore[import-not-found]
  21. from pyreqwest.request import ( # type: ignore[import-not-found]
  22. Request,
  23. OneOffRequestBuilder,
  24. SyncOneOffRequestBuilder,
  25. )
  26. from pyreqwest.middleware import Next, SyncNext # type: ignore[import-not-found]
  27. from pyreqwest.response import Response, SyncResponse # type: ignore[import-not-found]
  28. except ImportError:
  29. raise DidNotEnable("pyreqwest not installed or incompatible version installed")
  30. class PyreqwestIntegration(Integration):
  31. identifier = "pyreqwest"
  32. origin = f"auto.http.{identifier}"
  33. @staticmethod
  34. def setup_once() -> None:
  35. _patch_pyreqwest()
  36. def _patch_pyreqwest() -> None:
  37. # Patch Client Builders
  38. _patch_builder_method(ClientBuilder, "build", sentry_async_middleware)
  39. _patch_builder_method(SyncClientBuilder, "build", sentry_sync_middleware)
  40. # Patch Request Builders
  41. _patch_builder_method(OneOffRequestBuilder, "send", sentry_async_middleware)
  42. _patch_builder_method(SyncOneOffRequestBuilder, "send", sentry_sync_middleware)
  43. def _patch_builder_method(cls: type, method_name: str, middleware: "Any") -> None:
  44. if not hasattr(cls, method_name):
  45. return
  46. original_method = getattr(cls, method_name)
  47. def sentry_patched_method(self: "Any", *args: "Any", **kwargs: "Any") -> "Any":
  48. if not getattr(self, "_sentry_instrumented", False):
  49. integration = sentry_sdk.get_client().get_integration(PyreqwestIntegration)
  50. if integration is not None:
  51. self.with_middleware(middleware)
  52. try:
  53. self._sentry_instrumented = True
  54. except (TypeError, AttributeError):
  55. # In case the instance itself is immutable or doesn't allow extra attributes
  56. pass
  57. return original_method(self, *args, **kwargs)
  58. setattr(cls, method_name, sentry_patched_method)
  59. @contextmanager
  60. def _sentry_pyreqwest_span(request: "Request") -> "Generator[Any, None, None]":
  61. parsed_url = None
  62. with capture_internal_exceptions():
  63. parsed_url = parse_url(str(request.url), sanitize=False)
  64. with start_span(
  65. op=OP.HTTP_CLIENT,
  66. name=f"{request.method} {parsed_url.url if parsed_url else SENSITIVE_DATA_SUBSTITUTE}",
  67. origin=PyreqwestIntegration.origin,
  68. ) as span:
  69. span.set_data(SPANDATA.HTTP_METHOD, request.method)
  70. if parsed_url is not None:
  71. span.set_data("url", parsed_url.url)
  72. span.set_data(SPANDATA.HTTP_QUERY, parsed_url.query)
  73. span.set_data(SPANDATA.HTTP_FRAGMENT, parsed_url.fragment)
  74. if should_propagate_trace(sentry_sdk.get_client(), str(request.url)):
  75. for (
  76. key,
  77. value,
  78. ) in sentry_sdk.get_current_scope().iter_trace_propagation_headers():
  79. logger.debug(
  80. "[Tracing] Adding `{key}` header {value} to outgoing request to {url}.".format(
  81. key=key, value=value, url=request.url
  82. )
  83. )
  84. if key == BAGGAGE_HEADER_NAME:
  85. add_sentry_baggage_to_headers(request.headers, value)
  86. else:
  87. request.headers[key] = value
  88. yield span
  89. with capture_internal_exceptions():
  90. add_http_request_source(span)
  91. async def sentry_async_middleware(
  92. request: "Request", next_handler: "Next"
  93. ) -> "Response":
  94. if sentry_sdk.get_client().get_integration(PyreqwestIntegration) is None:
  95. return await next_handler.run(request)
  96. with _sentry_pyreqwest_span(request) as span:
  97. response = await next_handler.run(request)
  98. span.set_http_status(response.status)
  99. return response
  100. def sentry_sync_middleware(
  101. request: "Request", next_handler: "SyncNext"
  102. ) -> "SyncResponse":
  103. if sentry_sdk.get_client().get_integration(PyreqwestIntegration) is None:
  104. return next_handler.run(request)
  105. with _sentry_pyreqwest_span(request) as span:
  106. response = next_handler.run(request)
  107. span.set_http_status(response.status)
  108. return response