Skip to content

Commit

Permalink
DEBUG-3175 DI snapshot/rate limit benchmark (#4207)
Browse files Browse the repository at this point in the history
  • Loading branch information
p-datadog authored Dec 18, 2024
1 parent 11b9ae1 commit 7fd1feb
Show file tree
Hide file tree
Showing 6 changed files with 306 additions and 6 deletions.
257 changes: 257 additions & 0 deletions benchmarks/di_snapshot.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,257 @@
=begin
This benchmark verifies that the rate limits used for dynamic instrumentation
probes are attainable.
Each benchmark performs as many operations as the rate limit permits -
5000 for a basic probe and 1 for enriched probe. If the benchmark
produces a rate of fewer than 1 "instructions" per second, the rate limit is
not being reached. A result of more than 1 "instruction" per second
means the rate limit is being reached.
Note that the number of "instructions per second" reported by benchmark/ips
does not reflect how many times the instrumentation creates a snapshot -
there can (and normally are) invocations of the target method that do not
produce DI snapshots due to rate limit but these invocations are counted in
the "instructions per second" reported by benchmark/ips.
The default dynamic instrumentation settings for the probe notifier worker
(queue capacity of 100 and minimum send interval of 3 seconds) mean an
effective rate limit of 30 snapshots per second for basic probes,
which is shared across all probes in the application, which is significantly
below the 5000 snapshots per second per probe that DI is theoretically
supposed to achieve. However, to increase actual attainable snapshot rate
to 5000/second, the probe notifier worker needs to be changed to send
multiple network requests for a single queue processing run or be more
aggressive in flushing the snapshots to the network when the queue is getting
full. In either case care needs to be taken not to starve customer applications
of CPU.
=end

# Used to quickly run benchmark under RSpec as part of the usual test suite, to validate it didn't bitrot
VALIDATE_BENCHMARK_MODE = ENV['VALIDATE_BENCHMARK'] == 'true'

return unless __FILE__ == $PROGRAM_NAME || VALIDATE_BENCHMARK_MODE

require 'benchmark/ips'
require 'datadog'
require 'webrick'

class DISnapshotBenchmark
# If we are validating the benchmark a single operation is sufficient.
BASIC_RATE_LIMIT = VALIDATE_BENCHMARK_MODE ? 1 : 5000
ENRICHED_RATE_LIMIT = 1

def initialize

Datadog::DI.activate_tracking!

Datadog.configure do |c|
c.remote.enabled = true
c.dynamic_instrumentation.enabled = true
c.dynamic_instrumentation.internal.development = true

# Increase queue capacity and reduce min send interval
# to be able to send more snapshots out.
# The default settings will result in dropped snapshots
# way before non-enriched probe rate limit is reached.
c.dynamic_instrumentation.internal.snapshot_queue_capacity = 10000
c.dynamic_instrumentation.internal.min_send_interval = 1
end

Thread.new do
# If there is an actual Datadog agent running locally, the server
# used in this benchmark will fail to start.
# Using an actual Datadog agent instead of the fake server should not
# affect the indication of whether the rate limit is reachable
# since the agent shouldn't take longer to process than the fake
# web server (and the agent should also run on another core),
# however using a real agent would forego reports of the number of
# snapshots submitted and their size.
server.start
end

require_relative 'support/di_snapshot_target'
end

def run_benchmark
probe = Datadog::DI::Probe.new(
id: 1, type: :log,
type_name: 'DISnapshotTarget', method_name: 'test_method',
rate_limit: BASIC_RATE_LIMIT,
)

unless probe_manager.add_probe(probe)
raise "Failed to instrument method (without snapshot capture)"
end

@received_snapshot_count = 0
@received_snapshot_bytes = 0

Benchmark.ips do |x|
benchmark_time = VALIDATE_BENCHMARK_MODE ? { time: 0.01, warmup: 0 } : { time: 10, warmup: 2 }
x.config(
**benchmark_time,
)

x.report('method probe - basic') do
BASIC_RATE_LIMIT.times do
DISnapshotTarget.new.test_method
end
Datadog::DI.component.probe_notifier_worker.flush
end

x.save! 'di-snapshot-results.json' unless VALIDATE_BENCHMARK_MODE
x.compare!
end

# DI does not provide an API to remove a specific probe because
# this functionality is currently not needed by the product.
probe_manager.remove_other_probes([])

puts "Received #{@received_snapshot_count} snapshots, #{@received_snapshot_bytes} bytes total"

probe = Datadog::DI::Probe.new(
id: 1, type: :log,
type_name: 'DISnapshotTarget', method_name: 'test_method',
capture_snapshot: true,
# Normally rate limit for enriched probes is 1.
# To get a meaningful number of submissions, increase it to 20.
# We should get about 200 snapshots in the 10 seconds that the
# benchmark is supposed to run.
rate_limit: ENRICHED_RATE_LIMIT,
)

unless probe_manager.add_probe(probe)
raise "Failed to instrument method (with snapshot capture)"
end

@received_snapshot_count = 0
@received_snapshot_bytes = 0

Benchmark.ips do |x|
benchmark_time = VALIDATE_BENCHMARK_MODE ? { time: 0.01, warmup: 0 } : { time: 10, warmup: 2 }
x.config(
**benchmark_time,
)

x.report('method probe - enriched') do
ENRICHED_RATE_LIMIT.times do
DISnapshotTarget.new.test_method
end
Datadog::DI.component.probe_notifier_worker.flush
end

x.save! 'di-snapshot-results.json' unless VALIDATE_BENCHMARK_MODE
x.compare!
end

probe_manager.remove_other_probes([])

puts "Received #{@received_snapshot_count} snapshots, #{@received_snapshot_bytes} bytes total"

probe = Datadog::DI::Probe.new(
id: 1, type: :log,
file: 'di_snapshot_target.rb', line_no: 20,
capture_snapshot: false,
rate_limit: BASIC_RATE_LIMIT,
)

unless probe_manager.add_probe(probe)
raise "Failed to instrument line (with snapshot capture)"
end

@received_snapshot_count = 0
@received_snapshot_bytes = 0

Benchmark.ips do |x|
benchmark_time = VALIDATE_BENCHMARK_MODE ? { time: 0.01, warmup: 0 } : { time: 10, warmup: 2 }
x.config(
**benchmark_time,
)

x.report('line probe - basic') do
BASIC_RATE_LIMIT.times do
DISnapshotTarget.new.test_method
end
Datadog::DI.component.probe_notifier_worker.flush
end

x.save! 'di-snapshot-results.json' unless VALIDATE_BENCHMARK_MODE
x.compare!
end

probe_manager.remove_other_probes([])

puts "Received #{@received_snapshot_count} snapshots, #{@received_snapshot_bytes} bytes total"

probe = Datadog::DI::Probe.new(
id: 1, type: :log,
file: 'di_snapshot_target.rb', line_no: 20,
capture_snapshot: true,
rate_limit: ENRICHED_RATE_LIMIT,
)

unless probe_manager.add_probe(probe)
raise "Failed to instrument line (with snapshot capture)"
end

@received_snapshot_count = 0
@received_snapshot_bytes = 0

Benchmark.ips do |x|
benchmark_time = VALIDATE_BENCHMARK_MODE ? { time: 0.01, warmup: 0 } : { time: 10, warmup: 2 }
x.config(
**benchmark_time,
)

x.report('line probe - enriched') do
ENRICHED_RATE_LIMIT.times do
DISnapshotTarget.new.test_method
end
Datadog::DI.component.probe_notifier_worker.flush
end

x.save! 'di-snapshot-results.json' unless VALIDATE_BENCHMARK_MODE
x.compare!
end

probe_manager.remove_other_probes([])

puts "Received #{@received_snapshot_count} snapshots, #{@received_snapshot_bytes} bytes total"
end

private

def probe_manager
Datadog::DI.component.probe_manager
end

def server
WEBrick::HTTPServer.new(
Port: 8126,
).tap do |server|
@received_snapshot_count = 0
@received_snapshot_bytes = 0

server.mount_proc('/debugger/v1/diagnostics') do |req, res|
# This request is a multipart form post
end

server.mount_proc('/debugger/v1/input') do |req, res|
payload = JSON.parse(req.body)
@received_snapshot_count += payload.length
@received_snapshot_bytes += req.body.length
end
end
end

attr_reader :received_snapshot_count
end

puts "Current pid is #{Process.pid}"

DISnapshotBenchmark.new.instance_exec do
run_benchmark
end
22 changes: 22 additions & 0 deletions benchmarks/support/di_snapshot_target.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
class DISnapshotTarget
def test_method
# Perform some work to take up time
SecureRandom.uuid

v1 = Datadog.configuration
v2 = Datadog.configuration
v3 = Datadog.configuration
v4 = Datadog.configuration
v5 = Datadog.configuration
v6 = Datadog.configuration
v7 = Datadog.configuration
v8 = Datadog.configuration
v9 = Datadog.configuration

# Currently Ruby DI does not implement capture of local variables
# in method probes, or instance variables.
# Return a complex object which will be serialized for the
# enriched probe.
Datadog.configuration # Line 20
end
end
12 changes: 11 additions & 1 deletion lib/datadog/di/configuration/settings.rb
Original file line number Diff line number Diff line change
Expand Up @@ -166,10 +166,20 @@ def self.add_settings!(base)
# being sent out by the probe notifier worker) and creates a
# possibility of dropping payloads if the queue gets too long.
option :min_send_interval do |o|
o.type :int
o.type :float
o.default 3
end

# Number of snapshots that can be stored in the probe
# notifier worker queue. Larger capacity runs the risk of
# creating snapshots that exceed the agent's request size
# limit. Smaller capacity increases the risk of dropping
# snapshots.
option :snapshot_queue_capacity do |o|
o.type :int
o.default 100
end

# Enable dynamic instrumentation in development environments.
# Currently DI does not fully implement support for code
# unloading and reloading, and is not supported in
Expand Down
3 changes: 1 addition & 2 deletions lib/datadog/di/probe_notifier_worker.rb
Original file line number Diff line number Diff line change
Expand Up @@ -183,8 +183,7 @@ def io_in_progress?
define_method("add_#{event_type}") do |event|
@lock.synchronize do
queue = send("#{event_type}_queue")
# TODO determine a suitable limit via testing/benchmarking
if queue.length > 100
if queue.length > settings.dynamic_instrumentation.internal.snapshot_queue_capacity
logger.warn("#{self.class.name}: dropping #{event_type} because queue is full")
else
queue << event
Expand Down
1 change: 1 addition & 0 deletions spec/datadog/di/probe_notifier_worker_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@
allow(settings.dynamic_instrumentation.internal).to receive(:propagate_all_exceptions).and_return(false)
# Reduce to 1 to have the test run faster
allow(settings.dynamic_instrumentation.internal).to receive(:min_send_interval).and_return(1)
allow(settings.dynamic_instrumentation.internal).to receive(:snapshot_queue_capacity).and_return(10)
end

let(:transport) do
Expand Down
17 changes: 14 additions & 3 deletions spec/datadog/di/validate_benchmarks_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -10,18 +10,29 @@
end

benchmarks_to_validate = [
"di_instrument",
"di_instrument", "di_snapshot",
].freeze

benchmarks_to_validate.each do |benchmark|
describe benchmark do
it("runs without raising errors") { expect_in_fork { load "./benchmarks/#{benchmark}.rb" } }
timeout = if benchmark == 'di_snapshot'
20
else
10
end
it "runs without raising errors" do
expect_in_fork(timeout_seconds: timeout) do
load "./benchmarks/#{benchmark}.rb"
end
end
end
end

# This test validates that we don't forget to add new benchmarks to benchmarks_to_validate
it "tests all expected benchmarks in the benchmarks folder" do
all_benchmarks = Dir["./benchmarks/di_*"].map { |it| it.gsub("./benchmarks/", "").gsub(".rb", "") }
all_benchmarks = Dir["./benchmarks/di_*"].map do |it|
it.gsub("./benchmarks/", "").gsub(".rb", "")
end

expect(benchmarks_to_validate).to contain_exactly(*all_benchmarks)
end
Expand Down

0 comments on commit 7fd1feb

Please sign in to comment.