use std::future::Future;
use std::io::Read;
use std::sync::Arc;
use std::time::Duration;
use std::time::Instant;
use anyhow::anyhow;
use apollo_router::TestHarness;
use apollo_router::make_fake_batch;
use apollo_router::services::router;
use apollo_router::services::router::BoxCloneService;
use apollo_router::services::supergraph;
use axum::Extension;
use axum::Json;
use axum::body::Bytes;
use axum::routing::post;
use flate2::read::GzDecoder;
use http::header::ACCEPT;
use http_body_util::BodyExt as _;
use once_cell::sync::Lazy;
use prost::Message;
use proto::reports::Report;
use tokio::sync::Mutex;
use tokio::task::JoinHandle;
use tower::Service;
use tower::ServiceExt;
use tower_http::decompression::DecompressionLayer;
use tracing_common::proto;
mod tracing_common;
static ROUTER_SERVICE_RUNTIME: Lazy<Arc<tokio::runtime::Runtime>> = Lazy::new(|| {
Arc::new(tokio::runtime::Runtime::new().expect("must be able to create tokio runtime"))
});
async fn config(
use_legacy_request_span: bool,
reports: Arc<Mutex<Vec<Report>>>,
demand_control: bool,
experimental_field_stats: bool,
config_str: &str,
) -> (JoinHandle<()>, serde_json::Value) {
*apollo_router::_private::APOLLO_KEY.lock() = Some("test".to_string());
*apollo_router::_private::APOLLO_GRAPH_REF.lock() = Some("test".to_string());
let listener = tokio::net::TcpListener::bind("127.0.0.1:0").await.unwrap();
let addr = listener.local_addr().unwrap();
let app = axum::Router::new()
.route("/", post(report))
.layer(DecompressionLayer::new())
.layer(tower_http::add_extension::AddExtensionLayer::new(reports));
let task = ROUTER_SERVICE_RUNTIME.spawn(async move {
axum::serve(listener, app.into_make_service())
.await
.expect("could not start axum server")
});
let mut config = serde_yaml::from_str(config_str).expect("config yaml was invalid");
config = jsonpath_lib::replace_with(config, "$.telemetry.apollo.endpoint", &mut |_| {
Some(serde_json::Value::String(format!("http://{addr}")))
})
.expect("Could not sub in endpoint");
config =
jsonpath_lib::replace_with(config, "$.telemetry.spans.legacy_request_span", &mut |_| {
Some(serde_json::Value::Bool(use_legacy_request_span))
})
.expect("Could not sub in endpoint");
config = jsonpath_lib::replace_with(config, "$.demand_control.enabled", &mut |_| {
Some(serde_json::Value::Bool(demand_control))
})
.expect("Could not sub in demand_control");
config = jsonpath_lib::replace_with(
config,
"$.telemetry.apollo.experimental_local_field_metrics",
&mut |_| Some(serde_json::Value::Bool(experimental_field_stats)),
)
.expect("Could not sub in experimental_local_field_metrics");
(task, config)
}
async fn get_router_service(
reports: Arc<Mutex<Vec<Report>>>,
use_legacy_request_span: bool,
mocked: bool,
demand_control: bool,
experimental_local_field_metrics: bool,
config_str: Option<&str>,
) -> (JoinHandle<()>, BoxCloneService) {
let (task, config) = config(
use_legacy_request_span,
reports,
demand_control,
experimental_local_field_metrics,
config_str.unwrap_or(include_str!("fixtures/reports/apollo_reports.router.yaml")),
)
.await;
let builder = TestHarness::builder()
.try_log_level("INFO")
.configuration_json(config)
.expect("test harness had config errors")
.schema(include_str!("fixtures/supergraph.graphql"));
let builder = if mocked {
builder.subgraph_hook(|subgraph, _service| tracing_common::subgraph_mocks(subgraph))
} else {
builder.with_subgraph_network_requests()
};
(
task,
builder
.build_router()
.await
.expect("could create router test harness"),
)
}
async fn get_batch_router_service(
reports: Arc<Mutex<Vec<Report>>>,
use_legacy_request_span: bool,
mocked: bool,
demand_control: bool,
experimental_local_field_metrics: bool,
config_str: Option<&str>,
) -> (JoinHandle<()>, BoxCloneService) {
let (task, config) = config(
use_legacy_request_span,
reports,
demand_control,
experimental_local_field_metrics,
config_str.unwrap_or(include_str!(
"fixtures/reports/apollo_reports_batch.router.yaml"
)),
)
.await;
let builder = TestHarness::builder()
.try_log_level("INFO")
.configuration_json(config)
.expect("test harness had config errors")
.schema(include_str!("fixtures/supergraph.graphql"));
let builder = if mocked {
builder.subgraph_hook(|subgraph, _service| tracing_common::subgraph_mocks(subgraph))
} else {
builder.with_subgraph_network_requests()
};
(
task,
builder
.build_router()
.await
.expect("could create router test harness"),
)
}
macro_rules! assert_report {
($report: expr)=> {
insta::with_settings!({sort_maps => true}, {
insta::assert_yaml_snapshot!($report, {
".**.agent_version" => "[agent_version]",
".**.executable_schema_id" => "[executable_schema_id]",
".**.agent_id" => "[agent_id]",
".header.hostname" => "[hostname]",
".header.uname" => "[uname]",
".**.seconds" => "[seconds]",
".**.nanos" => "[nanos]",
".**.duration_ns" => "[duration_ns]",
".**.child[].start_time" => "[start_time]",
".**.child[].end_time" => "[end_time]",
".**.trace_id.value[]" => "[trace_id]",
".**.sent_time_offset" => "[sent_time_offset]",
".**.my_trace_id" => "[my_trace_id]",
".**.latency_count" => "[latency_count]",
".**.cache_latency_count" => "[cache_latency_count]",
".**.public_cache_ttl_count" => "[public_cache_ttl_count]",
".**.private_cache_ttl_count" => "[private_cache_ttl_count]",
});
});
}
}
pub(crate) mod plugins {
pub(crate) mod telemetry {
pub(crate) mod apollo_exporter {
use serde::ser::SerializeStruct;
pub(crate) fn serialize_timestamp<S>(
timestamp: &Option<prost_types::Timestamp>,
serializer: S,
) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
match timestamp {
Some(ts) => {
let mut ts_strukt = serializer.serialize_struct("Timestamp", 2)?;
ts_strukt.serialize_field("seconds", &ts.seconds)?;
ts_strukt.serialize_field("nanos", &ts.nanos)?;
ts_strukt.end()
}
None => serializer.serialize_none(),
}
}
}
}
}
async fn report(
Extension(state): Extension<Arc<Mutex<Vec<Report>>>>,
bytes: Bytes,
) -> Result<Json<()>, http::StatusCode> {
let mut gz = GzDecoder::new(&*bytes);
let mut buf = Vec::new();
gz.read_to_end(&mut buf)
.expect("could not decompress bytes");
let report = Report::decode(&*buf).expect("could not deserialize report");
state.lock().await.push(report);
Ok(Json(()))
}
async fn get_trace_report(
reports: Arc<Mutex<Vec<Report>>>,
request: router::Request,
use_legacy_request_span: bool,
demand_control: bool,
experimental_local_field_metrics: bool,
config_str: Option<&'static str>,
) -> Report {
get_report(
get_router_service,
reports,
use_legacy_request_span,
false,
request,
demand_control,
experimental_local_field_metrics,
|r| {
!r.traces_per_query
.values()
.next()
.expect("traces and stats required")
.trace
.is_empty()
},
config_str,
)
.await
}
async fn get_batch_trace_report(
reports: Arc<Mutex<Vec<Report>>>,
request: router::Request,
use_legacy_request_span: bool,
demand_control: bool,
experimental_local_field_metrics: bool,
config_str: Option<&'static str>,
) -> Report {
get_report(
get_batch_router_service,
reports,
use_legacy_request_span,
false,
request,
demand_control,
experimental_local_field_metrics,
|r| {
!r.traces_per_query
.values()
.next()
.expect("traces and stats required")
.trace
.is_empty()
},
config_str,
)
.await
}
fn has_metrics(r: &&Report) -> bool {
!r.traces_per_query
.values()
.next()
.expect("traces and stats required")
.stats_with_context
.is_empty()
}
async fn get_metrics_report(
reports: Arc<Mutex<Vec<Report>>>,
request: router::Request,
demand_control: bool,
experimental_local_field_metrics: bool,
config_str: Option<&'static str>,
) -> Report {
get_report(
get_router_service,
reports,
false,
false,
request,
demand_control,
experimental_local_field_metrics,
has_metrics,
config_str,
)
.await
}
async fn get_batch_metrics_report(
reports: Arc<Mutex<Vec<Report>>>,
request: router::Request,
) -> u64 {
get_batch_stats_report(reports, false, request, has_metrics).await
}
async fn get_metrics_report_mocked(
reports: Arc<Mutex<Vec<Report>>>,
request: router::Request,
config_str: Option<&'static str>,
) -> Report {
get_report(
get_router_service,
reports,
false,
true,
request,
false,
false,
has_metrics,
config_str,
)
.await
}
#[allow(clippy::too_many_arguments)]
async fn get_report<Fut, T: Fn(&&Report) -> bool + Send + Sync + Copy + 'static>(
service_fn: impl FnOnce(
Arc<Mutex<Vec<Report>>>,
bool,
bool,
bool,
bool,
Option<&'static str>,
) -> Fut,
reports: Arc<Mutex<Vec<Report>>>,
use_legacy_request_span: bool,
mocked: bool,
request: router::Request,
demand_control: bool,
experimental_local_field_metrics: bool,
filter: T,
config_str: Option<&'static str>,
) -> Report
where
Fut: Future<Output = (JoinHandle<()>, BoxCloneService)>,
{
reports.lock().await.clear();
let (task, mut service) = service_fn(
reports.clone(),
use_legacy_request_span,
mocked,
demand_control,
experimental_local_field_metrics,
config_str,
)
.await;
let response = service
.ready()
.await
.expect("router service was never ready")
.call(request)
.await
.expect("router service call failed");
let mut found_report = match response
.response
.into_body()
.collect()
.await
.map(|b| String::from_utf8(b.to_bytes().to_vec()))
{
Ok(Ok(response)) => {
if response.contains("errors") {
eprintln!("response had errors {response}");
}
Ok(None)
}
_ => Err(anyhow!("error retrieving response")),
};
let deadline = Instant::now() + Duration::from_secs(10);
loop {
let my_reports = reports.lock().await;
let report = my_reports.iter().find(filter);
if report.is_some() && matches!(found_report, Ok(None)) {
found_report = Ok(report.cloned());
break;
}
drop(my_reports);
assert!(
Instant::now() < deadline,
"timed out waiting for matching report"
);
tokio::time::sleep(Duration::from_millis(50)).await;
}
task.abort();
found_report
.expect("failed to get report")
.expect("failed to find report")
}
async fn get_batch_stats_report<T: Fn(&&Report) -> bool + Send + Sync + Copy + 'static>(
reports: Arc<Mutex<Vec<Report>>>,
mocked: bool,
request: router::Request,
filter: T,
) -> u64 {
reports.lock().await.clear();
let (task, mut service) =
get_batch_router_service(reports.clone(), mocked, false, false, false, None).await;
let response = service
.ready()
.await
.expect("router service was never ready")
.call(request)
.await
.expect("router service call failed");
let _found_report = response.response.into_body().collect().await;
let deadline = Instant::now() + Duration::from_secs(10);
let request_count = loop {
let mut count = 0;
for report in reports.lock().await.iter().filter(filter) {
let stats = &report
.traces_per_query
.values()
.next()
.expect("has something")
.stats_with_context;
count += stats[0].query_latency_stats.as_ref().unwrap().request_count;
}
if count > 0 {
break count;
}
assert!(
Instant::now() < deadline,
"timed out waiting for batch stats reports"
);
tokio::time::sleep(Duration::from_millis(50)).await;
};
task.abort();
request_count
}
#[tokio::test(flavor = "multi_thread")]
async fn non_defer() {
for use_legacy_request_span in [true, false] {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report =
get_trace_report(reports, req, use_legacy_request_span, false, false, None).await;
assert_report!(report);
}
}
#[tokio::test(flavor = "multi_thread")]
async fn test_condition_if() {
for use_legacy_request_span in [true, false] {
let request = supergraph::Request::fake_builder()
.query("query($if: Boolean!) {topProducts { name ... @defer(if: $if) { reviews { author { name } } reviews { author { name } } }}}")
.variable("if", true)
.header(ACCEPT, "multipart/mixed;deferSpec=20220824")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report =
get_trace_report(reports, req, use_legacy_request_span, false, false, None).await;
assert_report!(report);
}
}
#[tokio::test(flavor = "multi_thread")]
async fn test_condition_else() {
for use_legacy_request_span in [true, false] {
let request = supergraph::Request::fake_builder()
.query("query($if: Boolean!) {topProducts { name ... @defer(if: $if) { reviews { author { name } } reviews { author { name } } }}}")
.variable("if", false)
.header(ACCEPT, "multipart/mixed;deferSpec=20220824")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report =
get_trace_report(reports, req, use_legacy_request_span, false, false, None).await;
assert_report!(report);
}
}
#[tokio::test(flavor = "multi_thread")]
async fn test_trace_id() {
for use_legacy_request_span in [true, false] {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report =
get_trace_report(reports, req, use_legacy_request_span, false, false, None).await;
assert_report!(report);
}
}
#[tokio::test(flavor = "multi_thread")]
async fn test_batch_trace_id() {
for use_legacy_request_span in [true, false] {
let request = make_fake_batch(
supergraph::Request::fake_builder()
.query("query one {topProducts{name reviews {author{name}} reviews{author{name}}}}")
.operation_name("one")
.build()
.unwrap()
.supergraph_request,
Some(("one", "two")),
);
let reports = Arc::new(Mutex::new(vec![]));
let report = get_batch_trace_report(
reports,
request.into(),
use_legacy_request_span,
false,
false,
None,
)
.await;
assert_report!(report);
}
}
#[tokio::test(flavor = "multi_thread")]
async fn test_trace_with_client_name_http_header() {
for use_legacy_request_span in [true, false] {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.header("apollographql-client-name", "my client")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report =
get_trace_report(reports, req, use_legacy_request_span, false, false, None).await;
assert_report!(report);
}
}
#[tokio::test(flavor = "multi_thread")]
async fn test_trace_with_client_version_http_header() {
for use_legacy_request_span in [true, false] {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.header("apollographql-client-version", "my client version")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report =
get_trace_report(reports, req, use_legacy_request_span, false, false, None).await;
assert_report!(report);
}
}
#[tokio::test(flavor = "multi_thread")]
async fn test_metrics_with_client_name_http_header() {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.header("apollographql-client-name", "my client")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(reports, req, false, false, None).await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_metrics_with_client_version_http_header() {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.header("apollographql-client-version", "my client version")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(reports, req, false, false, None).await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_metrics_with_library_name_http_header() {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.header("apollographql-library-name", "apollo library")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(reports, req, false, false, None).await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_metrics_with_library_version_http_header() {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.header("apollographql-library-version", "apollo library version")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(reports, req, false, false, None).await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_metrics_with_library_name_request_extension() {
let mut clients_map = serde_json_bytes::map::Map::new();
clients_map.insert("name", "apollo library".into());
let mut extensions_map = serde_json_bytes::map::Map::new();
extensions_map.insert("clientLibrary", clients_map.into());
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.extensions(extensions_map.clone())
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(reports, req, false, false, None).await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_metrics_with_library_version_request_extension() {
let mut clients_map = serde_json_bytes::map::Map::new();
clients_map.insert("version", "apollo library version".into());
let mut extensions_map = serde_json_bytes::map::Map::new();
extensions_map.insert("clientLibrary", clients_map.into());
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.extensions(extensions_map.clone())
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(reports, req, false, false, None).await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_send_header() {
for use_legacy_request_span in [true, false] {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.header("send-header", "Header value")
.header("dont-send-header", "Header value")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report =
get_trace_report(reports, req, use_legacy_request_span, false, false, None).await;
assert_report!(report);
}
}
#[tokio::test(flavor = "multi_thread")]
async fn test_batch_send_header() {
for use_legacy_request_span in [true, false] {
let request = make_fake_batch(
supergraph::Request::fake_builder()
.query("query one {topProducts{name reviews {author{name}} reviews{author{name}}}}")
.operation_name("one")
.header("send-header", "Header value")
.header("dont-send-header", "Header value")
.build()
.unwrap()
.supergraph_request,
Some(("one", "two")),
);
let reports = Arc::new(Mutex::new(vec![]));
let report = get_batch_trace_report(
reports,
request.into(),
use_legacy_request_span,
false,
false,
None,
)
.await;
assert_report!(report);
}
}
#[tokio::test(flavor = "multi_thread")]
async fn test_send_variable_value() {
for use_legacy_request_span in [true, false] {
let request = supergraph::Request::fake_builder()
.query("query($sendValue:Boolean!, $dontSendValue: Boolean!){topProducts{name reviews @include(if: $sendValue) {author{name}} reviews @include(if: $dontSendValue){author{name}}}}")
.variable("sendValue", true)
.variable("dontSendValue", true)
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report =
get_trace_report(reports, req, use_legacy_request_span, false, false, None).await;
assert_report!(report);
}
}
#[tokio::test(flavor = "multi_thread")]
async fn test_stats() {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(reports, req, false, false, None).await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_batch_stats() {
let request = make_fake_batch(
supergraph::Request::fake_builder()
.query("query one {topProducts{name reviews {author{name}} reviews{author{name}}}}")
.operation_name("one")
.build()
.unwrap()
.supergraph_request,
Some(("one", "two")),
);
let reports = Arc::new(Mutex::new(vec![]));
let request_count = get_batch_metrics_report(reports, request.into()).await;
assert!(request_count == 1 || request_count == 2);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_stats_mocked() {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report_mocked(reports, req, None).await;
let per_query = report.traces_per_query.values().next().unwrap();
let stats = per_query.stats_with_context.first().unwrap();
insta::with_settings!({sort_maps => true}, {
insta::assert_yaml_snapshot!(stats, {
".query_latency_stats.latency_count" => "[latency_count]"
});
});
}
#[tokio::test(flavor = "multi_thread")]
async fn test_new_field_stats() {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(reports, req, true, true, None).await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_demand_control_stats() {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(reports, req, true, false, None).await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_demand_control_trace() {
for use_legacy_request_span in [true, false] {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report =
get_trace_report(reports, req, use_legacy_request_span, true, false, None).await;
assert_report!(report);
}
}
#[tokio::test(flavor = "multi_thread")]
async fn test_demand_control_trace_batched() {
for use_legacy_request_span in [true, false] {
let request = make_fake_batch(
supergraph::Request::fake_builder()
.query("query one {topProducts{name reviews {author{name}} reviews{author{name}}}}")
.operation_name("one")
.build()
.unwrap()
.supergraph_request,
Some(("one", "two")),
);
let req: router::Request = request.into();
let reports = Arc::new(Mutex::new(vec![]));
let report =
get_batch_trace_report(reports, req, use_legacy_request_span, true, false, None).await;
assert_report!(report);
}
}
#[tokio::test(flavor = "multi_thread")]
async fn test_features_enabled() {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(
reports,
req,
false,
false,
Some(include_str!(
"fixtures/reports/all_features_enabled.router.yaml"
)),
)
.await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_features_disabled() {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name reviews {author{name}} reviews{author{name}}}}")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(
reports,
req,
false,
false,
Some(include_str!(
"fixtures/reports/all_features_disabled.router.yaml"
)),
)
.await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_persisted_query_by_id_stats() {
let request = supergraph::Request::fake_builder()
.extension(
"persistedQuery",
serde_json::json!({
"version": 1,
"sha256Hash": "test_pq_id"
}),
)
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(
reports,
req,
false,
false,
Some(include_str!("fixtures/reports/pq_enabled.router.yaml")),
)
.await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_persisted_query_by_safelist_body_stats() {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name}}")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(
reports,
req,
false,
false,
Some(include_str!("fixtures/reports/pq_enabled.router.yaml")),
)
.await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_persisted_query_by_id_logging_only_stats() {
let request = supergraph::Request::fake_builder()
.extension(
"persistedQuery",
serde_json::json!({
"version": 1,
"sha256Hash": "test_pq_id"
}),
)
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(
reports,
req,
false,
false,
Some(include_str!("fixtures/reports/pq_logging.router.yaml")),
)
.await;
assert_report!(report);
}
#[tokio::test(flavor = "multi_thread")]
async fn test_persisted_query_by_safelist_body_logging_pq_only_stats() {
let request = supergraph::Request::fake_builder()
.query("query{topProducts{name}}")
.build()
.unwrap();
let req: router::Request = request.try_into().expect("could not convert request");
let reports = Arc::new(Mutex::new(vec![]));
let report = get_metrics_report(
reports,
req,
false,
false,
Some(include_str!("fixtures/reports/pq_logging.router.yaml")),
)
.await;
assert_report!(report);
}