# frozen_string_literal: true # Copyright The OpenTelemetry Authors # # SPDX-License-Identifier: Apache-2.0 module OpenTelemetry module Instrumentation module Net module HTTP module Patches # Module to prepend to Net::HTTP for instrumentation module Instrumentation HTTP_METHODS_TO_SPAN_NAMES = Hash.new { |h, k| h[k] = "HTTP #{k}" } USE_SSL_TO_SCHEME = { false => 'http', true => 'https' }.freeze def request(req, body = nil, &block) # Do not trace recursive call for starting the connection return super(req, body, &block) unless started? attributes = { OpenTelemetry::SemanticConventions::Trace::HTTP_METHOD => req.method, OpenTelemetry::SemanticConventions::Trace::HTTP_SCHEME => USE_SSL_TO_SCHEME[use_ssl?], OpenTelemetry::SemanticConventions::Trace::HTTP_TARGET => req.path, OpenTelemetry::SemanticConventions::Trace::NET_PEER_NAME => @address, OpenTelemetry::SemanticConventions::Trace::NET_PEER_PORT => @port }.merge!(OpenTelemetry::Common::HTTP::ClientContext.attributes) tracer.in_span( HTTP_METHODS_TO_SPAN_NAMES[req.method], attributes: attributes, kind: :client ) do |span| OpenTelemetry.propagation.inject(req) super(req, body, &block).tap do |response| annotate_span_with_response!(span, response) end end end private def connect if proxy? conn_address = proxy_address conn_port = proxy_port else conn_address = address 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) tracer.in_span('HTTP CONNECT', attributes: attributes) do super end end def annotate_span_with_response!(span, response) return unless response&.code status_code = response.code.to_i span.set_attribute(OpenTelemetry::SemanticConventions::Trace::HTTP_STATUS_CODE, status_code) span.status = OpenTelemetry::Trace::Status.error unless (100..399).include?(status_code.to_i) end def tracer Net::HTTP::Instrumentation.instance.tracer end end end end end end end