use anyhow::{Context, Result};
use opentelemetry::trace::TracerProvider as _;
use opentelemetry::{KeyValue, global};
use opentelemetry_otlp::{SpanExporter, WithExportConfig};
use opentelemetry_sdk::Resource;
use opentelemetry_sdk::runtime;
use opentelemetry_sdk::trace::TracerProvider;
use opentelemetry_semantic_conventions::resource::SERVICE_NAME;
use tracing_subscriber::EnvFilter;
use tracing_subscriber::layer::SubscriberExt;
use tracing_subscriber::util::SubscriberInitExt;
pub struct TracerGuard {
provider: TracerProvider,
}
impl Drop for TracerGuard {
fn drop(&mut self) {
if let Err(err) = self.provider.shutdown() {
eprintln!("lightshuttle-otel: tracer flush on shutdown failed: {err}");
}
}
}
pub fn init_orchestrator_tracer(endpoint: &str, service: &str) -> Result<TracerGuard> {
let exporter = SpanExporter::builder()
.with_tonic()
.with_endpoint(endpoint.to_owned())
.build()
.context("failed to build the OTLP span exporter")?;
let resource = Resource::new(vec![KeyValue::new(SERVICE_NAME, service.to_owned())]);
let provider = TracerProvider::builder()
.with_batch_exporter(exporter, runtime::Tokio)
.with_resource(resource)
.build();
global::set_tracer_provider(provider.clone());
let tracer = provider.tracer("lightshuttle");
let otel_layer = tracing_opentelemetry::layer().with_tracer(tracer);
let filter =
EnvFilter::try_from_env("LIGHTSHUTTLE_LOG").unwrap_or_else(|_| EnvFilter::new("info"));
tracing_subscriber::registry()
.with(filter)
.with(
tracing_subscriber::fmt::layer()
.with_target(false)
.with_level(true)
.compact(),
)
.with(otel_layer)
.try_init()
.context("a global tracing subscriber is already installed")?;
Ok(TracerGuard { provider })
}