UNPKG

@graphql-hive/cli

Version:

A CLI util to manage and control your GraphQL Hive

342 lines • 13.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); const tslib_1 = require("tslib"); const fs_1 = require("fs"); const graphql_1 = require("graphql"); const utils_1 = require("@graphql-tools/utils"); const core_1 = require("@oclif/core"); const base_command_1 = tslib_1.__importDefault(require("../../base-command")); const gql_1 = require("../../gql"); const config_1 = require("../../helpers/config"); const git_1 = require("../../helpers/git"); const schema_1 = require("../../helpers/schema"); const validation_1 = require("../../helpers/validation"); const schemaPublishMutation = (0, gql_1.graphql)(/* GraphQL */ ` mutation schemaPublish($input: SchemaPublishInput!, $usesGitHubApp: Boolean!) { schemaPublish(input: $input) { __typename ... on SchemaPublishSuccess @skip(if: $usesGitHubApp) { initial valid successMessage: message linkToWebsite changes { nodes { message(withSafeBasedOnUsageNote: false) criticality isSafeBasedOnUsage } total ...RenderChanges_schemaChanges } } ... on SchemaPublishError @skip(if: $usesGitHubApp) { valid linkToWebsite changes { nodes { message(withSafeBasedOnUsageNote: false) criticality isSafeBasedOnUsage } total ...RenderChanges_schemaChanges } errors { nodes { message } total } } ... on SchemaPublishMissingServiceError @skip(if: $usesGitHubApp) { missingServiceError: message } ... on SchemaPublishMissingUrlError @skip(if: $usesGitHubApp) { missingUrlError: message } ... on GitHubSchemaPublishSuccess @include(if: $usesGitHubApp) { message } ... on GitHubSchemaPublishError @include(if: $usesGitHubApp) { message } ... on SchemaPublishRetry { reason } } } `); class SchemaPublish extends base_command_1.default { resolveMetadata(metadata) { if (!metadata) { return; } try { JSON.parse(metadata); // If we are able to parse it, it means it's a valid JSON, let's use it as-is return metadata; } catch (e) { // If we can't parse it, we can try to load it from FS const exists = (0, fs_1.existsSync)(metadata); if (!exists) { throw new Error(`Failed to load metadata from "${metadata}": Please specify a path to an existing file, or a string with valid JSON.`); } try { const fileContent = (0, fs_1.readFileSync)(metadata, 'utf-8'); JSON.parse(fileContent); return fileContent; } catch (e) { throw new Error(`Failed to load metadata from file "${metadata}": Please make sure the file is readable and contains a valid JSON`); } } } async run() { var _a, _b; try { const { flags, args } = await this.parse(SchemaPublish); await this.require(flags); const endpoint = this.ensure({ key: 'registry.endpoint', args: flags, legacyFlagName: 'registry', defaultValue: config_1.graphqlEndpoint, env: 'HIVE_REGISTRY', }); const accessToken = this.ensure({ key: 'registry.accessToken', args: flags, legacyFlagName: 'token', env: 'HIVE_TOKEN', }); const service = flags.service; const url = flags.url; const file = args.file; const force = flags.force; const experimental_acceptBreakingChanges = flags.experimental_acceptBreakingChanges; const metadata = this.resolveMetadata(flags.metadata); const usesGitHubApp = flags.github; let commit = this.maybe({ key: 'commit', args: flags, env: 'HIVE_COMMIT', }); let author = this.maybe({ key: 'author', args: flags, env: 'HIVE_AUTHOR', }); let gitHub = null; if (!commit || !author) { const git = await (0, git_1.gitInfo)(() => { this.warn(`No git information found. Couldn't resolve author and commit.`); }); if (!commit) { commit = git.commit; } if (!author) { author = git.author; } } if (!author) { throw new core_1.Errors.CLIError(`Missing "author"`); } if (!commit) { throw new core_1.Errors.CLIError(`Missing "commit"`); } if (usesGitHubApp) { // eslint-disable-next-line no-process-env const repository = (_a = process.env['GITHUB_REPOSITORY']) !== null && _a !== void 0 ? _a : null; if (!repository) { throw new core_1.Errors.CLIError(`Missing "GITHUB_REPOSITORY" environment variable.`); } gitHub = { repository, commit, }; } let sdl; try { const rawSdl = await (0, schema_1.loadSchema)('introspection', file); (0, validation_1.invariant)(typeof rawSdl === 'string' && rawSdl.length > 0, 'Schema seems empty'); const transformedSDL = (0, graphql_1.print)((0, utils_1.transformCommentsToDescriptions)(rawSdl)); sdl = (0, schema_1.minifySchema)(transformedSDL); } catch (err) { if (err instanceof graphql_1.GraphQLError) { const location = (_b = err.locations) === null || _b === void 0 ? void 0 : _b[0]; const locationString = location ? ` at line ${location.line}, column ${location.column}` : ''; throw new Error(`The SDL is not valid${locationString}:\n ${err.message}`); } throw err; } let result = null; do { result = await this.registryApi(endpoint, accessToken).request({ operation: schemaPublishMutation, variables: { input: { service, url, author, commit, sdl, force, experimental_acceptBreakingChanges: experimental_acceptBreakingChanges === true, metadata, gitHub, supportsRetry: true, }, usesGitHubApp: !!gitHub, }, /** Gateway timeout is 60 seconds. */ timeout: 55000, }); if (result.schemaPublish.__typename === 'SchemaPublishSuccess') { const changes = result.schemaPublish.changes; if (result.schemaPublish.initial) { this.success('Published initial schema.'); } else if (result.schemaPublish.successMessage) { this.success(result.schemaPublish.successMessage); } else if (changes && changes.total === 0) { this.success('No changes. Skipping.'); } else { if (changes) { schema_1.renderChanges.call(this, changes); } this.success('Schema published'); } if (result.schemaPublish.linkToWebsite) { this.info(`Available at ${result.schemaPublish.linkToWebsite}`); } } else if (result.schemaPublish.__typename === 'SchemaPublishRetry') { this.log(result.schemaPublish.reason); this.log('Waiting for other schema publishes to complete...'); result = null; } else if (result.schemaPublish.__typename === 'SchemaPublishMissingServiceError') { this.fail(`${result.schemaPublish.missingServiceError} Please use the '--service <name>' parameter.`); this.exit(1); } else if (result.schemaPublish.__typename === 'SchemaPublishMissingUrlError') { this.fail(`${result.schemaPublish.missingUrlError} Please use the '--url <url>' parameter.`); this.exit(1); } else if (result.schemaPublish.__typename === 'SchemaPublishError') { const changes = result.schemaPublish.changes; const errors = result.schemaPublish.errors; schema_1.renderErrors.call(this, errors); if (changes && changes.total) { this.log(''); schema_1.renderChanges.call(this, changes); } this.log(''); if (!force) { this.fail('Failed to publish schema'); this.exit(1); } else { this.success('Schema published (forced)'); } if (result.schemaPublish.linkToWebsite) { this.info(`Available at ${result.schemaPublish.linkToWebsite}`); } } else if (result.schemaPublish.__typename === 'GitHubSchemaPublishSuccess') { this.success(result.schemaPublish.message); } else { this.error('message' in result.schemaPublish ? result.schemaPublish.message : 'Unknown error'); } } while (result === null); } catch (error) { if (error instanceof core_1.Errors.ExitError) { throw error; } else { this.fail('Failed to publish schema'); this.handleFetchError(error); } } } } SchemaPublish.description = 'publishes schema'; SchemaPublish.flags = { service: core_1.Flags.string({ description: 'service name (only for distributed schemas)', }), url: core_1.Flags.string({ description: 'service url (only for distributed schemas)', }), metadata: core_1.Flags.string({ description: 'additional metadata to attach to the GraphQL schema. This can be a string with a valid JSON, or a path to a file containing a valid JSON', }), 'registry.endpoint': core_1.Flags.string({ description: 'registry endpoint', }), /** @deprecated */ registry: core_1.Flags.string({ description: 'registry address', deprecated: { message: 'use --registry.endpoint instead', version: '0.21.0', }, }), 'registry.accessToken': core_1.Flags.string({ description: 'registry access token', }), /** @deprecated */ token: core_1.Flags.string({ description: 'api token', deprecated: { message: 'use --registry.accessToken instead', version: '0.21.0', }, }), author: core_1.Flags.string({ description: 'author of the change', }), commit: core_1.Flags.string({ description: 'associated commit sha', }), github: core_1.Flags.boolean({ description: 'Connect with GitHub Application', default: false, }), force: core_1.Flags.boolean({ description: 'force publish even on breaking changes', deprecated: { message: '--force is enabled by default for newly created projects', }, }), experimental_acceptBreakingChanges: core_1.Flags.boolean({ description: '(experimental) accept breaking changes and mark schema as valid (only if composable)', deprecated: { message: '--experimental_acceptBreakingChanges is enabled by default for newly created projects', }, }), require: core_1.Flags.string({ description: 'Loads specific require.extensions before running the codegen and reading the configuration', default: [], multiple: true, }), }; SchemaPublish.args = { file: core_1.Args.string({ name: 'file', required: true, description: 'Path to the schema file(s)', hidden: false, }), }; exports.default = SchemaPublish; //# sourceMappingURL=publish.js.map