graphql
Version:
A Query Language and Runtime which can target any service.
46 lines • 2.1 kB
JavaScript
import { GraphQLError } from "../../error/GraphQLError.mjs";
import { isInputObjectType, isInterfaceType, isObjectType, } from "../../type/definition.mjs";
export function UniqueFieldDefinitionNamesRule(context) {
const schema = context.getSchema();
const existingTypeMap = schema ? schema.getTypeMap() : Object.create(null);
const knownFieldNames = new Map();
return {
InputObjectTypeDefinition: checkFieldUniqueness,
InputObjectTypeExtension: checkFieldUniqueness,
InterfaceTypeDefinition: checkFieldUniqueness,
InterfaceTypeExtension: checkFieldUniqueness,
ObjectTypeDefinition: checkFieldUniqueness,
ObjectTypeExtension: checkFieldUniqueness,
};
function checkFieldUniqueness(node) {
const typeName = node.name.value;
let fieldNames = knownFieldNames.get(typeName);
if (fieldNames == null) {
fieldNames = new Map();
knownFieldNames.set(typeName, fieldNames);
}
const fieldNodes = node.fields ?? [];
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.`, { nodes: fieldDef.name }));
continue;
}
const knownFieldName = fieldNames.get(fieldName);
if (knownFieldName != null) {
context.reportError(new GraphQLError(`Field "${typeName}.${fieldName}" can only be defined once.`, { nodes: [knownFieldName, fieldDef.name] }));
}
else {
fieldNames.set(fieldName, fieldDef.name);
}
}
return false;
}
}
function hasField(type, fieldName) {
if (isObjectType(type) || isInterfaceType(type) || isInputObjectType(type)) {
return type.getFields()[fieldName] != null;
}
return false;
}
//# sourceMappingURL=UniqueFieldDefinitionNamesRule.js.map