ddtrace 1.12.1 → 1.14.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (275) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +158 -9
  3. data/ext/ddtrace_profiling_native_extension/collectors_thread_context.c +97 -14
  4. data/ext/ddtrace_profiling_native_extension/extconf.rb +6 -0
  5. data/ext/ddtrace_profiling_native_extension/http_transport.c +19 -6
  6. data/ext/ddtrace_profiling_native_extension/native_extension_helpers.rb +16 -1
  7. data/ext/ddtrace_profiling_native_extension/private_vm_api_access.c +41 -2
  8. data/ext/ddtrace_profiling_native_extension/private_vm_api_access.h +6 -0
  9. data/ext/ddtrace_profiling_native_extension/stack_recorder.c +6 -10
  10. data/ext/ddtrace_profiling_native_extension/time_helpers.c +40 -4
  11. data/ext/ddtrace_profiling_native_extension/time_helpers.h +14 -0
  12. data/lib/datadog/appsec/component.rb +9 -0
  13. data/lib/datadog/appsec/configuration/settings.rb +110 -195
  14. data/lib/datadog/appsec/configuration.rb +0 -79
  15. data/lib/datadog/appsec/contrib/auto_instrument.rb +2 -4
  16. data/lib/datadog/appsec/contrib/devise/event.rb +57 -0
  17. data/lib/datadog/appsec/contrib/devise/ext.rb +13 -0
  18. data/lib/datadog/appsec/contrib/devise/integration.rb +42 -0
  19. data/lib/datadog/appsec/contrib/devise/patcher/authenticatable_patch.rb +76 -0
  20. data/lib/datadog/appsec/contrib/devise/patcher/registration_controller_patch.rb +52 -0
  21. data/lib/datadog/appsec/contrib/devise/patcher.rb +45 -0
  22. data/lib/datadog/appsec/contrib/devise/resource.rb +35 -0
  23. data/lib/datadog/appsec/contrib/devise/tracking.rb +49 -0
  24. data/lib/datadog/appsec/contrib/rack/ext.rb +2 -1
  25. data/lib/datadog/appsec/contrib/rack/reactive/request.rb +1 -1
  26. data/lib/datadog/appsec/contrib/rack/reactive/request_body.rb +1 -1
  27. data/lib/datadog/appsec/contrib/rack/reactive/response.rb +1 -1
  28. data/lib/datadog/appsec/contrib/rack/request_middleware.rb +12 -7
  29. data/lib/datadog/appsec/contrib/rails/ext.rb +3 -2
  30. data/lib/datadog/appsec/contrib/rails/framework.rb +1 -3
  31. data/lib/datadog/appsec/contrib/rails/patcher.rb +8 -8
  32. data/lib/datadog/appsec/contrib/rails/reactive/action.rb +1 -1
  33. data/lib/datadog/appsec/contrib/sinatra/ext.rb +2 -1
  34. data/lib/datadog/appsec/contrib/sinatra/framework.rb +1 -3
  35. data/lib/datadog/appsec/contrib/sinatra/reactive/routed.rb +1 -1
  36. data/lib/datadog/appsec/event.rb +1 -1
  37. data/lib/datadog/appsec/extensions.rb +1 -130
  38. data/lib/datadog/appsec/monitor/reactive/set_user.rb +1 -1
  39. data/lib/datadog/appsec/processor.rb +1 -1
  40. data/lib/datadog/appsec/rate_limiter.rb +1 -1
  41. data/lib/datadog/appsec/remote.rb +1 -1
  42. data/lib/datadog/appsec.rb +1 -2
  43. data/lib/datadog/ci/configuration/settings.rb +6 -8
  44. data/lib/datadog/ci/contrib/cucumber/configuration/settings.rb +7 -5
  45. data/lib/datadog/ci/contrib/cucumber/ext.rb +10 -8
  46. data/lib/datadog/ci/contrib/minitest/configuration/settings.rb +35 -0
  47. data/lib/datadog/ci/contrib/minitest/ext.rb +21 -0
  48. data/lib/datadog/ci/contrib/minitest/integration.rb +49 -0
  49. data/lib/datadog/ci/contrib/minitest/patcher.rb +27 -0
  50. data/lib/datadog/ci/contrib/minitest/test_helper.rb +68 -0
  51. data/lib/datadog/ci/contrib/rspec/configuration/settings.rb +7 -5
  52. data/lib/datadog/ci/contrib/rspec/ext.rb +9 -7
  53. data/lib/datadog/ci.rb +1 -0
  54. data/lib/datadog/core/backport.rb +51 -0
  55. data/lib/datadog/core/configuration/agent_settings_resolver.rb +9 -5
  56. data/lib/datadog/core/configuration/base.rb +5 -5
  57. data/lib/datadog/core/configuration/components.rb +8 -1
  58. data/lib/datadog/core/configuration/ext.rb +7 -5
  59. data/lib/datadog/core/configuration/option.rb +269 -19
  60. data/lib/datadog/core/configuration/option_definition.rb +76 -11
  61. data/lib/datadog/core/configuration/options.rb +22 -10
  62. data/lib/datadog/core/configuration/settings.rb +139 -61
  63. data/lib/datadog/core/diagnostics/environment_logger.rb +130 -234
  64. data/lib/datadog/core/environment/execution.rb +65 -0
  65. data/lib/datadog/core/environment/ext.rb +13 -11
  66. data/lib/datadog/core/environment/yjit.rb +58 -0
  67. data/lib/datadog/core/git/ext.rb +24 -22
  68. data/lib/datadog/core/logging/ext.rb +3 -1
  69. data/lib/datadog/core/metrics/ext.rb +7 -5
  70. data/lib/datadog/core/remote/client/capabilities.rb +5 -0
  71. data/lib/datadog/core/remote/client.rb +3 -0
  72. data/lib/datadog/core/remote/component.rb +25 -34
  73. data/lib/datadog/core/remote/configuration/content.rb +28 -1
  74. data/lib/datadog/core/remote/configuration/repository.rb +3 -1
  75. data/lib/datadog/core/remote/ext.rb +1 -1
  76. data/lib/datadog/core/remote/negotiation.rb +17 -4
  77. data/lib/datadog/core/runtime/ext.rb +22 -12
  78. data/lib/datadog/core/runtime/metrics.rb +43 -0
  79. data/lib/datadog/core/telemetry/client.rb +12 -2
  80. data/lib/datadog/core/telemetry/collector.rb +10 -2
  81. data/lib/datadog/core/telemetry/emitter.rb +4 -2
  82. data/lib/datadog/core/telemetry/event.rb +19 -4
  83. data/lib/datadog/core/telemetry/ext.rb +4 -1
  84. data/lib/datadog/core/telemetry/heartbeat.rb +2 -4
  85. data/lib/datadog/core/telemetry/http/ext.rb +10 -8
  86. data/lib/datadog/core/telemetry/http/transport.rb +1 -0
  87. data/lib/datadog/core/telemetry/v2/app_client_configuration_change.rb +41 -0
  88. data/lib/datadog/core/telemetry/v2/request.rb +29 -0
  89. data/lib/datadog/core/transport/http/client.rb +1 -1
  90. data/lib/datadog/core/transport/http/config.rb +10 -0
  91. data/lib/datadog/core/utils/duration.rb +52 -0
  92. data/lib/datadog/core/utils/hash.rb +47 -0
  93. data/lib/datadog/core/utils/network.rb +1 -1
  94. data/lib/datadog/core/utils/safe_dup.rb +27 -20
  95. data/lib/datadog/core/utils.rb +1 -1
  96. data/lib/datadog/core/workers/async.rb +2 -2
  97. data/lib/datadog/kit/appsec/events.rb +139 -89
  98. data/lib/datadog/kit/identity.rb +80 -65
  99. data/lib/datadog/profiling/collectors/cpu_and_wall_time_worker.rb +3 -0
  100. data/lib/datadog/profiling/collectors/idle_sampling_helper.rb +1 -0
  101. data/lib/datadog/profiling/collectors/thread_context.rb +9 -2
  102. data/lib/datadog/profiling/component.rb +51 -9
  103. data/lib/datadog/profiling/diagnostics/environment_logger.rb +39 -0
  104. data/lib/datadog/profiling/exporter.rb +5 -1
  105. data/lib/datadog/profiling/flush.rb +7 -2
  106. data/lib/datadog/profiling/http_transport.rb +13 -3
  107. data/lib/datadog/profiling/load_native_extension.rb +7 -1
  108. data/lib/datadog/profiling.rb +12 -1
  109. data/lib/datadog/tracing/component.rb +60 -7
  110. data/lib/datadog/tracing/configuration/dynamic/option.rb +71 -0
  111. data/lib/datadog/tracing/configuration/dynamic.rb +64 -0
  112. data/lib/datadog/tracing/configuration/ext.rb +35 -32
  113. data/lib/datadog/tracing/configuration/http.rb +74 -0
  114. data/lib/datadog/tracing/configuration/settings.rb +106 -92
  115. data/lib/datadog/tracing/contrib/action_cable/configuration/settings.rb +9 -6
  116. data/lib/datadog/tracing/contrib/action_cable/ext.rb +20 -18
  117. data/lib/datadog/tracing/contrib/action_mailer/configuration/settings.rb +9 -6
  118. data/lib/datadog/tracing/contrib/action_mailer/ext.rb +20 -18
  119. data/lib/datadog/tracing/contrib/action_pack/configuration/settings.rb +8 -6
  120. data/lib/datadog/tracing/contrib/action_pack/ext.rb +10 -8
  121. data/lib/datadog/tracing/contrib/action_view/configuration/settings.rb +9 -6
  122. data/lib/datadog/tracing/contrib/action_view/ext.rb +12 -10
  123. data/lib/datadog/tracing/contrib/active_job/configuration/settings.rb +13 -7
  124. data/lib/datadog/tracing/contrib/active_job/ext.rb +25 -23
  125. data/lib/datadog/tracing/contrib/active_job/log_injection.rb +1 -1
  126. data/lib/datadog/tracing/contrib/active_job/patcher.rb +1 -1
  127. data/lib/datadog/tracing/contrib/active_model_serializers/configuration/settings.rb +9 -6
  128. data/lib/datadog/tracing/contrib/active_model_serializers/ext.rb +12 -10
  129. data/lib/datadog/tracing/contrib/active_record/configuration/resolver.rb +18 -11
  130. data/lib/datadog/tracing/contrib/active_record/configuration/settings.rb +9 -7
  131. data/lib/datadog/tracing/contrib/active_record/events/sql.rb +0 -8
  132. data/lib/datadog/tracing/contrib/active_record/ext.rb +17 -15
  133. data/lib/datadog/tracing/contrib/active_record/utils.rb +1 -1
  134. data/lib/datadog/tracing/contrib/active_support/cache/instrumentation.rb +0 -5
  135. data/lib/datadog/tracing/contrib/active_support/configuration/settings.rb +9 -7
  136. data/lib/datadog/tracing/contrib/active_support/ext.rb +18 -16
  137. data/lib/datadog/tracing/contrib/aws/configuration/settings.rb +14 -7
  138. data/lib/datadog/tracing/contrib/aws/ext.rb +37 -24
  139. data/lib/datadog/tracing/contrib/aws/instrumentation.rb +9 -5
  140. data/lib/datadog/tracing/contrib/concurrent_ruby/configuration/settings.rb +3 -2
  141. data/lib/datadog/tracing/contrib/concurrent_ruby/ext.rb +4 -2
  142. data/lib/datadog/tracing/contrib/dalli/configuration/settings.rb +14 -7
  143. data/lib/datadog/tracing/contrib/dalli/ext.rb +19 -11
  144. data/lib/datadog/tracing/contrib/dalli/instrumentation.rb +8 -6
  145. data/lib/datadog/tracing/contrib/delayed_job/configuration/settings.rb +13 -7
  146. data/lib/datadog/tracing/contrib/delayed_job/ext.rb +16 -14
  147. data/lib/datadog/tracing/contrib/elasticsearch/configuration/settings.rb +14 -7
  148. data/lib/datadog/tracing/contrib/elasticsearch/ext.rb +21 -15
  149. data/lib/datadog/tracing/contrib/elasticsearch/patcher.rb +99 -99
  150. data/lib/datadog/tracing/contrib/ethon/configuration/settings.rb +16 -9
  151. data/lib/datadog/tracing/contrib/ethon/easy_patch.rb +43 -3
  152. data/lib/datadog/tracing/contrib/ethon/ext.rb +19 -11
  153. data/lib/datadog/tracing/contrib/ethon/multi_patch.rb +0 -5
  154. data/lib/datadog/tracing/contrib/excon/configuration/settings.rb +19 -10
  155. data/lib/datadog/tracing/contrib/excon/ext.rb +16 -8
  156. data/lib/datadog/tracing/contrib/excon/middleware.rb +20 -5
  157. data/lib/datadog/tracing/contrib/ext.rb +23 -1
  158. data/lib/datadog/tracing/contrib/extensions.rb +32 -0
  159. data/lib/datadog/tracing/contrib/faraday/configuration/settings.rb +20 -10
  160. data/lib/datadog/tracing/contrib/faraday/ext.rb +16 -8
  161. data/lib/datadog/tracing/contrib/faraday/middleware.rb +16 -5
  162. data/lib/datadog/tracing/contrib/grape/configuration/settings.rb +8 -6
  163. data/lib/datadog/tracing/contrib/grape/ext.rb +16 -14
  164. data/lib/datadog/tracing/contrib/graphql/configuration/settings.rb +8 -6
  165. data/lib/datadog/tracing/contrib/graphql/ext.rb +7 -5
  166. data/lib/datadog/tracing/contrib/grpc/configuration/settings.rb +19 -9
  167. data/lib/datadog/tracing/contrib/grpc/datadog_interceptor/client.rb +29 -20
  168. data/lib/datadog/tracing/contrib/grpc/datadog_interceptor/server.rb +21 -20
  169. data/lib/datadog/tracing/contrib/grpc/ext.rb +16 -13
  170. data/lib/datadog/tracing/contrib/grpc/formatting.rb +127 -0
  171. data/lib/datadog/tracing/contrib/hanami/configuration/settings.rb +3 -2
  172. data/lib/datadog/tracing/contrib/hanami/ext.rb +10 -8
  173. data/lib/datadog/tracing/contrib/http/circuit_breaker.rb +4 -7
  174. data/lib/datadog/tracing/contrib/http/configuration/settings.rb +33 -11
  175. data/lib/datadog/tracing/contrib/http/ext.rb +16 -9
  176. data/lib/datadog/tracing/contrib/http/instrumentation.rb +22 -7
  177. data/lib/datadog/tracing/contrib/httpclient/configuration/settings.rb +33 -11
  178. data/lib/datadog/tracing/contrib/httpclient/ext.rb +17 -9
  179. data/lib/datadog/tracing/contrib/httpclient/instrumentation.rb +17 -5
  180. data/lib/datadog/tracing/contrib/httprb/configuration/settings.rb +33 -11
  181. data/lib/datadog/tracing/contrib/httprb/ext.rb +16 -9
  182. data/lib/datadog/tracing/contrib/httprb/instrumentation.rb +17 -5
  183. data/lib/datadog/tracing/contrib/kafka/configuration/settings.rb +9 -6
  184. data/lib/datadog/tracing/contrib/kafka/ext.rb +42 -39
  185. data/lib/datadog/tracing/contrib/lograge/configuration/settings.rb +3 -2
  186. data/lib/datadog/tracing/contrib/lograge/ext.rb +3 -1
  187. data/lib/datadog/tracing/contrib/lograge/instrumentation.rb +2 -17
  188. data/lib/datadog/tracing/contrib/mongodb/configuration/settings.rb +14 -7
  189. data/lib/datadog/tracing/contrib/mongodb/ext.rb +20 -16
  190. data/lib/datadog/tracing/contrib/mongodb/subscribers.rb +9 -5
  191. data/lib/datadog/tracing/contrib/mysql2/configuration/settings.rb +17 -14
  192. data/lib/datadog/tracing/contrib/mysql2/ext.rb +15 -10
  193. data/lib/datadog/tracing/contrib/mysql2/instrumentation.rb +9 -5
  194. data/lib/datadog/tracing/contrib/opensearch/configuration/settings.rb +52 -0
  195. data/lib/datadog/tracing/contrib/opensearch/ext.rb +37 -0
  196. data/lib/datadog/tracing/contrib/opensearch/integration.rb +44 -0
  197. data/lib/datadog/tracing/contrib/opensearch/patcher.rb +128 -0
  198. data/lib/datadog/tracing/contrib/opensearch/quantize.rb +81 -0
  199. data/lib/datadog/tracing/contrib/pg/configuration/settings.rb +17 -14
  200. data/lib/datadog/tracing/contrib/pg/ext.rb +22 -19
  201. data/lib/datadog/tracing/contrib/pg/instrumentation.rb +9 -5
  202. data/lib/datadog/tracing/contrib/presto/configuration/settings.rb +14 -7
  203. data/lib/datadog/tracing/contrib/presto/ext.rb +25 -20
  204. data/lib/datadog/tracing/contrib/presto/instrumentation.rb +9 -5
  205. data/lib/datadog/tracing/contrib/propagation/sql_comment/ext.rb +12 -10
  206. data/lib/datadog/tracing/contrib/qless/configuration/settings.rb +12 -8
  207. data/lib/datadog/tracing/contrib/qless/ext.rb +14 -12
  208. data/lib/datadog/tracing/contrib/que/configuration/settings.rb +21 -12
  209. data/lib/datadog/tracing/contrib/racecar/configuration/settings.rb +9 -7
  210. data/lib/datadog/tracing/contrib/racecar/event.rb +0 -5
  211. data/lib/datadog/tracing/contrib/racecar/ext.rb +20 -18
  212. data/lib/datadog/tracing/contrib/rack/configuration/settings.rb +16 -12
  213. data/lib/datadog/tracing/contrib/rack/ext.rb +18 -16
  214. data/lib/datadog/tracing/contrib/rack/header_collection.rb +3 -0
  215. data/lib/datadog/tracing/contrib/rack/header_tagging.rb +53 -0
  216. data/lib/datadog/tracing/contrib/rack/middlewares.rb +8 -49
  217. data/lib/datadog/tracing/contrib/rails/configuration/settings.rb +15 -11
  218. data/lib/datadog/tracing/contrib/rails/ext.rb +7 -5
  219. data/lib/datadog/tracing/contrib/rails/log_injection.rb +7 -10
  220. data/lib/datadog/tracing/contrib/rails/patcher.rb +10 -41
  221. data/lib/datadog/tracing/contrib/rails/railtie.rb +3 -3
  222. data/lib/datadog/tracing/contrib/rake/configuration/settings.rb +12 -9
  223. data/lib/datadog/tracing/contrib/rake/ext.rb +14 -12
  224. data/lib/datadog/tracing/contrib/redis/configuration/settings.rb +17 -9
  225. data/lib/datadog/tracing/contrib/redis/ext.rb +22 -15
  226. data/lib/datadog/tracing/contrib/redis/tags.rb +9 -5
  227. data/lib/datadog/tracing/contrib/resque/configuration/settings.rb +13 -7
  228. data/lib/datadog/tracing/contrib/resque/ext.rb +9 -7
  229. data/lib/datadog/tracing/contrib/rest_client/configuration/settings.rb +16 -9
  230. data/lib/datadog/tracing/contrib/rest_client/ext.rb +15 -8
  231. data/lib/datadog/tracing/contrib/rest_client/request_patch.rb +20 -5
  232. data/lib/datadog/tracing/contrib/roda/configuration/settings.rb +9 -6
  233. data/lib/datadog/tracing/contrib/semantic_logger/configuration/settings.rb +3 -2
  234. data/lib/datadog/tracing/contrib/semantic_logger/ext.rb +3 -1
  235. data/lib/datadog/tracing/contrib/semantic_logger/instrumentation.rb +4 -20
  236. data/lib/datadog/tracing/contrib/sequel/configuration/settings.rb +9 -6
  237. data/lib/datadog/tracing/contrib/sequel/ext.rb +10 -8
  238. data/lib/datadog/tracing/contrib/sequel/utils.rb +2 -7
  239. data/lib/datadog/tracing/contrib/shoryuken/configuration/settings.rb +14 -8
  240. data/lib/datadog/tracing/contrib/shoryuken/ext.rb +14 -12
  241. data/lib/datadog/tracing/contrib/sidekiq/configuration/settings.rb +18 -11
  242. data/lib/datadog/tracing/contrib/sidekiq/ext.rb +32 -30
  243. data/lib/datadog/tracing/contrib/sinatra/configuration/settings.rb +11 -9
  244. data/lib/datadog/tracing/contrib/sinatra/env.rb +0 -17
  245. data/lib/datadog/tracing/contrib/sinatra/ext.rb +21 -19
  246. data/lib/datadog/tracing/contrib/sinatra/tracer_middleware.rb +3 -14
  247. data/lib/datadog/tracing/contrib/sneakers/configuration/settings.rb +14 -8
  248. data/lib/datadog/tracing/contrib/sneakers/ext.rb +1 -0
  249. data/lib/datadog/tracing/contrib/sneakers/tracer.rb +1 -1
  250. data/lib/datadog/tracing/contrib/span_attribute_schema.rb +74 -10
  251. data/lib/datadog/tracing/contrib/stripe/configuration/settings.rb +9 -6
  252. data/lib/datadog/tracing/contrib/sucker_punch/configuration/settings.rb +9 -6
  253. data/lib/datadog/tracing/contrib/sucker_punch/ext.rb +15 -13
  254. data/lib/datadog/tracing/contrib/utils/database.rb +5 -3
  255. data/lib/datadog/tracing/contrib/utils/quantization/http.rb +9 -9
  256. data/lib/datadog/tracing/contrib.rb +1 -0
  257. data/lib/datadog/tracing/correlation.rb +29 -12
  258. data/lib/datadog/tracing/diagnostics/environment_logger.rb +159 -0
  259. data/lib/datadog/tracing/diagnostics/ext.rb +21 -19
  260. data/lib/datadog/tracing/distributed/b3_multi.rb +2 -2
  261. data/lib/datadog/tracing/distributed/b3_single.rb +1 -1
  262. data/lib/datadog/tracing/distributed/trace_context.rb +52 -17
  263. data/lib/datadog/tracing/metadata/ext.rb +9 -6
  264. data/lib/datadog/tracing/remote.rb +78 -0
  265. data/lib/datadog/tracing/sampling/rule_sampler.rb +29 -0
  266. data/lib/datadog/tracing/span_operation.rb +3 -15
  267. data/lib/datadog/tracing/trace_operation.rb +16 -3
  268. data/lib/datadog/tracing/trace_segment.rb +5 -2
  269. data/lib/datadog/tracing/tracer.rb +10 -1
  270. data/lib/datadog/tracing/workers/trace_writer.rb +1 -1
  271. data/lib/ddtrace/transport/ext.rb +15 -9
  272. data/lib/ddtrace/transport/trace_formatter.rb +9 -0
  273. data/lib/ddtrace/version.rb +9 -12
  274. metadata +37 -6
  275. data/lib/datadog/tracing/contrib/sinatra/headers.rb +0 -35
