UNPKG

@graphql-mesh/compose-cli

Version:
139 lines (138 loc) 5.36 kB
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; }