httpx.py 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  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. ensure_integration_enabled,
  15. logger,
  16. parse_url,
  17. )
  18. from typing import TYPE_CHECKING
  19. if TYPE_CHECKING:
  20. from typing import Any
  21. try:
  22. from httpx import AsyncClient, Client, Request, Response
  23. except ImportError:
  24. raise DidNotEnable("httpx is not installed")
  25. __all__ = ["HttpxIntegration"]
  26. class HttpxIntegration(Integration):
  27. identifier = "httpx"
  28. origin = f"auto.http.{identifier}"
  29. @staticmethod
  30. def setup_once() -> None:
  31. """
  32. httpx has its own transport layer and can be customized when needed,
  33. so patch Client.send and AsyncClient.send to support both synchronous and async interfaces.
  34. """
  35. _install_httpx_client()
  36. _install_httpx_async_client()
  37. def _install_httpx_client() -> None:
  38. real_send = Client.send
  39. @ensure_integration_enabled(HttpxIntegration, real_send)
  40. def send(self: "Client", request: "Request", **kwargs: "Any") -> "Response":
  41. parsed_url = None
  42. with capture_internal_exceptions():
  43. parsed_url = parse_url(str(request.url), sanitize=False)
  44. with start_span(
  45. op=OP.HTTP_CLIENT,
  46. name="%s %s"
  47. % (
  48. request.method,
  49. parsed_url.url if parsed_url else SENSITIVE_DATA_SUBSTITUTE,
  50. ),
  51. origin=HttpxIntegration.origin,
  52. ) as span:
  53. span.set_data(SPANDATA.HTTP_METHOD, request.method)
  54. if parsed_url is not None:
  55. span.set_data("url", parsed_url.url)
  56. span.set_data(SPANDATA.HTTP_QUERY, parsed_url.query)
  57. span.set_data(SPANDATA.HTTP_FRAGMENT, parsed_url.fragment)
  58. if should_propagate_trace(sentry_sdk.get_client(), str(request.url)):
  59. for (
  60. key,
  61. value,
  62. ) in sentry_sdk.get_current_scope().iter_trace_propagation_headers():
  63. logger.debug(
  64. "[Tracing] Adding `{key}` header {value} to outgoing request to {url}.".format(
  65. key=key, value=value, url=request.url
  66. )
  67. )
  68. if key == BAGGAGE_HEADER_NAME:
  69. add_sentry_baggage_to_headers(request.headers, value)
  70. else:
  71. request.headers[key] = value
  72. rv = real_send(self, request, **kwargs)
  73. span.set_http_status(rv.status_code)
  74. span.set_data("reason", rv.reason_phrase)
  75. with capture_internal_exceptions():
  76. add_http_request_source(span)
  77. return rv
  78. Client.send = send # type: ignore
  79. def _install_httpx_async_client() -> None:
  80. real_send = AsyncClient.send
  81. async def send(
  82. self: "AsyncClient", request: "Request", **kwargs: "Any"
  83. ) -> "Response":
  84. if sentry_sdk.get_client().get_integration(HttpxIntegration) is None:
  85. return await real_send(self, request, **kwargs)
  86. parsed_url = None
  87. with capture_internal_exceptions():
  88. parsed_url = parse_url(str(request.url), sanitize=False)
  89. with start_span(
  90. op=OP.HTTP_CLIENT,
  91. name="%s %s"
  92. % (
  93. request.method,
  94. parsed_url.url if parsed_url else SENSITIVE_DATA_SUBSTITUTE,
  95. ),
  96. origin=HttpxIntegration.origin,
  97. ) as span:
  98. span.set_data(SPANDATA.HTTP_METHOD, request.method)
  99. if parsed_url is not None:
  100. span.set_data("url", parsed_url.url)
  101. span.set_data(SPANDATA.HTTP_QUERY, parsed_url.query)
  102. span.set_data(SPANDATA.HTTP_FRAGMENT, parsed_url.fragment)
  103. if should_propagate_trace(sentry_sdk.get_client(), str(request.url)):
  104. for (
  105. key,
  106. value,
  107. ) in sentry_sdk.get_current_scope().iter_trace_propagation_headers():
  108. logger.debug(
  109. "[Tracing] Adding `{key}` header {value} to outgoing request to {url}.".format(
  110. key=key, value=value, url=request.url
  111. )
  112. )
  113. if key == BAGGAGE_HEADER_NAME:
  114. add_sentry_baggage_to_headers(request.headers, value)
  115. else:
  116. request.headers[key] = value
  117. rv = await real_send(self, request, **kwargs)
  118. span.set_http_status(rv.status_code)
  119. span.set_data("reason", rv.reason_phrase)
  120. with capture_internal_exceptions():
  121. add_http_request_source(span)
  122. return rv
  123. AsyncClient.send = send # type: ignore