Skip to content

Commit

Permalink
[COLIB-522]: Fix the trace propagation for engine gRPC calls
Browse files Browse the repository at this point in the history
  • Loading branch information
pavel-procopiuc committed Dec 3, 2024
1 parent 941d945 commit a0d0066
Show file tree
Hide file tree
Showing 6 changed files with 75 additions and 9 deletions.
3 changes: 3 additions & 0 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ auth0 = [
"dashmap",
"tracing",
]
grpc = ["tonic"]
gzip = ["reqwest/gzip"]
redis-tls = ["redis/tls", "redis/tokio-native-tls-comp"]
tracing_opentelemetry = ["tracing_opentelemetry_0_27"]
Expand Down Expand Up @@ -97,6 +98,7 @@ serde = { version = "1.0", features = ["derive"] }
serde_json = "1.0"
thiserror = "2"
tokio = { version = "1.16", features = ["macros", "rt-multi-thread", "fs"] }
tonic = { version = "0.12", default-features = false, optional = true }
tracing = { version = "0.1", optional = true }
uuid = { version = ">=0.7.0, <2.0.0", features = ["serde", "v4"] }
chacha20poly1305 = { version = "0.10.1", features = ["std"], optional = true }
Expand Down Expand Up @@ -130,6 +132,7 @@ tracing-opentelemetry_0_28_pkg = { package = "tracing-opentelemetry", version =
flate2 = "1.0"
mockito = "1.0"
tokio = { version = "1.16", features = ["macros", "rt-multi-thread"] }
tonic = "0.12"

[profile.release]
codegen-units = 1
Expand Down
2 changes: 1 addition & 1 deletion Makefile.toml
Original file line number Diff line number Diff line change
Expand Up @@ -140,7 +140,7 @@ args = [
[tasks.docs]
description = "Build docs as they are rendered on docs.rs"
command = "cargo"
args = ["doc", "--document-private-items", "--features=auth0,gzip", "--no-deps"]
args = ["doc", "--document-private-items", "--features=auth0,gzip,grpc", "--no-deps"]
env = { "RUSTDOCFLAGS" = "-Dwarnings" }

[tasks.release]
Expand Down
15 changes: 8 additions & 7 deletions src/lib.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
#![cfg_attr(docsrs, feature(doc_cfg))]

//! This crate gives an high level API to execute external HTTP requests.
//! This crate gives a high level API to execute external HTTP requests.
//!
//! It is supposed to give the basics building blocks for building bridges to other services
//! while abstracting the low level stuff like adding custom headers and request tracing.
Expand All @@ -21,12 +21,11 @@
//! * `auth0` - enable auth0 integration, allowing bridge.rs to retrieve tokens from auth0 for authentication
//! * `gzip` - provides response body gzip decompression.
//! * `redis-tls` - add support for connecting to redis with tls
//! * `tracing-opentelemetry` adds support for integration with opentelemetry.
//! This feature is an alias for the `tracing_opentelemetry_0_21` feature.
//! `tracing_opentelemetry_0_20` is also available as to support the 0.20 opentelemetry
//! libraries.
//!
//! We are going to support at least the last 3 versions of opentelemetry. After that we mightremove support for older otel version without it being a breaking change.
//! * `grpc` - provides the [GrpcOtelInterceptor] for adding the opentelemetry context to the gRPC requests
//! * `tracing_opentelemetry` - adds support for integration with opentelemetry.
//! This feature is an alias for the latest `tracing_opentelemetry_x_xx` feature.
//! * `tracing_opentelemetry_x_xx` (e.g. `tracing_opentelemetry_0_27`) - adds support for integration with a particular opentelemetry version.
//! We are going to support at least the last 3 versions of opentelemetry. After that we might remove support for older otel version without it being a breaking change.
use errors::PrimaBridgeError;
use http::{header::HeaderName, HeaderValue, Method};
Expand All @@ -43,6 +42,8 @@ pub use self::{
response::graphql::{Error, ParsedGraphqlResponse, ParsedGraphqlResponseExt, PossiblyParsedData},
response::Response,
};
#[cfg(all(feature = "grpc", feature = "_any_otel_version"))]
pub use request::grpc::{GrpcOtelInterceptedService, GrpcOtelInterceptor};

mod builder;
mod errors;
Expand Down
58 changes: 58 additions & 0 deletions src/request/grpc.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
use std::str::FromStr;

use tonic::metadata::{MetadataKey, MetadataMap, MetadataValue};
use tonic::service::interceptor::InterceptedService;
use tonic::service::Interceptor;
use tonic::{Request, Status};

use super::otel::{inject_context, Injector};

/// A gRPC interceptor that injects the current tracing context into the request metadata.
///
/// Use this interceptor to make sure that DataDog traces are connected between the services when calling the gRPC
/// service on another server.
///
/// # Example
///
/// ```
/// # struct QuotePreviewServiceClient<S> { _d: std::marker::PhantomData<S> }
/// # impl QuotePreviewServiceClient<GrpcOtelInterceptedService<Channel>> {
/// # pub fn with_interceptor(endpoint: Channel, interceptor: GrpcOtelInterceptor) -> Self { Self { _d: std::marker::PhantomData } }
/// # }
/// use tonic::transport::{Endpoint, Channel};
///
/// use prima_bridge::{GrpcOtelInterceptor, GrpcOtelInterceptedService};
///
/// async fn make_grpc_service() -> Result<QuotePreviewServiceClient<GrpcOtelInterceptedService<Channel>>, Box<dyn std::error::Error>> {
/// let url = "http://...";
/// let channel = Endpoint::new(url)?.connect().await?;
/// // QuotePreviewServiceClient is a tonic-generated gRPC client from [https://github.com/primait/es-engine-schema]
/// Ok(QuotePreviewServiceClient::with_interceptor(channel, GrpcOtelInterceptor))
/// }
/// ```
#[derive(Clone)]
pub struct GrpcOtelInterceptor;

/// Convenience type alias for a long type that's returned from the `with_interceptor()` function of the tonic-generated
/// gRPC client when used with [GrpcOtelInterceptor].
pub type GrpcOtelInterceptedService<Svc> = InterceptedService<Svc, GrpcOtelInterceptor>;

impl Interceptor for GrpcOtelInterceptor {
fn call(&mut self, mut request: Request<()>) -> Result<Request<()>, Status> {
inject_context(&mut MetadataMapInjector(request.metadata_mut()));
Ok(request)
}
}

pub struct MetadataMapInjector<'h>(&'h mut MetadataMap);

impl Injector for MetadataMapInjector<'_> {
fn set(&mut self, key: &str, value: String) {
let key_value = MetadataKey::from_str(key)
.ok()
.and_then(|key| Some((key, MetadataValue::from_str(&value).ok()?)));
if let Some((key, value)) = key_value {
self.0.insert(key, value);
}
}
}
2 changes: 2 additions & 0 deletions src/request/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,8 @@ use crate::{BridgeClient, BridgeImpl, Response};
mod body;
mod request_type;

#[cfg(all(feature = "grpc", feature = "_any_otel_version"))]
pub mod grpc;
#[cfg(feature = "_any_otel_version")]
mod otel;

Expand Down
4 changes: 3 additions & 1 deletion src/request/otel.rs
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,9 @@ mod otel_crates {

use otel_crates::*;

use opentelemetry::propagation::{Injector, TextMapPropagator};
pub use opentelemetry::propagation::Injector;

use opentelemetry::propagation::TextMapPropagator;
use opentelemetry_sdk::propagation::TraceContextPropagator;
use tracing_opentelemetry::OpenTelemetrySpanExt;

Expand Down

0 comments on commit a0d0066

Please sign in to comment.