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

Enable generate_query_fragments by default #6013

Merged
merged 29 commits into from
Nov 13, 2024
Merged
Show file tree
Hide file tree
Changes from 22 commits
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
be83e30
Enable `generate_query_fragments` by default
lrlna Sep 17, 2024
ee0975d
update config snapshot
lrlna Sep 17, 2024
cd3c249
Update docs/source/configuration/overview.mdx
lrlna Sep 17, 2024
1daae5d
use unwrap_or_else for generate_query_fragments defaults
lrlna Sep 17, 2024
51e89cb
update redis cache keys
lrlna Sep 17, 2024
6f43f20
new changelog
goto-bus-stop Oct 16, 2024
a6035eb
Merge branch 'dev' into lrlna/enable-generate-fragments
goto-bus-stop Oct 16, 2024
e807ff6
explain experimental/legacy state of fragment reuse in docs; do not s…
goto-bus-stop Oct 16, 2024
f1f45a7
keep links to the old header name working
goto-bus-stop Oct 16, 2024
0c40214
Update mocks: migrate some to fragment generation, disable fragment g…
goto-bus-stop Oct 16, 2024
544ce20
Merge branch 'dev' into lrlna/enable-generate-fragments
goto-bus-stop Oct 16, 2024
f5e1898
Update more mocks to use frag generation
goto-bus-stop Oct 23, 2024
b395dcc
Update at least one type conditioned fetching mock
goto-bus-stop Oct 23, 2024
592c45c
Fix redis tests
goto-bus-stop Oct 30, 2024
c51d105
Disable fragment generation for setContext integration tests
goto-bus-stop Oct 30, 2024
e638565
fix type condition test mocks
goto-bus-stop Oct 30, 2024
820977e
Update expectations in entity cache invalidation test
goto-bus-stop Oct 31, 2024
d2379fe
Merge branch 'dev' into lrlna/enable-generate-fragments
goto-bus-stop Nov 5, 2024
c734f9b
update mock for entity-cache + defer test
goto-bus-stop Nov 6, 2024
bc2ae12
update redis keys
goto-bus-stop Nov 6, 2024
eef215d
update core/defer mock
goto-bus-stop Nov 7, 2024
bf90de4
Merge branch 'dev' into lrlna/enable-generate-fragments
goto-bus-stop Nov 7, 2024
236fba6
Also swap the defaults inside apollo-federation
goto-bus-stop Nov 8, 2024
c3c86ac
Revert "Also swap the defaults inside apollo-federation"
goto-bus-stop Nov 8, 2024
2e235a8
Disable reuse fragments by default inside apollo-federation
goto-bus-stop Nov 8, 2024
e4190e1
linting once again
goto-bus-stop Nov 8, 2024
45fb226
Merge branch 'dev' into lrlna/enable-generate-fragments
goto-bus-stop Nov 8, 2024
c571242
update mock in benchmarks test
goto-bus-stop Nov 8, 2024
cb539c2
Merge branch 'dev' into lrlna/enable-generate-fragments
goto-bus-stop Nov 11, 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
Original file line number Diff line number Diff line change
@@ -0,0 +1,26 @@
### Compress subgraph operations by generating fragments

The router now compresses operations sent to subgraphs by default by generating fragment
definitions and using them in the operation.

Initially, the router is using a very simple transformation that is implemented in both
the JavaScript and Native query planners. We will improve the algorithm after the JavaScript
planner is no longer supported.

This replaces a previous experimental algorithm that was enabled by default.
`experimental_reuse_query_fragments` attempted to intelligently reuse the fragment definitions
from the original operation. Fragment generation is much faster, and in most cases produces
better outputs too.

If you are relying on the shape of fragments in your subgraph operations or tests, you can opt
out of the new algorithm with the configuration below. Note we strongly recommend against
relying on the shape of planned operations as new router features and optimizations may affect
it, and we intend to remove `experimental_reuse_query_fragments` in a future release.

```yaml
supergraph:
generate_query_fragments: false
experimental_reuse_query_fragments: true
```

