@graphql-hive/cli
Version:
A CLI util to manage and control your GraphQL Hive
342 lines • 13.3 kB
JavaScript
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
;