UNPKG

@theguild/federation-composition

Version:
140 lines (139 loc) 5.57 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.validateSubgraph = validateSubgraph; exports.validate = validate; const constant_case_1 = require("constant-case"); const helpers_js_1 = require("./graphql/helpers.js"); const federation_js_1 = require("./specifications/federation.js"); const state_js_1 = require("./subgraph/state.js"); const validate_subgraph_js_1 = require("./subgraph/validation/validate-subgraph.js"); const state_js_2 = require("./supergraph/state.js"); const validate_supergraph_js_1 = require("./supergraph/validation/validate-supergraph.js"); const numberAtStartRegex = /^\d/; function startsWithNumber(value) { return numberAtStartRegex.test(value); } function buildGraphList(subgraphs) { const errors = []; const graphs = []; const names = new Set(); const idCounter = new Map(); for (const subgraph of subgraphs) { const name = String(subgraph.name); const nameStartsWithNumber = startsWithNumber(name); if (names.has(name)) { throw new Error(`A subgraph named ${name} already exists`); } const { url, typeDefs } = subgraph; names.add(name); let proposedId = (0, constant_case_1.constantCase)(name.replace(/[^A-Z0-9]/gi, '_'), { stripRegexp: /[^A-Z0-9_]+/gi, }); if (nameStartsWithNumber) { proposedId = '_' + proposedId + '_'; } let count = idCounter.get(proposedId); if (typeof count === 'number') { if (count === 1) { graphs.find(g => g.id === proposedId).id += '_1'; } graphs.push({ name, id: proposedId + '_' + (count + 1), url, typeDefs: (0, helpers_js_1.moveSchemaAndDirectiveDefinitionsToTop)(typeDefs), }); idCounter.set(proposedId, count + 1); } else { idCounter.set(proposedId, 1); graphs.push({ name, id: proposedId, url, typeDefs: (0, helpers_js_1.moveSchemaAndDirectiveDefinitionsToTop)(typeDefs), }); } } if (errors.length > 0) { return { success: false, errors, }; } graphs.sort((a, b) => a.id.localeCompare(b.id)); return { success: true, graphs, }; } function validateSubgraph(subgraph) { const subgraphs = [subgraph]; const graphList = buildGraphList(subgraphs); if (!graphList.success) { return graphList.errors; } const corePerSubgraph = graphList.graphs.map(subgraph => (0, validate_subgraph_js_1.validateSubgraphCore)(subgraph)); const coreErrors = corePerSubgraph.map(core => core.errors ?? []).flat(1); if (coreErrors.length > 0) { return coreErrors; } const detectedFederationSpec = new Map(graphList.graphs.map(graph => [graph.id, (0, federation_js_1.detectFederationVersion)(graph.typeDefs)])); const subgraphStateBuilders = new Map(graphList.graphs.map((graph, i) => [ graph.id, (0, state_js_1.createSubgraphStateBuilder)(graph, graph.typeDefs, detectedFederationSpec.get(graph.id).version, corePerSubgraph[i].links ?? []), ])); const subgraphErrors = graphList.graphs .map(graph => (0, validate_subgraph_js_1.validateSubgraph)(graph, subgraphStateBuilders.get(graph.id), detectedFederationSpec.get(graph.id))) .flat(1); return subgraphErrors; } function validate(subgraphs, __internal) { const graphList = buildGraphList(subgraphs); if (!graphList.success) { return { success: false, errors: graphList.errors, }; } const corePerSubgraph = graphList.graphs.map(subgraph => (0, validate_subgraph_js_1.validateSubgraphCore)(subgraph)); const coreErrors = corePerSubgraph.map(core => core.errors ?? []).flat(1); if (coreErrors.length > 0) { return { success: false, errors: coreErrors, }; } const detectedFederationSpec = new Map(graphList.graphs.map(graph => [graph.id, (0, federation_js_1.detectFederationVersion)(graph.typeDefs)])); const subgraphStateBuilders = new Map(graphList.graphs.map((graph, i) => [ graph.id, (0, state_js_1.createSubgraphStateBuilder)(graph, graph.typeDefs, detectedFederationSpec.get(graph.id).version, corePerSubgraph[i].links ?? []), ])); const subgraphErrors = graphList.graphs .map(graph => (0, validate_subgraph_js_1.validateSubgraph)(graph, subgraphStateBuilders.get(graph.id), detectedFederationSpec.get(graph.id), __internal)) .flat(1); if (subgraphErrors.length > 0) { return { success: false, errors: subgraphErrors, }; } const state = (0, state_js_2.createSupergraphStateBuilder)(); const supergraphErrors = (0, validate_supergraph_js_1.validateSupergraph)(new Map(Array.from(subgraphStateBuilders.entries()).map(([id, builder]) => [ id, (0, state_js_1.cleanSubgraphStateFromFederationSpec)((0, state_js_1.cleanSubgraphStateFromLinkSpec)(builder.state)), ])), state, __internal); if (supergraphErrors.length > 0) { return { success: false, errors: supergraphErrors, }; } const nodes = state.build(); return { success: true, supergraph: nodes, links: state.links(), specs: state.getSupergraphState().specs, }; }