From f0f89edb73befe51964a88950e849c38444b2383 Mon Sep 17 00:00:00 2001 From: Duckki Oe Date: Tue, 16 Apr 2024 18:08:11 -0700 Subject: [PATCH] fixed a regression created by PR #2967 (#2982) - Changed the `Operation` class to inherit from `DirectiveTargetElement` class, which attaches directives properly. - Previously, directives applied on operations weren't properly attached to their parent elements. Related: https://github.com/apollographql/federation/issues/2961 --- gateway-js/src/resultShaping.ts | 4 +- internals-js/src/operations.ts | 23 ++++++----- .../src/__tests__/buildPlan.test.ts | 40 +++++++++++++++++++ query-planner-js/src/buildPlan.ts | 10 ++--- 4 files changed, 59 insertions(+), 18 deletions(-) diff --git a/gateway-js/src/resultShaping.ts b/gateway-js/src/resultShaping.ts index bb4697db5..f171a5c69 100644 --- a/gateway-js/src/resultShaping.ts +++ b/gateway-js/src/resultShaping.ts @@ -69,7 +69,7 @@ export function computeResponse({ } const parameters = { - schema: operation.schema, + schema: operation.schema(), variables: { ...operation.collectDefaultedVariableValues(), // overwrite any defaulted variables if they are provided @@ -87,7 +87,7 @@ export function computeResponse({ output: data, parameters, path: [], - parentType: operation.schema.schemaDefinition.rootType(operation.rootKind)!, + parentType: operation.schema().schemaDefinition.rootType(operation.rootKind)!, }); return { diff --git a/internals-js/src/operations.ts b/internals-js/src/operations.ts index dd2b2699f..26728c499 100644 --- a/internals-js/src/operations.ts +++ b/internals-js/src/operations.ts @@ -879,15 +879,16 @@ function computeFragmentsToKeep( return toExpand.size === 0 ? fragments : fragments.filter((f) => !toExpand.has(f.name)); } -export class Operation { +export class Operation extends DirectiveTargetElement { constructor( - readonly schema: Schema, + schema: Schema, readonly rootKind: SchemaRootKind, readonly selectionSet: SelectionSet, readonly variableDefinitions: VariableDefinitions, readonly fragments?: NamedFragments, readonly name?: string, - readonly directives?: readonly Directive[]) { + directives: readonly Directive[] = []) { + super(schema, directives); } // Returns a copy of this operation with the provided updated selection set. @@ -898,13 +899,13 @@ export class Operation { } return new Operation( - this.schema, + this.schema(), this.rootKind, newSelectionSet, this.variableDefinitions, this.fragments, this.name, - this.directives + this.appliedDirectives, ); } @@ -915,13 +916,13 @@ export class Operation { } return new Operation( - this.schema, + this.schema(), this.rootKind, newSelectionSet, this.variableDefinitions, newFragments, this.name, - this.directives + this.appliedDirectives, ); } @@ -980,13 +981,13 @@ export class Operation { generateQueryFragments(): Operation { const [minimizedSelectionSet, fragments] = this.selectionSet.minimizeSelectionSet(); return new Operation( - this.schema, + this.schema(), this.rootKind, minimizedSelectionSet, this.variableDefinitions, fragments, this.name, - this.directives + this.appliedDirectives, ); } @@ -1058,7 +1059,7 @@ export class Operation { } toString(expandFragments: boolean = false, prettyPrint: boolean = true): string { - return this.selectionSet.toOperationString(this.rootKind, this.variableDefinitions, this.fragments, this.name, this.directives, expandFragments, prettyPrint); + return this.selectionSet.toOperationString(this.rootKind, this.variableDefinitions, this.fragments, this.name, this.appliedDirectives, expandFragments, prettyPrint); } } @@ -3968,7 +3969,7 @@ export function operationToDocument(operation: Operation): DocumentNode { name: operation.name ? { kind: Kind.NAME, value: operation.name } : undefined, selectionSet: operation.selectionSet.toSelectionSetNode(), variableDefinitions: operation.variableDefinitions.toVariableDefinitionNodes(), - directives: directivesToDirectiveNodes(operation.directives), + directives: directivesToDirectiveNodes(operation.appliedDirectives), }; const fragmentASTs: DefinitionNode[] = operation.fragments ? operation.fragments?.toFragmentDefinitionNodes() diff --git a/query-planner-js/src/__tests__/buildPlan.test.ts b/query-planner-js/src/__tests__/buildPlan.test.ts index 9ec81d23b..932e827d4 100644 --- a/query-planner-js/src/__tests__/buildPlan.test.ts +++ b/query-planner-js/src/__tests__/buildPlan.test.ts @@ -8679,4 +8679,44 @@ describe('handles operations with directives', () => { } `); }); // end of `test` + + test('if directives with arguments applied on queries are ok', () => { + const subgraph1 = { + name: 'Subgraph1', + typeDefs: gql` + directive @noArgs on QUERY + directive @withArgs(arg1: String) on QUERY + + type Query { + test: String! + } + `, + }; + + const subgraph2 = { + name: 'Subgraph2', + typeDefs: gql` + directive @noArgs on QUERY + directive @withArgs(arg1: String) on QUERY + `, + }; + + const query = gql` + query @noArgs @withArgs(arg1: "hi") { + test + } + `; + + const [api, qp] = composeAndCreatePlanner(subgraph1, subgraph2); + const op = operationFromDocument(api, query); + const queryPlan = qp.buildQueryPlan(op); + const fetch_nodes = findFetchNodes(subgraph1.name, queryPlan.node); + expect(fetch_nodes).toHaveLength(1); + // Note: The query is expected to carry the `@noArgs` and `@withArgs` directive. + expect(parse(fetch_nodes[0].operation)).toMatchInlineSnapshot(` + query @noArgs @withArgs(arg1: "hi") { + test + } + `); + }); }); // end of `describe` diff --git a/query-planner-js/src/buildPlan.ts b/query-planner-js/src/buildPlan.ts index bf1fe75b4..cdf337135 100644 --- a/query-planner-js/src/buildPlan.ts +++ b/query-planner-js/src/buildPlan.ts @@ -3174,7 +3174,7 @@ export class QueryPlanner { variableDefinitions: operation.variableDefinitions, fragments: fragments ? new RebasedFragments(fragments) : undefined, operationName: operation.name, - directives: operation.directives, + directives: operation.appliedDirectives, assignedDeferLabels, }); @@ -3380,13 +3380,13 @@ export class QueryPlanner { return operation; } return new Operation( - operation.schema, + operation.schema(), operation.rootKind, updatedSelectionSet, operation.variableDefinitions, operation.fragments, operation.name, - operation.directives, + operation.appliedDirectives, ); } @@ -3535,13 +3535,13 @@ function withoutIntrospection(operation: Operation): Operation { } return new Operation( - operation.schema, + operation.schema(), operation.rootKind, operation.selectionSet.lazyMap((s) => isIntrospectionSelection(s) ? undefined : s), operation.variableDefinitions, operation.fragments, operation.name, - operation.directives, + operation.appliedDirectives, ); }