From 96f1b48690ff90934116d60657e5c69de08a600e Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Wed, 2 Aug 2023 17:41:31 +0200 Subject: [PATCH 01/13] Start PropagationContext impl --- sentry-ruby/lib/sentry/propagation_context.rb | 14 ++++++++++++++ sentry-ruby/lib/sentry/scope.rb | 5 +++++ 2 files changed, 19 insertions(+) create mode 100644 sentry-ruby/lib/sentry/propagation_context.rb diff --git a/sentry-ruby/lib/sentry/propagation_context.rb b/sentry-ruby/lib/sentry/propagation_context.rb new file mode 100644 index 000000000..02b7765c7 --- /dev/null +++ b/sentry-ruby/lib/sentry/propagation_context.rb @@ -0,0 +1,14 @@ +# frozen_string_literal: true + +require "securerandom" + +module Sentry + class PropagationContext + def initialize + @trace_id = SecureRandom.uuid.delete("-") + @span_id = SecureRandom.uuid.delete("-").slice(0, 16) + @parent_span_id = nil + @dynamic_sampling_context = nil + end + end +end diff --git a/sentry-ruby/lib/sentry/scope.rb b/sentry-ruby/lib/sentry/scope.rb index 7f98cd079..4c7b2e442 100644 --- a/sentry-ruby/lib/sentry/scope.rb +++ b/sentry-ruby/lib/sentry/scope.rb @@ -292,6 +292,7 @@ def set_default_value @rack_env = {} @span = nil @session = nil + generate_propagation_context set_new_breadcrumb_buffer end @@ -299,6 +300,10 @@ def set_new_breadcrumb_buffer @breadcrumbs = BreadcrumbBuffer.new(@max_breadcrumbs) end + def generate_propagation_context + @propagation_context = PropagationContext.new + end + class << self # @return [Hash] def os_context From 3bfaad51359802f7a6ae0f4b8aaea23f74af59cb Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Wed, 2 Aug 2023 17:41:49 +0200 Subject: [PATCH 02/13] Make span_id generation consistent --- sentry-ruby/lib/sentry/span.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/sentry-ruby/lib/sentry/span.rb b/sentry-ruby/lib/sentry/span.rb index 8fda79fd8..95dd62efd 100644 --- a/sentry-ruby/lib/sentry/span.rb +++ b/sentry-ruby/lib/sentry/span.rb @@ -75,7 +75,7 @@ def initialize( timestamp: nil ) @trace_id = trace_id || SecureRandom.uuid.delete("-") - @span_id = span_id || SecureRandom.hex(8) + @span_id = span_id || SecureRandom.uuid.delete("-").slice(0, 16) @parent_span_id = parent_span_id @sampled = sampled @start_timestamp = start_timestamp || Sentry.utc_now.to_f From ccc6634999f303893c1ef58714cd19c0256e380c Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Wed, 2 Aug 2023 18:09:42 +0200 Subject: [PATCH 03/13] Add get_trace_context and apply in scope --- sentry-ruby/lib/sentry/propagation_context.rb | 10 ++++++++++ sentry-ruby/lib/sentry/scope.rb | 10 ++++++++-- 2 files changed, 18 insertions(+), 2 deletions(-) diff --git a/sentry-ruby/lib/sentry/propagation_context.rb b/sentry-ruby/lib/sentry/propagation_context.rb index 02b7765c7..c6e69e48e 100644 --- a/sentry-ruby/lib/sentry/propagation_context.rb +++ b/sentry-ruby/lib/sentry/propagation_context.rb @@ -10,5 +10,15 @@ def initialize @parent_span_id = nil @dynamic_sampling_context = nil end + + # Returns the trace context that can be used to embed in an Event. + # @return [Hash] + def get_trace_context + { + trace_id: @trace_id, + span_id: @span_id, + parent_span_id: @parent_span_id + } + end end end diff --git a/sentry-ruby/lib/sentry/scope.rb b/sentry-ruby/lib/sentry/scope.rb index 4c7b2e442..a7be87e9e 100644 --- a/sentry-ruby/lib/sentry/scope.rb +++ b/sentry-ruby/lib/sentry/scope.rb @@ -1,6 +1,7 @@ # frozen_string_literal: true require "sentry/breadcrumb_buffer" +require "sentry/propagation_context" require "etc" module Sentry @@ -20,7 +21,8 @@ class Scope :event_processors, :rack_env, :span, - :session + :session, + :propagation_context ] attr_reader(*ATTRIBUTES) @@ -50,7 +52,9 @@ def apply_to_event(event, hint = nil) event.transaction_info = { source: transaction_source } if transaction_source if span - event.contexts[:trace] = span.get_trace_context + event.contexts[:trace] ||= span.get_trace_context + else + event.contexts[:trace] ||= propagation_context.get_trace_context end event.fingerprint = fingerprint @@ -95,6 +99,7 @@ def dup copy.fingerprint = fingerprint.deep_dup copy.span = span.deep_dup copy.session = session.deep_dup + copy.propagation_context = propagation_context.deep_dup copy end @@ -111,6 +116,7 @@ def update_from_scope(scope) self.transaction_sources = scope.transaction_sources self.fingerprint = scope.fingerprint self.span = scope.span + self.propagation_context = scope.propagation_context end # Updates the scope's data from the given options. From 51a521e83471654c8e736e2b59a198b5432956c0 Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Thu, 3 Aug 2023 15:01:12 +0200 Subject: [PATCH 04/13] Move dynamic_sampling_context to Event from TransactionEvent --- sentry-ruby/lib/sentry/event.rb | 5 +++++ sentry-ruby/lib/sentry/transaction_event.rb | 3 --- 2 files changed, 5 insertions(+), 3 deletions(-) diff --git a/sentry-ruby/lib/sentry/event.rb b/sentry-ruby/lib/sentry/event.rb index 0a76f3f4a..fa66c3187 100644 --- a/sentry-ruby/lib/sentry/event.rb +++ b/sentry-ruby/lib/sentry/event.rb @@ -37,6 +37,11 @@ class Event # @return [RequestInterface] attr_reader :request + # Dynamic Sampling Context (DSC) that gets attached + # as the trace envelope header in the transport. + # @return [Hash, nil] + attr_accessor :dynamic_sampling_context + # @param configuration [Configuration] # @param integration_meta [Hash, nil] # @param message [String, nil] diff --git a/sentry-ruby/lib/sentry/transaction_event.rb b/sentry-ruby/lib/sentry/transaction_event.rb index 9b3f6909f..20d25445e 100644 --- a/sentry-ruby/lib/sentry/transaction_event.rb +++ b/sentry-ruby/lib/sentry/transaction_event.rb @@ -8,9 +8,6 @@ class TransactionEvent < Event # @return [] attr_accessor :spans - # @return [Hash, nil] - attr_accessor :dynamic_sampling_context - # @return [Hash] attr_accessor :measurements From 951033589994af033c42b45bf6b2d649d2c25a6f Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Thu, 3 Aug 2023 15:01:43 +0200 Subject: [PATCH 05/13] Implement traceparent/baggage/DSC for PropagationContext and apply in scope --- sentry-ruby/lib/sentry/propagation_context.rb | 64 +++++++++++++++++-- sentry-ruby/lib/sentry/scope.rb | 3 +- 2 files changed, 61 insertions(+), 6 deletions(-) diff --git a/sentry-ruby/lib/sentry/propagation_context.rb b/sentry-ruby/lib/sentry/propagation_context.rb index c6e69e48e..1211c760b 100644 --- a/sentry-ruby/lib/sentry/propagation_context.rb +++ b/sentry-ruby/lib/sentry/propagation_context.rb @@ -1,24 +1,78 @@ # frozen_string_literal: true require "securerandom" +require "sentry/baggage" module Sentry class PropagationContext - def initialize + + # An uuid that can be used to identify a trace. + # @return [String] + attr_reader :trace_id + # An uuid that can be used to identify the span. + # @return [String] + attr_reader :span_id + # Span parent's span_id. + # @return [String] + attr_reader :parent_span_id + + def initialize(scope) + @scope = scope @trace_id = SecureRandom.uuid.delete("-") @span_id = SecureRandom.uuid.delete("-").slice(0, 16) @parent_span_id = nil - @dynamic_sampling_context = nil + @baggage = nil end # Returns the trace context that can be used to embed in an Event. # @return [Hash] def get_trace_context { - trace_id: @trace_id, - span_id: @span_id, - parent_span_id: @parent_span_id + trace_id: trace_id, + span_id: span_id, + parent_span_id: parent_span_id + } + end + + # Returns the sentry-trace header from the propagation context. + # @return [String] + def get_traceparent + "#{trace_id}-#{span_id}" + end + + # Returns the W3C baggage header from the propagation context. + # @return [String, nil] + def get_baggage + populate_head_baggage if @baggage.nil? || @baggage.mutable + @baggage + end + + # Returns the Dynamic Sampling Context from the baggage. + # @return [String, nil] + def get_dynamic_sampling_context + get_baggage&.dynamic_sampling_context + end + + private + + def populate_head_baggage + return unless Sentry.initialized? + + configuration = Sentry.configuration + + items = { + "trace_id" => trace_id, + "sample_rate" => configuration.traces_sample_rate, + "environment" => configuration.environment, + "release" => configuration.release, + "public_key" => configuration.dsn&.public_key } + + user = @scope&.user + items["user_segment"] = user["segment"] if user && user["segment"] + + items.compact! + @baggage = Baggage.new(items, mutable: false) end end end diff --git a/sentry-ruby/lib/sentry/scope.rb b/sentry-ruby/lib/sentry/scope.rb index a7be87e9e..f0ea77982 100644 --- a/sentry-ruby/lib/sentry/scope.rb +++ b/sentry-ruby/lib/sentry/scope.rb @@ -55,6 +55,7 @@ def apply_to_event(event, hint = nil) event.contexts[:trace] ||= span.get_trace_context else event.contexts[:trace] ||= propagation_context.get_trace_context + event.dynamic_sampling_context ||= propagation_context.get_dynamic_sampling_context end event.fingerprint = fingerprint @@ -307,7 +308,7 @@ def set_new_breadcrumb_buffer end def generate_propagation_context - @propagation_context = PropagationContext.new + @propagation_context = PropagationContext.new(self) end class << self From d1454ecbebaf9bfb74301e48d8ec7d1255be8703 Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Fri, 4 Aug 2023 13:11:25 +0200 Subject: [PATCH 06/13] Top level get_traceparent and get_baggage methods --- sentry-ruby/lib/sentry-ruby.rb | 18 ++++++++++++++++++ sentry-ruby/lib/sentry/hub.rb | 14 ++++++++++++++ 2 files changed, 32 insertions(+) diff --git a/sentry-ruby/lib/sentry-ruby.rb b/sentry-ruby/lib/sentry-ruby.rb index 9efb6e6ec..7a56054ee 100644 --- a/sentry-ruby/lib/sentry-ruby.rb +++ b/sentry-ruby/lib/sentry-ruby.rb @@ -489,6 +489,24 @@ def add_global_event_processor(&block) Scope.add_global_event_processor(&block) end + # Returns the traceparent (sentry-trace) header for distributed tracing. + # Can be either from the currently active span or the propagation context. + # + # @return [String, nil] + def get_traceparent + return nil unless initialized? + get_current_hub.get_traceparent + end + + # Returns the baggage header for distributed tracing. + # Can be either from the currently active span or the propagation context. + # + # @return [String, nil] + def get_baggage + return nil unless initialized? + get_current_hub.get_baggage + end + ##### Helpers ##### # @!visibility private diff --git a/sentry-ruby/lib/sentry/hub.rb b/sentry-ruby/lib/sentry/hub.rb index 2b87b2694..0d9c02dba 100644 --- a/sentry-ruby/lib/sentry/hub.rb +++ b/sentry-ruby/lib/sentry/hub.rb @@ -229,6 +229,20 @@ def with_session_tracking(&block) end_session end + def get_traceparent + return nil unless current_scope + + current_scope.get_span&.to_sentry_trace || + current_scope.propagation_context&.get_traceparent + end + + def get_baggage + return nil unless current_scope + + current_scope.get_span&.to_baggage || + current_scope.propagation_context&.get_baggage&.serialize + end + private def current_layer From 8dadf5a0d1ac6a115cf12d37a77ae082bf2235a9 Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Wed, 16 Aug 2023 13:34:17 +0200 Subject: [PATCH 07/13] Add new unified get_trace_propagation_headers and use in net/http --- sentry-ruby/lib/sentry-ruby.rb | 9 +++++++++ sentry-ruby/lib/sentry/client.rb | 6 +++++- sentry-ruby/lib/sentry/hub.rb | 9 +++++++++ sentry-ruby/lib/sentry/net/http.rb | 12 +++--------- sentry-ruby/lib/sentry/propagation_context.rb | 2 +- sentry-ruby/spec/sentry/net/http_spec.rb | 19 +------------------ 6 files changed, 28 insertions(+), 29 deletions(-) diff --git a/sentry-ruby/lib/sentry-ruby.rb b/sentry-ruby/lib/sentry-ruby.rb index 7a56054ee..03edc0a08 100644 --- a/sentry-ruby/lib/sentry-ruby.rb +++ b/sentry-ruby/lib/sentry-ruby.rb @@ -507,6 +507,15 @@ def get_baggage get_current_hub.get_baggage end + # Returns the a Hash containing sentry-trace and baggage. + # Can be either from the currently active span or the propagation context. + # + # @return [Hash, nil] + def get_trace_propagation_headers + return nil unless initialized? + get_current_hub.get_trace_propagation_headers + end + ##### Helpers ##### # @!visibility private diff --git a/sentry-ruby/lib/sentry/client.rb b/sentry-ruby/lib/sentry/client.rb index 3d8c0dba6..80f1165c8 100644 --- a/sentry-ruby/lib/sentry/client.rb +++ b/sentry-ruby/lib/sentry/client.rb @@ -148,6 +148,8 @@ def send_event(event, hint = nil) raise end + # @deprecated use Sentry.get_traceparent instead. + # # Generates a Sentry trace for distribted tracing from the given Span. # Returns `nil` if `config.propagate_traces` is `false`. # @param span [Span] the span to generate trace from. @@ -160,7 +162,9 @@ def generate_sentry_trace(span) trace end - # Generates a W3C Baggage header for distribted tracing from the given Span. + # @deprecated Use Sentry.get_baggage instead. + # + # Generates a W3C Baggage header for distributed tracing from the given Span. # Returns `nil` if `config.propagate_traces` is `false`. # @param span [Span] the span to generate trace from. # @return [String, nil] diff --git a/sentry-ruby/lib/sentry/hub.rb b/sentry-ruby/lib/sentry/hub.rb index 0d9c02dba..2bece1b03 100644 --- a/sentry-ruby/lib/sentry/hub.rb +++ b/sentry-ruby/lib/sentry/hub.rb @@ -243,6 +243,15 @@ def get_baggage current_scope.propagation_context&.get_baggage&.serialize end + def get_trace_propagation_headers + return nil unless configuration.propagate_traces + + { + SENTRY_TRACE_HEADER_NAME => get_traceparent, + BAGGAGE_HEADER_NAME => get_baggage + }.compact + end + private def current_layer diff --git a/sentry-ruby/lib/sentry/net/http.rb b/sentry-ruby/lib/sentry/net/http.rb index 989d8f393..af6e12aa6 100644 --- a/sentry-ruby/lib/sentry/net/http.rb +++ b/sentry-ruby/lib/sentry/net/http.rb @@ -31,7 +31,7 @@ def request(req, body = nil, &block) Sentry.with_child_span(op: OP_NAME, start_timestamp: Sentry.utc_now.to_f) do |sentry_span| request_info = extract_request_info(req) - set_sentry_trace_header(req, sentry_span, request_info) + set_propagation_headers(req, request_info) super.tap do |res| record_sentry_breadcrumb(request_info, res) @@ -49,17 +49,11 @@ def request(req, body = nil, &block) private - def set_sentry_trace_header(req, sentry_span, request_info) - return unless sentry_span - + def set_propagation_headers(req, request_info) client = Sentry.get_current_client return unless propagate_trace?(request_info[:url], client.configuration.trace_propagation_targets) - trace = client.generate_sentry_trace(sentry_span) - req[SENTRY_TRACE_HEADER_NAME] = trace if trace - - baggage = client.generate_baggage(sentry_span) - req[BAGGAGE_HEADER_NAME] = baggage if baggage && !baggage.empty? + Sentry.get_trace_propagation_headers&.each { |k, v| req[k] = v } end def record_sentry_breadcrumb(request_info, res) diff --git a/sentry-ruby/lib/sentry/propagation_context.rb b/sentry-ruby/lib/sentry/propagation_context.rb index 1211c760b..1e99d2756 100644 --- a/sentry-ruby/lib/sentry/propagation_context.rb +++ b/sentry-ruby/lib/sentry/propagation_context.rb @@ -62,7 +62,7 @@ def populate_head_baggage items = { "trace_id" => trace_id, - "sample_rate" => configuration.traces_sample_rate, + "sample_rate" => configuration.traces_sample_rate&.to_s, "environment" => configuration.environment, "release" => configuration.release, "public_key" => configuration.dsn&.public_key diff --git a/sentry-ruby/spec/sentry/net/http_spec.rb b/sentry-ruby/spec/sentry/net/http_spec.rb index eaa79e955..21eb96f7b 100644 --- a/sentry-ruby/spec/sentry/net/http_spec.rb +++ b/sentry-ruby/spec/sentry/net/http_spec.rb @@ -97,9 +97,6 @@ response = http.request(request) expect(response.code).to eq("200") - expect(string_io.string).to match( - /\[Tracing\] Adding sentry-trace header to outgoing request:/ - ) end it "adds baggage header to the request header as head SDK when no incoming trace" do @@ -115,10 +112,6 @@ response = http.request(request) expect(response.code).to eq("200") - expect(string_io.string).to match( - /\[Tracing\] Adding baggage header to outgoing request:/ - ) - request_span = transaction.span_recorder.spans.last expect(request["baggage"]).to eq(request_span.to_baggage) expect(request["baggage"]).to eq( @@ -151,10 +144,6 @@ response = http.request(request) expect(response.code).to eq("200") - expect(string_io.string).to match( - /\[Tracing\] Adding baggage header to outgoing request:/ - ) - request_span = transaction.span_recorder.spans.last expect(request["baggage"]).to eq(request_span.to_baggage) expect(request["baggage"]).to eq( @@ -165,7 +154,7 @@ ) end - context "with config.propagate_trace = false" do + context "with config.propagate_traces = false" do before do Sentry.configuration.propagate_traces = false end @@ -183,9 +172,6 @@ response = http.request(request) expect(response.code).to eq("200") - expect(string_io.string).not_to match( - /Adding sentry-trace header to outgoing request:/ - ) expect(request.key?("sentry-trace")).to eq(false) end @@ -210,9 +196,6 @@ response = http.request(request) expect(response.code).to eq("200") - expect(string_io.string).not_to match( - /Adding baggage header to outgoing request:/ - ) expect(request.key?("baggage")).to eq(false) end end From c8b8c4c89ef73229311767dacb38b25dd2881d0f Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Wed, 16 Aug 2023 14:02:11 +0200 Subject: [PATCH 08/13] Changelog --- CHANGELOG.md | 15 +++++++++++++++ sentry-ruby/lib/sentry/hub.rb | 13 +++++++++---- sentry-ruby/lib/sentry/net/http.rb | 10 +++++----- 3 files changed, 29 insertions(+), 9 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index f3e4cce68..3fd1a7f6a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -11,6 +11,21 @@ config.trace_propagation_targets = [/.*/] # default is to all targets config.trace_propagation_targets = [/example.com/, 'foobar.org/api/v2'] ``` +- Tracing without Performance + - Implement `PropagationContext` on `Scope` and add `Sentry.get_trace_propagation_headers` API [#2084](https://github.com/getsentry/sentry-ruby/pull/2084) + + The SDK now supports connecting arbitrary events (Errors / Transactions / Replays) across distributed services and not just Transactions. + To continue an incoming trace starting with this version of the SDK, use `Sentry.continue_trace` as follows. + + ```rb + # rack application + def call(env) + transaction = Sentry.continue_trace(env, name: 'transaction', op: 'op') + Sentry.start_transaction(transaction: transaction) + end + ``` + + To inject headers into outgoing requests, use `Sentry.get_trace_propagation_headers` to get a hash of headers to add to your request. ### Bug Fixes diff --git a/sentry-ruby/lib/sentry/hub.rb b/sentry-ruby/lib/sentry/hub.rb index 2bece1b03..48a131cd9 100644 --- a/sentry-ruby/lib/sentry/hub.rb +++ b/sentry-ruby/lib/sentry/hub.rb @@ -246,10 +246,15 @@ def get_baggage def get_trace_propagation_headers return nil unless configuration.propagate_traces - { - SENTRY_TRACE_HEADER_NAME => get_traceparent, - BAGGAGE_HEADER_NAME => get_baggage - }.compact + headers = {} + + traceparent = get_traceparent + headers[SENTRY_TRACE_HEADER_NAME] = traceparent if traceparent + + baggage = get_baggage + headers[BAGGAGE_HEADER_NAME] = baggage if baggage && !baggage.empty? + + headers end private diff --git a/sentry-ruby/lib/sentry/net/http.rb b/sentry-ruby/lib/sentry/net/http.rb index af6e12aa6..04eedb290 100644 --- a/sentry-ruby/lib/sentry/net/http.rb +++ b/sentry-ruby/lib/sentry/net/http.rb @@ -31,7 +31,10 @@ def request(req, body = nil, &block) Sentry.with_child_span(op: OP_NAME, start_timestamp: Sentry.utc_now.to_f) do |sentry_span| request_info = extract_request_info(req) - set_propagation_headers(req, request_info) + + if propagate_trace?(request_info[:url], Sentry.configuration.trace_propagation_targets) + set_propagation_headers(req) + end super.tap do |res| record_sentry_breadcrumb(request_info, res) @@ -49,10 +52,7 @@ def request(req, body = nil, &block) private - def set_propagation_headers(req, request_info) - client = Sentry.get_current_client - return unless propagate_trace?(request_info[:url], client.configuration.trace_propagation_targets) - + def set_propagation_headers(req) Sentry.get_trace_propagation_headers&.each { |k, v| req[k] = v } end From 0cf49960b3dc8f6c5b896fee996e21bae1b8e898 Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Mon, 28 Aug 2023 17:46:28 +0200 Subject: [PATCH 09/13] Review fixes --- sentry-ruby/lib/sentry/event.rb | 1 + sentry-ruby/lib/sentry/hub.rb | 4 ++-- sentry-ruby/lib/sentry/propagation_context.rb | 6 ++---- 3 files changed, 5 insertions(+), 6 deletions(-) diff --git a/sentry-ruby/lib/sentry/event.rb b/sentry-ruby/lib/sentry/event.rb index fa66c3187..b9f8a58e6 100644 --- a/sentry-ruby/lib/sentry/event.rb +++ b/sentry-ruby/lib/sentry/event.rb @@ -59,6 +59,7 @@ def initialize(configuration:, integration_meta: nil, message: nil) @tags = {} @fingerprint = [] + @dynamic_sampling_context = nil # configuration data that's directly used by events @server_name = configuration.server_name diff --git a/sentry-ruby/lib/sentry/hub.rb b/sentry-ruby/lib/sentry/hub.rb index 48a131cd9..0efd00b3c 100644 --- a/sentry-ruby/lib/sentry/hub.rb +++ b/sentry-ruby/lib/sentry/hub.rb @@ -233,14 +233,14 @@ def get_traceparent return nil unless current_scope current_scope.get_span&.to_sentry_trace || - current_scope.propagation_context&.get_traceparent + current_scope.propagation_context.get_traceparent end def get_baggage return nil unless current_scope current_scope.get_span&.to_baggage || - current_scope.propagation_context&.get_baggage&.serialize + current_scope.propagation_context.get_baggage&.serialize end def get_trace_propagation_headers diff --git a/sentry-ruby/lib/sentry/propagation_context.rb b/sentry-ruby/lib/sentry/propagation_context.rb index 1e99d2756..9bf163b72 100644 --- a/sentry-ruby/lib/sentry/propagation_context.rb +++ b/sentry-ruby/lib/sentry/propagation_context.rb @@ -65,12 +65,10 @@ def populate_head_baggage "sample_rate" => configuration.traces_sample_rate&.to_s, "environment" => configuration.environment, "release" => configuration.release, - "public_key" => configuration.dsn&.public_key + "public_key" => configuration.dsn&.public_key, + "user_segment" => @scope.user && @scope.user["segment"] } - user = @scope&.user - items["user_segment"] = user["segment"] if user && user["segment"] - items.compact! @baggage = Baggage.new(items, mutable: false) end From 94ce420ce59211d8a195108368d6c29b92c9b19e Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Tue, 29 Aug 2023 14:04:19 +0200 Subject: [PATCH 10/13] Specs --- sentry-ruby/lib/sentry/hub.rb | 2 - sentry-ruby/lib/sentry/net/http.rb | 8 ++- sentry-ruby/spec/sentry/event_spec.rb | 1 + sentry-ruby/spec/sentry/net/http_spec.rb | 15 ++++- .../spec/sentry/propagation_context_spec.rb | 63 +++++++++++++++++++ sentry-ruby/spec/sentry/scope_spec.rb | 9 ++- sentry-ruby/spec/sentry_spec.rb | 49 +++++++++++++++ 7 files changed, 140 insertions(+), 7 deletions(-) create mode 100644 sentry-ruby/spec/sentry/propagation_context_spec.rb diff --git a/sentry-ruby/lib/sentry/hub.rb b/sentry-ruby/lib/sentry/hub.rb index 0efd00b3c..c9f95eefe 100644 --- a/sentry-ruby/lib/sentry/hub.rb +++ b/sentry-ruby/lib/sentry/hub.rb @@ -244,8 +244,6 @@ def get_baggage end def get_trace_propagation_headers - return nil unless configuration.propagate_traces - headers = {} traceparent = get_traceparent diff --git a/sentry-ruby/lib/sentry/net/http.rb b/sentry-ruby/lib/sentry/net/http.rb index 04eedb290..4b0345f52 100644 --- a/sentry-ruby/lib/sentry/net/http.rb +++ b/sentry-ruby/lib/sentry/net/http.rb @@ -32,7 +32,7 @@ def request(req, body = nil, &block) Sentry.with_child_span(op: OP_NAME, start_timestamp: Sentry.utc_now.to_f) do |sentry_span| request_info = extract_request_info(req) - if propagate_trace?(request_info[:url], Sentry.configuration.trace_propagation_targets) + if propagate_trace?(request_info[:url], Sentry.configuration) set_propagation_headers(req) end @@ -90,8 +90,10 @@ def extract_request_info(req) result end - def propagate_trace?(url, trace_propagation_targets) - url && trace_propagation_targets.any? { |target| url.match?(target) } + def propagate_trace?(url, configuration) + url && + configuration.propagate_traces && + configuration.trace_propagation_targets.any? { |target| url.match?(target) } end end end diff --git a/sentry-ruby/spec/sentry/event_spec.rb b/sentry-ruby/spec/sentry/event_spec.rb index cfa19b259..10c4b448c 100644 --- a/sentry-ruby/spec/sentry/event_spec.rb +++ b/sentry-ruby/spec/sentry/event_spec.rb @@ -30,6 +30,7 @@ expect(event.environment).to eq("test") expect(event.release).to eq("721e41770371db95eee98ca2707686226b993eda") expect(event.sdk).to eq("name" => "sentry.ruby", "version" => Sentry::VERSION) + expect(event.dynamic_sampling_context).to eq(nil) end end diff --git a/sentry-ruby/spec/sentry/net/http_spec.rb b/sentry-ruby/spec/sentry/net/http_spec.rb index 21eb96f7b..1e96f7feb 100644 --- a/sentry-ruby/spec/sentry/net/http_spec.rb +++ b/sentry-ruby/spec/sentry/net/http_spec.rb @@ -373,7 +373,20 @@ def verify_spans(transaction) perform_basic_setup end - it "doesn't affect the HTTP lib anything" do + it "adds sentry-trace and baggage headers for tracing without performance" do + stub_normal_response + + uri = URI("http://example.com/path") + http = Net::HTTP.new(uri.host, uri.port) + request = Net::HTTP::Get.new(uri.request_uri) + response = http.request(request) + + expect(request["sentry-trace"]).to eq(Sentry.get_traceparent) + expect(request["baggage"]).to eq(Sentry.get_baggage) + expect(response.code).to eq("200") + end + + it "doesn't create transaction or breadcrumbs" do stub_normal_response response = Net::HTTP.get_response(URI("http://example.com/path")) diff --git a/sentry-ruby/spec/sentry/propagation_context_spec.rb b/sentry-ruby/spec/sentry/propagation_context_spec.rb new file mode 100644 index 000000000..ce2c337b2 --- /dev/null +++ b/sentry-ruby/spec/sentry/propagation_context_spec.rb @@ -0,0 +1,63 @@ +require "spec_helper" + +RSpec.describe Sentry::PropagationContext do + before do + perform_basic_setup + end + + let(:scope) { Sentry.get_current_scope } + let(:subject) { described_class.new(scope) } + + describe "#initialize" do + it "generates correct attributes" do + expect(subject.trace_id.length).to eq(32) + expect(subject.span_id.length).to eq(16) + expect(subject.parent_span_id).to be_nil + end + end + + describe "#get_trace_context" do + it "generates correct trace context" do + expect(subject.get_trace_context).to eq({ + trace_id: subject.trace_id, + span_id: subject.span_id, + parent_span_id: subject.parent_span_id + }) + end + end + + describe "#get_traceparent" do + it "generates correct traceparent" do + expect(subject.get_traceparent).to eq("#{subject.trace_id}-#{subject.span_id}") + end + end + + describe "#get_baggage" do + before do + perform_basic_setup do |config| + config.environment = "test" + config.release = "foobar" + config.traces_sample_rate = 0.5 + end + end + + it "populates head baggage" do + baggage = subject.get_baggage + + expect(baggage.mutable).to eq(false) + expect(baggage.items).to eq({ + "trace_id" => subject.trace_id, + "sample_rate" => "0.5", + "environment" => "test", + "release" => "foobar", + "public_key" => Sentry.configuration.dsn.public_key + }) + end + end + + describe "#get_dynamic_sampling_context" do + it "generates DSC from baggage" do + expect(subject.get_dynamic_sampling_context).to eq(subject.get_baggage.dynamic_sampling_context) + end + end +end diff --git a/sentry-ruby/spec/sentry/scope_spec.rb b/sentry-ruby/spec/sentry/scope_spec.rb index 1e6d92426..3573e3a6e 100644 --- a/sentry-ruby/spec/sentry/scope_spec.rb +++ b/sentry-ruby/spec/sentry/scope_spec.rb @@ -24,6 +24,7 @@ expect(subject.fingerprint).to eq([]) expect(subject.transaction_names).to eq([]) expect(subject.transaction_sources).to eq([]) + expect(subject.propagation_context).to be_a(Sentry::PropagationContext) end it "allows setting breadcrumb buffer's size limit" do @@ -276,7 +277,7 @@ end end - it "sets trace context if there's a span" do + it "sets trace context from span if there's a span" do transaction = Sentry::Transaction.new(op: "foo", hub: hub) subject.set_span(transaction) @@ -286,6 +287,12 @@ expect(event.contexts.dig(:trace, :op)).to eq("foo") end + it "sets trace context and dynamic_sampling_context from propagation context if there's no span" do + subject.apply_to_event(event) + expect(event.contexts[:trace]).to eq(subject.propagation_context.get_trace_context) + expect(event.dynamic_sampling_context).to eq(subject.propagation_context.get_dynamic_sampling_context) + end + context "with Rack", rack: true do let(:env) do Rack::MockRequest.env_for("/test", {}) diff --git a/sentry-ruby/spec/sentry_spec.rb b/sentry-ruby/spec/sentry_spec.rb index 72fb18970..beb931131 100644 --- a/sentry-ruby/spec/sentry_spec.rb +++ b/sentry-ruby/spec/sentry_spec.rb @@ -667,6 +667,55 @@ end end + describe ".get_traceparent" do + it "returns a valid traceparent header from scope propagation context" do + traceparent = described_class.get_traceparent + propagation_context = described_class.get_current_scope.propagation_context + + expect(traceparent).to match(Sentry::Transaction::SENTRY_TRACE_REGEXP) + expect(traceparent).to eq("#{propagation_context.trace_id}-#{propagation_context.span_id}") + end + + it "returns a valid traceparent header from scope current span" do + transaction = Sentry::Transaction.new(op: "foo", hub: Sentry.get_current_hub, sampled: true) + span = transaction.start_child(op: "parent") + described_class.get_current_scope.set_span(span) + + traceparent = described_class.get_traceparent + + expect(traceparent).to match(Sentry::Transaction::SENTRY_TRACE_REGEXP) + expect(traceparent).to eq("#{span.trace_id}-#{span.span_id}-1") + end + end + + describe ".get_baggage" do + it "returns a valid baggage header from scope propagation context" do + baggage = described_class.get_baggage + propagation_context = described_class.get_current_scope.propagation_context + + expect(baggage).to eq("sentry-trace_id=#{propagation_context.trace_id},sentry-environment=development,sentry-public_key=12345") + end + + it "returns a valid baggage header from scope current span" do + transaction = Sentry::Transaction.new(op: "foo", hub: Sentry.get_current_hub, sampled: true) + span = transaction.start_child(op: "parent") + described_class.get_current_scope.set_span(span) + + baggage = described_class.get_baggage + + expect(baggage).to eq("sentry-trace_id=#{span.trace_id},sentry-sampled=true,sentry-environment=development,sentry-public_key=12345") + end + end + + describe ".get_trace_propagation_headers" do + it "returns a Hash of sentry-trace and baggage" do + expect(described_class.get_trace_propagation_headers).to eq({ + "sentry-trace" => described_class.get_traceparent, + "baggage" => described_class.get_baggage + }) + end + end + describe 'release detection' do let(:fake_root) { "/tmp/sentry/" } From 143bdcec26b0d5e8c448168c7a736a6591288623 Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Thu, 31 Aug 2023 13:26:35 +0200 Subject: [PATCH 11/13] Tracing without performance: new continue_trace api (#2089) --- CHANGELOG.md | 1 + sentry-rails/lib/sentry/rails/action_cable.rb | 5 +- .../lib/sentry/rails/capture_exceptions.rb | 5 +- sentry-ruby/lib/sentry-ruby.rb | 9 +++ sentry-ruby/lib/sentry/hub.rb | 18 +++++ sentry-ruby/lib/sentry/propagation_context.rb | 71 ++++++++++++++++-- .../lib/sentry/rack/capture_exceptions.rb | 5 +- sentry-ruby/lib/sentry/scope.rb | 11 ++- sentry-ruby/lib/sentry/transaction.rb | 25 ++----- sentry-ruby/spec/sentry/net/http_spec.rb | 4 +- .../spec/sentry/propagation_context_spec.rb | 75 ++++++++++++++++++- .../sentry/rack/capture_exceptions_spec.rb | 26 +++++++ sentry-ruby/spec/sentry/scope_spec.rb | 13 ++++ sentry-ruby/spec/sentry/span_spec.rb | 4 +- sentry-ruby/spec/sentry_spec.rb | 73 ++++++++++++++++-- .../sidekiq/sentry_context_middleware.rb | 9 +-- .../sidekiq/sentry_context_middleware_spec.rb | 14 ++-- 17 files changed, 308 insertions(+), 60 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 3fd1a7f6a..40b8fe6e9 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -13,6 +13,7 @@ ``` - Tracing without Performance - Implement `PropagationContext` on `Scope` and add `Sentry.get_trace_propagation_headers` API [#2084](https://github.com/getsentry/sentry-ruby/pull/2084) + - Implement `Sentry.continue_trace` API [#2089](https://github.com/getsentry/sentry-ruby/pull/2089) The SDK now supports connecting arbitrary events (Errors / Transactions / Replays) across distributed services and not just Transactions. To continue an incoming trace starting with this version of the SDK, use `Sentry.continue_trace` as follows. diff --git a/sentry-rails/lib/sentry/rails/action_cable.rb b/sentry-rails/lib/sentry/rails/action_cable.rb index 5bf01f5da..155377466 100644 --- a/sentry-rails/lib/sentry/rails/action_cable.rb +++ b/sentry-rails/lib/sentry/rails/action_cable.rb @@ -33,11 +33,8 @@ def capture(connection, transaction_name:, extra_context: nil, &block) end def start_transaction(env, scope) - sentry_trace = env["HTTP_SENTRY_TRACE"] - baggage = env["HTTP_BAGGAGE"] - options = { name: scope.transaction_name, source: scope.transaction_source, op: OP_NAME } - transaction = Sentry::Transaction.from_sentry_trace(sentry_trace, baggage: baggage, **options) if sentry_trace + transaction = Sentry.continue_trace(env, **options) Sentry.start_transaction(transaction: transaction, **options) end diff --git a/sentry-rails/lib/sentry/rails/capture_exceptions.rb b/sentry-rails/lib/sentry/rails/capture_exceptions.rb index a145d53dd..8d93784ed 100644 --- a/sentry-rails/lib/sentry/rails/capture_exceptions.rb +++ b/sentry-rails/lib/sentry/rails/capture_exceptions.rb @@ -32,16 +32,13 @@ def capture_exception(exception, env) end def start_transaction(env, scope) - sentry_trace = env["HTTP_SENTRY_TRACE"] - baggage = env["HTTP_BAGGAGE"] - options = { name: scope.transaction_name, source: scope.transaction_source, op: transaction_op } if @assets_regexp && scope.transaction_name.match?(@assets_regexp) options.merge!(sampled: false) end - transaction = Sentry::Transaction.from_sentry_trace(sentry_trace, baggage: baggage, **options) if sentry_trace + transaction = Sentry.continue_trace(env, **options) Sentry.start_transaction(transaction: transaction, custom_sampling_context: { env: env }, **options) end diff --git a/sentry-ruby/lib/sentry-ruby.rb b/sentry-ruby/lib/sentry-ruby.rb index 03edc0a08..daed11803 100644 --- a/sentry-ruby/lib/sentry-ruby.rb +++ b/sentry-ruby/lib/sentry-ruby.rb @@ -516,6 +516,15 @@ def get_trace_propagation_headers get_current_hub.get_trace_propagation_headers end + # Continue an incoming trace from a rack env like hash. + # + # @param env [Hash] + # @return [Transaction, nil] + def continue_trace(env, **options) + return nil unless initialized? + get_current_hub.continue_trace(env, **options) + end + ##### Helpers ##### # @!visibility private diff --git a/sentry-ruby/lib/sentry/hub.rb b/sentry-ruby/lib/sentry/hub.rb index c9f95eefe..7d9dca4c7 100644 --- a/sentry-ruby/lib/sentry/hub.rb +++ b/sentry-ruby/lib/sentry/hub.rb @@ -255,6 +255,24 @@ def get_trace_propagation_headers headers end + def continue_trace(env, **options) + configure_scope { |s| s.generate_propagation_context(env) } + + return nil unless configuration.tracing_enabled? + + propagation_context = current_scope.propagation_context + return nil unless propagation_context.incoming_trace + + Transaction.new( + hub: self, + trace_id: propagation_context.trace_id, + parent_span_id: propagation_context.parent_span_id, + parent_sampled: propagation_context.parent_sampled, + baggage: propagation_context.baggage, + **options + ) + end + private def current_layer diff --git a/sentry-ruby/lib/sentry/propagation_context.rb b/sentry-ruby/lib/sentry/propagation_context.rb index 9bf163b72..c76e73e90 100644 --- a/sentry-ruby/lib/sentry/propagation_context.rb +++ b/sentry-ruby/lib/sentry/propagation_context.rb @@ -5,6 +5,13 @@ module Sentry class PropagationContext + SENTRY_TRACE_REGEXP = Regexp.new( + "^[ \t]*" + # whitespace + "([0-9a-f]{32})?" + # trace_id + "-?([0-9a-f]{16})?" + # span_id + "-?([01])?" + # sampled + "[ \t]*$" # whitespace + ) # An uuid that can be used to identify a trace. # @return [String] @@ -13,15 +20,67 @@ class PropagationContext # @return [String] attr_reader :span_id # Span parent's span_id. - # @return [String] + # @return [String, nil] attr_reader :parent_span_id + # The sampling decision of the parent transaction. + # @return [Boolean, nil] + attr_reader :parent_sampled + # Is there an incoming trace or not? + # @return [Boolean] + attr_reader :incoming_trace + # This is only for accessing the current baggage variable. + # Please use the #get_baggage method for interfacing outside this class. + # @return [Baggage, nil] + attr_reader :baggage - def initialize(scope) + def initialize(scope, env = nil) @scope = scope - @trace_id = SecureRandom.uuid.delete("-") - @span_id = SecureRandom.uuid.delete("-").slice(0, 16) @parent_span_id = nil + @parent_sampled = nil @baggage = nil + @incoming_trace = false + + if env + sentry_trace_header = env["HTTP_SENTRY_TRACE"] || env[SENTRY_TRACE_HEADER_NAME] + baggage_header = env["HTTP_BAGGAGE"] || env[BAGGAGE_HEADER_NAME] + + if sentry_trace_header + sentry_trace_data = self.class.extract_sentry_trace(sentry_trace_header) + + if sentry_trace_data + @trace_id, @parent_span_id, @parent_sampled = sentry_trace_data + + @baggage = if baggage_header && !baggage_header.empty? + Baggage.from_incoming_header(baggage_header) + else + # If there's an incoming sentry-trace but no incoming baggage header, + # for instance in traces coming from older SDKs, + # baggage will be empty and frozen and won't be populated as head SDK. + Baggage.new({}) + end + + @baggage.freeze! + @incoming_trace = true + end + end + end + + @trace_id ||= SecureRandom.uuid.delete("-") + @span_id = SecureRandom.uuid.delete("-").slice(0, 16) + end + + # Extract the trace_id, parent_span_id and parent_sampled values from a sentry-trace header. + # + # @param sentry_trace [String] the sentry-trace header value from the previous transaction. + # @return [Array, nil] + def self.extract_sentry_trace(sentry_trace) + match = SENTRY_TRACE_REGEXP.match(sentry_trace) + return nil if match.nil? + + trace_id, parent_span_id, sampled_flag = match[1..3] + parent_sampled = sampled_flag.nil? ? nil : sampled_flag != "0" + + [trace_id, parent_span_id, parent_sampled] end # Returns the trace context that can be used to embed in an Event. @@ -40,8 +99,8 @@ def get_traceparent "#{trace_id}-#{span_id}" end - # Returns the W3C baggage header from the propagation context. - # @return [String, nil] + # Returns the Baggage from the propagation context or populates as head SDK if empty. + # @return [Baggage, nil] def get_baggage populate_head_baggage if @baggage.nil? || @baggage.mutable @baggage diff --git a/sentry-ruby/lib/sentry/rack/capture_exceptions.rb b/sentry-ruby/lib/sentry/rack/capture_exceptions.rb index 36c8b1848..2248799a3 100644 --- a/sentry-ruby/lib/sentry/rack/capture_exceptions.rb +++ b/sentry-ruby/lib/sentry/rack/capture_exceptions.rb @@ -62,11 +62,8 @@ def capture_exception(exception, env) end def start_transaction(env, scope) - sentry_trace = env["HTTP_SENTRY_TRACE"] - baggage = env["HTTP_BAGGAGE"] - options = { name: scope.transaction_name, source: scope.transaction_source, op: transaction_op } - transaction = Sentry::Transaction.from_sentry_trace(sentry_trace, baggage: baggage, **options) if sentry_trace + transaction = Sentry.continue_trace(env, **options) Sentry.start_transaction(transaction: transaction, custom_sampling_context: { env: env }, **options) end diff --git a/sentry-ruby/lib/sentry/scope.rb b/sentry-ruby/lib/sentry/scope.rb index f0ea77982..afbc03ea0 100644 --- a/sentry-ruby/lib/sentry/scope.rb +++ b/sentry-ruby/lib/sentry/scope.rb @@ -279,6 +279,13 @@ def add_event_processor(&block) @event_processors << block end + # Generate a new propagation context either from the incoming env headers or from scratch. + # @param env [Hash, nil] + # @return [void] + def generate_propagation_context(env = nil) + @propagation_context = PropagationContext.new(self, env) + end + protected # for duplicating scopes internally @@ -307,10 +314,6 @@ def set_new_breadcrumb_buffer @breadcrumbs = BreadcrumbBuffer.new(@max_breadcrumbs) end - def generate_propagation_context - @propagation_context = PropagationContext.new(self) - end - class << self # @return [Hash] def os_context diff --git a/sentry-ruby/lib/sentry/transaction.rb b/sentry-ruby/lib/sentry/transaction.rb index 0dce5dcce..b154f71fb 100644 --- a/sentry-ruby/lib/sentry/transaction.rb +++ b/sentry-ruby/lib/sentry/transaction.rb @@ -2,16 +2,13 @@ require "sentry/baggage" require "sentry/profiler" +require "sentry/propagation_context" module Sentry class Transaction < Span - SENTRY_TRACE_REGEXP = Regexp.new( - "^[ \t]*" + # whitespace - "([0-9a-f]{32})?" + # trace_id - "-?([0-9a-f]{16})?" + # span_id - "-?([01])?" + # sampled - "[ \t]*$" # whitespace - ) + # @deprecated Use Sentry::PropagationContext::SENTRY_TRACE_REGEXP instead. + SENTRY_TRACE_REGEXP = PropagationContext::SENTRY_TRACE_REGEXP + UNLABELD_NAME = "".freeze MESSAGE_PREFIX = "[Tracing]" @@ -92,6 +89,8 @@ def initialize( init_span_recorder end + # @deprecated use Sentry.continue_trace instead. + # # Initalizes a Transaction instance with a Sentry trace string from another transaction (usually from an external request). # # The original transaction will become the parent of the new Transaction instance. And they will share the same `trace_id`. @@ -132,18 +131,10 @@ def self.from_sentry_trace(sentry_trace, baggage: nil, hub: Sentry.get_current_h ) end - # Extract the trace_id, parent_span_id and parent_sampled values from a sentry-trace header. - # - # @param sentry_trace [String] the sentry-trace header value from the previous transaction. + # @deprecated Use Sentry::PropagationContext.extract_sentry_trace instead. # @return [Array, nil] def self.extract_sentry_trace(sentry_trace) - match = SENTRY_TRACE_REGEXP.match(sentry_trace) - return nil if match.nil? - - trace_id, parent_span_id, sampled_flag = match[1..3] - parent_sampled = sampled_flag.nil? ? nil : sampled_flag != "0" - - [trace_id, parent_span_id, parent_sampled] + PropagationContext.extract_sentry_trace(sentry_trace) end # @return [Hash] diff --git a/sentry-ruby/spec/sentry/net/http_spec.rb b/sentry-ruby/spec/sentry/net/http_spec.rb index 1e96f7feb..49c628a22 100644 --- a/sentry-ruby/spec/sentry/net/http_spec.rb +++ b/sentry-ruby/spec/sentry/net/http_spec.rb @@ -138,7 +138,7 @@ "sentry-user_id=Am%C3%A9lie, "\ "other-vendor-value-2=foo;bar;" - transaction = Sentry::Transaction.from_sentry_trace(sentry_trace, baggage: baggage) + transaction = Sentry.continue_trace({ "sentry-trace" => sentry_trace, "baggage" => baggage }) Sentry.get_current_scope.set_span(transaction) response = http.request(request) @@ -190,7 +190,7 @@ "sentry-user_id=Am%C3%A9lie, "\ "other-vendor-value-2=foo;bar;" - transaction = Sentry::Transaction.from_sentry_trace(sentry_trace, baggage: baggage) + transaction = Sentry.continue_trace({ "sentry-trace" => sentry_trace, "baggage" => baggage }) Sentry.get_current_scope.set_span(transaction) response = http.request(request) diff --git a/sentry-ruby/spec/sentry/propagation_context_spec.rb b/sentry-ruby/spec/sentry/propagation_context_spec.rb index ce2c337b2..60728bc11 100644 --- a/sentry-ruby/spec/sentry/propagation_context_spec.rb +++ b/sentry-ruby/spec/sentry/propagation_context_spec.rb @@ -9,10 +9,83 @@ let(:subject) { described_class.new(scope) } describe "#initialize" do - it "generates correct attributes" do + it "generates correct attributes without env" do expect(subject.trace_id.length).to eq(32) expect(subject.span_id.length).to eq(16) expect(subject.parent_span_id).to be_nil + expect(subject.parent_sampled).to be_nil + expect(subject.baggage).to be_nil + expect(subject.incoming_trace).to eq(false) + end + + it "generates correct attributes when incoming sentry-trace and baggage" do + env = { + "sentry-trace" => "771a43a4192642f0b136d5159a501700-7c51afd529da4a2a", + "baggage" => "other-vendor-value-1=foo;bar;baz, "\ + "sentry-trace_id=771a43a4192642f0b136d5159a501700, "\ + "sentry-public_key=49d0f7386ad645858ae85020e393bef3, "\ + "sentry-sample_rate=0.01337, "\ + "sentry-user_id=Am%C3%A9lie, "\ + "other-vendor-value-2=foo;bar;" + } + + subject = described_class.new(scope, env) + expect(subject.trace_id).to eq("771a43a4192642f0b136d5159a501700") + expect(subject.span_id.length).to eq(16) + expect(subject.parent_span_id).to eq("7c51afd529da4a2a") + expect(subject.parent_sampled).to eq(nil) + expect(subject.incoming_trace).to eq(true) + expect(subject.baggage).to be_a(Sentry::Baggage) + expect(subject.baggage.mutable).to eq(false) + expect(subject.baggage.items).to eq({ + "public_key"=>"49d0f7386ad645858ae85020e393bef3", + "sample_rate"=>"0.01337", + "trace_id"=>"771a43a4192642f0b136d5159a501700", + "user_id"=>"Amélie" + }) + end + + it "generates correct attributes when incoming HTTP_SENTRY_TRACE and HTTP_BAGGAGE" do + env = { + "HTTP_SENTRY_TRACE" => "771a43a4192642f0b136d5159a501700-7c51afd529da4a2a", + "HTTP_BAGGAGE" => "other-vendor-value-1=foo;bar;baz, "\ + "sentry-trace_id=771a43a4192642f0b136d5159a501700, "\ + "sentry-public_key=49d0f7386ad645858ae85020e393bef3, "\ + "sentry-sample_rate=0.01337, "\ + "sentry-user_id=Am%C3%A9lie, "\ + "other-vendor-value-2=foo;bar;" + } + + subject = described_class.new(scope, env) + expect(subject.trace_id).to eq("771a43a4192642f0b136d5159a501700") + expect(subject.span_id.length).to eq(16) + expect(subject.parent_span_id).to eq("7c51afd529da4a2a") + expect(subject.parent_sampled).to eq(nil) + expect(subject.incoming_trace).to eq(true) + expect(subject.baggage).to be_a(Sentry::Baggage) + expect(subject.baggage.mutable).to eq(false) + expect(subject.baggage.items).to eq({ + "public_key"=>"49d0f7386ad645858ae85020e393bef3", + "sample_rate"=>"0.01337", + "trace_id"=>"771a43a4192642f0b136d5159a501700", + "user_id"=>"Amélie" + }) + end + + it "generates correct attributes when incoming sentry-trace only (from older SDKs)" do + env = { + "sentry-trace" => "771a43a4192642f0b136d5159a501700-7c51afd529da4a2a" + } + + subject = described_class.new(scope, env) + expect(subject.trace_id).to eq("771a43a4192642f0b136d5159a501700") + expect(subject.span_id.length).to eq(16) + expect(subject.parent_span_id).to eq("7c51afd529da4a2a") + expect(subject.parent_sampled).to eq(nil) + expect(subject.incoming_trace).to eq(true) + expect(subject.baggage).to be_a(Sentry::Baggage) + expect(subject.baggage.mutable).to eq(false) + expect(subject.baggage.items).to eq({}) end end diff --git a/sentry-ruby/spec/sentry/rack/capture_exceptions_spec.rb b/sentry-ruby/spec/sentry/rack/capture_exceptions_spec.rb index 7429b2d0d..22766342c 100644 --- a/sentry-ruby/spec/sentry/rack/capture_exceptions_spec.rb +++ b/sentry-ruby/spec/sentry/rack/capture_exceptions_spec.rb @@ -565,6 +565,32 @@ def will_be_sampled_by_sdk end end + describe "tracing without performance" do + let(:incoming_prop_context) { Sentry::PropagationContext.new(Sentry::Scope.new) } + let(:env) do + { + "HTTP_SENTRY_TRACE" => incoming_prop_context.get_traceparent, + "HTTP_BAGGAGE" => incoming_prop_context.get_baggage.serialize + } + end + + let(:stack) do + app = ->(_e) { raise exception } + described_class.new(app) + end + + it "captures exception with correct DSC and trace context" do + expect { stack.call(env) }.to raise_error(ZeroDivisionError) + + trace_context = last_sentry_event.contexts[:trace] + expect(trace_context[:trace_id]).to eq(incoming_prop_context.trace_id) + expect(trace_context[:parent_span_id]).to eq(incoming_prop_context.span_id) + expect(trace_context[:span_id].length).to eq(16) + + expect(last_sentry_event.dynamic_sampling_context).to eq(incoming_prop_context.get_dynamic_sampling_context) + end + end + describe "session capturing" do context "when auto_session_tracking is false" do before do diff --git a/sentry-ruby/spec/sentry/scope_spec.rb b/sentry-ruby/spec/sentry/scope_spec.rb index 3573e3a6e..95f1a396a 100644 --- a/sentry-ruby/spec/sentry/scope_spec.rb +++ b/sentry-ruby/spec/sentry/scope_spec.rb @@ -311,4 +311,17 @@ end end end + + describe "#generate_propagation_context" do + it "initializes new propagation context without env" do + expect(Sentry::PropagationContext).to receive(:new).with(subject, nil) + subject.generate_propagation_context + end + + it "initializes new propagation context without env" do + env = { foo: 42 } + expect(Sentry::PropagationContext).to receive(:new).with(subject, env) + subject.generate_propagation_context(env) + end + end end diff --git a/sentry-ruby/spec/sentry/span_spec.rb b/sentry-ruby/spec/sentry/span_spec.rb index 5518f23c5..1d50896c3 100644 --- a/sentry-ruby/spec/sentry/span_spec.rb +++ b/sentry-ruby/spec/sentry/span_spec.rb @@ -77,7 +77,7 @@ sentry_trace = subject.to_sentry_trace expect(sentry_trace).to eq("#{subject.trace_id}-#{subject.span_id}-1") - expect(sentry_trace).to match(Sentry::Transaction::SENTRY_TRACE_REGEXP) + expect(sentry_trace).to match(Sentry::PropagationContext::SENTRY_TRACE_REGEXP) end context "without sampled value" do @@ -87,7 +87,7 @@ sentry_trace = subject.to_sentry_trace expect(sentry_trace).to eq("#{subject.trace_id}-#{subject.span_id}-") - expect(sentry_trace).to match(Sentry::Transaction::SENTRY_TRACE_REGEXP) + expect(sentry_trace).to match(Sentry::PropagationContext::SENTRY_TRACE_REGEXP) end end end diff --git a/sentry-ruby/spec/sentry_spec.rb b/sentry-ruby/spec/sentry_spec.rb index beb931131..80c74180f 100644 --- a/sentry-ruby/spec/sentry_spec.rb +++ b/sentry-ruby/spec/sentry_spec.rb @@ -331,18 +331,18 @@ unsampled_trace = "d298e6b033f84659928a2267c3879aaa-2a35b8e9a1b974f4-0" not_sampled_trace = "d298e6b033f84659928a2267c3879aaa-2a35b8e9a1b974f4-" - transaction = Sentry::Transaction.from_sentry_trace(sampled_trace, op: "rack.request", name: "/payment") + transaction = Sentry.continue_trace({ "sentry-trace" => sampled_trace }, op: "rack.request", name: "/payment") described_class.start_transaction(transaction: transaction) expect(transaction.sampled).to eq(true) - transaction = Sentry::Transaction.from_sentry_trace(unsampled_trace, op: "rack.request", name: "/payment") + transaction = Sentry.continue_trace({ "sentry-trace" => unsampled_trace }, op: "rack.request", name: "/payment") described_class.start_transaction(transaction: transaction) expect(transaction.sampled).to eq(false) allow(Random).to receive(:rand).and_return(0.4) - transaction = Sentry::Transaction.from_sentry_trace(not_sampled_trace, op: "rack.request", name: "/payment") + transaction = Sentry.continue_trace({ "sentry-trace" => not_sampled_trace }, op: "rack.request", name: "/payment") described_class.start_transaction(transaction: transaction) expect(transaction.sampled).to eq(true) @@ -672,7 +672,7 @@ traceparent = described_class.get_traceparent propagation_context = described_class.get_current_scope.propagation_context - expect(traceparent).to match(Sentry::Transaction::SENTRY_TRACE_REGEXP) + expect(traceparent).to match(Sentry::PropagationContext::SENTRY_TRACE_REGEXP) expect(traceparent).to eq("#{propagation_context.trace_id}-#{propagation_context.span_id}") end @@ -683,7 +683,7 @@ traceparent = described_class.get_traceparent - expect(traceparent).to match(Sentry::Transaction::SENTRY_TRACE_REGEXP) + expect(traceparent).to match(Sentry::PropagationContext::SENTRY_TRACE_REGEXP) expect(traceparent).to eq("#{span.trace_id}-#{span.span_id}-1") end end @@ -716,6 +716,69 @@ end end + describe ".continue_trace" do + + context "without incoming sentry trace" do + let(:env) { { "HTTP_FOO" => "bar" } } + + it "returns nil with tracing disabled" do + expect(described_class.continue_trace(env)).to eq(nil) + end + + it "returns nil with tracing enabled" do + Sentry.configuration.traces_sample_rate = 1.0 + expect(described_class.continue_trace(env)).to eq(nil) + end + + it "sets new propagation context on scope" do + expect(Sentry.get_current_scope).to receive(:generate_propagation_context).and_call_original + described_class.continue_trace(env) + + propagation_context = Sentry.get_current_scope.propagation_context + expect(propagation_context.incoming_trace).to eq(false) + end + end + + context "with incoming sentry trace" do + let(:incoming_prop_context) { Sentry::PropagationContext.new(Sentry::Scope.new) } + let(:env) do + { + "HTTP_SENTRY_TRACE" => incoming_prop_context.get_traceparent, + "HTTP_BAGGAGE" => incoming_prop_context.get_baggage.serialize + } + end + + it "returns nil with tracing disabled" do + expect(described_class.continue_trace(env)).to eq(nil) + end + + it "sets new propagation context from env on scope" do + expect(Sentry.get_current_scope).to receive(:generate_propagation_context).and_call_original + described_class.continue_trace(env) + + propagation_context = Sentry.get_current_scope.propagation_context + expect(propagation_context.incoming_trace).to eq(true) + expect(propagation_context.trace_id).to eq(incoming_prop_context.trace_id) + expect(propagation_context.parent_span_id).to eq(incoming_prop_context.span_id) + expect(propagation_context.parent_sampled).to eq(nil) + expect(propagation_context.baggage.items).to eq(incoming_prop_context.get_baggage.items) + expect(propagation_context.baggage.mutable).to eq(false) + end + + it "returns new Transaction with tracing enabled" do + Sentry.configuration.traces_sample_rate = 1.0 + + transaction = described_class.continue_trace(env, name: "foobar") + expect(transaction).to be_a(Sentry::Transaction) + expect(transaction.name).to eq("foobar") + expect(transaction.trace_id).to eq(incoming_prop_context.trace_id) + expect(transaction.parent_span_id).to eq(incoming_prop_context.span_id) + expect(transaction.baggage.items).to eq(incoming_prop_context.get_baggage.items) + expect(transaction.baggage.mutable).to eq(false) + end + end + end + describe 'release detection' do let(:fake_root) { "/tmp/sentry/" } diff --git a/sentry-sidekiq/lib/sentry/sidekiq/sentry_context_middleware.rb b/sentry-sidekiq/lib/sentry/sidekiq/sentry_context_middleware.rb index 646c40e60..fcb6e0ada 100644 --- a/sentry-sidekiq/lib/sentry/sidekiq/sentry_context_middleware.rb +++ b/sentry-sidekiq/lib/sentry/sidekiq/sentry_context_middleware.rb @@ -19,7 +19,7 @@ def call(_worker, job, queue) scope.set_tags(build_tags(job["tags"])) scope.set_contexts(sidekiq: job.merge("queue" => queue)) scope.set_transaction_name(context_filter.transaction_name, source: :task) - transaction = start_transaction(scope, job["sentry_trace"]) + transaction = start_transaction(scope, job["trace_propagation_headers"]) scope.set_span(transaction) if transaction begin @@ -39,9 +39,9 @@ def build_tags(tags) Array(tags).each_with_object({}) { |name, tags_hash| tags_hash[:"sidekiq.#{name}"] = true } end - def start_transaction(scope, sentry_trace) + def start_transaction(scope, env) options = { name: scope.transaction_name, source: scope.transaction_source, op: OP_NAME } - transaction = Sentry::Transaction.from_sentry_trace(sentry_trace, **options) if sentry_trace + transaction = Sentry.continue_trace(env, **options) Sentry.start_transaction(transaction: transaction, **options) end @@ -58,9 +58,8 @@ def call(_worker_class, job, _queue, _redis_pool) return yield unless Sentry.initialized? user = Sentry.get_current_scope.user - transaction = Sentry.get_current_scope.get_transaction job["sentry_user"] = user unless user.empty? - job["sentry_trace"] = transaction.to_sentry_trace if transaction + job["trace_propagation_headers"] = Sentry.get_trace_propagation_headers yield end end diff --git a/sentry-sidekiq/spec/sentry/sidekiq/sentry_context_middleware_spec.rb b/sentry-sidekiq/spec/sentry/sidekiq/sentry_context_middleware_spec.rb index c07f18050..3a2b48de0 100644 --- a/sentry-sidekiq/spec/sentry/sidekiq/sentry_context_middleware_spec.rb +++ b/sentry-sidekiq/spec/sentry/sidekiq/sentry_context_middleware_spec.rb @@ -57,11 +57,12 @@ expect(event.tags.keys).to include(:"sidekiq.marvel", :"sidekiq.dc") end - context "with sentry_trace" do + context "with trace_propagation_headers" do let(:parent_transaction) { Sentry.start_transaction(op: "sidekiq") } it "starts the transaction from it" do - execute_worker(processor, HappyWorker, sentry_trace: parent_transaction.to_sentry_trace) + trace_propagation_headers = { "sentry-trace" => parent_transaction.to_sentry_trace } + execute_worker(processor, HappyWorker, trace_propagation_headers: trace_propagation_headers) expect(transport.events.count).to eq(1) transaction = transport.events.first @@ -93,12 +94,11 @@ end end - it "does not add user or sentry_trace to the job if they're absence in the current scope" do + it "does not add user to the job if they're absent in the current scope" do client.push('queue' => 'default', 'class' => HappyWorker, 'args' => []) expect(queue.size).to be(1) expect(queue.first["sentry_user"]).to be_nil - expect(queue.first["sentry_trace"]).to be_nil end describe "with user" do @@ -121,11 +121,13 @@ Sentry.get_current_scope.set_span(transaction) end - it "sets user of the current scope to the job" do + it "sets the correct trace_propagation_headers linked to the transaction" do client.push('queue' => 'default', 'class' => HappyWorker, 'args' => []) expect(queue.size).to be(1) - expect(queue.first["sentry_trace"]).to eq(transaction.to_sentry_trace) + headers = queue.first["trace_propagation_headers"] + expect(headers["sentry-trace"]).to eq(transaction.to_sentry_trace) + expect(headers["baggage"]).to eq(transaction.to_baggage) end end end From d251e9f4a75f930b6e69820863097a04d8b49651 Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Thu, 31 Aug 2023 13:32:37 +0200 Subject: [PATCH 12/13] Remove sample_rate from DSC from prop context --- sentry-ruby/lib/sentry/propagation_context.rb | 1 - sentry-ruby/spec/sentry/propagation_context_spec.rb | 2 -- 2 files changed, 3 deletions(-) diff --git a/sentry-ruby/lib/sentry/propagation_context.rb b/sentry-ruby/lib/sentry/propagation_context.rb index c76e73e90..1fe5bc1ea 100644 --- a/sentry-ruby/lib/sentry/propagation_context.rb +++ b/sentry-ruby/lib/sentry/propagation_context.rb @@ -121,7 +121,6 @@ def populate_head_baggage items = { "trace_id" => trace_id, - "sample_rate" => configuration.traces_sample_rate&.to_s, "environment" => configuration.environment, "release" => configuration.release, "public_key" => configuration.dsn&.public_key, diff --git a/sentry-ruby/spec/sentry/propagation_context_spec.rb b/sentry-ruby/spec/sentry/propagation_context_spec.rb index 60728bc11..644fc9530 100644 --- a/sentry-ruby/spec/sentry/propagation_context_spec.rb +++ b/sentry-ruby/spec/sentry/propagation_context_spec.rb @@ -110,7 +110,6 @@ perform_basic_setup do |config| config.environment = "test" config.release = "foobar" - config.traces_sample_rate = 0.5 end end @@ -120,7 +119,6 @@ expect(baggage.mutable).to eq(false) expect(baggage.items).to eq({ "trace_id" => subject.trace_id, - "sample_rate" => "0.5", "environment" => "test", "release" => "foobar", "public_key" => Sentry.configuration.dsn.public_key From 1e3916a271ce2bc7322f8c7431cb39ec9fd6d6f6 Mon Sep 17 00:00:00 2001 From: Neel Shah Date: Thu, 31 Aug 2023 13:57:12 +0200 Subject: [PATCH 13/13] Fix spec --- sentry-ruby/spec/sentry/rack/capture_exceptions_spec.rb | 2 ++ 1 file changed, 2 insertions(+) diff --git a/sentry-ruby/spec/sentry/rack/capture_exceptions_spec.rb b/sentry-ruby/spec/sentry/rack/capture_exceptions_spec.rb index 22766342c..24a90c87b 100644 --- a/sentry-ruby/spec/sentry/rack/capture_exceptions_spec.rb +++ b/sentry-ruby/spec/sentry/rack/capture_exceptions_spec.rb @@ -579,6 +579,8 @@ def will_be_sampled_by_sdk described_class.new(app) end + before { perform_basic_setup } + it "captures exception with correct DSC and trace context" do expect { stack.call(env) }.to raise_error(ZeroDivisionError)