datadog 2.20.0 → 2.26.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (310) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +212 -1
  3. data/README.md +0 -1
  4. data/ext/LIBDATADOG_DEVELOPMENT.md +3 -0
  5. data/ext/datadog_profiling_native_extension/collectors_cpu_and_wall_time_worker.c +93 -23
  6. data/ext/datadog_profiling_native_extension/collectors_discrete_dynamic_sampler.c +1 -1
  7. data/ext/datadog_profiling_native_extension/collectors_stack.c +21 -5
  8. data/ext/datadog_profiling_native_extension/crashtracking_runtime_stacks.c +239 -0
  9. data/ext/datadog_profiling_native_extension/datadog_ruby_common.h +1 -1
  10. data/ext/datadog_profiling_native_extension/extconf.rb +9 -4
  11. data/ext/datadog_profiling_native_extension/heap_recorder.c +1 -1
  12. data/ext/datadog_profiling_native_extension/http_transport.c +1 -0
  13. data/ext/datadog_profiling_native_extension/private_vm_api_access.c +12 -0
  14. data/ext/datadog_profiling_native_extension/private_vm_api_access.h +4 -0
  15. data/ext/datadog_profiling_native_extension/profiling.c +2 -0
  16. data/ext/libdatadog_api/datadog_ruby_common.h +1 -1
  17. data/ext/libdatadog_api/ddsketch.c +106 -0
  18. data/ext/libdatadog_api/feature_flags.c +554 -0
  19. data/ext/libdatadog_api/feature_flags.h +5 -0
  20. data/ext/libdatadog_api/init.c +5 -0
  21. data/ext/libdatadog_api/library_config.c +34 -25
  22. data/ext/libdatadog_api/process_discovery.c +24 -18
  23. data/ext/libdatadog_extconf_helpers.rb +1 -1
  24. data/lib/datadog/ai_guard/api_client.rb +82 -0
  25. data/lib/datadog/ai_guard/component.rb +42 -0
  26. data/lib/datadog/ai_guard/configuration/ext.rb +17 -0
  27. data/lib/datadog/ai_guard/configuration/settings.rb +98 -0
  28. data/lib/datadog/ai_guard/configuration.rb +11 -0
  29. data/lib/datadog/ai_guard/evaluation/message.rb +25 -0
  30. data/lib/datadog/ai_guard/evaluation/no_op_result.rb +34 -0
  31. data/lib/datadog/ai_guard/evaluation/request.rb +81 -0
  32. data/lib/datadog/ai_guard/evaluation/result.rb +43 -0
  33. data/lib/datadog/ai_guard/evaluation/tool_call.rb +18 -0
  34. data/lib/datadog/ai_guard/evaluation.rb +72 -0
  35. data/lib/datadog/ai_guard/ext.rb +16 -0
  36. data/lib/datadog/ai_guard.rb +153 -0
  37. data/lib/datadog/appsec/api_security/endpoint_collection/grape_route_serializer.rb +26 -0
  38. data/lib/datadog/appsec/api_security/endpoint_collection/rails_collector.rb +59 -0
  39. data/lib/datadog/appsec/api_security/endpoint_collection/rails_route_serializer.rb +29 -0
  40. data/lib/datadog/appsec/api_security/endpoint_collection/sinatra_route_serializer.rb +26 -0
  41. data/lib/datadog/appsec/api_security/endpoint_collection.rb +10 -0
  42. data/lib/datadog/appsec/api_security/route_extractor.rb +26 -5
  43. data/lib/datadog/appsec/api_security/sampler.rb +7 -4
  44. data/lib/datadog/appsec/assets/blocked.html +8 -0
  45. data/lib/datadog/appsec/assets/blocked.json +1 -1
  46. data/lib/datadog/appsec/assets/blocked.text +3 -1
  47. data/lib/datadog/appsec/assets/waf_rules/README.md +30 -36
  48. data/lib/datadog/appsec/assets/waf_rules/recommended.json +359 -4
  49. data/lib/datadog/appsec/assets/waf_rules/strict.json +43 -2
  50. data/lib/datadog/appsec/assets.rb +1 -1
  51. data/lib/datadog/appsec/autoload.rb +1 -1
  52. data/lib/datadog/appsec/compressed_json.rb +1 -1
  53. data/lib/datadog/appsec/configuration/settings.rb +9 -0
  54. data/lib/datadog/appsec/context.rb +2 -1
  55. data/lib/datadog/appsec/contrib/active_record/instrumentation.rb +3 -1
  56. data/lib/datadog/appsec/contrib/excon/ssrf_detection_middleware.rb +3 -2
  57. data/lib/datadog/appsec/contrib/faraday/ssrf_detection_middleware.rb +3 -1
  58. data/lib/datadog/appsec/contrib/graphql/gateway/watcher.rb +3 -1
  59. data/lib/datadog/appsec/contrib/rack/gateway/watcher.rb +9 -4
  60. data/lib/datadog/appsec/contrib/rack/request_middleware.rb +5 -1
  61. data/lib/datadog/appsec/contrib/rails/gateway/watcher.rb +7 -2
  62. data/lib/datadog/appsec/contrib/rails/patcher.rb +30 -0
  63. data/lib/datadog/appsec/contrib/rest_client/request_ssrf_detection_patch.rb +3 -1
  64. data/lib/datadog/appsec/contrib/sinatra/gateway/watcher.rb +10 -4
  65. data/lib/datadog/appsec/event.rb +12 -14
  66. data/lib/datadog/appsec/metrics/collector.rb +19 -3
  67. data/lib/datadog/appsec/metrics/telemetry_exporter.rb +2 -1
  68. data/lib/datadog/appsec/monitor/gateway/watcher.rb +4 -4
  69. data/lib/datadog/appsec/remote.rb +34 -25
  70. data/lib/datadog/appsec/response.rb +18 -4
  71. data/lib/datadog/appsec/security_engine/engine.rb +3 -3
  72. data/lib/datadog/appsec/security_engine/result.rb +29 -9
  73. data/lib/datadog/appsec/security_engine/runner.rb +19 -9
  74. data/lib/datadog/appsec/security_event.rb +5 -7
  75. data/lib/datadog/core/configuration/agent_settings_resolver.rb +4 -4
  76. data/lib/datadog/core/configuration/components.rb +59 -11
  77. data/lib/datadog/core/configuration/config_helper.rb +100 -0
  78. data/lib/datadog/core/configuration/deprecations.rb +36 -0
  79. data/lib/datadog/core/configuration/ext.rb +0 -1
  80. data/lib/datadog/core/configuration/option.rb +38 -43
  81. data/lib/datadog/core/configuration/option_definition.rb +4 -11
  82. data/lib/datadog/core/configuration/options.rb +9 -10
  83. data/lib/datadog/core/configuration/settings.rb +38 -9
  84. data/lib/datadog/core/configuration/stable_config.rb +10 -0
  85. data/lib/datadog/core/configuration/supported_configurations.rb +373 -0
  86. data/lib/datadog/core/configuration.rb +2 -2
  87. data/lib/datadog/core/ddsketch.rb +19 -0
  88. data/lib/datadog/core/deprecations.rb +2 -2
  89. data/lib/datadog/core/environment/cgroup.rb +52 -25
  90. data/lib/datadog/core/environment/container.rb +140 -46
  91. data/lib/datadog/core/environment/ext.rb +7 -2
  92. data/lib/datadog/core/environment/git.rb +2 -2
  93. data/lib/datadog/core/environment/process.rb +87 -0
  94. data/lib/datadog/core/environment/variable_helpers.rb +3 -3
  95. data/lib/datadog/core/environment/yjit.rb +2 -1
  96. data/lib/datadog/core/error.rb +6 -6
  97. data/lib/datadog/core/feature_flags.rb +61 -0
  98. data/lib/datadog/core/metrics/client.rb +2 -2
  99. data/lib/datadog/core/pin.rb +8 -8
  100. data/lib/datadog/core/process_discovery/tracer_memfd.rb +2 -4
  101. data/lib/datadog/core/process_discovery.rb +48 -23
  102. data/lib/datadog/core/rate_limiter.rb +9 -1
  103. data/lib/datadog/core/remote/client/capabilities.rb +7 -0
  104. data/lib/datadog/core/remote/client.rb +14 -6
  105. data/lib/datadog/core/remote/component.rb +10 -10
  106. data/lib/datadog/core/remote/configuration/content.rb +15 -2
  107. data/lib/datadog/core/remote/configuration/digest.rb +14 -7
  108. data/lib/datadog/core/remote/configuration/repository.rb +1 -1
  109. data/lib/datadog/core/remote/configuration/target.rb +13 -6
  110. data/lib/datadog/core/remote/transport/config.rb +4 -25
  111. data/lib/datadog/core/remote/transport/http/config.rb +10 -50
  112. data/lib/datadog/core/remote/transport/http/negotiation.rb +14 -44
  113. data/lib/datadog/core/remote/transport/http.rb +15 -24
  114. data/lib/datadog/core/remote/transport/negotiation.rb +8 -33
  115. data/lib/datadog/core/remote/worker.rb +25 -37
  116. data/lib/datadog/core/runtime/ext.rb +0 -1
  117. data/lib/datadog/core/runtime/metrics.rb +11 -1
  118. data/lib/datadog/core/semaphore.rb +1 -4
  119. data/lib/datadog/core/tag_builder.rb +0 -4
  120. data/lib/datadog/core/tag_normalizer.rb +84 -0
  121. data/lib/datadog/core/telemetry/component.rb +69 -15
  122. data/lib/datadog/core/telemetry/emitter.rb +6 -6
  123. data/lib/datadog/core/telemetry/event/app_endpoints_loaded.rb +30 -0
  124. data/lib/datadog/core/telemetry/event/app_started.rb +89 -51
  125. data/lib/datadog/core/telemetry/event/synth_app_client_configuration_change.rb +27 -4
  126. data/lib/datadog/core/telemetry/event.rb +1 -0
  127. data/lib/datadog/core/telemetry/logger.rb +2 -2
  128. data/lib/datadog/core/telemetry/logging.rb +2 -8
  129. data/lib/datadog/core/telemetry/metrics_manager.rb +9 -0
  130. data/lib/datadog/core/telemetry/request.rb +17 -3
  131. data/lib/datadog/core/telemetry/transport/http/telemetry.rb +3 -34
  132. data/lib/datadog/core/telemetry/transport/http.rb +21 -16
  133. data/lib/datadog/core/telemetry/transport/telemetry.rb +3 -11
  134. data/lib/datadog/core/telemetry/worker.rb +88 -32
  135. data/lib/datadog/core/transport/ext.rb +2 -0
  136. data/lib/datadog/core/transport/http/api/endpoint.rb +9 -4
  137. data/lib/datadog/core/transport/http/api/instance.rb +4 -21
  138. data/lib/datadog/core/transport/http/builder.rb +9 -5
  139. data/lib/datadog/core/transport/http/client.rb +80 -0
  140. data/lib/datadog/core/transport/http.rb +22 -19
  141. data/lib/datadog/core/transport/response.rb +15 -1
  142. data/lib/datadog/core/transport/transport.rb +90 -0
  143. data/lib/datadog/core/utils/array.rb +29 -0
  144. data/lib/datadog/{appsec/api_security → core/utils}/lru_cache.rb +10 -21
  145. data/lib/datadog/core/utils/network.rb +22 -1
  146. data/lib/datadog/core/utils/only_once_successful.rb +8 -2
  147. data/lib/datadog/core/utils/safe_dup.rb +2 -2
  148. data/lib/datadog/core/utils/sequence.rb +2 -0
  149. data/lib/datadog/core/utils/time.rb +1 -1
  150. data/lib/datadog/core/utils.rb +2 -0
  151. data/lib/datadog/core/workers/async.rb +10 -1
  152. data/lib/datadog/core/workers/interval_loop.rb +44 -3
  153. data/lib/datadog/core/workers/polling.rb +2 -0
  154. data/lib/datadog/core/workers/queue.rb +100 -1
  155. data/lib/datadog/core.rb +2 -0
  156. data/lib/datadog/data_streams/configuration/settings.rb +49 -0
  157. data/lib/datadog/data_streams/configuration.rb +11 -0
  158. data/lib/datadog/data_streams/ext.rb +11 -0
  159. data/lib/datadog/data_streams/extensions.rb +16 -0
  160. data/lib/datadog/data_streams/pathway_context.rb +169 -0
  161. data/lib/datadog/data_streams/processor.rb +509 -0
  162. data/lib/datadog/data_streams/transport/http/stats.rb +52 -0
  163. data/lib/datadog/data_streams/transport/http.rb +40 -0
  164. data/lib/datadog/data_streams/transport/stats.rb +46 -0
  165. data/lib/datadog/data_streams.rb +100 -0
  166. data/lib/datadog/di/boot.rb +7 -3
  167. data/lib/datadog/di/component.rb +14 -16
  168. data/lib/datadog/di/context.rb +70 -0
  169. data/lib/datadog/di/contrib/active_record.rb +30 -5
  170. data/lib/datadog/di/el/compiler.rb +168 -0
  171. data/lib/datadog/di/el/evaluator.rb +159 -0
  172. data/lib/datadog/di/el/expression.rb +42 -0
  173. data/lib/datadog/di/el.rb +5 -0
  174. data/lib/datadog/di/error.rb +34 -0
  175. data/lib/datadog/di/instrumenter.rb +189 -55
  176. data/lib/datadog/di/logger.rb +2 -2
  177. data/lib/datadog/di/probe.rb +55 -15
  178. data/lib/datadog/di/probe_builder.rb +41 -2
  179. data/lib/datadog/di/probe_file_loader/railtie.rb +1 -1
  180. data/lib/datadog/di/probe_file_loader.rb +1 -1
  181. data/lib/datadog/di/probe_manager.rb +50 -35
  182. data/lib/datadog/di/probe_notification_builder.rb +121 -70
  183. data/lib/datadog/di/probe_notifier_worker.rb +5 -5
  184. data/lib/datadog/di/proc_responder.rb +32 -0
  185. data/lib/datadog/di/remote.rb +89 -84
  186. data/lib/datadog/di/serializer.rb +151 -7
  187. data/lib/datadog/di/transport/diagnostics.rb +8 -36
  188. data/lib/datadog/di/transport/http/diagnostics.rb +1 -33
  189. data/lib/datadog/di/transport/http/input.rb +1 -33
  190. data/lib/datadog/di/transport/http.rb +32 -17
  191. data/lib/datadog/di/transport/input.rb +67 -34
  192. data/lib/datadog/di.rb +61 -5
  193. data/lib/datadog/error_tracking/filters.rb +2 -2
  194. data/lib/datadog/kit/appsec/events/v2.rb +2 -3
  195. data/lib/datadog/open_feature/component.rb +60 -0
  196. data/lib/datadog/open_feature/configuration.rb +27 -0
  197. data/lib/datadog/open_feature/evaluation_engine.rb +70 -0
  198. data/lib/datadog/open_feature/exposures/batch_builder.rb +32 -0
  199. data/lib/datadog/open_feature/exposures/buffer.rb +43 -0
  200. data/lib/datadog/open_feature/exposures/deduplicator.rb +30 -0
  201. data/lib/datadog/open_feature/exposures/event.rb +60 -0
  202. data/lib/datadog/open_feature/exposures/reporter.rb +40 -0
  203. data/lib/datadog/open_feature/exposures/worker.rb +116 -0
  204. data/lib/datadog/open_feature/ext.rb +14 -0
  205. data/lib/datadog/open_feature/native_evaluator.rb +38 -0
  206. data/lib/datadog/open_feature/noop_evaluator.rb +26 -0
  207. data/lib/datadog/open_feature/provider.rb +141 -0
  208. data/lib/datadog/open_feature/remote.rb +67 -0
  209. data/lib/datadog/open_feature/resolution_details.rb +35 -0
  210. data/lib/datadog/open_feature/transport.rb +70 -0
  211. data/lib/datadog/open_feature.rb +19 -0
  212. data/lib/datadog/opentelemetry/api/baggage.rb +1 -1
  213. data/lib/datadog/opentelemetry/configuration/settings.rb +159 -0
  214. data/lib/datadog/opentelemetry/metrics.rb +117 -0
  215. data/lib/datadog/opentelemetry/sdk/configurator.rb +26 -2
  216. data/lib/datadog/opentelemetry/sdk/metrics_exporter.rb +35 -0
  217. data/lib/datadog/opentelemetry.rb +3 -0
  218. data/lib/datadog/profiling/collectors/code_provenance.rb +41 -7
  219. data/lib/datadog/profiling/collectors/cpu_and_wall_time_worker.rb +3 -2
  220. data/lib/datadog/profiling/collectors/idle_sampling_helper.rb +1 -1
  221. data/lib/datadog/profiling/collectors/info.rb +6 -5
  222. data/lib/datadog/profiling/component.rb +12 -11
  223. data/lib/datadog/profiling/ext/dir_monkey_patches.rb +18 -0
  224. data/lib/datadog/profiling/ext.rb +2 -1
  225. data/lib/datadog/profiling/http_transport.rb +5 -2
  226. data/lib/datadog/profiling/profiler.rb +4 -0
  227. data/lib/datadog/profiling/tag_builder.rb +36 -3
  228. data/lib/datadog/profiling/tasks/exec.rb +2 -2
  229. data/lib/datadog/profiling.rb +1 -2
  230. data/lib/datadog/single_step_instrument.rb +1 -1
  231. data/lib/datadog/tracing/component.rb +6 -17
  232. data/lib/datadog/tracing/configuration/dynamic.rb +2 -2
  233. data/lib/datadog/tracing/configuration/ext.rb +9 -3
  234. data/lib/datadog/tracing/configuration/settings.rb +89 -10
  235. data/lib/datadog/tracing/contrib/action_pack/action_controller/instrumentation.rb +4 -4
  236. data/lib/datadog/tracing/contrib/action_pack/utils.rb +1 -2
  237. data/lib/datadog/tracing/contrib/active_job/log_injection.rb +21 -7
  238. data/lib/datadog/tracing/contrib/active_job/patcher.rb +5 -1
  239. data/lib/datadog/tracing/contrib/aws/instrumentation.rb +4 -2
  240. data/lib/datadog/tracing/contrib/component.rb +2 -2
  241. data/lib/datadog/tracing/contrib/ethon/easy_patch.rb +4 -1
  242. data/lib/datadog/tracing/contrib/excon/configuration/settings.rb +11 -3
  243. data/lib/datadog/tracing/contrib/extensions.rb +10 -2
  244. data/lib/datadog/tracing/contrib/faraday/configuration/settings.rb +11 -7
  245. data/lib/datadog/tracing/contrib/grape/configuration/settings.rb +7 -3
  246. data/lib/datadog/tracing/contrib/graphql/configuration/settings.rb +7 -0
  247. data/lib/datadog/tracing/contrib/graphql/ext.rb +1 -0
  248. data/lib/datadog/tracing/contrib/graphql/unified_trace.rb +84 -43
  249. data/lib/datadog/tracing/contrib/http/configuration/settings.rb +11 -3
  250. data/lib/datadog/tracing/contrib/httpclient/configuration/settings.rb +11 -3
  251. data/lib/datadog/tracing/contrib/httprb/configuration/settings.rb +11 -3
  252. data/lib/datadog/tracing/contrib/kafka/instrumentation/consumer.rb +66 -0
  253. data/lib/datadog/tracing/contrib/kafka/instrumentation/producer.rb +66 -0
  254. data/lib/datadog/tracing/contrib/kafka/patcher.rb +14 -0
  255. data/lib/datadog/tracing/contrib/karafka/framework.rb +30 -0
  256. data/lib/datadog/tracing/contrib/karafka/monitor.rb +11 -0
  257. data/lib/datadog/tracing/contrib/karafka/patcher.rb +35 -4
  258. data/lib/datadog/tracing/contrib/rack/middlewares.rb +59 -27
  259. data/lib/datadog/tracing/contrib/rack/request_queue.rb +1 -0
  260. data/lib/datadog/tracing/contrib/rack/route_inference.rb +53 -0
  261. data/lib/datadog/tracing/contrib/rack/trace_proxy_middleware.rb +7 -1
  262. data/lib/datadog/tracing/contrib/rails/ext.rb +2 -1
  263. data/lib/datadog/tracing/contrib/rails/integration.rb +1 -1
  264. data/lib/datadog/tracing/contrib/rails/middlewares.rb +2 -2
  265. data/lib/datadog/tracing/contrib/rest_client/request_patch.rb +4 -1
  266. data/lib/datadog/tracing/contrib/roda/instrumentation.rb +3 -1
  267. data/lib/datadog/tracing/contrib/sinatra/tracer_middleware.rb +3 -1
  268. data/lib/datadog/tracing/contrib/span_attribute_schema.rb +1 -1
  269. data/lib/datadog/tracing/contrib/status_range_matcher.rb +9 -1
  270. data/lib/datadog/tracing/contrib/utils/quantization/hash.rb +3 -1
  271. data/lib/datadog/tracing/contrib/waterdrop/configuration/settings.rb +27 -0
  272. data/lib/datadog/tracing/contrib/waterdrop/distributed/propagation.rb +48 -0
  273. data/lib/datadog/tracing/contrib/waterdrop/ext.rb +17 -0
  274. data/lib/datadog/tracing/contrib/waterdrop/integration.rb +43 -0
  275. data/lib/datadog/tracing/contrib/waterdrop/middleware.rb +46 -0
  276. data/lib/datadog/tracing/contrib/waterdrop/patcher.rb +49 -0
  277. data/lib/datadog/tracing/contrib/waterdrop/producer.rb +50 -0
  278. data/lib/datadog/tracing/contrib/waterdrop.rb +41 -0
  279. data/lib/datadog/tracing/contrib.rb +1 -0
  280. data/lib/datadog/tracing/diagnostics/environment_logger.rb +1 -1
  281. data/lib/datadog/tracing/distributed/baggage.rb +3 -2
  282. data/lib/datadog/tracing/metadata/ext.rb +9 -1
  283. data/lib/datadog/tracing/remote.rb +1 -9
  284. data/lib/datadog/tracing/sampling/priority_sampler.rb +3 -1
  285. data/lib/datadog/tracing/span.rb +1 -1
  286. data/lib/datadog/tracing/span_event.rb +2 -2
  287. data/lib/datadog/tracing/span_operation.rb +20 -9
  288. data/lib/datadog/tracing/trace_operation.rb +44 -6
  289. data/lib/datadog/tracing/tracer.rb +42 -16
  290. data/lib/datadog/tracing/transport/http/client.rb +12 -26
  291. data/lib/datadog/tracing/transport/http/traces.rb +2 -50
  292. data/lib/datadog/tracing/transport/http.rb +15 -9
  293. data/lib/datadog/tracing/transport/io/client.rb +1 -1
  294. data/lib/datadog/tracing/transport/trace_formatter.rb +11 -0
  295. data/lib/datadog/tracing/transport/traces.rb +9 -71
  296. data/lib/datadog/tracing/workers/trace_writer.rb +5 -0
  297. data/lib/datadog/tracing/writer.rb +1 -0
  298. data/lib/datadog/version.rb +2 -2
  299. data/lib/datadog.rb +3 -0
  300. metadata +110 -24
  301. data/ext/libdatadog_api/macos_development.md +0 -26
  302. data/lib/datadog/core/remote/transport/http/api.rb +0 -53
  303. data/lib/datadog/core/remote/transport/http/client.rb +0 -49
  304. data/lib/datadog/core/telemetry/transport/http/api.rb +0 -43
  305. data/lib/datadog/core/telemetry/transport/http/client.rb +0 -49
  306. data/lib/datadog/core/transport/http/api/spec.rb +0 -36
  307. data/lib/datadog/di/transport/http/api.rb +0 -42
  308. data/lib/datadog/di/transport/http/client.rb +0 -47
  309. data/lib/datadog/opentelemetry/api/baggage.rbs +0 -26
  310. data/lib/datadog/tracing/transport/http/api.rb +0 -44
