Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(excon)! Add a connect span to excon and add more span attributes to the tracer middleware #712

Merged
merged 27 commits into from
Nov 28, 2023
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
ad99f5a
Add a connect span to excon and add more span attributes to the trace…
misalcedo Nov 6, 2023
997d5cd
Skip matching on the error message as it varies by platform
misalcedo Nov 6, 2023
b399dd8
Rescue the IOError that can occur on accept
misalcedo Nov 6, 2023
9c4e787
Switch to must_be_empty assertion instead of size
misalcedo Nov 16, 2023
854125b
Move allowing and disallowing connect to setup and after respectively.
misalcedo Nov 16, 2023
8c8eabc
use dig when getting hostname and port of proxy
misalcedo Nov 16, 2023
175818c
Call untraced when we hit an untraced host.
misalcedo Nov 16, 2023
6415ba8
Switch to using recording? to test whether we should finish a span.
misalcedo Nov 16, 2023
441f1d0
Switch to handle_error instead of debug logging.
misalcedo Nov 16, 2023
55bf3f9
Record the exception on error
misalcedo Nov 16, 2023
4bf5b3d
Perform the next step in the middleware stack in the context of the c…
misalcedo Nov 16, 2023
99d8c62
Add assertions on http spans to connect tests.
misalcedo Nov 16, 2023
fd148ef
Fix rubocop lints
misalcedo Nov 21, 2023
d6f6c2b
Remove interpolation from status message on span now that we capture …
misalcedo Nov 21, 2023
2cdce0b
Switch to attach and detach instead of with_span
misalcedo Nov 21, 2023
d1aa1d4
Include untraced context into untraced? check for the middleware and …
misalcedo Nov 21, 2023
e3fde0c
Add test for untraced.
misalcedo Nov 21, 2023
a0b428d
Merge branch 'main' into misalcedo/excon
arielvalentin Nov 21, 2023
2fbc8e9
Add a module that centralizes the untraced hosts concern.
misalcedo Nov 21, 2023
1721eea
Expand doc comment.
misalcedo Nov 21, 2023
c0a2404
Move module to the excon gem.
misalcedo Nov 22, 2023
64ec771
Merge branch 'main' into misalcedo/excon
misalcedo Nov 22, 2023
64011e8
Add doc comment for untraced? in the concern
misalcedo Nov 22, 2023
9fb3072
Update instrumentation/excon/lib/opentelemetry/instrumentation/excon/…
misalcedo Nov 22, 2023
851aa89
Merge branch 'main' into misalcedo/excon
arielvalentin Nov 22, 2023
1b9c62d
Merge branch 'main' into misalcedo/excon
arielvalentin Nov 22, 2023
802100f
Merge branch 'main' into misalcedo/excon
misalcedo Nov 28, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ class Instrumentation < OpenTelemetry::Instrumentation::Base
install do |_config|
require_dependencies
add_middleware
patch
end

present do
Expand All @@ -21,15 +22,25 @@ class Instrumentation < OpenTelemetry::Instrumentation::Base

option :peer_service, default: nil, validate: :string

# untraced_hosts: if a request's address matches any of the `String`
# or `Regexp` in this array, the instrumentation will not record a
# `kind = :client` representing the request and will not propagate
# context in the request.
option :untraced_hosts, default: [], validate: :array

private

def require_dependencies
require_relative 'middlewares/tracer_middleware'
require_relative 'patches/socket'
end

def add_middleware
::Excon.defaults[:middlewares] =
Middlewares::TracerMiddleware.around_default_stack
::Excon.defaults[:middlewares] = Middlewares::TracerMiddleware.around_default_stack
end

def patch
::Excon::Socket.prepend(Patches::Socket)
end
end
end
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -22,26 +22,34 @@ class TracerMiddleware < ::Excon::Middleware::Base
end.freeze

def request_call(datum)
begin
unless datum.key?(:otel_span)
http_method = HTTP_METHODS_TO_UPPERCASE[datum[:method]]
attributes = span_creation_attributes(datum, http_method)
tracer.start_span(
HTTP_METHODS_TO_SPAN_NAMES[http_method],
attributes: attributes,
kind: :client
).tap do |span|
datum[:otel_span] = span
OpenTelemetry::Trace.with_span(span) do
OpenTelemetry.propagation.inject(datum[:headers])
end
end
if skip_trace?(datum)
return OpenTelemetry::Common::Utilities.untraced do
@stack.request_call(datum)
end
rescue StandardError => e
OpenTelemetry.logger.debug(e.message)
end

@stack.request_call(datum)
http_method = HTTP_METHODS_TO_UPPERCASE[datum[:method]]

attributes = {
OpenTelemetry::SemanticConventions::Trace::HTTP_METHOD => http_method,
OpenTelemetry::SemanticConventions::Trace::HTTP_SCHEME => datum[:scheme],
OpenTelemetry::SemanticConventions::Trace::HTTP_TARGET => datum[:path],
OpenTelemetry::SemanticConventions::Trace::HTTP_HOST => datum[:host],
OpenTelemetry::SemanticConventions::Trace::NET_PEER_NAME => datum[:hostname],
OpenTelemetry::SemanticConventions::Trace::NET_PEER_PORT => datum[:port]
}

peer_service = Excon::Instrumentation.instance.config[:peer_service]
attributes[OpenTelemetry::SemanticConventions::Trace::PEER_SERVICE] = peer_service if peer_service
attributes.merge!(OpenTelemetry::Common::HTTP::ClientContext.attributes)

datum[:otel_span] = tracer.start_span(HTTP_METHODS_TO_SPAN_NAMES[http_method], attributes: attributes, kind: :client)

OpenTelemetry::Trace.with_span(datum[:otel_span]) do
misalcedo marked this conversation as resolved.
Show resolved Hide resolved
OpenTelemetry.propagation.inject(datum[:headers])

@stack.request_call(datum)
end
end

def response_call(datum)
Expand Down Expand Up @@ -71,43 +79,35 @@ def self.around_default_stack
private

def handle_response(datum)
if datum.key?(:otel_span)
datum[:otel_span].tap do |span|
return span if span.end_timestamp
datum.delete(:otel_span)&.tap do |span|
return span unless span.recording?
misalcedo marked this conversation as resolved.
Show resolved Hide resolved

if datum.key?(:response)
response = datum[:response]
span.set_attribute('http.status_code', response[:status])
span.status = OpenTelemetry::Trace::Status.error unless (100..399).include?(response[:status].to_i)
end

span.status = OpenTelemetry::Trace::Status.error("Request has failed: #{datum[:error]}") if datum.key?(:error)
if datum.key?(:response)
response = datum[:response]
span.set_attribute(OpenTelemetry::SemanticConventions::Trace::HTTP_STATUS_CODE, response[:status])
span.status = OpenTelemetry::Trace::Status.error unless (100..399).include?(response[:status].to_i)
arielvalentin marked this conversation as resolved.
Show resolved Hide resolved
end

span.finish
datum.delete(:otel_span)
if datum.key?(:error)
span.status = OpenTelemetry::Trace::Status.error("Request has failed: #{datum[:error]}")
misalcedo marked this conversation as resolved.
Show resolved Hide resolved
span.record_exception(datum[:error])
end

span.finish
end
rescue StandardError => e
OpenTelemetry.logger.debug(e.message)
end

def span_creation_attributes(datum, http_method)
instrumentation_attrs = {
'http.host' => datum[:host],
'http.method' => http_method,
'http.scheme' => datum[:scheme],
'http.target' => datum[:path]
}
config = Excon::Instrumentation.instance.config
instrumentation_attrs['peer.service'] = config[:peer_service] if config[:peer_service]
instrumentation_attrs.merge!(
OpenTelemetry::Common::HTTP::ClientContext.attributes
)
OpenTelemetry.handle_error(e)
end

def tracer
Excon::Instrumentation.instance.tracer
end

def skip_trace?(datum)
datum.key?(:otel_span) || Excon::Instrumentation.instance.config[:untraced_hosts].any? do |host|
host.is_a?(Regexp) ? host.match?(datum[:host]) : host == datum[:host]
end
end
end
end
end
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,62 @@
# frozen_string_literal: true

