@graphql-tools/stitching-directives
Version:
A set of utils for faster development of GraphQL tools
117 lines (116 loc) • 7.51 kB
JavaScript
import { getNullableType, isAbstractType, isInterfaceType, isListType, isNamedType, isObjectType, isUnionType, parseValue, } from 'graphql';
import { getDirective, getImplementingTypes, isSome, MapperKind, mapSchema, parseSelectionSet, } from '@graphql-tools/utils';
import { defaultStitchingDirectiveOptions } from './defaultStitchingDirectiveOptions.js';
import { parseMergeArgsExpr } from './parseMergeArgsExpr.js';
const dottedNameRegEx = /^[_A-Za-z][_0-9A-Za-z]*(.[_A-Za-z][_0-9A-Za-z]*)*$/;
export function stitchingDirectivesValidator(options = {}) {
const { keyDirectiveName, computedDirectiveName, mergeDirectiveName, pathToDirectivesInExtensions } = {
...defaultStitchingDirectiveOptions,
...options,
};
return (schema) => {
var _a;
const queryTypeName = (_a = schema.getQueryType()) === null || _a === void 0 ? void 0 : _a.name;
mapSchema(schema, {
[MapperKind.OBJECT_TYPE]: type => {
var _a;
const keyDirective = (_a = getDirective(schema, type, keyDirectiveName, pathToDirectivesInExtensions)) === null || _a === void 0 ? void 0 : _a[0];
if (keyDirective != null) {
parseSelectionSet(keyDirective['selectionSet']);
}
return undefined;
},
[MapperKind.OBJECT_FIELD]: (fieldConfig, _fieldName, typeName) => {
var _a, _b, _c;
const computedDirective = (_a = getDirective(schema, fieldConfig, computedDirectiveName, pathToDirectivesInExtensions)) === null || _a === void 0 ? void 0 : _a[0];
if (computedDirective != null) {
parseSelectionSet(computedDirective['selectionSet']);
}
const mergeDirective = (_b = getDirective(schema, fieldConfig, mergeDirectiveName, pathToDirectivesInExtensions)) === null || _b === void 0 ? void 0 : _b[0];
if (mergeDirective != null) {
if (typeName !== queryTypeName) {
throw new Error('@merge directive may be used only for root fields of the root Query type.');
}
let returnType = getNullableType(fieldConfig.type);
if (isListType(returnType)) {
returnType = getNullableType(returnType.ofType);
}
if (!isNamedType(returnType)) {
throw new Error('@merge directive must be used on a field that returns an object or a list of objects.');
}
const mergeArgsExpr = mergeDirective['argsExpr'];
if (mergeArgsExpr != null) {
parseMergeArgsExpr(mergeArgsExpr);
}
const args = Object.keys((_c = fieldConfig.args) !== null && _c !== void 0 ? _c : {});
const keyArg = mergeDirective['keyArg'];
if (keyArg == null) {
if (!mergeArgsExpr && args.length !== 1) {
throw new Error('Cannot use @merge directive without `keyArg` argument if resolver takes more than one argument.');
}
}
else if (!keyArg.match(dottedNameRegEx)) {
throw new Error('`keyArg` argument for @merge directive must be a set of valid GraphQL SDL names separated by periods.');
// TODO: ideally we should check that the arg exists for the resolver
}
const keyField = mergeDirective['keyField'];
if (keyField != null && !keyField.match(dottedNameRegEx)) {
throw new Error('`keyField` argument for @merge directive must be a set of valid GraphQL SDL names separated by periods.');
// TODO: ideally we should check that it is part of the key
}
const key = mergeDirective['key'];
if (key != null) {
if (keyField != null) {
throw new Error('Cannot use @merge directive with both `keyField` and `key` arguments.');
}
for (const keyDef of key) {
let [aliasOrKeyPath, keyPath] = keyDef.split(':');
let aliasPath;
if (keyPath == null) {
keyPath = aliasPath = aliasOrKeyPath;
}
else {
aliasPath = aliasOrKeyPath;
}
if (keyPath != null && !keyPath.match(dottedNameRegEx)) {
throw new Error('Each partial key within the `key` argument for @merge directive must be a set of valid GraphQL SDL names separated by periods.');
// TODO: ideally we should check that it is part of the key
}
if (aliasPath != null && !aliasOrKeyPath.match(dottedNameRegEx)) {
throw new Error('Each alias within the `key` argument for @merge directive must be a set of valid GraphQL SDL names separated by periods.');
// TODO: ideally we should check that the arg exists within the resolver
}
}
}
const additionalArgs = mergeDirective['additionalArgs'];
if (additionalArgs != null) {
parseValue(`{ ${additionalArgs} }`, { noLocation: true });
}
if (mergeArgsExpr != null && (keyArg != null || additionalArgs != null)) {
throw new Error('Cannot use @merge directive with both `argsExpr` argument and any additional argument.');
}
if (!isInterfaceType(returnType) && !isUnionType(returnType) && !isObjectType(returnType)) {
throw new Error('@merge directive may be used only with resolver that return an object, interface, or union.');
}
const typeNames = mergeDirective['types'];
if (typeNames != null) {
if (!isAbstractType(returnType)) {
throw new Error('Types argument can only be used with a field that returns an abstract type.');
}
const implementingTypes = isInterfaceType(returnType)
? getImplementingTypes(returnType.name, schema).map(typeName => schema.getType(typeName))
: returnType.getTypes();
const implementingTypeNames = implementingTypes.map(type => type === null || type === void 0 ? void 0 : type.name).filter(isSome);
for (const typeName of typeNames) {
if (!implementingTypeNames.includes(typeName)) {
throw new Error(`Types argument can only include only type names that implement the field return type's abstract type.`);
}
}
}
}
return undefined;
},
});
return schema;
};
}