@@ -6,84 +6,99 @@ module Datadog
6
6
  module Kit
7
7
  # Tracking identity via traces
8
8
  module Identity
9
- # Attach user information to the trace
10
- #
11
- # @param trace [TraceOperation] Trace to attach data to. Defaults to
12
- # active trace.
13
- # @param span [SpanOperation] Span to attach data to. Defaults to
14
- # active span on trace. Note that this should be a service entry span.
15
- # When AppSec is enabled, the expected span and trace are automatically
16
- # used as defaults.
17
- # @param id [String] Mandatory. Username or client id extracted
18
- # from the access token or Authorization header in the inbound request
19
- # from outside the system.
20
- # @param email [String] Email of the authenticated user associated
21
- # to the trace.
22
- # @param name [String] User-friendly name. To be displayed in the
23
- # UI if set.
24
- # @param session_id [String] Session ID of the authenticated user.
25
- # @param role [String] Actual/assumed role the client is making
26
- # the request under extracted from token or application security
27
- # context.
28
- # @param scope [String] Scopes or granted authorities the client
29
- # currently possesses extracted from token or application security
30
- # context. The value would come from the scope associated with an OAuth
31
- # 2.0 Access Token or an attribute value in a SAML 2.0 Assertion.
32
- # @param others [Hash<Symbol, String>] Additional free-form
33
- # user information to attach to the trace.
34
- #
35
- # rubocop:disable Metrics/CyclomaticComplexity
36
- # rubocop:disable Metrics/PerceivedComplexity
37
- # rubocop:disable Metrics/AbcSize
38
- def self.set_user(
39
- trace = nil, span = nil, id:, email: nil, name: nil, session_id: nil, role: nil, scope: nil, **others
40
- )
41
- raise ArgumentError, 'missing required key: :id' if id.nil?
9
+ class << self
10
+ # Attach user information to the trace
11
+ #
12
+ # @param trace [TraceOperation] Trace to attach data to. Defaults to
13
+ # active trace.
14
+ # @param span [SpanOperation] Span to attach data to. Defaults to
15
+ # active span on trace. Note that this should be a service entry span.
16
+ # When AppSec is enabled, the expected span and trace are automatically
17
+ # used as defaults.
18
+ # @param id [String] Mandatory. Username or client id extracted
19
+ # from the access token or Authorization header in the inbound request
20
+ # from outside the system.
21
+ # @param email [String] Email of the authenticated user associated
22
+ # to the trace.
23
+ # @param name [String] User-friendly name. To be displayed in the
24
+ # UI if set.
25
+ # @param session_id [String] Session ID of the authenticated user.
26
+ # @param role [String] Actual/assumed role the client is making
27
+ # the request under extracted from token or application security
28
+ # context.
29
+ # @param scope [String] Scopes or granted authorities the client
30
+ # currently possesses extracted from token or application security
31
+ # context. The value would come from the scope associated with an OAuth
32
+ # 2.0 Access Token or an attribute value in a SAML 2.0 Assertion.
33
+ # @param others [Hash<Symbol, String>] Additional free-form
34
+ # user information to attach to the trace.
35
+ #
36
+ # rubocop:disable Metrics/CyclomaticComplexity
37
+ # rubocop:disable Metrics/PerceivedComplexity
38
+ def set_user(
39
+ trace = nil, span = nil, id:, email: nil, name: nil, session_id: nil, role: nil, scope: nil, **others
40
+ )
41
+ raise ArgumentError, 'missing required key: :id' if id.nil?
42
42
 
