opentelemetry_otlp/
metric.rs1#[cfg(any(feature = "http-proto", feature = "http-json", feature = "grpc-tonic"))]
7use crate::HasExportConfig;
8
9#[cfg(any(feature = "http-proto", feature = "http-json"))]
10use crate::{exporter::http::HttpExporterBuilder, HasHttpConfig, HttpExporterBuilderSet};
11
12#[cfg(feature = "grpc-tonic")]
13use crate::{exporter::tonic::TonicExporterBuilder, HasTonicConfig, TonicExporterBuilderSet};
14
15use crate::{ExporterBuildError, NoExporterBuilderSet};
16
17use core::fmt;
18use opentelemetry_sdk::error::OTelSdkResult;
19
20use opentelemetry_sdk::metrics::{
21 data::ResourceMetrics, exporter::PushMetricExporter, Temporality,
22};
23use std::fmt::{Debug, Formatter};
24
25pub const OTEL_EXPORTER_OTLP_METRICS_ENDPOINT: &str = "OTEL_EXPORTER_OTLP_METRICS_ENDPOINT";
29pub const OTEL_EXPORTER_OTLP_METRICS_TIMEOUT: &str = "OTEL_EXPORTER_OTLP_METRICS_TIMEOUT";
31pub const OTEL_EXPORTER_OTLP_METRICS_COMPRESSION: &str = "OTEL_EXPORTER_OTLP_METRICS_COMPRESSION";
33pub const OTEL_EXPORTER_OTLP_METRICS_HEADERS: &str = "OTEL_EXPORTER_OTLP_METRICS_HEADERS";
38
39#[derive(Debug, Default, Clone)]
40pub struct MetricExporterBuilder<C> {
41 client: C,
42 temporality: Temporality,
43}
44
45impl MetricExporterBuilder<NoExporterBuilderSet> {
46 pub fn new() -> Self {
47 MetricExporterBuilder::default()
48 }
49}
50
51impl<C> MetricExporterBuilder<C> {
52 #[cfg(feature = "grpc-tonic")]
53 pub fn with_tonic(self) -> MetricExporterBuilder<TonicExporterBuilderSet> {
54 MetricExporterBuilder {
55 client: TonicExporterBuilderSet(TonicExporterBuilder::default()),
56 temporality: self.temporality,
57 }
58 }
59
60 #[cfg(any(feature = "http-proto", feature = "http-json"))]
61 pub fn with_http(self) -> MetricExporterBuilder<HttpExporterBuilderSet> {
62 MetricExporterBuilder {
63 client: HttpExporterBuilderSet(HttpExporterBuilder::default()),
64 temporality: self.temporality,
65 }
66 }
67
68 pub fn with_temporality(self, temporality: Temporality) -> MetricExporterBuilder<C> {
69 MetricExporterBuilder {
70 client: self.client,
71 temporality,
72 }
73 }
74}
75
76#[cfg(feature = "grpc-tonic")]
77impl MetricExporterBuilder<TonicExporterBuilderSet> {
78 pub fn build(self) -> Result<MetricExporter, ExporterBuildError> {
79 let exporter = self.client.0.build_metrics_exporter(self.temporality)?;
80 opentelemetry::otel_debug!(name: "MetricExporterBuilt");
81 Ok(exporter)
82 }
83}
84
85#[cfg(any(feature = "http-proto", feature = "http-json"))]
86impl MetricExporterBuilder<HttpExporterBuilderSet> {
87 pub fn build(self) -> Result<MetricExporter, ExporterBuildError> {
88 let exporter = self.client.0.build_metrics_exporter(self.temporality)?;
89 Ok(exporter)
90 }
91}
92
93#[cfg(feature = "grpc-tonic")]
94impl HasExportConfig for MetricExporterBuilder<TonicExporterBuilderSet> {
95 fn export_config(&mut self) -> &mut crate::ExportConfig {
96 &mut self.client.0.exporter_config
97 }
98}
99
100#[cfg(any(feature = "http-proto", feature = "http-json"))]
101impl HasExportConfig for MetricExporterBuilder<HttpExporterBuilderSet> {
102 fn export_config(&mut self) -> &mut crate::ExportConfig {
103 &mut self.client.0.exporter_config
104 }
105}
106
107#[cfg(feature = "grpc-tonic")]
108impl HasTonicConfig for MetricExporterBuilder<TonicExporterBuilderSet> {
109 fn tonic_config(&mut self) -> &mut crate::TonicConfig {
110 &mut self.client.0.tonic_config
111 }
112}
113
114#[cfg(any(feature = "http-proto", feature = "http-json"))]
115impl HasHttpConfig for MetricExporterBuilder<HttpExporterBuilderSet> {
116 fn http_client_config(&mut self) -> &mut crate::exporter::http::HttpConfig {
117 &mut self.client.0.http_config
118 }
119}
120
121pub(crate) trait MetricsClient: fmt::Debug + Send + Sync + 'static {
123 fn export(
124 &self,
125 metrics: &mut ResourceMetrics,
126 ) -> impl std::future::Future<Output = OTelSdkResult> + Send;
127 fn shutdown(&self) -> OTelSdkResult;
128}
129
130pub struct MetricExporter {
132 client: SupportedTransportClient,
133 temporality: Temporality,
134}
135
136#[derive(Debug)]
137enum SupportedTransportClient {
138 #[cfg(feature = "grpc-tonic")]
139 Tonic(crate::exporter::tonic::metrics::TonicMetricsClient),
140 #[cfg(any(feature = "http-proto", feature = "http-json"))]
141 Http(crate::exporter::http::OtlpHttpClient),
142}
143
144impl Debug for MetricExporter {
145 fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
146 f.debug_struct("MetricExporter").finish()
147 }
148}
149
150impl PushMetricExporter for MetricExporter {
151 async fn export(&self, metrics: &mut ResourceMetrics) -> OTelSdkResult {
152 match &self.client {
153 #[cfg(feature = "grpc-tonic")]
154 SupportedTransportClient::Tonic(client) => client.export(metrics).await,
155 #[cfg(any(feature = "http-proto", feature = "http-json"))]
156 SupportedTransportClient::Http(client) => client.export(metrics).await,
157 }
158 }
159
160 fn force_flush(&self) -> OTelSdkResult {
161 Ok(())
163 }
164
165 fn shutdown(&self) -> OTelSdkResult {
166 match &self.client {
167 #[cfg(feature = "grpc-tonic")]
168 SupportedTransportClient::Tonic(client) => client.shutdown(),
169 #[cfg(any(feature = "http-proto", feature = "http-json"))]
170 SupportedTransportClient::Http(client) => client.shutdown(),
171 }
172 }
173
174 fn temporality(&self) -> Temporality {
175 self.temporality
176 }
177}
178
179impl MetricExporter {
180 pub fn builder() -> MetricExporterBuilder<NoExporterBuilderSet> {
182 MetricExporterBuilder::default()
183 }
184
185 #[cfg(feature = "grpc-tonic")]
186 pub(crate) fn from_tonic(
187 client: crate::exporter::tonic::metrics::TonicMetricsClient,
188 temporality: Temporality,
189 ) -> Self {
190 Self {
191 client: SupportedTransportClient::Tonic(client),
192 temporality,
193 }
194 }
195
196 #[cfg(any(feature = "http-proto", feature = "http-json"))]
197 pub(crate) fn from_http(
198 client: crate::exporter::http::OtlpHttpClient,
199 temporality: Temporality,
200 ) -> Self {
201 Self {
202 client: SupportedTransportClient::Http(client),
203 temporality,
204 }
205 }
206}