# Copyright The OpenTelemetry Authors
#
# SPDX-License-Identifier: Apache-2.0

module OpenTelemetry
module Instrumentation
module Excon
module Patches
# Module to prepend to an Excon Socket for instrumentation
module Socket
private

def connect
if untraced?
return OpenTelemetry::Common::Utilities.untraced { super }
end

if @data[:proxy]
conn_address = @data.dig(:proxy, :hostname)
conn_port = @data.dig(:proxy, :port)
else
conn_address = @data[:hostname]
conn_port = @port
end

attributes = { OpenTelemetry::SemanticConventions::Trace::NET_PEER_NAME => conn_address, OpenTelemetry::SemanticConventions::Trace::NET_PEER_PORT => conn_port }.merge!(OpenTelemetry::Common::HTTP::ClientContext.attributes)

if is_a?(::Excon::SSLSocket) && @data[:proxy]
arielvalentin marked this conversation as resolved.
Show resolved Hide resolved
span_name = 'HTTP CONNECT'
span_kind = :client
else
span_name = 'connect'
span_kind = :internal
end

tracer.in_span(span_name, attributes: attributes, kind: span_kind) do
super
end
end

def tracer
Excon::Instrumentation.instance.tracer
end

def untraced?
arielvalentin marked this conversation as resolved.
Show resolved Hide resolved
address = if @data[:proxy]
@data.dig(:proxy, :hostname)
else
@data[:hostname]
end

Excon::Instrumentation.instance.config[:untraced_hosts].any? do |host|
host.is_a?(Regexp) ? host.match?(address) : host == address
end
end
end
end
end
end
end
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@

require_relative '../../../../lib/opentelemetry/instrumentation/excon'
require_relative '../../../../lib/opentelemetry/instrumentation/excon/middlewares/tracer_middleware'
require_relative '../../../../lib/opentelemetry/instrumentation/excon/patches/socket'

describe OpenTelemetry::Instrumentation::Excon::Instrumentation do
let(:instrumentation) { OpenTelemetry::Instrumentation::Excon::Instrumentation.instance }
Expand Down Expand Up @@ -39,7 +40,7 @@
end

it 'before request' do
_(exporter.finished_spans.size).must_equal 0
_(exporter.finished_spans).must_be_empty
end

it 'after request with success code' do
Expand Down Expand Up @@ -104,6 +105,10 @@
'http://example.com/timeout',
headers: { 'Traceparent' => "00-#{span.hex_trace_id}-#{span.hex_span_id}-01" }
)

exception_event = span.events.first
_(exception_event.attributes['exception.type']).must_equal('Excon::Error::Timeout')
_(exception_event.attributes['exception.message']).must_equal('Excon::Error::Timeout')
end

it 'merges HTTP client context' do
Expand Down Expand Up @@ -150,4 +155,156 @@
_(span.attributes['peer.service']).must_equal 'example:custom'
end
end

describe 'untraced?' do
before do
instrumentation.install(untraced_hosts: ['foobar.com', /bazqux\.com/])

stub_request(:get, 'http://example.com/body').to_return(status: 200)
stub_request(:get, 'http://foobar.com/body').to_return(status: 200)
stub_request(:get, 'http://bazqux.com/body').to_return(status: 200)
end

it 'does not create a span when request ignored using a string' do
Excon.get('http://foobar.com/body')
_(exporter.finished_spans).must_be_empty
end

it 'does not create a span when request ignored using a regexp' do
Excon.get('http://bazqux.com/body')
_(exporter.finished_spans).must_be_empty
end

it 'does not create a span on connect when request ignored using a regexp' do
uri = URI.parse('http://bazqux.com')

Excon::Socket.new(hostname: uri.host, port: uri.port)

_(exporter.finished_spans).must_be_empty
end

it 'creates a span for a non-ignored request' do
Excon.get('http://example.com/body')

_(exporter.finished_spans.size).must_equal 1
_(span.name).must_equal 'HTTP GET'
_(span.attributes['http.method']).must_equal 'GET'
_(span.attributes['http.host']).must_equal 'example.com'
end

it 'creates a span on connect for a non-ignored request' do
uri = URI.parse('http://example.com')