43
- # enforce types
43
+ # enforce types
44
44
 
45
- raise TypeError, ':id must be a String' unless id.is_a?(String)
46
- raise TypeError, ':email must be a String' unless email.nil? || email.is_a?(String)
47
- raise TypeError, ':name must be a String' unless name.nil? || name.is_a?(String)
48
- raise TypeError, ':session_id must be a String' unless session_id.nil? || session_id.is_a?(String)
49
- raise TypeError, ':role must be a String' unless role.nil? || role.is_a?(String)
50
- raise TypeError, ':scope must be a String' unless scope.nil? || scope.is_a?(String)
45
+ raise TypeError, ':id must be a String' unless id.is_a?(String)
46
+ raise TypeError, ':email must be a String' unless email.nil? || email.is_a?(String)
47
+ raise TypeError, ':name must be a String' unless name.nil? || name.is_a?(String)
48
+ raise TypeError, ':session_id must be a String' unless session_id.nil? || session_id.is_a?(String)
49
+ raise TypeError, ':role must be a String' unless role.nil? || role.is_a?(String)
50
+ raise TypeError, ':scope must be a String' unless scope.nil? || scope.is_a?(String)
51
51
 
52
- others.each do |k, v|
53
- raise TypeError, "#{k.inspect} must be a String" unless v.nil? || v.is_a?(String)
54
- end
52
+ others.each do |k, v|
53
+ raise TypeError, "#{k.inspect} must be a String" unless v.nil? || v.is_a?(String)
54
+ end
55
+
56
+ set_trace_and_span_context('set_user', trace, span) do |_active_trace, active_span|
57
+ # set tags once data is known consistent
58
+ active_span.set_tag('usr.id', id)
59
+ active_span.set_tag('usr.email', email) unless email.nil?
60
+ active_span.set_tag('usr.name', name) unless name.nil?
61
+ active_span.set_tag('usr.session_id', session_id) unless session_id.nil?
62
+ active_span.set_tag('usr.role', role) unless role.nil?
63
+ active_span.set_tag('usr.scope', scope) unless scope.nil?
55
64
 
