@graphql-codegen/plugin-helpers
Version:
GraphQL Code Generator common utils and types
240 lines (238 loc) • 9.17 kB
JavaScript
import { astFromObjectType, getRootTypeNames, MapperKind, mapSchema } from '@graphql-tools/utils';
import { GraphQLObjectType, isObjectType, parse, } from 'graphql';
import merge from 'lodash/merge.js';
import { oldVisit } from './index.js';
import { getBaseType } from './utils.js';
/**
* Federation Spec
*/
export const federationSpec = parse(/* GraphQL */ `
scalar _FieldSet
directive on FIELD_DEFINITION
directive on FIELD_DEFINITION
directive on FIELD_DEFINITION
directive on OBJECT | INTERFACE
`);
/**
* Adds `__resolveReference` in each ObjectType involved in Federation.
* @param schema
*/
export function addFederationReferencesToSchema(schema) {
return mapSchema(schema, {
[MapperKind.OBJECT_TYPE]: type => {
if (isFederationObjectType(type, schema)) {
const typeConfig = type.toConfig();
typeConfig.fields = {
[resolveReferenceFieldName]: {
type,
},
...typeConfig.fields,
};
return new GraphQLObjectType(typeConfig);
}
return type;
},
});
}
/**
* Removes Federation Spec from GraphQL Schema
* @param schema
* @param config
*/
export function removeFederation(schema) {
return mapSchema(schema, {
[MapperKind.QUERY]: queryType => {
const queryTypeConfig = queryType.toConfig();
delete queryTypeConfig.fields._entities;
delete queryTypeConfig.fields._service;
return new GraphQLObjectType(queryTypeConfig);
},
[MapperKind.UNION_TYPE]: unionType => {
const unionTypeName = unionType.name;
if (unionTypeName === '_Entity' || unionTypeName === '_Any') {
return null;
}
return unionType;
},
[MapperKind.OBJECT_TYPE]: objectType => {
if (objectType.name === '_Service') {
return null;
}
return objectType;
},
});
}
const resolveReferenceFieldName = '__resolveReference';
export class ApolloFederation {
constructor({ enabled, schema }) {
this.enabled = false;
this.enabled = enabled;
this.schema = schema;
this.providesMap = this.createMapOfProvides();
}
/**
* Excludes types definde by Federation
* @param typeNames List of type names
*/
filterTypeNames(typeNames) {
return this.enabled ? typeNames.filter(t => t !== '_FieldSet') : typeNames;
}
/**
* Excludes `__resolveReference` fields
* @param fieldNames List of field names
*/
filterFieldNames(fieldNames) {
return this.enabled ? fieldNames.filter(t => t !== resolveReferenceFieldName) : fieldNames;
}
/**
* Decides if directive should not be generated
* @param name directive's name
*/
skipDirective(name) {
return this.enabled && ['external', 'requires', 'provides', 'key'].includes(name);
}
/**
* Decides if scalar should not be generated
* @param name directive's name
*/
skipScalar(name) {
return this.enabled && name === '_FieldSet';
}
/**
* Decides if field should not be generated
* @param data
*/
skipField({ fieldNode, parentType }) {
if (!this.enabled || !isObjectType(parentType) || !isFederationObjectType(parentType, this.schema)) {
return false;
}
return this.isExternalAndNotProvided(fieldNode, parentType);
}
isResolveReferenceField(fieldNode) {
const name = typeof fieldNode.name === 'string' ? fieldNode.name : fieldNode.name.value;
return this.enabled && name === resolveReferenceFieldName;
}
/**
* Transforms ParentType signature in ObjectTypes involved in Federation
* @param data
*/
transformParentType({ fieldNode, parentType, parentTypeSignature, }) {
if (this.enabled &&
isObjectType(parentType) &&
isFederationObjectType(parentType, this.schema) &&
(isTypeExtension(parentType, this.schema) || fieldNode.name.value === resolveReferenceFieldName)) {
const keys = getDirectivesByName('key', parentType);
if (keys.length) {
const outputs = [`{ __typename: '${parentType.name}' } &`];
// Look for @requires and see what the service needs and gets
const requires = getDirectivesByName('requires', fieldNode).map(this.extractFieldSet);
const requiredFields = this.translateFieldSet(merge({}, ...requires), parentTypeSignature);
// @key() @key() - "primary keys" in Federation
const primaryKeys = keys.map(def => {
const fields = this.extractFieldSet(def);
return this.translateFieldSet(fields, parentTypeSignature);
});
const [open, close] = primaryKeys.length > 1 ? ['(', ')'] : ['', ''];
outputs.push([open, primaryKeys.join(' | '), close].join(''));
// include required fields
if (requires.length) {
outputs.push(`& ${requiredFields}`);
}
return outputs.join(' ');
}
}
return parentTypeSignature;
}
isExternalAndNotProvided(fieldNode, objectType) {
return this.isExternal(fieldNode) && !this.hasProvides(objectType, fieldNode);
}
isExternal(node) {
return getDirectivesByName('external', node).length > 0;
}
hasProvides(objectType, node) {
const fields = this.providesMap[isObjectType(objectType) ? objectType.name : objectType.name.value];
if (fields?.length) {
return fields.includes(node.name.value);
}
return false;
}
translateFieldSet(fields, parentTypeRef) {
return `GraphQLRecursivePick<${parentTypeRef}, ${JSON.stringify(fields)}>`;
}
extractFieldSet(directive) {
const arg = directive.arguments.find(arg => arg.name.value === 'fields');
const { value } = arg.value;
return oldVisit(parse(`{${value}}`), {
leave: {
SelectionSet(node) {
return node.selections.reduce((accum, field) => {
accum[field.name] = field.selection;
return accum;
}, {});
},
Field(node) {
return {
name: node.name.value,
selection: node.selectionSet || true,
};
},
Document(node) {
return node.definitions.find((def) => def.kind === 'OperationDefinition' && def.operation === 'query').selectionSet;
},
},
});
}
createMapOfProvides() {
const providesMap = {};
for (const typename of Object.keys(this.schema.getTypeMap())) {
const objectType = this.schema.getType(typename);
if (isObjectType(objectType)) {
for (const field of Object.values(objectType.getFields())) {
const provides = getDirectivesByName('provides', field.astNode)
.map(this.extractFieldSet)
.reduce((prev, curr) => [...prev, ...Object.keys(curr)], []);
const ofType = getBaseType(field.type);
providesMap[ofType.name] ||= [];
providesMap[ofType.name].push(...provides);
}
}
}
return providesMap;
}
}
/**
* Checks if Object Type is involved in Federation. Based on `@key` directive
* @param node Type
*/
function isFederationObjectType(node, schema) {
const { name: { value: name }, directives, } = isObjectType(node) ? astFromObjectType(node, schema) : node;
const rootTypeNames = getRootTypeNames(schema);
const isNotRoot = !rootTypeNames.has(name);
const isNotIntrospection = !name.startsWith('__');
const hasKeyDirective = directives.some(d => d.name.value === 'key');
return isNotRoot && isNotIntrospection && hasKeyDirective;
}
/**
* Extracts directives from a node based on directive's name
* @param name directive name
* @param node ObjectType or Field
*/
function getDirectivesByName(name, node) {
let astNode;
if (isObjectType(node)) {
astNode = node.astNode;
}
else {
astNode = node;
}
return astNode?.directives?.filter(d => d.name.value === name) || [];
}
/**
* Checks if the Object Type extends a federated type from a remote schema.
* Based on if any of its fields contain the `@external` directive
* @param node Type
*/
function isTypeExtension(node, schema) {
const definition = isObjectType(node) ? node.astNode || astFromObjectType(node, schema) : node;
return definition.fields?.some(field => getDirectivesByName('external', field).length);
}