UNPKG

@graphql-mesh/compose-cli

Version:
129 lines (128 loc) 5.44 kB
import 'dotenv/config'; // inject dotenv options to process.env // eslint-disable-next-line import/no-nodejs-modules import { Console } from 'node:console'; // eslint-disable-next-line import/no-nodejs-modules import { promises as fsPromises } from 'node:fs'; // eslint-disable-next-line import/no-nodejs-modules import module from 'node:module'; // eslint-disable-next-line import/no-nodejs-modules import { isAbsolute, join } from 'node:path'; // eslint-disable-next-line import/no-nodejs-modules import { pathToFileURL } from 'node:url'; import { parse } from 'graphql'; import { Command, Option } from '@commander-js/extra-typings'; import { DefaultLogger } from '@graphql-mesh/utils'; import { getComposedSchemaFromConfig } from './getComposedSchemaFromConfig.js'; /** Default config paths sorted by priority. */ const defaultConfigPaths = [ 'mesh.config.ts', 'mesh.config.mts', 'mesh.config.cts', 'mesh.config.js', 'mesh.config.mjs', 'mesh.config.cjs', ]; let program = new Command() .addOption(new Option('-c, --config-path <path>', `path to the configuration file. defaults to the following files respectively in the current working directory: ${defaultConfigPaths.join(', ')}`).env('CONFIG_PATH')) .option('--subgraph <name>', 'name of the subgraph to compose') .option('-o, --output <path>', 'path to the output file'); export async function run({ log: rootLog = new DefaultLogger(undefined, undefined, undefined, new Console(process.stderr)), productName = 'Mesh Compose', productDescription = 'compose a GraphQL federated schema from any API service(s)', binName = 'mesh-compose', version, }) { module.register('@graphql-mesh/include/hooks', // eslint-disable-next-line @typescript-eslint/ban-ts-comment // @ts-ignore bob will complain when bundling for cjs import.meta.url); program = program.name(binName).description(productDescription); if (version) program = program.version(version); if (process.env.NODE_ENV === 'test') program = program.allowUnknownOption(); const opts = program.parse().opts(); const log = rootLog.child(` ${productName}`); let importedConfig; if (!opts.configPath) { log.debug(`Searching for default config files`); for (const configPath of defaultConfigPaths) { const absoluteConfigPath = join(process.cwd(), configPath); const exists = await fsPromises .lstat(absoluteConfigPath) .then(() => true) .catch(() => false); if (exists) { log.info(`Found default config file ${absoluteConfigPath}`); const importUrl = pathToFileURL(absoluteConfigPath).toString(); const module = await import(importUrl); importedConfig = Object(module).composeConfig; if (!importedConfig) { throw new Error(`No "composeConfig" exported from default config at ${configPath}`); } break; } } if (!importedConfig) { throw new Error(`Cannot find default config file at ${defaultConfigPaths.join(' or ')} in the current working directory`); } } else { // using user-provided config const configPath = isAbsolute(opts.configPath) ? opts.configPath : join(process.cwd(), opts.configPath); log.debug(`Loading config file at path ${configPath}`); const exists = await fsPromises .lstat(configPath) .then(() => true) .catch(() => false); if (!exists) { throw new Error(`Cannot find config file at ${configPath}`); } const importUrl = pathToFileURL(configPath).toString(); const module = await import(importUrl); importedConfig = Object(module).composeConfig; if (!importedConfig) { throw new Error(`No "composeConfig" exported from config at ${configPath}`); } } log.info('Loaded config'); const config = { ...importedConfig, ...opts, }; log.info('Composing'); const supergraphSdl = await getComposedSchemaFromConfig(config, log); let output = config.output; if (!output) { if (typeof process === 'object') { process.stdout.write(supergraphSdl + '\n'); } else { console.log(supergraphSdl); } log.info('Done!'); return; } log.info(`Writing schema to ${output}`); output = isAbsolute(output) ? output : join(process.cwd(), output); let writtenData; if (output.endsWith('.json')) { writtenData = JSON.stringify(parse(supergraphSdl, { noLocation: true }), null, 2); } else if (output.endsWith('.graphql') || output.endsWith('.gql') || output.endsWith('.graphqls') || output.endsWith('.gqls')) { writtenData = supergraphSdl; } else if (output.endsWith('.ts') || output.endsWith('.cts') || output.endsWith('.mts') || output.endsWith('.js') || output.endsWith('.cjs') || output.endsWith('.mjs')) { writtenData = `export default ${JSON.stringify(supergraphSdl)}`; } else { throw new Error(`Unsupported file extension for ${output}`); } await fsPromises.writeFile(output, writtenData, 'utf8'); log.info('Done!'); }