Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. import re
  2. from collections.abc import Mapping
  3. from typing import Any, Optional
  4. def is_valid_trace_id(trace_id: str) -> bool:
  5. """
  6. Check if the trace_id is valid.
  7. Requirements: 1-128 characters, only letters, numbers, '-', and '_'.
  8. """
  9. return bool(re.match(r"^[a-zA-Z0-9\-_]{1,128}$", trace_id))
  10. def get_external_trace_id(request: Any) -> Optional[str]:
  11. """
  12. Retrieve the trace_id from the request.
  13. Priority:
  14. 1. header ('X-Trace-Id')
  15. 2. parameters
  16. 3. JSON body
  17. 4. Current OpenTelemetry context (if enabled)
  18. 5. OpenTelemetry traceparent header (if present and valid)
  19. Returns None if no valid trace_id is provided.
  20. """
  21. trace_id = request.headers.get("X-Trace-Id")
  22. if not trace_id:
  23. trace_id = request.args.get("trace_id")
  24. if not trace_id and getattr(request, "is_json", False):
  25. json_data = getattr(request, "json", None)
  26. if json_data:
  27. trace_id = json_data.get("trace_id")
  28. if not trace_id:
  29. trace_id = get_trace_id_from_otel_context()
  30. if not trace_id:
  31. traceparent = request.headers.get("traceparent")
  32. if traceparent:
  33. trace_id = parse_traceparent_header(traceparent)
  34. if isinstance(trace_id, str) and is_valid_trace_id(trace_id):
  35. return trace_id
  36. return None
  37. def extract_external_trace_id_from_args(args: Mapping[str, Any]) -> dict:
  38. """
  39. Extract 'external_trace_id' from args.
  40. Returns a dict suitable for use in extras. Returns an empty dict if not found.
  41. """
  42. trace_id = args.get("external_trace_id")
  43. if trace_id:
  44. return {"external_trace_id": trace_id}
  45. return {}
  46. def get_trace_id_from_otel_context() -> Optional[str]:
  47. """
  48. Retrieve the current trace ID from the active OpenTelemetry trace context.
  49. Returns None if:
  50. 1. OpenTelemetry SDK is not installed or enabled.
  51. 2. There is no active span or trace context.
  52. """
  53. try:
  54. from opentelemetry.trace import SpanContext, get_current_span
  55. from opentelemetry.trace.span import INVALID_TRACE_ID
  56. span = get_current_span()
  57. if not span:
  58. return None
  59. span_context: SpanContext = span.get_span_context()
  60. if not span_context or span_context.trace_id == INVALID_TRACE_ID:
  61. return None
  62. trace_id_hex = f"{span_context.trace_id:032x}"
  63. return trace_id_hex
  64. except Exception:
  65. return None
  66. def parse_traceparent_header(traceparent: str) -> Optional[str]:
  67. """
  68. Parse the `traceparent` header to extract the trace_id.
  69. Expected format:
  70. 'version-trace_id-span_id-flags'
  71. Reference:
  72. W3C Trace Context Specification: https://www.w3.org/TR/trace-context/
  73. """
  74. try:
  75. parts = traceparent.split("-")
  76. if len(parts) == 4 and len(parts[1]) == 32:
  77. return parts[1]
  78. except Exception:
  79. pass
  80. return None