Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add ability for router to deal with query plans with contextual rewrites #5097

Merged
merged 78 commits into from
May 23, 2024
Merged
Show file tree
Hide file tree
Changes from 41 commits
Commits
Show all changes
78 commits
Select commit Hold shift + click to select a range
842175f
set context branch
clenfest Apr 29, 2024
17be8d7
Merge branch 'dev' into clenfest/set_context_working
o0Ignition0o Apr 29, 2024
88d898a
wip
o0Ignition0o Apr 29, 2024
a156efc
checkpoint
clenfest Apr 30, 2024
2ac6b46
it s not working yet but contextRewrites makes it to the router
o0Ignition0o Apr 30, 2024
631e1af
ok context makes it to variables.
o0Ignition0o Apr 30, 2024
0a50a26
ok we have a test now
o0Ignition0o Apr 30, 2024
c97e43c
wip
o0Ignition0o Apr 30, 2024
644cb7a
update tests
clenfest Apr 30, 2024
da6b2b5
checkpoint. 1 contextual value -> many consumers test
clenfest Apr 30, 2024
af096e4
5/1 evening checkpoint
clenfest May 2, 2024
5d71bf3
update mock
o0Ignition0o May 2, 2024
5810a9a
wip
o0Ignition0o May 2, 2024
a7efdce
Update to detct condition about when to rename args.
clenfest May 2, 2024
f2177cd
Adding query batching support for outgoing contextual requests
clenfest May 3, 2024
8412cbd
working test
clenfest May 3, 2024
54d5c78
updating pinned releases
clenfest May 3, 2024
db394b5
updates to git paths
clenfest May 3, 2024
4b8572b
use apollo parser primitives to spread variables in the subgraph oper…
o0Ignition0o May 4, 2024
3db9aa1
We allow context_rewrites to contain multiple rewrites for a single v…
clenfest May 7, 2024
5c2dd1f
Merge branch 'dev' into clenfest/set_context_working
clenfest May 7, 2024
0cf5d3d
make changes to apollo-federation to use context_rewrites
clenfest May 7, 2024
b2c2db4
Updates from linter
clenfest May 7, 2024
bb6bd95
add union test and bump pin
clenfest May 12, 2024
47bf117
cleaning up code
clenfest May 13, 2024
10b6eb7
Merge branch 'dev' into clenfest/set_context_working
clenfest May 13, 2024
8e882c3
lint
o0Ignition0o May 13, 2024
8c183ca
remove unwraps
o0Ignition0o May 13, 2024
5504835
change federation-rs pin
clenfest May 13, 2024
0ee4eb6
update snapshots that now expose contextRewrites
o0Ignition0o May 13, 2024
9376226
updating snapshots for latest version of federation
clenfest May 13, 2024
35aca9f
checkpoint
clenfest May 15, 2024
44154e5
another_checkpoint
clenfest May 15, 2024
2f5e374
refactor done, tests not working yet
clenfest May 16, 2024
b69175f
updating tests to be more DRY and snapshots
clenfest May 16, 2024
9e64881
adding some comments
clenfest May 16, 2024
49c5c82
update dependency
o0Ignition0o May 16, 2024
66992c2
update dependency requirements across examples
o0Ignition0o May 16, 2024
a50e786
wip
o0Ignition0o May 16, 2024
d413e87
wip
o0Ignition0o May 16, 2024
dbf8ce0
wip
o0Ignition0o May 16, 2024
65473b1
adding set context to license_enforcement
clenfest May 16, 2024
67ffcc3
checkpoint
clenfest May 17, 2024
d99229e
add a couple of tests
clenfest May 17, 2024
b848d22
validate subgraph operations + update snapshot ordering
o0Ignition0o May 17, 2024
403841a
lint
o0Ignition0o May 17, 2024
87c3280
clippy
o0Ignition0o May 17, 2024
44d64e3
duplicate fixture to address non ordering of variables
o0Ignition0o May 17, 2024
f76788a
Merge branch 'dev' into clenfest/set_context_working
clenfest May 17, 2024
3da3d82
cargo lock update
clenfest May 17, 2024
c6acd13
Add changeset
clenfest May 17, 2024
faaf790
updating redis cache keys
clenfest May 17, 2024
2212b9d
redis key update
clenfest May 17, 2024
323b263
Update apollo-router/src/query_planner/fetch.rs
clenfest May 17, 2024
089270f
Update apollo-router/src/query_planner/mod.rs
clenfest May 17, 2024
ec6cd98
add some failure tests
clenfest May 20, 2024
1730293
split out transform_operation as a separate function
clenfest May 20, 2024
7c23876
fmt and linter
clenfest May 20, 2024
991685b
Merge branch 'dev' into clenfest/set_context_working
clenfest May 20, 2024
323a5c7
accidently messed up test name
clenfest May 20, 2024
4cc3704
Merge branch 'dev' into clenfest/set_context_working
clenfest May 20, 2024
5abdfa8
update snapshots
o0Ignition0o May 21, 2024
d0736f2
update snapshots
o0Ignition0o May 21, 2024
1528d43
Merge branch 'dev' into clenfest/set_context_working
o0Ignition0o May 21, 2024
aa75f82
the test doesnt panic anymore?
o0Ignition0o May 21, 2024
d5942b1
Merge branch 'dev' into clenfest/set_context_working
clenfest May 21, 2024
9fe931d
Set required number of threads for any integration test to 2 to preve…
o0Ignition0o May 21, 2024
abb8b73
lint
o0Ignition0o May 21, 2024
55c51ba
override the dev profile for tests since there s no default
o0Ignition0o May 21, 2024
a64e3a2
move the nexttest profile to apollo-router so xtask picks it up
o0Ignition0o May 21, 2024
e0a3dae
update the nextest threads-required to be less agressive with the CI …
o0Ignition0o May 21, 2024
4f330c6
change the override so it applies to the CI profile
o0Ignition0o May 21, 2024
2ca8a9a
ok we re making progress it seems
o0Ignition0o May 21, 2024
b29b4f6
bump router-bridge version
clenfest May 23, 2024
26ddc08
Merge branch 'dev' into clenfest/set_context_working
clenfest May 23, 2024
df344c7
update redis cache key
clenfest May 23, 2024
9613cf6
bump bridge dependency
o0Ignition0o May 23, 2024
1307fe2
explicitly call out caches will be repopulated
o0Ignition0o May 23, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
679 changes: 273 additions & 406 deletions Cargo.lock

