UNPKG

@graphql-mesh/transform-federation

Version:
155 lines (152 loc) • 7.87 kB
import { isNonNullType, GraphQLNonNull, GraphQLID, isListType, isObjectType, GraphQLObjectType, GraphQLUnionType } from 'graphql'; import { loadFromModuleExportExpression } from '@graphql-mesh/utils'; import { addFederationAnnotations } from 'graphql-transform-federation/dist/transform-sdl.js'; import { entitiesField, serviceField, EntityType } from '@apollo/subgraph/dist/types.js'; import { printSchemaWithDirectives, mapSchema, MapperKind } from '@graphql-tools/utils'; class FederationTransform { constructor({ apiName, baseDir, config, importFn }) { this.apiName = apiName; this.config = config; this.baseDir = baseDir; this.importFn = importFn; } transformSchema(schema, rawSource) { var _a, _b, _c, _d; const federationConfig = {}; rawSource.merge = {}; if ((_a = this.config) === null || _a === void 0 ? void 0 : _a.types) { const queryType = schema.getQueryType(); const queryTypeFields = queryType.getFields(); for (const type of this.config.types) { rawSource.merge[type.name] = {}; const fields = {}; if ((_b = type.config) === null || _b === void 0 ? void 0 : _b.fields) { for (const field of type.config.fields) { fields[field.name] = field.config; rawSource.merge[type.name].fields = rawSource.merge[type.name].fields || {}; rawSource.merge[type.name].fields[field.name] = rawSource.merge[type.name].fields[field.name] || {}; if (field.config.requires) { rawSource.merge[type.name].fields[field.name].computed = true; rawSource.merge[type.name].fields[field.name].selectionSet = `{ ${field.config.requires} }`; } } } // If a field is a key field, it should be GraphQLID if ((_c = type.config) === null || _c === void 0 ? void 0 : _c.keyFields) { rawSource.merge[type.name].selectionSet = `{ ${type.config.keyFields.join(' ')} }`; for (const fieldName of type.config.keyFields) { const objectType = schema.getType(type.name); if (objectType) { const existingType = objectType.getFields()[fieldName].type; objectType.getFields()[fieldName].type = isNonNullType(existingType) ? new GraphQLNonNull(GraphQLID) : GraphQLID; } } } let resolveReference; if ((_d = type.config) === null || _d === void 0 ? void 0 : _d.resolveReference) { const resolveReferenceConfig = type.config.resolveReference; if (typeof resolveReferenceConfig === 'string') { const fn$ = loadFromModuleExportExpression(resolveReferenceConfig, { cwd: this.baseDir, defaultExportName: 'default', importFn: this.importFn, }); resolveReference = (...args) => fn$.then(fn => fn(...args)); } else if (typeof resolveReferenceConfig === 'function') { resolveReference = resolveReferenceConfig; } else { const queryField = queryTypeFields[resolveReferenceConfig.queryFieldName]; const keyArg = resolveReferenceConfig.keyArg || queryField.args[0].name; const keyField = type.config.keyFields[0]; const isBatch = isListType(queryField.args.find(arg => arg.name === keyArg)); resolveReference = async (root, context, info) => { const result = await context[this.apiName].Query[queryField.name]({ root, ...(isBatch ? { key: root[keyField], argsFromKeys: (keys) => ({ [keyArg]: keys, }), } : { args: { [keyArg]: root[keyField], }, }), context, info, }); return { ...root, ...result, }; }; } rawSource.merge[type.name].resolve = resolveReference; } federationConfig[type.name] = { ...type.config, resolveReference, fields, }; } } const entityTypes = Object.fromEntries(Object.entries(federationConfig) .filter(([, { keyFields }]) => keyFields === null || keyFields === void 0 ? void 0 : keyFields.length) .map(([objectName]) => { const type = schema.getType(objectName); if (!isObjectType(type)) { throw new Error(`Type "${objectName}" is not an object type and can't have a key directive`); } return [objectName, type]; })); const hasEntities = !!Object.keys(entityTypes).length; const sdlWithFederationDirectives = addFederationAnnotations(printSchemaWithDirectives(schema), federationConfig); const schemaWithFederationQueryType = mapSchema(schema, { [MapperKind.QUERY]: type => { const config = type.toConfig(); return new GraphQLObjectType({ ...config, fields: { ...config.fields, ...(hasEntities && { _entities: entitiesField, _service: { ...serviceField, resolve: () => ({ sdl: sdlWithFederationDirectives }), }, }), }, }); }, }); const schemaWithUnionType = mapSchema(schemaWithFederationQueryType, { [MapperKind.UNION_TYPE]: type => { if (type.name === EntityType.name) { return new GraphQLUnionType({ ...EntityType.toConfig(), types: Object.values(entityTypes), }); } return type; }, }); // Not using transformSchema since it will remove resolveReference Object.entries(federationConfig).forEach(([objectName, currentFederationConfig]) => { if (currentFederationConfig.resolveReference) { const type = schemaWithUnionType.getType(objectName); if (!isObjectType(type)) { throw new Error(`Type "${objectName}" is not an object type and can't have a resolveReference function`); } type.resolveObject = currentFederationConfig.resolveReference; } }); return schemaWithUnionType; } } export default FederationTransform;