-
Notifications
You must be signed in to change notification settings - Fork 42
/
Copy pathprefer-web-first-assertions.ts
211 lines (186 loc) · 6.03 KB
/
prefer-web-first-assertions.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
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
import {
dereference,
findParent,
getRawValue,
getStringValue,
isBooleanLiteral,
} from '../utils/ast.js'
import { createRule } from '../utils/createRule.js'
import { parseFnCall } from '../utils/parseFnCall.js'
type MethodConfig = {
inverse?: string
matcher: string
prop?: string
type: 'boolean' | 'string'
}
const methods: Record<string, MethodConfig> = {
getAttribute: {
matcher: 'toHaveAttribute',
type: 'string',
},
innerText: { matcher: 'toHaveText', type: 'string' },
inputValue: { matcher: 'toHaveValue', type: 'string' },
isChecked: {
matcher: 'toBeChecked',
prop: 'checked',
type: 'boolean',
},
isDisabled: {
inverse: 'toBeEnabled',
matcher: 'toBeDisabled',
type: 'boolean',
},
isEditable: { matcher: 'toBeEditable', type: 'boolean' },
isEnabled: {
inverse: 'toBeDisabled',
matcher: 'toBeEnabled',
type: 'boolean',
},
isHidden: {
inverse: 'toBeVisible',
matcher: 'toBeHidden',
type: 'boolean',
},
isVisible: {
inverse: 'toBeHidden',
matcher: 'toBeVisible',
type: 'boolean',
},
textContent: { matcher: 'toHaveText', type: 'string' },
}
const supportedMatchers = new Set([
'toBe',
'toEqual',
'toBeTruthy',
'toBeFalsy',
])
export default createRule({
create(context) {
return {
CallExpression(node) {
const call = parseFnCall(context, node)
if (call?.type !== 'expect') return
const expect = findParent(call.head.node, 'CallExpression')
if (!expect) return
const arg = dereference(context, call.args[0])
if (
!arg ||
arg.type !== 'AwaitExpression' ||
arg.argument.type !== 'CallExpression' ||
arg.argument.callee.type !== 'MemberExpression'
) {
return
}
// Matcher must be supported
if (!supportedMatchers.has(call.matcherName)) return
// Playwright method must be supported
const method = getStringValue(arg.argument.callee.property)
const methodConfig = methods[method]
if (!methodConfig) return
// Change the matcher
const notModifier = call.modifiers.find(
(mod) => getStringValue(mod) === 'not',
)
const isFalsy =
methodConfig.type === 'boolean' &&
((!!call.matcherArgs.length &&
isBooleanLiteral(call.matcherArgs[0], false)) ||
call.matcherName === 'toBeFalsy')
const isInverse = methodConfig.inverse
? notModifier || isFalsy
: notModifier && isFalsy
// Replace the old matcher with the new matcher. The inverse
// matcher should only be used if the old statement was not a
// double negation.
const newMatcher =
(+!!notModifier ^ +isFalsy && methodConfig.inverse) ||
methodConfig.matcher
const { callee } = arg.argument
context.report({
data: {
matcher: newMatcher,
method,
},
fix: (fixer) => {
const methodArgs =
arg.argument.type === 'CallExpression'
? arg.argument.arguments
: []
const methodEnd = methodArgs.length
? methodArgs.at(-1)!.range![1] + 1
: callee.property.range![1] + 2
const fixes = [
// Add await to the expect call
fixer.insertTextBefore(expect, 'await '),
// Remove the await keyword
fixer.replaceTextRange(
[arg.range![0], arg.argument.range![0]],
'',
),
// Remove the old Playwright method and any arguments
fixer.replaceTextRange(
[callee.property.range![0] - 1, methodEnd],
'',
),
]
// Remove not from matcher chain if no longer needed
if (isInverse && notModifier) {
const notRange = notModifier.range!
fixes.push(fixer.removeRange([notRange[0], notRange[1] + 1]))
}
// Add not to the matcher chain if no inverse matcher exists
if (!methodConfig.inverse && !notModifier && isFalsy) {
fixes.push(fixer.insertTextBefore(call.matcher, 'not.'))
}
fixes.push(fixer.replaceText(call.matcher, newMatcher))
// Remove boolean argument if it exists
const [matcherArg] = call.matcherArgs ?? []
if (matcherArg && isBooleanLiteral(matcherArg)) {
fixes.push(fixer.remove(matcherArg))
}
// Add the prop argument if needed
else if (methodConfig.prop && matcherArg) {
const propArg = methodConfig.prop
const variable = getStringValue(matcherArg)
const args = `{ ${propArg}: ${variable} }`
fixes.push(fixer.replaceText(matcherArg, args))
}
// Add the new matcher arguments if needed
const hasOtherArgs = !!methodArgs.filter(
(arg) => !isBooleanLiteral(arg),
).length
if (methodArgs) {
const range = call.matcher.range!
const stringArgs = methodArgs
.map((arg) => getRawValue(arg))
.concat(hasOtherArgs ? '' : [])
.join(', ')
fixes.push(
fixer.insertTextAfterRange(
[range[0], range[1] + 1],
stringArgs,
),
)
}
return fixes
},
messageId: 'useWebFirstAssertion',
node: expect,
})
},
}
},
meta: {
docs: {
category: 'Best Practices',
description: 'Prefer web first assertions',
recommended: true,
url: 'https://github.com/playwright-community/eslint-plugin-playwright/tree/main/docs/rules/prefer-web-first-assertions.md',
},
fixable: 'code',
messages: {
useWebFirstAssertion: 'Replace {{method}}() with {{matcher}}().',
},
type: 'suggestion',
},
})