Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/francoposa/tower-otel-http-metrics
OpenTelemetry HTTP Metrics Middleware Layer for Tower-Compatible Servers (Axum, Hyper)
https://github.com/francoposa/tower-otel-http-metrics
axum axum-middleware hacktoberfest hacktoberfest2024 http hyper metrics middleware opentelemetry opentelemetry-rust tower tower-middleware
Last synced: 2 months ago
JSON representation
OpenTelemetry HTTP Metrics Middleware Layer for Tower-Compatible Servers (Axum, Hyper)
- Host: GitHub
- URL: https://github.com/francoposa/tower-otel-http-metrics
- Owner: francoposa
- License: mit
- Created: 2023-11-26T03:31:55.000Z (about 1 year ago)
- Default Branch: main
- Last Pushed: 2024-10-21T21:37:28.000Z (3 months ago)
- Last Synced: 2024-10-22T17:16:06.435Z (3 months ago)
- Topics: axum, axum-middleware, hacktoberfest, hacktoberfest2024, http, hyper, metrics, middleware, opentelemetry, opentelemetry-rust, tower, tower-middleware
- Language: Rust
- Homepage: https://docs.rs/tower-otel-http-metrics/latest/tower_otel_http_metrics
- Size: 84 KB
- Stars: 8
- Watchers: 1
- Forks: 3
- Open Issues: 2
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# Tower OTEL Metrics Middleware
OpenTelemetry Metrics Middleware for Tower-compatible Rust HTTP servers.
## Examples
See `examples` directory in repo for runnable code and supporting config files.
Attempts are made to keep the code here synced, but it will not be perfect.OTEL libraries in particular are sensitive to minor version changes at this point,
so the examples may only work with the OTEL crate versions pinned in `examples`.### Axum Server
Adding OpenTelementry HTTP Server metrics using the [`Axum`](https://docs.rs/axum/latest/axum) framework
over a Tower-compatible [`Hyper`](https://docs.rs/hyper/latest/hyper) Service:```rust
use std::time::Duration;use axum::routing::{get, post, put, Router};
use bytes::Bytes;
use opentelemetry::{global, KeyValue};
use opentelemetry_otlp::{
WithExportConfig, {self},
};
use opentelemetry_sdk::resource::{
EnvResourceDetector, SdkProvidedResourceDetector, TelemetryResourceDetector,
};
use opentelemetry_sdk::Resource;
use tower_otel_http_metrics;const SERVICE_NAME: &str = "example-axum-http-service";
fn init_otel_resource() -> Resource {
let otlp_resource_detected = Resource::from_detectors(
Duration::from_secs(3),
vec![
Box::new(SdkProvidedResourceDetector),
Box::new(EnvResourceDetector::new()),
Box::new(TelemetryResourceDetector),
],
);
let otlp_resource_override = Resource::new(vec![KeyValue {
key: opentelemetry_semantic_conventions::resource::SERVICE_NAME.into(),
value: SERVICE_NAME.into(),
}]);
otlp_resource_detected.merge(&otlp_resource_override)
}async fn handle() -> Bytes {
Bytes::from("hello, world")
}#[tokio::main]
async fn main() {
// init otel metrics pipeline
// https://docs.rs/opentelemetry-otlp/latest/opentelemetry_otlp/#kitchen-sink-full-configuration
// this configuration interface is annoyingly slightly different from the tracing one
// also the above documentation is outdated, it took awhile to get this correct one working
let meter_provider = opentelemetry_otlp::new_pipeline()
.metrics(opentelemetry_sdk::runtime::Tokio)
.with_exporter(
opentelemetry_otlp::new_exporter()
.tonic()
.with_endpoint("http://localhost:4317"),
)
.with_resource(init_otel_resource().clone())
.with_period(Duration::from_secs(10))
.build() // build registers the global meter provider
.unwrap();global::set_meter_provider(meter_provider);
// init our otel metrics middleware
let global_meter = global::meter(SERVICE_NAME);
let otel_metrics_service_layer = tower_otel_http_metrics::HTTPMetricsLayerBuilder::new()
.with_meter(global_meter)
.build()
.unwrap();let app = Router::new()
.route("/", get(handle))
.route("/", post(handle))
.route("/", put(handle))
.layer(otel_metrics_service_layer);let listener = tokio::net::TcpListener::bind("0.0.0.0:5000").await.unwrap();
let server = axum::serve(listener, app);if let Err(err) = server.await {
eprintln!("server error: {}", err);
}
}
```### Hyper Server
Adding OpenTelementry HTTP Server metrics to a bare-bones Tower-compatible Service
using [`Hyper`](https://docs.rs/crate/hyper/latest):```rust
use std::convert::Infallible;
use std::net::SocketAddr;
use std::time::Duration;use http_body_util::Full;
use hyper::body::Bytes;
use hyper::server::conn::http1;
use hyper::{Request, Response};
use opentelemetry::{global, KeyValue};
use opentelemetry_otlp::{
WithExportConfig, {self},
};
use opentelemetry_sdk::resource::{
EnvResourceDetector, SdkProvidedResourceDetector, TelemetryResourceDetector,
};
use opentelemetry_sdk::Resource;
use tokio::net::TcpListener;
use tower::ServiceBuilder;
use tower_otel_http_metrics;const SERVICE_NAME: &str = "example-tower-http-service";
fn init_otel_resource() -> Resource {
let otlp_resource_detected = Resource::from_detectors(
Duration::from_secs(3),
vec![
Box::new(SdkProvidedResourceDetector),
Box::new(EnvResourceDetector::new()),
Box::new(TelemetryResourceDetector),
],
);
let otlp_resource_override = Resource::new(vec![KeyValue {
key: opentelemetry_semantic_conventions::resource::SERVICE_NAME.into(),
value: SERVICE_NAME.into(),
}]);
otlp_resource_detected.merge(&otlp_resource_override)
}async fn handle(_req: Request) -> Result>, Infallible> {
Ok(Response::new(Full::new(Bytes::from("hello, world"))))
}#[tokio::main]
async fn main() {
// init otel metrics pipeline
// https://docs.rs/opentelemetry-otlp/latest/opentelemetry_otlp/#kitchen-sink-full-configuration
// this configuration interface is annoyingly slightly different from the tracing one
// also the above documentation is outdated, it took awhile to get this correct one working
let meter_provider = opentelemetry_otlp::new_pipeline()
.metrics(opentelemetry_sdk::runtime::Tokio)
.with_exporter(
opentelemetry_otlp::new_exporter()
.tonic()
.with_endpoint("http://localhost:4317"),
)
.with_resource(init_otel_resource().clone())
.with_period(Duration::from_secs(10))
.build() // build registers the global meter provider
.unwrap();global::set_meter_provider(meter_provider);
// init our otel metrics middleware
let global_meter = global::meter(SERVICE_NAME);
let otel_metrics_service_layer = tower_otel_http_metrics::HTTPMetricsLayerBuilder::new()
.with_meter(global_meter)
.build()
.unwrap();let tower_service = ServiceBuilder::new()
.layer(otel_metrics_service_layer)
.service_fn(handle);
let hyper_service = hyper_util::service::TowerToHyperService::new(tower_service);let addr = SocketAddr::from(([127, 0, 0, 1], 5000));
let listener = TcpListener::bind(addr).await.unwrap();loop {
let (stream, _) = listener.accept().await.unwrap();let io = hyper_util::rt::TokioIo::new(stream);
let service_clone = hyper_service.clone();tokio::task::spawn(async move {
if let Err(err) = http1::Builder::new()
.serve_connection(io, service_clone)
.await
{
eprintln!("server error: {}", err);
}
});
}
}
```