@@ -0,0 +1,100 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'data_streams/processor'
4
+ require_relative 'data_streams/pathway_context'
5
+ require_relative 'data_streams/configuration/settings'
6
+ require_relative 'data_streams/extensions'
7
+ require_relative 'core/utils/time'
8
+
9
+ module Datadog
10
+ # Datadog Data Streams Monitoring public API.
11
+ #
12
+ # The Datadog team ensures that public methods in this module
13
+ # only receive backwards compatible changes, and breaking changes
14
+ # will only occur in new major versions releases.
15
+ # @public_api
16
+ module DataStreams
17
+ class << self
18
+ # Set a produce checkpoint for Data Streams Monitoring
19
+ #
20
+ # @param type [String] The type of the checkpoint (e.g., 'kafka', 'kinesis', 'sns')
21
+ # @param destination [String] The destination (e.g., topic, exchange, stream name)
22
+ # @param auto_instrumentation [Boolean] Whether this checkpoint was set by auto-instrumentation (default: false)
23
+ # @param tags [Hash] Additional tags to include
24
+ # @yield [key, value] Block to inject context into carrier
25
+ # @return [String, nil] Base64 encoded pathway context or nil if disabled
26
+ # @public_api
27
+ def set_produce_checkpoint(type:, destination:, auto_instrumentation: false, tags: {}, &block)
28
+ processor&.set_produce_checkpoint(
29
+ type: type,
30
+ destination: destination,
31
+ manual_checkpoint: !auto_instrumentation,
32
+ tags: tags,
33
+ &block
34
+ )
35
+ end
36
+
37
+ # Set a consume checkpoint for Data Streams Monitoring
38
+ #
39
+ # @param type [String] The type of the checkpoint (e.g., 'kafka', 'kinesis', 'sns')
40
+ # @param source [String] The source (e.g., topic, exchange, stream name)
41
+ # @param auto_instrumentation [Boolean] Whether this checkpoint was set by auto-instrumentation (default: false)
42
+ # @param tags [Hash] Additional tags to include
43
+ # @yield [key] Block to extract context from carrier
44
+ # @return [String, nil] Base64 encoded pathway context or nil if disabled
45
+ # @public_api
46
+ def set_consume_checkpoint(type:, source:, auto_instrumentation: false, tags: {}, &block)
47
+ processor&.set_consume_checkpoint(
48
+ type: type,
49
+ source: source,
50
+ manual_checkpoint: !auto_instrumentation,
51
+ tags: tags,
52
+ &block
53
+ )
54
+ end
55
+
56
+ # Track Kafka produce offset for lag monitoring
57
+ #
58
+ # @param topic [String] The Kafka topic name
59
+ # @param partition [Integer] The partition number
60
+ # @param offset [Integer] The offset of the produced message
61
+ # @return [Boolean, nil] true if tracking succeeded, nil if disabled
62
+ # @!visibility private
63
+ def track_kafka_produce(topic, partition, offset)
64
+ processor&.track_kafka_produce(topic, partition, offset, Core::Utils::Time.now)
65
+ end
66
+
67
+ # Track Kafka message consumption for consumer lag monitoring
68
+ #
69
+ # @param topic [String] The Kafka topic name
70
+ # @param partition [Integer] The partition number
71
+ # @param offset [Integer] The offset of the consumed message
72
+ # @return [Boolean, nil] true if tracking succeeded, nil if disabled
73
+ # @!visibility private
74
+ def track_kafka_consume(topic, partition, offset)
75
+ processor&.track_kafka_consume(topic, partition, offset, Core::Utils::Time.now)
76
+ end
77
+
78
+ # Check if Data Streams Monitoring is enabled and available
79
+ #
80
+ # @return [Boolean] true if the processor is available
81
+ # @public_api
82
+ def enabled?
83
+ !processor.nil?
84
+ end
85
+
86
+ private
87
+
88
+ def processor
89
+ components.data_streams
90
+ end
91
+
92
+ def components
93
+ Datadog.send(:components)
94
+ end
95
+ end
96
+
97
+ # Expose Data Streams to global shared objects
98
+ Extensions.activate!
99
+ end
100
+ end
@@ -5,6 +5,7 @@ require_relative 'base'
5
5
  require_relative 'error'
