UNPKG

@graphql-eslint/eslint-plugin

Version:
133 lines (132 loc) 4.08 kB
import { Kind } from "graphql"; import depthLimit from "graphql-depth-limit"; import { ARRAY_DEFAULT_OPTIONS, logger, requireGraphQLOperations } from "../../utils.js"; const RULE_ID = "selection-set-depth", schema = { type: "array", minItems: 1, maxItems: 1, items: { type: "object", additionalProperties: !1, required: ["maxDepth"], properties: { maxDepth: { type: "number" }, ignore: ARRAY_DEFAULT_OPTIONS } } }, rule = { meta: { type: "suggestion", hasSuggestions: !0, docs: { category: "Operations", description: "Limit the complexity of the GraphQL operations solely by their depth. Based on [graphql-depth-limit](https://npmjs.com/package/graphql-depth-limit).", url: `https://the-guild.dev/graphql/eslint/rules/${RULE_ID}`, requiresSiblings: !0, examples: [ { title: "Incorrect", usage: [{ maxDepth: 1 }], code: ` query deep2 { viewer { # Level 0 albums { # Level 1 title # Level 2 } } } ` }, { title: "Correct", usage: [{ maxDepth: 4 }], code: ` query deep2 { viewer { # Level 0 albums { # Level 1 title # Level 2 } } } ` }, { title: "Correct (ignored field)", usage: [{ maxDepth: 1, ignore: ["albums"] }], code: ` query deep2 { viewer { # Level 0 albums { # Level 1 title # Level 2 } } } ` } ], recommended: !0, configOptions: [{ maxDepth: 7 }] }, schema }, create(context) { let siblings = null; try { siblings = requireGraphQLOperations(RULE_ID, context); } catch { logger.warn( `Rule "${RULE_ID}" works best with siblings operations loaded. See https://the-guild.dev/graphql/eslint/docs/usage#providing-operations for more info` ); } const { maxDepth, ignore = [] } = context.options[0], checkFn = depthLimit(maxDepth, { ignore }); return { "OperationDefinition, FragmentDefinition"(node) { try { const rawNode = node.rawNode(), fragmentsInUse = siblings ? siblings.getFragmentsInUse(rawNode) : [], document = { kind: Kind.DOCUMENT, definitions: [rawNode, ...fragmentsInUse] }; checkFn({ getDocument: () => document, reportError(error) { const { line, column } = error.locations[0], token = context.sourceCode.getAncestors(node)[0].tokens.find( (token2) => token2.loc.start.line === line && token2.loc.start.column === column - 1 ); context.report({ loc: { line, column: column - 1 }, message: error.message, // Don't provide suggestions for fragment that can be in a separate file ...token && { suggest: [ { desc: "Remove selections", fix(fixer) { const foundNode = context.getSourceCode().getNodeByRangeIndex(token.range[0]), parentNode = foundNode.parent.parent; return fixer.remove( foundNode.kind === "Name" ? parentNode.parent : parentNode ); } } ] } }); } }); } catch (e) { logger.warn( `Rule "${RULE_ID}" check failed due to a missing siblings operations. See https://the-guild.dev/graphql/eslint/docs/usage#providing-operations for more info`, e ); } } }; } }; export { rule };