use crate::{OtelData, PreSampledTracer};
use once_cell::unsync;
use opentelemetry::{
trace::{self as otel, noop, TraceContextExt},
Context as OtelContext, Key, KeyValue, Value,
};
use std::any::TypeId;
use std::borrow::Cow;
use std::fmt;
use std::marker;
use std::thread;
use std::time::{Instant, SystemTime};
use tracing_core::span::{self, Attributes, Id, Record};
use tracing_core::{field, Event, Subscriber};
#[cfg(feature = "tracing-log")]
use tracing_log::NormalizeEvent;
use tracing_subscriber::layer::Context;
use tracing_subscriber::registry::LookupSpan;
use tracing_subscriber::Layer;
const SPAN_NAME_FIELD: &str = "otel.name";
const SPAN_KIND_FIELD: &str = "otel.kind";
const SPAN_STATUS_CODE_FIELD: &str = "otel.status_code";
const SPAN_STATUS_MESSAGE_FIELD: &str = "otel.status_message";
const FIELD_EXCEPTION_MESSAGE: &str = "exception.message";
const FIELD_EXCEPTION_STACKTRACE: &str = "exception.stacktrace";
pub struct OpenTelemetryLayer<S, T> {
tracer: T,
location: bool,
tracked_inactivity: bool,
with_threads: bool,
exception_config: ExceptionFieldConfig,
get_context: WithContext,
_registry: marker::PhantomData<S>,
}
impl<S> Default for OpenTelemetryLayer<S, noop::NoopTracer>
where
S: Subscriber + for<'span> LookupSpan<'span>,
{
fn default() -> Self {
OpenTelemetryLayer::new(noop::NoopTracer::new())
}
}
pub fn layer<S>() -> OpenTelemetryLayer<S, noop::NoopTracer>
where
S: Subscriber + for<'span> LookupSpan<'span>,
{
OpenTelemetryLayer::default()
}
pub(crate) struct WithContext(
fn(&tracing::Dispatch, &span::Id, f: &mut dyn FnMut(&mut OtelData, &dyn PreSampledTracer)),
);
impl WithContext {
pub(crate) fn with_context<'a>(
&self,
dispatch: &'a tracing::Dispatch,
id: &span::Id,
mut f: impl FnMut(&mut OtelData, &dyn PreSampledTracer),
) {
(self.0)(dispatch, id, &mut f)
}
}
fn str_to_span_kind(s: &str) -> Option<otel::SpanKind> {
match s {
s if s.eq_ignore_ascii_case("server") => Some(otel::SpanKind::Server),
s if s.eq_ignore_ascii_case("client") => Some(otel::SpanKind::Client),
s if s.eq_ignore_ascii_case("producer") => Some(otel::SpanKind::Producer),
s if s.eq_ignore_ascii_case("consumer") => Some(otel::SpanKind::Consumer),
s if s.eq_ignore_ascii_case("internal") => Some(otel::SpanKind::Internal),
_ => None,
}
}
fn str_to_status_code(s: &str) -> Option<otel::StatusCode> {
match s {
s if s.eq_ignore_ascii_case("unset") => Some(otel::StatusCode::Unset),
s if s.eq_ignore_ascii_case("ok") => Some(otel::StatusCode::Ok),
s if s.eq_ignore_ascii_case("error") => Some(otel::StatusCode::Error),
_ => None,
}
}
struct SpanEventVisitor<'a, 'b> {
event_builder: &'a mut otel::Event,
span_builder: Option<&'b mut otel::SpanBuilder>,
exception_config: ExceptionFieldConfig,
}
impl<'a, 'b> field::Visit for SpanEventVisitor<'a, 'b> {
fn record_bool(&mut self, field: &field::Field, value: bool) {
match field.name() {
"message" => self.event_builder.name = value.to_string().into(),
#[cfg(feature = "tracing-log")]
name if name.starts_with("log.") => (),
name => {
self.event_builder
.attributes
.push(KeyValue::new(name, value));
}
}
}
fn record_f64(&mut self, field: &field::Field, value: f64) {
match field.name() {
"message" => self.event_builder.name = value.to_string().into(),
#[cfg(feature = "tracing-log")]
name if name.starts_with("log.") => (),
name => {
self.event_builder
.attributes
.push(KeyValue::new(name, value));
}
}
}
fn record_i64(&mut self, field: &field::Field, value: i64) {
match field.name() {
"message" => self.event_builder.name = value.to_string().into(),
#[cfg(feature = "tracing-log")]
name if name.starts_with("log.") => (),
name => {
self.event_builder
.attributes
.push(KeyValue::new(name, value));
}
}
}
fn record_str(&mut self, field: &field::Field, value: &str) {
match field.name() {
"message" => self.event_builder.name = value.to_string().into(),
#[cfg(feature = "tracing-log")]
name if name.starts_with("log.") => (),
name => {
self.event_builder
.attributes
.push(KeyValue::new(name, value.to_string()));
}
}
}
fn record_debug(&mut self, field: &field::Field, value: &dyn fmt::Debug) {
match field.name() {
"message" => self.event_builder.name = format!("{:?}", value).into(),
#[cfg(feature = "tracing-log")]
name if name.starts_with("log.") => (),
name => {
self.event_builder
.attributes
.push(KeyValue::new(name, format!("{:?}", value)));
}
}
}
fn record_error(
&mut self,
field: &tracing_core::Field,
value: &(dyn std::error::Error + 'static),
) {
let mut chain = Vec::new();
let mut next_err = value.source();
while let Some(err) = next_err {
chain.push(Cow::Owned(err.to_string()));
next_err = err.source();
}
let error_msg = value.to_string();
if self.exception_config.record {
self.event_builder
.attributes
.push(Key::new(FIELD_EXCEPTION_MESSAGE).string(error_msg.clone()));
self.event_builder
.attributes
.push(Key::new(FIELD_EXCEPTION_STACKTRACE).array(chain.clone()));
}
if self.exception_config.propagate {
if let Some(span) = &mut self.span_builder {
if let Some(attrs) = span.attributes.as_mut() {
attrs.push(Key::new(FIELD_EXCEPTION_MESSAGE).string(error_msg.clone()));
attrs.push(Key::new(FIELD_EXCEPTION_STACKTRACE).array(chain.clone()));
}
}
}
self.event_builder
.attributes
.push(Key::new(field.name()).string(error_msg));
self.event_builder
.attributes
.push(Key::new(format!("{}.chain", field.name())).array(chain));
}
}
#[derive(Clone, Copy)]
struct ExceptionFieldConfig {
record: bool,
propagate: bool,
}
struct SpanAttributeVisitor<'a> {
span_builder: &'a mut otel::SpanBuilder,
exception_config: ExceptionFieldConfig,
}
impl<'a> SpanAttributeVisitor<'a> {
fn record(&mut self, attribute: KeyValue) {
debug_assert!(self.span_builder.attributes.is_some());
if let Some(v) = self.span_builder.attributes.as_mut() {
v.push(attribute);
}
}
}
impl<'a> field::Visit for SpanAttributeVisitor<'a> {
fn record_bool(&mut self, field: &field::Field, value: bool) {
self.record(KeyValue::new(field.name(), value));
}
fn record_f64(&mut self, field: &field::Field, value: f64) {
self.record(KeyValue::new(field.name(), value));
}
fn record_i64(&mut self, field: &field::Field, value: i64) {
self.record(KeyValue::new(field.name(), value));
}
fn record_str(&mut self, field: &field::Field, value: &str) {
match field.name() {
SPAN_NAME_FIELD => self.span_builder.name = value.to_string().into(),
SPAN_KIND_FIELD => self.span_builder.span_kind = str_to_span_kind(value),
SPAN_STATUS_CODE_FIELD => self.span_builder.status_code = str_to_status_code(value),
SPAN_STATUS_MESSAGE_FIELD => {
self.span_builder.status_message = Some(value.to_owned().into())
}
_ => self.record(KeyValue::new(field.name(), value.to_string())),
}
}
fn record_debug(&mut self, field: &field::Field, value: &dyn fmt::Debug) {
match field.name() {
SPAN_NAME_FIELD => self.span_builder.name = format!("{:?}", value).into(),
SPAN_KIND_FIELD => {
self.span_builder.span_kind = str_to_span_kind(&format!("{:?}", value))
}
SPAN_STATUS_CODE_FIELD => {
self.span_builder.status_code = str_to_status_code(&format!("{:?}", value))
}
SPAN_STATUS_MESSAGE_FIELD => {
self.span_builder.status_message = Some(format!("{:?}", value).into())
}
_ => self.record(Key::new(field.name()).string(format!("{:?}", value))),
}
}
fn record_error(
&mut self,
field: &tracing_core::Field,
value: &(dyn std::error::Error + 'static),
) {
let mut chain = Vec::new();
let mut next_err = value.source();
while let Some(err) = next_err {
chain.push(Cow::Owned(err.to_string()));
next_err = err.source();
}
let error_msg = value.to_string();
if self.exception_config.record {
self.record(Key::new(FIELD_EXCEPTION_MESSAGE).string(error_msg.clone()));
self.record(Key::new(FIELD_EXCEPTION_STACKTRACE).array(chain.clone()));
}
self.record(Key::new(field.name()).string(error_msg));
self.record(Key::new(format!("{}.chain", field.name())).array(chain));
}
}
impl<S, T> OpenTelemetryLayer<S, T>
where
S: Subscriber + for<'span> LookupSpan<'span>,
T: otel::Tracer + PreSampledTracer + 'static,
{
pub fn new(tracer: T) -> Self {
OpenTelemetryLayer {
tracer,
location: true,
tracked_inactivity: true,
with_threads: true,
exception_config: ExceptionFieldConfig {
record: false,
propagate: false,
},
get_context: WithContext(Self::get_context),
_registry: marker::PhantomData,
}
}
pub fn with_tracer<Tracer>(self, tracer: Tracer) -> OpenTelemetryLayer<S, Tracer>
where
Tracer: otel::Tracer + PreSampledTracer + 'static,
{
OpenTelemetryLayer {
tracer,
location: self.location,
tracked_inactivity: self.tracked_inactivity,
with_threads: self.with_threads,
exception_config: self.exception_config,
get_context: WithContext(OpenTelemetryLayer::<S, Tracer>::get_context),
_registry: self._registry,
}
}
pub fn with_exception_fields(self, exception_fields: bool) -> Self {
Self {
exception_config: ExceptionFieldConfig {
record: exception_fields,
..self.exception_config
},
..self
}
}
pub fn with_exception_field_propagation(self, exception_field_propagation: bool) -> Self {
Self {
exception_config: ExceptionFieldConfig {
propagate: exception_field_propagation,
..self.exception_config
},
..self
}
}
pub fn with_location(self, location: bool) -> Self {
Self { location, ..self }
}
#[deprecated(
since = "0.17.3",
note = "renamed to `OpenTelemetrySubscriber::with_location`"
)]
pub fn with_event_location(self, event_location: bool) -> Self {
Self {
location: event_location,
..self
}
}
pub fn with_tracked_inactivity(self, tracked_inactivity: bool) -> Self {
Self {
tracked_inactivity,
..self
}
}
pub fn with_threads(self, threads: bool) -> Self {
Self {
with_threads: threads,
..self
}
}
fn parent_context(&self, attrs: &Attributes<'_>, ctx: &Context<'_, S>) -> OtelContext {
if let Some(parent) = attrs.parent() {
let span = ctx.span(parent).expect("Span not found, this is a bug");
let mut extensions = span.extensions_mut();
extensions
.get_mut::<OtelData>()
.map(|builder| self.tracer.sampled_context(builder))
.unwrap_or_default()
} else if attrs.is_contextual() {
ctx.lookup_current()
.and_then(|span| {
let mut extensions = span.extensions_mut();
extensions
.get_mut::<OtelData>()
.map(|builder| self.tracer.sampled_context(builder))
})
.unwrap_or_else(OtelContext::current)
} else {
OtelContext::new()
}
}
fn get_context(
dispatch: &tracing::Dispatch,
id: &span::Id,
f: &mut dyn FnMut(&mut OtelData, &dyn PreSampledTracer),
) {
let subscriber = dispatch
.downcast_ref::<S>()
.expect("subscriber should downcast to expected type; this is a bug!");
let span = subscriber
.span(id)
.expect("registry should have a span for the current ID");
let layer = dispatch
.downcast_ref::<OpenTelemetryLayer<S, T>>()
.expect("layer should downcast to expected type; this is a bug!");
let mut extensions = span.extensions_mut();
if let Some(builder) = extensions.get_mut::<OtelData>() {
f(builder, &layer.tracer);
}
}
fn extra_span_attrs(&self) -> usize {
let mut extra_attrs = 0;
if self.location {
extra_attrs += 3;
}
if self.with_threads {
extra_attrs += 2;
}
extra_attrs
}
}
thread_local! {
static THREAD_ID: unsync::Lazy<u64> = unsync::Lazy::new(|| {
thread_id_integer(thread::current().id())
});
}
impl<S, T> Layer<S> for OpenTelemetryLayer<S, T>
where
S: Subscriber + for<'span> LookupSpan<'span>,
T: otel::Tracer + PreSampledTracer + 'static,
{
fn on_new_span(&self, attrs: &Attributes<'_>, id: &span::Id, ctx: Context<'_, S>) {
let span = ctx.span(id).expect("Span not found, this is a bug");
let mut extensions = span.extensions_mut();
if self.tracked_inactivity && extensions.get_mut::<Timings>().is_none() {
extensions.insert(Timings::new());
}
let parent_cx = self.parent_context(attrs, &ctx);
let mut builder = self
.tracer
.span_builder(attrs.metadata().name())
.with_start_time(SystemTime::now())
.with_span_id(self.tracer.new_span_id());
if !parent_cx.has_active_span() {
builder.trace_id = Some(self.tracer.new_trace_id());
}
let builder_attrs = builder.attributes.get_or_insert(Vec::with_capacity(
attrs.fields().len() + self.extra_span_attrs(),
));
if self.location {
let meta = attrs.metadata();
if let Some(filename) = meta.file() {
builder_attrs.push(KeyValue::new("code.filepath", filename));
}
if let Some(module) = meta.module_path() {
builder_attrs.push(KeyValue::new("code.namespace", module));
}
if let Some(line) = meta.line() {
builder_attrs.push(KeyValue::new("code.lineno", line as i64));
}
}
if self.with_threads {
THREAD_ID.with(|id| builder_attrs.push(KeyValue::new("thread.id", **id as i64)));
if let Some(name) = std::thread::current().name() {
builder_attrs.push(KeyValue::new("thread.name", name.to_owned()));
}
}
attrs.record(&mut SpanAttributeVisitor {
span_builder: &mut builder,
exception_config: self.exception_config,
});
extensions.insert(OtelData { builder, parent_cx });
}
fn on_enter(&self, id: &span::Id, ctx: Context<'_, S>) {
if !self.tracked_inactivity {
return;
}
let span = ctx.span(id).expect("Span not found, this is a bug");
let mut extensions = span.extensions_mut();
if let Some(timings) = extensions.get_mut::<Timings>() {
let now = Instant::now();
timings.idle += (now - timings.last).as_nanos() as i64;
timings.last = now;
}
}
fn on_exit(&self, id: &span::Id, ctx: Context<'_, S>) {
if !self.tracked_inactivity {
return;
}
let span = ctx.span(id).expect("Span not found, this is a bug");
let mut extensions = span.extensions_mut();
if let Some(timings) = extensions.get_mut::<Timings>() {
let now = Instant::now();
timings.busy += (now - timings.last).as_nanos() as i64;
timings.last = now;
}
}
fn on_record(&self, id: &Id, values: &Record<'_>, ctx: Context<'_, S>) {
let span = ctx.span(id).expect("Span not found, this is a bug");
let mut extensions = span.extensions_mut();
if let Some(data) = extensions.get_mut::<OtelData>() {
values.record(&mut SpanAttributeVisitor {
span_builder: &mut data.builder,
exception_config: self.exception_config,
});
}
}
fn on_follows_from(&self, id: &Id, follows: &Id, ctx: Context<S>) {
let span = ctx.span(id).expect("Span not found, this is a bug");
let mut extensions = span.extensions_mut();
let data = extensions
.get_mut::<OtelData>()
.expect("Missing otel data span extensions");
let follows_span = ctx
.span(follows)
.expect("Span to follow not found, this is a bug");
let mut follows_extensions = follows_span.extensions_mut();
let follows_data = follows_extensions
.get_mut::<OtelData>()
.expect("Missing otel data span extensions");
let follows_context = self
.tracer
.sampled_context(follows_data)
.span()
.span_context()
.clone();
let follows_link = otel::Link::new(follows_context, Vec::new());
if let Some(ref mut links) = data.builder.links {
links.push(follows_link);
} else {
data.builder.links = Some(vec![follows_link]);
}
}
fn on_event(&self, event: &Event<'_>, ctx: Context<'_, S>) {
if let Some(span) = ctx.lookup_current() {
#[cfg(feature = "tracing-log")]
let normalized_meta = event.normalized_metadata();
#[cfg(feature = "tracing-log")]
let meta = normalized_meta.as_ref().unwrap_or_else(|| event.metadata());
#[cfg(not(feature = "tracing-log"))]
let meta = event.metadata();
let target = Key::new("target");
#[cfg(feature = "tracing-log")]
let target = if normalized_meta.is_some() {
target.string(meta.target().to_owned())
} else {
target.string(event.metadata().target())
};
#[cfg(not(feature = "tracing-log"))]
let target = target.string(meta.target());
let mut extensions = span.extensions_mut();
let span_builder = extensions
.get_mut::<OtelData>()
.map(|data| &mut data.builder);
let mut otel_event = otel::Event::new(
String::new(),
SystemTime::now(),
vec![Key::new("level").string(meta.level().as_str()), target],
0,
);
event.record(&mut SpanEventVisitor {
event_builder: &mut otel_event,
span_builder,
exception_config: self.exception_config,
});
if let Some(OtelData { builder, .. }) = extensions.get_mut::<OtelData>() {
if builder.status_code.is_none() && *meta.level() == tracing_core::Level::ERROR {
builder.status_code = Some(otel::StatusCode::Error);
}
if self.location {
#[cfg(not(feature = "tracing-log"))]
let normalized_meta: Option<tracing_core::Metadata<'_>> = None;
let (file, module) = match &normalized_meta {
Some(meta) => (
meta.file().map(|s| Value::from(s.to_owned())),
meta.module_path().map(|s| Value::from(s.to_owned())),
),
None => (
event.metadata().file().map(Value::from),
event.metadata().module_path().map(Value::from),
),
};
if let Some(file) = file {
otel_event
.attributes
.push(KeyValue::new("code.filepath", file));
}
if let Some(module) = module {
otel_event
.attributes
.push(KeyValue::new("code.namespace", module));
}
if let Some(line) = meta.line() {
otel_event
.attributes
.push(KeyValue::new("code.lineno", line as i64));
}
}
if let Some(ref mut events) = builder.events {
events.push(otel_event);
} else {
builder.events = Some(vec![otel_event]);
}
}
};
}
fn on_close(&self, id: span::Id, ctx: Context<'_, S>) {
let span = ctx.span(&id).expect("Span not found, this is a bug");
let mut extensions = span.extensions_mut();
if let Some(OtelData {
mut builder,
parent_cx,
}) = extensions.remove::<OtelData>()
{
if self.tracked_inactivity {
if let Some(timings) = extensions.get_mut::<Timings>() {
let busy_ns = KeyValue::new("busy_ns", timings.busy);
let idle_ns = KeyValue::new("idle_ns", timings.idle);
if let Some(ref mut attributes) = builder.attributes {
attributes.push(busy_ns);
attributes.push(idle_ns);
} else {
builder.attributes = Some(vec![busy_ns, idle_ns]);
}
}
}
builder
.with_end_time(SystemTime::now())
.start_with_context(&self.tracer, &parent_cx);
}
}
unsafe fn downcast_raw(&self, id: TypeId) -> Option<*const ()> {
match id {
id if id == TypeId::of::<Self>() => Some(self as *const _ as *const ()),
id if id == TypeId::of::<WithContext>() => {
Some(&self.get_context as *const _ as *const ())
}
_ => None,
}
}
}
struct Timings {
idle: i64,
busy: i64,
last: Instant,
}
impl Timings {
fn new() -> Self {
Self {
idle: 0,
busy: 0,
last: Instant::now(),
}
}
}
fn thread_id_integer(id: thread::ThreadId) -> u64 {
let thread_id = format!("{:?}", id);
thread_id
.trim_start_matches("ThreadId(")
.trim_end_matches(')')
.parse::<u64>()
.expect("thread ID should parse as an integer")
}
#[cfg(test)]
mod tests {
use super::*;
use crate::OtelData;
use opentelemetry::trace::{noop, SpanKind, TraceFlags};
use std::{
borrow::Cow,
collections::HashMap,
error::Error,
fmt::Display,
sync::{Arc, Mutex},
thread,
time::SystemTime,
};
use tracing_subscriber::prelude::*;
#[derive(Debug, Clone)]
struct TestTracer(Arc<Mutex<Option<OtelData>>>);
impl otel::Tracer for TestTracer {
type Span = noop::NoopSpan;
fn start_with_context<T>(&self, _name: T, _context: &OtelContext) -> Self::Span
where
T: Into<Cow<'static, str>>,
{
noop::NoopSpan::new()
}
fn span_builder<T>(&self, name: T) -> otel::SpanBuilder
where
T: Into<Cow<'static, str>>,
{
otel::SpanBuilder::from_name(name)
}
fn build_with_context(
&self,
builder: otel::SpanBuilder,
parent_cx: &OtelContext,
) -> Self::Span {
*self.0.lock().unwrap() = Some(OtelData {
builder,
parent_cx: parent_cx.clone(),
});
noop::NoopSpan::new()
}
}
impl PreSampledTracer for TestTracer {
fn sampled_context(&self, _builder: &mut crate::OtelData) -> OtelContext {
OtelContext::new()
}
fn new_trace_id(&self) -> otel::TraceId {
otel::TraceId::INVALID
}
fn new_span_id(&self) -> otel::SpanId {
otel::SpanId::INVALID
}
}
impl TestTracer {
fn with_data<T>(&self, f: impl FnOnce(&OtelData) -> T) -> T {
let lock = self.0.lock().unwrap();
let data = lock.as_ref().expect("no span data has been recorded yet");
f(data)
}
}
#[derive(Debug, Clone)]
struct TestSpan(otel::SpanContext);
impl otel::Span for TestSpan {
fn add_event_with_timestamp<T: Into<Cow<'static, str>>>(
&mut self,
_: T,
_: SystemTime,
_: Vec<KeyValue>,
) {
}
fn span_context(&self) -> &otel::SpanContext {
&self.0
}
fn is_recording(&self) -> bool {
false
}
fn set_attribute(&mut self, _attribute: KeyValue) {}
fn set_status(&mut self, _code: otel::StatusCode, _message: String) {}
fn update_name<T: Into<Cow<'static, str>>>(&mut self, _new_name: T) {}
fn end_with_timestamp(&mut self, _timestamp: SystemTime) {}
}
#[derive(Debug)]
struct TestDynError {
msg: &'static str,
source: Option<Box<TestDynError>>,
}
impl Display for TestDynError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "{}", self.msg)
}
}
impl Error for TestDynError {
fn source(&self) -> Option<&(dyn Error + 'static)> {
match &self.source {
Some(source) => Some(source),
None => None,
}
}
}
impl TestDynError {
fn new(msg: &'static str) -> Self {
Self { msg, source: None }
}
fn with_parent(self, parent_msg: &'static str) -> Self {
Self {
msg: parent_msg,
source: Some(Box::new(self)),
}
}
}
#[test]
fn dynamic_span_names() {
let dynamic_name = "GET http://example.com".to_string();
let tracer = TestTracer(Arc::new(Mutex::new(None)));
let subscriber = tracing_subscriber::registry().with(layer().with_tracer(tracer.clone()));
tracing::subscriber::with_default(subscriber, || {
tracing::debug_span!("static_name", otel.name = dynamic_name.as_str());
});
let recorded_name = tracer
.0
.lock()
.unwrap()
.as_ref()
.map(|b| b.builder.name.clone());
assert_eq!(recorded_name, Some(dynamic_name.into()))
}
#[test]
fn span_kind() {
let tracer = TestTracer(Arc::new(Mutex::new(None)));
let subscriber = tracing_subscriber::registry().with(layer().with_tracer(tracer.clone()));
tracing::subscriber::with_default(subscriber, || {
tracing::debug_span!("request", otel.kind = %SpanKind::Server);
});
let recorded_kind = tracer.with_data(|data| data.builder.span_kind.clone());
assert_eq!(recorded_kind, Some(otel::SpanKind::Server))
}
#[test]
fn span_status_code() {
let tracer = TestTracer(Arc::new(Mutex::new(None)));
let subscriber = tracing_subscriber::registry().with(layer().with_tracer(tracer.clone()));
tracing::subscriber::with_default(subscriber, || {
tracing::debug_span!("request", otel.status_code = ?otel::StatusCode::Ok);
});
let recorded_status_code = tracer.with_data(|data| data.builder.status_code);
assert_eq!(recorded_status_code, Some(otel::StatusCode::Ok))
}
#[test]
fn span_status_message() {
let tracer = TestTracer(Arc::new(Mutex::new(None)));
let subscriber = tracing_subscriber::registry().with(layer().with_tracer(tracer.clone()));
let message = "message";
tracing::subscriber::with_default(subscriber, || {
tracing::debug_span!("request", otel.status_message = message);
});
let recorded_status_message = tracer.with_data(|data| data.builder.status_message.clone());
assert_eq!(recorded_status_message, Some(message.into()))
}
#[test]
fn trace_id_from_existing_context() {
let tracer = TestTracer(Arc::new(Mutex::new(None)));
let subscriber = tracing_subscriber::registry().with(layer().with_tracer(tracer.clone()));
let trace_id = otel::TraceId::from(42u128.to_be_bytes());
let existing_cx = OtelContext::current_with_span(TestSpan(otel::SpanContext::new(
trace_id,
otel::SpanId::from(1u64.to_be_bytes()),
TraceFlags::default(),
false,
Default::default(),
)));
let _g = existing_cx.attach();
tracing::subscriber::with_default(subscriber, || {
tracing::debug_span!("request", otel.kind = %SpanKind::Server);
});
let recorded_trace_id =
tracer.with_data(|data| data.parent_cx.span().span_context().trace_id());
assert_eq!(recorded_trace_id, trace_id)
}
#[test]
fn includes_timings() {
let tracer = TestTracer(Arc::new(Mutex::new(None)));
let subscriber = tracing_subscriber::registry().with(
layer()
.with_tracer(tracer.clone())
.with_tracked_inactivity(true),
);
tracing::subscriber::with_default(subscriber, || {
tracing::debug_span!("request");
});
let attributes = tracer.with_data(|data| data.builder.attributes.as_ref().unwrap().clone());
let keys = attributes
.iter()
.map(|attr| attr.key.as_str())
.collect::<Vec<&str>>();
assert!(keys.contains(&"idle_ns"));
assert!(keys.contains(&"busy_ns"));
}
#[test]
fn records_error_fields() {
let tracer = TestTracer(Arc::new(Mutex::new(None)));
let subscriber = tracing_subscriber::registry().with(
layer()
.with_tracer(tracer.clone())
.with_exception_fields(true),
);
let err = TestDynError::new("base error")
.with_parent("intermediate error")
.with_parent("user error");
tracing::subscriber::with_default(subscriber, || {
tracing::debug_span!(
"request",
error = &err as &(dyn std::error::Error + 'static)
);
});
let attributes = tracer
.0
.lock()
.unwrap()
.as_ref()
.unwrap()
.builder
.attributes
.as_ref()
.unwrap()
.clone();
let key_values = attributes
.into_iter()
.map(|attr| (attr.key.as_str().to_owned(), attr.value))
.collect::<HashMap<_, _>>();
assert_eq!(key_values["error"].as_str(), "user error");
assert_eq!(
key_values["error.chain"],
Value::Array(
vec![
Cow::Borrowed("intermediate error"),
Cow::Borrowed("base error")
]
.into()
)
);
assert_eq!(key_values[FIELD_EXCEPTION_MESSAGE].as_str(), "user error");
assert_eq!(
key_values[FIELD_EXCEPTION_STACKTRACE],
Value::Array(
vec![
Cow::Borrowed("intermediate error"),
Cow::Borrowed("base error")
]
.into()
)
);
}
#[test]
fn includes_span_location() {
let tracer = TestTracer(Arc::new(Mutex::new(None)));
let subscriber = tracing_subscriber::registry()
.with(layer().with_tracer(tracer.clone()).with_location(true));
tracing::subscriber::with_default(subscriber, || {
tracing::debug_span!("request");
});
let attributes = tracer.with_data(|data| data.builder.attributes.as_ref().unwrap().clone());
let keys = attributes
.iter()
.map(|attr| attr.key.as_str())
.collect::<Vec<&str>>();
assert!(keys.contains(&"code.filepath"));
assert!(keys.contains(&"code.namespace"));
assert!(keys.contains(&"code.lineno"));
}
#[test]
fn excludes_span_location() {
let tracer = TestTracer(Arc::new(Mutex::new(None)));
let subscriber = tracing_subscriber::registry()
.with(layer().with_tracer(tracer.clone()).with_location(false));
tracing::subscriber::with_default(subscriber, || {
tracing::debug_span!("request");
});
let attributes = tracer.with_data(|data| data.builder.attributes.as_ref().unwrap().clone());
let keys = attributes
.iter()
.map(|attr| attr.key.as_str())
.collect::<Vec<&str>>();
assert!(!keys.contains(&"code.filepath"));
assert!(!keys.contains(&"code.namespace"));
assert!(!keys.contains(&"code.lineno"));
}
#[test]
fn includes_thread() {
let thread = thread::current();
let expected_name = thread
.name()
.map(|name| Value::String(Cow::Owned(name.to_owned())));
let expected_id = Value::I64(thread_id_integer(thread.id()) as i64);
let tracer = TestTracer(Arc::new(Mutex::new(None)));
let subscriber = tracing_subscriber::registry()
.with(layer().with_tracer(tracer.clone()).with_threads(true));
tracing::subscriber::with_default(subscriber, || {
tracing::debug_span!("request");
});
let attributes = tracer
.with_data(|data| data.builder.attributes.as_ref().unwrap().clone())
.drain(..)
.map(|keyval| (keyval.key.as_str().to_string(), keyval.value))
.collect::<HashMap<_, _>>();
assert_eq!(attributes.get("thread.name"), expected_name.as_ref());
assert_eq!(attributes.get("thread.id"), Some(&expected_id));
}
#[test]
fn excludes_thread() {
let tracer = TestTracer(Arc::new(Mutex::new(None)));
let subscriber = tracing_subscriber::registry()
.with(layer().with_tracer(tracer.clone()).with_threads(false));
tracing::subscriber::with_default(subscriber, || {
tracing::debug_span!("request");
});
let attributes = tracer.with_data(|data| data.builder.attributes.as_ref().unwrap().clone());
let keys = attributes
.iter()
.map(|attr| attr.key.as_str())
.collect::<Vec<&str>>();
assert!(!keys.contains(&"thread.name"));
assert!(!keys.contains(&"thread.id"));
}
#[test]
fn propagates_error_fields_from_event_to_span() {
let tracer = TestTracer(Arc::new(Mutex::new(None)));
let subscriber = tracing_subscriber::registry().with(
layer()
.with_tracer(tracer.clone())
.with_exception_field_propagation(true),
);
let err = TestDynError::new("base error")
.with_parent("intermediate error")
.with_parent("user error");
tracing::subscriber::with_default(subscriber, || {
let _guard = tracing::debug_span!("request",).entered();
tracing::error!(
error = &err as &(dyn std::error::Error + 'static),
"request error!"
)
});
let attributes = tracer
.0
.lock()
.unwrap()
.as_ref()
.unwrap()
.builder
.attributes
.as_ref()
.unwrap()
.clone();
let key_values = attributes
.into_iter()
.map(|attr| (attr.key.as_str().to_owned(), attr.value))
.collect::<HashMap<_, _>>();
assert_eq!(key_values[FIELD_EXCEPTION_MESSAGE].as_str(), "user error");
assert_eq!(
key_values[FIELD_EXCEPTION_STACKTRACE],
Value::Array(
vec![
Cow::Borrowed("intermediate error"),
Cow::Borrowed("base error")
]
.into()
)
);
}
}