UNPKG

@graphql-inspector/core

Version:

Tooling for GraphQL. Compare GraphQL Schemas, check documents, find breaking changes, find similar types.

38 lines (37 loc) 1.75 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.validateComplexity = validateComplexity; exports.calculateOperationComplexity = calculateOperationComplexity; const graphql_1 = require("graphql"); function validateComplexity({ source, doc, maxComplexityScore, config, fragmentGraph, }) { const getFragmentByFragmentName = (fragmentName) => fragmentGraph.getNodeData(fragmentName); for (const definition of doc.definitions) { if (definition.kind !== graphql_1.Kind.OPERATION_DEFINITION) { continue; } const complexityScore = calculateOperationComplexity(definition, config, getFragmentByFragmentName); if (complexityScore > maxComplexityScore) { return new graphql_1.GraphQLError(`Too high complexity score (${complexityScore}). Maximum allowed is ${maxComplexityScore}`, [definition], source, definition.loc?.start ? [definition.loc.start] : undefined); } } } function calculateOperationComplexity(node, config, getFragmentByName, depth = 0) { let cost = config.scalarCost; if ('selectionSet' in node && node.selectionSet) { cost = config.objectCost; for (const child of node.selectionSet.selections) { cost += config.depthCostFactor * calculateOperationComplexity(child, config, getFragmentByName, depth + 1); } } if (node.kind === graphql_1.Kind.FRAGMENT_SPREAD) { const fragment = getFragmentByName(node.name.value); if (fragment) { cost += config.depthCostFactor * calculateOperationComplexity(fragment, config, getFragmentByName, depth + 1); } } return cost; }