Initial Save
This commit is contained in:
491
node_modules/graphql/utilities/buildASTSchema.js.flow
generated
vendored
Normal file
491
node_modules/graphql/utilities/buildASTSchema.js.flow
generated
vendored
Normal file
@@ -0,0 +1,491 @@
|
||||
// @flow strict
|
||||
|
||||
import objectValues from '../polyfills/objectValues';
|
||||
|
||||
import keyMap from '../jsutils/keyMap';
|
||||
import inspect from '../jsutils/inspect';
|
||||
import invariant from '../jsutils/invariant';
|
||||
import devAssert from '../jsutils/devAssert';
|
||||
import keyValMap from '../jsutils/keyValMap';
|
||||
import { type ObjMap } from '../jsutils/ObjMap';
|
||||
|
||||
import { Kind } from '../language/kinds';
|
||||
import { type Source } from '../language/source';
|
||||
import { TokenKind } from '../language/tokenKind';
|
||||
import { type ParseOptions, parse } from '../language/parser';
|
||||
import { isTypeDefinitionNode } from '../language/predicates';
|
||||
import { dedentBlockStringValue } from '../language/blockString';
|
||||
import { type DirectiveLocationEnum } from '../language/directiveLocation';
|
||||
import {
|
||||
type DocumentNode,
|
||||
type NameNode,
|
||||
type TypeNode,
|
||||
type NamedTypeNode,
|
||||
type SchemaDefinitionNode,
|
||||
type TypeDefinitionNode,
|
||||
type ScalarTypeDefinitionNode,
|
||||
type ObjectTypeDefinitionNode,
|
||||
type FieldDefinitionNode,
|
||||
type InputValueDefinitionNode,
|
||||
type InterfaceTypeDefinitionNode,
|
||||
type UnionTypeDefinitionNode,
|
||||
type EnumTypeDefinitionNode,
|
||||
type EnumValueDefinitionNode,
|
||||
type InputObjectTypeDefinitionNode,
|
||||
type DirectiveDefinitionNode,
|
||||
type StringValueNode,
|
||||
type Location,
|
||||
} from '../language/ast';
|
||||
|
||||
import { assertValidSDL } from '../validation/validate';
|
||||
|
||||
import { getDirectiveValues } from '../execution/values';
|
||||
|
||||
import { specifiedScalarTypes } from '../type/scalars';
|
||||
import { introspectionTypes } from '../type/introspection';
|
||||
import {
|
||||
type GraphQLSchemaValidationOptions,
|
||||
GraphQLSchema,
|
||||
} from '../type/schema';
|
||||
import {
|
||||
GraphQLDirective,
|
||||
GraphQLSkipDirective,
|
||||
GraphQLIncludeDirective,
|
||||
GraphQLDeprecatedDirective,
|
||||
} from '../type/directives';
|
||||
import {
|
||||
type GraphQLType,
|
||||
type GraphQLNamedType,
|
||||
type GraphQLFieldConfig,
|
||||
type GraphQLArgumentConfig,
|
||||
type GraphQLEnumValueConfig,
|
||||
type GraphQLInputFieldConfig,
|
||||
GraphQLScalarType,
|
||||
GraphQLObjectType,
|
||||
GraphQLInterfaceType,
|
||||
GraphQLUnionType,
|
||||
GraphQLEnumType,
|
||||
GraphQLInputObjectType,
|
||||
GraphQLList,
|
||||
GraphQLNonNull,
|
||||
} from '../type/definition';
|
||||
|
||||
import { valueFromAST } from './valueFromAST';
|
||||
|
||||
export type BuildSchemaOptions = {
|
||||
...GraphQLSchemaValidationOptions,
|
||||
|
||||
/**
|
||||
* Descriptions are defined as preceding string literals, however an older
|
||||
* experimental version of the SDL supported preceding comments as
|
||||
* descriptions. Set to true to enable this deprecated behavior.
|
||||
* This option is provided to ease adoption and will be removed in v16.
|
||||
*
|
||||
* Default: false
|
||||
*/
|
||||
commentDescriptions?: boolean,
|
||||
|
||||
/**
|
||||
* Set to true to assume the SDL is valid.
|
||||
*
|
||||
* Default: false
|
||||
*/
|
||||
assumeValidSDL?: boolean,
|
||||
|
||||
...
|
||||
};
|
||||
|
||||
/**
|
||||
* This takes the ast of a schema document produced by the parse function in
|
||||
* src/language/parser.js.
|
||||
*
|
||||
* If no schema definition is provided, then it will look for types named Query
|
||||
* and Mutation.
|
||||
*
|
||||
* Given that AST it constructs a GraphQLSchema. The resulting schema
|
||||
* has no resolve methods, so execution will use default resolvers.
|
||||
*
|
||||
* Accepts options as a second argument:
|
||||
*
|
||||
* - commentDescriptions:
|
||||
* Provide true to use preceding comments as the description.
|
||||
*
|
||||
*/
|
||||
export function buildASTSchema(
|
||||
documentAST: DocumentNode,
|
||||
options?: BuildSchemaOptions,
|
||||
): GraphQLSchema {
|
||||
devAssert(
|
||||
documentAST && documentAST.kind === Kind.DOCUMENT,
|
||||
'Must provide valid Document AST',
|
||||
);
|
||||
|
||||
if (!options || !(options.assumeValid || options.assumeValidSDL)) {
|
||||
assertValidSDL(documentAST);
|
||||
}
|
||||
|
||||
let schemaDef: ?SchemaDefinitionNode;
|
||||
const typeDefs: Array<TypeDefinitionNode> = [];
|
||||
const directiveDefs: Array<DirectiveDefinitionNode> = [];
|
||||
|
||||
for (const def of documentAST.definitions) {
|
||||
if (def.kind === Kind.SCHEMA_DEFINITION) {
|
||||
schemaDef = def;
|
||||
} else if (isTypeDefinitionNode(def)) {
|
||||
typeDefs.push(def);
|
||||
} else if (def.kind === Kind.DIRECTIVE_DEFINITION) {
|
||||
directiveDefs.push(def);
|
||||
}
|
||||
}
|
||||
|
||||
const astBuilder = new ASTDefinitionBuilder(options, typeName => {
|
||||
const type = typeMap[typeName];
|
||||
if (type === undefined) {
|
||||
throw new Error(`Type "${typeName}" not found in document.`);
|
||||
}
|
||||
return type;
|
||||
});
|
||||
|
||||
const typeMap = keyByNameNode(typeDefs, node => astBuilder.buildType(node));
|
||||
|
||||
const operationTypes = schemaDef
|
||||
? getOperationTypes(schemaDef)
|
||||
: {
|
||||
query: 'Query',
|
||||
mutation: 'Mutation',
|
||||
subscription: 'Subscription',
|
||||
};
|
||||
|
||||
const directives = directiveDefs.map(def => astBuilder.buildDirective(def));
|
||||
|
||||
// If specified directives were not explicitly declared, add them.
|
||||
if (!directives.some(directive => directive.name === 'skip')) {
|
||||
directives.push(GraphQLSkipDirective);
|
||||
}
|
||||
|
||||
if (!directives.some(directive => directive.name === 'include')) {
|
||||
directives.push(GraphQLIncludeDirective);
|
||||
}
|
||||
|
||||
if (!directives.some(directive => directive.name === 'deprecated')) {
|
||||
directives.push(GraphQLDeprecatedDirective);
|
||||
}
|
||||
|
||||
return new GraphQLSchema({
|
||||
// Note: While this could make early assertions to get the correctly
|
||||
// typed values below, that would throw immediately while type system
|
||||
// validation with validateSchema() will produce more actionable results.
|
||||
query: operationTypes.query ? (typeMap[operationTypes.query]: any) : null,
|
||||
mutation: operationTypes.mutation
|
||||
? (typeMap[operationTypes.mutation]: any)
|
||||
: null,
|
||||
subscription: operationTypes.subscription
|
||||
? (typeMap[operationTypes.subscription]: any)
|
||||
: null,
|
||||
|
||||
types: objectValues(typeMap),
|
||||
directives,
|
||||
astNode: schemaDef,
|
||||
assumeValid: options && options.assumeValid,
|
||||
allowedLegacyNames: options && options.allowedLegacyNames,
|
||||
});
|
||||
|
||||
function getOperationTypes(schema: SchemaDefinitionNode) {
|
||||
const opTypes = {};
|
||||
for (const operationType of schema.operationTypes) {
|
||||
opTypes[operationType.operation] = operationType.type.name.value;
|
||||
}
|
||||
return opTypes;
|
||||
}
|
||||
}
|
||||
|
||||
type TypeResolver = (typeName: string) => GraphQLNamedType;
|
||||
|
||||
const stdTypeMap = keyMap(
|
||||
specifiedScalarTypes.concat(introspectionTypes),
|
||||
type => type.name,
|
||||
);
|
||||
|
||||
export class ASTDefinitionBuilder {
|
||||
_options: ?BuildSchemaOptions;
|
||||
_resolveType: TypeResolver;
|
||||
|
||||
constructor(options: ?BuildSchemaOptions, resolveType: TypeResolver) {
|
||||
this._options = options;
|
||||
this._resolveType = resolveType;
|
||||
}
|
||||
|
||||
getNamedType(node: NamedTypeNode): GraphQLNamedType {
|
||||
const name = node.name.value;
|
||||
return stdTypeMap[name] || this._resolveType(name);
|
||||
}
|
||||
|
||||
getWrappedType(node: TypeNode): GraphQLType {
|
||||
if (node.kind === Kind.LIST_TYPE) {
|
||||
return new GraphQLList(this.getWrappedType(node.type));
|
||||
}
|
||||
if (node.kind === Kind.NON_NULL_TYPE) {
|
||||
return new GraphQLNonNull(this.getWrappedType(node.type));
|
||||
}
|
||||
return this.getNamedType(node);
|
||||
}
|
||||
|
||||
buildDirective(directive: DirectiveDefinitionNode): GraphQLDirective {
|
||||
const locations = directive.locations.map(
|
||||
({ value }) => ((value: any): DirectiveLocationEnum),
|
||||
);
|
||||
|
||||
return new GraphQLDirective({
|
||||
name: directive.name.value,
|
||||
description: getDescription(directive, this._options),
|
||||
locations,
|
||||
isRepeatable: directive.repeatable,
|
||||
args: keyByNameNode(directive.arguments || [], arg => this.buildArg(arg)),
|
||||
astNode: directive,
|
||||
});
|
||||
}
|
||||
|
||||
buildField(field: FieldDefinitionNode): GraphQLFieldConfig<mixed, mixed> {
|
||||
return {
|
||||
// Note: While this could make assertions to get the correctly typed
|
||||
// value, that would throw immediately while type system validation
|
||||
// with validateSchema() will produce more actionable results.
|
||||
type: (this.getWrappedType(field.type): any),
|
||||
description: getDescription(field, this._options),
|
||||
args: keyByNameNode(field.arguments || [], arg => this.buildArg(arg)),
|
||||
deprecationReason: getDeprecationReason(field),
|
||||
astNode: field,
|
||||
};
|
||||
}
|
||||
|
||||
buildArg(value: InputValueDefinitionNode): GraphQLArgumentConfig {
|
||||
// Note: While this could make assertions to get the correctly typed
|
||||
// value, that would throw immediately while type system validation
|
||||
// with validateSchema() will produce more actionable results.
|
||||
const type: any = this.getWrappedType(value.type);
|
||||
return {
|
||||
type,
|
||||
description: getDescription(value, this._options),
|
||||
defaultValue: valueFromAST(value.defaultValue, type),
|
||||
astNode: value,
|
||||
};
|
||||
}
|
||||
|
||||
buildInputField(value: InputValueDefinitionNode): GraphQLInputFieldConfig {
|
||||
// Note: While this could make assertions to get the correctly typed
|
||||
// value, that would throw immediately while type system validation
|
||||
// with validateSchema() will produce more actionable results.
|
||||
const type: any = this.getWrappedType(value.type);
|
||||
return {
|
||||
type,
|
||||
description: getDescription(value, this._options),
|
||||
defaultValue: valueFromAST(value.defaultValue, type),
|
||||
astNode: value,
|
||||
};
|
||||
}
|
||||
|
||||
buildEnumValue(value: EnumValueDefinitionNode): GraphQLEnumValueConfig {
|
||||
return {
|
||||
description: getDescription(value, this._options),
|
||||
deprecationReason: getDeprecationReason(value),
|
||||
astNode: value,
|
||||
};
|
||||
}
|
||||
|
||||
buildType(astNode: TypeDefinitionNode): GraphQLNamedType {
|
||||
const name = astNode.name.value;
|
||||
if (stdTypeMap[name]) {
|
||||
return stdTypeMap[name];
|
||||
}
|
||||
|
||||
switch (astNode.kind) {
|
||||
case Kind.OBJECT_TYPE_DEFINITION:
|
||||
return this._makeTypeDef(astNode);
|
||||
case Kind.INTERFACE_TYPE_DEFINITION:
|
||||
return this._makeInterfaceDef(astNode);
|
||||
case Kind.ENUM_TYPE_DEFINITION:
|
||||
return this._makeEnumDef(astNode);
|
||||
case Kind.UNION_TYPE_DEFINITION:
|
||||
return this._makeUnionDef(astNode);
|
||||
case Kind.SCALAR_TYPE_DEFINITION:
|
||||
return this._makeScalarDef(astNode);
|
||||
case Kind.INPUT_OBJECT_TYPE_DEFINITION:
|
||||
return this._makeInputObjectDef(astNode);
|
||||
}
|
||||
|
||||
// Not reachable. All possible type definition nodes have been considered.
|
||||
invariant(
|
||||
false,
|
||||
'Unexpected type definition node: ' + inspect((astNode: empty)),
|
||||
);
|
||||
}
|
||||
|
||||
_makeTypeDef(astNode: ObjectTypeDefinitionNode) {
|
||||
const interfaceNodes = astNode.interfaces;
|
||||
const fieldNodes = astNode.fields;
|
||||
|
||||
// Note: While this could make assertions to get the correctly typed
|
||||
// values below, that would throw immediately while type system
|
||||
// validation with validateSchema() will produce more actionable results.
|
||||
const interfaces =
|
||||
interfaceNodes && interfaceNodes.length > 0
|
||||
? () => interfaceNodes.map(ref => (this.getNamedType(ref): any))
|
||||
: [];
|
||||
|
||||
const fields =
|
||||
fieldNodes && fieldNodes.length > 0
|
||||
? () => keyByNameNode(fieldNodes, field => this.buildField(field))
|
||||
: Object.create(null);
|
||||
|
||||
return new GraphQLObjectType({
|
||||
name: astNode.name.value,
|
||||
description: getDescription(astNode, this._options),
|
||||
interfaces,
|
||||
fields,
|
||||
astNode,
|
||||
});
|
||||
}
|
||||
|
||||
_makeInterfaceDef(astNode: InterfaceTypeDefinitionNode) {
|
||||
const fieldNodes = astNode.fields;
|
||||
|
||||
const fields =
|
||||
fieldNodes && fieldNodes.length > 0
|
||||
? () => keyByNameNode(fieldNodes, field => this.buildField(field))
|
||||
: Object.create(null);
|
||||
|
||||
return new GraphQLInterfaceType({
|
||||
name: astNode.name.value,
|
||||
description: getDescription(astNode, this._options),
|
||||
fields,
|
||||
astNode,
|
||||
});
|
||||
}
|
||||
|
||||
_makeEnumDef(astNode: EnumTypeDefinitionNode) {
|
||||
const valueNodes = astNode.values || [];
|
||||
|
||||
return new GraphQLEnumType({
|
||||
name: astNode.name.value,
|
||||
description: getDescription(astNode, this._options),
|
||||
values: keyByNameNode(valueNodes, value => this.buildEnumValue(value)),
|
||||
astNode,
|
||||
});
|
||||
}
|
||||
|
||||
_makeUnionDef(astNode: UnionTypeDefinitionNode) {
|
||||
const typeNodes = astNode.types;
|
||||
|
||||
// Note: While this could make assertions to get the correctly typed
|
||||
// values below, that would throw immediately while type system
|
||||
// validation with validateSchema() will produce more actionable results.
|
||||
const types =
|
||||
typeNodes && typeNodes.length > 0
|
||||
? () => typeNodes.map(ref => (this.getNamedType(ref): any))
|
||||
: [];
|
||||
|
||||
return new GraphQLUnionType({
|
||||
name: astNode.name.value,
|
||||
description: getDescription(astNode, this._options),
|
||||
types,
|
||||
astNode,
|
||||
});
|
||||
}
|
||||
|
||||
_makeScalarDef(astNode: ScalarTypeDefinitionNode) {
|
||||
return new GraphQLScalarType({
|
||||
name: astNode.name.value,
|
||||
description: getDescription(astNode, this._options),
|
||||
astNode,
|
||||
});
|
||||
}
|
||||
|
||||
_makeInputObjectDef(def: InputObjectTypeDefinitionNode) {
|
||||
const { fields } = def;
|
||||
|
||||
return new GraphQLInputObjectType({
|
||||
name: def.name.value,
|
||||
description: getDescription(def, this._options),
|
||||
fields: fields
|
||||
? () => keyByNameNode(fields, field => this.buildInputField(field))
|
||||
: Object.create(null),
|
||||
astNode: def,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
function keyByNameNode<T: { +name: NameNode, ... }, V>(
|
||||
list: $ReadOnlyArray<T>,
|
||||
valFn: (item: T) => V,
|
||||
): ObjMap<V> {
|
||||
return keyValMap(list, ({ name }) => name.value, valFn);
|
||||
}
|
||||
|
||||
/**
|
||||
* Given a field or enum value node, returns the string value for the
|
||||
* deprecation reason.
|
||||
*/
|
||||
function getDeprecationReason(
|
||||
node: EnumValueDefinitionNode | FieldDefinitionNode,
|
||||
): ?string {
|
||||
const deprecated = getDirectiveValues(GraphQLDeprecatedDirective, node);
|
||||
return deprecated && (deprecated.reason: any);
|
||||
}
|
||||
|
||||
/**
|
||||
* Given an ast node, returns its string description.
|
||||
* @deprecated: provided to ease adoption and will be removed in v16.
|
||||
*
|
||||
* Accepts options as a second argument:
|
||||
*
|
||||
* - commentDescriptions:
|
||||
* Provide true to use preceding comments as the description.
|
||||
*
|
||||
*/
|
||||
export function getDescription(
|
||||
node: { +description?: StringValueNode, +loc?: Location, ... },
|
||||
options: ?BuildSchemaOptions,
|
||||
): void | string {
|
||||
if (node.description) {
|
||||
return node.description.value;
|
||||
}
|
||||
if (options && options.commentDescriptions) {
|
||||
const rawValue = getLeadingCommentBlock(node);
|
||||
if (rawValue !== undefined) {
|
||||
return dedentBlockStringValue('\n' + rawValue);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function getLeadingCommentBlock(node): void | string {
|
||||
const loc = node.loc;
|
||||
if (!loc) {
|
||||
return;
|
||||
}
|
||||
const comments = [];
|
||||
let token = loc.startToken.prev;
|
||||
while (
|
||||
token &&
|
||||
token.kind === TokenKind.COMMENT &&
|
||||
token.next &&
|
||||
token.prev &&
|
||||
token.line + 1 === token.next.line &&
|
||||
token.line !== token.prev.line
|
||||
) {
|
||||
const value = String(token.value);
|
||||
comments.push(value);
|
||||
token = token.prev;
|
||||
}
|
||||
return comments.reverse().join('\n');
|
||||
}
|
||||
|
||||
/**
|
||||
* A helper function to build a GraphQLSchema directly from a source
|
||||
* document.
|
||||
*/
|
||||
export function buildSchema(
|
||||
source: string | Source,
|
||||
options?: BuildSchemaOptions & ParseOptions,
|
||||
): GraphQLSchema {
|
||||
return buildASTSchema(parse(source, options), options);
|
||||
}
|
||||
Reference in New Issue
Block a user