6
6
  require_relative 'code_tracker'
7
7
  require_relative 'component'
8
+ require_relative 'context'
8
9
  require_relative 'instrumenter'
9
10
  require_relative 'probe'
10
11
  require_relative 'probe_builder'
@@ -16,7 +17,9 @@ require_relative 'serializer'
16
17
  require_relative 'transport/http'
17
18
  require_relative 'utils'
18
19
 
19
- if %w[1 true yes].include?(ENV['DD_DYNAMIC_INSTRUMENTATION_ENABLED']) # steep:ignore
20
+ # Steep: https://github.com/ruby/rbs/pull/2715
21
+ if %w[1 true yes].include?(Datadog::DATADOG_ENV['DD_DYNAMIC_INSTRUMENTATION_ENABLED']) # steep:ignore ArgumentTypeMismatch
22
+
20
23
  # For initial release of Dynamic Instrumentation, activate code tracking
21
24
  # only if DI is explicitly requested in the environment.
22
25
  # Code tracking is required for line probes to work; see the comments
@@ -33,8 +36,9 @@ require_relative 'contrib'
33
36
 
34
37
  Datadog::DI::Contrib.load_now_or_later
35
38
 
36
- if %w[1 true yes].include?(ENV['DD_DYNAMIC_INSTRUMENTATION_ENABLED']) # steep:ignore
37
- if ENV['DD_DYNAMIC_INSTRUMENTATION_PROBE_FILE']
39
+ # Steep: https://github.com/ruby/rbs/pull/2715
40
+ if %w[1 true yes].include?(Datadog::DATADOG_ENV['DD_DYNAMIC_INSTRUMENTATION_ENABLED']) # steep:ignore ArgumentTypeMismatch
41
+ if Datadog::DATADOG_ENV['DD_DYNAMIC_INSTRUMENTATION_PROBE_FILE']
38
42
  require_relative 'probe_file_loader'
