@graphql-eslint/eslint-plugin
Version:
GraphQL plugin for ESLint
120 lines (119 loc) • 5.04 kB
JavaScript
import { isScalarType, Kind } from 'graphql';
import { requireGraphQLSchemaFromContext } from '../utils.js';
const RULE_ID = 'relay-arguments';
const MISSING_ARGUMENTS = 'MISSING_ARGUMENTS';
const schema = {
type: 'array',
maxItems: 1,
items: {
type: 'object',
additionalProperties: false,
minProperties: 1,
properties: {
includeBoth: {
type: 'boolean',
default: true,
description: 'Enforce including both forward and backward pagination arguments',
},
},
},
};
export const rule = {
meta: {
type: 'problem',
docs: {
category: 'Schema',
description: [
'Set of rules to follow Relay specification for Arguments.',
'',
'- A field that returns a Connection type must include forward pagination arguments (`first` and `after`), backward pagination arguments (`last` and `before`), or both',
'',
'Forward pagination arguments',
'',
'- `first` takes a non-negative integer',
'- `after` takes the Cursor type',
'',
'Backward pagination arguments',
'',
'- `last` takes a non-negative integer',
'- `before` takes the Cursor type',
].join('\n'),
url: `https://the-guild.dev/graphql/eslint/rules/${RULE_ID}`,
examples: [
{
title: 'Incorrect',
code: /* GraphQL */ `
type User {
posts: PostConnection
}
`,
},
{
title: 'Correct',
code: /* GraphQL */ `
type User {
posts(after: String, first: Int, before: String, last: Int): PostConnection
}
`,
},
],
isDisabledForAllConfig: true,
},
messages: {
[MISSING_ARGUMENTS]: 'A field that returns a Connection type must include forward pagination arguments (`first` and `after`), backward pagination arguments (`last` and `before`), or both.',
},
schema,
},
create(context) {
const schema = requireGraphQLSchemaFromContext(RULE_ID, context);
const { includeBoth = true } = context.options[0] || {};
return {
'FieldDefinition > .gqlType Name[value=/Connection$/]'(node) {
var _a;
let fieldNode = node.parent;
while (fieldNode.kind !== Kind.FIELD_DEFINITION) {
fieldNode = fieldNode.parent;
}
const args = Object.fromEntries(((_a = fieldNode.arguments) === null || _a === void 0 ? void 0 : _a.map(argument => [argument.name.value, argument])) || []);
const hasForwardPagination = !!(args.first && args.after);
const hasBackwardPagination = !!(args.last && args.before);
if (!hasForwardPagination && !hasBackwardPagination) {
context.report({
node: fieldNode.name,
messageId: MISSING_ARGUMENTS,
});
return;
}
function checkField(typeName, argumentName) {
const argument = args[argumentName];
const hasArgument = !!argument;
let type = argument;
if (hasArgument && type.gqlType.kind === Kind.NON_NULL_TYPE) {
type = type.gqlType;
}
const isAllowedNonNullType = hasArgument &&
type.gqlType.kind === Kind.NAMED_TYPE &&
(type.gqlType.name.value === typeName ||
(typeName === 'String' && isScalarType(schema.getType(type.gqlType.name.value))));
if (!isAllowedNonNullType) {
const returnType = typeName === 'String' ? 'String or Scalar' : typeName;
context.report({
node: (argument || fieldNode).name,
message: hasArgument
? `Argument \`${argumentName}\` must return ${returnType}.`
: `Field \`${fieldNode.name.value}\` must contain an argument \`${argumentName}\`, that return ${returnType}.`,
});
}
}
if (includeBoth || args.first || args.after) {
checkField('Int', 'first');
checkField('String', 'after');
}
if (includeBoth || args.last || args.before) {
checkField('Int', 'last');
checkField('String', 'before');
}
},
};
},
};