UNPKG

graphql-config

Version:

The easiest way to configure your development environment with your GraphQL schema (supported by most tools, editors & IDEs)

121 lines (120 loc) 4.75 kB
import { dirname } from 'path'; import { GraphQLProjectConfig } from './project-config.js'; import { isMultipleProjectConfig, isSingleProjectConfig, findConfig, getConfig, getConfigSync, findConfigSync, isLegacyProjectConfig, } from './helpers/index.js'; import { ProjectNotFoundError, ConfigNotFoundError, ConfigEmptyError } from './errors.js'; import { GraphQLExtensionsRegistry } from './extension.js'; import { EndpointsExtension } from './extensions/endpoints.js'; import { isLegacyConfig } from './helpers/cosmiconfig.js'; const CWD = process.cwd(); const defaultConfigName = 'graphql'; const defaultLoadConfigOptions = { rootDir: CWD, extensions: [], throwOnMissing: true, throwOnEmpty: true, configName: defaultConfigName, legacy: true, }; export async function loadConfig(options) { const { filepath, configName, rootDir, extensions, throwOnEmpty, throwOnMissing, legacy } = { ...defaultLoadConfigOptions, ...options, }; try { const found = filepath ? await getConfig({ filepath, configName, legacy }) : await findConfig({ rootDir, configName, legacy }); return new GraphQLConfig(found, extensions); } catch (error) { return handleError(error, { throwOnMissing, throwOnEmpty }); } } export function loadConfigSync(options) { const { filepath, configName, rootDir, extensions, throwOnEmpty, throwOnMissing, legacy } = { ...defaultLoadConfigOptions, ...options, }; try { const found = filepath ? getConfigSync({ filepath, configName, legacy }) : findConfigSync({ rootDir, configName, legacy }); return new GraphQLConfig(found, extensions); } catch (error) { return handleError(error, { throwOnMissing, throwOnEmpty }); } } function handleError(error, options) { if ((!options.throwOnMissing && error instanceof ConfigNotFoundError) || (!options.throwOnEmpty && error instanceof ConfigEmptyError)) { return; } throw error; } export class GraphQLConfig { constructor(raw, extensions) { // TODO: in v5 change projects to `Object.create(null)` and refactor `graphql-codegen-cli` to remove `projects.hasOwnProperty` // https://github.com/dotansimha/graphql-code-generator/blob/3c6abbde7a20515d9a1d55b4003ef365d248efb5/packages/graphql-codegen-cli/src/graphql-config.ts#L62-L72 this.projects = {}; this._rawConfig = raw.config; this.filepath = raw.filepath; this.dirpath = dirname(raw.filepath); this.extensions = new GraphQLExtensionsRegistry({ cwd: this.dirpath }); // Register Endpoints this.extensions.register(EndpointsExtension); for (const extension of extensions) { this.extensions.register(extension); } if (isMultipleProjectConfig(this._rawConfig)) { for (const [projectName, config] of Object.entries(this._rawConfig.projects)) { this.projects[projectName] = new GraphQLProjectConfig({ filepath: this.filepath, name: projectName, config, extensionsRegistry: this.extensions, }); } } else if (isSingleProjectConfig(this._rawConfig) || isLegacyProjectConfig(this._rawConfig)) { this.projects.default = new GraphQLProjectConfig({ filepath: this.filepath, name: 'default', config: this._rawConfig, extensionsRegistry: this.extensions, }); } } getProject(name) { if (!name) { return this.getDefault(); } const project = this.projects[name]; if (!project) { throw new ProjectNotFoundError(`Project '${name}' not found`); } return project; } getProjectForFile(filepath) { // Looks for a project that includes the file or the file is a part of schema or documents for (const project of Object.values(this.projects)) { if (project.match(filepath)) { return project; } } // The file doesn't match any of the project // Looks for a first project that has no `include` and `exclude` for (const project of Object.values(this.projects)) { if (!project.include && !project.exclude) { return project; } } throw new ProjectNotFoundError(`File '${filepath}' doesn't match any project`); } getDefault() { return this.getProject('default'); } isLegacy() { return isLegacyConfig(this.filepath); } }