From 2b50a8b5bb62276752308f339c47d883e7e1f211 Mon Sep 17 00:00:00 2001 From: Yaacov Rydzinski Date: Sat, 14 Jan 2023 23:26:46 +0200 Subject: [PATCH 1/7] introduce GroupedFieldSet and FieldGroup types --- src/execution/collectFields.ts | 12 +- src/execution/execute.ts | 194 ++++++++++-------- .../rules/SingleFieldSubscriptionsRule.ts | 8 +- 3 files changed, 115 insertions(+), 99 deletions(-) diff --git a/src/execution/collectFields.ts b/src/execution/collectFields.ts index 17468b791f..faa273ee32 100644 --- a/src/execution/collectFields.ts +++ b/src/execution/collectFields.ts @@ -26,13 +26,17 @@ import { typeFromAST } from '../utilities/typeFromAST.js'; import { getDirectiveValues } from './values.js'; +export type FieldGroup = ReadonlyArray; + +export type GroupedFieldSet = Map; + export interface PatchFields { label: string | undefined; - fields: Map>; + fields: GroupedFieldSet; } export interface FieldsAndPatches { - fields: Map>; + fields: GroupedFieldSet; patches: Array; } @@ -85,7 +89,7 @@ export function collectSubfields( variableValues: { [variable: string]: unknown }, operation: OperationDefinitionNode, returnType: GraphQLObjectType, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, ): FieldsAndPatches { const subFieldNodes = new AccumulatorMap(); const visitedFragmentNames = new Set(); @@ -96,7 +100,7 @@ export function collectSubfields( patches: subPatches, }; - for (const node of fieldNodes) { + for (const node of fieldGroup) { if (node.selectionSet) { collectFieldsImpl( schema, diff --git a/src/execution/execute.ts b/src/execution/execute.ts index 1bc6c4267b..2485ead567 100644 --- a/src/execution/execute.ts +++ b/src/execution/execute.ts @@ -19,7 +19,6 @@ import { locatedError } from '../error/locatedError.js'; import type { DocumentNode, - FieldNode, FragmentDefinitionNode, OperationDefinitionNode, } from '../language/ast.js'; @@ -48,6 +47,7 @@ import { GraphQLStreamDirective } from '../type/directives.js'; import type { GraphQLSchema } from '../type/schema.js'; import { assertValidSchema } from '../type/validate.js'; +import type { FieldGroup, GroupedFieldSet } from './collectFields.js'; import { collectFields, collectSubfields as _collectSubfields, @@ -72,7 +72,7 @@ const collectSubfields = memoize3( ( exeContext: ExecutionContext, returnType: GraphQLObjectType, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, ) => _collectSubfields( exeContext.schema, @@ -80,7 +80,7 @@ const collectSubfields = memoize3( exeContext.variableValues, exeContext.operation, returnType, - fieldNodes, + fieldGroup, ), ); @@ -536,7 +536,7 @@ function executeOperation( ); } - const { fields: rootFields, patches } = collectFields( + const { fields: groupedFieldSet, patches } = collectFields( schema, fragments, variableValues, @@ -548,7 +548,13 @@ function executeOperation( switch (operation.operation) { case OperationTypeNode.QUERY: - result = executeFields(exeContext, rootType, rootValue, path, rootFields); + result = executeFields( + exeContext, + rootType, + rootValue, + path, + groupedFieldSet, + ); break; case OperationTypeNode.MUTATION: result = executeFieldsSerially( @@ -556,13 +562,19 @@ function executeOperation( rootType, rootValue, path, - rootFields, + groupedFieldSet, ); break; case OperationTypeNode.SUBSCRIPTION: // TODO: deprecate `subscribe` and move all logic here // Temporary solution until we finish merging execute and subscribe together - result = executeFields(exeContext, rootType, rootValue, path, rootFields); + result = executeFields( + exeContext, + rootType, + rootValue, + path, + groupedFieldSet, + ); } for (const patch of patches) { @@ -589,17 +601,17 @@ function executeFieldsSerially( parentType: GraphQLObjectType, sourceValue: unknown, path: Path | undefined, - fields: Map>, + groupedFieldSet: GroupedFieldSet, ): PromiseOrValue> { return promiseReduce( - fields, - (results, [responseName, fieldNodes]) => { + groupedFieldSet, + (results, [responseName, fieldGroup]) => { const fieldPath = addPath(path, responseName, parentType.name); const result = executeField( exeContext, parentType, sourceValue, - fieldNodes, + fieldGroup, fieldPath, ); if (result === undefined) { @@ -627,20 +639,20 @@ function executeFields( parentType: GraphQLObjectType, sourceValue: unknown, path: Path | undefined, - fields: Map>, + groupedFieldSet: GroupedFieldSet, asyncPayloadRecord?: AsyncPayloadRecord, ): PromiseOrValue> { const results = Object.create(null); let containsPromise = false; try { - for (const [responseName, fieldNodes] of fields) { + for (const [responseName, fieldGroup] of groupedFieldSet) { const fieldPath = addPath(path, responseName, parentType.name); const result = executeField( exeContext, parentType, sourceValue, - fieldNodes, + fieldGroup, fieldPath, asyncPayloadRecord, ); @@ -683,12 +695,12 @@ function executeField( exeContext: ExecutionContext, parentType: GraphQLObjectType, source: unknown, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, path: Path, asyncPayloadRecord?: AsyncPayloadRecord, ): PromiseOrValue { const errors = asyncPayloadRecord?.errors ?? exeContext.errors; - const fieldName = fieldNodes[0].name.value; + const fieldName = fieldGroup[0].name.value; const fieldDef = exeContext.schema.getField(parentType, fieldName); if (!fieldDef) { return; @@ -700,7 +712,7 @@ function executeField( const info = buildResolveInfo( exeContext, fieldDef, - fieldNodes, + fieldGroup, parentType, path, ); @@ -712,7 +724,7 @@ function executeField( // TODO: find a way to memoize, in case this field is within a List type. const args = getArgumentValues( fieldDef, - fieldNodes[0], + fieldGroup[0], exeContext.variableValues, ); @@ -727,7 +739,7 @@ function executeField( return completePromisedValue( exeContext, returnType, - fieldNodes, + fieldGroup, info, path, result, @@ -738,7 +750,7 @@ function executeField( const completed = completeValue( exeContext, returnType, - fieldNodes, + fieldGroup, info, path, result, @@ -749,7 +761,7 @@ function executeField( // Note: we don't rely on a `catch` method, but we do expect "thenable" // to take a second callback for the error case. return completed.then(undefined, (rawError) => { - const error = locatedError(rawError, fieldNodes, pathToArray(path)); + const error = locatedError(rawError, fieldGroup, pathToArray(path)); const handledError = handleFieldError(error, returnType, errors); filterSubsequentPayloads(exeContext, path, asyncPayloadRecord); return handledError; @@ -757,7 +769,7 @@ function executeField( } return completed; } catch (rawError) { - const error = locatedError(rawError, fieldNodes, pathToArray(path)); + const error = locatedError(rawError, fieldGroup, pathToArray(path)); const handledError = handleFieldError(error, returnType, errors); filterSubsequentPayloads(exeContext, path, asyncPayloadRecord); return handledError; @@ -771,7 +783,7 @@ function executeField( export function buildResolveInfo( exeContext: ExecutionContext, fieldDef: GraphQLField, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, parentType: GraphQLObjectType, path: Path, ): GraphQLResolveInfo { @@ -779,7 +791,7 @@ export function buildResolveInfo( // information about the current execution state. return { fieldName: fieldDef.name, - fieldNodes, + fieldNodes: fieldGroup, returnType: fieldDef.type, parentType, path, @@ -832,7 +844,7 @@ function handleFieldError( function completeValue( exeContext: ExecutionContext, returnType: GraphQLOutputType, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, info: GraphQLResolveInfo, path: Path, result: unknown, @@ -849,7 +861,7 @@ function completeValue( const completed = completeValue( exeContext, returnType.ofType, - fieldNodes, + fieldGroup, info, path, result, @@ -873,7 +885,7 @@ function completeValue( return completeListValue( exeContext, returnType, - fieldNodes, + fieldGroup, info, path, result, @@ -893,7 +905,7 @@ function completeValue( return completeAbstractValue( exeContext, returnType, - fieldNodes, + fieldGroup, info, path, result, @@ -906,7 +918,7 @@ function completeValue( return completeObjectValue( exeContext, returnType, - fieldNodes, + fieldGroup, info, path, result, @@ -924,7 +936,7 @@ function completeValue( async function completePromisedValue( exeContext: ExecutionContext, returnType: GraphQLOutputType, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, info: GraphQLResolveInfo, path: Path, result: Promise, @@ -935,7 +947,7 @@ async function completePromisedValue( let completed = completeValue( exeContext, returnType, - fieldNodes, + fieldGroup, info, path, resolved, @@ -947,7 +959,7 @@ async function completePromisedValue( return completed; } catch (rawError) { const errors = asyncPayloadRecord?.errors ?? exeContext.errors; - const error = locatedError(rawError, fieldNodes, pathToArray(path)); + const error = locatedError(rawError, fieldGroup, pathToArray(path)); const handledError = handleFieldError(error, returnType, errors); filterSubsequentPayloads(exeContext, path, asyncPayloadRecord); return handledError; @@ -961,7 +973,7 @@ async function completePromisedValue( */ function getStreamValues( exeContext: ExecutionContext, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, path: Path, ): | undefined @@ -978,7 +990,7 @@ function getStreamValues( // safe to only check the first fieldNode for the stream directive const stream = getDirectiveValues( GraphQLStreamDirective, - fieldNodes[0], + fieldGroup[0], exeContext.variableValues, ); @@ -1018,14 +1030,14 @@ function getStreamValues( async function completeAsyncIteratorValue( exeContext: ExecutionContext, itemType: GraphQLOutputType, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, info: GraphQLResolveInfo, path: Path, iterator: AsyncIterator, asyncPayloadRecord?: AsyncPayloadRecord, ): Promise> { const errors = asyncPayloadRecord?.errors ?? exeContext.errors; - const stream = getStreamValues(exeContext, fieldNodes, path); + const stream = getStreamValues(exeContext, fieldGroup, path); let containsPromise = false; const completedResults: Array = []; let index = 0; @@ -1041,7 +1053,7 @@ async function completeAsyncIteratorValue( index, iterator, exeContext, - fieldNodes, + fieldGroup, info, itemType, path, @@ -1060,7 +1072,7 @@ async function completeAsyncIteratorValue( break; } } catch (rawError) { - const error = locatedError(rawError, fieldNodes, pathToArray(itemPath)); + const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); completedResults.push(handleFieldError(error, itemType, errors)); break; } @@ -1072,7 +1084,7 @@ async function completeAsyncIteratorValue( errors, exeContext, itemType, - fieldNodes, + fieldGroup, info, itemPath, asyncPayloadRecord, @@ -1092,7 +1104,7 @@ async function completeAsyncIteratorValue( function completeListValue( exeContext: ExecutionContext, returnType: GraphQLList, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, info: GraphQLResolveInfo, path: Path, result: unknown, @@ -1107,7 +1119,7 @@ function completeListValue( return completeAsyncIteratorValue( exeContext, itemType, - fieldNodes, + fieldGroup, info, path, iterator, @@ -1121,7 +1133,7 @@ function completeListValue( ); } - const stream = getStreamValues(exeContext, fieldNodes, path); + const stream = getStreamValues(exeContext, fieldGroup, path); // This is specified as a simple map, however we're optimizing the path // where the list contains no Promises by avoiding creating another Promise. @@ -1144,7 +1156,7 @@ function completeListValue( itemPath, item, exeContext, - fieldNodes, + fieldGroup, info, itemType, stream.label, @@ -1161,7 +1173,7 @@ function completeListValue( errors, exeContext, itemType, - fieldNodes, + fieldGroup, info, itemPath, asyncPayloadRecord, @@ -1187,7 +1199,7 @@ function completeListItemValue( errors: Array, exeContext: ExecutionContext, itemType: GraphQLOutputType, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, info: GraphQLResolveInfo, itemPath: Path, asyncPayloadRecord?: AsyncPayloadRecord, @@ -1197,7 +1209,7 @@ function completeListItemValue( completePromisedValue( exeContext, itemType, - fieldNodes, + fieldGroup, info, itemPath, item, @@ -1212,7 +1224,7 @@ function completeListItemValue( const completedItem = completeValue( exeContext, itemType, - fieldNodes, + fieldGroup, info, itemPath, item, @@ -1226,7 +1238,7 @@ function completeListItemValue( completedItem.then(undefined, (rawError) => { const error = locatedError( rawError, - fieldNodes, + fieldGroup, pathToArray(itemPath), ); const handledError = handleFieldError(error, itemType, errors); @@ -1240,7 +1252,7 @@ function completeListItemValue( completedResults.push(completedItem); } catch (rawError) { - const error = locatedError(rawError, fieldNodes, pathToArray(itemPath)); + const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); const handledError = handleFieldError(error, itemType, errors); filterSubsequentPayloads(exeContext, itemPath, asyncPayloadRecord); completedResults.push(handledError); @@ -1274,7 +1286,7 @@ function completeLeafValue( function completeAbstractValue( exeContext: ExecutionContext, returnType: GraphQLAbstractType, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, info: GraphQLResolveInfo, path: Path, result: unknown, @@ -1292,11 +1304,11 @@ function completeAbstractValue( resolvedRuntimeType, exeContext, returnType, - fieldNodes, + fieldGroup, info, result, ), - fieldNodes, + fieldGroup, info, path, result, @@ -1311,11 +1323,11 @@ function completeAbstractValue( runtimeType, exeContext, returnType, - fieldNodes, + fieldGroup, info, result, ), - fieldNodes, + fieldGroup, info, path, result, @@ -1327,14 +1339,14 @@ function ensureValidRuntimeType( runtimeTypeName: unknown, exeContext: ExecutionContext, returnType: GraphQLAbstractType, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, info: GraphQLResolveInfo, result: unknown, ): GraphQLObjectType { if (runtimeTypeName == null) { throw new GraphQLError( `Abstract type "${returnType.name}" must resolve to an Object type at runtime for field "${info.parentType.name}.${info.fieldName}". Either the "${returnType.name}" type should provide a "resolveType" function or each possible type should provide an "isTypeOf" function.`, - { nodes: fieldNodes }, + { nodes: fieldGroup }, ); } @@ -1357,21 +1369,21 @@ function ensureValidRuntimeType( if (runtimeType == null) { throw new GraphQLError( `Abstract type "${returnType.name}" was resolved to a type "${runtimeTypeName}" that does not exist inside the schema.`, - { nodes: fieldNodes }, + { nodes: fieldGroup }, ); } if (!isObjectType(runtimeType)) { throw new GraphQLError( `Abstract type "${returnType.name}" was resolved to a non-object type "${runtimeTypeName}".`, - { nodes: fieldNodes }, + { nodes: fieldGroup }, ); } if (!exeContext.schema.isSubType(returnType, runtimeType)) { throw new GraphQLError( `Runtime Object type "${runtimeType.name}" is not a possible type for "${returnType.name}".`, - { nodes: fieldNodes }, + { nodes: fieldGroup }, ); } @@ -1384,7 +1396,7 @@ function ensureValidRuntimeType( function completeObjectValue( exeContext: ExecutionContext, returnType: GraphQLObjectType, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, info: GraphQLResolveInfo, path: Path, result: unknown, @@ -1399,12 +1411,12 @@ function completeObjectValue( if (isPromise(isTypeOf)) { return isTypeOf.then((resolvedIsTypeOf) => { if (!resolvedIsTypeOf) { - throw invalidReturnTypeError(returnType, result, fieldNodes); + throw invalidReturnTypeError(returnType, result, fieldGroup); } return collectAndExecuteSubfields( exeContext, returnType, - fieldNodes, + fieldGroup, path, result, asyncPayloadRecord, @@ -1413,14 +1425,14 @@ function completeObjectValue( } if (!isTypeOf) { - throw invalidReturnTypeError(returnType, result, fieldNodes); + throw invalidReturnTypeError(returnType, result, fieldGroup); } } return collectAndExecuteSubfields( exeContext, returnType, - fieldNodes, + fieldGroup, path, result, asyncPayloadRecord, @@ -1430,18 +1442,18 @@ function completeObjectValue( function invalidReturnTypeError( returnType: GraphQLObjectType, result: unknown, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, ): GraphQLError { return new GraphQLError( `Expected value of type "${returnType.name}" but got: ${inspect(result)}.`, - { nodes: fieldNodes }, + { nodes: fieldGroup }, ); } function collectAndExecuteSubfields( exeContext: ExecutionContext, returnType: GraphQLObjectType, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, path: Path, result: unknown, asyncPayloadRecord?: AsyncPayloadRecord, @@ -1450,7 +1462,7 @@ function collectAndExecuteSubfields( const { fields: subFieldNodes, patches: subPatches } = collectSubfields( exeContext, returnType, - fieldNodes, + fieldGroup, ); const subFields = executeFields( @@ -1691,7 +1703,7 @@ function executeSubscription( ); } - const { fields: rootFields } = collectFields( + const { fields: groupedFieldSet } = collectFields( schema, fragments, variableValues, @@ -1699,15 +1711,15 @@ function executeSubscription( operation, ); - const firstRootField = rootFields.entries().next().value; - const [responseName, fieldNodes] = firstRootField; - const fieldName = fieldNodes[0].name.value; + const firstRootField = groupedFieldSet.entries().next().value; + const [responseName, fieldGroup] = firstRootField; + const fieldName = fieldGroup[0].name.value; const fieldDef = schema.getField(rootType, fieldName); if (!fieldDef) { throw new GraphQLError( `The subscription field "${fieldName}" is not defined.`, - { nodes: fieldNodes }, + { nodes: fieldGroup }, ); } @@ -1715,7 +1727,7 @@ function executeSubscription( const info = buildResolveInfo( exeContext, fieldDef, - fieldNodes, + fieldGroup, rootType, path, ); @@ -1726,7 +1738,7 @@ function executeSubscription( // Build a JS object of arguments from the field.arguments AST, using the // variables scope to fulfill any variable references. - const args = getArgumentValues(fieldDef, fieldNodes[0], variableValues); + const args = getArgumentValues(fieldDef, fieldGroup[0], variableValues); // The resolve function's optional third argument is a context value that // is provided to every resolve function within an execution. It is commonly @@ -1740,13 +1752,13 @@ function executeSubscription( if (isPromise(result)) { return result.then(assertEventStream).then(undefined, (error) => { - throw locatedError(error, fieldNodes, pathToArray(path)); + throw locatedError(error, fieldGroup, pathToArray(path)); }); } return assertEventStream(result); } catch (error) { - throw locatedError(error, fieldNodes, pathToArray(path)); + throw locatedError(error, fieldGroup, pathToArray(path)); } } @@ -1770,7 +1782,7 @@ function executeDeferredFragment( exeContext: ExecutionContext, parentType: GraphQLObjectType, sourceValue: unknown, - fields: Map>, + groupedFieldSet: GroupedFieldSet, label?: string, path?: Path, parentContext?: AsyncPayloadRecord, @@ -1788,7 +1800,7 @@ function executeDeferredFragment( parentType, sourceValue, path, - fields, + groupedFieldSet, asyncPayloadRecord, ); @@ -1810,7 +1822,7 @@ function executeStreamField( itemPath: Path, item: PromiseOrValue, exeContext: ExecutionContext, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, info: GraphQLResolveInfo, itemType: GraphQLOutputType, label?: string, @@ -1826,7 +1838,7 @@ function executeStreamField( const completedItems = completePromisedValue( exeContext, itemType, - fieldNodes, + fieldGroup, info, itemPath, item, @@ -1850,14 +1862,14 @@ function executeStreamField( completedItem = completeValue( exeContext, itemType, - fieldNodes, + fieldGroup, info, itemPath, item, asyncPayloadRecord, ); } catch (rawError) { - const error = locatedError(rawError, fieldNodes, pathToArray(itemPath)); + const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); completedItem = handleFieldError( error, itemType, @@ -1875,7 +1887,7 @@ function executeStreamField( if (isPromise(completedItem)) { const completedItems = completedItem .then(undefined, (rawError) => { - const error = locatedError(rawError, fieldNodes, pathToArray(itemPath)); + const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); const handledError = handleFieldError( error, itemType, @@ -1904,7 +1916,7 @@ function executeStreamField( async function executeStreamIteratorItem( iterator: AsyncIterator, exeContext: ExecutionContext, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, info: GraphQLResolveInfo, itemType: GraphQLOutputType, asyncPayloadRecord: StreamRecord, @@ -1919,7 +1931,7 @@ async function executeStreamIteratorItem( } item = value; } catch (rawError) { - const error = locatedError(rawError, fieldNodes, pathToArray(itemPath)); + const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); const value = handleFieldError(error, itemType, asyncPayloadRecord.errors); // don't continue if iterator throws return { done: true, value }; @@ -1929,7 +1941,7 @@ async function executeStreamIteratorItem( completedItem = completeValue( exeContext, itemType, - fieldNodes, + fieldGroup, info, itemPath, item, @@ -1938,7 +1950,7 @@ async function executeStreamIteratorItem( if (isPromise(completedItem)) { completedItem = completedItem.then(undefined, (rawError) => { - const error = locatedError(rawError, fieldNodes, pathToArray(itemPath)); + const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); const handledError = handleFieldError( error, itemType, @@ -1950,7 +1962,7 @@ async function executeStreamIteratorItem( } return { done: false, value: completedItem }; } catch (rawError) { - const error = locatedError(rawError, fieldNodes, pathToArray(itemPath)); + const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); const value = handleFieldError(error, itemType, asyncPayloadRecord.errors); filterSubsequentPayloads(exeContext, itemPath, asyncPayloadRecord); return { done: false, value }; @@ -1961,7 +1973,7 @@ async function executeStreamIterator( initialIndex: number, iterator: AsyncIterator, exeContext: ExecutionContext, - fieldNodes: ReadonlyArray, + fieldGroup: FieldGroup, info: GraphQLResolveInfo, itemType: GraphQLOutputType, path: Path, @@ -1987,7 +1999,7 @@ async function executeStreamIterator( iteration = await executeStreamIteratorItem( iterator, exeContext, - fieldNodes, + fieldGroup, info, itemType, asyncPayloadRecord, diff --git a/src/validation/rules/SingleFieldSubscriptionsRule.ts b/src/validation/rules/SingleFieldSubscriptionsRule.ts index 4a3d834124..bc68586e01 100644 --- a/src/validation/rules/SingleFieldSubscriptionsRule.ts +++ b/src/validation/rules/SingleFieldSubscriptionsRule.ts @@ -49,15 +49,15 @@ export function SingleFieldSubscriptionsRule( node, ); if (fields.size > 1) { - const fieldSelectionLists = [...fields.values()]; - const extraFieldSelectionLists = fieldSelectionLists.slice(1); - const extraFieldSelections = extraFieldSelectionLists.flat(); + const fieldGroups = [...fields.values()]; + const extraFieldGroups = fieldGroups.slice(1); + const extraFields = extraFieldGroups.flat(); context.reportError( new GraphQLError( operationName != null ? `Subscription "${operationName}" must select only one top level field.` : 'Anonymous Subscription must select only one top level field.', - { nodes: extraFieldSelections }, + { nodes: extraFields }, ), ); } From 5a02b0858ea6d28b6613a43c4b3a575321f42bf6 Mon Sep 17 00:00:00 2001 From: Yaacov Rydzinski Date: Sun, 8 Jan 2023 15:44:55 +0200 Subject: [PATCH 2/7] remove labels --- src/execution/__tests__/defer-test.ts | 24 +-- src/execution/__tests__/mutations-test.ts | 6 +- src/execution/__tests__/stream-test.ts | 20 +- src/execution/__tests__/sync-test.ts | 2 +- src/execution/collectFields.ts | 19 +- src/execution/execute.ts | 37 +--- src/type/directives.ts | 8 - .../DeferStreamDirectiveLabelRule-test.ts | 171 ------------------ .../OverlappingFieldsCanBeMergedRule-test.ts | 30 +-- src/validation/index.ts | 3 - .../rules/DeferStreamDirectiveLabelRule.ts | 55 ------ src/validation/specifiedRules.ts | 3 - 12 files changed, 32 insertions(+), 346 deletions(-) delete mode 100644 src/validation/__tests__/DeferStreamDirectiveLabelRule-test.ts delete mode 100644 src/validation/rules/DeferStreamDirectiveLabelRule.ts diff --git a/src/execution/__tests__/defer-test.ts b/src/execution/__tests__/defer-test.ts index 5cad95bbc3..712ef87579 100644 --- a/src/execution/__tests__/defer-test.ts +++ b/src/execution/__tests__/defer-test.ts @@ -206,7 +206,7 @@ describe('Execute: defer directive', () => { it('Can defer fragments on the top level Query field', async () => { const document = parse(` query HeroNameQuery { - ...QueryFragment @defer(label: "DeferQuery") + ...QueryFragment @defer } fragment QueryFragment on Query { hero { @@ -230,7 +230,6 @@ describe('Execute: defer directive', () => { }, }, path: [], - label: 'DeferQuery', }, ], hasNext: false, @@ -240,7 +239,7 @@ describe('Execute: defer directive', () => { it('Can defer fragments with errors on the top level Query field', async () => { const document = parse(` query HeroNameQuery { - ...QueryFragment @defer(label: "DeferQuery") + ...QueryFragment @defer } fragment QueryFragment on Query { hero { @@ -271,7 +270,6 @@ describe('Execute: defer directive', () => { }, ], path: [], - label: 'DeferQuery', }, ], hasNext: false, @@ -283,12 +281,12 @@ describe('Execute: defer directive', () => { query HeroNameQuery { hero { id - ...TopFragment @defer(label: "DeferTop") + ...TopFragment @defer } } fragment TopFragment on Hero { name - ...NestedFragment @defer(label: "DeferNested") + ...NestedFragment @defer } fragment NestedFragment on Hero { friends { @@ -314,14 +312,12 @@ describe('Execute: defer directive', () => { friends: [{ name: 'Han' }, { name: 'Leia' }, { name: 'C-3PO' }], }, path: ['hero'], - label: 'DeferNested', }, { data: { name: 'Luke', }, path: ['hero'], - label: 'DeferTop', }, ], hasNext: false, @@ -333,7 +329,7 @@ describe('Execute: defer directive', () => { query HeroNameQuery { hero { id - ...TopFragment @defer(label: "DeferTop") + ...TopFragment @defer ...TopFragment } } @@ -359,7 +355,6 @@ describe('Execute: defer directive', () => { name: 'Luke', }, path: ['hero'], - label: 'DeferTop', }, ], hasNext: false, @@ -372,7 +367,7 @@ describe('Execute: defer directive', () => { hero { id ...TopFragment - ...TopFragment @defer(label: "DeferTop") + ...TopFragment @defer } } fragment TopFragment on Hero { @@ -397,7 +392,6 @@ describe('Execute: defer directive', () => { name: 'Luke', }, path: ['hero'], - label: 'DeferTop', }, ], hasNext: false, @@ -410,7 +404,7 @@ describe('Execute: defer directive', () => { query HeroNameQuery { hero { id - ... on Hero @defer(label: "InlineDeferred") { + ... on Hero @defer { name } } @@ -424,9 +418,7 @@ describe('Execute: defer directive', () => { hasNext: true, }, { - incremental: [ - { data: { name: 'Luke' }, path: ['hero'], label: 'InlineDeferred' }, - ], + incremental: [{ data: { name: 'Luke' }, path: ['hero'] }], hasNext: false, }, ]); diff --git a/src/execution/__tests__/mutations-test.ts b/src/execution/__tests__/mutations-test.ts index fa533c75ea..ce3b6c5fc7 100644 --- a/src/execution/__tests__/mutations-test.ts +++ b/src/execution/__tests__/mutations-test.ts @@ -206,7 +206,7 @@ describe('Execute: Handles mutation execution ordering', () => { const document = parse(` mutation M { first: promiseToChangeTheNumber(newNumber: 1) { - ...DeferFragment @defer(label: "defer-label") + ...DeferFragment @defer }, second: immediatelyChangeTheNumber(newNumber: 2) { theNumber @@ -242,7 +242,6 @@ describe('Execute: Handles mutation execution ordering', () => { { incremental: [ { - label: 'defer-label', path: ['first'], data: { promiseToGetTheNumber: 2, @@ -281,7 +280,7 @@ describe('Execute: Handles mutation execution ordering', () => { it('Mutation with @defer is not executed serially', async () => { const document = parse(` mutation M { - ...MutationFragment @defer(label: "defer-label") + ...MutationFragment @defer second: immediatelyChangeTheNumber(newNumber: 2) { theNumber } @@ -317,7 +316,6 @@ describe('Execute: Handles mutation execution ordering', () => { { incremental: [ { - label: 'defer-label', path: [], data: { first: { diff --git a/src/execution/__tests__/stream-test.ts b/src/execution/__tests__/stream-test.ts index cd9b9b3965..7a168269c9 100644 --- a/src/execution/__tests__/stream-test.ts +++ b/src/execution/__tests__/stream-test.ts @@ -210,9 +210,7 @@ describe('Execute: stream directive', () => { }); }); it('Returns label from stream directive', async () => { - const document = parse( - '{ scalarList @stream(initialCount: 1, label: "scalar-stream") }', - ); + const document = parse('{ scalarList @stream(initialCount: 1) }'); const result = await complete(document, { scalarList: () => ['apple', 'banana', 'coconut'], }); @@ -228,7 +226,6 @@ describe('Execute: stream directive', () => { { items: ['banana'], path: ['scalarList', 1], - label: 'scalar-stream', }, ], hasNext: true, @@ -238,7 +235,6 @@ describe('Execute: stream directive', () => { { items: ['coconut'], path: ['scalarList', 2], - label: 'scalar-stream', }, ], hasNext: false, @@ -1661,8 +1657,8 @@ describe('Execute: stream directive', () => { const document = parse(` query { - friendList @stream(initialCount: 1, label:"stream-label") { - ...NameFragment @defer(label: "DeferName") @defer(label: "DeferName") + friendList @stream(initialCount: 1) { + ...NameFragment @defer id } } @@ -1705,12 +1701,10 @@ describe('Execute: stream directive', () => { { data: { name: 'Luke' }, path: ['friendList', 0], - label: 'DeferName', }, { items: [{ id: '2' }], path: ['friendList', 1], - label: 'stream-label', }, ], hasNext: true, @@ -1727,7 +1721,6 @@ describe('Execute: stream directive', () => { { data: { name: 'Han' }, path: ['friendList', 1], - label: 'DeferName', }, ], hasNext: false, @@ -1747,8 +1740,8 @@ describe('Execute: stream directive', () => { const document = parse(` query { - friendList @stream(initialCount: 1, label:"stream-label") { - ...NameFragment @defer(label: "DeferName") @defer(label: "DeferName") + friendList @stream(initialCount: 1) { + ...NameFragment @defer id } } @@ -1791,12 +1784,10 @@ describe('Execute: stream directive', () => { { data: { name: 'Luke' }, path: ['friendList', 0], - label: 'DeferName', }, { items: [{ id: '2' }], path: ['friendList', 1], - label: 'stream-label', }, ], hasNext: true, @@ -1811,7 +1802,6 @@ describe('Execute: stream directive', () => { { data: { name: 'Han' }, path: ['friendList', 1], - label: 'DeferName', }, ], hasNext: true, diff --git a/src/execution/__tests__/sync-test.ts b/src/execution/__tests__/sync-test.ts index f5efa4097c..aa68dcf441 100644 --- a/src/execution/__tests__/sync-test.ts +++ b/src/execution/__tests__/sync-test.ts @@ -117,7 +117,7 @@ describe('Execute: synchronously when possible', () => { it('throws if encountering async iterable execution', () => { const doc = ` query Example { - ...deferFrag @defer(label: "deferLabel") + ...deferFrag @defer } fragment deferFrag on Query { syncField diff --git a/src/execution/collectFields.ts b/src/execution/collectFields.ts index faa273ee32..d4c92fe0ca 100644 --- a/src/execution/collectFields.ts +++ b/src/execution/collectFields.ts @@ -31,7 +31,6 @@ export type FieldGroup = ReadonlyArray; export type GroupedFieldSet = Map; export interface PatchFields { - label: string | undefined; fields: GroupedFieldSet; } @@ -147,7 +146,7 @@ function collectFieldsImpl( continue; } - const defer = getDeferValues(operation, variableValues, selection); + const defer = isFragmentDeferred(operation, variableValues, selection); if (defer) { const patchFields = new AccumulatorMap(); @@ -163,7 +162,6 @@ function collectFieldsImpl( visitedFragmentNames, ); patches.push({ - label: defer.label, fields: patchFields, }); } else { @@ -188,7 +186,7 @@ function collectFieldsImpl( continue; } - const defer = getDeferValues(operation, variableValues, selection); + const defer = isFragmentDeferred(operation, variableValues, selection); if (visitedFragmentNames.has(fragName) && !defer) { continue; } @@ -219,7 +217,6 @@ function collectFieldsImpl( visitedFragmentNames, ); patches.push({ - label: defer.label, fields: patchFields, }); } else { @@ -246,19 +243,19 @@ function collectFieldsImpl( * deferred based on the experimental flag, defer directive present and * not disabled by the "if" argument. */ -function getDeferValues( +function isFragmentDeferred( operation: OperationDefinitionNode, variableValues: { [variable: string]: unknown }, node: FragmentSpreadNode | InlineFragmentNode, -): undefined | { label: string | undefined } { +): boolean { const defer = getDirectiveValues(GraphQLDeferDirective, node, variableValues); if (!defer) { - return; + return false; } if (defer.if === false) { - return; + return false; } invariant( @@ -266,9 +263,7 @@ function getDeferValues( '`@defer` directive not supported on subscription operations. Disable `@defer` by setting the `if` argument to `false`.', ); - return { - label: typeof defer.label === 'string' ? defer.label : undefined, - }; + return true; } /** diff --git a/src/execution/execute.ts b/src/execution/execute.ts index 2485ead567..5b1608e229 100644 --- a/src/execution/execute.ts +++ b/src/execution/execute.ts @@ -204,7 +204,6 @@ export interface IncrementalDeferResult< TExtensions = ObjMap, > extends ExecutionResult { path?: ReadonlyArray; - label?: string; } export interface FormattedIncrementalDeferResult< @@ -212,7 +211,6 @@ export interface FormattedIncrementalDeferResult< TExtensions = ObjMap, > extends FormattedExecutionResult { path?: ReadonlyArray; - label?: string; } export interface IncrementalStreamResult< @@ -222,7 +220,6 @@ export interface IncrementalStreamResult< errors?: ReadonlyArray; items?: TData | null; path?: ReadonlyArray; - label?: string; extensions?: TExtensions; } @@ -233,7 +230,6 @@ export interface FormattedIncrementalStreamResult< errors?: ReadonlyArray; items?: TData | null; path?: ReadonlyArray; - label?: string; extensions?: TExtensions; } @@ -578,15 +574,8 @@ function executeOperation( } for (const patch of patches) { - const { label, fields: patchFields } = patch; - executeDeferredFragment( - exeContext, - rootType, - rootValue, - patchFields, - label, - path, - ); + const { fields: patchFields } = patch; + executeDeferredFragment(exeContext, rootType, rootValue, patchFields, path); } return result; @@ -979,7 +968,6 @@ function getStreamValues( | undefined | { initialCount: number | undefined; - label: string | undefined; } { // do not stream inner lists of multi-dimensional lists if (typeof path.key === 'number') { @@ -1019,7 +1007,6 @@ function getStreamValues( return { initialCount: stream.initialCount, - label: typeof stream.label === 'string' ? stream.label : undefined, }; } @@ -1057,7 +1044,6 @@ async function completeAsyncIteratorValue( info, itemType, path, - stream.label, asyncPayloadRecord, ); break; @@ -1159,7 +1145,6 @@ function completeListValue( fieldGroup, info, itemType, - stream.label, previousAsyncPayloadRecord, ); index++; @@ -1475,13 +1460,12 @@ function collectAndExecuteSubfields( ); for (const subPatch of subPatches) { - const { label, fields: subPatchFieldNodes } = subPatch; + const { fields: subPatchFieldNodes } = subPatch; executeDeferredFragment( exeContext, returnType, result, subPatchFieldNodes, - label, path, asyncPayloadRecord, ); @@ -1783,12 +1767,10 @@ function executeDeferredFragment( parentType: GraphQLObjectType, sourceValue: unknown, groupedFieldSet: GroupedFieldSet, - label?: string, path?: Path, parentContext?: AsyncPayloadRecord, ): void { const asyncPayloadRecord = new DeferredFragmentRecord({ - label, path, parentContext, exeContext, @@ -1825,11 +1807,9 @@ function executeStreamField( fieldGroup: FieldGroup, info: GraphQLResolveInfo, itemType: GraphQLOutputType, - label?: string, parentContext?: AsyncPayloadRecord, ): AsyncPayloadRecord { const asyncPayloadRecord = new StreamRecord({ - label, path: itemPath, parentContext, exeContext, @@ -1977,7 +1957,6 @@ async function executeStreamIterator( info: GraphQLResolveInfo, itemType: GraphQLOutputType, path: Path, - label?: string, parentContext?: AsyncPayloadRecord, ): Promise { let index = initialIndex; @@ -1986,7 +1965,6 @@ async function executeStreamIterator( while (true) { const itemPath = addPath(path, index, undefined); const asyncPayloadRecord = new StreamRecord({ - label, path: itemPath, parentContext: previousAsyncPayloadRecord, iterator, @@ -2094,9 +2072,6 @@ function getCompletedIncrementalResults( } incrementalResult.path = asyncPayloadRecord.path; - if (asyncPayloadRecord.label) { - incrementalResult.label = asyncPayloadRecord.label; - } if (asyncPayloadRecord.errors.length > 0) { incrementalResult.errors = asyncPayloadRecord.errors; } @@ -2181,7 +2156,6 @@ function yieldSubsequentPayloads( class DeferredFragmentRecord { type: 'defer'; errors: Array; - label: string | undefined; path: Array; promise: Promise; data: ObjMap | null; @@ -2190,13 +2164,11 @@ class DeferredFragmentRecord { _exeContext: ExecutionContext; _resolve?: (arg: PromiseOrValue | null>) => void; constructor(opts: { - label: string | undefined; path: Path | undefined; parentContext: AsyncPayloadRecord | undefined; exeContext: ExecutionContext; }) { this.type = 'defer'; - this.label = opts.label; this.path = pathToArray(opts.path); this.parentContext = opts.parentContext; this.errors = []; @@ -2227,7 +2199,6 @@ class DeferredFragmentRecord { class StreamRecord { type: 'stream'; errors: Array; - label: string | undefined; path: Array; items: Array | null; promise: Promise; @@ -2238,7 +2209,6 @@ class StreamRecord { _exeContext: ExecutionContext; _resolve?: (arg: PromiseOrValue | null>) => void; constructor(opts: { - label: string | undefined; path: Path | undefined; iterator?: AsyncIterator; parentContext: AsyncPayloadRecord | undefined; @@ -2246,7 +2216,6 @@ class StreamRecord { }) { this.type = 'stream'; this.items = null; - this.label = opts.label; this.path = pathToArray(opts.path); this.parentContext = opts.parentContext; this.iterator = opts.iterator; diff --git a/src/type/directives.ts b/src/type/directives.ts index 8fd5a6a62e..3f5664a96a 100644 --- a/src/type/directives.ts +++ b/src/type/directives.ts @@ -170,10 +170,6 @@ export const GraphQLDeferDirective = new GraphQLDirective({ description: 'Deferred when true or undefined.', defaultValue: true, }, - label: { - type: GraphQLString, - description: 'Unique name', - }, }, }); @@ -191,10 +187,6 @@ export const GraphQLStreamDirective = new GraphQLDirective({ description: 'Stream when true or undefined.', defaultValue: true, }, - label: { - type: GraphQLString, - description: 'Unique name', - }, initialCount: { defaultValue: 0, type: GraphQLInt, diff --git a/src/validation/__tests__/DeferStreamDirectiveLabelRule-test.ts b/src/validation/__tests__/DeferStreamDirectiveLabelRule-test.ts deleted file mode 100644 index a5ffd1cfa9..0000000000 --- a/src/validation/__tests__/DeferStreamDirectiveLabelRule-test.ts +++ /dev/null @@ -1,171 +0,0 @@ -import { describe, it } from 'mocha'; - -import { DeferStreamDirectiveLabelRule } from '../rules/DeferStreamDirectiveLabelRule.js'; - -import { expectValidationErrors } from './harness.js'; - -function expectErrors(queryStr: string) { - return expectValidationErrors(DeferStreamDirectiveLabelRule, queryStr); -} - -function expectValid(queryStr: string) { - expectErrors(queryStr).toDeepEqual([]); -} - -describe('Validate: Defer/Stream directive on root field', () => { - it('Defer fragments with no label', () => { - expectValid(` - { - dog { - ...dogFragmentA @defer - ...dogFragmentB @defer - } - } - fragment dogFragmentA on Dog { - name - } - fragment dogFragmentB on Dog { - nickname - } - `); - }); - - it('Defer fragments, one with label, one without', () => { - expectValid(` - { - dog { - ...dogFragmentA @defer(label: "fragA") - ...dogFragmentB @defer - } - } - fragment dogFragmentA on Dog { - name - } - fragment dogFragmentB on Dog { - nickname - } - `); - }); - - it('Defer fragment with variable label', () => { - expectErrors(` - query($label: String) { - dog { - ...dogFragmentA @defer(label: $label) - ...dogFragmentB @defer(label: "fragA") - } - } - fragment dogFragmentA on Dog { - name - } - fragment dogFragmentB on Dog { - nickname - } - `).toDeepEqual([ - { - message: 'Directive "defer"\'s label argument must be a static string.', - locations: [{ line: 4, column: 25 }], - }, - ]); - }); - - it('Defer fragments with different labels', () => { - expectValid(` - { - dog { - ...dogFragmentA @defer(label: "fragB") - ...dogFragmentB @defer(label: "fragA") - } - } - fragment dogFragmentA on Dog { - name - } - fragment dogFragmentB on Dog { - nickname - } - `); - }); - it('Defer fragments with same label', () => { - expectErrors(` - { - dog { - ...dogFragmentA @defer(label: "fragA") - ...dogFragmentB @defer(label: "fragA") - } - } - fragment dogFragmentA on Dog { - name - } - fragment dogFragmentB on Dog { - nickname - } - `).toDeepEqual([ - { - message: 'Defer/Stream directive label argument must be unique.', - locations: [ - { line: 4, column: 25 }, - { line: 5, column: 25 }, - ], - }, - ]); - }); - it('Defer and stream with no label', () => { - expectValid(` - { - dog { - ...dogFragment @defer - } - pets @stream(initialCount: 0) @stream { - name - } - } - fragment dogFragment on Dog { - name - } - `); - }); - it('Stream with variable label', () => { - expectErrors(` - query ($label: String!) { - dog { - ...dogFragment @defer - } - pets @stream(initialCount: 0) @stream(label: $label) { - name - } - } - fragment dogFragment on Dog { - name - } - `).toDeepEqual([ - { - message: - 'Directive "stream"\'s label argument must be a static string.', - locations: [{ line: 6, column: 39 }], - }, - ]); - }); - it('Defer and stream with the same label', () => { - expectErrors(` - { - dog { - ...dogFragment @defer(label: "MyLabel") - } - pets @stream(initialCount: 0) @stream(label: "MyLabel") { - name - } - } - fragment dogFragment on Dog { - name - } - `).toDeepEqual([ - { - message: 'Defer/Stream directive label argument must be unique.', - locations: [ - { line: 4, column: 26 }, - { line: 6, column: 39 }, - ], - }, - ]); - }); -}); diff --git a/src/validation/__tests__/OverlappingFieldsCanBeMergedRule-test.ts b/src/validation/__tests__/OverlappingFieldsCanBeMergedRule-test.ts index 52c2deb1a0..66e7a7721c 100644 --- a/src/validation/__tests__/OverlappingFieldsCanBeMergedRule-test.ts +++ b/src/validation/__tests__/OverlappingFieldsCanBeMergedRule-test.ts @@ -101,35 +101,17 @@ describe('Validate: Overlapping fields can be merged', () => { it('Same stream directives supported', () => { expectValid(` fragment differentDirectivesWithDifferentAliases on Dog { - name @stream(label: "streamLabel", initialCount: 1) - name @stream(label: "streamLabel", initialCount: 1) + name @stream(initialCount: 1) + name @stream(initialCount: 1) } `); }); - it('different stream directive label', () => { - expectErrors(` - fragment conflictingArgs on Dog { - name @stream(label: "streamLabel", initialCount: 1) - name @stream(label: "anotherLabel", initialCount: 1) - } - `).toDeepEqual([ - { - message: - 'Fields "name" conflict because they have differing stream directives. Use different aliases on the fields to fetch both if this was intentional.', - locations: [ - { line: 3, column: 9 }, - { line: 4, column: 9 }, - ], - }, - ]); - }); - it('different stream directive initialCount', () => { expectErrors(` fragment conflictingArgs on Dog { - name @stream(label: "streamLabel", initialCount: 1) - name @stream(label: "streamLabel", initialCount: 2) + name @stream(initialCount: 1) + name @stream(initialCount: 2) } `).toDeepEqual([ { @@ -147,7 +129,7 @@ describe('Validate: Overlapping fields can be merged', () => { expectErrors(` fragment conflictingArgs on Dog { name @stream - name @stream(label: "streamLabel", initialCount: 1) + name @stream(initialCount: 1) } `).toDeepEqual([ { @@ -164,7 +146,7 @@ describe('Validate: Overlapping fields can be merged', () => { it('different stream directive second missing args', () => { expectErrors(` fragment conflictingArgs on Dog { - name @stream(label: "streamLabel", initialCount: 1) + name @stream(initialCount: 1) name @stream } `).toDeepEqual([ diff --git a/src/validation/index.ts b/src/validation/index.ts index b0cc754490..ceea29e84b 100644 --- a/src/validation/index.ts +++ b/src/validation/index.ts @@ -6,9 +6,6 @@ export type { ValidationRule } from './ValidationContext.js'; // All validation rules in the GraphQL Specification. export { specifiedRules } from './specifiedRules.js'; -// Spec Section: "Defer And Stream Directive Labels Are Unique" -export { DeferStreamDirectiveLabelRule } from './rules/DeferStreamDirectiveLabelRule.js'; - // Spec Section: "Defer And Stream Directives Are Used On Valid Root Field" export { DeferStreamDirectiveOnRootFieldRule } from './rules/DeferStreamDirectiveOnRootFieldRule.js'; diff --git a/src/validation/rules/DeferStreamDirectiveLabelRule.ts b/src/validation/rules/DeferStreamDirectiveLabelRule.ts deleted file mode 100644 index a0fc3cc424..0000000000 --- a/src/validation/rules/DeferStreamDirectiveLabelRule.ts +++ /dev/null @@ -1,55 +0,0 @@ -import { GraphQLError } from '../../error/GraphQLError.js'; - -import { Kind } from '../../language/kinds.js'; -import type { ASTVisitor } from '../../language/visitor.js'; - -import { - GraphQLDeferDirective, - GraphQLStreamDirective, -} from '../../type/directives.js'; - -import type { ValidationContext } from '../ValidationContext.js'; - -/** - * Defer and stream directive labels are unique - * - * A GraphQL document is only valid if defer and stream directives' label argument is static and unique. - */ -export function DeferStreamDirectiveLabelRule( - context: ValidationContext, -): ASTVisitor { - const knownLabels = Object.create(null); - return { - Directive(node) { - if ( - node.name.value === GraphQLDeferDirective.name || - node.name.value === GraphQLStreamDirective.name - ) { - const labelArgument = node.arguments?.find( - (arg) => arg.name.value === 'label', - ); - const labelValue = labelArgument?.value; - if (!labelValue) { - return; - } - if (labelValue.kind !== Kind.STRING) { - context.reportError( - new GraphQLError( - `Directive "${node.name.value}"'s label argument must be a static string.`, - { nodes: node }, - ), - ); - } else if (knownLabels[labelValue.value]) { - context.reportError( - new GraphQLError( - 'Defer/Stream directive label argument must be unique.', - { nodes: [knownLabels[labelValue.value], node] }, - ), - ); - } else { - knownLabels[labelValue.value] = node; - } - } - }, - }; -} diff --git a/src/validation/specifiedRules.ts b/src/validation/specifiedRules.ts index 60c967f8f0..6187ac0182 100644 --- a/src/validation/specifiedRules.ts +++ b/src/validation/specifiedRules.ts @@ -1,5 +1,3 @@ -// Spec Section: "Defer And Stream Directive Labels Are Unique" -import { DeferStreamDirectiveLabelRule } from './rules/DeferStreamDirectiveLabelRule.js'; // Spec Section: "Defer And Stream Directives Are Used On Valid Root Field" import { DeferStreamDirectiveOnRootFieldRule } from './rules/DeferStreamDirectiveOnRootFieldRule.js'; // Spec Section: "Defer And Stream Directives Are Used On Valid Operations" @@ -103,7 +101,6 @@ export const specifiedRules: ReadonlyArray = Object.freeze([ UniqueDirectivesPerLocationRule, DeferStreamDirectiveOnRootFieldRule, DeferStreamDirectiveOnValidOperationsRule, - DeferStreamDirectiveLabelRule, StreamDirectiveOnListFieldRule, KnownArgumentNamesRule, UniqueArgumentNamesRule, From 086c92e2ef18fe0003fea85fce9df6a82b673b90 Mon Sep 17 00:00:00 2001 From: Yaacov Rydzinski Date: Sun, 8 Jan 2023 15:33:25 +0200 Subject: [PATCH 3/7] try aggressive merging --- src/execution/__tests__/defer-test.ts | 409 +++++++++++++++++- src/execution/collectFields.ts | 192 ++++---- src/execution/execute.ts | 256 ++++++++--- .../rules/SingleFieldSubscriptionsRule.ts | 13 +- 4 files changed, 701 insertions(+), 169 deletions(-) diff --git a/src/execution/__tests__/defer-test.ts b/src/execution/__tests__/defer-test.ts index 712ef87579..c8701b3eb0 100644 --- a/src/execution/__tests__/defer-test.ts +++ b/src/execution/__tests__/defer-test.ts @@ -40,6 +40,66 @@ const friends = [ { name: 'C-3PO', id: 4 }, ]; +const deeperObject = new GraphQLObjectType({ + fields: { + foo: { type: GraphQLString, resolve: () => 'foo' }, + bar: { type: GraphQLString, resolve: () => 'bar' }, + baz: { type: GraphQLString, resolve: () => 'baz' }, + bak: { type: GraphQLString, resolve: () => 'bak' }, + }, + name: 'DeeperObject', +}); + +const nestedObject = new GraphQLObjectType({ + fields: { + deeperObject: { type: deeperObject, resolve: () => ({}) }, + }, + name: 'NestedObject', +}); + +const anotherNestedObject = new GraphQLObjectType({ + fields: { + deeperObject: { type: deeperObject, resolve: () => ({}) }, + }, + name: 'AnotherNestedObject', +}); + +const c = new GraphQLObjectType({ + fields: { + d: { type: GraphQLString, resolve: () => 'd' }, + }, + name: 'c', +}); + +const e = new GraphQLObjectType({ + fields: { + f: { type: GraphQLString, resolve: () => 'f' }, + }, + name: 'e', +}); + +const b = new GraphQLObjectType({ + fields: { + c: { type: c, resolve: () => ({}) }, + e: { type: e, resolve: () => ({}) }, + }, + name: 'b', +}); + +const a = new GraphQLObjectType({ + fields: { + b: { type: b, resolve: () => ({}) }, + }, + name: 'a', +}); + +const g = new GraphQLObjectType({ + fields: { + h: { type: GraphQLString, resolve: () => 'h' }, + }, + name: 'g', +}); + const heroType = new GraphQLObjectType({ fields: { id: { type: GraphQLID }, @@ -75,6 +135,8 @@ const heroType = new GraphQLObjectType({ yield await Promise.resolve(friends[0]); }, }, + nestedObject: { type: nestedObject, resolve: () => ({}) }, + anotherNestedObject: { type: anotherNestedObject, resolve: () => ({}) }, }, name: 'Hero', }); @@ -87,6 +149,8 @@ const query = new GraphQLObjectType({ type: heroType, resolve: () => hero, }, + a: { type: a, resolve: () => ({}) }, + g: { type: g, resolve: () => ({}) }, }, name: 'Query', }); @@ -141,7 +205,6 @@ describe('Execute: defer directive', () => { incremental: [ { data: { - id: '1', name: 'Luke', }, path: ['hero'], @@ -307,15 +370,10 @@ describe('Execute: defer directive', () => { }, { incremental: [ - { - data: { - friends: [{ name: 'Han' }, { name: 'Leia' }, { name: 'C-3PO' }], - }, - path: ['hero'], - }, { data: { name: 'Luke', + friends: [{ name: 'Han' }, { name: 'Leia' }, { name: 'C-3PO' }], }, path: ['hero'], }, @@ -351,9 +409,7 @@ describe('Execute: defer directive', () => { { incremental: [ { - data: { - name: 'Luke', - }, + data: {}, path: ['hero'], }, ], @@ -388,9 +444,7 @@ describe('Execute: defer directive', () => { { incremental: [ { - data: { - name: 'Luke', - }, + data: {}, path: ['hero'], }, ], @@ -423,6 +477,335 @@ describe('Execute: defer directive', () => { }, ]); }); + + it('Can deduplicate multiple defers on the same object', async () => { + const document = parse(` + query { + hero { + friends { + ... @defer { + ...FriendFrag + ... @defer { + ...FriendFrag + ... @defer { + ...FriendFrag + ... @defer { + ...FriendFrag + } + } + } + } + } + } + } + + fragment FriendFrag on Friend { + id + name + } + `); + const result = await complete(document); + + expectJSON(result).toDeepEqual([ + { + data: { hero: { friends: [{}, {}, {}] } }, + hasNext: true, + }, + { + incremental: [ + { data: { id: '2', name: 'Han' }, path: ['hero', 'friends', 0] }, + { data: { id: '3', name: 'Leia' }, path: ['hero', 'friends', 1] }, + { data: { id: '4', name: 'C-3PO' }, path: ['hero', 'friends', 2] }, + ], + hasNext: false, + }, + ]); + }); + + it('Can deduplicate leaf fields present in the initial payload', async () => { + const document = parse(` + query { + hero { + nestedObject { + deeperObject { + foo + } + } + anotherNestedObject { + deeperObject { + foo + } + } + ... @defer { + nestedObject { + deeperObject { + bar + } + } + anotherNestedObject { + deeperObject { + foo + } + } + } + } + } + `); + const result = await complete(document); + expectJSON(result).toDeepEqual([ + { + data: { + hero: { + nestedObject: { + deeperObject: { + foo: 'foo', + }, + }, + anotherNestedObject: { + deeperObject: { + foo: 'foo', + }, + }, + }, + }, + hasNext: true, + }, + { + incremental: [ + { + data: { + nestedObject: { + deeperObject: { + bar: 'bar', + }, + }, + anotherNestedObject: { + deeperObject: {}, + }, + }, + path: ['hero'], + }, + ], + hasNext: false, + }, + ]); + }); + + it('Can deduplicate initial fields with deferred fragments at multiple levels', async () => { + const document = parse(` + query { + hero { + nestedObject { + deeperObject { + foo + } + } + ... @defer { + nestedObject { + deeperObject { + foo + bar + } + ... @defer { + deeperObject { + foo + bar + baz + ... @defer { + foo + bar + baz + bak + } + } + } + } + } + } + } + `); + const result = await complete(document); + expectJSON(result).toDeepEqual([ + { + data: { + hero: { + nestedObject: { + deeperObject: { + foo: 'foo', + }, + }, + }, + }, + hasNext: true, + }, + { + incremental: [ + { + data: { + bar: 'bar', + baz: 'baz', + bak: 'bak', + }, + path: ['hero', 'nestedObject', 'deeperObject'], + }, + { + data: { + deeperObject: { + bar: 'bar', + baz: 'baz', + }, + }, + path: ['hero', 'nestedObject'], + }, + { + data: { + nestedObject: { + deeperObject: { + bar: 'bar', + }, + }, + }, + path: ['hero'], + }, + ], + hasNext: false, + }, + ]); + }); + + it('Can combine multiple fields from deferred fragments from different branches occurring at the same level', async () => { + const document = parse(` + query { + hero { + nestedObject { + deeperObject { + ... @defer { + foo + } + } + } + ... @defer { + nestedObject { + deeperObject { + ... @defer { + foo + bar + } + } + } + } + } + } + `); + const result = await complete(document); + expectJSON(result).toDeepEqual([ + { + data: { + hero: { + nestedObject: { + deeperObject: {}, + }, + }, + }, + hasNext: true, + }, + { + incremental: [ + { + data: { + foo: 'foo', + bar: 'bar', + }, + path: ['hero', 'nestedObject', 'deeperObject'], + }, + { + data: { + nestedObject: { + deeperObject: {}, + }, + }, + path: ['hero'], + }, + ], + hasNext: false, + }, + ]); + }); + + it('can deduplicate initial fields with deferred fragments in different branches at multiple non-overlapping levels', async () => { + const document = parse(` + query { + a { + b { + c { + d + } + ... @defer { + e { + f + } + } + } + } + ... @defer { + a { + b { + e { + f + } + } + } + g { + h + } + } + } + `); + const result = await complete(document); + expectJSON(result).toDeepEqual([ + { + data: { + a: { + b: { + c: { + d: 'd', + }, + }, + }, + }, + hasNext: true, + }, + { + incremental: [ + { + data: { + e: { + f: 'f', + }, + }, + path: ['a', 'b'], + }, + { + data: { + a: { + b: { + e: { + f: 'f', + }, + }, + }, + g: { + h: 'h', + }, + }, + path: [], + }, + ], + hasNext: false, + }, + ]); + }); + it('Handles errors thrown in deferred fragments', async () => { const document = parse(` query HeroNameQuery { diff --git a/src/execution/collectFields.ts b/src/execution/collectFields.ts index d4c92fe0ca..a953b38467 100644 --- a/src/execution/collectFields.ts +++ b/src/execution/collectFields.ts @@ -26,17 +26,29 @@ import { typeFromAST } from '../utilities/typeFromAST.js'; import { getDirectiveValues } from './values.js'; -export type FieldGroup = ReadonlyArray; +export type FieldGroup = ReadonlyArray; export type GroupedFieldSet = Map; -export interface PatchFields { - fields: GroupedFieldSet; -} - -export interface FieldsAndPatches { - fields: GroupedFieldSet; - patches: Array; +/** + * A tagged field node includes metadata necessary to determine whether a field should + * be executed. + * + * A field's depth is equivalent to the number of fields between the given field and + * the operation root. For example, root fields have a depth of 0, their sub-fields + * have a depth of 1, and so on. Tagging fields with their depth is necessary only to + * compute a field's "defer depth". + * + * A field's defer depth is the depth of the closest containing defer directive , or + * undefined, if the field is not contained by a deferred fragment. + * + * Because deferred fragments at a given level are merged, the defer depth may be used + * as a unique id to tag the fields for inclusion within a given deferred payload. + */ +export interface TaggedFieldNode { + fieldNode: FieldNode; + depth: number; + deferDepth: number | undefined; } /** @@ -54,21 +66,27 @@ export function collectFields( variableValues: { [variable: string]: unknown }, runtimeType: GraphQLObjectType, operation: OperationDefinitionNode, -): FieldsAndPatches { - const fields = new AccumulatorMap(); - const patches: Array = []; - collectFieldsImpl( +): { + groupedFieldSet: GroupedFieldSet; + newDeferDepth: number | undefined; +} { + const groupedFieldSet = new AccumulatorMap(); + const newDeferDepth = collectFieldsImpl( schema, fragments, variableValues, operation, runtimeType, operation.selectionSet, - fields, - patches, + groupedFieldSet, new Set(), + 0, + undefined, ); - return { fields, patches }; + return { + groupedFieldSet, + newDeferDepth, + }; } /** @@ -89,32 +107,38 @@ export function collectSubfields( operation: OperationDefinitionNode, returnType: GraphQLObjectType, fieldGroup: FieldGroup, -): FieldsAndPatches { - const subFieldNodes = new AccumulatorMap(); +): { + groupedFieldSet: GroupedFieldSet; + newDeferDepth: number | undefined; +} { + const groupedFieldSet = new AccumulatorMap(); + let newDeferDepth: number | undefined; const visitedFragmentNames = new Set(); - const subPatches: Array = []; - const subFieldsAndPatches = { - fields: subFieldNodes, - patches: subPatches, - }; - - for (const node of fieldGroup) { - if (node.selectionSet) { - collectFieldsImpl( + for (const field of fieldGroup) { + if (field.fieldNode.selectionSet) { + const nestedNewDeferDepth = collectFieldsImpl( schema, fragments, variableValues, operation, returnType, - node.selectionSet, - subFieldNodes, - subPatches, + field.fieldNode.selectionSet, + groupedFieldSet, visitedFragmentNames, + fieldGroup[0].depth + 1, + field.deferDepth, ); + if (nestedNewDeferDepth !== undefined) { + newDeferDepth = nestedNewDeferDepth; + } } } - return subFieldsAndPatches; + + return { + groupedFieldSet, + newDeferDepth, + }; } // eslint-disable-next-line max-params @@ -125,17 +149,23 @@ function collectFieldsImpl( operation: OperationDefinitionNode, runtimeType: GraphQLObjectType, selectionSet: SelectionSetNode, - fields: AccumulatorMap, - patches: Array, + groupedFieldSet: AccumulatorMap, visitedFragmentNames: Set, -): void { + depth: number, + deferDepth: number | undefined, +): number | undefined { + let hasNewDefer = false; for (const selection of selectionSet.selections) { switch (selection.kind) { case Kind.FIELD: { if (!shouldIncludeNode(variableValues, selection)) { continue; } - fields.add(getFieldEntryKey(selection), selection); + groupedFieldSet.add(getFieldEntryKey(selection), { + fieldNode: selection, + depth, + deferDepth, + }); break; } case Kind.INLINE_FRAGMENT: { @@ -148,35 +178,21 @@ function collectFieldsImpl( const defer = isFragmentDeferred(operation, variableValues, selection); - if (defer) { - const patchFields = new AccumulatorMap(); - collectFieldsImpl( - schema, - fragments, - variableValues, - operation, - runtimeType, - selection.selectionSet, - patchFields, - patches, - visitedFragmentNames, - ); - patches.push({ - fields: patchFields, - }); - } else { - collectFieldsImpl( - schema, - fragments, - variableValues, - operation, - runtimeType, - selection.selectionSet, - fields, - patches, - visitedFragmentNames, - ); - } + const nestedHasNewDefer = collectFieldsImpl( + schema, + fragments, + variableValues, + operation, + runtimeType, + selection.selectionSet, + groupedFieldSet, + visitedFragmentNames, + depth, + defer ? depth : deferDepth, + ); + + hasNewDefer ||= defer || nestedHasNewDefer !== undefined; + break; } case Kind.FRAGMENT_SPREAD: { @@ -203,45 +219,31 @@ function collectFieldsImpl( visitedFragmentNames.add(fragName); } - if (defer) { - const patchFields = new AccumulatorMap(); - collectFieldsImpl( - schema, - fragments, - variableValues, - operation, - runtimeType, - fragment.selectionSet, - patchFields, - patches, - visitedFragmentNames, - ); - patches.push({ - fields: patchFields, - }); - } else { - collectFieldsImpl( - schema, - fragments, - variableValues, - operation, - runtimeType, - fragment.selectionSet, - fields, - patches, - visitedFragmentNames, - ); - } + const nestedNewDeferDepth = collectFieldsImpl( + schema, + fragments, + variableValues, + operation, + runtimeType, + fragment.selectionSet, + groupedFieldSet, + visitedFragmentNames, + depth, + defer ? depth : deferDepth, + ); + + hasNewDefer ||= defer || nestedNewDeferDepth !== undefined; + break; } } } + return hasNewDefer ? depth : undefined; } /** - * Returns an object containing the `@defer` arguments if a field should be - * deferred based on the experimental flag, defer directive present and - * not disabled by the "if" argument. + * Returns whether a fragment should be deferred based on the presence of a + * defer directive and whether it is disabled by the "if" argument. */ function isFragmentDeferred( operation: OperationDefinitionNode, diff --git a/src/execution/execute.ts b/src/execution/execute.ts index 5b1608e229..8b9e6b5544 100644 --- a/src/execution/execute.ts +++ b/src/execution/execute.ts @@ -19,6 +19,7 @@ import { locatedError } from '../error/locatedError.js'; import type { DocumentNode, + FieldNode, FragmentDefinitionNode, OperationDefinitionNode, } from '../language/ast.js'; @@ -37,6 +38,7 @@ import type { GraphQLTypeResolver, } from '../type/definition.js'; import { + getNamedType, isAbstractType, isLeafType, isListType, @@ -122,6 +124,7 @@ export interface ExecutionContext { subscribeFieldResolver: GraphQLFieldResolver; errors: Array; subsequentPayloads: Set; + branches: WeakMap>; } /** @@ -500,6 +503,7 @@ export function buildExecutionContext( typeResolver: typeResolver ?? defaultTypeResolver, subscribeFieldResolver: subscribeFieldResolver ?? defaultFieldResolver, subsequentPayloads: new Set(), + branches: new WeakMap(), errors: [], }; } @@ -512,10 +516,29 @@ function buildPerEventExecutionContext( ...exeContext, rootValue: payload, subsequentPayloads: new Set(), + branches: new WeakMap(), errors: [], }; } +function shouldBranch( + groupedFieldSet: GroupedFieldSet, + exeContext: ExecutionContext, + path: Path | undefined, +): boolean { + const set = exeContext.branches.get(groupedFieldSet); + const key = pathToArray(path).join('.'); + if (set === undefined) { + exeContext.branches.set(groupedFieldSet, new Set([key])); + return true; + } + if (!set.has(key)) { + set.add(key); + return true; + } + return false; +} + /** * Implements the "Executing operations" section of the spec. */ @@ -532,7 +555,7 @@ function executeOperation( ); } - const { fields: groupedFieldSet, patches } = collectFields( + const { groupedFieldSet, newDeferDepth } = collectFields( schema, fragments, variableValues, @@ -573,9 +596,18 @@ function executeOperation( ); } - for (const patch of patches) { - const { fields: patchFields } = patch; - executeDeferredFragment(exeContext, rootType, rootValue, patchFields, path); + if ( + newDeferDepth !== undefined && + shouldBranch(groupedFieldSet, exeContext, path) + ) { + executeDeferredFragment( + exeContext, + rootType, + rootValue, + groupedFieldSet, + newDeferDepth, + path, + ); } return result; @@ -596,16 +628,27 @@ function executeFieldsSerially( groupedFieldSet, (results, [responseName, fieldGroup]) => { const fieldPath = addPath(path, responseName, parentType.name); + + const fieldName = fieldGroup[0].fieldNode.name.value; + const fieldDef = exeContext.schema.getField(parentType, fieldName); + if (!fieldDef) { + return results; + } + + const returnType = fieldDef.type; + + if (!shouldExecute(fieldGroup, returnType)) { + return results; + } const result = executeField( exeContext, parentType, + fieldDef, + returnType, sourceValue, fieldGroup, fieldPath, ); - if (result === undefined) { - return results; - } if (isPromise(result)) { return result.then((resolvedResult) => { results[responseName] = resolvedResult; @@ -619,6 +662,29 @@ function executeFieldsSerially( ); } +function shouldExecute( + fieldGroup: FieldGroup, + returnType: GraphQLOutputType, + deferDepth?: number | undefined, +): boolean { + if (deferDepth === undefined || !isLeafType(getNamedType(returnType))) { + return fieldGroup.some( + ({ deferDepth: fieldDeferDepth }) => fieldDeferDepth === deferDepth, + ); + } + + let hasDepth = false; + for (const { deferDepth: fieldDeferDepth } of fieldGroup) { + if (fieldDeferDepth === undefined) { + return false; + } + if (fieldDeferDepth === deferDepth) { + hasDepth = true; + } + } + return hasDepth; +} + /** * Implements the "Executing selection sets" section of the spec * for fields that may be executed in parallel. @@ -637,19 +703,34 @@ function executeFields( try { for (const [responseName, fieldGroup] of groupedFieldSet) { const fieldPath = addPath(path, responseName, parentType.name); - const result = executeField( - exeContext, - parentType, - sourceValue, - fieldGroup, - fieldPath, - asyncPayloadRecord, - ); - if (result !== undefined) { - results[responseName] = result; - if (isPromise(result)) { - containsPromise = true; + const fieldName = fieldGroup[0].fieldNode.name.value; + const fieldDef = exeContext.schema.getField(parentType, fieldName); + if (!fieldDef) { + continue; + } + + const returnType = fieldDef.type; + + if ( + shouldExecute(fieldGroup, returnType, asyncPayloadRecord?.deferDepth) + ) { + const result = executeField( + exeContext, + parentType, + fieldDef, + returnType, + sourceValue, + fieldGroup, + fieldPath, + asyncPayloadRecord, + ); + + if (result !== undefined) { + results[responseName] = result; + if (isPromise(result)) { + containsPromise = true; + } } } } @@ -674,6 +755,9 @@ function executeFields( return promiseForObject(results); } +function toNodes(fieldGroup: FieldGroup): ReadonlyArray { + return fieldGroup.map(({ fieldNode }) => fieldNode); +} /** * Implements the "Executing fields" section of the spec * In particular, this function figures out the value that the field returns by @@ -683,19 +767,15 @@ function executeFields( function executeField( exeContext: ExecutionContext, parentType: GraphQLObjectType, + fieldDef: GraphQLField, + returnType: GraphQLOutputType, source: unknown, fieldGroup: FieldGroup, path: Path, asyncPayloadRecord?: AsyncPayloadRecord, ): PromiseOrValue { const errors = asyncPayloadRecord?.errors ?? exeContext.errors; - const fieldName = fieldGroup[0].name.value; - const fieldDef = exeContext.schema.getField(parentType, fieldName); - if (!fieldDef) { - return; - } - const returnType = fieldDef.type; const resolveFn = fieldDef.resolve ?? exeContext.fieldResolver; const info = buildResolveInfo( @@ -713,7 +793,7 @@ function executeField( // TODO: find a way to memoize, in case this field is within a List type. const args = getArgumentValues( fieldDef, - fieldGroup[0], + fieldGroup[0].fieldNode, exeContext.variableValues, ); @@ -750,7 +830,11 @@ function executeField( // Note: we don't rely on a `catch` method, but we do expect "thenable" // to take a second callback for the error case. return completed.then(undefined, (rawError) => { - const error = locatedError(rawError, fieldGroup, pathToArray(path)); + const error = locatedError( + rawError, + toNodes(fieldGroup), + pathToArray(path), + ); const handledError = handleFieldError(error, returnType, errors); filterSubsequentPayloads(exeContext, path, asyncPayloadRecord); return handledError; @@ -758,7 +842,11 @@ function executeField( } return completed; } catch (rawError) { - const error = locatedError(rawError, fieldGroup, pathToArray(path)); + const error = locatedError( + rawError, + toNodes(fieldGroup), + pathToArray(path), + ); const handledError = handleFieldError(error, returnType, errors); filterSubsequentPayloads(exeContext, path, asyncPayloadRecord); return handledError; @@ -780,7 +868,7 @@ export function buildResolveInfo( // information about the current execution state. return { fieldName: fieldDef.name, - fieldNodes: fieldGroup, + fieldNodes: toNodes(fieldGroup), returnType: fieldDef.type, parentType, path, @@ -948,7 +1036,11 @@ async function completePromisedValue( return completed; } catch (rawError) { const errors = asyncPayloadRecord?.errors ?? exeContext.errors; - const error = locatedError(rawError, fieldGroup, pathToArray(path)); + const error = locatedError( + rawError, + toNodes(fieldGroup), + pathToArray(path), + ); const handledError = handleFieldError(error, returnType, errors); filterSubsequentPayloads(exeContext, path, asyncPayloadRecord); return handledError; @@ -978,7 +1070,7 @@ function getStreamValues( // safe to only check the first fieldNode for the stream directive const stream = getDirectiveValues( GraphQLStreamDirective, - fieldGroup[0], + fieldGroup[0].fieldNode, exeContext.variableValues, ); @@ -1058,7 +1150,11 @@ async function completeAsyncIteratorValue( break; } } catch (rawError) { - const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); + const error = locatedError( + rawError, + toNodes(fieldGroup), + pathToArray(itemPath), + ); completedResults.push(handleFieldError(error, itemType, errors)); break; } @@ -1124,6 +1220,7 @@ function completeListValue( // This is specified as a simple map, however we're optimizing the path // where the list contains no Promises by avoiding creating another Promise. let containsPromise = false; + const deferDepth = asyncPayloadRecord?.deferDepth; let previousAsyncPayloadRecord = asyncPayloadRecord; const completedResults: Array = []; let index = 0; @@ -1145,6 +1242,7 @@ function completeListValue( fieldGroup, info, itemType, + deferDepth, previousAsyncPayloadRecord, ); index++; @@ -1223,7 +1321,7 @@ function completeListItemValue( completedItem.then(undefined, (rawError) => { const error = locatedError( rawError, - fieldGroup, + toNodes(fieldGroup), pathToArray(itemPath), ); const handledError = handleFieldError(error, itemType, errors); @@ -1237,7 +1335,11 @@ function completeListItemValue( completedResults.push(completedItem); } catch (rawError) { - const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); + const error = locatedError( + rawError, + toNodes(fieldGroup), + pathToArray(itemPath), + ); const handledError = handleFieldError(error, itemType, errors); filterSubsequentPayloads(exeContext, itemPath, asyncPayloadRecord); completedResults.push(handledError); @@ -1331,7 +1433,7 @@ function ensureValidRuntimeType( if (runtimeTypeName == null) { throw new GraphQLError( `Abstract type "${returnType.name}" must resolve to an Object type at runtime for field "${info.parentType.name}.${info.fieldName}". Either the "${returnType.name}" type should provide a "resolveType" function or each possible type should provide an "isTypeOf" function.`, - { nodes: fieldGroup }, + { nodes: toNodes(fieldGroup) }, ); } @@ -1354,21 +1456,21 @@ function ensureValidRuntimeType( if (runtimeType == null) { throw new GraphQLError( `Abstract type "${returnType.name}" was resolved to a type "${runtimeTypeName}" that does not exist inside the schema.`, - { nodes: fieldGroup }, + { nodes: toNodes(fieldGroup) }, ); } if (!isObjectType(runtimeType)) { throw new GraphQLError( `Abstract type "${returnType.name}" was resolved to a non-object type "${runtimeTypeName}".`, - { nodes: fieldGroup }, + { nodes: toNodes(fieldGroup) }, ); } if (!exeContext.schema.isSubType(returnType, runtimeType)) { throw new GraphQLError( `Runtime Object type "${runtimeType.name}" is not a possible type for "${returnType.name}".`, - { nodes: fieldGroup }, + { nodes: toNodes(fieldGroup) }, ); } @@ -1431,7 +1533,7 @@ function invalidReturnTypeError( ): GraphQLError { return new GraphQLError( `Expected value of type "${returnType.name}" but got: ${inspect(result)}.`, - { nodes: fieldGroup }, + { nodes: toNodes(fieldGroup) }, ); } @@ -1444,7 +1546,7 @@ function collectAndExecuteSubfields( asyncPayloadRecord?: AsyncPayloadRecord, ): PromiseOrValue> { // Collect sub-fields to execute to complete this value. - const { fields: subFieldNodes, patches: subPatches } = collectSubfields( + const { groupedFieldSet, newDeferDepth } = collectSubfields( exeContext, returnType, fieldGroup, @@ -1455,17 +1557,20 @@ function collectAndExecuteSubfields( returnType, result, path, - subFieldNodes, + groupedFieldSet, asyncPayloadRecord, ); - for (const subPatch of subPatches) { - const { fields: subPatchFieldNodes } = subPatch; + if ( + newDeferDepth !== undefined && + shouldBranch(groupedFieldSet, exeContext, path) + ) { executeDeferredFragment( exeContext, returnType, result, - subPatchFieldNodes, + groupedFieldSet, + newDeferDepth, path, asyncPayloadRecord, ); @@ -1687,7 +1792,7 @@ function executeSubscription( ); } - const { fields: groupedFieldSet } = collectFields( + const { groupedFieldSet } = collectFields( schema, fragments, variableValues, @@ -1695,15 +1800,18 @@ function executeSubscription( operation, ); - const firstRootField = groupedFieldSet.entries().next().value; + const firstRootField = groupedFieldSet.entries().next().value as [ + string, + FieldGroup, + ]; const [responseName, fieldGroup] = firstRootField; - const fieldName = fieldGroup[0].name.value; + const fieldName = fieldGroup[0].fieldNode.name.value; const fieldDef = schema.getField(rootType, fieldName); if (!fieldDef) { throw new GraphQLError( `The subscription field "${fieldName}" is not defined.`, - { nodes: fieldGroup }, + { nodes: toNodes(fieldGroup) }, ); } @@ -1722,7 +1830,11 @@ function executeSubscription( // Build a JS object of arguments from the field.arguments AST, using the // variables scope to fulfill any variable references. - const args = getArgumentValues(fieldDef, fieldGroup[0], variableValues); + const args = getArgumentValues( + fieldDef, + fieldGroup[0].fieldNode, + variableValues, + ); // The resolve function's optional third argument is a context value that // is provided to every resolve function within an execution. It is commonly @@ -1736,13 +1848,13 @@ function executeSubscription( if (isPromise(result)) { return result.then(assertEventStream).then(undefined, (error) => { - throw locatedError(error, fieldGroup, pathToArray(path)); + throw locatedError(error, toNodes(fieldGroup), pathToArray(path)); }); } return assertEventStream(result); } catch (error) { - throw locatedError(error, fieldGroup, pathToArray(path)); + throw locatedError(error, toNodes(fieldGroup), pathToArray(path)); } } @@ -1767,10 +1879,12 @@ function executeDeferredFragment( parentType: GraphQLObjectType, sourceValue: unknown, groupedFieldSet: GroupedFieldSet, + newDeferDepth: number, path?: Path, parentContext?: AsyncPayloadRecord, ): void { const asyncPayloadRecord = new DeferredFragmentRecord({ + deferDepth: newDeferDepth, path, parentContext, exeContext, @@ -1807,9 +1921,11 @@ function executeStreamField( fieldGroup: FieldGroup, info: GraphQLResolveInfo, itemType: GraphQLOutputType, + deferDepth: number | undefined, parentContext?: AsyncPayloadRecord, ): AsyncPayloadRecord { const asyncPayloadRecord = new StreamRecord({ + deferDepth, path: itemPath, parentContext, exeContext, @@ -1849,7 +1965,11 @@ function executeStreamField( asyncPayloadRecord, ); } catch (rawError) { - const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); + const error = locatedError( + rawError, + toNodes(fieldGroup), + pathToArray(itemPath), + ); completedItem = handleFieldError( error, itemType, @@ -1867,7 +1987,11 @@ function executeStreamField( if (isPromise(completedItem)) { const completedItems = completedItem .then(undefined, (rawError) => { - const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); + const error = locatedError( + rawError, + toNodes(fieldGroup), + pathToArray(itemPath), + ); const handledError = handleFieldError( error, itemType, @@ -1911,7 +2035,11 @@ async function executeStreamIteratorItem( } item = value; } catch (rawError) { - const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); + const error = locatedError( + rawError, + toNodes(fieldGroup), + pathToArray(itemPath), + ); const value = handleFieldError(error, itemType, asyncPayloadRecord.errors); // don't continue if iterator throws return { done: true, value }; @@ -1930,7 +2058,11 @@ async function executeStreamIteratorItem( if (isPromise(completedItem)) { completedItem = completedItem.then(undefined, (rawError) => { - const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); + const error = locatedError( + rawError, + toNodes(fieldGroup), + pathToArray(itemPath), + ); const handledError = handleFieldError( error, itemType, @@ -1942,7 +2074,11 @@ async function executeStreamIteratorItem( } return { done: false, value: completedItem }; } catch (rawError) { - const error = locatedError(rawError, fieldGroup, pathToArray(itemPath)); + const error = locatedError( + rawError, + toNodes(fieldGroup), + pathToArray(itemPath), + ); const value = handleFieldError(error, itemType, asyncPayloadRecord.errors); filterSubsequentPayloads(exeContext, itemPath, asyncPayloadRecord); return { done: false, value }; @@ -1960,12 +2096,14 @@ async function executeStreamIterator( parentContext?: AsyncPayloadRecord, ): Promise { let index = initialIndex; + const deferDepth = parentContext?.deferDepth; let previousAsyncPayloadRecord = parentContext ?? undefined; // eslint-disable-next-line no-constant-condition while (true) { const itemPath = addPath(path, index, undefined); const asyncPayloadRecord = new StreamRecord({ path: itemPath, + deferDepth, parentContext: previousAsyncPayloadRecord, iterator, exeContext, @@ -2157,6 +2295,7 @@ class DeferredFragmentRecord { type: 'defer'; errors: Array; path: Array; + deferDepth: number | undefined; promise: Promise; data: ObjMap | null; parentContext: AsyncPayloadRecord | undefined; @@ -2165,11 +2304,13 @@ class DeferredFragmentRecord { _resolve?: (arg: PromiseOrValue | null>) => void; constructor(opts: { path: Path | undefined; + deferDepth: number | undefined; parentContext: AsyncPayloadRecord | undefined; exeContext: ExecutionContext; }) { this.type = 'defer'; this.path = pathToArray(opts.path); + this.deferDepth = opts.deferDepth; this.parentContext = opts.parentContext; this.errors = []; this._exeContext = opts.exeContext; @@ -2200,6 +2341,7 @@ class StreamRecord { type: 'stream'; errors: Array; path: Array; + deferDepth: number | undefined; items: Array | null; promise: Promise; parentContext: AsyncPayloadRecord | undefined; @@ -2210,6 +2352,7 @@ class StreamRecord { _resolve?: (arg: PromiseOrValue | null>) => void; constructor(opts: { path: Path | undefined; + deferDepth: number | undefined; iterator?: AsyncIterator; parentContext: AsyncPayloadRecord | undefined; exeContext: ExecutionContext; @@ -2217,6 +2360,7 @@ class StreamRecord { this.type = 'stream'; this.items = null; this.path = pathToArray(opts.path); + this.deferDepth = opts.deferDepth; this.parentContext = opts.parentContext; this.iterator = opts.iterator; this.errors = []; diff --git a/src/validation/rules/SingleFieldSubscriptionsRule.ts b/src/validation/rules/SingleFieldSubscriptionsRule.ts index bc68586e01..9406b119da 100644 --- a/src/validation/rules/SingleFieldSubscriptionsRule.ts +++ b/src/validation/rules/SingleFieldSubscriptionsRule.ts @@ -41,17 +41,19 @@ export function SingleFieldSubscriptionsRule( fragments[definition.name.value] = definition; } } - const { fields } = collectFields( + const { groupedFieldSet } = collectFields( schema, fragments, variableValues, subscriptionType, node, ); - if (fields.size > 1) { - const fieldGroups = [...fields.values()]; + if (groupedFieldSet.size > 1) { + const fieldGroups = [...groupedFieldSet.values()]; const extraFieldGroups = fieldGroups.slice(1); - const extraFields = extraFieldGroups.flat(); + const extraFields = extraFieldGroups + .flat() + .map(({ fieldNode }) => fieldNode); context.reportError( new GraphQLError( operationName != null @@ -61,7 +63,8 @@ export function SingleFieldSubscriptionsRule( ), ); } - for (const fieldNodes of fields.values()) { + for (const fieldSet of groupedFieldSet.values()) { + const fieldNodes = fieldSet.map(({ fieldNode }) => fieldNode); const fieldName = fieldNodes[0].name.value; if (fieldName.startsWith('__')) { context.reportError( From 1236b1f7d4ded34555d45888dcd7977b979b4814 Mon Sep 17 00:00:00 2001 From: Yaacov Rydzinski Date: Sat, 14 Jan 2023 21:13:43 +0200 Subject: [PATCH 4/7] memoize Path generation so paths can be used as keys directly across branches --- src/execution/__tests__/defer-test.ts | 4 +- src/execution/__tests__/executor-test.ts | 35 ++++++++----- src/execution/execute.ts | 28 +++++----- src/jsutils/Path.ts | 65 ++++++++++++++++++++---- src/jsutils/__tests__/Path-test.ts | 18 +++---- src/utilities/coerceInputValue.ts | 13 ++++- 6 files changed, 113 insertions(+), 50 deletions(-) diff --git a/src/execution/__tests__/defer-test.ts b/src/execution/__tests__/defer-test.ts index c8701b3eb0..9f55fbc732 100644 --- a/src/execution/__tests__/defer-test.ts +++ b/src/execution/__tests__/defer-test.ts @@ -591,7 +591,7 @@ describe('Execute: defer directive', () => { ]); }); - it('Can deduplicate initial fields with deferred fragments at multiple levels', async () => { + it('Can deduplicate fields with deferred fragments at multiple levels', async () => { const document = parse(` query { hero { @@ -732,7 +732,7 @@ describe('Execute: defer directive', () => { ]); }); - it('can deduplicate initial fields with deferred fragments in different branches at multiple non-overlapping levels', async () => { + it('can deduplicate fields with deferred fragments in different branches at multiple non-overlapping levels', async () => { const document = parse(` query { a { diff --git a/src/execution/__tests__/executor-test.ts b/src/execution/__tests__/executor-test.ts index 5e25dddb5f..4199600fc1 100644 --- a/src/execution/__tests__/executor-test.ts +++ b/src/execution/__tests__/executor-test.ts @@ -5,10 +5,12 @@ import { expectJSON } from '../../__testUtils__/expectJSON.js'; import { resolveOnNextTick } from '../../__testUtils__/resolveOnNextTick.js'; import { inspect } from '../../jsutils/inspect.js'; +import type { Path } from '../../jsutils/Path.js'; import { Kind } from '../../language/kinds.js'; import { parse } from '../../language/parser.js'; +import type { GraphQLResolveInfo } from '../../type/definition.js'; import { GraphQLInterfaceType, GraphQLList, @@ -191,7 +193,7 @@ describe('Execute: Handles basic execution tasks', () => { }); it('provides info about current execution state', () => { - let resolvedInfo; + let resolvedInfo: GraphQLResolveInfo | undefined; const testType = new GraphQLObjectType({ name: 'Test', fields: { @@ -239,13 +241,18 @@ describe('Execute: Handles basic execution tasks', () => { const field = operation.selectionSet.selections[0]; expect(resolvedInfo).to.deep.include({ fieldNodes: [field], - path: { prev: undefined, key: 'result', typename: 'Test' }, variableValues: { var: 'abc' }, }); + + expect(resolvedInfo?.path).to.deep.include({ + prev: undefined, + key: 'result', + typename: 'Test', + }); }); it('populates path correctly with complex types', () => { - let path; + let path: Path | undefined; const someObject = new GraphQLObjectType({ name: 'SomeObject', fields: { @@ -288,18 +295,20 @@ describe('Execute: Handles basic execution tasks', () => { executeSync({ schema, document, rootValue }); - expect(path).to.deep.equal({ + expect(path).to.deep.include({ key: 'l2', typename: 'SomeObject', - prev: { - key: 0, - typename: undefined, - prev: { - key: 'l1', - typename: 'SomeQuery', - prev: undefined, - }, - }, + }); + + expect(path?.prev).to.deep.include({ + key: 0, + typename: undefined, + }); + + expect(path?.prev?.prev).to.deep.include({ + key: 'l1', + typename: 'SomeQuery', + prev: undefined, }); }); diff --git a/src/execution/execute.ts b/src/execution/execute.ts index 8b9e6b5544..5025a622c1 100644 --- a/src/execution/execute.ts +++ b/src/execution/execute.ts @@ -7,8 +7,8 @@ import { isPromise } from '../jsutils/isPromise.js'; import type { Maybe } from '../jsutils/Maybe.js'; import { memoize3 } from '../jsutils/memoize3.js'; import type { ObjMap } from '../jsutils/ObjMap.js'; -import type { Path } from '../jsutils/Path.js'; -import { addPath, pathToArray } from '../jsutils/Path.js'; +import type { Path, PathFactory } from '../jsutils/Path.js'; +import { createPathFactory, pathToArray } from '../jsutils/Path.js'; import { promiseForObject } from '../jsutils/promiseForObject.js'; import type { PromiseOrValue } from '../jsutils/PromiseOrValue.js'; import { promiseReduce } from '../jsutils/promiseReduce.js'; @@ -122,9 +122,10 @@ export interface ExecutionContext { fieldResolver: GraphQLFieldResolver; typeResolver: GraphQLTypeResolver; subscribeFieldResolver: GraphQLFieldResolver; + addPath: PathFactory; errors: Array; subsequentPayloads: Set; - branches: WeakMap>; + branches: WeakMap>; } /** @@ -502,6 +503,7 @@ export function buildExecutionContext( fieldResolver: fieldResolver ?? defaultFieldResolver, typeResolver: typeResolver ?? defaultTypeResolver, subscribeFieldResolver: subscribeFieldResolver ?? defaultFieldResolver, + addPath: createPathFactory(), subsequentPayloads: new Set(), branches: new WeakMap(), errors: [], @@ -515,6 +517,7 @@ function buildPerEventExecutionContext( return { ...exeContext, rootValue: payload, + addPath: createPathFactory(), subsequentPayloads: new Set(), branches: new WeakMap(), errors: [], @@ -527,13 +530,12 @@ function shouldBranch( path: Path | undefined, ): boolean { const set = exeContext.branches.get(groupedFieldSet); - const key = pathToArray(path).join('.'); if (set === undefined) { - exeContext.branches.set(groupedFieldSet, new Set([key])); + exeContext.branches.set(groupedFieldSet, new Set([path])); return true; } - if (!set.has(key)) { - set.add(key); + if (!set.has(path)) { + set.add(path); return true; } return false; @@ -627,7 +629,7 @@ function executeFieldsSerially( return promiseReduce( groupedFieldSet, (results, [responseName, fieldGroup]) => { - const fieldPath = addPath(path, responseName, parentType.name); + const fieldPath = exeContext.addPath(path, responseName, parentType.name); const fieldName = fieldGroup[0].fieldNode.name.value; const fieldDef = exeContext.schema.getField(parentType, fieldName); @@ -702,7 +704,7 @@ function executeFields( try { for (const [responseName, fieldGroup] of groupedFieldSet) { - const fieldPath = addPath(path, responseName, parentType.name); + const fieldPath = exeContext.addPath(path, responseName, parentType.name); const fieldName = fieldGroup[0].fieldNode.name.value; const fieldDef = exeContext.schema.getField(parentType, fieldName); @@ -1141,7 +1143,7 @@ async function completeAsyncIteratorValue( break; } - const itemPath = addPath(path, index, undefined); + const itemPath = exeContext.addPath(path, index, undefined); let iteration; try { // eslint-disable-next-line no-await-in-loop @@ -1227,7 +1229,7 @@ function completeListValue( for (const item of result) { // No need to modify the info object containing the path, // since from here on it is not ever accessed by resolver functions. - const itemPath = addPath(path, index, undefined); + const itemPath = exeContext.addPath(path, index, undefined); if ( stream && @@ -1815,7 +1817,7 @@ function executeSubscription( ); } - const path = addPath(undefined, responseName, rootType.name); + const path = exeContext.addPath(undefined, responseName, rootType.name); const info = buildResolveInfo( exeContext, fieldDef, @@ -2100,7 +2102,7 @@ async function executeStreamIterator( let previousAsyncPayloadRecord = parentContext ?? undefined; // eslint-disable-next-line no-constant-condition while (true) { - const itemPath = addPath(path, index, undefined); + const itemPath = exeContext.addPath(path, index, undefined); const asyncPayloadRecord = new StreamRecord({ path: itemPath, deferDepth, diff --git a/src/jsutils/Path.ts b/src/jsutils/Path.ts index d223b6e752..2a8eb9d599 100644 --- a/src/jsutils/Path.ts +++ b/src/jsutils/Path.ts @@ -1,20 +1,39 @@ import type { Maybe } from './Maybe.js'; -export interface Path { +/** + * @internal + */ +export class Path { readonly prev: Path | undefined; readonly key: string | number; readonly typename: string | undefined; -} -/** - * Given a Path and a key, return a new Path containing the new key. - */ -export function addPath( - prev: Readonly | undefined, - key: string | number, - typename: string | undefined, -): Path { - return { prev, key, typename }; + readonly _subPaths: Map; + + constructor( + prev: Path | undefined, + key: string | number, + typename: string | undefined, + ) { + this.prev = prev; + this.key = key; + this.typename = typename; + this._subPaths = new Map(); + } + + /** + * Given a Path and a key, return a new Path containing the new key. + */ + addPath(key: string | number, typeName: string | undefined): Path { + let path = this._subPaths.get(key); + if (path !== undefined) { + return path; + } + + path = new Path(this, key, typeName); + this._subPaths.set(key, path); + return path; + } } /** @@ -31,3 +50,27 @@ export function pathToArray( } return flattened.reverse(); } + +export type PathFactory = ( + path: Path | undefined, + key: string | number, + typeName: string | undefined, +) => Path; + +export function createPathFactory(): PathFactory { + const paths = new Map(); + return (path, key, typeName) => { + if (path !== undefined) { + return path.addPath(key, typeName); + } + + let newPath = paths.get(key as string); + if (newPath !== undefined) { + return newPath; + } + + newPath = new Path(undefined, key, typeName); + paths.set(key as string, newPath); + return newPath; + }; +} diff --git a/src/jsutils/__tests__/Path-test.ts b/src/jsutils/__tests__/Path-test.ts index 0484377db9..c671eafc4e 100644 --- a/src/jsutils/__tests__/Path-test.ts +++ b/src/jsutils/__tests__/Path-test.ts @@ -1,13 +1,13 @@ import { expect } from 'chai'; import { describe, it } from 'mocha'; -import { addPath, pathToArray } from '../Path.js'; +import { Path, pathToArray } from '../Path.js'; describe('Path', () => { it('can create a Path', () => { - const first = addPath(undefined, 1, 'First'); + const first = new Path(undefined, 1, 'First'); - expect(first).to.deep.equal({ + expect(first).to.deep.include({ prev: undefined, key: 1, typename: 'First', @@ -15,10 +15,10 @@ describe('Path', () => { }); it('can add a new key to an existing Path', () => { - const first = addPath(undefined, 1, 'First'); - const second = addPath(first, 'two', 'Second'); + const first = new Path(undefined, 1, 'First'); + const second = first.addPath('two', 'Second'); - expect(second).to.deep.equal({ + expect(second).to.deep.include({ prev: first, key: 'two', typename: 'Second', @@ -26,9 +26,9 @@ describe('Path', () => { }); it('can convert a Path to an array of its keys', () => { - const root = addPath(undefined, 0, 'Root'); - const first = addPath(root, 'one', 'First'); - const second = addPath(first, 2, 'Second'); + const root = new Path(undefined, 0, 'Root'); + const first = root.addPath('one', 'First'); + const second = first.addPath(2, 'Second'); const path = pathToArray(second); expect(path).to.deep.equal([0, 'one', 2]); diff --git a/src/utilities/coerceInputValue.ts b/src/utilities/coerceInputValue.ts index d1decf86a1..002f5d1192 100644 --- a/src/utilities/coerceInputValue.ts +++ b/src/utilities/coerceInputValue.ts @@ -3,8 +3,7 @@ import { inspect } from '../jsutils/inspect.js'; import { invariant } from '../jsutils/invariant.js'; import { isIterableObject } from '../jsutils/isIterableObject.js'; import { isObjectLike } from '../jsutils/isObjectLike.js'; -import type { Path } from '../jsutils/Path.js'; -import { addPath, pathToArray } from '../jsutils/Path.js'; +import { Path, pathToArray } from '../jsutils/Path.js'; import { printPathArray } from '../jsutils/printPathArray.js'; import { suggestionList } from '../jsutils/suggestionList.js'; @@ -48,6 +47,16 @@ function defaultOnError( throw error; } +function addPath( + path: Path | undefined, + key: string | number, + typeName: string | undefined, +): Path { + return path + ? path.addPath(key, typeName) + : new Path(undefined, key, typeName); +} + function coerceInputValueImpl( inputValue: unknown, type: GraphQLInputType, From 8bbf2d805c0590c45098b0c50b4b3181943d1520 Mon Sep 17 00:00:00 2001 From: Yaacov Rydzinski Date: Sat, 14 Jan 2023 23:56:26 +0200 Subject: [PATCH 5/7] expose all of the execution environment to resolvers presumably we should do this --- src/execution/__tests__/defer-test.ts | 22 +++++--- src/execution/__tests__/executor-test.ts | 8 +-- src/execution/execute.ts | 68 +++++++----------------- src/type/definition.ts | 7 ++- 4 files changed, 46 insertions(+), 59 deletions(-) diff --git a/src/execution/__tests__/defer-test.ts b/src/execution/__tests__/defer-test.ts index 9f55fbc732..9320945eeb 100644 --- a/src/execution/__tests__/defer-test.ts +++ b/src/execution/__tests__/defer-test.ts @@ -205,6 +205,7 @@ describe('Execute: defer directive', () => { incremental: [ { data: { + id: '1', name: 'Luke', }, path: ['hero'], @@ -409,7 +410,9 @@ describe('Execute: defer directive', () => { { incremental: [ { - data: {}, + data: { + name: 'Luke', + }, path: ['hero'], }, ], @@ -444,7 +447,9 @@ describe('Execute: defer directive', () => { { incremental: [ { - data: {}, + data: { + name: 'Luke', + }, path: ['hero'], }, ], @@ -522,7 +527,7 @@ describe('Execute: defer directive', () => { ]); }); - it('Can deduplicate leaf fields present in the initial payload', async () => { + it('Does not deduplicate leaf fields present in the initial payload', async () => { const document = parse(` query { hero { @@ -580,7 +585,9 @@ describe('Execute: defer directive', () => { }, }, anotherNestedObject: { - deeperObject: {}, + deeperObject: { + foo: 'foo', + }, }, }, path: ['hero'], @@ -591,7 +598,7 @@ describe('Execute: defer directive', () => { ]); }); - it('Can deduplicate fields with deferred fragments at multiple levels', async () => { + it('Does not deduplicate fields with deferred fragments at multiple levels', async () => { const document = parse(` query { hero { @@ -642,6 +649,7 @@ describe('Execute: defer directive', () => { incremental: [ { data: { + foo: 'foo', bar: 'bar', baz: 'baz', bak: 'bak', @@ -651,6 +659,7 @@ describe('Execute: defer directive', () => { { data: { deeperObject: { + foo: 'foo', bar: 'bar', baz: 'baz', }, @@ -661,6 +670,7 @@ describe('Execute: defer directive', () => { data: { nestedObject: { deeperObject: { + foo: 'foo', bar: 'bar', }, }, @@ -732,7 +742,7 @@ describe('Execute: defer directive', () => { ]); }); - it('can deduplicate fields with deferred fragments in different branches at multiple non-overlapping levels', async () => { + it('Can deduplicate fields with deferred fragments in different branches at multiple non-overlapping levels', async () => { const document = parse(` query { a { diff --git a/src/execution/__tests__/executor-test.ts b/src/execution/__tests__/executor-test.ts index 4199600fc1..0ec1de3ff0 100644 --- a/src/execution/__tests__/executor-test.ts +++ b/src/execution/__tests__/executor-test.ts @@ -215,7 +215,8 @@ describe('Execute: Handles basic execution tasks', () => { expect(resolvedInfo).to.have.all.keys( 'fieldName', - 'fieldNodes', + 'fieldGroup', + 'deferDepth', 'returnType', 'parentType', 'path', @@ -238,9 +239,10 @@ describe('Execute: Handles basic execution tasks', () => { operation, }); - const field = operation.selectionSet.selections[0]; + const fieldNode = operation.selectionSet.selections[0]; expect(resolvedInfo).to.deep.include({ - fieldNodes: [field], + fieldGroup: [{ fieldNode, depth: 0, deferDepth: undefined }], + deferDepth: undefined, variableValues: { var: 'abc' }, }); diff --git a/src/execution/execute.ts b/src/execution/execute.ts index 5025a622c1..38a73d1328 100644 --- a/src/execution/execute.ts +++ b/src/execution/execute.ts @@ -38,7 +38,6 @@ import type { GraphQLTypeResolver, } from '../type/definition.js'; import { - getNamedType, isAbstractType, isLeafType, isListType, @@ -631,26 +630,19 @@ function executeFieldsSerially( (results, [responseName, fieldGroup]) => { const fieldPath = exeContext.addPath(path, responseName, parentType.name); - const fieldName = fieldGroup[0].fieldNode.name.value; - const fieldDef = exeContext.schema.getField(parentType, fieldName); - if (!fieldDef) { - return results; - } - - const returnType = fieldDef.type; - - if (!shouldExecute(fieldGroup, returnType)) { + if (!shouldExecute(fieldGroup)) { return results; } const result = executeField( exeContext, parentType, - fieldDef, - returnType, sourceValue, fieldGroup, fieldPath, ); + if (result === undefined) { + return results; + } if (isPromise(result)) { return result.then((resolvedResult) => { results[responseName] = resolvedResult; @@ -666,25 +658,11 @@ function executeFieldsSerially( function shouldExecute( fieldGroup: FieldGroup, - returnType: GraphQLOutputType, deferDepth?: number | undefined, ): boolean { - if (deferDepth === undefined || !isLeafType(getNamedType(returnType))) { - return fieldGroup.some( - ({ deferDepth: fieldDeferDepth }) => fieldDeferDepth === deferDepth, - ); - } - - let hasDepth = false; - for (const { deferDepth: fieldDeferDepth } of fieldGroup) { - if (fieldDeferDepth === undefined) { - return false; - } - if (fieldDeferDepth === deferDepth) { - hasDepth = true; - } - } - return hasDepth; + return fieldGroup.some( + ({ deferDepth: fieldDeferDepth }) => fieldDeferDepth === deferDepth, + ); } /** @@ -706,22 +684,10 @@ function executeFields( for (const [responseName, fieldGroup] of groupedFieldSet) { const fieldPath = exeContext.addPath(path, responseName, parentType.name); - const fieldName = fieldGroup[0].fieldNode.name.value; - const fieldDef = exeContext.schema.getField(parentType, fieldName); - if (!fieldDef) { - continue; - } - - const returnType = fieldDef.type; - - if ( - shouldExecute(fieldGroup, returnType, asyncPayloadRecord?.deferDepth) - ) { + if (shouldExecute(fieldGroup, asyncPayloadRecord?.deferDepth)) { const result = executeField( exeContext, parentType, - fieldDef, - returnType, sourceValue, fieldGroup, fieldPath, @@ -769,15 +735,19 @@ function toNodes(fieldGroup: FieldGroup): ReadonlyArray { function executeField( exeContext: ExecutionContext, parentType: GraphQLObjectType, - fieldDef: GraphQLField, - returnType: GraphQLOutputType, source: unknown, fieldGroup: FieldGroup, path: Path, asyncPayloadRecord?: AsyncPayloadRecord, ): PromiseOrValue { const errors = asyncPayloadRecord?.errors ?? exeContext.errors; + const fieldName = fieldGroup[0].fieldNode.name.value; + const fieldDef = exeContext.schema.getField(parentType, fieldName); + if (!fieldDef) { + return; + } + const returnType = fieldDef.type; const resolveFn = fieldDef.resolve ?? exeContext.fieldResolver; const info = buildResolveInfo( @@ -786,6 +756,7 @@ function executeField( fieldGroup, parentType, path, + asyncPayloadRecord, ); // Get the resolve function, regardless of if its result is normal or abrupt (error). @@ -865,12 +836,14 @@ export function buildResolveInfo( fieldGroup: FieldGroup, parentType: GraphQLObjectType, path: Path, + asyncPayloadRecord?: AsyncPayloadRecord | undefined, ): GraphQLResolveInfo { // The resolve function's optional fourth argument is a collection of // information about the current execution state. return { fieldName: fieldDef.name, - fieldNodes: toNodes(fieldGroup), + fieldGroup, + deferDepth: asyncPayloadRecord?.deferDepth, returnType: fieldDef.type, parentType, path, @@ -1222,7 +1195,6 @@ function completeListValue( // This is specified as a simple map, however we're optimizing the path // where the list contains no Promises by avoiding creating another Promise. let containsPromise = false; - const deferDepth = asyncPayloadRecord?.deferDepth; let previousAsyncPayloadRecord = asyncPayloadRecord; const completedResults: Array = []; let index = 0; @@ -1244,7 +1216,6 @@ function completeListValue( fieldGroup, info, itemType, - deferDepth, previousAsyncPayloadRecord, ); index++; @@ -1923,11 +1894,10 @@ function executeStreamField( fieldGroup: FieldGroup, info: GraphQLResolveInfo, itemType: GraphQLOutputType, - deferDepth: number | undefined, parentContext?: AsyncPayloadRecord, ): AsyncPayloadRecord { const asyncPayloadRecord = new StreamRecord({ - deferDepth, + deferDepth: parentContext?.deferDepth, path: itemPath, parentContext, exeContext, diff --git a/src/type/definition.ts b/src/type/definition.ts index 81488efb39..83208b18d3 100644 --- a/src/type/definition.ts +++ b/src/type/definition.ts @@ -890,7 +890,12 @@ export type GraphQLFieldResolver< export interface GraphQLResolveInfo { readonly fieldName: string; - readonly fieldNodes: ReadonlyArray; + readonly fieldGroup: ReadonlyArray<{ + fieldNode: FieldNode; + depth: number; + deferDepth: number | undefined; + }>; + readonly deferDepth: number | undefined; readonly returnType: GraphQLOutputType; readonly parentType: GraphQLObjectType; readonly path: Path; From 54d2684ab846a7874e1d1cbd208869f56b5a6e08 Mon Sep 17 00:00:00 2001 From: Yaacov Rydzinski Date: Sun, 15 Jan 2023 22:01:42 +0200 Subject: [PATCH 6/7] deduplicate leaf fields from initial payload --- src/execution/__tests__/defer-test.ts | 20 +++------ src/execution/execute.ts | 58 ++++++++++++++++++++------- 2 files changed, 49 insertions(+), 29 deletions(-) diff --git a/src/execution/__tests__/defer-test.ts b/src/execution/__tests__/defer-test.ts index 9320945eeb..19d49cd95e 100644 --- a/src/execution/__tests__/defer-test.ts +++ b/src/execution/__tests__/defer-test.ts @@ -205,7 +205,6 @@ describe('Execute: defer directive', () => { incremental: [ { data: { - id: '1', name: 'Luke', }, path: ['hero'], @@ -410,9 +409,7 @@ describe('Execute: defer directive', () => { { incremental: [ { - data: { - name: 'Luke', - }, + data: {}, path: ['hero'], }, ], @@ -447,9 +444,7 @@ describe('Execute: defer directive', () => { { incremental: [ { - data: { - name: 'Luke', - }, + data: {}, path: ['hero'], }, ], @@ -527,7 +522,7 @@ describe('Execute: defer directive', () => { ]); }); - it('Does not deduplicate leaf fields present in the initial payload', async () => { + it('Can deduplicate leaf fields present in the initial payload', async () => { const document = parse(` query { hero { @@ -585,9 +580,7 @@ describe('Execute: defer directive', () => { }, }, anotherNestedObject: { - deeperObject: { - foo: 'foo', - }, + deeperObject: {}, }, }, path: ['hero'], @@ -598,7 +591,7 @@ describe('Execute: defer directive', () => { ]); }); - it('Does not deduplicate fields with deferred fragments at multiple levels', async () => { + it('Can deduplicate fields with deferred fragments at multiple levels', async () => { const document = parse(` query { hero { @@ -649,7 +642,6 @@ describe('Execute: defer directive', () => { incremental: [ { data: { - foo: 'foo', bar: 'bar', baz: 'baz', bak: 'bak', @@ -659,7 +651,6 @@ describe('Execute: defer directive', () => { { data: { deeperObject: { - foo: 'foo', bar: 'bar', baz: 'baz', }, @@ -670,7 +661,6 @@ describe('Execute: defer directive', () => { data: { nestedObject: { deeperObject: { - foo: 'foo', bar: 'bar', }, }, diff --git a/src/execution/execute.ts b/src/execution/execute.ts index 38a73d1328..72b32260a3 100644 --- a/src/execution/execute.ts +++ b/src/execution/execute.ts @@ -38,6 +38,7 @@ import type { GraphQLTypeResolver, } from '../type/definition.js'; import { + getNamedType, isAbstractType, isLeafType, isListType, @@ -630,19 +631,26 @@ function executeFieldsSerially( (results, [responseName, fieldGroup]) => { const fieldPath = exeContext.addPath(path, responseName, parentType.name); - if (!shouldExecute(fieldGroup)) { + const fieldName = fieldGroup[0].fieldNode.name.value; + const fieldDef = exeContext.schema.getField(parentType, fieldName); + if (!fieldDef) { + return results; + } + + const returnType = fieldDef.type; + + if (!shouldExecute(fieldGroup, returnType)) { return results; } const result = executeField( exeContext, parentType, + fieldDef, + returnType, sourceValue, fieldGroup, fieldPath, ); - if (result === undefined) { - return results; - } if (isPromise(result)) { return result.then((resolvedResult) => { results[responseName] = resolvedResult; @@ -658,11 +666,25 @@ function executeFieldsSerially( function shouldExecute( fieldGroup: FieldGroup, + returnType: GraphQLOutputType, deferDepth?: number | undefined, ): boolean { - return fieldGroup.some( - ({ deferDepth: fieldDeferDepth }) => fieldDeferDepth === deferDepth, - ); + if (deferDepth === undefined || !isLeafType(getNamedType(returnType))) { + return fieldGroup.some( + ({ deferDepth: fieldDeferDepth }) => fieldDeferDepth === deferDepth, + ); + } + + let hasDepth = false; + for (const { deferDepth: fieldDeferDepth } of fieldGroup) { + if (fieldDeferDepth === undefined) { + return false; + } + if (fieldDeferDepth === deferDepth) { + hasDepth = true; + } + } + return hasDepth; } /** @@ -684,10 +706,22 @@ function executeFields( for (const [responseName, fieldGroup] of groupedFieldSet) { const fieldPath = exeContext.addPath(path, responseName, parentType.name); - if (shouldExecute(fieldGroup, asyncPayloadRecord?.deferDepth)) { + const fieldName = fieldGroup[0].fieldNode.name.value; + const fieldDef = exeContext.schema.getField(parentType, fieldName); + if (!fieldDef) { + continue; + } + + const returnType = fieldDef.type; + + if ( + shouldExecute(fieldGroup, returnType, asyncPayloadRecord?.deferDepth) + ) { const result = executeField( exeContext, parentType, + fieldDef, + returnType, sourceValue, fieldGroup, fieldPath, @@ -735,19 +769,15 @@ function toNodes(fieldGroup: FieldGroup): ReadonlyArray { function executeField( exeContext: ExecutionContext, parentType: GraphQLObjectType, + fieldDef: GraphQLField, + returnType: GraphQLOutputType, source: unknown, fieldGroup: FieldGroup, path: Path, asyncPayloadRecord?: AsyncPayloadRecord, ): PromiseOrValue { const errors = asyncPayloadRecord?.errors ?? exeContext.errors; - const fieldName = fieldGroup[0].fieldNode.name.value; - const fieldDef = exeContext.schema.getField(parentType, fieldName); - if (!fieldDef) { - return; - } - const returnType = fieldDef.type; const resolveFn = fieldDef.resolve ?? exeContext.fieldResolver; const info = buildResolveInfo( From 95198f911bd10dc8f95dcdc81b2a8d55c839b48e Mon Sep 17 00:00:00 2001 From: Yaacov Rydzinski Date: Sat, 14 Jan 2023 21:15:50 +0200 Subject: [PATCH 7/7] deduplicate (mask) leaf values that have already been sent causes some additional minor changes to `hasNext` property on some payloads due to modification of returned promises resulting in an extra tick --- src/execution/__tests__/defer-test.ts | 13 ++---- src/execution/__tests__/stream-test.ts | 6 --- src/execution/execute.ts | 55 ++++++++++++++++++++++---- 3 files changed, 50 insertions(+), 24 deletions(-) diff --git a/src/execution/__tests__/defer-test.ts b/src/execution/__tests__/defer-test.ts index 19d49cd95e..6b26a37e68 100644 --- a/src/execution/__tests__/defer-test.ts +++ b/src/execution/__tests__/defer-test.ts @@ -650,19 +650,14 @@ describe('Execute: defer directive', () => { }, { data: { - deeperObject: { - bar: 'bar', - baz: 'baz', - }, + deeperObject: {}, }, path: ['hero', 'nestedObject'], }, { data: { nestedObject: { - deeperObject: { - bar: 'bar', - }, + deeperObject: {}, }, }, path: ['hero'], @@ -789,9 +784,7 @@ describe('Execute: defer directive', () => { data: { a: { b: { - e: { - f: 'f', - }, + e: {}, }, }, g: { diff --git a/src/execution/__tests__/stream-test.ts b/src/execution/__tests__/stream-test.ts index 7a168269c9..2d8c56cc19 100644 --- a/src/execution/__tests__/stream-test.ts +++ b/src/execution/__tests__/stream-test.ts @@ -1170,9 +1170,6 @@ describe('Execute: stream directive', () => { ], }, ], - hasNext: true, - }, - { hasNext: false, }, ]); @@ -1355,9 +1352,6 @@ describe('Execute: stream directive', () => { ], }, ], - hasNext: true, - }, - { hasNext: false, }, ]); diff --git a/src/execution/execute.ts b/src/execution/execute.ts index 72b32260a3..8b8c5b1dfe 100644 --- a/src/execution/execute.ts +++ b/src/execution/execute.ts @@ -126,6 +126,7 @@ export interface ExecutionContext { errors: Array; subsequentPayloads: Set; branches: WeakMap>; + leaves: WeakMap>; } /** @@ -506,6 +507,7 @@ export function buildExecutionContext( addPath: createPathFactory(), subsequentPayloads: new Set(), branches: new WeakMap(), + leaves: new WeakMap(), errors: [], }; } @@ -520,6 +522,7 @@ function buildPerEventExecutionContext( addPath: createPathFactory(), subsequentPayloads: new Set(), branches: new WeakMap(), + leaves: new WeakMap(), errors: [], }; } @@ -639,7 +642,9 @@ function executeFieldsSerially( const returnType = fieldDef.type; - if (!shouldExecute(fieldGroup, returnType)) { + const isLeaf = isLeafType(getNamedType(returnType)); + + if (!shouldExecute(fieldGroup, isLeaf)) { return results; } const result = executeField( @@ -666,10 +671,10 @@ function executeFieldsSerially( function shouldExecute( fieldGroup: FieldGroup, - returnType: GraphQLOutputType, + isLeaf: boolean, deferDepth?: number | undefined, ): boolean { - if (deferDepth === undefined || !isLeafType(getNamedType(returnType))) { + if (deferDepth === undefined || !isLeaf) { return fieldGroup.some( ({ deferDepth: fieldDeferDepth }) => fieldDeferDepth === deferDepth, ); @@ -702,6 +707,8 @@ function executeFields( const results = Object.create(null); let containsPromise = false; + const shouldMask = Object.create(null); + try { for (const [responseName, fieldGroup] of groupedFieldSet) { const fieldPath = exeContext.addPath(path, responseName, parentType.name); @@ -714,9 +721,27 @@ function executeFields( const returnType = fieldDef.type; - if ( - shouldExecute(fieldGroup, returnType, asyncPayloadRecord?.deferDepth) - ) { + const isLeaf = isLeafType(getNamedType(returnType)); + + if (shouldExecute(fieldGroup, isLeaf, asyncPayloadRecord?.deferDepth)) { + if ( + asyncPayloadRecord !== undefined && + isLeafType(getNamedType(returnType)) + ) { + shouldMask[responseName] = () => { + const set = exeContext.leaves.get(groupedFieldSet); + if (set === undefined) { + exeContext.leaves.set(groupedFieldSet, new Set([fieldPath])); + return false; + } + if (set.has(fieldPath)) { + return true; + } + set.add(fieldPath); + return false; + }; + } + const result = executeField( exeContext, parentType, @@ -748,13 +773,27 @@ function executeFields( // If there are no promises, we can just return the object if (!containsPromise) { - return results; + return asyncPayloadRecord === undefined + ? results + : new Proxy(results, { + ownKeys: (target) => + Reflect.ownKeys(target).filter((key) => !shouldMask[key]?.()), + }); } // Otherwise, results is a map from field name to the result of resolving that // field, which is possibly a promise. Return a promise that will return this // same map, but with any promises replaced with the values they resolved to. - return promiseForObject(results); + const promisedResult = promiseForObject(results); + return asyncPayloadRecord === undefined + ? promisedResult + : promisedResult.then( + (resolved) => + new Proxy(resolved, { + ownKeys: (target) => + Reflect.ownKeys(target).filter((key) => !shouldMask[key]?.()), + }), + ); } function toNodes(fieldGroup: FieldGroup): ReadonlyArray {