mirror of
https://github.com/denoland/deno.git
synced 2025-01-21 13:00:36 -05:00
feat(unstable): add metrics to otel (#27143)
Refs: https://github.com/denoland/deno/issues/26852 Initial support for exporting metrics. Co-authored-by: Luca Casonato <hello@lcas.dev>
This commit is contained in:
parent
6dd2d5e49e
commit
7c036772df
11 changed files with 1174 additions and 35 deletions
|
@ -16,6 +16,7 @@ use once_cell::sync::OnceCell;
|
||||||
use opentelemetry::logs::AnyValue;
|
use opentelemetry::logs::AnyValue;
|
||||||
use opentelemetry::logs::LogRecord as LogRecordTrait;
|
use opentelemetry::logs::LogRecord as LogRecordTrait;
|
||||||
use opentelemetry::logs::Severity;
|
use opentelemetry::logs::Severity;
|
||||||
|
use opentelemetry::otel_error;
|
||||||
use opentelemetry::trace::SpanContext;
|
use opentelemetry::trace::SpanContext;
|
||||||
use opentelemetry::trace::SpanId;
|
use opentelemetry::trace::SpanId;
|
||||||
use opentelemetry::trace::SpanKind;
|
use opentelemetry::trace::SpanKind;
|
||||||
|
@ -27,15 +28,21 @@ use opentelemetry::KeyValue;
|
||||||
use opentelemetry::StringValue;
|
use opentelemetry::StringValue;
|
||||||
use opentelemetry::Value;
|
use opentelemetry::Value;
|
||||||
use opentelemetry_otlp::HttpExporterBuilder;
|
use opentelemetry_otlp::HttpExporterBuilder;
|
||||||
|
use opentelemetry_otlp::MetricExporter;
|
||||||
use opentelemetry_otlp::Protocol;
|
use opentelemetry_otlp::Protocol;
|
||||||
use opentelemetry_otlp::WithExportConfig;
|
use opentelemetry_otlp::WithExportConfig;
|
||||||
use opentelemetry_otlp::WithHttpConfig;
|
use opentelemetry_otlp::WithHttpConfig;
|
||||||
use opentelemetry_sdk::export::trace::SpanData;
|
use opentelemetry_sdk::export::trace::SpanData;
|
||||||
use opentelemetry_sdk::logs::BatchLogProcessor;
|
use opentelemetry_sdk::logs::BatchLogProcessor;
|
||||||
use opentelemetry_sdk::logs::LogProcessor as LogProcessorTrait;
|
use opentelemetry_sdk::logs::LogProcessor;
|
||||||
use opentelemetry_sdk::logs::LogRecord;
|
use opentelemetry_sdk::logs::LogRecord;
|
||||||
|
use opentelemetry_sdk::metrics::data::Metric;
|
||||||
|
use opentelemetry_sdk::metrics::data::ResourceMetrics;
|
||||||
|
use opentelemetry_sdk::metrics::data::ScopeMetrics;
|
||||||
|
use opentelemetry_sdk::metrics::exporter::PushMetricExporter;
|
||||||
|
use opentelemetry_sdk::metrics::Temporality;
|
||||||
use opentelemetry_sdk::trace::BatchSpanProcessor;
|
use opentelemetry_sdk::trace::BatchSpanProcessor;
|
||||||
use opentelemetry_sdk::trace::SpanProcessor as SpanProcessorTrait;
|
use opentelemetry_sdk::trace::SpanProcessor;
|
||||||
use opentelemetry_sdk::Resource;
|
use opentelemetry_sdk::Resource;
|
||||||
use opentelemetry_semantic_conventions::resource::PROCESS_RUNTIME_NAME;
|
use opentelemetry_semantic_conventions::resource::PROCESS_RUNTIME_NAME;
|
||||||
use opentelemetry_semantic_conventions::resource::PROCESS_RUNTIME_VERSION;
|
use opentelemetry_semantic_conventions::resource::PROCESS_RUNTIME_VERSION;
|
||||||
|
@ -54,9 +61,6 @@ use std::thread;
|
||||||
use std::time::Duration;
|
use std::time::Duration;
|
||||||
use std::time::SystemTime;
|
use std::time::SystemTime;
|
||||||
|
|
||||||
type SpanProcessor = BatchSpanProcessor<OtelSharedRuntime>;
|
|
||||||
type LogProcessor = BatchLogProcessor<OtelSharedRuntime>;
|
|
||||||
|
|
||||||
deno_core::extension!(
|
deno_core::extension!(
|
||||||
deno_telemetry,
|
deno_telemetry,
|
||||||
ops = [
|
ops = [
|
||||||
|
@ -71,6 +75,23 @@ deno_core::extension!(
|
||||||
op_otel_span_attribute3,
|
op_otel_span_attribute3,
|
||||||
op_otel_span_set_dropped,
|
op_otel_span_set_dropped,
|
||||||
op_otel_span_flush,
|
op_otel_span_flush,
|
||||||
|
op_otel_metrics_resource_attribute,
|
||||||
|
op_otel_metrics_resource_attribute2,
|
||||||
|
op_otel_metrics_resource_attribute3,
|
||||||
|
op_otel_metrics_scope,
|
||||||
|
op_otel_metrics_sum,
|
||||||
|
op_otel_metrics_gauge,
|
||||||
|
op_otel_metrics_sum_or_gauge_data_point,
|
||||||
|
op_otel_metrics_histogram,
|
||||||
|
op_otel_metrics_histogram_data_point,
|
||||||
|
op_otel_metrics_histogram_data_point_entry_final,
|
||||||
|
op_otel_metrics_histogram_data_point_entry1,
|
||||||
|
op_otel_metrics_histogram_data_point_entry2,
|
||||||
|
op_otel_metrics_histogram_data_point_entry3,
|
||||||
|
op_otel_metrics_data_point_attribute,
|
||||||
|
op_otel_metrics_data_point_attribute2,
|
||||||
|
op_otel_metrics_data_point_attribute3,
|
||||||
|
op_otel_metrics_submit,
|
||||||
],
|
],
|
||||||
esm = ["telemetry.ts", "util.ts"],
|
esm = ["telemetry.ts", "util.ts"],
|
||||||
);
|
);
|
||||||
|
@ -322,8 +343,69 @@ mod hyper_client {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static OTEL_PROCESSORS: OnceCell<(SpanProcessor, LogProcessor)> =
|
enum MetricProcessorMessage {
|
||||||
OnceCell::new();
|
ResourceMetrics(ResourceMetrics),
|
||||||
|
Flush(tokio::sync::oneshot::Sender<()>),
|
||||||
|
}
|
||||||
|
|
||||||
|
struct MetricProcessor {
|
||||||
|
tx: tokio::sync::mpsc::Sender<MetricProcessorMessage>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl MetricProcessor {
|
||||||
|
fn new(exporter: MetricExporter) -> Self {
|
||||||
|
let (tx, mut rx) = tokio::sync::mpsc::channel(2048);
|
||||||
|
let future = async move {
|
||||||
|
while let Some(message) = rx.recv().await {
|
||||||
|
match message {
|
||||||
|
MetricProcessorMessage::ResourceMetrics(mut rm) => {
|
||||||
|
if let Err(err) = exporter.export(&mut rm).await {
|
||||||
|
otel_error!(
|
||||||
|
name: "MetricProcessor.Export.Error",
|
||||||
|
error = format!("{}", err)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
MetricProcessorMessage::Flush(tx) => {
|
||||||
|
if let Err(()) = tx.send(()) {
|
||||||
|
otel_error!(
|
||||||
|
name: "MetricProcessor.Flush.SendResultError",
|
||||||
|
error = "()",
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
(*OTEL_SHARED_RUNTIME_SPAWN_TASK_TX)
|
||||||
|
.unbounded_send(Box::pin(future))
|
||||||
|
.expect("failed to send task to shared OpenTelemetry runtime");
|
||||||
|
|
||||||
|
Self { tx }
|
||||||
|
}
|
||||||
|
|
||||||
|
fn submit(&self, rm: ResourceMetrics) {
|
||||||
|
let _ = self
|
||||||
|
.tx
|
||||||
|
.try_send(MetricProcessorMessage::ResourceMetrics(rm));
|
||||||
|
}
|
||||||
|
|
||||||
|
fn force_flush(&self) -> Result<(), anyhow::Error> {
|
||||||
|
let (tx, rx) = tokio::sync::oneshot::channel();
|
||||||
|
self.tx.try_send(MetricProcessorMessage::Flush(tx))?;
|
||||||
|
deno_core::futures::executor::block_on(rx)?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Processors {
|
||||||
|
spans: BatchSpanProcessor<OtelSharedRuntime>,
|
||||||
|
logs: BatchLogProcessor<OtelSharedRuntime>,
|
||||||
|
metrics: MetricProcessor,
|
||||||
|
}
|
||||||
|
|
||||||
|
static OTEL_PROCESSORS: OnceCell<Processors> = OnceCell::new();
|
||||||
|
|
||||||
static BUILT_IN_INSTRUMENTATION_SCOPE: OnceCell<
|
static BUILT_IN_INSTRUMENTATION_SCOPE: OnceCell<
|
||||||
opentelemetry::InstrumentationScope,
|
opentelemetry::InstrumentationScope,
|
||||||
|
@ -404,6 +486,12 @@ pub fn init(config: OtelConfig) -> anyhow::Result<()> {
|
||||||
BatchSpanProcessor::builder(span_exporter, OtelSharedRuntime).build();
|
BatchSpanProcessor::builder(span_exporter, OtelSharedRuntime).build();
|
||||||
span_processor.set_resource(&resource);
|
span_processor.set_resource(&resource);
|
||||||
|
|
||||||
|
let metric_exporter = HttpExporterBuilder::default()
|
||||||
|
.with_http_client(client.clone())
|
||||||
|
.with_protocol(protocol)
|
||||||
|
.build_metrics_exporter(Temporality::Cumulative)?;
|
||||||
|
let metric_processor = MetricProcessor::new(metric_exporter);
|
||||||
|
|
||||||
let log_exporter = HttpExporterBuilder::default()
|
let log_exporter = HttpExporterBuilder::default()
|
||||||
.with_http_client(client)
|
.with_http_client(client)
|
||||||
.with_protocol(protocol)
|
.with_protocol(protocol)
|
||||||
|
@ -413,7 +501,11 @@ pub fn init(config: OtelConfig) -> anyhow::Result<()> {
|
||||||
log_processor.set_resource(&resource);
|
log_processor.set_resource(&resource);
|
||||||
|
|
||||||
OTEL_PROCESSORS
|
OTEL_PROCESSORS
|
||||||
.set((span_processor, log_processor))
|
.set(Processors {
|
||||||
|
spans: span_processor,
|
||||||
|
logs: log_processor,
|
||||||
|
metrics: metric_processor,
|
||||||
|
})
|
||||||
.map_err(|_| anyhow!("failed to init otel"))?;
|
.map_err(|_| anyhow!("failed to init otel"))?;
|
||||||
|
|
||||||
let builtin_instrumentation_scope =
|
let builtin_instrumentation_scope =
|
||||||
|
@ -431,16 +523,22 @@ pub fn init(config: OtelConfig) -> anyhow::Result<()> {
|
||||||
/// `process::exit()`, to ensure that all OpenTelemetry logs are properly
|
/// `process::exit()`, to ensure that all OpenTelemetry logs are properly
|
||||||
/// flushed before the process terminates.
|
/// flushed before the process terminates.
|
||||||
pub fn flush() {
|
pub fn flush() {
|
||||||
if let Some((span_processor, log_processor)) = OTEL_PROCESSORS.get() {
|
if let Some(Processors {
|
||||||
let _ = span_processor.force_flush();
|
spans,
|
||||||
let _ = log_processor.force_flush();
|
logs,
|
||||||
|
metrics,
|
||||||
|
}) = OTEL_PROCESSORS.get()
|
||||||
|
{
|
||||||
|
let _ = spans.force_flush();
|
||||||
|
let _ = logs.force_flush();
|
||||||
|
let _ = metrics.force_flush();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn handle_log(record: &log::Record) {
|
pub fn handle_log(record: &log::Record) {
|
||||||
use log::Level;
|
use log::Level;
|
||||||
|
|
||||||
let Some((_, log_processor)) = OTEL_PROCESSORS.get() else {
|
let Some(Processors { logs, .. }) = OTEL_PROCESSORS.get() else {
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -490,7 +588,7 @@ pub fn handle_log(record: &log::Record) {
|
||||||
|
|
||||||
let _ = record.key_values().visit(&mut Visitor(&mut log_record));
|
let _ = record.key_values().visit(&mut Visitor(&mut log_record));
|
||||||
|
|
||||||
log_processor.emit(
|
logs.emit(
|
||||||
&mut log_record,
|
&mut log_record,
|
||||||
BUILT_IN_INSTRUMENTATION_SCOPE.get().unwrap(),
|
BUILT_IN_INSTRUMENTATION_SCOPE.get().unwrap(),
|
||||||
);
|
);
|
||||||
|
@ -648,7 +746,7 @@ fn op_otel_log(
|
||||||
span_id: v8::Local<'_, v8::Value>,
|
span_id: v8::Local<'_, v8::Value>,
|
||||||
#[smi] trace_flags: u8,
|
#[smi] trace_flags: u8,
|
||||||
) {
|
) {
|
||||||
let Some((_, log_processor)) = OTEL_PROCESSORS.get() else {
|
let Some(Processors { logs, .. }) = OTEL_PROCESSORS.get() else {
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -678,12 +776,25 @@ fn op_otel_log(
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
log_processor.emit(
|
logs.emit(
|
||||||
&mut log_record,
|
&mut log_record,
|
||||||
BUILT_IN_INSTRUMENTATION_SCOPE.get().unwrap(),
|
BUILT_IN_INSTRUMENTATION_SCOPE.get().unwrap(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn owned_string<'s>(
|
||||||
|
scope: &mut v8::HandleScope<'s>,
|
||||||
|
string: v8::Local<'s, v8::String>,
|
||||||
|
) -> String {
|
||||||
|
let x = v8::ValueView::new(scope, string);
|
||||||
|
match x.data() {
|
||||||
|
v8::ValueViewData::OneByte(bytes) => {
|
||||||
|
String::from_utf8_lossy(bytes).into_owned()
|
||||||
|
}
|
||||||
|
v8::ValueViewData::TwoByte(bytes) => String::from_utf16_lossy(bytes),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
struct TemporarySpan(SpanData);
|
struct TemporarySpan(SpanData);
|
||||||
|
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
|
@ -700,10 +811,10 @@ fn op_otel_span_start<'s>(
|
||||||
end_time: f64,
|
end_time: f64,
|
||||||
) -> Result<(), anyhow::Error> {
|
) -> Result<(), anyhow::Error> {
|
||||||
if let Some(temporary_span) = state.try_take::<TemporarySpan>() {
|
if let Some(temporary_span) = state.try_take::<TemporarySpan>() {
|
||||||
let Some((span_processor, _)) = OTEL_PROCESSORS.get() else {
|
let Some(Processors { spans, .. }) = OTEL_PROCESSORS.get() else {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
};
|
};
|
||||||
span_processor.on_end(temporary_span.0);
|
spans.on_end(temporary_span.0);
|
||||||
};
|
};
|
||||||
|
|
||||||
let Some(InstrumentationScope(instrumentation_scope)) =
|
let Some(InstrumentationScope(instrumentation_scope)) =
|
||||||
|
@ -724,15 +835,7 @@ fn op_otel_span_start<'s>(
|
||||||
|
|
||||||
let parent_span_id = parse_span_id(scope, parent_span_id);
|
let parent_span_id = parse_span_id(scope, parent_span_id);
|
||||||
|
|
||||||
let name = {
|
let name = owned_string(scope, name.try_cast()?);
|
||||||
let x = v8::ValueView::new(scope, name.try_cast()?);
|
|
||||||
match x.data() {
|
|
||||||
v8::ValueViewData::OneByte(bytes) => {
|
|
||||||
String::from_utf8_lossy(bytes).into_owned()
|
|
||||||
}
|
|
||||||
v8::ValueViewData::TwoByte(bytes) => String::from_utf16_lossy(bytes),
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let temporary_span = TemporarySpan(SpanData {
|
let temporary_span = TemporarySpan(SpanData {
|
||||||
span_context: SpanContext::new(
|
span_context: SpanContext::new(
|
||||||
|
@ -866,9 +969,598 @@ fn op_otel_span_flush(state: &mut OpState) {
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
let Some((span_processor, _)) = OTEL_PROCESSORS.get() else {
|
let Some(Processors { spans, .. }) = OTEL_PROCESSORS.get() else {
|
||||||
return;
|
return;
|
||||||
};
|
};
|
||||||
|
|
||||||
span_processor.on_end(temporary_span.0);
|
spans.on_end(temporary_span.0);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Holds data being built from JS before
|
||||||
|
// it is submitted to the rust processor.
|
||||||
|
struct TemporaryMetricsExport {
|
||||||
|
resource_attributes: Vec<KeyValue>,
|
||||||
|
scope_metrics: Vec<ScopeMetrics>,
|
||||||
|
metric: Option<TemporaryMetric>,
|
||||||
|
}
|
||||||
|
|
||||||
|
struct TemporaryMetric {
|
||||||
|
name: String,
|
||||||
|
description: String,
|
||||||
|
unit: String,
|
||||||
|
data: TemporaryMetricData,
|
||||||
|
}
|
||||||
|
|
||||||
|
enum TemporaryMetricData {
|
||||||
|
Sum(opentelemetry_sdk::metrics::data::Sum<f64>),
|
||||||
|
Gauge(opentelemetry_sdk::metrics::data::Gauge<f64>),
|
||||||
|
Histogram(opentelemetry_sdk::metrics::data::Histogram<f64>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<TemporaryMetric> for Metric {
|
||||||
|
fn from(value: TemporaryMetric) -> Self {
|
||||||
|
Metric {
|
||||||
|
name: Cow::Owned(value.name),
|
||||||
|
description: Cow::Owned(value.description),
|
||||||
|
unit: Cow::Owned(value.unit),
|
||||||
|
data: match value.data {
|
||||||
|
TemporaryMetricData::Sum(sum) => Box::new(sum),
|
||||||
|
TemporaryMetricData::Gauge(gauge) => Box::new(gauge),
|
||||||
|
TemporaryMetricData::Histogram(histogram) => Box::new(histogram),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_resource_attribute<'s>(
|
||||||
|
scope: &mut v8::HandleScope<'s>,
|
||||||
|
state: &mut OpState,
|
||||||
|
#[smi] capacity: u32,
|
||||||
|
key: v8::Local<'s, v8::Value>,
|
||||||
|
value: v8::Local<'s, v8::Value>,
|
||||||
|
) {
|
||||||
|
let metrics_export = if let Some(metrics_export) =
|
||||||
|
state.try_borrow_mut::<TemporaryMetricsExport>()
|
||||||
|
{
|
||||||
|
metrics_export.resource_attributes.reserve_exact(
|
||||||
|
(capacity as usize) - metrics_export.resource_attributes.capacity(),
|
||||||
|
);
|
||||||
|
metrics_export
|
||||||
|
} else {
|
||||||
|
state.put(TemporaryMetricsExport {
|
||||||
|
resource_attributes: Vec::with_capacity(capacity as usize),
|
||||||
|
scope_metrics: vec![],
|
||||||
|
metric: None,
|
||||||
|
});
|
||||||
|
state.borrow_mut()
|
||||||
|
};
|
||||||
|
attr!(scope, metrics_export.resource_attributes, key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_resource_attribute2<'s>(
|
||||||
|
scope: &mut v8::HandleScope<'s>,
|
||||||
|
state: &mut OpState,
|
||||||
|
#[smi] capacity: u32,
|
||||||
|
key1: v8::Local<'s, v8::Value>,
|
||||||
|
value1: v8::Local<'s, v8::Value>,
|
||||||
|
key2: v8::Local<'s, v8::Value>,
|
||||||
|
value2: v8::Local<'s, v8::Value>,
|
||||||
|
) {
|
||||||
|
let metrics_export = if let Some(metrics_export) =
|
||||||
|
state.try_borrow_mut::<TemporaryMetricsExport>()
|
||||||
|
{
|
||||||
|
metrics_export.resource_attributes.reserve_exact(
|
||||||
|
(capacity as usize) - metrics_export.resource_attributes.capacity(),
|
||||||
|
);
|
||||||
|
metrics_export
|
||||||
|
} else {
|
||||||
|
state.put(TemporaryMetricsExport {
|
||||||
|
resource_attributes: Vec::with_capacity(capacity as usize),
|
||||||
|
scope_metrics: vec![],
|
||||||
|
metric: None,
|
||||||
|
});
|
||||||
|
state.borrow_mut()
|
||||||
|
};
|
||||||
|
attr!(scope, metrics_export.resource_attributes, key1, value1);
|
||||||
|
attr!(scope, metrics_export.resource_attributes, key2, value2);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::too_many_arguments)]
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_resource_attribute3<'s>(
|
||||||
|
scope: &mut v8::HandleScope<'s>,
|
||||||
|
state: &mut OpState,
|
||||||
|
#[smi] capacity: u32,
|
||||||
|
key1: v8::Local<'s, v8::Value>,
|
||||||
|
value1: v8::Local<'s, v8::Value>,
|
||||||
|
key2: v8::Local<'s, v8::Value>,
|
||||||
|
value2: v8::Local<'s, v8::Value>,
|
||||||
|
key3: v8::Local<'s, v8::Value>,
|
||||||
|
value3: v8::Local<'s, v8::Value>,
|
||||||
|
) {
|
||||||
|
let metrics_export = if let Some(metrics_export) =
|
||||||
|
state.try_borrow_mut::<TemporaryMetricsExport>()
|
||||||
|
{
|
||||||
|
metrics_export.resource_attributes.reserve_exact(
|
||||||
|
(capacity as usize) - metrics_export.resource_attributes.capacity(),
|
||||||
|
);
|
||||||
|
metrics_export
|
||||||
|
} else {
|
||||||
|
state.put(TemporaryMetricsExport {
|
||||||
|
resource_attributes: Vec::with_capacity(capacity as usize),
|
||||||
|
scope_metrics: vec![],
|
||||||
|
metric: None,
|
||||||
|
});
|
||||||
|
state.borrow_mut()
|
||||||
|
};
|
||||||
|
attr!(scope, metrics_export.resource_attributes, key1, value1);
|
||||||
|
attr!(scope, metrics_export.resource_attributes, key2, value2);
|
||||||
|
attr!(scope, metrics_export.resource_attributes, key3, value3);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_scope<'s>(
|
||||||
|
scope: &mut v8::HandleScope<'s>,
|
||||||
|
state: &mut OpState,
|
||||||
|
name: v8::Local<'s, v8::Value>,
|
||||||
|
schema_url: v8::Local<'s, v8::Value>,
|
||||||
|
version: v8::Local<'s, v8::Value>,
|
||||||
|
) {
|
||||||
|
let name = owned_string(scope, name.cast());
|
||||||
|
|
||||||
|
let scope_builder = opentelemetry::InstrumentationScope::builder(name);
|
||||||
|
let scope_builder = if schema_url.is_null_or_undefined() {
|
||||||
|
scope_builder
|
||||||
|
} else {
|
||||||
|
scope_builder.with_schema_url(owned_string(scope, schema_url.cast()))
|
||||||
|
};
|
||||||
|
let scope_builder = if version.is_null_or_undefined() {
|
||||||
|
scope_builder
|
||||||
|
} else {
|
||||||
|
scope_builder.with_version(owned_string(scope, version.cast()))
|
||||||
|
};
|
||||||
|
let scope = scope_builder.build();
|
||||||
|
let scope_metric = ScopeMetrics {
|
||||||
|
scope,
|
||||||
|
metrics: vec![],
|
||||||
|
};
|
||||||
|
|
||||||
|
match state.try_borrow_mut::<TemporaryMetricsExport>() {
|
||||||
|
Some(temp) => {
|
||||||
|
if let Some(current_metric) = temp.metric.take() {
|
||||||
|
let metric = Metric::from(current_metric);
|
||||||
|
temp.scope_metrics.last_mut().unwrap().metrics.push(metric);
|
||||||
|
}
|
||||||
|
temp.scope_metrics.push(scope_metric);
|
||||||
|
}
|
||||||
|
None => {
|
||||||
|
state.put(TemporaryMetricsExport {
|
||||||
|
resource_attributes: vec![],
|
||||||
|
scope_metrics: vec![scope_metric],
|
||||||
|
metric: None,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_sum<'s>(
|
||||||
|
scope: &mut v8::HandleScope<'s>,
|
||||||
|
state: &mut OpState,
|
||||||
|
name: v8::Local<'s, v8::Value>,
|
||||||
|
description: v8::Local<'s, v8::Value>,
|
||||||
|
unit: v8::Local<'s, v8::Value>,
|
||||||
|
#[smi] temporality: u8,
|
||||||
|
is_monotonic: bool,
|
||||||
|
) {
|
||||||
|
let Some(temp) = state.try_borrow_mut::<TemporaryMetricsExport>() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(current_metric) = temp.metric.take() {
|
||||||
|
let metric = Metric::from(current_metric);
|
||||||
|
temp.scope_metrics.last_mut().unwrap().metrics.push(metric);
|
||||||
|
}
|
||||||
|
|
||||||
|
let name = owned_string(scope, name.cast());
|
||||||
|
let description = owned_string(scope, description.cast());
|
||||||
|
let unit = owned_string(scope, unit.cast());
|
||||||
|
let temporality = match temporality {
|
||||||
|
0 => Temporality::Delta,
|
||||||
|
1 => Temporality::Cumulative,
|
||||||
|
_ => return,
|
||||||
|
};
|
||||||
|
let sum = opentelemetry_sdk::metrics::data::Sum {
|
||||||
|
data_points: vec![],
|
||||||
|
temporality,
|
||||||
|
is_monotonic,
|
||||||
|
};
|
||||||
|
|
||||||
|
temp.metric = Some(TemporaryMetric {
|
||||||
|
name,
|
||||||
|
description,
|
||||||
|
unit,
|
||||||
|
data: TemporaryMetricData::Sum(sum),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_gauge<'s>(
|
||||||
|
scope: &mut v8::HandleScope<'s>,
|
||||||
|
state: &mut OpState,
|
||||||
|
name: v8::Local<'s, v8::Value>,
|
||||||
|
description: v8::Local<'s, v8::Value>,
|
||||||
|
unit: v8::Local<'s, v8::Value>,
|
||||||
|
) {
|
||||||
|
let Some(temp) = state.try_borrow_mut::<TemporaryMetricsExport>() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(current_metric) = temp.metric.take() {
|
||||||
|
let metric = Metric::from(current_metric);
|
||||||
|
temp.scope_metrics.last_mut().unwrap().metrics.push(metric);
|
||||||
|
}
|
||||||
|
|
||||||
|
let name = owned_string(scope, name.cast());
|
||||||
|
let description = owned_string(scope, description.cast());
|
||||||
|
let unit = owned_string(scope, unit.cast());
|
||||||
|
|
||||||
|
let gauge = opentelemetry_sdk::metrics::data::Gauge {
|
||||||
|
data_points: vec![],
|
||||||
|
};
|
||||||
|
|
||||||
|
temp.metric = Some(TemporaryMetric {
|
||||||
|
name,
|
||||||
|
description,
|
||||||
|
unit,
|
||||||
|
data: TemporaryMetricData::Gauge(gauge),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_sum_or_gauge_data_point(
|
||||||
|
state: &mut OpState,
|
||||||
|
value: f64,
|
||||||
|
start_time: f64,
|
||||||
|
time: f64,
|
||||||
|
) {
|
||||||
|
let Some(temp) = state.try_borrow_mut::<TemporaryMetricsExport>() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
let start_time = SystemTime::UNIX_EPOCH
|
||||||
|
.checked_add(std::time::Duration::from_secs_f64(start_time))
|
||||||
|
.unwrap();
|
||||||
|
let time = SystemTime::UNIX_EPOCH
|
||||||
|
.checked_add(std::time::Duration::from_secs_f64(time))
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let data_point = opentelemetry_sdk::metrics::data::DataPoint {
|
||||||
|
value,
|
||||||
|
start_time: Some(start_time),
|
||||||
|
time: Some(time),
|
||||||
|
attributes: vec![],
|
||||||
|
exemplars: vec![],
|
||||||
|
};
|
||||||
|
|
||||||
|
match &mut temp.metric {
|
||||||
|
Some(TemporaryMetric {
|
||||||
|
data: TemporaryMetricData::Sum(sum),
|
||||||
|
..
|
||||||
|
}) => sum.data_points.push(data_point),
|
||||||
|
Some(TemporaryMetric {
|
||||||
|
data: TemporaryMetricData::Gauge(gauge),
|
||||||
|
..
|
||||||
|
}) => gauge.data_points.push(data_point),
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_histogram<'s>(
|
||||||
|
scope: &mut v8::HandleScope<'s>,
|
||||||
|
state: &mut OpState,
|
||||||
|
name: v8::Local<'s, v8::Value>,
|
||||||
|
description: v8::Local<'s, v8::Value>,
|
||||||
|
unit: v8::Local<'s, v8::Value>,
|
||||||
|
#[smi] temporality: u8,
|
||||||
|
) {
|
||||||
|
let Some(temp) = state.try_borrow_mut::<TemporaryMetricsExport>() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(current_metric) = temp.metric.take() {
|
||||||
|
let metric = Metric::from(current_metric);
|
||||||
|
temp.scope_metrics.last_mut().unwrap().metrics.push(metric);
|
||||||
|
}
|
||||||
|
|
||||||
|
let name = owned_string(scope, name.cast());
|
||||||
|
let description = owned_string(scope, description.cast());
|
||||||
|
let unit = owned_string(scope, unit.cast());
|
||||||
|
|
||||||
|
let temporality = match temporality {
|
||||||
|
0 => Temporality::Delta,
|
||||||
|
1 => Temporality::Cumulative,
|
||||||
|
_ => return,
|
||||||
|
};
|
||||||
|
let histogram = opentelemetry_sdk::metrics::data::Histogram {
|
||||||
|
data_points: vec![],
|
||||||
|
temporality,
|
||||||
|
};
|
||||||
|
|
||||||
|
temp.metric = Some(TemporaryMetric {
|
||||||
|
name,
|
||||||
|
description,
|
||||||
|
unit,
|
||||||
|
data: TemporaryMetricData::Histogram(histogram),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::too_many_arguments)]
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_histogram_data_point(
|
||||||
|
state: &mut OpState,
|
||||||
|
#[number] count: u64,
|
||||||
|
min: f64,
|
||||||
|
max: f64,
|
||||||
|
sum: f64,
|
||||||
|
start_time: f64,
|
||||||
|
time: f64,
|
||||||
|
#[smi] buckets: u32,
|
||||||
|
) {
|
||||||
|
let Some(temp) = state.try_borrow_mut::<TemporaryMetricsExport>() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
let min = if min.is_nan() { None } else { Some(min) };
|
||||||
|
let max = if max.is_nan() { None } else { Some(max) };
|
||||||
|
|
||||||
|
let start_time = SystemTime::UNIX_EPOCH
|
||||||
|
.checked_add(std::time::Duration::from_secs_f64(start_time))
|
||||||
|
.unwrap();
|
||||||
|
let time = SystemTime::UNIX_EPOCH
|
||||||
|
.checked_add(std::time::Duration::from_secs_f64(time))
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let data_point = opentelemetry_sdk::metrics::data::HistogramDataPoint {
|
||||||
|
bounds: Vec::with_capacity(buckets as usize),
|
||||||
|
bucket_counts: Vec::with_capacity((buckets as usize) + 1),
|
||||||
|
count,
|
||||||
|
sum,
|
||||||
|
min,
|
||||||
|
max,
|
||||||
|
start_time,
|
||||||
|
time,
|
||||||
|
attributes: vec![],
|
||||||
|
exemplars: vec![],
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(TemporaryMetric {
|
||||||
|
data: TemporaryMetricData::Histogram(histogram),
|
||||||
|
..
|
||||||
|
}) = &mut temp.metric
|
||||||
|
{
|
||||||
|
histogram.data_points.push(data_point);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_histogram_data_point_entry_final(
|
||||||
|
state: &mut OpState,
|
||||||
|
#[number] count1: u64,
|
||||||
|
) {
|
||||||
|
let Some(temp) = state.try_borrow_mut::<TemporaryMetricsExport>() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(TemporaryMetric {
|
||||||
|
data: TemporaryMetricData::Histogram(histogram),
|
||||||
|
..
|
||||||
|
}) = &mut temp.metric
|
||||||
|
{
|
||||||
|
histogram
|
||||||
|
.data_points
|
||||||
|
.last_mut()
|
||||||
|
.unwrap()
|
||||||
|
.bucket_counts
|
||||||
|
.push(count1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_histogram_data_point_entry1(
|
||||||
|
state: &mut OpState,
|
||||||
|
#[number] count1: u64,
|
||||||
|
bound1: f64,
|
||||||
|
) {
|
||||||
|
let Some(temp) = state.try_borrow_mut::<TemporaryMetricsExport>() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(TemporaryMetric {
|
||||||
|
data: TemporaryMetricData::Histogram(histogram),
|
||||||
|
..
|
||||||
|
}) = &mut temp.metric
|
||||||
|
{
|
||||||
|
let data_point = histogram.data_points.last_mut().unwrap();
|
||||||
|
data_point.bucket_counts.push(count1);
|
||||||
|
data_point.bounds.push(bound1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_histogram_data_point_entry2(
|
||||||
|
state: &mut OpState,
|
||||||
|
#[number] count1: u64,
|
||||||
|
bound1: f64,
|
||||||
|
#[number] count2: u64,
|
||||||
|
bound2: f64,
|
||||||
|
) {
|
||||||
|
let Some(temp) = state.try_borrow_mut::<TemporaryMetricsExport>() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(TemporaryMetric {
|
||||||
|
data: TemporaryMetricData::Histogram(histogram),
|
||||||
|
..
|
||||||
|
}) = &mut temp.metric
|
||||||
|
{
|
||||||
|
let data_point = histogram.data_points.last_mut().unwrap();
|
||||||
|
data_point.bucket_counts.push(count1);
|
||||||
|
data_point.bounds.push(bound1);
|
||||||
|
data_point.bucket_counts.push(count2);
|
||||||
|
data_point.bounds.push(bound2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_histogram_data_point_entry3(
|
||||||
|
state: &mut OpState,
|
||||||
|
#[number] count1: u64,
|
||||||
|
bound1: f64,
|
||||||
|
#[number] count2: u64,
|
||||||
|
bound2: f64,
|
||||||
|
#[number] count3: u64,
|
||||||
|
bound3: f64,
|
||||||
|
) {
|
||||||
|
let Some(temp) = state.try_borrow_mut::<TemporaryMetricsExport>() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(TemporaryMetric {
|
||||||
|
data: TemporaryMetricData::Histogram(histogram),
|
||||||
|
..
|
||||||
|
}) = &mut temp.metric
|
||||||
|
{
|
||||||
|
let data_point = histogram.data_points.last_mut().unwrap();
|
||||||
|
data_point.bucket_counts.push(count1);
|
||||||
|
data_point.bounds.push(bound1);
|
||||||
|
data_point.bucket_counts.push(count2);
|
||||||
|
data_point.bounds.push(bound2);
|
||||||
|
data_point.bucket_counts.push(count3);
|
||||||
|
data_point.bounds.push(bound3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_data_point_attribute<'s>(
|
||||||
|
scope: &mut v8::HandleScope<'s>,
|
||||||
|
state: &mut OpState,
|
||||||
|
#[smi] capacity: u32,
|
||||||
|
key: v8::Local<'s, v8::Value>,
|
||||||
|
value: v8::Local<'s, v8::Value>,
|
||||||
|
) {
|
||||||
|
if let Some(TemporaryMetricsExport {
|
||||||
|
metric: Some(metric),
|
||||||
|
..
|
||||||
|
}) = state.try_borrow_mut::<TemporaryMetricsExport>()
|
||||||
|
{
|
||||||
|
let attributes = match &mut metric.data {
|
||||||
|
TemporaryMetricData::Sum(sum) => {
|
||||||
|
&mut sum.data_points.last_mut().unwrap().attributes
|
||||||
|
}
|
||||||
|
TemporaryMetricData::Gauge(gauge) => {
|
||||||
|
&mut gauge.data_points.last_mut().unwrap().attributes
|
||||||
|
}
|
||||||
|
TemporaryMetricData::Histogram(histogram) => {
|
||||||
|
&mut histogram.data_points.last_mut().unwrap().attributes
|
||||||
|
}
|
||||||
|
};
|
||||||
|
attributes.reserve_exact((capacity as usize) - attributes.capacity());
|
||||||
|
attr!(scope, attributes, key, value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_data_point_attribute2<'s>(
|
||||||
|
scope: &mut v8::HandleScope<'s>,
|
||||||
|
state: &mut OpState,
|
||||||
|
#[smi] capacity: u32,
|
||||||
|
key1: v8::Local<'s, v8::Value>,
|
||||||
|
value1: v8::Local<'s, v8::Value>,
|
||||||
|
key2: v8::Local<'s, v8::Value>,
|
||||||
|
value2: v8::Local<'s, v8::Value>,
|
||||||
|
) {
|
||||||
|
if let Some(TemporaryMetricsExport {
|
||||||
|
metric: Some(metric),
|
||||||
|
..
|
||||||
|
}) = state.try_borrow_mut::<TemporaryMetricsExport>()
|
||||||
|
{
|
||||||
|
let attributes = match &mut metric.data {
|
||||||
|
TemporaryMetricData::Sum(sum) => {
|
||||||
|
&mut sum.data_points.last_mut().unwrap().attributes
|
||||||
|
}
|
||||||
|
TemporaryMetricData::Gauge(gauge) => {
|
||||||
|
&mut gauge.data_points.last_mut().unwrap().attributes
|
||||||
|
}
|
||||||
|
TemporaryMetricData::Histogram(histogram) => {
|
||||||
|
&mut histogram.data_points.last_mut().unwrap().attributes
|
||||||
|
}
|
||||||
|
};
|
||||||
|
attributes.reserve_exact((capacity as usize) - attributes.capacity());
|
||||||
|
attr!(scope, attributes, key1, value1);
|
||||||
|
attr!(scope, attributes, key2, value2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::too_many_arguments)]
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_data_point_attribute3<'s>(
|
||||||
|
scope: &mut v8::HandleScope<'s>,
|
||||||
|
state: &mut OpState,
|
||||||
|
#[smi] capacity: u32,
|
||||||
|
key1: v8::Local<'s, v8::Value>,
|
||||||
|
value1: v8::Local<'s, v8::Value>,
|
||||||
|
key2: v8::Local<'s, v8::Value>,
|
||||||
|
value2: v8::Local<'s, v8::Value>,
|
||||||
|
key3: v8::Local<'s, v8::Value>,
|
||||||
|
value3: v8::Local<'s, v8::Value>,
|
||||||
|
) {
|
||||||
|
if let Some(TemporaryMetricsExport {
|
||||||
|
metric: Some(metric),
|
||||||
|
..
|
||||||
|
}) = state.try_borrow_mut::<TemporaryMetricsExport>()
|
||||||
|
{
|
||||||
|
let attributes = match &mut metric.data {
|
||||||
|
TemporaryMetricData::Sum(sum) => {
|
||||||
|
&mut sum.data_points.last_mut().unwrap().attributes
|
||||||
|
}
|
||||||
|
TemporaryMetricData::Gauge(gauge) => {
|
||||||
|
&mut gauge.data_points.last_mut().unwrap().attributes
|
||||||
|
}
|
||||||
|
TemporaryMetricData::Histogram(histogram) => {
|
||||||
|
&mut histogram.data_points.last_mut().unwrap().attributes
|
||||||
|
}
|
||||||
|
};
|
||||||
|
attributes.reserve_exact((capacity as usize) - attributes.capacity());
|
||||||
|
attr!(scope, attributes, key1, value1);
|
||||||
|
attr!(scope, attributes, key2, value2);
|
||||||
|
attr!(scope, attributes, key3, value3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[op2(fast)]
|
||||||
|
fn op_otel_metrics_submit(state: &mut OpState) {
|
||||||
|
let Some(mut temp) = state.try_take::<TemporaryMetricsExport>() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
let Some(Processors { metrics, .. }) = OTEL_PROCESSORS.get() else {
|
||||||
|
return;
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(current_metric) = temp.metric {
|
||||||
|
let metric = Metric::from(current_metric);
|
||||||
|
temp.scope_metrics.last_mut().unwrap().metrics.push(metric);
|
||||||
|
}
|
||||||
|
|
||||||
|
let resource = Resource::new(temp.resource_attributes);
|
||||||
|
let scope_metrics = temp.scope_metrics;
|
||||||
|
|
||||||
|
metrics.submit(ResourceMetrics {
|
||||||
|
resource,
|
||||||
|
scope_metrics,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,23 @@ import {
|
||||||
op_otel_instrumentation_scope_enter,
|
op_otel_instrumentation_scope_enter,
|
||||||
op_otel_instrumentation_scope_enter_builtin,
|
op_otel_instrumentation_scope_enter_builtin,
|
||||||
op_otel_log,
|
op_otel_log,
|
||||||
|
op_otel_metrics_data_point_attribute,
|
||||||
|
op_otel_metrics_data_point_attribute2,
|
||||||
|
op_otel_metrics_data_point_attribute3,
|
||||||
|
op_otel_metrics_gauge,
|
||||||
|
op_otel_metrics_histogram,
|
||||||
|
op_otel_metrics_histogram_data_point,
|
||||||
|
op_otel_metrics_histogram_data_point_entry1,
|
||||||
|
op_otel_metrics_histogram_data_point_entry2,
|
||||||
|
op_otel_metrics_histogram_data_point_entry3,
|
||||||
|
op_otel_metrics_histogram_data_point_entry_final,
|
||||||
|
op_otel_metrics_resource_attribute,
|
||||||
|
op_otel_metrics_resource_attribute2,
|
||||||
|
op_otel_metrics_resource_attribute3,
|
||||||
|
op_otel_metrics_scope,
|
||||||
|
op_otel_metrics_submit,
|
||||||
|
op_otel_metrics_sum,
|
||||||
|
op_otel_metrics_sum_or_gauge_data_point,
|
||||||
op_otel_span_attribute,
|
op_otel_span_attribute,
|
||||||
op_otel_span_attribute2,
|
op_otel_span_attribute2,
|
||||||
op_otel_span_attribute3,
|
op_otel_span_attribute3,
|
||||||
|
@ -186,7 +203,7 @@ const instrumentationScopes = new SafeWeakMap<
|
||||||
>();
|
>();
|
||||||
let activeInstrumentationLibrary: WeakRef<InstrumentationLibrary> | null = null;
|
let activeInstrumentationLibrary: WeakRef<InstrumentationLibrary> | null = null;
|
||||||
|
|
||||||
function submit(
|
function submitSpan(
|
||||||
spanId: string | Uint8Array,
|
spanId: string | Uint8Array,
|
||||||
traceId: string | Uint8Array,
|
traceId: string | Uint8Array,
|
||||||
traceFlags: number,
|
traceFlags: number,
|
||||||
|
@ -411,7 +428,7 @@ export class Span {
|
||||||
|
|
||||||
endSpan = (span: Span) => {
|
endSpan = (span: Span) => {
|
||||||
const endTime = now();
|
const endTime = now();
|
||||||
submit(
|
submitSpan(
|
||||||
span.#spanId,
|
span.#spanId,
|
||||||
span.#traceId,
|
span.#traceId,
|
||||||
span.#traceFlags,
|
span.#traceFlags,
|
||||||
|
@ -571,7 +588,7 @@ class SpanExporter {
|
||||||
for (let i = 0; i < spans.length; i += 1) {
|
for (let i = 0; i < spans.length; i += 1) {
|
||||||
const span = spans[i];
|
const span = spans[i];
|
||||||
const context = span.spanContext();
|
const context = span.spanContext();
|
||||||
submit(
|
submitSpan(
|
||||||
context.spanId,
|
context.spanId,
|
||||||
context.traceId,
|
context.traceId,
|
||||||
context.traceFlags,
|
context.traceFlags,
|
||||||
|
@ -671,6 +688,262 @@ class ContextManager {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function attributeValue(value: IAnyValue) {
|
||||||
|
return value.boolValue ?? value.stringValue ?? value.doubleValue ??
|
||||||
|
value.intValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
function submitMetrics(resource, scopeMetrics) {
|
||||||
|
let i = 0;
|
||||||
|
while (i < resource.attributes.length) {
|
||||||
|
if (i + 2 < resource.attributes.length) {
|
||||||
|
op_otel_metrics_resource_attribute3(
|
||||||
|
resource.attributes.length,
|
||||||
|
resource.attributes[i].key,
|
||||||
|
attributeValue(resource.attributes[i].value),
|
||||||
|
resource.attributes[i + 1].key,
|
||||||
|
attributeValue(resource.attributes[i + 1].value),
|
||||||
|
resource.attributes[i + 2].key,
|
||||||
|
attributeValue(resource.attributes[i + 2].value),
|
||||||
|
);
|
||||||
|
i += 3;
|
||||||
|
} else if (i + 1 < resource.attributes.length) {
|
||||||
|
op_otel_metrics_resource_attribute2(
|
||||||
|
resource.attributes.length,
|
||||||
|
resource.attributes[i].key,
|
||||||
|
attributeValue(resource.attributes[i].value),
|
||||||
|
resource.attributes[i + 1].key,
|
||||||
|
attributeValue(resource.attributes[i + 1].value),
|
||||||
|
);
|
||||||
|
i += 2;
|
||||||
|
} else {
|
||||||
|
op_otel_metrics_resource_attribute(
|
||||||
|
resource.attributes.length,
|
||||||
|
resource.attributes[i].key,
|
||||||
|
attributeValue(resource.attributes[i].value),
|
||||||
|
);
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let smi = 0; smi < scopeMetrics.length; smi += 1) {
|
||||||
|
const { scope, metrics } = scopeMetrics[smi];
|
||||||
|
|
||||||
|
op_otel_metrics_scope(scope.name, scope.schemaUrl, scope.version);
|
||||||
|
|
||||||
|
for (let mi = 0; mi < metrics.length; mi += 1) {
|
||||||
|
const metric = metrics[mi];
|
||||||
|
switch (metric.dataPointType) {
|
||||||
|
case 3:
|
||||||
|
op_otel_metrics_sum(
|
||||||
|
metric.descriptor.name,
|
||||||
|
// deno-lint-ignore prefer-primordials
|
||||||
|
metric.descriptor.description,
|
||||||
|
metric.descriptor.unit,
|
||||||
|
metric.aggregationTemporality,
|
||||||
|
metric.isMonotonic,
|
||||||
|
);
|
||||||
|
for (let di = 0; di < metric.dataPoints.length; di += 1) {
|
||||||
|
const dataPoint = metric.dataPoints[di];
|
||||||
|
op_otel_metrics_sum_or_gauge_data_point(
|
||||||
|
dataPoint.value,
|
||||||
|
hrToSecs(dataPoint.startTime),
|
||||||
|
hrToSecs(dataPoint.endTime),
|
||||||
|
);
|
||||||
|
const attributes = ObjectEntries(dataPoint.attributes);
|
||||||
|
let i = 0;
|
||||||
|
while (i < attributes.length) {
|
||||||
|
if (i + 2 < attributes.length) {
|
||||||
|
op_otel_metrics_data_point_attribute3(
|
||||||
|
attributes.length,
|
||||||
|
attributes[i][0],
|
||||||
|
attributes[i][1],
|
||||||
|
attributes[i + 1][0],
|
||||||
|
attributes[i + 1][1],
|
||||||
|
attributes[i + 2][0],
|
||||||
|
attributes[i + 2][1],
|
||||||
|
);
|
||||||
|
i += 3;
|
||||||
|
} else if (i + 1 < attributes.length) {
|
||||||
|
op_otel_metrics_data_point_attribute2(
|
||||||
|
attributes.length,
|
||||||
|
attributes[i][0],
|
||||||
|
attributes[i][1],
|
||||||
|
attributes[i + 1][0],
|
||||||
|
attributes[i + 1][1],
|
||||||
|
);
|
||||||
|
i += 2;
|
||||||
|
} else {
|
||||||
|
op_otel_metrics_data_point_attribute(
|
||||||
|
attributes.length,
|
||||||
|
attributes[i][0],
|
||||||
|
attributes[i][1],
|
||||||
|
);
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
op_otel_metrics_gauge(
|
||||||
|
metric.descriptor.name,
|
||||||
|
// deno-lint-ignore prefer-primordials
|
||||||
|
metric.descriptor.description,
|
||||||
|
metric.descriptor.unit,
|
||||||
|
);
|
||||||
|
for (let di = 0; di < metric.dataPoints.length; di += 1) {
|
||||||
|
const dataPoint = metric.dataPoints[di];
|
||||||
|
op_otel_metrics_sum_or_gauge_data_point(
|
||||||
|
dataPoint.value,
|
||||||
|
hrToSecs(dataPoint.startTime),
|
||||||
|
hrToSecs(dataPoint.endTime),
|
||||||
|
);
|
||||||
|
const attributes = ObjectEntries(dataPoint.attributes);
|
||||||
|
let i = 0;
|
||||||
|
while (i < attributes.length) {
|
||||||
|
if (i + 2 < attributes.length) {
|
||||||
|
op_otel_metrics_data_point_attribute3(
|
||||||
|
attributes.length,
|
||||||
|
attributes[i][0],
|
||||||
|
attributes[i][1],
|
||||||
|
attributes[i + 1][0],
|
||||||
|
attributes[i + 1][1],
|
||||||
|
attributes[i + 2][0],
|
||||||
|
attributes[i + 2][1],
|
||||||
|
);
|
||||||
|
i += 3;
|
||||||
|
} else if (i + 1 < attributes.length) {
|
||||||
|
op_otel_metrics_data_point_attribute2(
|
||||||
|
attributes.length,
|
||||||
|
attributes[i][0],
|
||||||
|
attributes[i][1],
|
||||||
|
attributes[i + 1][0],
|
||||||
|
attributes[i + 1][1],
|
||||||
|
);
|
||||||
|
i += 2;
|
||||||
|
} else {
|
||||||
|
op_otel_metrics_data_point_attribute(
|
||||||
|
attributes.length,
|
||||||
|
attributes[i][0],
|
||||||
|
attributes[i][1],
|
||||||
|
);
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 0:
|
||||||
|
op_otel_metrics_histogram(
|
||||||
|
metric.descriptor.name,
|
||||||
|
// deno-lint-ignore prefer-primordials
|
||||||
|
metric.descriptor.description,
|
||||||
|
metric.descriptor.unit,
|
||||||
|
metric.aggregationTemporality,
|
||||||
|
);
|
||||||
|
for (let di = 0; di < metric.dataPoints.length; di += 1) {
|
||||||
|
const dataPoint = metric.dataPoints[di];
|
||||||
|
const { boundaries, counts } = dataPoint.value.buckets;
|
||||||
|
op_otel_metrics_histogram_data_point(
|
||||||
|
dataPoint.value.count,
|
||||||
|
dataPoint.value.min ?? NaN,
|
||||||
|
dataPoint.value.max ?? NaN,
|
||||||
|
dataPoint.value.sum,
|
||||||
|
hrToSecs(dataPoint.startTime),
|
||||||
|
hrToSecs(dataPoint.endTime),
|
||||||
|
boundaries.length,
|
||||||
|
);
|
||||||
|
let j = 0;
|
||||||
|
while (j < boundaries.length) {
|
||||||
|
if (j + 3 < boundaries.length) {
|
||||||
|
op_otel_metrics_histogram_data_point_entry3(
|
||||||
|
counts[j],
|
||||||
|
boundaries[j],
|
||||||
|
counts[j + 1],
|
||||||
|
boundaries[j + 1],
|
||||||
|
counts[j + 2],
|
||||||
|
boundaries[j + 2],
|
||||||
|
);
|
||||||
|
j += 3;
|
||||||
|
} else if (j + 2 < boundaries.length) {
|
||||||
|
op_otel_metrics_histogram_data_point_entry2(
|
||||||
|
counts[j],
|
||||||
|
boundaries[j],
|
||||||
|
counts[j + 1],
|
||||||
|
boundaries[j + 1],
|
||||||
|
);
|
||||||
|
j += 2;
|
||||||
|
} else {
|
||||||
|
op_otel_metrics_histogram_data_point_entry1(
|
||||||
|
counts[j],
|
||||||
|
boundaries[j],
|
||||||
|
);
|
||||||
|
j += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
op_otel_metrics_histogram_data_point_entry_final(counts[j]);
|
||||||
|
const attributes = ObjectEntries(dataPoint.attributes);
|
||||||
|
let i = 0;
|
||||||
|
while (i < attributes.length) {
|
||||||
|
if (i + 2 < attributes.length) {
|
||||||
|
op_otel_metrics_data_point_attribute3(
|
||||||
|
attributes.length,
|
||||||
|
attributes[i][0],
|
||||||
|
attributes[i][1],
|
||||||
|
attributes[i + 1][0],
|
||||||
|
attributes[i + 1][1],
|
||||||
|
attributes[i + 2][0],
|
||||||
|
attributes[i + 2][1],
|
||||||
|
);
|
||||||
|
i += 3;
|
||||||
|
} else if (i + 1 < attributes.length) {
|
||||||
|
op_otel_metrics_data_point_attribute2(
|
||||||
|
attributes.length,
|
||||||
|
attributes[i][0],
|
||||||
|
attributes[i][1],
|
||||||
|
attributes[i + 1][0],
|
||||||
|
attributes[i + 1][1],
|
||||||
|
);
|
||||||
|
i += 2;
|
||||||
|
} else {
|
||||||
|
op_otel_metrics_data_point_attribute(
|
||||||
|
attributes.length,
|
||||||
|
attributes[i][0],
|
||||||
|
attributes[i][1],
|
||||||
|
);
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
op_otel_metrics_submit();
|
||||||
|
}
|
||||||
|
|
||||||
|
class MetricExporter {
|
||||||
|
export(metrics, resultCallback: (result: ExportResult) => void) {
|
||||||
|
try {
|
||||||
|
submitMetrics(metrics.resource, metrics.scopeMetrics);
|
||||||
|
resultCallback({ code: 0 });
|
||||||
|
} catch (error) {
|
||||||
|
resultCallback({
|
||||||
|
code: 1,
|
||||||
|
error: ObjectPrototypeIsPrototypeOf(error, Error)
|
||||||
|
? error as Error
|
||||||
|
: new Error(String(error)),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async forceFlush() {}
|
||||||
|
|
||||||
|
async shutdown() {}
|
||||||
|
}
|
||||||
|
|
||||||
const otelConsoleConfig = {
|
const otelConsoleConfig = {
|
||||||
ignore: 0,
|
ignore: 0,
|
||||||
capture: 1,
|
capture: 1,
|
||||||
|
@ -708,4 +981,5 @@ export function bootstrap(
|
||||||
export const telemetry = {
|
export const telemetry = {
|
||||||
SpanExporter,
|
SpanExporter,
|
||||||
ContextManager,
|
ContextManager,
|
||||||
|
MetricExporter,
|
||||||
};
|
};
|
||||||
|
|
File diff suppressed because one or more lines are too long
|
@ -15,6 +15,10 @@
|
||||||
{
|
{
|
||||||
"args": "run -A main.ts uncaught.ts",
|
"args": "run -A main.ts uncaught.ts",
|
||||||
"output": "uncaught.out"
|
"output": "uncaught.out"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"args": "run -A main.ts metric.ts",
|
||||||
|
"output": "metric.out"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
|
@ -188,5 +188,6 @@
|
||||||
"traceId": "00000000000000000000000000000003",
|
"traceId": "00000000000000000000000000000003",
|
||||||
"spanId": "1000000000000002"
|
"spanId": "1000000000000002"
|
||||||
}
|
}
|
||||||
]
|
],
|
||||||
|
"metrics": []
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,5 +15,6 @@
|
||||||
"traceId": "",
|
"traceId": "",
|
||||||
"spanId": ""
|
"spanId": ""
|
||||||
}
|
}
|
||||||
]
|
],
|
||||||
|
"metrics": []
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,6 +3,7 @@
|
||||||
const data = {
|
const data = {
|
||||||
spans: [],
|
spans: [],
|
||||||
logs: [],
|
logs: [],
|
||||||
|
metrics: [],
|
||||||
};
|
};
|
||||||
|
|
||||||
const server = Deno.serve(
|
const server = Deno.serve(
|
||||||
|
@ -45,6 +46,11 @@ const server = Deno.serve(
|
||||||
data.spans.push(...sSpans.spans);
|
data.spans.push(...sSpans.spans);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
body.resourceMetrics?.forEach((rMetrics) => {
|
||||||
|
rMetrics.scopeMetrics.forEach((sMetrics) => {
|
||||||
|
data.metrics.push(...sMetrics.metrics);
|
||||||
|
});
|
||||||
|
});
|
||||||
return Response.json({ partialSuccess: {} }, { status: 200 });
|
return Response.json({ partialSuccess: {} }, { status: 200 });
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
124
tests/specs/cli/otel_basic/metric.out
Normal file
124
tests/specs/cli/otel_basic/metric.out
Normal file
|
@ -0,0 +1,124 @@
|
||||||
|
{
|
||||||
|
"spans": [],
|
||||||
|
"logs": [],
|
||||||
|
"metrics": [
|
||||||
|
{
|
||||||
|
"name": "counter",
|
||||||
|
"description": "Example of a Counter",
|
||||||
|
"unit": "",
|
||||||
|
"metadata": [],
|
||||||
|
"sum": {
|
||||||
|
"dataPoints": [
|
||||||
|
{
|
||||||
|
"attributes": [
|
||||||
|
{
|
||||||
|
"key": "attribute",
|
||||||
|
"value": {
|
||||||
|
"doubleValue": 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"startTimeUnixNano": "[WILDCARD]",
|
||||||
|
"timeUnixNano": "[WILDCARD]",
|
||||||
|
"exemplars": [],
|
||||||
|
"flags": 0,
|
||||||
|
"asDouble": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"aggregationTemporality": 2,
|
||||||
|
"isMonotonic": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "up_down_counter",
|
||||||
|
"description": "Example of a UpDownCounter",
|
||||||
|
"unit": "",
|
||||||
|
"metadata": [],
|
||||||
|
"sum": {
|
||||||
|
"dataPoints": [
|
||||||
|
{
|
||||||
|
"attributes": [
|
||||||
|
{
|
||||||
|
"key": "attribute",
|
||||||
|
"value": {
|
||||||
|
"doubleValue": 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"startTimeUnixNano": "[WILDCARD]",
|
||||||
|
"timeUnixNano": "[WILDCARD]",
|
||||||
|
"exemplars": [],
|
||||||
|
"flags": 0,
|
||||||
|
"asDouble": -1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"aggregationTemporality": 2,
|
||||||
|
"isMonotonic": false
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "histogram",
|
||||||
|
"description": "Example of a Histogram",
|
||||||
|
"unit": "",
|
||||||
|
"metadata": [],
|
||||||
|
"histogram": {
|
||||||
|
"dataPoints": [
|
||||||
|
{
|
||||||
|
"attributes": [
|
||||||
|
{
|
||||||
|
"key": "attribute",
|
||||||
|
"value": {
|
||||||
|
"doubleValue": 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"startTimeUnixNano": "[WILDCARD]",
|
||||||
|
"timeUnixNano": "[WILDCARD]",
|
||||||
|
"count": 1,
|
||||||
|
"sum": 1,
|
||||||
|
"bucketCounts": [
|
||||||
|
0,
|
||||||
|
1,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
],
|
||||||
|
"explicitBounds": [
|
||||||
|
0,
|
||||||
|
5,
|
||||||
|
10,
|
||||||
|
25,
|
||||||
|
50,
|
||||||
|
75,
|
||||||
|
100,
|
||||||
|
250,
|
||||||
|
500,
|
||||||
|
750,
|
||||||
|
1000,
|
||||||
|
2500,
|
||||||
|
5000,
|
||||||
|
7500,
|
||||||
|
10000
|
||||||
|
],
|
||||||
|
"exemplars": [],
|
||||||
|
"flags": 0,
|
||||||
|
"min": 1,
|
||||||
|
"max": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"aggregationTemporality": 2
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
34
tests/specs/cli/otel_basic/metric.ts
Normal file
34
tests/specs/cli/otel_basic/metric.ts
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
import {
|
||||||
|
MeterProvider,
|
||||||
|
PeriodicExportingMetricReader,
|
||||||
|
} from "npm:@opentelemetry/sdk-metrics@1.28.0";
|
||||||
|
|
||||||
|
const meterProvider = new MeterProvider();
|
||||||
|
|
||||||
|
meterProvider.addMetricReader(
|
||||||
|
new PeriodicExportingMetricReader({
|
||||||
|
exporter: new Deno.telemetry.MetricExporter(),
|
||||||
|
exportIntervalMillis: 100,
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
const meter = meterProvider.getMeter("m");
|
||||||
|
|
||||||
|
const counter = meter.createCounter("counter", {
|
||||||
|
description: "Example of a Counter",
|
||||||
|
});
|
||||||
|
|
||||||
|
const upDownCounter = meter.createUpDownCounter("up_down_counter", {
|
||||||
|
description: "Example of a UpDownCounter",
|
||||||
|
});
|
||||||
|
|
||||||
|
const histogram = meter.createHistogram("histogram", {
|
||||||
|
description: "Example of a Histogram",
|
||||||
|
});
|
||||||
|
|
||||||
|
const attributes = { attribute: 1 };
|
||||||
|
counter.add(1, attributes);
|
||||||
|
upDownCounter.add(-1, attributes);
|
||||||
|
histogram.record(1, attributes);
|
||||||
|
|
||||||
|
await meterProvider.forceFlush();
|
|
@ -15,5 +15,6 @@
|
||||||
"traceId": "",
|
"traceId": "",
|
||||||
"spanId": ""
|
"spanId": ""
|
||||||
}
|
}
|
||||||
]
|
],
|
||||||
|
"metrics": []
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,5 +33,6 @@ throw new Error("uncaught");
|
||||||
"traceId": "",
|
"traceId": "",
|
||||||
"spanId": ""
|
"spanId": ""
|
||||||
}
|
}
|
||||||
]
|
],
|
||||||
|
"metrics": []
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue