UNPKG

@graphql-hive/cli

Version:

A CLI util to manage and control your GraphQL Hive

139 lines 5.45 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.renderErrors = renderErrors; exports.renderChanges = renderChanges; exports.renderWarnings = renderWarnings; exports.loadSchema = loadSchema; exports.minifySchema = minifySchema; const tslib_1 = require("tslib"); const colors_1 = tslib_1.__importDefault(require("colors")); const graphql_1 = require("graphql"); const code_file_loader_1 = require("@graphql-tools/code-file-loader"); const graphql_file_loader_1 = require("@graphql-tools/graphql-file-loader"); const json_file_loader_1 = require("@graphql-tools/json-file-loader"); const load_1 = require("@graphql-tools/load"); const url_loader_1 = require("@graphql-tools/url-loader"); const base_command_1 = require("../base-command"); const gql_1 = require("../gql"); const graphql_2 = require("../gql/graphql"); const indent = ' '; const criticalityMap = { [graphql_2.CriticalityLevel.Breaking]: colors_1.default.red('-'), [graphql_2.CriticalityLevel.Safe]: colors_1.default.green('-'), [graphql_2.CriticalityLevel.Dangerous]: colors_1.default.green('-'), }; function renderErrors(errors) { this.fail(`Detected ${errors.total} error${errors.total > 1 ? 's' : ''}`); this.log(''); errors.nodes.forEach(error => { this.log(String(indent), colors_1.default.red('-'), this.bolderize(error.message)); }); } const RenderChanges_SchemaChanges = (0, gql_1.graphql)(` fragment RenderChanges_schemaChanges on SchemaChangeConnection { total nodes { criticality isSafeBasedOnUsage message(withSafeBasedOnUsageNote: false) approval { approvedBy { displayName } } } } `); function renderChanges(maskedChanges) { const changes = (0, gql_1.useFragment)(RenderChanges_SchemaChanges, maskedChanges); const writeChanges = (changes) => { changes.forEach(change => { var _a, _b; const messageParts = [ String(indent), criticalityMap[change.isSafeBasedOnUsage ? graphql_2.CriticalityLevel.Safe : change.criticality], this.bolderize(change.message), ]; if (change.isSafeBasedOnUsage) { messageParts.push(colors_1.default.green('(Safe based on usage ✓)')); } if (change.approval) { messageParts.push(colors_1.default.green(`(Approved by ${(_b = (_a = change.approval.approvedBy) === null || _a === void 0 ? void 0 : _a.displayName) !== null && _b !== void 0 ? _b : '<unknown>'} ✓)`)); } this.log(...messageParts); }); }; this.info(`Detected ${changes.total} change${changes.total > 1 ? 's' : ''}`); this.log(''); const breakingChanges = changes.nodes.filter(change => change.criticality === graphql_2.CriticalityLevel.Breaking); const safeChanges = changes.nodes.filter(change => change.criticality !== graphql_2.CriticalityLevel.Breaking); if (breakingChanges.length) { this.log(String(indent), `Breaking changes:`); writeChanges(breakingChanges); } if (safeChanges.length) { this.log(String(indent), `Safe changes:`); writeChanges(safeChanges); } } function renderWarnings(warnings) { this.log(''); this.infoWarning(`Detected ${warnings.total} warning${warnings.total > 1 ? 's' : ''}`); this.log(''); warnings.nodes.forEach(warning => { const details = [warning.source ? `source: ${this.bolderize(warning.source)}` : undefined] .filter(Boolean) .join(', '); this.log(indent, `- ${this.bolderize(warning.message)}${details ? ` (${details})` : ''}`); }); } async function loadSchema( /** * This is used to determine the correct loader to use. * * If a user is simply introspecting a schema, the 'introspection' should be used. * In case of federation, we should skip the UrlLoader, * because it will try to introspect the schema * instead of fetching the SDL with directives. */ intent, file, options) { const loaders = [new code_file_loader_1.CodeFileLoader(), new graphql_file_loader_1.GraphQLFileLoader(), new json_file_loader_1.JsonFileLoader()]; if (intent === 'federation-subgraph-introspection') { loaders.push(new FederationSubgraphUrlLoader()); } else { loaders.push(new url_loader_1.UrlLoader()); } const sources = await (0, load_1.loadTypedefs)(file, Object.assign(Object.assign({}, options), { cwd: process.cwd(), loaders })); return (0, graphql_1.print)((0, graphql_1.concatAST)(sources.map(s => s.document))); } function minifySchema(schema) { return schema.replace(/\s+/g, ' ').trim(); } class FederationSubgraphUrlLoader { async load(pointer) { if (!pointer.startsWith('http://') && !pointer.startsWith('https://')) { return null; } const response = await (0, base_command_1.graphqlRequest)({ endpoint: pointer, }).request({ operation: (0, graphql_1.parse)(` query GetFederationSchema { _service { sdl } } `), variables: undefined, }); const sdl = minifySchema(response._service.sdl); return [ { document: (0, graphql_1.parse)(sdl), rawSDL: sdl, }, ]; } } //# sourceMappingURL=schema.js.map