By [@lrlna](https://github.com/lrlna) in https://github.com/apollographql/router/pull/6013
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ expression: parts
},
"errors": [
{
"message": "couldn't find mock for query {\"query\":\"query($representations: [_Any!]!) { _entities(representations: $representations) { ... on Product { reviews { __typename id product { __typename upc } } } } }\",\"variables\":{\"representations\":[{\"__typename\":\"Product\",\"upc\":\"1\"},{\"__typename\":\"Product\",\"upc\":\"2\"}]}}",
"message": "couldn't find mock for query {\"query\":\"query($representations: [_Any!]!) { _entities(representations: $representations) { ..._generated_onProduct1_0 } } fragment _generated_onProduct1_0 on Product { reviews { __typename id product { __typename upc } } }\",\"variables\":{\"representations\":[{\"__typename\":\"Product\",\"upc\":\"1\"},{\"__typename\":\"Product\",\"upc\":\"2\"}]}}",
"path": [
"topProducts",
"@"
Expand Down
2 changes: 1 addition & 1 deletion apollo-router/src/axum_factory/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1847,7 +1847,7 @@ async fn http_compressed_service() -> impl Service<
"apollo.include_subgraph_errors": {
"all": true
}
}
},
}))
.unwrap()
.supergraph_hook(move |service| {
Expand Down
10 changes: 7 additions & 3 deletions apollo-router/src/configuration/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -681,7 +681,7 @@ pub(crate) struct Supergraph {
pub(crate) reuse_query_fragments: Option<bool>,

/// Enable QP generation of fragments for subgraph requests
/// Default: false
/// Default: true
pub(crate) generate_query_fragments: bool,

/// Set to false to disable defer support
Expand All @@ -701,6 +701,10 @@ pub(crate) struct Supergraph {
pub(crate) experimental_log_on_broken_pipe: bool,
}

const fn default_generate_query_fragments() -> bool {
true
}

#[derive(Debug, Copy, Clone, PartialEq, Eq, Deserialize, Serialize, JsonSchema)]
#[serde(rename_all = "snake_case", untagged)]
pub(crate) enum AvailableParallelism {
Expand Down Expand Up @@ -753,7 +757,7 @@ impl Supergraph {
Some(false)
} else { reuse_query_fragments }
),
generate_query_fragments: generate_query_fragments.unwrap_or_default(),
generate_query_fragments: generate_query_fragments.unwrap_or_else(default_generate_query_fragments),
early_cancel: early_cancel.unwrap_or_default(),
experimental_log_on_broken_pipe: experimental_log_on_broken_pipe.unwrap_or_default(),
}
Expand Down Expand Up @@ -790,7 +794,7 @@ impl Supergraph {
Some(false)
} else { reuse_query_fragments }
),
generate_query_fragments: generate_query_fragments.unwrap_or_default(),
generate_query_fragments: generate_query_fragments.unwrap_or_else(default_generate_query_fragments),
early_cancel: early_cancel.unwrap_or_default(),
experimental_log_on_broken_pipe: experimental_log_on_broken_pipe.unwrap_or_default(),
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -6604,8 +6604,8 @@ expression: "&schema"
"type": "boolean"
},
"generate_query_fragments": {
"default": false,
"description": "Enable QP generation of fragments for subgraph requests Default: false",
"default": true,
"description": "Enable QP generation of fragments for subgraph requests Default: true",
"type": "boolean"
},
"introspection": {
Expand Down
2 changes: 1 addition & 1 deletion apollo-router/src/plugin/test/mock/canned.rs
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,7 @@ pub(crate) fn reviews_subgraph() -> MockSubgraph {
let review_mocks = vec![
(
json! {{
"query": "query TopProducts__reviews__1($representations:[_Any!]!){_entities(representations:$representations){...on Product{reviews{id product{__typename upc}author{__typename id}}}}}",
"query": "query TopProducts__reviews__1($representations:[_Any!]!){_entities(representations:$representations){..._generated_onProduct1_0}}fragment _generated_onProduct1_0 on Product{reviews{id product{__typename upc}author{__typename id}}}",
"operationName": "TopProducts__reviews__1",
"variables": {
"representations":[
Expand Down
2 changes: 1 addition & 1 deletion apollo-router/src/plugins/authorization/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ async fn authenticated_request() {
let subgraphs = MockedSubgraphs([
("user", MockSubgraph::builder().with_json(
serde_json::json!{{
"query": "query($representations:[_Any!]!){_entities(representations:$representations){...on User{name phone}}}",
"query": "query($representations:[_Any!]!){_entities(representations:$representations){..._generated_onUser2_0}}fragment _generated_onUser2_0 on User{name phone}",
"variables": {
"representations": [
{ "__typename": "User", "id":0 }
Expand Down
6 changes: 3 additions & 3 deletions apollo-router/src/plugins/cache/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -160,7 +160,7 @@ async fn insert() {
).with_header(CACHE_CONTROL, HeaderValue::from_static("public")).build()),
("orga", MockSubgraph::builder().with_json(
serde_json::json!{{
"query": "query($representations:[_Any!]!){_entities(representations:$representations){...on Organization{creatorUser{__typename id}}}}",
"query": "query($representations:[_Any!]!){_entities(representations:$representations){..._generated_onOrganization1_0}}fragment _generated_onOrganization1_0 on Organization{creatorUser{__typename id}}",
"variables": {
"representations": [
{
Expand Down Expand Up @@ -274,7 +274,7 @@ async fn no_cache_control() {
).build()),
("orga", MockSubgraph::builder().with_json(
serde_json::json!{{
"query": "query($representations:[_Any!]!){_entities(representations:$representations){...on Organization{creatorUser{__typename id}}}}",
"query": "query($representations:[_Any!]!){_entities(representations:$representations){..._generated_onOrganization1_0}}fragment _generated_onOrganization1_0 on Organization{creatorUser{__typename id}}",
"variables": {
"representations": [
{
Expand Down Expand Up @@ -365,7 +365,7 @@ async fn private() {
.build()),
("orga", MockSubgraph::builder().with_json(
serde_json::json!{{
"query": "query($representations:[_Any!]!){_entities(representations:$representations){...on Organization{creatorUser{__typename id}}}}",
"query": "query($representations:[_Any!]!){_entities(representations:$representations){..._generated_onOrganization1_0}}fragment _generated_onOrganization1_0 on Organization{creatorUser{__typename id}}",
"variables": {
"representations": [
{
Expand Down
12 changes: 12 additions & 0 deletions apollo-router/src/plugins/expose_query_plan.rs
Original file line number Diff line number Diff line change
Expand Up @@ -218,6 +218,10 @@ mod tests {
build_mock_supergraph(serde_json::json! {{
"plugins": {
"experimental.expose_query_plan": true
},
"supergraph": {
// TODO(@goto-bus-stop): need to update the mocks and remove this, #6013
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

should this be linking to a new issue (6013 is the existing PR#)?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The work would likely be done as part of removing reuse_query_fragments, I linked the PR as a place where you can find context

"generate_query_fragments": false,
}
}})
.await,
Expand All @@ -231,6 +235,10 @@ mod tests {
build_mock_supergraph(serde_json::json! {{
"plugins": {
"experimental.expose_query_plan": true
},
"supergraph": {
// TODO(@goto-bus-stop): need to update the mocks and remove this, #6013
"generate_query_fragments": false,
}
}})
.await,
Expand All @@ -245,6 +253,10 @@ mod tests {
let supergraph = build_mock_supergraph(serde_json::json! {{
"plugins": {
"experimental.expose_query_plan": false
},
"supergraph": {
// TODO(@goto-bus-stop): need to update the mocks and remove this, #6013
"generate_query_fragments": false,
}
}})
.await;
Expand Down
28 changes: 14 additions & 14 deletions apollo-router/src/plugins/include_subgraph_errors.rs
Original file line number Diff line number Diff line change
Expand Up @@ -191,13 +191,19 @@ mod test {

let product_service = MockSubgraph::new(product_mocks).with_extensions(extensions);

let mut configuration = Configuration::default();
// TODO(@goto-bus-stop): need to update the mocks and remove this, #6013
configuration.supergraph.generate_query_fragments = false;
let configuration = Arc::new(configuration);

let schema =
include_str!("../../../apollo-router-benchmarks/benches/fixtures/supergraph.graphql");
let schema = Schema::parse(schema, &Default::default()).unwrap();
let schema = Schema::parse(schema, &configuration).unwrap();

let planner = BridgeQueryPlannerPool::new(
Vec::new(),
schema.into(),
Default::default(),
Arc::clone(&configuration),
NonZeroUsize::new(1).unwrap(),
)
.await
Expand All @@ -207,15 +213,9 @@ mod test {

let builder = PluggableSupergraphServiceBuilder::new(planner);

let mut plugins = create_plugins(
&Configuration::default(),
&schema,
subgraph_schemas,
None,
None,
)
.await
.unwrap();
let mut plugins = create_plugins(&configuration, &schema, subgraph_schemas, None, None)
.await
.unwrap();

plugins.insert("apollo.include_subgraph_errors".to_string(), plugin);

Expand All @@ -228,10 +228,10 @@ mod test {
let supergraph_creator = builder.build().await.expect("should build");

RouterCreator::new(
QueryAnalysisLayer::new(supergraph_creator.schema(), Default::default()).await,
Arc::new(PersistedQueryLayer::new(&Default::default()).await.unwrap()),
QueryAnalysisLayer::new(supergraph_creator.schema(), Arc::clone(&configuration)).await,
Arc::new(PersistedQueryLayer::new(&configuration).await.unwrap()),
Arc::new(supergraph_creator),
Arc::new(Configuration::default()),
configuration,
)
.await
.unwrap()
Expand Down
3 changes: 3 additions & 0 deletions apollo-router/src/plugins/traffic_shaping/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -578,6 +578,9 @@ mod test {
r#"
traffic_shaping:
deduplicate_variables: true
supergraph:
# TODO(@goto-bus-stop): need to update the mocks and remove this, #6013
generate_query_fragments: false
"#,
)
.unwrap();
Expand Down
40 changes: 35 additions & 5 deletions apollo-router/src/query_planner/tests.rs
Original file line number Diff line number Diff line change
Expand Up @@ -821,7 +821,13 @@ async fn alias_renaming() {
].into_iter().collect());

let service = TestHarness::builder()
.configuration_json(serde_json::json!({"include_subgraph_errors": { "all": true } }))
.configuration_json(serde_json::json!({
"include_subgraph_errors": { "all": true },
"supergraph": {
// TODO(@goto-bus-stop): need to update the mocks and remove this, #6013
"generate_query_fragments": false,
}
}))
.unwrap()
.schema(schema)
.extra_plugin(subgraphs)
Expand Down Expand Up @@ -1264,7 +1270,13 @@ async fn missing_typename_and_fragments_in_requires2() {
].into_iter().collect());

let service = TestHarness::builder()
.configuration_json(serde_json::json!({"include_subgraph_errors": { "all": true } }))
.configuration_json(serde_json::json!({
"include_subgraph_errors": { "all": true },
"supergraph": {
// TODO(@goto-bus-stop): need to update the mocks and remove this, #6013
"generate_query_fragments": false,
}
}))
.unwrap()
.schema(schema)
.extra_plugin(subgraphs)
Expand Down Expand Up @@ -1551,7 +1563,13 @@ async fn typename_propagation() {
);

let service = TestHarness::builder()
.configuration_json(serde_json::json!({"include_subgraph_errors": { "all": true } }))
.configuration_json(serde_json::json!({
"include_subgraph_errors": { "all": true },
"supergraph": {
// TODO(@goto-bus-stop): need to update the mocks and remove this, #6013
"generate_query_fragments": false,
}
}))
.unwrap()
.schema(TYPENAME_PROPAGATION_SCHEMA)
.extra_plugin(subgraphs)
Expand Down Expand Up @@ -1648,7 +1666,13 @@ async fn typename_propagation2() {
);

let service = TestHarness::builder()
.configuration_json(serde_json::json!({"include_subgraph_errors": { "all": true } }))
.configuration_json(serde_json::json!({
"include_subgraph_errors": { "all": true },
"supergraph": {
// TODO(@goto-bus-stop): need to update the mocks and remove this, #6013
"generate_query_fragments": false,
}
}))
.unwrap()
.schema(TYPENAME_PROPAGATION_SCHEMA)
.extra_plugin(subgraphs)
Expand Down Expand Up @@ -1746,7 +1770,13 @@ async fn typename_propagation3() {
);

let service = TestHarness::builder()
.configuration_json(serde_json::json!({"include_subgraph_errors": { "all": true } }))
.configuration_json(serde_json::json!({
"include_subgraph_errors": { "all": true },
"supergraph": {
// TODO(@goto-bus-stop): need to update the mocks and remove this, #6013
"generate_query_fragments": false,
}
}))
.unwrap()
.schema(TYPENAME_PROPAGATION_SCHEMA)
.extra_plugin(subgraphs)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ expression: "(graphql_response, &subgraph_query_log)"
(
"reviews",
Some(
"query TopProducts__reviews__1($representations:[_Any!]!){_entities(representations:$representations){...on Product{reviewsForAuthor(authorID:\"\\\"1\\\"\"){body}}}}",
"query TopProducts__reviews__1($representations:[_Any!]!){_entities(representations:$representations){..._generated_onProduct1_0}}fragment _generated_onProduct1_0 on Product{reviewsForAuthor(authorID:\"\\\"1\\\"\"){body}}",
),
),
],
Expand Down
Loading
Loading