@graphql-mesh/merger-federation
Version:
118 lines (117 loc) • 5.06 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
const graphql_1 = require("graphql");
const gateway_1 = require("@apollo/gateway");
const cross_helpers_1 = require("@graphql-mesh/cross-helpers");
const store_1 = require("@graphql-mesh/store");
const utils_1 = require("@graphql-mesh/utils");
const schema_1 = require("@graphql-tools/schema");
const utils_2 = require("@graphql-tools/utils");
const wrap_1 = require("@graphql-tools/wrap");
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 = (0, wrap_1.wrapSchema)(rawSource);
rawSourceMap.set(rawSource.name, rawSource);
sourceMap.set(rawSource, transformedSchema);
const sdl = await this.store
.proxy(`${rawSource.name}_sdl`, store_1.PredefinedProxyOptions.StringWithoutValidation)
.getWithSet(async () => {
this.logger.debug(`Fetching Apollo Federated Service SDL for ${rawSource.name}`);
const sdlQueryResult = await (0, graphql_1.execute)({
schema: transformedSchema,
document: (0, graphql_1.parse)(gateway_1.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: (0, graphql_1.parse)(sdl),
});
}));
this.logger.debug(`Creating ApolloGateway`);
const gateway = new gateway_1.ApolloGateway({
localServiceList,
buildService: ({ name }) => {
this.logger.debug(`Building federation service: ${name}`);
const rawSource = rawSourceMap.get(name);
const transformedSchema = sourceMap.get(rawSource);
return new gateway_1.LocalGraphQLDataSource(transformedSchema);
},
logger: this.logger,
debug: !!cross_helpers_1.process.env.DEBUG,
serviceHealthCheck: true,
});
this.logger.debug(`Loading gateway`);
const { schema, executor: gatewayExecutor } = await gateway.load();
const schemaHash = (0, utils_2.printSchemaWithDirectives)(schema);
let remoteSchema = schema;
this.logger.debug(`Wrapping gateway executor in a unified schema`);
const executor = ({ document, info, variables, context, operationName, }) => {
const documentStr = (0, utils_1.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 = (0, graphql_1.extendSchema)(remoteSchema, typeDef);
});
if (resolvers) {
this.logger.debug(`Applying additionalResolvers`);
for (const resolversObj of (0, utils_2.asArray)(resolvers)) {
remoteSchema = (0, schema_1.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,
};
}
}
exports.default = FederationMerger;
;