2024-05-10 14:40:20 +02:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
# Set OTEL_* environment variables according to OTel docs:
|
|
|
|
# https://opentelemetry.io/docs/concepts/sdk-configuration/
|
|
|
|
|
|
|
|
if ENV.keys.any? { |name| name.match?(/OTEL_.*_ENDPOINT/) }
|
|
|
|
require 'opentelemetry/sdk'
|
|
|
|
require 'opentelemetry/exporter/otlp'
|
|
|
|
|
|
|
|
require 'opentelemetry/instrumentation/active_job'
|
|
|
|
require 'opentelemetry/instrumentation/active_model_serializers'
|
|
|
|
require 'opentelemetry/instrumentation/concurrent_ruby'
|
|
|
|
require 'opentelemetry/instrumentation/excon'
|
|
|
|
require 'opentelemetry/instrumentation/faraday'
|
|
|
|
require 'opentelemetry/instrumentation/http'
|
|
|
|
require 'opentelemetry/instrumentation/http_client'
|
|
|
|
require 'opentelemetry/instrumentation/net/http'
|
|
|
|
require 'opentelemetry/instrumentation/pg'
|
|
|
|
require 'opentelemetry/instrumentation/rack'
|
|
|
|
require 'opentelemetry/instrumentation/rails'
|
|
|
|
require 'opentelemetry/instrumentation/redis'
|
|
|
|
require 'opentelemetry/instrumentation/sidekiq'
|
|
|
|
|
|
|
|
OpenTelemetry::SDK.configure do |c|
|
|
|
|
# use_all() attempts to load ALL the auto-instrumentations
|
|
|
|
# currently loaded by Ruby requires.
|
|
|
|
#
|
|
|
|
# Load attempts will emit an INFO or WARN to the console
|
|
|
|
# about the success/failure to wire up an auto-instrumentation.
|
|
|
|
# "WARN -- : Instrumentation: <X> failed to install" is most
|
|
|
|
# likely caused by <X> not being a Ruby library loaded by
|
|
|
|
# the application or the instrumentation has been explicitly
|
|
|
|
# disabled.
|
|
|
|
#
|
|
|
|
# To disable an instrumentation, set an environment variable
|
|
|
|
# along this pattern:
|
|
|
|
#
|
|
|
|
# OTEL_RUBY_INSTRUMENTATION_<X>_ENABLED=false
|
|
|
|
#
|
|
|
|
# For example, PostgreSQL and Redis produce a lot of child spans
|
|
|
|
# in the course of this application doing its business. To turn
|
|
|
|
# them off, set the env vars below, but recognize that you will
|
|
|
|
# be missing details about what particular calls to the
|
|
|
|
# datastores are slow.
|
|
|
|
#
|
|
|
|
# OTEL_RUBY_INSTRUMENTATION_PG_ENABLED=false
|
|
|
|
# OTEL_RUBY_INSTRUMENTATION_REDIS_ENABLED=false
|
|
|
|
|
|
|
|
c.use_all({
|
|
|
|
'OpenTelemetry::Instrumentation::Rack' => {
|
|
|
|
use_rack_events: false, # instead of events, use middleware; allows for untraced_endpoints to ignore child spans
|
|
|
|
untraced_endpoints: ['/health'],
|
|
|
|
},
|
2024-05-20 10:01:04 +02:00
|
|
|
'OpenTelemetry::Instrumentation::Sidekiq' => {
|
|
|
|
span_naming: :job_class, # Use the job class as the span name, otherwise this is the queue name and not very helpful
|
|
|
|
},
|
2024-12-13 09:21:38 +01:00
|
|
|
'OpenTelemetry::Instrumentation::Redis' => {
|
|
|
|
trace_root_spans: false, # don't start traces with Redis spans
|
|
|
|
},
|
2024-05-10 14:40:20 +02:00
|
|
|
})
|
|
|
|
|
2024-09-23 10:55:35 +02:00
|
|
|
prefix = ENV.fetch('OTEL_SERVICE_NAME_PREFIX', 'mastodon')
|
|
|
|
separator = ENV.fetch('OTEL_SERVICE_NAME_SEPARATOR', '/')
|
2024-05-16 11:28:10 +02:00
|
|
|
|
2024-05-10 14:40:20 +02:00
|
|
|
c.service_name = case $PROGRAM_NAME
|
2024-09-23 10:55:35 +02:00
|
|
|
when /puma/ then "#{prefix}#{separator}web"
|
2024-05-10 14:40:20 +02:00
|
|
|
else
|
2024-09-23 10:55:35 +02:00
|
|
|
"#{prefix}#{separator}#{$PROGRAM_NAME.split('/').last}"
|
2024-05-10 14:40:20 +02:00
|
|
|
end
|
|
|
|
c.service_version = Mastodon::Version.to_s
|
2024-11-18 09:41:09 +01:00
|
|
|
|
|
|
|
if Mastodon::Version.source_commit.present?
|
|
|
|
c.resource = OpenTelemetry::SDK::Resources::Resource.create(
|
|
|
|
'vcs.repository.ref.revision' => Mastodon::Version.source_commit,
|
|
|
|
'vcs.repository.url.full' => Mastodon::Version.source_base_url
|
|
|
|
)
|
|
|
|
end
|
2024-05-10 14:40:20 +02:00
|
|
|
end
|
2024-12-18 08:34:20 +01:00
|
|
|
|
|
|
|
# This middleware adds the trace_id and span_id to the Rails logging tags for every requests
|
|
|
|
class TelemetryLoggingMiddleware
|
|
|
|
def initialize(app)
|
|
|
|
@app = app
|
|
|
|
end
|
|
|
|
|
|
|
|
def call(env)
|
|
|
|
span = OpenTelemetry::Trace.current_span
|
|
|
|
|
|
|
|
unless span.recording?
|
|
|
|
@app.call(env)
|
|
|
|
return
|
|
|
|
end
|
|
|
|
|
|
|
|
span_id = span.context.hex_span_id
|
|
|
|
trace_id = span.context.hex_trace_id
|
|
|
|
|
|
|
|
Rails.logger.tagged("trace_id=#{trace_id}", "span_id=#{span_id}") do
|
|
|
|
@app.call(env)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
Rails.application.configure do
|
|
|
|
config.middleware.insert_before Rails::Rack::Logger, TelemetryLoggingMiddleware
|
|
|
|
end
|
|
|
|
|
2024-05-10 14:40:20 +02:00
|
|
|
end
|
2024-05-24 15:13:23 +02:00
|
|
|
|
|
|
|
MastodonOTELTracer = OpenTelemetry.tracer_provider.tracer('mastodon')
|