This commit is contained in:
87
node_modules/graphql/validation/rules/UniqueFieldDefinitionNamesRule.js.flow
generated
vendored
Normal file
87
node_modules/graphql/validation/rules/UniqueFieldDefinitionNamesRule.js.flow
generated
vendored
Normal file
@@ -0,0 +1,87 @@
|
||||
// @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;
|
||||
}
|
||||
Reference in New Issue
Block a user