/
no-restricted-matchers.ts
77 lines (70 loc) · 1.84 KB
/
no-restricted-matchers.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
import {
ModifierName,
createRule,
getAccessorValue,
parseJestFnCall,
} from './utils';
const isChainRestricted = (chain: string, restriction: string): boolean => {
if (
ModifierName.hasOwnProperty(restriction) ||
restriction.endsWith('.not')
) {
return chain.startsWith(restriction);
}
return chain === restriction;
};
export default createRule<
[Record<string, string | null>],
'restrictedChain' | 'restrictedChainWithMessage'
>({
name: __filename,
meta: {
docs: {
category: 'Best Practices',
description: 'Disallow specific matchers & modifiers',
recommended: false,
},
type: 'suggestion',
schema: [
{
type: 'object',
additionalProperties: {
type: ['string', 'null'],
},
},
],
messages: {
restrictedChain: 'Use of `{{ restriction }}` is disallowed',
restrictedChainWithMessage: '{{ message }}',
},
},
defaultOptions: [{}],
create(context, [restrictedChains]) {
return {
CallExpression(node) {
const jestFnCall = parseJestFnCall(node, context);
if (jestFnCall?.type !== 'expect') {
return;
}
const chain = jestFnCall.members
.map(nod => getAccessorValue(nod))
.join('.');
for (const [restriction, message] of Object.entries(restrictedChains)) {
if (isChainRestricted(chain, restriction)) {
context.report({
messageId: message
? 'restrictedChainWithMessage'
: 'restrictedChain',
data: { message, restriction },
loc: {
start: jestFnCall.members[0].loc.start,
end: jestFnCall.members[jestFnCall.members.length - 1].loc.end,
},
});
break;
}
}
},
};
},
});