import { GraphQLError } from '../../error/GraphQLError';
import type {
EnumTypeDefinitionNode,
EnumTypeExtensionNode,
} from '../../language/ast';
import type { ASTVisitor } from '../../language/visitor';
import { isEnumType } from '../../type/definition';
import type { SDLValidationContext } from '../ValidationContext';
export function UniqueEnumValueNamesRule(
context: SDLValidationContext,
): ASTVisitor {
const schema = context.getSchema();
const existingTypeMap = schema ? schema.getTypeMap() : Object.create(null);
const knownValueNames = Object.create(null);
return {
EnumTypeDefinition: checkValueUniqueness,
EnumTypeExtension: checkValueUniqueness,
};
function checkValueUniqueness(
node: EnumTypeDefinitionNode | EnumTypeExtensionNode,
) {
const typeName = node.name.value;
if (!knownValueNames[typeName]) {
knownValueNames[typeName] = Object.create(null);
}
const valueNodes = node.values ?? [];
const valueNames = knownValueNames[typeName];
for (const valueDef of valueNodes) {
const valueName = valueDef.name.value;
const existingType = existingTypeMap[typeName];
if (isEnumType(existingType) && existingType.getValue(valueName)) {
context.reportError(
new GraphQLError(
`Enum value "${typeName}.${valueName}" already exists in the schema. It cannot also be defined in this type extension.`,
{ nodes: valueDef.name },
),
);
} else if (valueNames[valueName]) {
context.reportError(
new GraphQLError(
`Enum value "${typeName}.${valueName}" can only be defined once.`,
{ nodes: [valueNames[valueName], valueDef.name] },
),
);
} else {
valueNames[valueName] = valueDef.name;
}
}
return false;
}
}