async-graphql/src/extensions/apollo_tracing.rs

137 lines
4.3 KiB
Rust
Raw Normal View History

2020-09-29 12:47:37 +00:00
use crate::extensions::{Extension, ExtensionContext, ExtensionFactory, ResolveInfo};
2020-10-10 02:32:43 +00:00
use crate::{Value, Variables};
2020-04-28 07:01:19 +00:00
use chrono::{DateTime, Utc};
use serde::ser::SerializeMap;
use serde::{Serialize, Serializer};
use std::collections::BTreeMap;
2020-10-10 02:32:43 +00:00
use std::convert::TryInto;
2020-04-28 07:01:19 +00:00
use std::ops::Deref;
struct PendingResolve {
path: serde_json::Value,
field_name: String,
parent_type: String,
return_type: String,
start_time: DateTime<Utc>,
}
struct ResolveStat {
pending_resolve: PendingResolve,
end_time: DateTime<Utc>,
start_offset: i64,
}
impl Deref for ResolveStat {
type Target = PendingResolve;
fn deref(&self) -> &Self::Target {
&self.pending_resolve
}
}
impl Serialize for ResolveStat {
fn serialize<S: Serializer>(&self, serializer: S) -> std::result::Result<S::Ok, S::Error> {
let mut map = serializer.serialize_map(None)?;
map.serialize_entry("path", &self.path)?;
map.serialize_entry("fieldName", &self.field_name)?;
map.serialize_entry("parentType", &self.parent_type)?;
map.serialize_entry("returnType", &self.return_type)?;
map.serialize_entry("startOffset", &self.start_offset)?;
map.serialize_entry(
"duration",
&(self.end_time - self.start_time).num_nanoseconds(),
)?;
map.end()
}
}
/// Apollo tracing extension for performance tracing
///
2020-09-14 18:38:41 +00:00
/// Apollo Tracing works by including data in the extensions field of the GraphQL response, which is
/// reserved by the GraphQL spec for extra information that a server wants to return. That way, you
/// have access to performance traces alongside the data returned by your query.
/// It's already supported by `Apollo Engine`, and we're excited to see what other kinds of
/// integrations people can build on top of this format.
2020-09-15 18:32:13 +00:00
#[cfg_attr(feature = "nightly", doc(cfg(feature = "apollo_tracing")))]
2020-09-29 12:47:37 +00:00
pub struct ApolloTracing;
2020-04-28 07:01:19 +00:00
2020-09-29 12:47:37 +00:00
impl ExtensionFactory for ApolloTracing {
fn create(&self) -> Box<dyn Extension> {
Box::new(ApolloTracingExtension {
2020-04-28 07:01:19 +00:00
start_time: Utc::now(),
end_time: Utc::now(),
pending_resolves: Default::default(),
resolves: Default::default(),
2020-09-29 12:47:37 +00:00
})
2020-04-28 07:01:19 +00:00
}
}
2020-09-29 12:47:37 +00:00
struct ApolloTracingExtension {
start_time: DateTime<Utc>,
end_time: DateTime<Utc>,
pending_resolves: BTreeMap<usize, PendingResolve>,
resolves: Vec<ResolveStat>,
}
impl Extension for ApolloTracingExtension {
2020-04-28 07:01:19 +00:00
fn name(&self) -> Option<&'static str> {
Some("tracing")
}
2020-09-29 12:47:37 +00:00
fn parse_start(
&mut self,
_ctx: &ExtensionContext<'_>,
_query_source: &str,
_variables: &Variables,
) {
self.start_time = Utc::now();
2020-04-28 07:01:19 +00:00
}
2020-09-29 12:47:37 +00:00
fn execution_end(&mut self, _ctx: &ExtensionContext<'_>) {
self.end_time = Utc::now();
2020-04-28 07:01:19 +00:00
}
2020-09-29 12:47:37 +00:00
fn resolve_start(&mut self, _ctx: &ExtensionContext<'_>, info: &ResolveInfo<'_>) {
self.pending_resolves.insert(
2020-04-28 07:01:19 +00:00
info.resolve_id.current,
PendingResolve {
2020-07-15 10:05:24 +00:00
path: serde_json::to_value(info.path_node).unwrap(),
2020-04-28 07:01:19 +00:00
field_name: info.path_node.field_name().to_string(),
parent_type: info.parent_type.to_string(),
return_type: info.return_type.to_string(),
start_time: Utc::now(),
},
);
}
2020-09-29 12:47:37 +00:00
fn resolve_end(&mut self, _ctx: &ExtensionContext<'_>, info: &ResolveInfo<'_>) {
if let Some(pending_resolve) = self.pending_resolves.remove(&info.resolve_id.current) {
let start_offset = (pending_resolve.start_time - self.start_time)
2020-04-28 07:01:19 +00:00
.num_nanoseconds()
.unwrap();
self.resolves.push(ResolveStat {
2020-04-28 07:01:19 +00:00
pending_resolve,
start_offset,
end_time: Utc::now(),
});
}
}
2020-10-10 02:32:43 +00:00
fn result(&mut self, _ctx: &ExtensionContext<'_>) -> Option<Value> {
self.resolves
2020-04-28 07:01:19 +00:00
.sort_by(|a, b| a.start_offset.cmp(&b.start_offset));
2020-10-10 02:32:43 +00:00
serde_json::json!({
2020-04-28 07:01:19 +00:00
"version": 1,
"startTime": self.start_time.to_rfc3339(),
"endTime": self.end_time.to_rfc3339(),
"duration": (self.end_time - self.start_time).num_nanoseconds(),
2020-04-28 07:01:19 +00:00
"execution": {
"resolvers": self.resolves
2020-04-28 07:01:19 +00:00
}
2020-10-10 02:32:43 +00:00
})
.try_into()
.ok()
2020-04-28 07:01:19 +00:00
}
}