@graphql-hive/cli
Version:
A CLI util to manage and control your GraphQL Hive
311 lines • 11.5 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const tslib_1 = require("tslib");
const node_fs_1 = tslib_1.__importDefault(require("node:fs"));
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 errors_1 = require("../../helpers/errors");
const git_1 = require("../../helpers/git");
const schema_1 = require("../../helpers/schema");
const TargetInput = tslib_1.__importStar(require("../../helpers/target-input"));
const schemaCheckMutation = (0, gql_1.graphql)(/* GraphQL */ `
mutation schemaCheck($input: SchemaCheckInput!) {
schemaCheck(input: $input) {
__typename
... on SchemaCheckSuccess {
valid
initial
warnings {
nodes {
message
source
line
column
}
total
}
changes {
nodes {
message(withSafeBasedOnUsageNote: false)
criticality
isSafeBasedOnUsage
approval {
approvedBy {
id
displayName
}
}
}
total
...RenderChanges_schemaChanges
}
schemaCheck {
webUrl
}
}
... on SchemaCheckError {
valid
changes {
nodes {
message(withSafeBasedOnUsageNote: false)
criticality
isSafeBasedOnUsage
}
total
...RenderChanges_schemaChanges
}
warnings {
nodes {
message
source
line
column
}
total
}
errors {
nodes {
message
}
total
}
schemaCheck {
webUrl
}
}
... on GitHubSchemaCheckSuccess {
message
}
... on GitHubSchemaCheckError {
message
}
}
}
`);
class SchemaCheck extends base_command_1.default {
async run() {
var _a, _b, _c;
try {
const { flags, args } = await this.parse(SchemaCheck);
await this.require(flags);
let target = null;
if (flags.target) {
const result = TargetInput.parse(flags.target);
if (result.type === 'error') {
throw new errors_1.InvalidTargetError();
}
target = result.data;
}
const service = flags.service;
const forceSafe = flags.forceSafe;
const usesGitHubApp = flags.github === true;
let endpoint, accessToken;
try {
endpoint = this.ensure({
key: 'registry.endpoint',
args: flags,
legacyFlagName: 'registry',
defaultValue: config_1.graphqlEndpoint,
env: 'HIVE_REGISTRY',
description: SchemaCheck.flags['registry.endpoint'].description,
});
}
catch (e) {
throw new errors_1.MissingEndpointError();
}
const file = args.file;
try {
accessToken = this.ensure({
key: 'registry.accessToken',
args: flags,
legacyFlagName: 'token',
env: 'HIVE_TOKEN',
description: SchemaCheck.flags['registry.accessToken'].description,
});
}
catch (e) {
throw new errors_1.MissingRegistryTokenError();
}
const sdl = await (0, schema_1.loadSchema)(file).catch(e => {
throw new errors_1.SchemaFileNotFoundError(file, e);
});
const git = await (0, git_1.gitInfo)(() => {
// noop
});
const commit = flags.commit || (git === null || git === void 0 ? void 0 : git.commit);
const author = flags.author || (git === null || git === void 0 ? void 0 : git.author);
if (typeof sdl !== 'string' || sdl.length === 0) {
throw new errors_1.SchemaFileEmptyError(file);
}
let github = null;
if (usesGitHubApp) {
if (!commit) {
throw new errors_1.GithubCommitRequiredError();
}
if (!git.repository) {
throw new errors_1.GithubRepositoryRequiredError();
}
if (!git.pullRequestNumber) {
this.warn("Could not resolve pull request number. Are you running this command on a 'pull_request' event?\n" +
'See https://the-guild.dev/graphql/hive/docs/other-integrations/ci-cd#github-workflow-for-ci');
}
github = {
commit: commit,
repository: git.repository,
pullRequestNumber: git.pullRequestNumber,
};
}
const result = await this.registryApi(endpoint, accessToken).request({
operation: schemaCheckMutation,
variables: {
input: {
service,
sdl: (0, schema_1.minifySchema)(sdl),
github,
meta: !!commit && !!author
? {
commit,
author,
}
: null,
contextId: (_a = flags.contextId) !== null && _a !== void 0 ? _a : undefined,
target,
url: flags.url,
},
},
});
if (flags.experimentalJsonFile) {
node_fs_1.default.writeFileSync(flags.experimentalJsonFile, JSON.stringify(result, null, 2));
}
if (result.schemaCheck.__typename === 'SchemaCheckSuccess') {
const changes = result.schemaCheck.changes;
if (result.schemaCheck.initial) {
this.logSuccess('Schema registry is empty, nothing to compare your schema with.');
}
else if (!(changes === null || changes === void 0 ? void 0 : changes.total)) {
this.logSuccess('No changes');
}
else {
this.log((0, schema_1.renderChanges)(changes));
}
const warnings = result.schemaCheck.warnings;
if (warnings === null || warnings === void 0 ? void 0 : warnings.total) {
this.log((0, schema_1.renderWarnings)(warnings));
}
if ((_b = result.schemaCheck.schemaCheck) === null || _b === void 0 ? void 0 : _b.webUrl) {
this.log(`View full report:\n${result.schemaCheck.schemaCheck.webUrl}`);
}
}
else if (result.schemaCheck.__typename === 'SchemaCheckError') {
const changes = result.schemaCheck.changes;
const errors = result.schemaCheck.errors;
const warnings = result.schemaCheck.warnings;
this.log((0, schema_1.renderErrors)(errors));
if (warnings === null || warnings === void 0 ? void 0 : warnings.total) {
this.log((0, schema_1.renderWarnings)(warnings));
}
if (changes && changes.total) {
this.log((0, schema_1.renderChanges)(changes));
}
if ((_c = result.schemaCheck.schemaCheck) === null || _c === void 0 ? void 0 : _c.webUrl) {
this.log(`View full report:\n${result.schemaCheck.schemaCheck.webUrl}`);
}
this.log('');
if (forceSafe) {
this.logSuccess('Breaking changes were expected (forced)');
}
else {
this.exit(1);
}
}
else if (result.schemaCheck.__typename === 'GitHubSchemaCheckSuccess') {
this.logSuccess(result.schemaCheck.message);
}
else {
throw new errors_1.APIError(result.schemaCheck.message);
}
}
catch (error) {
if (error instanceof core_1.Errors.CLIError) {
throw error;
}
else {
this.logFailure('Failed to check schema');
throw new errors_1.UnexpectedError(error);
}
}
}
}
SchemaCheck.description = 'checks schema';
SchemaCheck.flags = {
service: core_1.Flags.string({
description: 'service name (only for distributed schemas)',
}),
'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',
},
}),
experimentalJsonFile: core_1.Flags.string({
name: 'experimental-json-file',
description: "File path to output a JSON file containing the command's result. Useful for e.g. CI scripting with `jq`.",
}),
forceSafe: core_1.Flags.boolean({
description: 'mark the check as safe, breaking changes are expected',
}),
github: core_1.Flags.boolean({
description: 'Connect with GitHub Application',
default: false,
}),
require: core_1.Flags.string({
description: 'Loads specific require.extensions before running the codegen and reading the configuration',
default: [],
multiple: true,
}),
author: core_1.Flags.string({
description: 'Author of the change',
}),
commit: core_1.Flags.string({
description: 'Associated commit sha',
}),
contextId: core_1.Flags.string({
description: 'Context ID for grouping the schema check.',
}),
target: core_1.Flags.string({
description: 'The target against which to check the schema (slug or ID).' +
' This can either be a slug following the format "$organizationSlug/$projectSlug/$targetSlug" (e.g "the-guild/graphql-hive/staging")' +
' or an UUID (e.g. "a0f4c605-6541-4350-8cfe-b31f21a4bf80").',
}),
url: core_1.Flags.string({
description: 'If checking a service, then you can optionally provide the service URL to see the difference in the supergraph during the check.',
}),
};
SchemaCheck.args = {
file: core_1.Args.string({
name: 'file',
required: true,
description: 'Path to the schema file(s)',
hidden: false,
}),
};
exports.default = SchemaCheck;
//# sourceMappingURL=check.js.map