39
43
  Datadog::DI::ProbeFileLoader.load_now_or_later
40
44
  end
@@ -29,22 +29,6 @@ module Datadog
29
29
  end
30
30
  end
31
31
 
32
- def build!(settings, agent_settings, logger, telemetry: nil)
33
- unless settings.respond_to?(:dynamic_instrumentation) && settings.dynamic_instrumentation.enabled
34
- raise "Requested DI component but DI is not enabled in settings"
35
- end
36
-
37
- unless settings.respond_to?(:remote) && settings.remote.enabled
38
- raise "Requested DI component but remote config is not enabled in settings"
39
- end
40
-
41
- unless environment_supported?(settings, logger)
42
- raise "DI does not support the environment (development or Ruby version too low or not MRI)"
43
- end
44
-
45
- new(settings, agent_settings, logger, code_tracker: DI.code_tracker, telemetry: telemetry)
46
- end
47
-
48
32
  # Checks whether the runtime environment is supported by
49
33
  # dynamic instrumentation. Currently we only require that, if Rails
50
34
  # is used, that Rails environment is not development because
@@ -115,6 +99,20 @@ module Datadog
115
99
 
116
100
  def parse_probe_spec_and_notify(probe_spec)
117
101
  probe = ProbeBuilder.build_from_remote_config(probe_spec)