56
- if (appsec_scope = Datadog::AppSec.active_scope)
57
- trace = appsec_scope.trace
58
- span = appsec_scope.service_entry_span
65
+ others.each do |k, v|
66
+ active_span.set_tag("usr.#{k}", v) unless v.nil?
67
+ end
68
+
69
+ if Datadog::AppSec.active_scope
70
+ user = ::Datadog::AppSec::Instrumentation::Gateway::User.new(id)
71
+ ::Datadog::AppSec::Instrumentation.gateway.push('identity.set_user', user)
72
+ end
73
+ end
59
74
  end
75
+ # rubocop:enable Metrics/PerceivedComplexity
76
+ # rubocop:enable Metrics/CyclomaticComplexity
60
77
 
61
- trace ||= Datadog::Tracing.active_trace
62
- span ||= trace.active_span || Datadog::Tracing.active_span
78
+ private
63
79
 
64
- raise ArgumentError, "span #{span.span_id} does not belong to trace #{trace.id}" if trace.id != span.trace_id
80
+ def set_trace_and_span_context(method, trace = nil, span = nil)
81
+ if (appsec_scope = Datadog::AppSec.active_scope)
82
+ trace = appsec_scope.trace
83
+ span = appsec_scope.service_entry_span
84
+ end
65
85
 
