Skip to content

Commit

Permalink
Set proper traceparent in the case of root spans
Browse files Browse the repository at this point in the history
  • Loading branch information
miguelff committed Jan 27, 2023
1 parent f97eea9 commit 4febed0
Show file tree
Hide file tree
Showing 2 changed files with 49 additions and 30 deletions.
23 changes: 18 additions & 5 deletions query-engine/core/src/telemetry/helpers.rs
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,24 @@ pub fn set_span_link_from_traceparent(span: &Span, traceparent: Option<String>)
}
}

pub fn get_trace_parent_from_span(span: &Span) -> String {
let cx = span.context();
let binding = cx.span();
let span_context = binding.span_context();

format!("00-{}-{}-01", span_context.trace_id(), span_context.span_id())
}

pub fn get_trace_id_from_span(span: &Span) -> TraceId {
let cx = span.context();
get_trace_id_from_context(&cx)
}

pub fn get_trace_id_from_context(context: &Context) -> TraceId {
let context_span = context.span();
context_span.span_context().trace_id()
}

pub fn get_trace_id_from_traceparent(traceparent: Option<&str>) -> TraceId {
traceparent
.unwrap_or("0-0-0-0")
Expand All @@ -50,11 +68,6 @@ pub fn get_trace_id_from_traceparent(traceparent: Option<&str>) -> TraceId {
.unwrap()
}

pub fn get_trace_id_from_context(context: &Context) -> TraceId {
let context_span = context.span();
context_span.span_context().trace_id()
}

pub enum QueryEngineLogLevel {
FromEnv,
Override(String),
Expand Down
56 changes: 31 additions & 25 deletions query-engine/query-engine/src/server/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ use hyper::service::{make_service_fn, service_fn};
use hyper::{header::CONTENT_TYPE, Body, HeaderMap, Method, Request, Response, Server, StatusCode};
use opentelemetry::trace::TraceContextExt;
use opentelemetry::{global, propagation::Extractor};
use query_core::helpers::get_trace_id_from_traceparent;
use query_core::helpers::*;
use query_core::{
schema::QuerySchemaRenderer, telemetry, ExtendedTransactionUserFacingError, TransactionOptions, TxId,
};
Expand Down Expand Up @@ -132,30 +132,36 @@ async fn graphql_handler(state: State, req: Request<Body>) -> Result<Response<Bo
let mut traceparent = traceparent(headers);
let mut trace_id = get_trace_id_from_traceparent(traceparent.as_deref());

// If telemetry needs to be captured, we use the span trace_id to correlate the logs happening
// during the different operations within a transaction. The trace_id is propagated in the
// traceparent header, but if it's not present, we need to synthetically create one for the
// transaction. This is needed, in case the client is interested in capturing logs and not
// traces, because:
// - The client won't send a traceparent header
// - A transaction initial span is created here (prisma:engine:itx_runner) and stored in the
// ITXServer for that transaction
// - When a query comes in, the graphql handler process it, but we need to tell the capturer
// to start capturing logs, and for that we need a trace_id. There are two places were we
// could get that information from:
// - First, it's the traceparent, but the client didn't send it, because they are only
// interested in logs.
// - Second, it's the root span for the transaction, but it's not in scope but instead
// stored in the ITXServer, in a different tokio task.
//
// For the above reasons, we need to create a trace_id that we can predict and use accross the
// different operations happening within a transaction. So we do it by converting the tx_id
// into a trace_id, leaning on the fact that the tx_id has more entropy, and there's no
// information loss.
if capture_settings.logs_enabled() && traceparent.is_none() && tx_id.is_some() {
let tx_id = tx_id.clone().unwrap();
traceparent = Some(tx_id.as_traceparent());
trace_id = tx_id.into();
if traceparent.is_none() {
// If telemetry needs to be captured, we use the span trace_id to correlate the logs happening
// during the different operations within a transaction. The trace_id is propagated in the
// traceparent header, but if it's not present, we need to synthetically create one for the
// transaction. This is needed, in case the client is interested in capturing logs and not
// traces, because:
// - The client won't send a traceparent header
// - A transaction initial span is created here (prisma:engine:itx_runner) and stored in the
// ITXServer for that transaction
// - When a query comes in, the graphql handler process it, but we need to tell the capturer
// to start capturing logs, and for that we need a trace_id. There are two places were we
// could get that information from:
// - First, it's the traceparent, but the client didn't send it, because they are only
// interested in logs.
// - Second, it's the root span for the transaction, but it's not in scope but instead
// stored in the ITXServer, in a different tokio task.
//
// For the above reasons, we need to create a trace_id that we can predict and use accross the
// different operations happening within a transaction. So we do it by converting the tx_id
// into a trace_id, leaning on the fact that the tx_id has more entropy, and there's no
// information loss.
if capture_settings.logs_enabled() && tx_id.is_some() {
let tx_id = tx_id.clone().unwrap();
traceparent = Some(tx_id.as_traceparent());
trace_id = tx_id.into();
} else {
// this is the root span, and we are in a single operation.
traceparent = Some(get_trace_parent_from_span(&span));
trace_id = get_trace_id_from_span(&span);
}
}
let capture_config = telemetry::capturing::capturer(trace_id, capture_settings);

Expand Down

0 comments on commit 4febed0

Please sign in to comment.