UNPKG

@graphql-mesh/merger-federation

Version:
115 lines (114 loc) 4.91 kB
import { execute, extendSchema, parse, } from 'graphql'; import { ApolloGateway, LocalGraphQLDataSource, SERVICE_DEFINITION_QUERY } from '@apollo/gateway'; import { process } from '@graphql-mesh/cross-helpers'; import { PredefinedProxyOptions } from '@graphql-mesh/store'; import { printWithCache } from '@graphql-mesh/utils'; import { addResolversToSchema } from '@graphql-tools/schema'; import { asArray, printSchemaWithDirectives } from '@graphql-tools/utils'; import { wrapSchema } from '@graphql-tools/wrap'; export default class FederationMerger { constructor(options) { this.name = 'federation'; this.logger = options.logger; this.cache = options.cache; this.pubsub = options.pubsub; this.store = options.store; } async getUnifiedSchema({ rawSources, typeDefs, resolvers }) { this.logger.debug(`Creating localServiceList for gateway`); const rawSourceMap = new Map(); const localServiceList = []; const sourceMap = new Map(); await Promise.all(rawSources.map(async (rawSource) => { const transformedSchema = wrapSchema(rawSource); rawSourceMap.set(rawSource.name, rawSource); sourceMap.set(rawSource, transformedSchema); const sdl = await this.store .proxy(`${rawSource.name}_sdl`, PredefinedProxyOptions.StringWithoutValidation) .getWithSet(async () => { this.logger.debug(`Fetching Apollo Federated Service SDL for ${rawSource.name}`); const sdlQueryResult = await execute({ schema: transformedSchema, document: parse(SERVICE_DEFINITION_QUERY), }); if (sdlQueryResult.errors?.length) { throw new AggregateError(sdlQueryResult.errors, `Failed on fetching Federated SDL for ${rawSource.name}`); } return sdlQueryResult.data._service.sdl; }); localServiceList.push({ name: rawSource.name, typeDefs: parse(sdl), }); })); this.logger.debug(`Creating ApolloGateway`); const gateway = new ApolloGateway({ localServiceList, buildService: ({ name }) => { this.logger.debug(`Building federation service: ${name}`); const rawSource = rawSourceMap.get(name); const transformedSchema = sourceMap.get(rawSource); return new LocalGraphQLDataSource(transformedSchema); }, logger: this.logger, debug: !!process.env.DEBUG, serviceHealthCheck: true, }); this.logger.debug(`Loading gateway`); const { schema, executor: gatewayExecutor } = await gateway.load(); const schemaHash = printSchemaWithDirectives(schema); let remoteSchema = schema; this.logger.debug(`Wrapping gateway executor in a unified schema`); const executor = ({ document, info, variables, context, operationName, }) => { const documentStr = printWithCache(document); const { operation } = info; // const operationName = operation.name?.value; return gatewayExecutor({ document, request: { query: documentStr, operationName, variables, }, operationName, cache: this.cache, context, queryHash: documentStr, logger: this.logger, metrics: {}, source: documentStr, operation, schema, schemaHash, overallCachePolicy: undefined, }); }; const id = this.pubsub.subscribe('destroy', async () => { this.pubsub.unsubscribe(id); await gateway.stop(); }); this.logger.debug(`Applying additionalTypeDefs`); typeDefs?.forEach(typeDef => { remoteSchema = extendSchema(remoteSchema, typeDef); }); if (resolvers) { this.logger.debug(`Applying additionalResolvers`); for (const resolversObj of asArray(resolvers)) { remoteSchema = addResolversToSchema({ schema: remoteSchema, resolvers: resolversObj, updateResolversInPlace: true, }); } } this.logger.debug(`Attaching sourceMap to the unified schema`); remoteSchema.extensions = remoteSchema.extensions || {}; Object.defineProperty(remoteSchema.extensions, 'sourceMap', { get: () => sourceMap, }); return { schema: remoteSchema, executor, }; } }