66
- # set tags once data is known consistent
86
+ trace ||= Datadog::Tracing.active_trace
87
+ span ||= trace && trace.active_span || Datadog::Tracing.active_span
67
88
 
68
- span.set_tag('usr.id', id)
69
- span.set_tag('usr.email', email) unless email.nil?
70
- span.set_tag('usr.name', name) unless name.nil?
71
- span.set_tag('usr.session_id', session_id) unless session_id.nil?
72
- span.set_tag('usr.role', role) unless role.nil?
73
- span.set_tag('usr.scope', scope) unless scope.nil?
89
+ unless trace && span
90
+ Datadog.logger.debug(
91
+ "Tracing not enabled. Method ##{method} is a no-op. Please enable tracing if you want ##{method}"\
92
+ ' to track this events'
93
+ )
94
+ return
95
+ end
74
96
 
75
- others.each do |k, v|
76
- span.set_tag("usr.#{k}", v) unless v.nil?
77
- end
97
+ raise ArgumentError, "span #{span.span_id} does not belong to trace #{trace.id}" if trace.id != span.trace_id
78
98
 
79
- if appsec_scope
80
- user = ::Datadog::AppSec::Instrumentation::Gateway::User.new(id)
81
- ::Datadog::AppSec::Instrumentation.gateway.push('identity.set_user', user)
99
+ yield(trace, span)
82
100
  end
