Module: Datadog::CI::Configuration::Components
- Defined in:
- lib/datadog/ci/configuration/components.rb
Overview
Adds CI behavior to Datadog trace components
Instance Attribute Summary collapse
-
#agentless_logs_submission ⇒ Object
readonly
Returns the value of attribute agentless_logs_submission.
-
#ci_remote ⇒ Object
readonly
Returns the value of attribute ci_remote.
-
#code_coverage ⇒ Object
readonly
Returns the value of attribute code_coverage.
-
#git_tree_upload_worker ⇒ Object
readonly
Returns the value of attribute git_tree_upload_worker.
-
#impacted_tests_detection ⇒ Object
readonly
Returns the value of attribute impacted_tests_detection.
-
#test_discovery ⇒ Object
readonly
Returns the value of attribute test_discovery.
-
#test_management ⇒ Object
readonly
Returns the value of attribute test_management.
-
#test_optimisation ⇒ Object
readonly
Returns the value of attribute test_optimisation.
-
#test_retries ⇒ Object
readonly
Returns the value of attribute test_retries.
-
#test_visibility ⇒ Object
readonly
Returns the value of attribute test_visibility.
Instance Method Summary collapse
- #activate_ci!(settings) ⇒ Object
- #build_agentless_logs_component(settings, api) ⇒ Object
- #build_code_coverage(settings, api) ⇒ Object
- #build_coverage_writer(settings, api) ⇒ Object
- #build_git_upload_worker(settings, api) ⇒ Object
- #build_known_tests_client(settings, api) ⇒ Object
- #build_library_settings_client(settings, api) ⇒ Object
- #build_logs_writer(settings, api) ⇒ Object
- #build_test_optimisation(settings, test_visibility_api) ⇒ Object
- #build_test_visibility_api(settings) ⇒ Object
- #build_tracing_transport(settings, api) ⇒ Object
- #check_dd_site(settings) ⇒ Object
- #configure_telemetry(settings) ⇒ Object
-
#configure_time_providers(settings) ⇒ Object
When timecop is present: - Time.now is mocked and .now_without_mock_time is added on Time to get the current time without the mock.
-
#custom_configuration(settings) ⇒ Object
fetch custom tags provided by the user in DD_TAGS env var with prefix test.configuration.
- #initialize(settings) ⇒ Object
- #serializers_factory(settings) ⇒ Object
- #shutdown!(replacement = nil) ⇒ Object
- #timecop? ⇒ Boolean
Instance Attribute Details
#agentless_logs_submission ⇒ Object (readonly)
Returns the value of attribute agentless_logs_submission.
47 48 49 |
# File 'lib/datadog/ci/configuration/components.rb', line 47 def agentless_logs_submission @agentless_logs_submission end |
#ci_remote ⇒ Object (readonly)
Returns the value of attribute ci_remote.
47 48 49 |
# File 'lib/datadog/ci/configuration/components.rb', line 47 def ci_remote @ci_remote end |
#code_coverage ⇒ Object (readonly)
Returns the value of attribute code_coverage.
47 48 49 |
# File 'lib/datadog/ci/configuration/components.rb', line 47 def code_coverage @code_coverage end |
#git_tree_upload_worker ⇒ Object (readonly)
Returns the value of attribute git_tree_upload_worker.
47 48 49 |
# File 'lib/datadog/ci/configuration/components.rb', line 47 def git_tree_upload_worker @git_tree_upload_worker end |
#impacted_tests_detection ⇒ Object (readonly)
Returns the value of attribute impacted_tests_detection.
47 48 49 |
# File 'lib/datadog/ci/configuration/components.rb', line 47 def impacted_tests_detection @impacted_tests_detection end |
#test_discovery ⇒ Object (readonly)
Returns the value of attribute test_discovery.
47 48 49 |
# File 'lib/datadog/ci/configuration/components.rb', line 47 def test_discovery @test_discovery end |
#test_management ⇒ Object (readonly)
Returns the value of attribute test_management.
47 48 49 |
# File 'lib/datadog/ci/configuration/components.rb', line 47 def test_management @test_management end |
#test_optimisation ⇒ Object (readonly)
Returns the value of attribute test_optimisation.
47 48 49 |
# File 'lib/datadog/ci/configuration/components.rb', line 47 def test_optimisation @test_optimisation end |
#test_retries ⇒ Object (readonly)
Returns the value of attribute test_retries.
47 48 49 |
# File 'lib/datadog/ci/configuration/components.rb', line 47 def test_retries @test_retries end |
#test_visibility ⇒ Object (readonly)
Returns the value of attribute test_visibility.
47 48 49 |
# File 'lib/datadog/ci/configuration/components.rb', line 47 def test_visibility @test_visibility end |
Instance Method Details
#activate_ci!(settings) ⇒ Object
80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 |
# File 'lib/datadog/ci/configuration/components.rb', line 80 def activate_ci!(settings) unless settings.tracing.enabled Datadog.logger.error( "Test Optimization requires tracing to be enabled. Disabling Test Optimization. " \ "NOTE: if you didn't disable tracing intentionally, add `c.tracing.enabled = true` to " \ "your Datadog.configure block." ) settings.ci.enabled = false return end # Builds test visibility API layer in agentless or EvP proxy mode test_visibility_api = build_test_visibility_api(settings) # bail out early if api is misconfigured return unless settings.ci.enabled # Configure datadog gem for test visibility mode configure_telemetry(settings) # Test visibility uses its own remote settings settings.remote.enabled = false # startup logs are useless for test visibility and create noise settings.diagnostics.startup_logs.enabled = false # timecop configuration configure_time_providers(settings) # first check if we are in test discovery mode and configure library accordingly # @type ivar @test_discovery: Datadog::CI::TestDiscovery::Component @test_discovery = TestDiscovery::Component.new( enabled: settings.ci.test_discovery_enabled, output_path: settings.ci.test_discovery_output_path ) @test_discovery.disable_features_for_test_discovery!(settings) # Configure Datadog::Tracing module # No need not use 128-bit trace ids for test visibility, # they are used for OTEL compatibility in Datadog tracer settings.tracing.trace_id_128_bit_generation_enabled = false # Activate underlying tracing test mode with async worker settings.tracing.test_mode.enabled = true settings.tracing.test_mode.async = true settings.tracing.test_mode.trace_flush = settings.ci.trace_flush || CI::TestVisibility::Flush::Partial.new = settings.ci. [:shutdown_timeout] = 60 [:buffer_size] = 10_000 tracing_transport = build_tracing_transport(settings, test_visibility_api) [:transport] = tracing_transport if tracing_transport settings.tracing.test_mode. = @git_tree_upload_worker = build_git_upload_worker(settings, test_visibility_api) @ci_remote = Remote::Component.new( library_settings_client: build_library_settings_client(settings, test_visibility_api), test_discovery_enabled: settings.ci.test_discovery_enabled ) @test_retries = TestRetries::Component.new( retry_failed_tests_enabled: settings.ci.retry_failed_tests_enabled, retry_failed_tests_max_attempts: settings.ci.retry_failed_tests_max_attempts, retry_failed_tests_total_limit: settings.ci.retry_failed_tests_total_limit, retry_new_tests_enabled: settings.ci.retry_new_tests_enabled, retry_flaky_fixed_tests_enabled: settings.ci.test_management_enabled, retry_flaky_fixed_tests_max_attempts: settings.ci.test_management_attempt_to_fix_retries_count ) @test_management = TestManagement::Component.new( enabled: settings.ci.test_management_enabled, tests_properties_client: TestManagement::TestsProperties.new(api: test_visibility_api) ) # @type ivar @test_optimisation: Datadog::CI::TestOptimisation::Component @test_optimisation = build_test_optimisation(settings, test_visibility_api) @test_visibility = TestVisibility::Component.new( test_suite_level_visibility_enabled: !settings.ci.force_test_level_visibility, logical_test_session_name: settings.ci.test_session_name, known_tests_client: build_known_tests_client(settings, test_visibility_api), context_service_uri: settings.ci.test_visibility_drb_server_uri ) @agentless_logs_submission = build_agentless_logs_component(settings, test_visibility_api) @impacted_tests_detection = ImpactedTestsDetection::Component.new(enabled: settings.ci.impacted_tests_detection_enabled) @code_coverage = build_code_coverage(settings, test_visibility_api) end |
#build_agentless_logs_component(settings, api) ⇒ Object
302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 |
# File 'lib/datadog/ci/configuration/components.rb', line 302 def build_agentless_logs_component(settings, api) if settings.ci.agentless_logs_submission_enabled && !settings.ci.agentless_mode_enabled Datadog.logger.warn( "Agentless logs submission is enabled but agentless mode is not enabled. " \ "Logs will not be submitted. " \ "Please make sure to set DD_CIVISIBILITY_AGENTLESS_ENABLED to true if you want to submit logs in agentless mode. " \ "Otherwise, set DD_AGENTLESS_LOG_SUBMISSION_ENABLED to 0 and use Datadog Agent to submit logs." ) settings.ci.agentless_logs_submission_enabled = false end Logs::Component.new( enabled: settings.ci.agentless_logs_submission_enabled, writer: build_logs_writer(settings, api) ) end |
#build_code_coverage(settings, api) ⇒ Object
293 294 295 296 297 298 299 300 |
# File 'lib/datadog/ci/configuration/components.rb', line 293 def build_code_coverage(settings, api) return CodeCoverage::NullComponent.new if api.nil? || settings.ci.discard_traces CodeCoverage::Component.new( enabled: settings.ci.code_coverage_report_upload_enabled, transport: CodeCoverage::Transport.new(api: api) ) end |
#build_coverage_writer(settings, api) ⇒ Object
257 258 259 260 261 262 263 264 |
# File 'lib/datadog/ci/configuration/components.rb', line 257 def build_coverage_writer(settings, api) # nil means that coverage event will be ignored return nil if api.nil? || settings.ci.discard_traces AsyncWriter.new( transport: TestOptimisation::Coverage::Transport.new(api: api) ) end |
#build_git_upload_worker(settings, api) ⇒ Object
266 267 268 269 270 271 272 273 274 275 |
# File 'lib/datadog/ci/configuration/components.rb', line 266 def build_git_upload_worker(settings, api) if settings.ci. git_tree_uploader = Git::TreeUploader.new(api: api, force_unshallow: settings.ci.impacted_tests_detection_enabled) Worker.new do |repository_url| git_tree_uploader.call(repository_url) end else DummyWorker.new end end |
#build_known_tests_client(settings, api) ⇒ Object
285 286 287 288 289 290 291 |
# File 'lib/datadog/ci/configuration/components.rb', line 285 def build_known_tests_client(settings, api) TestVisibility::KnownTests.new( api: api, dd_env: settings.env, config_tags: custom_configuration(settings) ) end |
#build_library_settings_client(settings, api) ⇒ Object
277 278 279 280 281 282 283 |
# File 'lib/datadog/ci/configuration/components.rb', line 277 def build_library_settings_client(settings, api) Remote::LibrarySettingsClient.new( api: api, dd_env: settings.env, config_tags: custom_configuration(settings) ) end |
#build_logs_writer(settings, api) ⇒ Object
319 320 321 322 323 |
# File 'lib/datadog/ci/configuration/components.rb', line 319 def build_logs_writer(settings, api) return nil if api.nil? || settings.ci.discard_traces AsyncWriter.new(transport: Logs::Transport.new(api: api), options: {buffer_size: 1024}) end |
#build_test_optimisation(settings, test_visibility_api) ⇒ Object
170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 |
# File 'lib/datadog/ci/configuration/components.rb', line 170 def build_test_optimisation(settings, test_visibility_api) if settings.ci.itr_code_coverage_use_single_threaded_mode && settings.ci.itr_test_impact_analysis_use_allocation_tracing Datadog.logger.warn( "Test Impact Analysis: Single threaded coverage mode is incompatible with allocation tracing. " \ "Allocation tracing will be disabled. It means that test impact analysis will not be able to detect " \ "instantiations of objects in your code, which is important for ActiveRecord models. " \ "Please add your app/model folder to the list of tracked files or disable single threaded coverage mode." ) settings.ci.itr_test_impact_analysis_use_allocation_tracing = false end if RUBY_VERSION.start_with?("3.2.") && RUBY_VERSION < "3.2.3" && settings.ci.itr_test_impact_analysis_use_allocation_tracing Datadog.logger.warn( "Test Impact Analysis: Allocation tracing is not supported in Ruby versions 3.2.0, 3.2.1 and 3.2.2 and will be forcibly " \ "disabled. This is due to a VM bug that can lead to crashes (https://bugs.ruby-lang.org/issues/19482). " \ "Please update your Ruby version or add your app/model folder to the list of tracked files." \ "Set env variable DD_CIVISIBILITY_ITR_TEST_IMPACT_ANALYSIS_USE_ALLOCATION_TRACING to 0 to disable this warning." ) settings.ci.itr_test_impact_analysis_use_allocation_tracing = false end TestOptimisation::Component.new( api: test_visibility_api, dd_env: settings.env, config_tags: custom_configuration(settings), coverage_writer: build_coverage_writer(settings, test_visibility_api), enabled: settings.ci.enabled && settings.ci.itr_enabled, bundle_location: settings.ci.itr_code_coverage_excluded_bundle_path, use_single_threaded_coverage: settings.ci.itr_code_coverage_use_single_threaded_mode, use_allocation_tracing: settings.ci.itr_test_impact_analysis_use_allocation_tracing, static_dependencies_tracking_enabled: settings.ci.tia_static_dependencies_tracking_enabled ) end |
#build_test_visibility_api(settings) ⇒ Object
207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 |
# File 'lib/datadog/ci/configuration/components.rb', line 207 def build_test_visibility_api(settings) if settings.ci.agentless_mode_enabled check_dd_site(settings) Datadog.logger.debug("Test Optimization configured to use agentless transport") api = Transport::Api::Builder.build_agentless_api(settings) if api.nil? Datadog.logger.error do "DATADOG CONFIGURATION - TEST OPTIMIZATION - ATTENTION - " \ "Agentless mode was enabled but DD_API_KEY is not set: Test Optimization is disabled. " \ "Please make sure to set valid api key in DD_API_KEY environment variable" end # Tests are running without Test Optimization enabled settings.ci.enabled = false end else Datadog.logger.debug("Test Optimization configured to use agent transport via EVP proxy") api = Transport::Api::Builder.build_evp_proxy_api(settings) if api.nil? Datadog.logger.debug( "Old agent version detected, no evp_proxy support. Forcing test level visibility mode" ) # only legacy APM protocol is supported, so no test suite level visibility settings.ci.force_test_level_visibility = true # ITR is not supported with APM protocol settings.ci.itr_enabled = false end end api end |
#build_tracing_transport(settings, api) ⇒ Object
244 245 246 247 248 249 250 251 252 253 254 255 |
# File 'lib/datadog/ci/configuration/components.rb', line 244 def build_tracing_transport(settings, api) # NullTransport ignores traces return TestVisibility::NullTransport.new if settings.ci.discard_traces # nil means that default legacy APM transport will be used (only for very old Datadog Agent versions) return nil if api.nil? TestVisibility::Transport.new( api: api, serializers_factory: serializers_factory(settings), dd_env: settings.env ) end |
#check_dd_site(settings) ⇒ Object
339 340 341 342 343 344 345 346 347 348 |
# File 'lib/datadog/ci/configuration/components.rb', line 339 def check_dd_site(settings) return if settings.site.nil? return if Ext::Settings::DD_SITE_ALLOWLIST.include?(settings.site) Datadog.logger.warn do "TEST OPTIMIZATION CONFIGURATION " \ "Agentless mode was enabled but DD_SITE is not set to one of the following: #{Ext::Settings::DD_SITE_ALLOWLIST.join(", ")}. " \ "Please make sure to set valid site in DD_SITE environment variable" end end |
#configure_telemetry(settings) ⇒ Object
350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 |
# File 'lib/datadog/ci/configuration/components.rb', line 350 def configure_telemetry(settings) # in development environment Datadog's telemetry is disabled by default # for test visibility we want to enable it by default unless explicitly disabled # NOTE: before agentless mode is released, we only enable telemetry when running with Datadog Agent env_telemetry_enabled = ENV[Core::Telemetry::Ext::ENV_ENABLED] settings.telemetry.enabled = env_telemetry_enabled.nil? || Utils::Parsing.convert_to_bool(env_telemetry_enabled) return unless settings.telemetry.enabled settings.telemetry.agentless_enabled = true if settings.ci.agentless_mode_enabled settings.telemetry.shutdown_timeout_seconds = 60.0 begin require "datadog/core/transport/http/adapters/net" # patch gem's core transport layer to use Net::HTTP instead of WebMock's Net::HTTP Core::Transport::HTTP::Adapters::Net.include(CI::Transport::Adapters::TelemetryWebmockSafeAdapter) rescue LoadError, StandardError => e Datadog.logger.warn("Failed to patch Datadog gem's telemetry layer: #{e}") end # for compatibility with old telemetry transport begin require "datadog/core/telemetry/http/adapters/net" Core::Telemetry::Http::Adapters::Net.include(CI::Transport::Adapters::TelemetryWebmockSafeAdapter) rescue LoadError, StandardError => e Datadog.logger.debug("The old telemetry transport layer is not available: #{e}") end end |
#configure_time_providers(settings) ⇒ Object
When timecop is present:
-
Time.now is mocked and .now_without_mock_time is added on Time to get the current time without the mock.
-
Process.clock_gettime is mocked and .clock_gettime_without_mock is added on Process to get the monotonic time without the mock.
383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 |
# File 'lib/datadog/ci/configuration/components.rb', line 383 def configure_time_providers(settings) return unless timecop? settings.time_now_provider = -> do Time.now_without_mock_time rescue NoMethodError # fallback to normal Time.now if Time.now_without_mock_time is not defined for any reason Time.now end if defined?(Process.clock_gettime_without_mock) settings.get_time_provider = ->(unit = :float_second) do ::Process.clock_gettime_without_mock(::Process::CLOCK_MONOTONIC, unit) rescue NoMethodError # fallback to normal Process.clock_gettime if Process.clock_gettime_without_mock is not defined for any reason Process.clock_gettime(::Process::CLOCK_MONOTONIC, unit) end end end |
#custom_configuration(settings) ⇒ Object
fetch custom tags provided by the user in DD_TAGS env var with prefix test.configuration.
327 328 329 |
# File 'lib/datadog/ci/configuration/components.rb', line 327 def custom_configuration(settings) @custom_configuration ||= Utils::TestRun.custom_configuration(settings.) end |
#initialize(settings) ⇒ Object
50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 |
# File 'lib/datadog/ci/configuration/components.rb', line 50 def initialize(settings) @test_optimisation = TestOptimisation::NullComponent.new @test_visibility = TestVisibility::NullComponent.new @git_tree_upload_worker = DummyWorker.new @ci_remote = Remote::NullComponent.new @test_retries = TestRetries::NullComponent.new @test_management = TestManagement::NullComponent.new @impacted_tests_detection = ImpactedTestsDetection::NullComponent.new @test_discovery = TestDiscovery::NullComponent.new @code_coverage = CodeCoverage::NullComponent.new # Activate CI mode if enabled if settings.ci.enabled activate_ci!(settings) end super end |
#serializers_factory(settings) ⇒ Object
331 332 333 334 335 336 337 |
# File 'lib/datadog/ci/configuration/components.rb', line 331 def serializers_factory(settings) if settings.ci.force_test_level_visibility TestVisibility::Serializers::Factories::TestLevel else TestVisibility::Serializers::Factories::TestSuiteLevel end end |
#shutdown!(replacement = nil) ⇒ Object
69 70 71 72 73 74 75 76 77 78 |
# File 'lib/datadog/ci/configuration/components.rb', line 69 def shutdown!(replacement = nil) super @test_visibility&.shutdown! @test_optimisation&.shutdown! @agentless_logs_submission&.shutdown! @test_discovery&.shutdown! @code_coverage&.shutdown! @git_tree_upload_worker&.stop end |
#timecop? ⇒ Boolean
403 404 405 |
# File 'lib/datadog/ci/configuration/components.rb', line 403 def timecop? Gem.loaded_specs.key?("timecop") || !!defined?(Timecop) end |