use libdd_trace_protobuf::opentelemetry::proto as otel_proto;
use std::default::Default;
#[derive(serde::Serialize, Debug)]
pub struct TracerMetadata {
pub schema_version: u8,
#[serde(skip_serializing_if = "Option::is_none")]
pub runtime_id: Option<String>,
pub tracer_language: String,
pub tracer_version: String,
pub hostname: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub service_name: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub service_env: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub service_version: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub process_tags: Option<String>,
#[serde(skip_serializing_if = "Option::is_none")]
pub container_id: Option<String>,
}
impl Default for TracerMetadata {
fn default() -> Self {
TracerMetadata {
schema_version: 2,
runtime_id: None,
tracer_language: String::new(),
tracer_version: String::new(),
hostname: String::new(),
service_name: None,
service_env: None,
service_version: None,
process_tags: None,
container_id: None,
}
}
}
impl TracerMetadata {
const OTEL_SDK_NAME: &str = "libdatadog";
pub fn to_otel_process_ctx(&self) -> otel_proto::common::v1::ProcessContext {
use otel_proto::common::v1::{any_value, AnyValue, KeyValue};
fn key_value(key: &'static str, val: String) -> KeyValue {
KeyValue {
key: key.to_owned(),
value: Some(AnyValue {
value: Some(any_value::Value::StringValue(val)),
}),
key_ref: 0,
}
}
fn key_value_opt(key: &'static str, val: &Option<String>) -> KeyValue {
key_value(key, val.as_ref().cloned().unwrap_or_default())
}
let TracerMetadata {
schema_version: _,
runtime_id,
tracer_language,
tracer_version,
hostname,
service_name,
service_env,
service_version,
process_tags,
container_id,
} = self;
otel_proto::common::v1::ProcessContext {
resource: Some(otel_proto::resource::v1::Resource {
attributes: vec![
key_value_opt("service.name", service_name),
key_value_opt("service.instance.id", runtime_id),
key_value_opt("service.version", service_version),
key_value_opt("deployment.environment.name", service_env),
key_value("telemetry.sdk.language", tracer_language.clone()),
key_value("telemetry.sdk.version", tracer_version.clone()),
key_value("telemetry.sdk.name", Self::OTEL_SDK_NAME.to_owned()),
key_value("host.name", hostname.clone()),
key_value_opt("container.id", container_id),
],
dropped_attributes_count: 0,
entity_refs: vec![],
}),
extra_attributes: vec![key_value_opt("datadog.process_tags", process_tags)],
}
}
}
pub enum AnonymousFileHandle {
#[cfg(target_os = "linux")]
Linux(memfd::Memfd),
#[cfg(not(target_os = "linux"))]
Other(()),
}
#[cfg(target_os = "linux")]
mod linux {
use anyhow::Context;
use rand::distributions::Alphanumeric;
use rand::Rng;
use std::io::Write;
pub fn store_tracer_metadata(
data: &super::TracerMetadata,
) -> anyhow::Result<super::AnonymousFileHandle> {
let _ = crate::otel_process_ctx::linux::publish(&data.to_otel_process_ctx());
let uid: String = rand::thread_rng()
.sample_iter(&Alphanumeric)
.take(8)
.map(char::from)
.collect();
let mfd_name: String = format!("datadog-tracer-info-{uid}");
let mfd = memfd::MemfdOptions::default()
.close_on_exec(true)
.allow_sealing(true)
.create::<&str>(mfd_name.as_ref())
.context("unable to create memfd")?;
let buf = rmp_serde::to_vec_named(data).context("failed serialization")?;
mfd.as_file()
.write_all(&buf)
.context("unable to write into memfd")?;
mfd.add_seals(&[
memfd::FileSeal::SealShrink,
memfd::FileSeal::SealGrow,
memfd::FileSeal::SealSeal,
])
.context("unable to seal memfd")?;
Ok(super::AnonymousFileHandle::Linux(mfd))
}
}
#[cfg(not(target_os = "linux"))]
mod other {
pub fn store_tracer_metadata(
_data: &super::TracerMetadata,
) -> anyhow::Result<super::AnonymousFileHandle> {
Ok(super::AnonymousFileHandle::Other(()))
}
}
#[cfg(target_os = "linux")]
pub use linux::*;
#[cfg(not(target_os = "linux"))]
pub use other::*;