@graphql-mesh/serve-cli
Version:
137 lines (136 loc) • 5.2 kB
JavaScript
import cluster from 'node:cluster';
import { createGatewayRuntime } from '@graphql-mesh/serve-runtime';
import { isUrl, PubSub } from '@graphql-mesh/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('proxy')
.description('serve a proxy to a GraphQL API and add additional features such as monitoring/tracing, caching, rate limiting, security, and more')
.argument('[endpoint]', 'URL of the endpoint GraphQL API to proxy')
.option('--schema <schemaPathOrUrl>', 'path to the GraphQL schema file or a url from where to pull the schema')
.action(async function proxy(endpoint) {
const { hiveCdnEndpoint, hiveCdnKey, hiveRegistryToken, maskedErrors, polling, hivePersistedDocumentsEndpoint, hivePersistedDocumentsToken, ...opts } = this.optsWithGlobals();
const loadedConfig = await loadConfig({
log: ctx.log,
configPath: opts.configPath,
quiet: !cluster.isPrimary,
configFileName: ctx.configFileName,
});
let proxy;
if (endpoint) {
proxy = { endpoint };
}
else if ('proxy' in loadedConfig) {
proxy = loadedConfig.proxy;
// TODO: how to provide hive-cdn-key?
}
if (!proxy) {
ctx.log.error('Proxy endpoint not defined. Please provide it in the [endpoint] argument or in the config file.');
process.exit(1);
}
let schema;
const hiveCdnEndpointOpt = opts.schema || hiveCdnEndpoint;
if (hiveCdnEndpointOpt) {
if (hiveCdnKey) {
if (!isUrl(hiveCdnEndpointOpt)) {
ctx.log.error('Hive CDN endpoint must be a URL when providing --hive-cdn-key but got ' +
hiveCdnEndpointOpt);
process.exit(1);
}
schema = {
type: 'hive',
endpoint: hiveCdnEndpointOpt, // see validation above
key: hiveCdnKey,
};
}
else {
schema = opts.schema;
}
}
else if ('schema' in loadedConfig) {
schema = loadedConfig.schema;
// TODO: how to provide hive-cdn-key?
}
if (hiveCdnKey && !schema) {
process.stderr.write(`error: option '--schema <schemaPathOrUrl>' is required when providing '--hive-cdn-key <key>'\n`);
process.exit(1);
}
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),
},
}
: {}),
proxy,
schema,
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 runProxy(ctx, config);
});
export async function runProxy({ log }, config) {
if (handleFork(log, config)) {
return;
}
log.info(`Proxying requests to ${config.proxy.endpoint}`);
const runtime = createGatewayRuntime(config);
await startServerForRuntime(runtime, {
...config,
log,
});
}