From cc66527bea1b334634d9410a894f6b9f39eea0bb Mon Sep 17 00:00:00 2001 From: Ivan Goncharov Date: Mon, 13 Mar 2017 16:43:37 +0200 Subject: [PATCH] Add support for appliedDirectives in extendSchema --- src/utilities/__tests__/extendSchema-test.js | 15 ++----- src/utilities/extendSchema.js | 44 +++++++++++++++++++- 2 files changed, 46 insertions(+), 13 deletions(-) diff --git a/src/utilities/__tests__/extendSchema-test.js b/src/utilities/__tests__/extendSchema-test.js index 7064e9ac18d..7f425652fca 100644 --- a/src/utilities/__tests__/extendSchema-test.js +++ b/src/utilities/__tests__/extendSchema-test.js @@ -213,17 +213,10 @@ union SomeUnion = Foo | Biz expect(deprecatedFieldDef.isDeprecated).to.equal(true); expect(deprecatedFieldDef.deprecationReason).to.equal('not used anymore'); - const deprecatedEnumDef = extendedSchema - .getType('EnumWithDeprecatedValue'); - expect(deprecatedEnumDef.getValues()).to.deep.equal([ - { - name: 'DEPRECATED', - description: '', - isDeprecated: true, - deprecationReason: 'do not use', - value: 'DEPRECATED' - } - ]); + const deprecatedEnumValueDef = extendedSchema + .getType('EnumWithDeprecatedValue').getValues()[0]; + expect(deprecatedEnumValueDef.isDeprecated).to.equal(true); + expect(deprecatedEnumValueDef.deprecationReason).to.equal('do not use'); }); it('extends objects with deprecated fields', () => { diff --git a/src/utilities/extendSchema.js b/src/utilities/extendSchema.js index 77c45b383a4..d05a92624e0 100644 --- a/src/utilities/extendSchema.js +++ b/src/utilities/extendSchema.js @@ -28,6 +28,7 @@ import { GraphQLScalarType, GraphQLEnumType, GraphQLInputObjectType, + GraphQLAppliedDirectives, isInputType, isOutputType, } from '../type/definition'; @@ -82,6 +83,7 @@ import type { import type { DocumentNode, + DirectiveNode, InputValueDefinitionNode, TypeNode, NamedTypeNode, @@ -95,6 +97,7 @@ import type { DirectiveDefinitionNode, } from '../language/ast'; +import { getArgumentValues } from '../execution/values'; /** * Produces a new schema given an existing schema and a document which may @@ -245,15 +248,38 @@ export function extendSchema( types.push(getTypeFromAST(typeDefinitionMap[typeName])); }); + const directives = getMergedDirectives(); + const innerDirectivesMap = keyValMap( + directives, + directive => directive.name, + directive => directive + ); + // Then produce and return a Schema with these types. return new GraphQLSchema({ query: queryType, mutation: mutationType, subscription: subscriptionType, types, - directives: getMergedDirectives(), + directives, + appliedDirectives: schema.getAppliedDirectives(), }); + function makeAppliedDirectives(appliedDirectives: ?Array) { + return appliedDirectives && new GraphQLAppliedDirectives(keyValMap( + appliedDirectives, + directive => directive.name.value, + directive => (() => { + const directiveName = directive.name.value; + if (!innerDirectivesMap[directiveName]) { + throw new Error( + `Directive "${directiveName}" not found in document.`); + } + return getArgumentValues(innerDirectivesMap[directiveName], directive); + }) + )); + } + // Below are functions used for producing this schema that have closed over // this scope and have access to the schema, cache, and newly defined types. @@ -353,6 +379,7 @@ export function extendSchema( description: type.description, interfaces: () => extendImplementedInterfaces(type), fields: () => extendFieldMap(type), + appliedDirectives: type.appliedDirectives, isTypeOf: type.isTypeOf, }); } @@ -364,6 +391,7 @@ export function extendSchema( name: type.name, description: type.description, fields: () => extendFieldMap(type), + appliedDirectives: type.appliedDirectives, resolveType: type.resolveType, }); } @@ -373,6 +401,7 @@ export function extendSchema( name: type.name, description: type.description, types: type.getTypes().map(getTypeFromDef), + appliedDirectives: type.appliedDirectives, resolveType: type.resolveType, }); } @@ -413,6 +442,7 @@ export function extendSchema( deprecationReason: field.deprecationReason, type: extendFieldType(field.type), args: keyMap(field.args, arg => arg.name), + appliedDirectives: field.appliedDirectives, resolve: field.resolve, }; }); @@ -435,6 +465,7 @@ export function extendSchema( type: buildOutputFieldType(field.type), args: buildInputValues(field.arguments), deprecationReason: getDeprecationReason(field.directives), + appliedDirectives: makeAppliedDirectives(field.directives), }; }); }); @@ -471,6 +502,7 @@ export function extendSchema( description: getDescription(typeNode), interfaces: () => buildImplementedInterfaces(typeNode), fields: () => buildFieldMap(typeNode), + appliedDirectives: makeAppliedDirectives(typeNode.directives), }); } @@ -479,6 +511,7 @@ export function extendSchema( name: typeNode.name.value, description: getDescription(typeNode), fields: () => buildFieldMap(typeNode), + appliedDirectives: makeAppliedDirectives(typeNode.directives), resolveType: cannotExecuteExtendedSchema, }); } @@ -488,6 +521,7 @@ export function extendSchema( name: typeNode.name.value, description: getDescription(typeNode), types: typeNode.types.map(getObjectTypeFromAST), + appliedDirectives: makeAppliedDirectives(typeNode.directives), resolveType: cannotExecuteExtendedSchema, }); } @@ -496,6 +530,7 @@ export function extendSchema( return new GraphQLScalarType({ name: typeNode.name.value, description: getDescription(typeNode), + appliedDirectives: makeAppliedDirectives(typeNode.directives), serialize: id => id, // Note: validation calls the parse functions to determine if a // literal value is correct. Returning null would cause use of custom @@ -516,8 +551,10 @@ export function extendSchema( enumValue => ({ description: getDescription(enumValue), deprecationReason: getDeprecationReason(enumValue.directives), + appliedDirectives: makeAppliedDirectives(enumValue.directives), }), ), + appliedDirectives: makeAppliedDirectives(typeNode.directives), }); } @@ -526,6 +563,7 @@ export function extendSchema( name: typeNode.name.value, description: getDescription(typeNode), fields: () => buildInputValues(typeNode.fields), + appliedDirectives: makeAppliedDirectives(typeNode.directives), }); } @@ -556,6 +594,7 @@ export function extendSchema( description: getDescription(field), args: buildInputValues(field.arguments), deprecationReason: getDeprecationReason(field.directives), + appliedDirectives: makeAppliedDirectives(field.directives), }) ); } @@ -569,7 +608,8 @@ export function extendSchema( return { type, description: getDescription(value), - defaultValue: valueFromAST(value.defaultValue, type) + defaultValue: valueFromAST(value.defaultValue, type), + appliedDirectives: makeAppliedDirectives(value.directives), }; } );