83
101
  end
84
- # rubocop:enable Metrics/AbcSize
85
- # rubocop:enable Metrics/PerceivedComplexity
86
- # rubocop:enable Metrics/CyclomaticComplexity
87
102
  end
88
103
  end
89
104
  end
@@ -21,11 +21,13 @@ module Datadog
21
21
  gc_profiling_enabled:,
22
22
  allocation_counting_enabled:,
23
23
  no_signals_workaround_enabled:,
24
+ timeline_enabled:,
24
25
  thread_context_collector: ThreadContext.new(
25
26
  recorder: recorder,
26
27
  max_frames: max_frames,
27
28
  tracer: tracer,
28
29
  endpoint_collection_enabled: endpoint_collection_enabled,
30
+ timeline_enabled: timeline_enabled,
29
31
  ),
30
32
  idle_sampling_helper: IdleSamplingHelper.new,
31
33
  # **NOTE**: This should only be used for testing; disabling the dynamic sampling rate will increase the
@@ -76,6 +78,7 @@ module Datadog
76
78
  )
77
79
  end
78
80
  end
81
+ @worker_thread.name = self.class.name # Repeated from above to make sure thread gets named asap
79
82
  end
80
83
 
81
84
  true
@@ -42,6 +42,7 @@ module Datadog
42
42
  )
43
43
  end
44
44
  end
45
+ @worker_thread.name = self.class.name # Repeated from above to make sure thread gets named asap
45
46
  end
46
47
 
47
48
  true
@@ -14,9 +14,16 @@ module Datadog
14
14
  #
15
15
  # Methods prefixed with _native_ are implemented in `collectors_thread_context.c`
16
16
  class ThreadContext
17
- def initialize(recorder:, max_frames:, tracer:, endpoint_collection_enabled:)
17
+ def initialize(recorder:, max_frames:, tracer:, endpoint_collection_enabled:, timeline_enabled:)
18
18
  tracer_context_key = safely_extract_context_key_from(tracer)
19
- self.class._native_initialize(self, recorder, max_frames, tracer_context_key, endpoint_collection_enabled)
19
+ self.class._native_initialize(
20
+ self,
21
+ recorder,
22
+ max_frames,
23
+ tracer_context_key,
24
+ endpoint_collection_enabled,
25
+ timeline_enabled,
26
+ )
20
27
  end
21
28
 
22
29
  def inspect
@@ -7,7 +7,10 @@ module Datadog
7
7
  # Passing in a `nil` tracer is supported and will disable the following profiling features:
8
8
  # * Code Hotspots panel in the trace viewer, as well as scoping a profile down to a span
9
9
  # * Endpoint aggregation in the profiler UX, including normalization (resource per endpoint call)
10
- def self.build_profiler_component(settings:, agent_settings:, optional_tracer:)
10
+ def self.build_profiler_component(settings:, agent_settings:, optional_tracer:) # rubocop:disable Metrics/MethodLength
11
+ require_relative '../profiling/diagnostics/environment_logger'
12
+ Profiling::Diagnostics::EnvironmentLogger.collect_and_log!
13
+
11
14
  return unless settings.profiling.enabled
12
15
 
13
16
  # Workaround for weird dependency direction: the Core::Configuration::Components class currently has a
@@ -64,7 +67,13 @@ module Datadog
64
67
 
65
68
  # NOTE: Please update the Initialization section of ProfilingDevelopment.md with any changes to this method
66
69
 
70
+ no_signals_workaround_enabled = false
71
+ timeline_enabled = false
72
+
67
73
  if enable_new_profiler?(settings)
74
+ no_signals_workaround_enabled = no_signals_workaround_enabled?(settings)
75
+ timeline_enabled = settings.profiling.advanced.experimental_timeline_enabled
76
+
68
77
  recorder = Datadog::Profiling::StackRecorder.new(
69
78
  cpu_time_enabled: RUBY_PLATFORM.include?('linux'), # Only supported on Linux currently
70
79
  alloc_samples_enabled: false, # Always disabled for now -- work in progress
@@ -76,7 +85,8 @@ module Datadog
76
85
  endpoint_collection_enabled: settings.profiling.advanced.endpoint.collection.enabled,
77
86
  gc_profiling_enabled: enable_gc_profiling?(settings),
78
87
  allocation_counting_enabled: settings.profiling.advanced.allocation_counting_enabled,
79
- no_signals_workaround_enabled: no_signals_workaround_enabled?(settings),
88
+ no_signals_workaround_enabled: no_signals_workaround_enabled,
89
+ timeline_enabled: timeline_enabled,
80
90
  )
81
91
  else
82
92
  load_pprof_support
@@ -85,7 +95,12 @@ module Datadog
85
95
  collector = build_profiler_oldstack_collector(settings, recorder, optional_tracer)
86
96
  end
87
97
 
88
- exporter = build_profiler_exporter(settings, recorder)
98
+ internal_metadata = {
99
+ no_signals_workaround_enabled: no_signals_workaround_enabled,
100
+ timeline_enabled: timeline_enabled,
101
+ }.freeze
102
+
103
+ exporter = build_profiler_exporter(settings, recorder, internal_metadata: internal_metadata)
89
104
  transport = build_profiler_transport(settings, agent_settings)
90
105
  scheduler = Profiling::Scheduler.new(exporter: exporter, transport: transport)
91
106
 
