Skip to content

Commit

Permalink
XXX Better code documentation
Browse files Browse the repository at this point in the history
  • Loading branch information
nshalman committed Dec 12, 2024
1 parent 4092c21 commit 1db15e7
Show file tree
Hide file tree
Showing 3 changed files with 43 additions and 36 deletions.
4 changes: 4 additions & 0 deletions dropshot/examples/multiple-servers.rs
Original file line number Diff line number Diff line change
Expand Up @@ -100,6 +100,10 @@ use tokio::sync::Mutex;

#[tokio::main]
async fn main() -> Result<(), String> {
// XXX Is there interest in adding the optional integration into some of the existing examples?
#[cfg(feature = "otel-tracing")]
let _otel_guard = equinix_otel_tools::init(env!("CARGO_CRATE_NAME"));

// Initial set of servers to start. Once they're running, we may add or
// remove servers based on client requests.
let initial_servers = [("A", "127.0.0.1:12345"), ("B", "127.0.0.1:12346")];
Expand Down
10 changes: 4 additions & 6 deletions dropshot/examples/otel.rs
Original file line number Diff line number Diff line change
Expand Up @@ -46,11 +46,8 @@ use opentelemetry_http::{Bytes, HeaderInjector};
async fn main() -> Result<(), String> {
let _otel_guard = equinix_otel_tools::init("otel-demo");

let config_dropshot: ConfigDropshot = ConfigDropshot {
bind_address: std::net::SocketAddr::new(
std::net::IpAddr::V4(std::net::Ipv4Addr::new(127, 0, 0, 1)),
4000,
),
let config_dropshot = ConfigDropshot {
bind_address: "127.0.0.1:4000".parse().unwrap(),
..Default::default()
};

Expand Down Expand Up @@ -248,6 +245,7 @@ async fn example_api_error(
method = GET,
path = "/panic",
}]
#[cfg_attr(feature = "tokio-tracing", tracing::instrument(skip_all, err))]
async fn example_api_panic(
_rqctx: RequestContext<ExampleContext>,
) -> Result<HttpResponseOk<CounterValue>, HttpError> {
Expand All @@ -264,7 +262,7 @@ async fn example_api_panic(
method = PUT,
path = "/counter",
}]
#[cfg_attr(feature = "tokio-tracing", tracing::instrument)]
#[cfg_attr(feature = "tokio-tracing", tracing::instrument(skip_all, err))]
async fn example_api_put_counter(
rqctx: RequestContext<ExampleContext>,
update: TypedBody<CounterValue>,
Expand Down
65 changes: 35 additions & 30 deletions dropshot/src/otel.rs
Original file line number Diff line number Diff line change
@@ -1,33 +1,35 @@
// Copyright 2024 Oxide Computer Company
//! Opentelemetry tracing support
//!
// XXX Not sure if we want to just mimic reqwest-tracing
// or not...
//! Fields that we want to produce to provide comparable
//! functionality to reqwest-tracing[1]:
//! functionality to [reqwest-tracing]:
//!
//! - http.request.method
//! - url.scheme
//! - server.address
//! - server.port
//! - otel.kind
//! - otel.name
//! - otel.status_code
//! - user_agent.original
//! - http.response.status_code
//! - error.message
//! - error.cause_chain
//! - [ ] error.cause_chain
//! - [ ] error.message
//! - [x] http.request.method
//! - [x] http.response.status_code
//! - [x] otel.kind (showing up as span.kind?)
//! - [x] otel.name
//! - [ ] otel.status_code (needs investigation)
//! - [x] server.address
//! - [x] server.port
//! - [ ] url.scheme
//! - [x] user_agent.original
//!
//! [1] <https://docs.rs/reqwest-tracing/0.5.4/reqwest_tracing/macro.reqwest_otel_span.html>
//! Where possible we use the [opentelemetry-semantic-conventions] crate for naming.
//!
//! [reqwest-tracing]: https://docs.rs/reqwest-tracing/0.5.4/reqwest_tracing/macro.reqwest_otel_span.html
//! [opentelemetry-semantic-conventions]:
//! <https://docs.rs/opentelemetry-semantic-conventions/latest/opentelemetry_semantic_conventions/trace/index.html>
use opentelemetry::{
global, trace::Span, trace::Tracer, trace::TracerProvider,
};
use opentelemetry_http::HeaderExtractor;
use opentelemetry_semantic_conventions::trace;

// - url.scheme
// - server.address
// - server.port
// - user_agent.original
#[derive(Debug, Clone, serde::Serialize)]
pub(crate) struct RequestInfo {
pub id: String,
Expand All @@ -39,16 +41,14 @@ pub(crate) struct RequestInfo {
pub user_agent: String,
}

// - http.response.status_code
// - error.message
// - error.cause_chain
// - otel.status_code
pub(crate) struct ResponseInfo<'a> {
pub status_code: u16,
pub message: String,
pub error: Option<&'a crate::handler::HandlerError>,
}

/// Generate an opentelementry Context based on the headers
/// in the incoming hyper request.
fn extract_context_from_request(
request: &hyper::Request<hyper::body::Incoming>,
) -> opentelemetry::Context {
Expand All @@ -57,6 +57,7 @@ fn extract_context_from_request(
})
}

/// Create an opentelemetry Span to represent the handling of the incoming request.
pub fn create_request_span(
request: &hyper::Request<hyper::body::Incoming>,
) -> opentelemetry::global::BoxedSpan {
Expand All @@ -74,8 +75,14 @@ pub fn create_request_span(
.start_with_context(&tracer, &parent_cx)
}

/// This trait contains all functionality needed for dropshot to annotate
/// opentelemetry spans with information about a request and the corresponding response.
pub trait TraceDropshot {
/// Attach attributes to the span based on the provided
/// RequestInfo
fn trace_request(&mut self, request: RequestInfo);
/// Attach the information from the ResponseInfo to the span
/// including marking the span as an error if an error occurred.
fn trace_response(&mut self, response: ResponseInfo);
}

Expand All @@ -84,30 +91,28 @@ impl TraceDropshot for opentelemetry::global::BoxedSpan {
self.set_attributes(vec![
// Rename to dropshot.id ????
opentelemetry::KeyValue::new("http.id".to_string(), request.id),
opentelemetry::KeyValue::new(trace::URL_PATH, request.path),
opentelemetry::KeyValue::new(
"http.method".to_string(),
trace::HTTP_REQUEST_METHOD,
request.method,
),
opentelemetry::KeyValue::new("http.path".to_string(), request.path),
opentelemetry::KeyValue::new(
"server.address".to_string(),
trace::SERVER_ADDRESS,
request.local_addr.ip().to_string(),
),
opentelemetry::KeyValue::new(
"server.port".to_string(),
trace::SERVER_PORT,
request.local_addr.port().to_string(),
),
opentelemetry::KeyValue::new(
"user_agent.original".to_string(),
trace::USER_AGENT_ORIGINAL,
request.user_agent,
),
]);
}

// - [x] http.response.status_code
// - [x] error.message
// - [ ] error.cause_chain
// - [ ] otel.status_code
/// TODO: Do we want to differentiate between 4xx vs 5xx errors
/// and not mark 4xx spans as error spans?
fn trace_response(&mut self, response: ResponseInfo) {
self.set_attributes(vec![
opentelemetry::KeyValue::new(
Expand Down

0 comments on commit 1db15e7

Please sign in to comment.