@graphql-mesh/compose-cli
Version:
139 lines (138 loc) • 5.36 kB
JavaScript
import { buildSchema, Kind, parse, print, visit, } from 'graphql';
import { composeSubgraphs, getAnnotatedSubgraphs, } from '@graphql-mesh/fusion-composition';
import { parseWithCache } from '@graphql-mesh/utils';
import { CodeFileLoader } from '@graphql-tools/code-file-loader';
import { GraphQLFileLoader } from '@graphql-tools/graphql-file-loader';
import { loadTypedefs } from '@graphql-tools/load';
import { mergeSchemas } from '@graphql-tools/schema';
import { printSchemaWithDirectives } from '@graphql-tools/utils';
import { fetch as defaultFetch } from '@whatwg-node/fetch';
const isDebug = ['1', 'y', 'yes', 't', 'true'].includes(String(process.env.DEBUG));
export async function getComposedSchemaFromConfig(config, logger) {
const ctx = {
fetch: config.fetch || defaultFetch,
cwd: config.cwd || globalThis.process?.cwd?.(),
logger,
};
const subgraphConfigsForComposition = await Promise.all(config.subgraphs.map(async (subgraphCLIConfig) => {
const { name: subgraphName, schema$ } = subgraphCLIConfig.sourceHandler(ctx);
const log = logger.child(`[${subgraphName}]`);
log.debug(`Loading subgraph`);
let subgraphSchema;
try {
subgraphSchema = await schema$;
}
catch (e) {
if (isDebug) {
log.error(`Failed to load subgraph`, e);
}
else {
log.error(e.message || e);
log.error(`Failed to load subgraph`);
}
process.exit(1);
}
return {
name: subgraphName,
schema: subgraphSchema,
transforms: subgraphCLIConfig.transforms,
};
}));
let additionalTypeDefs;
if (config.additionalTypeDefs != null) {
let additionalFieldDirectiveUsed = false;
if (typeof config.additionalTypeDefs === 'string' && config.additionalTypeDefs?.includes(' ')) {
try {
additionalTypeDefs = [parseWithCache(config.additionalTypeDefs)];
}
catch (e) {
logger.error(`Failed to parse additional type definitions: ${e.message || e}`);
process.exit(1);
}
}
else {
const result = await loadTypedefs(config.additionalTypeDefs, {
noLocation: true,
assumeValid: true,
assumeValidSDL: true,
loaders: [new CodeFileLoader(), new GraphQLFileLoader()],
});
additionalTypeDefs = result.map(source => source.document ||
parseWithCache(source.rawSDL || printSchemaWithDirectives(source.schema)));
}
additionalTypeDefs = additionalTypeDefs.map(doc => visit(doc, {
[Kind.FIELD_DEFINITION](node) {
additionalFieldDirectiveUsed = true;
return {
...node,
directives: [
...(node.directives || []),
{
kind: Kind.DIRECTIVE,
name: { kind: Kind.NAME, value: 'additionalField' },
},
],
};
},
}));
if (additionalFieldDirectiveUsed) {
additionalTypeDefs.unshift(parse(/* GraphQL */ `
directive @additionalField on FIELD_DEFINITION
`));
}
}
if (config.subgraph) {
const annotatedSubgraphs = getAnnotatedSubgraphs(subgraphConfigsForComposition, {
ignoreSemanticConventions: config.ignoreSemanticConventions,
alwaysAddTransportDirective: true,
});
const subgraph = annotatedSubgraphs.find(sg => sg.name === config.subgraph);
if (!subgraph) {
logger.error(`Subgraph ${config.subgraph} not found`);
process.exit(1);
}
return print(subgraph.typeDefs);
}
const result = composeSubgraphs(subgraphConfigsForComposition);
if (result.errors?.length) {
logger.error(`Failed to compose subgraphs`);
for (const error of result.errors) {
if (isDebug) {
logger.error(error);
}
else {
logger.error(error.message || error);
}
}
process.exit(1);
}
if (!result.supergraphSdl) {
logger.error(`Unknown error: Supergraph is empty`);
process.exit(1);
}
if (additionalTypeDefs?.length /* TODO || config.transforms?.length */) {
let composedSchema = buildSchema(result.supergraphSdl, {
noLocation: true,
assumeValid: true,
assumeValidSDL: true,
});
if (additionalTypeDefs?.length) {
composedSchema = mergeSchemas({
schemas: [composedSchema],
typeDefs: additionalTypeDefs,
assumeValid: true,
assumeValidSDL: true,
});
}
/* TODO
if (config.transforms?.length) {
logger.info('Applying transforms');
for (const transform of config.transforms) {
composedSchema = transform(composedSchema);
}
}
*/
return printSchemaWithDirectives(composedSchema);
}
return result.supergraphSdl;
}