Excon::Socket.new(hostname: uri.host, port: uri.port)

_(exporter.finished_spans.size).must_equal 1
_(span.name).must_equal('connect')
_(span.kind).must_equal(:internal)
_(span.attributes['net.peer.name']).must_equal('example.com')
_(span.attributes['net.peer.port']).must_equal(80)
end
end

# NOTE: WebMock introduces an extra HTTP request span due to the way the mocking is implemented.
describe '#connect' do
before do
instrumentation.install
WebMock.allow_net_connect!
end

after do
WebMock.disable_net_connect!
end

it 'emits span on connect' do
TCPServer.open('localhost', 0) do |server|
Thread.start do
server.accept
rescue IOError
nil
end

port = server.addr[1]

_(-> { Excon.get("http://localhost:#{port}/example", read_timeout: 0) }).must_raise(Excon::Error::Timeout)
end

_(exporter.finished_spans.size).must_equal(3)
_(span.name).must_equal 'connect'
_(span.attributes['net.peer.name']).must_equal('localhost')
_(span.attributes['net.peer.port']).wont_be_nil

assert_http_spans(target: '/example', exception: 'Excon::Error::Timeout')
end

it 'captures errors' do
_(-> { Excon.get('http://invalid.com:99999/example') }).must_raise

_(exporter.finished_spans.size).must_equal(3)
_(span.name).must_equal 'connect'
_(span.attributes['net.peer.name']).must_equal('invalid.com')
_(span.attributes['net.peer.port']).must_equal(99_999)

span_event = span.events.first

_(span_event.name).must_equal 'exception'
_(span_event.attributes['exception.type']).must_equal(SocketError.name)

assert_http_spans(host: 'invalid.com', target: '/example')
end

it '[BUG] fails to emit an HTTP CONNECT span when connecting through an SSL proxy for an HTTP service' do
_(-> { Excon.get('http://localhost/', proxy: 'https://proxy_user:proxy_pass@localhost') }).must_raise(Excon::Error::Socket)

_(exporter.finished_spans.size).must_equal(3)
_(span.name).must_equal 'connect'
_(span.kind).must_equal(:internal)
_(span.attributes['net.peer.name']).must_equal('localhost')
_(span.attributes['net.peer.port']).must_equal(443)

assert_http_spans
end

it 'emits an HTTP CONNECT span when connecting through an SSL proxy' do
_(-> { Excon.get('https://localhost/', proxy: 'https://proxy_user:proxy_pass@localhost') }).must_raise(Excon::Error::Socket)

_(exporter.finished_spans.size).must_equal(3)
_(span.name).must_equal 'HTTP CONNECT'
_(span.kind).must_equal(:client)
_(span.attributes['net.peer.name']).must_equal('localhost')
_(span.attributes['net.peer.port']).must_equal(443)

assert_http_spans(scheme: "https")
end

it 'emits a "connect" span when connecting through an non-ssl proxy' do
_(-> { Excon.get('http://localhost', proxy: 'https://proxy_user:proxy_pass@localhost') }).must_raise(Excon::Error::Socket)

_(exporter.finished_spans.size).must_equal(3)
misalcedo marked this conversation as resolved.
Show resolved Hide resolved
_(span.name).must_equal 'connect'
_(span.kind).must_equal(:internal)
_(span.attributes['net.peer.name']).must_equal('localhost')
_(span.attributes['net.peer.port']).must_equal(443)

assert_http_spans(exception: 'Excon::Error::Socket')
end
end

def assert_http_spans(scheme: 'http', host: 'localhost', target: '/', exception: nil)
exporter.finished_spans[1..].each do |http_span|
_(http_span.name).must_equal 'HTTP GET'
_(http_span.attributes['http.method']).must_equal 'GET'
_(http_span.attributes['http.scheme']).must_equal scheme
_(http_span.attributes['http.host']).must_equal host
_(http_span.attributes['http.target']).must_equal target
_(http_span.status.code).must_equal(
OpenTelemetry::Trace::Status::ERROR
)

if exception
exception_event = http_span.events.first
_(exception_event.attributes['exception.type']).must_equal(exception)
end
end
end
end
Loading