102
+ rescue => exc
103
+ begin
104
+ probe = Struct.new(:id).new(
105
+ probe_spec['id'],
106
+ )
107
+ payload = probe_notification_builder.build_errored(probe, exc)
108
+ probe_notifier_worker.add_status(payload)
109
+ rescue # standard:disable Lint/UselessRescue
110
+ # TODO report via instrumentation telemetry?
111
+ raise
112
+ end
113
+
114
+ raise
115
+ else
118
116
  payload = probe_notification_builder.build_received(probe)
119
117
  probe_notifier_worker.add_status(payload)
120
118
  probe
@@ -0,0 +1,70 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Datadog
4
+ module DI
5
+ # Contains local and instance variables used when evaluating
6
+ # expressions in DI Expression Language.
7
+ #
8
+ # @api private
9
+ class Context
10
+ def initialize(probe:, settings:, serializer:, locals: nil,
11
+ # In Ruby everything is a method, therefore we should always have
12
+ # a target self. However, if we are not capturing a snapshot,
13
+ # there is no need to pass in the target self.
14
+ target_self: nil,
15
+ path: nil, caller_locations: nil,
16
+ serialized_entry_args: nil,
17
+ return_value: nil, duration: nil, exception: nil)
18
+ @probe = probe
19
+ @settings = settings
20
+ @serializer = serializer
21
+ @locals = locals
22
+ @target_self = target_self
23
+ @path = path
24
+ @caller_locations = caller_locations
25
+ @serialized_entry_args = serialized_entry_args
26
+ @return_value = return_value
27
+ @duration = duration
28
+ @exception = exception
29
+ end
30
+
31
+ attr_reader :probe
32
+ attr_reader :settings
33
+ attr_reader :serializer
34
+ attr_reader :locals
35
+ attr_reader :target_self
36
+ # Actual path of the instrumented file.
37
+ attr_reader :path
38
+ # TODO check how many stack frames we should be keeping/sending,
39
+ # this should be all frames for enriched probes and no frames for
40
+ # non-enriched probes?
41
+ attr_reader :caller_locations
42
+ attr_reader :serialized_entry_args
43
+ # Return value for the method, for a method probe
44
+ attr_reader :return_value
45
+ # How long the method took to execute, for a method probe
46
+ attr_reader :duration
47
+ # Exception raised by the method, if any, for a method probe
48
+ attr_reader :exception
49
+
50
+ def serialized_locals
51
+ # TODO cache?
52
+ locals && serializer.serialize_vars(locals,
53
+ depth: probe.max_capture_depth || settings.dynamic_instrumentation.max_capture_depth,
54
+ attribute_count: probe.max_capture_attribute_count || settings.dynamic_instrumentation.max_capture_attribute_count,)
55
+ end
56
+
57
+ def fetch(var_name)
58
+ unless locals
59
+ # TODO return "undefined" instead?
60
+ return nil
61
+ end
62
+ locals[var_name.to_sym]
63
+ end
64
+
65
+ def fetch_ivar(var_name)
66
+ target_self.instance_variable_get(var_name)
67
+ end
68
+ end
69
+ end
70
+ end
@@ -1,12 +1,37 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- Datadog::DI::Serializer.register(condition: lambda { |value| ActiveRecord::Base === value }) do |serializer, value, name:, depth:| # steep:ignore
4
- # steep thinks all of the arguments are nil here
5
- # steep:ignore:start
3
+ Datadog::DI::Serializer.register(
4
+ # This serializer uses a dynamic condition to determine its applicability
5
+ # to a particular value. A simpler case could have been a serializer for
6
+ # a particular class, but in this case any ActiveRecord model is covered
7
+ # and they all have different classes.
8
+ #
9
+ # An alternative could have been to make DI specifically provide lookup
10
+ # logic for "instances of classes derived from X", but a condition Proc
11
+ # is more universal.
12
+ condition: lambda { |value| ActiveRecord::Base === value }
13
+ ) do |serializer, value, name:, depth:|
14
+ # +serializer+ is an instance of DI::Serializer.
15
+ # Use it to perform the serialization to primitive values.
16
+ #
17
+ # +value+ is the value to serialize. It should match the condition
18
+ # provided above, meaning it would be an ActiveRecord::Base instance.
19
+ #
20
+ # +name+ is the name of the (local/instance) variable being serialized.
21
+ # The name is used by DI for redaction (upstream of serialization logic),
22
+ # and could potentially be used for redaction here also.
23
+ #
24
+ # +depth+ is the remaining depth for serializing collections and objects.
25
+ # It should always be an integer.
26
+ # Reduce it by 1 when invoking +serialize_value+ on the contents of +value+.
27
+ # This serializer could also potentially do its own depth limiting.
28
+ #
29
+ # Steep: steep thinks all of the arguments are nil here
30
+ # Looks like it cannot handle kwargs in lambdas
31
+ # @type var depth: Integer
6
32
  value_to_serialize = {
7
33
  attributes: value.attributes,
8
34
  new_record: value.new_record?,
9
35
  }
