UNPKG

@graphql-mesh/serve-cli

Version:
129 lines (128 loc) 4.89 kB
import cluster from 'node:cluster'; import { lstat } from 'node:fs/promises'; import { isAbsolute, resolve } from 'node:path'; import { createGatewayRuntime, } from '@graphql-mesh/serve-runtime'; import { isUrl, PubSub } from '@graphql-mesh/utils'; import { isValidPath } from '@graphql-tools/utils'; import { defaultOptions, } from '../cli.js'; import { getBuiltinPluginsFromConfig, getCacheInstanceFromConfig, loadConfig } from '../config.js'; import { startServerForRuntime } from '../server.js'; import { handleFork } from './handleFork.js'; export const addCommand = (ctx, cli) => cli .command('subgraph') .description('serve a Federation subgraph that can be used with any Federation compatible router like Apollo Router/Gateway') .argument('[schemaPathOrUrl]', 'path to the subgraph schema file or a url from where to pull the subgraph schema (default: "subgraph.graphql")') .action(async function subgraph(schemaPathOrUrl) { const { maskedErrors, hiveRegistryToken, polling, hivePersistedDocumentsEndpoint, hivePersistedDocumentsToken, ...opts } = this.optsWithGlobals(); const loadedConfig = await loadConfig({ log: ctx.log, configPath: opts.configPath, quiet: !cluster.isPrimary, configFileName: ctx.configFileName, }); let subgraph = 'subgraph.graphql'; if (schemaPathOrUrl) { subgraph = schemaPathOrUrl; } else if ('subgraph' in loadedConfig) { subgraph = loadedConfig.subgraph; } const pubsub = loadedConfig.pubsub || new PubSub(); const cache = await getCacheInstanceFromConfig(loadedConfig, { pubsub, logger: ctx.log, }); const builtinPlugins = await getBuiltinPluginsFromConfig({ ...loadedConfig, ...opts, }, { cache, logger: ctx.log, }); const config = { ...defaultOptions, ...loadedConfig, ...opts, ...(hiveRegistryToken ? { reporting: { ...loadedConfig.reporting, type: 'hive', token: hiveRegistryToken, }, } : {}), ...(polling ? { pollingInterval: polling } : {}), ...(hivePersistedDocumentsEndpoint ? { persistedDocuments: { type: 'hive', endpoint: hivePersistedDocumentsEndpoint || (loadedConfig.persistedDocuments && 'endpoint' in loadedConfig.persistedDocuments && loadedConfig.persistedDocuments?.endpoint), token: hivePersistedDocumentsToken || (loadedConfig.persistedDocuments && 'token' in loadedConfig.persistedDocuments && loadedConfig.persistedDocuments?.token), }, } : {}), subgraph, logging: loadedConfig.logging ?? ctx.log, productName: ctx.productName, productDescription: ctx.productDescription, productPackageName: ctx.productPackageName, productLogo: ctx.productLogo, productLink: ctx.productLink, pubsub, cache, plugins(ctx) { const userPlugins = loadedConfig.plugins?.(ctx) ?? []; return [...builtinPlugins, ...userPlugins]; }, }; if (maskedErrors != null) { // overwrite masked errors from loaded config only when provided config.maskedErrors = maskedErrors; } if (typeof config.pollingInterval === 'number' && config.pollingInterval < 10_000) { process.stderr.write(`error: polling interval duration too short, use at least 10 seconds\n`); process.exit(1); } return runSubgraph(ctx, config); }); export async function runSubgraph({ log }, config) { let absSchemaPath = null; if (typeof config.subgraph === 'string' && isValidPath(config.subgraph) && !isUrl(config.subgraph)) { const subgraphPath = config.subgraph; absSchemaPath = isAbsolute(subgraphPath) ? String(subgraphPath) : resolve(process.cwd(), subgraphPath); try { await lstat(absSchemaPath); } catch { throw new Error(`Subgraph schema at ${absSchemaPath} does not exist`); } } if (handleFork(log, config)) { return; } const runtime = createGatewayRuntime(config); if (absSchemaPath) { log.info(`Serving local subgraph from ${absSchemaPath}`); } else if (isUrl(String(config.subgraph))) { log.info(`Serving remote subgraph from ${config.subgraph}`); } else { log.info('Serving subgraph from config'); } await startServerForRuntime(runtime, { ...config, log, }); }