Large diffs are not rendered by default.

4 changes: 4 additions & 0 deletions apollo-federation/src/link/join_spec_definition.rs
Original file line number Diff line number Diff line change
Expand Up @@ -336,6 +336,10 @@ lazy_static! {
Version { major: 0, minor: 3 },
Some(Version { major: 2, minor: 0 }),
));
definitions.add(JoinSpecDefinition::new(
Version { major: 0, minor: 5 },
Some(Version { major: 2, minor: 8 }),
));
definitions
};

Expand Down
1 change: 1 addition & 0 deletions apollo-federation/src/query_plan/display.rs
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,7 @@ impl FetchNode {
operation_kind: _,
input_rewrites: _,
output_rewrites: _,
context_rewrites: _,
} = self;
state.write(format_args!("Fetch(service: {subgraph_name:?}"))?;
if let Some(id) = id {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -1307,6 +1307,7 @@ impl FetchDependencyGraphNode {
operation_kind: self.root_kind.into(),
input_rewrites: self.input_rewrites.clone(),
output_rewrites,
context_rewrites: Default::default(),
}));

Ok(Some(if let Some(path) = self.merge_at.clone() {
Expand Down
3 changes: 3 additions & 0 deletions apollo-federation/src/query_plan/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,9 @@ pub struct FetchNode {
/// Similar to `input_rewrites`, but for optional "rewrites" to apply to the data that is
/// received from a fetch (and before it is applied to the current in-memory results).
pub output_rewrites: Vec<Arc<FetchDataRewrite>>,
/// Similar to the other kinds of rewrites. This is a mechanism to convert a contextual path into
/// an argument to a resolver
pub context_rewrites: Vec<Arc<FetchDataRewrite>>,
}

#[derive(Debug, Clone)]
Expand Down
1 change: 1 addition & 0 deletions apollo-federation/src/query_plan/query_planner.rs
Original file line number Diff line number Diff line change
Expand Up @@ -342,6 +342,7 @@ impl QueryPlanner {
requires: Default::default(),
input_rewrites: Default::default(),
output_rewrites: Default::default(),
context_rewrites: Default::default(),
};

return Ok(QueryPlan::new(node, statistics));
Expand Down
4 changes: 2 additions & 2 deletions apollo-router/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -79,7 +79,7 @@ axum = { version = "0.6.20", features = ["headers", "json", "original-uri"] }
base64 = "0.21.7"
bloomfilter = "1.0.13"
buildstructor = "0.5.4"
bytes = "1.5.0"
bytes = "1.6.0"
clap = { version = "4.5.1", default-features = false, features = [
"env",
"derive",
Expand Down Expand Up @@ -185,7 +185,7 @@ regex = "1.10.3"
reqwest.workspace = true

# note: this dependency should _always_ be pinned, prefix the version with an `=`
router-bridge = "=0.5.21+v2.7.5"
router-bridge = { git = "https://github.com/apollographql/federation-rs", rev = "ced6eed5c4394bb3cbf3867798b8b7f63cb82d66" }

rust-embed = "8.2.0"
rustls = "0.21.11"
Expand Down
22 changes: 11 additions & 11 deletions apollo-router/build/main.rs
Original file line number Diff line number Diff line change
Expand Up @@ -10,23 +10,23 @@ fn main() -> Result<(), Box<dyn std::error::Error>> {
)
.expect("could not parse Cargo.toml");

let router_bridge = cargo_manifest
let _router_bridge = cargo_manifest
.get("dependencies")
.expect("Cargo.toml does not contain dependencies")
.as_object()
.expect("Cargo.toml dependencies key is not an object")
.get("router-bridge")
.expect("Cargo.toml dependencies does not have an entry for router-bridge");
let router_bridge_version = router_bridge
.as_str()
.or_else(|| {
router_bridge
.as_object()
.and_then(|o| o.get("version"))
.and_then(|version| version.as_str())
})
.expect("router-bridge does not have a version");

// let router_bridge_version = router_bridge
// .as_str()
// .or_else(|| {
// router_bridge
// .as_object()
// .and_then(|o| o.get("version"))
// .and_then(|version| version.as_str())
// })
// .expect("router-bridge does not have a version");
let router_bridge_version = "0.5.20+v2.8.0";
clenfest marked this conversation as resolved.
Show resolved Hide resolved
let mut it = router_bridge_version.split('+');
let _ = it.next();
let fed_version = it.next().expect("invalid router-bridge version format");
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ expression: query_plan
"id": null,
"inputRewrites": null,
"outputRewrites": null,
"contextRewrites": null,
"schemaAwareHash": "12dda6193654ae4fe6e38bc09d4f81cc73d0c9e098692096f72d2158eef4776f",
"authorization": {
"is_authenticated": false,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ expression: query_plan
"id": null,
"inputRewrites": null,
"outputRewrites": null,
"contextRewrites": null,
"schemaAwareHash": "00ad582ea45fc1bce436b36b21512f3d2c47b74fdbdc61e4b349289722c9ecf2",
"authorization": {
"is_authenticated": false,
Expand Down Expand Up @@ -61,6 +62,7 @@ expression: query_plan
"id": null,
"inputRewrites": null,
"outputRewrites": null,
"contextRewrites": null,
"schemaAwareHash": "a8ebdc2151a2e5207882e43c6906c0c64167fd9a8e0c7c4becc47736a5105096",
"authorization": {
"is_authenticated": false,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,7 @@ expression: "serde_json::to_value(response).unwrap()"
"id": null,
"inputRewrites": null,
"outputRewrites": null,
"contextRewrites": null,
"schemaAwareHash": "7245d488e97c3b2ac9f5fa4dd4660940b94ad81af070013305b2c0f76337b2f9",
"authorization": {
"is_authenticated": false,
Expand Down Expand Up @@ -107,6 +108,7 @@ expression: "serde_json::to_value(response).unwrap()"
"id": null,
"inputRewrites": null,
"outputRewrites": null,
"contextRewrites": null,
"schemaAwareHash": "6e0b4156706ea0cf924500cfdc99dd44b9f0ed07e2d3f888d4aff156e6a33238",
"authorization": {
"is_authenticated": false,
Expand Down Expand Up @@ -153,6 +155,7 @@ expression: "serde_json::to_value(response).unwrap()"
"id": null,
"inputRewrites": null,
"outputRewrites": null,
"contextRewrites": null,
"schemaAwareHash": "ff649f3d70241d5a8cd5f5d03ff4c41ecff72b0e4129a480207b05ac92318042",
"authorization": {
"is_authenticated": false,
Expand Down Expand Up @@ -196,6 +199,7 @@ expression: "serde_json::to_value(response).unwrap()"
"id": null,
"inputRewrites": null,
"outputRewrites": null,
"contextRewrites": null,
"schemaAwareHash": "bf9f3beda78a7a565e47c862157bad4ec871d724d752218da1168455dddca074",
"authorization": {
"is_authenticated": false,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -68,6 +68,7 @@ expression: "serde_json::to_value(response).unwrap()"
"id": null,
"inputRewrites": null,
"outputRewrites": null,
"contextRewrites": null,
"schemaAwareHash": "7245d488e97c3b2ac9f5fa4dd4660940b94ad81af070013305b2c0f76337b2f9",
"authorization": {
"is_authenticated": false,
Expand Down Expand Up @@ -107,6 +108,7 @@ expression: "serde_json::to_value(response).unwrap()"
"id": null,
"inputRewrites": null,
"outputRewrites": null,
"contextRewrites": null,
"schemaAwareHash": "6e0b4156706ea0cf924500cfdc99dd44b9f0ed07e2d3f888d4aff156e6a33238",
"authorization": {
"is_authenticated": false,
Expand Down Expand Up @@ -153,6 +155,7 @@ expression: "serde_json::to_value(response).unwrap()"
"id": null,
"inputRewrites": null,
"outputRewrites": null,
"contextRewrites": null,
"schemaAwareHash": "ff649f3d70241d5a8cd5f5d03ff4c41ecff72b0e4129a480207b05ac92318042",
"authorization": {
"is_authenticated": false,
Expand Down Expand Up @@ -196,6 +199,7 @@ expression: "serde_json::to_value(response).unwrap()"
"id": null,
"inputRewrites": null,
"outputRewrites": null,
"contextRewrites": null,
"schemaAwareHash": "bf9f3beda78a7a565e47c862157bad4ec871d724d752218da1168455dddca074",
"authorization": {
"is_authenticated": false,
Expand Down
3 changes: 3 additions & 0 deletions apollo-router/src/query_planner/convert.rs
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,7 @@ impl From<&'_ Box<next::FetchNode>> for plan::PlanNode {
operation_kind,
input_rewrites,
output_rewrites,
context_rewrites,
} = &**value;
Self::Fetch(super::fetch::FetchNode {
service_name: subgraph_name.clone(),
Expand All @@ -86,6 +87,7 @@ impl From<&'_ Box<next::FetchNode>> for plan::PlanNode {
id: id.map(|id| id.to_string().into()),
input_rewrites: option_vec(input_rewrites),
output_rewrites: option_vec(output_rewrites),
context_rewrites: option_vec(context_rewrites),
schema_aware_hash: Default::default(),
authorization: Default::default(),
})
Expand Down Expand Up @@ -153,6 +155,7 @@ impl From<&'_ next::FetchNode> for subscription::SubscriptionNode {
operation_kind,
input_rewrites,
output_rewrites,
context_rewrites: _,
} = value;
Self {
service_name: subgraph_name.clone(),
Expand Down
65 changes: 54 additions & 11 deletions apollo-router/src/query_planner/fetch.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ use std::collections::HashMap;
use std::fmt::Display;
use std::sync::Arc;

use apollo_compiler::ast;
use apollo_compiler::validation::Valid;
use apollo_compiler::ExecutableDocument;
use apollo_compiler::NodeStr;
Expand All @@ -17,6 +18,8 @@ use super::execution::ExecutionParameters;
use super::rewrites;
use super::selection::execute_selection_set;
use super::selection::Selection;
use super::subgraph_context::build_operation_with_aliasing;
use super::subgraph_context::ContextualArguments;
use crate::error::Error;
use crate::error::FetchError;
use crate::error::ValidationErrors;
Expand All @@ -34,6 +37,8 @@ use crate::services::SubgraphRequest;
use crate::spec::query::change::QueryHashVisitor;
use crate::spec::Schema;

use crate::query_planner::subgraph_context::SubgraphContext;

/// GraphQL operation type.
#[derive(Copy, Clone, Debug, Default, PartialEq, Eq, Hash, Deserialize, Serialize)]
#[serde(rename_all = "camelCase")]
Expand Down Expand Up @@ -70,22 +75,22 @@ impl OperationKind {
}
}

impl From<OperationKind> for apollo_compiler::ast::OperationType {
impl From<OperationKind> for ast::OperationType {
fn from(value: OperationKind) -> Self {
match value {
OperationKind::Query => apollo_compiler::ast::OperationType::Query,
OperationKind::Mutation => apollo_compiler::ast::OperationType::Mutation,
OperationKind::Subscription => apollo_compiler::ast::OperationType::Subscription,
OperationKind::Query => ast::OperationType::Query,
OperationKind::Mutation => ast::OperationType::Mutation,
OperationKind::Subscription => ast::OperationType::Subscription,
}
}
}

impl From<apollo_compiler::ast::OperationType> for OperationKind {
fn from(value: apollo_compiler::ast::OperationType) -> Self {
impl From<ast::OperationType> for OperationKind {
fn from(value: ast::OperationType) -> Self {
match value {
apollo_compiler::ast::OperationType::Query => OperationKind::Query,
apollo_compiler::ast::OperationType::Mutation => OperationKind::Mutation,
apollo_compiler::ast::OperationType::Subscription => OperationKind::Subscription,
ast::OperationType::Query => OperationKind::Query,
ast::OperationType::Mutation => OperationKind::Mutation,
ast::OperationType::Subscription => OperationKind::Subscription,
}
}
}
Expand Down Expand Up @@ -125,6 +130,9 @@ pub(crate) struct FetchNode {
// Optionally describes a number of "rewrites" to apply to the data that received from a fetch (and before it is applied to the current in-memory results).
pub(crate) output_rewrites: Option<Vec<rewrites::DataRewrite>>,

// Optionally describes a number of "rewrites" to apply to the data that has already been received further up the tree
pub(crate) context_rewrites: Option<Vec<rewrites::DataRewrite>>,

// hash for the query and relevant parts of the schema. if two different schemas provide the exact same types, fields and directives
// affecting the query, then they will have the same hash
#[serde(default)]
Expand Down Expand Up @@ -243,6 +251,7 @@ impl Display for QueryHash {
pub(crate) struct Variables {
pub(crate) variables: Object,
pub(crate) inverted_paths: Vec<Vec<Path>>,
pub(crate) contextual_arguments: Option<ContextualArguments>,
}

impl Variables {
Expand All @@ -256,8 +265,11 @@ impl Variables {
request: &Arc<http::Request<Request>>,
schema: &Schema,
input_rewrites: &Option<Vec<rewrites::DataRewrite>>,
context_rewrites: &Option<Vec<rewrites::DataRewrite>>,
) -> Option<Variables> {
let body = request.body();
let mut subgraph_context =
SubgraphContext::new(data, current_dir, schema, context_rewrites);
if !requires.is_empty() {
let mut variables = Object::with_capacity(1 + variable_usages.len());

Expand All @@ -269,8 +281,12 @@ impl Variables {

let mut inverted_paths: Vec<Vec<Path>> = Vec::new();
let mut values: IndexSet<Value> = IndexSet::new();

data.select_values_and_paths(schema, current_dir, |path, value| {
// first get contextual values that are required
if let Some(context) = subgraph_context.as_mut() {
context.execute_on_path(path);
}

let mut value = execute_selection_set(value, requires, schema, None);
if value.as_object().map(|o| !o.is_empty()).unwrap_or(false) {
rewrites::apply_rewrites(schema, &mut value, input_rewrites);
Expand All @@ -292,11 +308,16 @@ impl Variables {
}

let representations = Value::Array(Vec::from_iter(values));
let contextual_arguments = match subgraph_context.as_mut() {
Some(context) => context.add_variables_and_get_args(&mut variables),
None => None,
};

variables.insert("representations", representations);
Some(Variables {
variables,
inverted_paths,
contextual_arguments,
})
} else {
// with nested operations (Query or Mutation has an operation returning a Query or Mutation),
Expand All @@ -323,6 +344,7 @@ impl Variables {
})
.collect::<Object>(),
inverted_paths: Vec::new(),
contextual_arguments: None,
})
}
}
Expand Down Expand Up @@ -355,6 +377,7 @@ impl FetchNode {
let Variables {
variables,
inverted_paths: paths,
contextual_arguments,
} = match Variables::new(
&self.requires,
&self.variable_usages,
Expand All @@ -364,13 +387,33 @@ impl FetchNode {
parameters.supergraph_request,
parameters.schema,
&self.input_rewrites,
&self.context_rewrites,
) {
Some(variables) => variables,
None => {
return (Value::Object(Object::default()), Vec::new());
}
};

let alias_query_string; // this exists outside the if block to allow the as_str() to be longer lived
let aliased_operation = if let Some(ctx_arg) = contextual_arguments {
match build_operation_with_aliasing(operation, &ctx_arg, parameters.schema) {
Ok(op) => {
alias_query_string = op.serialize().no_indent().to_string();
alias_query_string.as_str()
}
Err(errors) => {
tracing::debug!(
"couldn't generate a valid executable document? {:?}",
errors
);
operation.as_serialized()
}
}
} else {
operation.as_serialized()
};

let mut subgraph_request = SubgraphRequest::builder()
.supergraph_request(parameters.supergraph_request.clone())
.subgraph_request(
Expand All @@ -389,7 +432,7 @@ impl FetchNode {
)
.body(
Request::builder()
.query(operation.as_serialized())
.query(aliased_operation)
.and_operation_name(operation_name.as_ref().map(|n| n.to_string()))
.variables(variables.clone())
.build(),
Expand Down
1 change: 1 addition & 0 deletions apollo-router/src/query_planner/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ mod labeler;
mod plan;
pub(crate) mod rewrites;
mod selection;
pub(crate) mod subgraph_context;
clenfest marked this conversation as resolved.
Show resolved Hide resolved
pub(crate) mod subscription;

pub(crate) const FETCH_SPAN_NAME: &str = "fetch";
Expand Down
Loading
Loading