123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687 |
- // @flow strict
- import { GraphQLError } from '../../error/GraphQLError';
- import type { ASTVisitor } from '../../language/visitor';
- import type {
- NameNode,
- FieldDefinitionNode,
- InputValueDefinitionNode,
- } from '../../language/ast';
- import type { GraphQLNamedType } from '../../type/definition';
- import {
- isObjectType,
- isInterfaceType,
- isInputObjectType,
- } from '../../type/definition';
- import type { SDLValidationContext } from '../ValidationContext';
- /**
- * Unique field definition names
- *
- * A GraphQL complex type is only valid if all its fields are uniquely named.
- */
- export function UniqueFieldDefinitionNamesRule(
- context: SDLValidationContext,
- ): ASTVisitor {
- const schema = context.getSchema();
- const existingTypeMap = schema ? schema.getTypeMap() : Object.create(null);
- const knownFieldNames = Object.create(null);
- return {
- InputObjectTypeDefinition: checkFieldUniqueness,
- InputObjectTypeExtension: checkFieldUniqueness,
- InterfaceTypeDefinition: checkFieldUniqueness,
- InterfaceTypeExtension: checkFieldUniqueness,
- ObjectTypeDefinition: checkFieldUniqueness,
- ObjectTypeExtension: checkFieldUniqueness,
- };
- function checkFieldUniqueness(node: {
- +name: NameNode,
- +fields?: $ReadOnlyArray<InputValueDefinitionNode | FieldDefinitionNode>,
- ...
- }) {
- const typeName = node.name.value;
- if (!knownFieldNames[typeName]) {
- knownFieldNames[typeName] = Object.create(null);
- }
- // istanbul ignore next (See: 'https://github.com/graphql/graphql-js/issues/2203')
- const fieldNodes = node.fields ?? [];
- const fieldNames = knownFieldNames[typeName];
- for (const fieldDef of fieldNodes) {
- const fieldName = fieldDef.name.value;
- if (hasField(existingTypeMap[typeName], fieldName)) {
- context.reportError(
- new GraphQLError(
- `Field "${typeName}.${fieldName}" already exists in the schema. It cannot also be defined in this type extension.`,
- fieldDef.name,
- ),
- );
- } else if (fieldNames[fieldName]) {
- context.reportError(
- new GraphQLError(
- `Field "${typeName}.${fieldName}" can only be defined once.`,
- [fieldNames[fieldName], fieldDef.name],
- ),
- );
- } else {
- fieldNames[fieldName] = fieldDef.name;
- }
- }
- return false;
- }
- }
- function hasField(type: GraphQLNamedType, fieldName: string): boolean {
- if (isObjectType(type) || isInterfaceType(type) || isInputObjectType(type)) {
- return type.getFields()[fieldName] != null;
- }
- return false;
- }
|