UNPKG

@graphql-tools/graphql

Version:
45 lines (44 loc) 2.19 kB
import { didYouMean } from '../../jsutils/didYouMean.js'; import { suggestionList } from '../../jsutils/suggestionList.js'; import { GraphQLError } from '../../error/GraphQLError.js'; import { isTypeDefinitionNode, isTypeSystemDefinitionNode, isTypeSystemExtensionNode, } from '../../language/predicates.js'; import { introspectionTypes } from '../../type/introspection.js'; import { specifiedScalarTypes } from '../../type/scalars.js'; /** * Known type names * * A GraphQL document is only valid if referenced types (specifically * variable definitions and fragment conditions) are defined by the type schema. * * See https://spec.graphql.org/draft/#sec-Fragment-Spread-Type-Existence */ export function KnownTypeNamesRule(context) { const schema = context.getSchema(); const existingTypesMap = schema ? schema.getTypeMap() : Object.create(null); const definedTypes = Object.create(null); for (const def of context.getDocument().definitions) { if (isTypeDefinitionNode(def)) { definedTypes[def.name.value] = true; } } const typeNames = [...Object.keys(existingTypesMap), ...Object.keys(definedTypes)]; return { NamedType(node, _1, parent, _2, ancestors) { var _a; const typeName = node.name.value; if (!existingTypesMap[typeName] && !definedTypes[typeName]) { const definitionNode = (_a = ancestors[2]) !== null && _a !== void 0 ? _a : parent; const isSDL = definitionNode != null && isSDLNode(definitionNode); if (isSDL && standardTypeNames.includes(typeName)) { return; } const suggestedTypes = suggestionList(typeName, isSDL ? standardTypeNames.concat(typeNames) : typeNames); context.reportError(new GraphQLError(`Unknown type "${typeName}".` + didYouMean(suggestedTypes), { nodes: node })); } }, }; } const standardTypeNames = [...specifiedScalarTypes, ...introspectionTypes].map(type => type.name); function isSDLNode(value) { return 'kind' in value && (isTypeSystemDefinitionNode(value) || isTypeSystemExtensionNode(value)); }