10
- serializer.serialize_value(value_to_serialize, depth: depth ? depth - 1 : nil, type: value.class)
11
- # steep:ignore:end
36
+ serializer.serialize_value(value_to_serialize, depth: depth - 1, type: value.class)
12
37
  end
@@ -0,0 +1,168 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Datadog
4
+ module DI
5
+ module EL
6
+ # DI Expression Language compiler.
7
+ #
8
+ # Converts AST in probe definitions into Expression objects.
9
+ #
10
+ # WARNING: this class produces strings that are then eval'd as
11
+ # Ruby code. Input ASTs are user-controlled. As such the compiler
12
+ # must sanitize and escape all input to avoid injection.
13
+ #
14
+ # Besides quotes and backslashes we must also escape # which is
15
+ # starting string interpolation (#{...}).
16
+ #
17
+ # @api private
18
+ class Compiler
19
+ def compile(ast)
20
+ compile_partial(ast)
21
+ end
22
+
23
+ private
24
+
25
+ # Steep: https://github.com/soutaro/steep/issues/363
26
+ OPERATORS = { # steep:ignore IncompatibleAssignment
27
+ 'eq' => '==',
28
+ 'ne' => '!=',
29
+ 'ge' => '>=',
30
+ 'gt' => '>',
31
+ 'le' => '<=',
32
+ 'lt' => '<',
33
+ }.freeze
34
+
35
+ # Steep: https://github.com/soutaro/steep/issues/363
36
+ SINGLE_ARG_METHODS = %w[
37
+ len isEmpty isUndefined
38
+ ].freeze # steep:ignore IncompatibleAssignment
39
+
40
+ # Steep: https://github.com/soutaro/steep/issues/363
41
+ TWO_ARG_METHODS = %w[
42
+ startsWith endsWith contains matches
43
+ getmember index instanceof
44
+ ].freeze # steep:ignore IncompatibleAssignment
45
+
46
+ # Steep: https://github.com/soutaro/steep/issues/363
47
+ MULTI_ARG_METHODS = { # steep:ignore IncompatibleAssignment
48
+ 'and' => '&&',
49
+ 'or' => '||',
50
+ }.freeze
51
+
52
+ def compile_partial(ast)
53
+ case ast
54
+ when Hash
55
+ if ast.length != 1
56
+ raise DI::Error::InvalidExpression, "Expected hash of length 1: #{ast}"
57
+ end
58
+ op, target = ast.first
59
+ case op
60
+ when 'ref'
61
+ unless String === target
62
+ raise DI::Error::InvalidExpression, "Bad ref value type: #{target.class}: #{target}"
63
+ end
64
+ case target
65
+ when '@it'
66
+ 'current_item'
67
+ when '@key'
68
+ 'current_key'
69
+ when '@value'
70
+ 'current_value'
71
+ when '@return'
72
+ # For @return, @duration and @exception we shadow
73
+ # instance variables.
74
+ "context.return_value"
75
+ when '@duration'
76
+ # There is no way to explicitly format the duration.
77
+ # TODO come up with better formatting?
78
+ # We could format to a string here but what if customer
79
+ # has @duration as part of an expression and wants
80
+ # to retain it as a number?
81
+ "(context.duration * 1000)"
82
+ when '@exception'
83
+ "context.exception"
84
+ else
85
+ # Ruby technically allows all kinds of symbols in variable
86
+ # names, for example spaces and many characters.
87
+ # Start out with strict validation to avoid possible
88
+ # surprises and need to escape.
89
+ unless target =~ %r{\A(@?)([a-zA-Z0-9_]+)\z}
90
+ raise DI::Error::BadVariableName, "Bad variable name: #{target}"
91
+ end
92
+ method_name = (($1 == '@') ? 'iref' : 'ref')
93
+ "#{method_name}('#{target}')"
94
+ end
95
+ when *SINGLE_ARG_METHODS
96
+ method_name = op.gsub(/[A-Z]/) { |m| "_#{m.downcase}" }
97
+ "#{method_name}(#{compile_partial(target)}, '#{var_name_maybe(target)}')"
98
+ when *TWO_ARG_METHODS
99
+ method_name = op.gsub(/[A-Z]/) { |m| "_#{m.downcase}" }
100
+ unless Array === target && target.length == 2
101
+ raise DI::Error::InvalidExpression, "Improper #{op} syntax"
102
+ end
103
+ first, second = target
104
+ "#{method_name}(#{compile_partial(first)}, (#{compile_partial(second)}))"
105
+ when *MULTI_ARG_METHODS.keys
106
+ unless Array === target && target.length >= 1
107
+ raise DI::Error::InvalidExpression, "Improper #{op} syntax"
108
+ end
109
+ compiled_targets = target.map do |item|
110
+ "(#{compile_partial(item)})"
111
+ end
112
+ compiled_op = MULTI_ARG_METHODS[op]
113
+ "(#{compiled_targets.join(" #{compiled_op} ")})"
114
+ when 'substring'
115
+ unless Array === target && target.length == 3
116
+ raise DI::Error::InvalidExpression, "Improper #{op} syntax"
117
+ end
118
+ "#{op}(#{target.map { |arg| "(#{compile_partial(arg)})" }.join(", ")})"
119
+ when 'not'
120
+ "!(#{compile_partial(target)})"
121
+ when *OPERATORS.keys
122
+ unless Array === target && target.length == 2
123
+ raise DI::Error::InvalidExpression, "Improper #{op} syntax"
124
+ end
125
+ first, second = target
126
+ operator = OPERATORS.fetch(op)
127
+ "(#{compile_partial(first)}) #{operator} (#{compile_partial(second)})"
128
+ when 'any', 'all', 'filter'
129
+ "#{op}(#{compile_partial(target.first)}) { |current_item, current_key, current_value| #{compile_partial(target.last)} }"
130
+ else
131
+ raise DI::Error::InvalidExpression, "Unknown operation: #{op}"
132
+ end
133
+ when Numeric, true, false, nil
134
+ # No escaping is needed for the values here.
135
+ ast.inspect
136
+ when String
137
+ "\"#{escape(ast)}\""
138
+ when Array
139
+ # Arrays are commonly used as arguments of operators/methods,
140
+ # but there are no arrays at the top level in the syntax that
141
+ # we currently understand. Provide a helpful error message in case
142
+ # syntax is expanded in the future.
143
+ raise DI::Error::InvalidExpression, "Array is not valid at its location, do you need to upgrade dd-trace-rb? #{ast}"
144
+ else
145
+ raise DI::Error::InvalidExpression, "Unknown type in AST: #{ast}"
146
+ end
147
+ end
148
+
149
+ # Returns a textual description of +target+ for use in exception
150
+ # messages. +target+ could be any expression language expression.
151
+ # WARNING: the result of this method is included in eval'd code,
152
+ # it must be sanitized to avoid injection.
153
+ def var_name_maybe(target)
154
+ if Hash === target && target.length == 1 && target.keys.first == 'ref' &&
155
+ String === (value = target.values.first)
156
+ escape(value)
157
+ else
158
+ '(expression)'
159
+ end
160
+ end
161
+
162
+ def escape(needle)
163
+ needle.gsub("\\") { "\\\\" }.gsub('"') { "\\\"" }.gsub('#') { "\\#" }
164
+ end
165
+ end
166
+ end
167
+ end
168
+ end
@@ -0,0 +1,159 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Datadog
4
+ module DI
5
+ module EL
6
+ # Evaluator for expression language.
7
+ #
8
+ # @api private
9
+ class Evaluator
10
+ def ref(var)
11
+ @context.fetch(var)
12
+ end
13
+
14
+ def iref(var)
15
+ @context.fetch_ivar(var)
16
+ end
17
+
18
+ def len(var, var_name)
19
+ case var
20
+ when Array, String, Hash
21
+ var.length
22
+ else
23
+ raise DI::Error::ExpressionEvaluationError, "Unsupported type for length: #{var.class}: #{var_name}"
24
+ end
25
+ end
26
+
27
+ def is_empty(var, var_name)
28
+ case var
29
+ when nil, Numeric
30
+ false
31
+ when Array, String
32
+ var.empty?
33
+ else
34
+ raise DI::Error::ExpressionEvaluationError, "Unsupported type for isEmpty: #{var.class}: #{var_name}"
35
+ end
36
+ end
37
+
38
+ def is_undefined(var, var_name)
39
+ var.nil?
40
+ end
41
+
42
+ def contains(haystack, needle)
43
+ if String === haystack && String === needle or # standard:disable Style/AndOr
44
+ Array === haystack
45
+ haystack.include?(needle)
46
+ else
47
+ raise DI::Error::ExpressionEvaluationError, "Invalid arguments for contains: #{haystack}, #{needle}"
48
+ end
49
+ end
50
+
51
+ def matches(haystack, needle)
52
+ re = Regexp.compile(needle)
53
+ !!(haystack =~ re)
54
+ end
55
+
56
+ def getmember(object, field)
57
+ object.instance_variable_get("@#{field}")
58
+ end
59
+
60
+ def index(array_or_hash, index_or_key)
61
+ case array_or_hash
62
+ when Array
63
+ case index_or_key
64
+ when Integer
65
+ array_or_hash[index_or_key]
66
+ else
67
+ raise DI::Error::ExpressionEvaluationError, "Invalid index value: #{index_or_key}"
68
+ end
69
+ when Hash
70
+ array_or_hash[index_or_key]
71
+ else
72
+ raise DI::Error::ExpressionEvaluationError, "Invalid argument for index: #{array_or_hash}"
73
+ end
74
+ end
75
+
76
+ def substring(object, from, to)
77
+ unless String === object
78
+ raise DI::Error::ExpressionEvaluationError, "Invalid type for substring: #{object}"
79
+ end
80
+ object[from...to]
81
+ end
82
+
83
+ def starts_with(haystack, needle)
84
+ # To guard against running arbitrary customer code, check that
85
+ # the haystack is a string. This does not help if customer
86
+ # overrode String#start_with? but at least it's better than nothing.
87
+ String === haystack && haystack.start_with?(needle)
88
+ end
89
+
90
+ def ends_with(haystack, needle)
91
+ String === haystack && haystack.end_with?(needle)
92
+ end
93
+
94
+ def all(collection, &block)
95
+ case collection
96
+ when Array
97
+ collection.all? do |item|
98
+ block.call(item)
99
+ end
100
+ when Hash
101
+ # For hashes, the expression language has both @it and
102
+ # @key/@value. Manufacture @it from the key and value.
103
+ collection.all? do |key, value|
104
+ block.call([key, value], key, value)
105
+ end
106
+ else
107
+ raise DI::Error::ExpressionEvaluationError, "Bad collection type for all: #{collection.class}"
108
+ end
109
+ end
110
+
111
+ def any(collection, &block)
112
+ case collection
113
+ when Array
114
+ collection.any? do |item|
115
+ block.call(item)
116
+ end
117
+ when Hash
118
+ collection.any? do |key, value|
119
+ # For hashes, the expression language has both @it and
120
+ # @key/@value. Manufacture @it from the key and value.
121
+ block.call([key, value], key, value)
122
+ end
123
+ else
124
+ raise DI::Error::ExpressionEvaluationError, "Bad collection type for any: #{collection.class}"
125
+ end
126
+ end
127
+
128
+ def filter(collection, &block)
129
+ case collection
130
+ when Array
131
+ collection.select do |item|
132
+ block.call(item)
133
+ end
134
+ when Hash
135
+ collection.select do |key, value|
136
+ block.call([key, value], key, value)
137
+ end.to_h
138
+ else
139
+ raise DI::Error::ExpressionEvaluationError, "Bad collection type for filter: #{collection.class}"
140
+ end
141
+ end
142
+
143
+ def instanceof(object, cls_name)
144
+ cls = object.class
145
+ loop do
146
+ if cls.name == cls_name
147
+ return true
148
+ end
149
+ if supercls = cls.superclass # standard:disable Lint/AssignmentInCondition
150
+ cls = supercls
151
+ else
152
+ return false
153
+ end
154
+ end
155
+ end
156
+ end
157
+ end
158
+ end
159
+ end
@@ -0,0 +1,42 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Datadog
4
+ module DI
5
+ module EL
6
+ # Represents an Expression Language expression.
7
+ #
8
+ # @api private
9
+ class Expression
10
+ def initialize(dsl_expr, compiled_expr)
11
+ unless String === compiled_expr
12
+ raise ArgumentError, "compiled_expr must be a string"
13
+ end
14
+
15
+ @dsl_expr = dsl_expr
16
+
17
+ cls = Class.new(Evaluator)
18
+ cls.class_exec do
19
+ eval(<<-RUBY, Object.new.send(:binding), __FILE__, __LINE__ + 1) # standard:disable Security/Eval
20
+ def evaluate(context)
21
+ @context = context
22
+ #{compiled_expr}
23
+ end
24
+ RUBY
25
+ end
26
+ @evaluator = cls.new
27
+ end
28
+
29
+ attr_reader :dsl_expr
30
+ attr_reader :evaluator
31
+
32
+ def evaluate(context)
33
+ @evaluator.evaluate(context)
34
+ end
35
+
36
+ def satisfied?(context)
37
+ !!evaluate(context)
38
+ end
39
+ end
40
+ end
41
+ end
42
+ end
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'el/expression'
4
+ require_relative 'el/compiler'
5
+ require_relative 'el/evaluator'