UNPKG

@graphql-eslint/eslint-plugin

Version:
191 lines (189 loc) • 6.95 kB
import { Kind, TokenKind } from 'graphql'; import { getLocation, requireGraphQLSchemaFromContext, TYPES_KINDS } from '../utils.js'; import { getRootTypeNames } from '@graphql-tools/utils'; const RULE_ID = 'require-description'; const ALLOWED_KINDS = [ ...TYPES_KINDS, Kind.DIRECTIVE_DEFINITION, Kind.FIELD_DEFINITION, Kind.INPUT_VALUE_DEFINITION, Kind.ENUM_VALUE_DEFINITION, Kind.OPERATION_DEFINITION, ]; function getNodeName(node) { const DisplayNodeNameMap = { [Kind.OBJECT_TYPE_DEFINITION]: 'type', [Kind.INTERFACE_TYPE_DEFINITION]: 'interface', [Kind.ENUM_TYPE_DEFINITION]: 'enum', [Kind.SCALAR_TYPE_DEFINITION]: 'scalar', [Kind.INPUT_OBJECT_TYPE_DEFINITION]: 'input', [Kind.UNION_TYPE_DEFINITION]: 'union', [Kind.DIRECTIVE_DEFINITION]: 'directive', }; switch (node.kind) { case Kind.OBJECT_TYPE_DEFINITION: case Kind.INTERFACE_TYPE_DEFINITION: case Kind.ENUM_TYPE_DEFINITION: case Kind.SCALAR_TYPE_DEFINITION: case Kind.INPUT_OBJECT_TYPE_DEFINITION: case Kind.UNION_TYPE_DEFINITION: return `${DisplayNodeNameMap[node.kind]} ${node.name.value}`; case Kind.DIRECTIVE_DEFINITION: return `${DisplayNodeNameMap[node.kind]} @${node.name.value}`; case Kind.FIELD_DEFINITION: case Kind.INPUT_VALUE_DEFINITION: case Kind.ENUM_VALUE_DEFINITION: return `${node.parent.name.value}.${node.name.value}`; case Kind.OPERATION_DEFINITION: return node.name ? `${node.operation} ${node.name.value}` : node.operation; } } const schema = { type: 'array', minItems: 1, maxItems: 1, items: { type: 'object', additionalProperties: false, minProperties: 1, properties: { types: { type: 'boolean', description: `Includes:\n${TYPES_KINDS.map(kind => `- \`${kind}\``).join('\n')}`, }, rootField: { type: 'boolean', description: 'Definitions within `Query`, `Mutation`, and `Subscription` root types.', }, ...Object.fromEntries([...ALLOWED_KINDS].sort().map(kind => { let description = `Read more about this kind on [spec.graphql.org](https://spec.graphql.org/October2021/#${kind}).`; if (kind === Kind.OPERATION_DEFINITION) { description += '\n> You must use only comment syntax `#` and not description syntax `"""` or `"`.'; } return [kind, { type: 'boolean', description }]; })), }, }, }; export const rule = { meta: { docs: { category: 'Schema', description: 'Enforce descriptions in type definitions and operations.', url: `https://github.com/B2o5T/graphql-eslint/blob/master/docs/rules/${RULE_ID}.md`, examples: [ { title: 'Incorrect', usage: [{ types: true, FieldDefinition: true }], code: /* GraphQL */ ` type someTypeName { name: String } `, }, { title: 'Correct', usage: [{ types: true, FieldDefinition: true }], code: /* GraphQL */ ` """ Some type description """ type someTypeName { """ Name description """ name: String } `, }, { title: 'Correct', usage: [{ OperationDefinition: true }], code: /* GraphQL */ ` # Create a new user mutation createUser { # ... } `, }, { title: 'Correct', usage: [{ rootField: true }], code: /* GraphQL */ ` type Mutation { "Create a new user" createUser: User } type User { name: String } `, }, ], configOptions: [ { types: true, [Kind.DIRECTIVE_DEFINITION]: true, // rootField: true TODO enable in graphql-eslint v4 }, ], recommended: true, }, type: 'suggestion', messages: { [RULE_ID]: 'Description is required for `{{ nodeName }}`.', }, schema, }, create(context) { const { types, rootField, ...restOptions } = context.options[0] || {}; const kinds = new Set(types ? TYPES_KINDS : []); for (const [kind, isEnabled] of Object.entries(restOptions)) { if (isEnabled) { kinds.add(kind); } else { kinds.delete(kind); } } if (rootField) { const schema = requireGraphQLSchemaFromContext(RULE_ID, context); const rootTypeNames = getRootTypeNames(schema); kinds.add(`:matches(ObjectTypeDefinition, ObjectTypeExtension)[name.value=/^(${[ ...rootTypeNames, ].join(',')})$/] > FieldDefinition`); } const selector = [...kinds].join(','); return { [selector](node) { var _a; let description = ''; const isOperation = node.kind === Kind.OPERATION_DEFINITION; if (isOperation) { const rawNode = node.rawNode(); const { prev, line } = rawNode.loc.startToken; if ((prev === null || prev === void 0 ? void 0 : prev.kind) === TokenKind.COMMENT) { const value = prev.value.trim(); // TODO: remove `!` when drop support of graphql@15 const linesBefore = line - prev.line; if (!value.startsWith('eslint') && linesBefore === 1) { description = value; } } } else { description = ((_a = node.description) === null || _a === void 0 ? void 0 : _a.value.trim()) || ''; } if (description.length === 0) { context.report({ loc: isOperation ? getLocation(node.loc.start, node.operation) : node.name.loc, messageId: RULE_ID, data: { nodeName: getNodeName(node), }, }); } }, }; }, };