2020-08-25 23:57:08 +00:00
|
|
|
"use strict";
|
|
|
|
|
|
|
|
Object.defineProperty(exports, "__esModule", {
|
|
|
|
value: true
|
|
|
|
});
|
|
|
|
exports.default = void 0;
|
2022-11-10 10:43:16 +00:00
|
|
|
var _utils = require("@typescript-eslint/utils");
|
|
|
|
var _utils2 = require("./utils");
|
2020-08-25 23:57:08 +00:00
|
|
|
const trimFXprefix = word => ['f', 'x'].includes(word.charAt(0)) ? word.substr(1) : word;
|
|
|
|
const doesBinaryExpressionContainStringNode = binaryExp => {
|
2022-11-10 10:43:16 +00:00
|
|
|
if ((0, _utils2.isStringNode)(binaryExp.right)) {
|
2020-08-25 23:57:08 +00:00
|
|
|
return true;
|
|
|
|
}
|
2022-11-10 10:43:16 +00:00
|
|
|
if (binaryExp.left.type === _utils.AST_NODE_TYPES.BinaryExpression) {
|
2020-08-25 23:57:08 +00:00
|
|
|
return doesBinaryExpressionContainStringNode(binaryExp.left);
|
|
|
|
}
|
2022-11-10 10:43:16 +00:00
|
|
|
return (0, _utils2.isStringNode)(binaryExp.left);
|
|
|
|
};
|
|
|
|
const quoteStringValue = node => node.type === _utils.AST_NODE_TYPES.TemplateLiteral ? `\`${node.quasis[0].value.raw}\`` : node.raw;
|
|
|
|
const compileMatcherPattern = matcherMaybeWithMessage => {
|
|
|
|
const [matcher, message] = Array.isArray(matcherMaybeWithMessage) ? matcherMaybeWithMessage : [matcherMaybeWithMessage];
|
|
|
|
return [new RegExp(matcher, 'u'), message];
|
2020-08-25 23:57:08 +00:00
|
|
|
};
|
|
|
|
const compileMatcherPatterns = matchers => {
|
2022-11-10 10:43:16 +00:00
|
|
|
if (typeof matchers === 'string' || Array.isArray(matchers)) {
|
|
|
|
const compiledMatcher = compileMatcherPattern(matchers);
|
2020-08-25 23:57:08 +00:00
|
|
|
return {
|
2022-11-10 10:43:16 +00:00
|
|
|
describe: compiledMatcher,
|
|
|
|
test: compiledMatcher,
|
|
|
|
it: compiledMatcher
|
2020-08-25 23:57:08 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
return {
|
2022-11-10 10:43:16 +00:00
|
|
|
describe: matchers.describe ? compileMatcherPattern(matchers.describe) : null,
|
|
|
|
test: matchers.test ? compileMatcherPattern(matchers.test) : null,
|
|
|
|
it: matchers.it ? compileMatcherPattern(matchers.it) : null
|
2020-08-25 23:57:08 +00:00
|
|
|
};
|
|
|
|
};
|
2022-11-10 10:43:16 +00:00
|
|
|
const MatcherAndMessageSchema = {
|
|
|
|
type: 'array',
|
|
|
|
items: {
|
|
|
|
type: 'string'
|
|
|
|
},
|
|
|
|
minItems: 1,
|
|
|
|
maxItems: 2,
|
|
|
|
additionalItems: false
|
|
|
|
};
|
|
|
|
var _default = (0, _utils2.createRule)({
|
2020-08-25 23:57:08 +00:00
|
|
|
name: __filename,
|
|
|
|
meta: {
|
|
|
|
docs: {
|
|
|
|
category: 'Best Practices',
|
|
|
|
description: 'Enforce valid titles',
|
2021-02-26 03:58:33 +00:00
|
|
|
recommended: 'error'
|
2020-08-25 23:57:08 +00:00
|
|
|
},
|
|
|
|
messages: {
|
|
|
|
titleMustBeString: 'Title must be a string',
|
|
|
|
emptyTitle: '{{ jestFunctionName }} should not have an empty title',
|
|
|
|
duplicatePrefix: 'should not have duplicate prefix',
|
|
|
|
accidentalSpace: 'should not have leading or trailing spaces',
|
|
|
|
disallowedWord: '"{{ word }}" is not allowed in test titles.',
|
|
|
|
mustNotMatch: '{{ jestFunctionName }} should not match {{ pattern }}',
|
2022-11-10 10:43:16 +00:00
|
|
|
mustMatch: '{{ jestFunctionName }} should match {{ pattern }}',
|
|
|
|
mustNotMatchCustom: '{{ message }}',
|
|
|
|
mustMatchCustom: '{{ message }}'
|
2020-08-25 23:57:08 +00:00
|
|
|
},
|
|
|
|
type: 'suggestion',
|
|
|
|
schema: [{
|
|
|
|
type: 'object',
|
|
|
|
properties: {
|
|
|
|
ignoreTypeOfDescribeName: {
|
|
|
|
type: 'boolean',
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
disallowedWords: {
|
|
|
|
type: 'array',
|
|
|
|
items: {
|
|
|
|
type: 'string'
|
|
|
|
}
|
2022-11-10 10:43:16 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
patternProperties: {
|
|
|
|
[/^must(?:Not)?Match$/u.source]: {
|
2020-08-25 23:57:08 +00:00
|
|
|
oneOf: [{
|
|
|
|
type: 'string'
|
2022-11-10 10:43:16 +00:00
|
|
|
}, MatcherAndMessageSchema, {
|
2020-08-25 23:57:08 +00:00
|
|
|
type: 'object',
|
2022-11-10 10:43:16 +00:00
|
|
|
propertyNames: {
|
|
|
|
enum: ['describe', 'test', 'it']
|
2020-08-25 23:57:08 +00:00
|
|
|
},
|
2022-11-10 10:43:16 +00:00
|
|
|
additionalProperties: {
|
|
|
|
oneOf: [{
|
2020-08-25 23:57:08 +00:00
|
|
|
type: 'string'
|
2022-11-10 10:43:16 +00:00
|
|
|
}, MatcherAndMessageSchema]
|
|
|
|
}
|
2020-08-25 23:57:08 +00:00
|
|
|
}]
|
|
|
|
}
|
|
|
|
},
|
|
|
|
additionalProperties: false
|
|
|
|
}],
|
|
|
|
fixable: 'code'
|
|
|
|
},
|
|
|
|
defaultOptions: [{
|
|
|
|
ignoreTypeOfDescribeName: false,
|
|
|
|
disallowedWords: []
|
|
|
|
}],
|
|
|
|
create(context, [{
|
|
|
|
ignoreTypeOfDescribeName,
|
|
|
|
disallowedWords = [],
|
|
|
|
mustNotMatch,
|
|
|
|
mustMatch
|
|
|
|
}]) {
|
|
|
|
const disallowedWordsRegexp = new RegExp(`\\b(${disallowedWords.join('|')})\\b`, 'iu');
|
2022-11-10 10:43:16 +00:00
|
|
|
const mustNotMatchPatterns = compileMatcherPatterns(mustNotMatch ?? {});
|
|
|
|
const mustMatchPatterns = compileMatcherPatterns(mustMatch ?? {});
|
2020-08-25 23:57:08 +00:00
|
|
|
return {
|
|
|
|
CallExpression(node) {
|
2022-11-10 10:43:16 +00:00
|
|
|
const jestFnCall = (0, _utils2.parseJestFnCall)(node, context);
|
|
|
|
if ((jestFnCall === null || jestFnCall === void 0 ? void 0 : jestFnCall.type) !== 'describe' && (jestFnCall === null || jestFnCall === void 0 ? void 0 : jestFnCall.type) !== 'test') {
|
2020-08-25 23:57:08 +00:00
|
|
|
return;
|
|
|
|
}
|
2022-11-10 10:43:16 +00:00
|
|
|
const [argument] = node.arguments;
|
2020-08-25 23:57:08 +00:00
|
|
|
if (!argument) {
|
|
|
|
return;
|
|
|
|
}
|
2022-11-10 10:43:16 +00:00
|
|
|
if (!(0, _utils2.isStringNode)(argument)) {
|
|
|
|
if (argument.type === _utils.AST_NODE_TYPES.BinaryExpression && doesBinaryExpressionContainStringNode(argument)) {
|
2020-08-25 23:57:08 +00:00
|
|
|
return;
|
|
|
|
}
|
2022-11-10 10:43:16 +00:00
|
|
|
if (argument.type !== _utils.AST_NODE_TYPES.TemplateLiteral && !(ignoreTypeOfDescribeName && jestFnCall.type === 'describe')) {
|
2020-08-25 23:57:08 +00:00
|
|
|
context.report({
|
|
|
|
messageId: 'titleMustBeString',
|
|
|
|
loc: argument.loc
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2022-11-10 10:43:16 +00:00
|
|
|
const title = (0, _utils2.getStringValue)(argument);
|
2020-08-25 23:57:08 +00:00
|
|
|
if (!title) {
|
|
|
|
context.report({
|
|
|
|
messageId: 'emptyTitle',
|
|
|
|
data: {
|
2022-11-10 10:43:16 +00:00
|
|
|
jestFunctionName: jestFnCall.type === 'describe' ? _utils2.DescribeAlias.describe : _utils2.TestCaseName.test
|
2020-08-25 23:57:08 +00:00
|
|
|
},
|
|
|
|
node
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (disallowedWords.length > 0) {
|
|
|
|
const disallowedMatch = disallowedWordsRegexp.exec(title);
|
|
|
|
if (disallowedMatch) {
|
|
|
|
context.report({
|
|
|
|
data: {
|
|
|
|
word: disallowedMatch[1]
|
|
|
|
},
|
|
|
|
messageId: 'disallowedWord',
|
|
|
|
node: argument
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (title.trim().length !== title.length) {
|
|
|
|
context.report({
|
|
|
|
messageId: 'accidentalSpace',
|
|
|
|
node: argument,
|
|
|
|
fix: fixer => [fixer.replaceTextRange(argument.range, quoteStringValue(argument).replace(/^([`'"]) +?/u, '$1').replace(/ +?([`'"])$/u, '$1'))]
|
|
|
|
});
|
|
|
|
}
|
2022-11-10 10:43:16 +00:00
|
|
|
const unprefixedName = trimFXprefix(jestFnCall.name);
|
2020-08-25 23:57:08 +00:00
|
|
|
const [firstWord] = title.split(' ');
|
2022-11-10 10:43:16 +00:00
|
|
|
if (firstWord.toLowerCase() === unprefixedName) {
|
2020-08-25 23:57:08 +00:00
|
|
|
context.report({
|
|
|
|
messageId: 'duplicatePrefix',
|
|
|
|
node: argument,
|
|
|
|
fix: fixer => [fixer.replaceTextRange(argument.range, quoteStringValue(argument).replace(/^([`'"]).+? /u, '$1'))]
|
|
|
|
});
|
|
|
|
}
|
2022-11-10 10:43:16 +00:00
|
|
|
const jestFunctionName = unprefixedName;
|
|
|
|
const [mustNotMatchPattern, mustNotMatchMessage] = mustNotMatchPatterns[jestFunctionName] ?? [];
|
2020-08-25 23:57:08 +00:00
|
|
|
if (mustNotMatchPattern) {
|
|
|
|
if (mustNotMatchPattern.test(title)) {
|
|
|
|
context.report({
|
2022-11-10 10:43:16 +00:00
|
|
|
messageId: mustNotMatchMessage ? 'mustNotMatchCustom' : 'mustNotMatch',
|
2020-08-25 23:57:08 +00:00
|
|
|
node: argument,
|
|
|
|
data: {
|
|
|
|
jestFunctionName,
|
2022-11-10 10:43:16 +00:00
|
|
|
pattern: mustNotMatchPattern,
|
|
|
|
message: mustNotMatchMessage
|
2020-08-25 23:57:08 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2022-11-10 10:43:16 +00:00
|
|
|
const [mustMatchPattern, mustMatchMessage] = mustMatchPatterns[jestFunctionName] ?? [];
|
2020-08-25 23:57:08 +00:00
|
|
|
if (mustMatchPattern) {
|
|
|
|
if (!mustMatchPattern.test(title)) {
|
|
|
|
context.report({
|
2022-11-10 10:43:16 +00:00
|
|
|
messageId: mustMatchMessage ? 'mustMatchCustom' : 'mustMatch',
|
2020-08-25 23:57:08 +00:00
|
|
|
node: argument,
|
|
|
|
data: {
|
|
|
|
jestFunctionName,
|
2022-11-10 10:43:16 +00:00
|
|
|
pattern: mustMatchPattern,
|
|
|
|
message: mustMatchMessage
|
2020-08-25 23:57:08 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
});
|
|
|
|
exports.default = _default;
|