@@ -96,11 +111,15 @@ module Datadog
96
111
  Profiling::OldRecorder.new([Profiling::Events::StackSample], settings.profiling.advanced.max_events)
97
112
  end
98
113
 
99
- private_class_method def self.build_profiler_exporter(settings, recorder)
114
+ private_class_method def self.build_profiler_exporter(settings, recorder, internal_metadata:)
100
115
  code_provenance_collector =
101
116
  (Profiling::Collectors::CodeProvenance.new if settings.profiling.advanced.code_provenance_enabled)
102
117
 
103
- Profiling::Exporter.new(pprof_recorder: recorder, code_provenance_collector: code_provenance_collector)
118
+ Profiling::Exporter.new(
119
+ pprof_recorder: recorder,
120
+ code_provenance_collector: code_provenance_collector,
121
+ internal_metadata: internal_metadata,
122
+ )
104
123
  end
105
124
 
106
125
  private_class_method def self.build_profiler_oldstack_collector(settings, old_recorder, tracer)
@@ -198,7 +217,7 @@ module Datadog
198
217
  if Gem.loaded_specs['mysql2'] && incompatible_libmysqlclient_version?(settings)
199
218
  Datadog.logger.warn(
200
219
  'Enabling the profiling "no signals" workaround because an incompatible version of the mysql2 gem is ' \
201
- 'installed. Profiling data will have lower quality.' \
220
+ 'installed. Profiling data will have lower quality. ' \
202
221
  'To fix this, upgrade the libmysqlclient in your OS image to version 8.0.0 or above.'
203
222
  )
204
223
  return true
@@ -214,6 +233,16 @@ module Datadog
214
233
  return true
215
234
  end
216
235
 
236
+ if defined?(::PhusionPassenger)
237
+ Datadog.logger.warn(
238
+ 'Enabling the profiling "no signals" workaround because the passenger web server is in use. ' \
239
+ 'This is needed because passenger is currently incompatible with the normal working mode ' \
240
+ 'of the profiler, as detailed in <https://github.com/DataDog/dd-trace-rb/issues/2976>. ' \
241
+ 'Profiling data will have lower quality.'
242
+ )
243
+ return true
244
+ end
245
+
217
246
  false
218
247
  end
219
248
 
@@ -236,9 +265,22 @@ module Datadog
236
265
  begin
237
266
  require 'mysql2'
238
267
 
239
- return true unless defined?(Mysql2::Client) && Mysql2::Client.respond_to?(:info)
240
-
241
- libmysqlclient_version = Gem::Version.new(Mysql2::Client.info[:version])
268
+ # The mysql2-aurora gem likes to monkey patch itself in replacement of Mysql2::Client, and uses
269
+ # `method_missing` to delegate to the original BUT unfortunately does not implement `respond_to_missing?` and
270
+ # thus our `respond_to?(:info)` below was failing.
271
+ #
272
+ # But on the bright side, the gem does stash a reference to the original Mysql2::Client class in a constant,
273
+ # so if that constant exists, we use that for our probing.
274
+ mysql2_client_class =
275
+ if defined?(Mysql2::Aurora::ORIGINAL_CLIENT_CLASS)
276
+ Mysql2::Aurora::ORIGINAL_CLIENT_CLASS
277
+ elsif defined?(Mysql2::Client)
278
+ Mysql2::Client
279
+ end
280
+
281
+ return true unless mysql2_client_class && mysql2_client_class.respond_to?(:info)
282
+
283
+ libmysqlclient_version = Gem::Version.new(mysql2_client_class.info[:version])
242
284
 
243
285
  compatible = libmysqlclient_version >= Gem::Version.new('8.0.0')
244
286
 
@@ -0,0 +1,39 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'date'
4
+ require 'json'
5
+ require 'rbconfig'
6
+ require_relative '../../core/diagnostics/environment_logger'
7
+
8
+ module Datadog
9
+ module Profiling
10
+ module Diagnostics
11
+ # Collects and logs Profiling diagnostic information
12
+ module EnvironmentLogger
13
+ extend Core::Diagnostics::EnvironmentLogging
14
+
15
+ def self.collect_and_log!
16
+ log_once! do
17
+ data = EnvironmentCollector.collect_config!
18
+ log_configuration!('PROFILING', data.to_json)
19
+ end
20
+ rescue => e
21
+ logger.warn("Failed to collect profiling environment information: #{e} Location: #{Array(e.backtrace).first}")
22
+ end
23
+ end
24
+
25
+ # Collects environment information for Profiling diagnostic logging
26
+ module EnvironmentCollector
27
+ def self.collect_config!(*args)
28
+ {
29
+ profiling_enabled: profiling_enabled
30
+ }
31
+ end
32
+
33
+ def self.profiling_enabled
34
+ !!Datadog.configuration.profiling.enabled
35
+ end
36
+ end
37
+ end
38
+ end
39
+ end
@@ -22,13 +22,15 @@ module Datadog
22
22
  :minimum_duration_seconds,
23
23
  :time_provider,
24
24
  :last_flush_finish_at,
25
- :created_at
25
+ :created_at,
26
+ :internal_metadata
26
27
 
27
28
  public
28
29
 
29
30
  def initialize(
30
31
  pprof_recorder:,
31
32
  code_provenance_collector:,
33
+ internal_metadata:,
32
34
  minimum_duration_seconds: PROFILE_DURATION_THRESHOLD_SECONDS,
33
35
  time_provider: Time
34
36
  )
@@ -38,6 +40,7 @@ module Datadog
38
40
  @time_provider = time_provider
39
41
  @last_flush_finish_at = nil
40
42
  @created_at = time_provider.now.utc
43
+ @internal_metadata = internal_metadata
41
44
  end
42
45
 
43
46
  def flush
@@ -61,6 +64,7 @@ module Datadog
61
64
  code_provenance_file_name: Datadog::Profiling::Ext::Transport::HTTP::CODE_PROVENANCE_FILENAME,
62
65
  code_provenance_data: uncompressed_code_provenance,
63
66
  tags_as_array: Datadog::Profiling::TagBuilder.call(settings: Datadog.configuration).to_a,
67
+ internal_metadata: internal_metadata,
64
68
  )
65
69
  end
66
70
 
@@ -1,5 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
+ require 'json'
4
+
3
5
  module Datadog
