require 'pp'
require 'thread'
require 'logger'
require 'pathname'
require 'ddtrace/span'
require 'ddtrace/buffer'
require 'ddtrace/logger'
require 'ddtrace/writer'
require 'ddtrace/sampler'
# \Datadog global namespace that includes all tracing functionality for Tracer and Span classes.
module Datadog
# A \Tracer keeps track of the time spent by an application processing a single operation. For
# example, a trace can be used to track the entire time spent processing a complicated web request.
# Even though the request may require multiple resources and machines to handle the request, all
# of these function calls and sub-requests would be encapsulated within a single trace.
class Tracer
attr_reader :writer, :sampler, :services, :tags
attr_accessor :enabled
attr_writer :default_service
# Global, memoized, lazy initialized instance of a logger that is used within the the Datadog
# namespace. This logger outputs to +STDOUT+ by default, and is considered thread-safe.
def self.log
unless defined? @logger
@logger = Datadog::Logger.new(STDOUT)
@logger.level = Logger::WARN
end
@logger
end
# Override the default logger with a custom one.
def self.log=(logger)
return unless logger
return unless logger.respond_to? :methods
return unless logger.respond_to? :error
if logger.respond_to? :methods
unimplemented = Logger.new(STDOUT).methods - logger.methods
unless unimplemented.empty?
logger.error("logger #{logger} does not implement #{unimplemented}")
return
end
end
@logger = logger
end
# Activate the debug mode providing more information related to tracer usage
def self.debug_logging=(value)
log.level = value ? Logger::DEBUG : Logger::WARN
end
# Return if the debug mode is activated or not
def self.debug_logging
log.level == Logger::DEBUG
end
# Initialize a new \Tracer used to create, sample and submit spans that measure the
# time of sections of code. Available +options+ are:
#
# * +enabled+: set if the tracer submits or not spans to the local agent. It's enabled
# by default.
def initialize(options = {})
@enabled = options.fetch(:enabled, true)
@writer = options.fetch(:writer, Datadog::Writer.new)
@sampler = options.fetch(:sampler, Datadog::AllSampler.new)
@buffer = Datadog::SpanBuffer.new()
@mutex = Mutex.new
@spans = []
@services = {}
@tags = {}
end
# Updates the current \Tracer instance, so that the tracer can be configured after the
# initialization. Available +options+ are:
#
# * +enabled+: set if the tracer submits or not spans to the trace agent
# * +hostname+: change the location of the trace agent
# * +port+: change the port of the trace agent
#
# For instance, if the trace agent runs in a different location, just:
#
# tracer.configure(hostname: 'agent.service.consul', port: '8777')
#
def configure(options = {})
enabled = options.fetch(:enabled, nil)
hostname = options.fetch(:hostname, nil)
port = options.fetch(:port, nil)
sampler = options.fetch(:sampler, nil)
@enabled = enabled unless enabled.nil?
@writer.transport.hostname = hostname unless hostname.nil?
@writer.transport.port = port unless port.nil?
@sampler = sampler unless sampler.nil?
end
# Set the information about the given service. A valid example is:
#
# tracer.set_service_info('web-application', 'rails', 'web')
def set_service_info(service, app, app_type)
@services[service] = {
'app' => app,
'app_type' => app_type
}
return unless Datadog::Tracer.debug_logging
Datadog::Tracer.log.debug("set_service_info: service: #{service} app: #{app} type: #{app_type}")
end
# A default value for service. One should really override this one
# for non-root spans which have a parent. However, root spans without
# a service would be invalid and rejected.
def default_service
return @default_service if @default_service
begin
@default_service = File.basename($PROGRAM_NAME, '.*')
rescue => e
Datadog::Tracer.log.error("unable to guess default service: #{e}")
@default_service = 'ruby'.freeze
end
@default_service
end
# Set the given key / value tag pair at the tracer level. These tags will be
# appended to each span created by the tracer. Keys and values must be strings.
# A valid example is:
#
# tracer.set_tags('env' => 'prod', 'component' => 'core')
def set_tags(tags)
@tags.update(tags)
end
# Return a +span+ that will trace an operation called +name+. You could trace your code
# using a do-block like:
#
# tracer.trace('web.request') do |span|
# span.service = 'my-web-site'
# span.resource = '/'
# span.set_tag('http.method', request.request_method)
# do_something()
# end
#
# The tracer.trace() method can also be used without a block in this way:
#
# span = tracer.trace('web.request', service: 'my-web-site')
# do_something()
# span.finish()
#
# Remember that in this case, calling span.finish() is mandatory.
#
# When a Trace is started, trace() will store the created span; subsequent spans will
# become it's children and will inherit some properties:
#
# parent = tracer.trace('parent') # has no parent span
# child = tracer.trace('child') # is a child of 'parent'
# child.finish()
# parent.finish()
# parent2 = tracer.trace('parent2') # has no parent span
# parent2.finish()
#
def trace(name, options = {})
span = Span.new(self, name, options)
# set up inheritance
parent = @buffer.get()
span.set_parent(parent)
@buffer.set(span)
@tags.each { |k, v| span.set_tag(k, v) } unless @tags.empty?
# sampling
if parent.nil?
@sampler.sample(span)
else
span.sampled = span.parent.sampled
end
# call the finish only if a block is given; this ensures
# that a call to tracer.trace() without a block, returns
# a span that should be manually finished.
if block_given?
begin
yield(span)
rescue StandardError => e
span.set_error(e)
raise
ensure
span.finish()
end
else
span
end
end
# Record the given finished span in the +spans+ list. When a +span+ is recorded, it will be sent
# to the Datadog trace agent as soon as the trace is finished.
def record(span)
span.service ||= default_service
spans = []
@mutex.synchronize do
@spans << span
parent = span.parent
# Bubble up until we find a non-finished parent. This is necessary for
# the case when the parent finished after its parent.
parent = parent.parent while !parent.nil? && parent.finished?
@buffer.set(parent)
return unless parent.nil?
spans = @spans
@spans = []
end
return if spans.empty? || !span.sampled
write(spans)
end
# Return the current active span or +nil+.
def active_span
@buffer.get()
end
def write(spans)
return if @writer.nil? || !@enabled
if Datadog::Tracer.debug_logging
Datadog::Tracer.log.debug("Writing #{spans.length} spans (enabled: #{@enabled})")
PP.pp(spans)
end
@writer.write(spans, @services)
end
private :write
end
end