4
6
  module Profiling
5
7
  # Represents a collection of events of a specific type being flushed.
@@ -14,7 +16,8 @@ module Datadog
14
16
  :pprof_data, # gzipped pprof bytes
15
17
  :code_provenance_file_name,
16
18
  :code_provenance_data, # gzipped json bytes
17
- :tags_as_array
19
+ :tags_as_array,
20
+ :internal_metadata_json
18
21
 
19
22
  def initialize(
20
23
  start:,
@@ -23,7 +26,8 @@ module Datadog
23
26
  pprof_data:,
24
27
  code_provenance_file_name:,
25
28
  code_provenance_data:,
26
- tags_as_array:
29
+ tags_as_array:,
30
+ internal_metadata:
27
31
  )
28
32
  @start = start
29
33
  @finish = finish
@@ -32,6 +36,7 @@ module Datadog
32
36
  @code_provenance_file_name = code_provenance_file_name
33
37
  @code_provenance_data = code_provenance_data
34
38
  @tags_as_array = tags_as_array
39
+ @internal_metadata_json = JSON.fast_generate(internal_metadata.map { |k, v| [k, v.to_s] }.to_h)
35
40
  end
36
41
  end
37
42
  end
@@ -40,6 +40,7 @@ module Datadog
40
40
  code_provenance_data: flush.code_provenance_data,
41
41
 
42
42
  tags_as_array: flush.tags_as_array,
43
+ internal_metadata_json: flush.internal_metadata_json,
43
44
  )
44
45
 
45
46
  if status == :ok
@@ -47,11 +48,14 @@ module Datadog
47
48
  Datadog.logger.debug('Successfully reported profiling data')
48
49
  true
49
50
  else
50
- Datadog.logger.error("Failed to report profiling data: server returned unexpected HTTP #{result} status code")
51
+ Datadog.logger.error(
52
+ "Failed to report profiling data (#{config_without_api_key}): " \
53
+ "server returned unexpected HTTP #{result} status code"
54
+ )
51
55
  false
52
56
  end
53
57
  else
54
- Datadog.logger.error("Failed to report profiling data: #{result}")
58
+ Datadog.logger.error("Failed to report profiling data (#{config_without_api_key}): #{result}")
55
59
  false
56
60
  end
57
61
  end
@@ -109,7 +113,8 @@ module Datadog
109
113
  pprof_data:,
110
114
  code_provenance_file_name:,
111
115
  code_provenance_data:,
112
- tags_as_array:
116
+ tags_as_array:,
117
+ internal_metadata_json:
113
118
  )
114
119
  self.class._native_do_export(
115
120
  exporter_configuration,
@@ -123,8 +128,13 @@ module Datadog
123
128
  code_provenance_file_name,
124
129
  code_provenance_data,
125
130
  tags_as_array,
131
+ internal_metadata_json,
126
132
  )
127
133
  end
134
+
135
+ def config_without_api_key
136
+ [@exporter_configuration[0..1]].to_h
137
+ end
128
138
  end
129
139
  end
130
140
  end
@@ -9,7 +9,13 @@
9
9
  # All code on this file is on-purpose at the top-level; this makes it so this file is executed only once,
10
10
  # the first time it gets required, to avoid any issues with the native extension being initialized more than once.
11
11
 
12
- require "ddtrace_profiling_loader.#{RUBY_VERSION}_#{RUBY_PLATFORM}"
12
+ begin
13
+ require "ddtrace_profiling_loader.#{RUBY_VERSION}_#{RUBY_PLATFORM}"
14
+ rescue LoadError => e
15
+ raise LoadError,
16
+ 'Failed to load the profiling loader extension. To fix this, please remove and then reinstall ddtrace ' \
17
+ "(Details: #{e.message})"
18
+ end
13
19
 
14
20
  extension_name = "ddtrace_profiling_native_extension.#{RUBY_VERSION}_#{RUBY_PLATFORM}"
15
21
  full_file_path = "#{__dir__}/../../#{extension_name}.#{RbConfig::CONFIG['DLEXT']}"
@@ -66,6 +66,11 @@ module Datadog
66
66
  nil
67
67
  end
68
68
 
69
+ def self.enabled?
70
+ profiler = Datadog.send(:components).profiler
71
+ !!(profiler.scheduler.running? if profiler)
72
+ end
73
+
69
74
  private_class_method def self.replace_noop_allocation_count
70
75
  def self.allocation_count # rubocop:disable Lint/DuplicateMethods, Lint/NestedMethodDefinition (On purpose!)
71
76
  Datadog::Profiling::Collectors::CpuAndWallTimeWorker._native_allocation_count
@@ -178,6 +183,10 @@ module Datadog
178
183
  end
179
184
  end
180
185
 
186
+ # All requires for the profiler should be directly added here; and everything should be loaded eagerly.
187
+ # (Currently there's a few exceptions for the old profiler, but we should avoid other exceptions.)
188
+ #
189
+ # All of the profiler should be loaded and ready to go when this method returns `true`.
181
190
  private_class_method def self.load_profiling
182
191
  return false unless supported?
183
192
 
@@ -189,6 +198,7 @@ module Datadog
189
198
  require_relative 'profiling/collectors/old_stack'
190
199
  require_relative 'profiling/collectors/stack'
191
200
  require_relative 'profiling/collectors/thread_context'
201
+ require_relative 'profiling/diagnostics/environment_logger'
192
202
  require_relative 'profiling/stack_recorder'
193
203
  require_relative 'profiling/old_recorder'
194
204
  require_relative 'profiling/exporter'
@@ -197,7 +207,8 @@ module Datadog
197
207
  require_relative 'profiling/profiler'
198
208
  require_relative 'profiling/native_extension'
199
209
  require_relative 'profiling/trace_identifiers/helper'
200
- # This file is loaded in Profiling::Component#load_pprof_support; see notes there for why
210
+ # This file is no longer eagerly loaded as a workaround for an issue. It only gets loaded dynamically if the old
211
+ # profiler is in use. See Profiling::Component#load_pprof_support for more details.
201
212
  # require_relative 'profiling/pprof/pprof_pb'
202
213
  require_relative 'profiling/tag_builder'
203
214
  require_relative 'profiling/http_transport'