Skip to content

Commit 3860f1e

Browse files
authoredJul 1, 2024··
fix(no-conditional-expect): Fix false positive with asymmetric matchers (#304)
* Expect parsing * Get it working * Fix up * Tests * Re-org
1 parent 92e9240 commit 3860f1e

5 files changed

+129
-118
lines changed
 

‎src/rules/no-conditional-expect.test.ts

+25-12
Original file line numberDiff line numberDiff line change
@@ -1,22 +1,36 @@
1-
import dedent from 'dedent'
21
import rule from '../../src/rules/no-conditional-expect'
3-
import { runRuleTester } from '../utils/rule-tester'
2+
import { javascript, runRuleTester } from '../utils/rule-tester'
43

54
const messageId = 'conditionalExpect'
65

76
runRuleTester('common tests', rule, {
87
invalid: [],
98
valid: [
10-
`
9+
javascript`
1110
test('foo', () => {
1211
expect(1).toBe(2);
1312
});
1413
`,
15-
`
14+
javascript`
1615
test('foo', () => {
1716
expect(!true).toBe(false);
1817
});
1918
`,
19+
{
20+
code: javascript`
21+
test('foo', () => {
22+
const expected = arr.map((x) => {
23+
if (typeof x === 'string') {
24+
return expect.arrayContaining([x])
25+
} else {
26+
return b
27+
}
28+
})
29+
30+
expect([1, 2, 3]).toEqual(expected);
31+
});
32+
`,
33+
},
2034
],
2135
})
2236

@@ -154,7 +168,7 @@ runRuleTester('logical conditions', rule, {
154168
`,
155169
// Global aliases
156170
{
157-
code: dedent`
171+
code: javascript`
158172
it('foo', () => {
159173
process.env.FAIL && setNum(1);
160174
@@ -331,7 +345,6 @@ runRuleTester('catch conditions', rule, {
331345
code: `
332346
test('foo', () => {
333347
try {
334-
335348
} catch (err) {
336349
expect(err).toMatch('Error');
337350
}
@@ -397,7 +410,7 @@ runRuleTester('catch conditions', rule, {
397410
runRuleTester('promises', rule, {
398411
invalid: [
399412
{
400-
code: dedent`
413+
code: javascript`
401414
test('works', async () => {
402415
await Promise.resolve()
403416
.then(() => { throw new Error('oh noes!'); })
@@ -407,7 +420,7 @@ runRuleTester('promises', rule, {
407420
errors: [{ messageId }],
408421
},
409422
{
410-
code: dedent`
423+
code: javascript`
411424
test('works', async () => {
412425
await Promise.resolve()
413426
.then(() => { throw new Error('oh noes!'); })
@@ -421,7 +434,7 @@ runRuleTester('promises', rule, {
421434
errors: [{ messageId }],
422435
},
423436
{
424-
code: dedent`
437+
code: javascript`
425438
test('works', async () => {
426439
await Promise.resolve()
427440
.catch(error => expect(error).toBeInstanceOf(Error))
@@ -432,7 +445,7 @@ runRuleTester('promises', rule, {
432445
errors: [{ messageId }],
433446
},
434447
{
435-
code: dedent`
448+
code: javascript`
436449
test('works', async () => {
437450
await Promise.resolve()
438451
.catch(error => expect(error).toBeInstanceOf(Error))
@@ -444,7 +457,7 @@ runRuleTester('promises', rule, {
444457
errors: [{ messageId }],
445458
},
446459
{
447-
code: dedent`
460+
code: javascript`
448461
test('works', async () => {
449462
await somePromise
450463
.then(() => { throw new Error('oh noes!'); })
@@ -454,7 +467,7 @@ runRuleTester('promises', rule, {
454467
errors: [{ messageId }],
455468
},
456469
{
457-
code: dedent`
470+
code: javascript`
458471
test('works', async () => {
459472
await somePromise.catch(error => expect(error).toBeInstanceOf(Error));
460473
});

‎src/rules/valid-expect.test.ts

+5
Original file line numberDiff line numberDiff line change
@@ -215,6 +215,11 @@ runRuleTester('valid-expect', rule, {
215215
valid: [
216216
{ code: 'expectPayButtonToBeEnabled()' },
217217
{ code: 'expect("something").toBe("else")' },
218+
{ code: 'expect.anything()' },
219+
{ code: 'expect.arrayContaining()' },
220+
{ code: 'expect.not.arrayContaining()' },
221+
{ code: 'expect.objectContaining(expected)' },
222+
{ code: 'expect.not.objectContaining(expected)' },
218223
{ code: 'expect("something").toBe(expect.anything())' },
219224
{ code: 'expect("something").toEqual({ foo: expect.anything() })' },
220225
{ code: 'expect("something").toBe(expect.arrayContaining([1, 2, 3]))' },

‎src/utils/parseFnCall.test.ts

+13-82
Original file line numberDiff line numberDiff line change
@@ -260,36 +260,6 @@ runRuleTester('expect', rule, {
260260
code: dedent`
261261
import { expect } from '@playwright/test';
262262
263-
expect.assertions();
264-
`,
265-
errors: [
266-
{
267-
column: 1,
268-
data: expectedParsedFnCallResultData({
269-
args: [],
270-
group: 'expect',
271-
head: {
272-
local: 'expect',
273-
node: 'expect',
274-
original: null,
275-
},
276-
matcher: 'assertions',
277-
matcherArgs: [],
278-
matcherName: 'assertions',
279-
members: ['assertions'],
280-
modifiers: [],
281-
name: 'expect',
282-
type: 'expect',
283-
}),
284-
line: 3,
285-
messageId: 'details',
286-
},
287-
],
288-
},
289-
{
290-
code: dedent`
291-
import { expect } from '@playwright/test';
292-
293263
expect(x).toBe(y);
294264
`,
295265
errors: [
@@ -380,58 +350,6 @@ runRuleTester('expect', rule, {
380350
code: dedent`
381351
import { expect } from '@playwright/test';
382352
383-
expect.anything();
384-
expect.not.arrayContaining();
385-
`,
386-
errors: [
387-
{
388-
column: 1,
389-
data: expectedParsedFnCallResultData({
390-
args: [],
391-
group: 'expect',
392-
head: {
393-
local: 'expect',
394-
node: 'expect',
395-
original: null,
396-
},
397-
matcher: 'anything',
398-
matcherArgs: [],
399-
matcherName: 'anything',
400-
members: ['anything'],
401-
modifiers: [],
402-
name: 'expect',
403-
type: 'expect',
404-
}),
405-
line: 3,
406-
messageId: 'details',
407-
},
408-
{
409-
column: 1,
410-
data: expectedParsedFnCallResultData({
411-
args: [],
412-
group: 'expect',
413-
head: {
414-
local: 'expect',
415-
node: 'expect',
416-
original: null,
417-
},
418-
matcher: 'arrayContaining',
419-
matcherArgs: [],
420-
matcherName: 'arrayContaining',
421-
members: ['not', 'arrayContaining'],
422-
modifiers: ['not'],
423-
name: 'expect',
424-
type: 'expect',
425-
}),
426-
line: 4,
427-
messageId: 'details',
428-
},
429-
],
430-
},
431-
{
432-
code: dedent`
433-
import { expect } from '@playwright/test';
434-
435353
expect;
436354
expect(x);
437355
expect(x).toBe;
@@ -1187,5 +1105,18 @@ runTSRuleTester('typescript', rule, {
11871105
},
11881106
"it('is not a function', () => {});",
11891107
'dedent()',
1108+
'expect.assertions()',
1109+
'expect.anything()',
1110+
'expect.arrayContaining()',
1111+
'expect.objectContaining(expected)',
1112+
'expect.not.objectContaining(expected)',
1113+
{
1114+
code: dedent`
1115+
import { expect } from '@playwright/test';
1116+
1117+
expect.assertions();
1118+
expect.anything();
1119+
`,
1120+
},
11901121
],
11911122
})

‎src/utils/parseFnCall.ts

+83-24
Original file line numberDiff line numberDiff line change
@@ -94,21 +94,51 @@ export const isSupportedAccessor = (
9494
): node is AccessorNode =>
9595
isIdentifier(node, value) || isStringNode(node, value)
9696

97-
function getNodeChain(node: ESTree.Node): AccessorNode[] | null {
98-
if (isSupportedAccessor(node)) {
99-
return [node]
97+
class Chain {
98+
#nodes: AccessorNode[] | null = null
99+
#leaves: WeakSet<AccessorNode> = new WeakSet()
100+
101+
constructor(node: ESTree.Node) {
102+
this.#nodes = this.#buildChain(node)
103+
}
104+
105+
isLeaf(node: AccessorNode): boolean {
106+
return this.#leaves.has(node)
100107
}
101108

102-
switch (node.type) {
103-
case 'TaggedTemplateExpression':
104-
return getNodeChain(node.tag)
105-
case 'MemberExpression':
106-
return joinChains(getNodeChain(node.object), getNodeChain(node.property))
107-
case 'CallExpression':
108-
return getNodeChain(node.callee)
109+
get nodes() {
110+
return this.#nodes
109111
}
110112

111-
return null
113+
#buildChain(node: ESTree.Node, insideCall = false): AccessorNode[] | null {
114+
if (isSupportedAccessor(node)) {
115+
// If we are inside a call expression, then the current node is a leaf,
116+
// that is, the end of the sub-chain. For example, in
117+
// `expect.soft(x).not.toBe()`, `soft` and `toBe` are leaves.
118+
if (insideCall) {
119+
this.#leaves.add(node)
120+
}
121+
122+
return [node]
123+
}
124+
125+
switch (node.type) {
126+
case 'TaggedTemplateExpression':
127+
return this.#buildChain(node.tag)
128+
129+
case 'MemberExpression':
130+
return joinChains(
131+
this.#buildChain(node.object),
132+
this.#buildChain(node.property, insideCall),
133+
)
134+
135+
case 'CallExpression':
136+
return this.#buildChain(node.callee, true)
137+
138+
default:
139+
return null
140+
}
141+
}
112142
}
113143

114144
const resolvePossibleAliasedGlobal = (
@@ -166,12 +196,13 @@ function determinePlaywrightFnGroup(name: string): FnGroup {
166196
export const modifiers = new Set(['not', 'resolves', 'rejects'])
167197

168198
const findModifiersAndMatcher = (
199+
chain: Chain,
169200
members: KnownMemberExpressionProperty[],
170-
): ModifiersAndMatcher | string => {
201+
stage: ExpectParseStage,
202+
): ModifiersAndMatcher | string | null => {
171203
const modifiers: KnownMemberExpressionProperty[] = []
172204

173205
for (const member of members) {
174-
// Otherwise, it should be a modifier
175206
const name = getStringValue(member)
176207

177208
if (name === 'soft' || name === 'poll') {
@@ -187,6 +218,12 @@ const findModifiersAndMatcher = (
187218
return 'modifier-unknown'
188219
}
189220
} else if (name !== 'not') {
221+
// If we're in the "modifiers" stage and we find an unknown modifier,
222+
// then it's actually an asymmetric matcher which we don't care about.
223+
if (stage === 'modifiers') {
224+
return null
225+
}
226+
190227
// Check if the member is being called, which means it is the matcher
191228
// (and also the end of the entire "expect" call chain).
192229
if (
@@ -205,6 +242,13 @@ const findModifiersAndMatcher = (
205242
return 'modifier-unknown'
206243
}
207244

245+
// When we find a leaf node, we're done with the modifiers and are moving
246+
// on to the matchers.
247+
if (chain.isLeaf(member)) {
248+
stage = 'matchers'
249+
}
250+
251+
// Add the modifier to the list of modifiers
208252
modifiers.push(member)
209253
}
210254

@@ -263,10 +307,22 @@ export interface ParsedExpectFnCall
263307

264308
export type ParsedFnCall = ParsedGeneralFnCall | ParsedExpectFnCall
265309

310+
type ExpectParseStage = 'matchers' | 'modifiers'
311+
266312
const parseExpectCall = (
313+
chain: Chain,
267314
call: Omit<ParsedFnCall, 'group' | 'type'>,
268-
): ParsedExpectFnCall | string => {
269-
const modifiersAndMatcher = findModifiersAndMatcher(call.members)
315+
stage: ExpectParseStage,
316+
): ParsedExpectFnCall | string | null => {
317+
const modifiersAndMatcher = findModifiersAndMatcher(
318+
chain,
319+
call.members,
320+
stage,
321+
)
322+
323+
if (!modifiersAndMatcher) {
324+
return null
325+
}
270326

271327
if (typeof modifiersAndMatcher === 'string') {
272328
return modifiersAndMatcher
@@ -316,13 +372,10 @@ function parse(
316372
context: Rule.RuleContext,
317373
node: ESTree.CallExpression,
318374
): ParsedFnCall | string | null {
319-
const chain = getNodeChain(node)
320-
321-
if (!chain?.length) {
322-
return null
323-
}
375+
const chain = new Chain(node)
376+
if (!chain.nodes?.length) return null
324377

325-
const [first, ...rest] = chain
378+
const [first, ...rest] = chain.nodes
326379
const resolved = resolveToPlaywrightFn(context, first)
327380
if (!resolved) return null
328381

@@ -355,14 +408,20 @@ function parse(
355408
const group = determinePlaywrightFnGroup(name)
356409

357410
if (group === 'expect') {
411+
let stage: ExpectParseStage = chain.isLeaf(parsedFnCall.head.node)
412+
? 'matchers'
413+
: 'modifiers'
414+
358415
// If using `test.expect` style, the `rest` array will start with `expect`
359416
// and we need to remove it to ensure the chain accurately represents the
360417
// `expect` call chain.
361418
if (isIdentifier(rest[0], 'expect')) {
419+
stage = chain.isLeaf(rest[0]) ? 'matchers' : 'modifiers'
362420
parsedFnCall.members.shift()
363421
}
364422

365-
const result = parseExpectCall(parsedFnCall)
423+
const result = parseExpectCall(chain, parsedFnCall, stage)
424+
if (!result) return null
366425

367426
// If the `expect` call chain is not valid, only report on the topmost node
368427
// since all members in the chain are likely to get flagged for some reason
@@ -384,8 +443,8 @@ function parse(
384443

385444
// Check that every link in the chain except the last is a member expression
386445
if (
387-
chain
388-
.slice(0, chain.length - 1)
446+
chain.nodes
447+
.slice(0, chain.nodes.length - 1)
389448
.some((n) => getParent(n)?.type !== 'MemberExpression')
390449
) {
391450
return null

‎src/utils/rule-tester.ts

+3
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
import dedent from 'dedent'
12
import { RuleTester } from 'eslint'
23
import { describe, it } from 'vitest'
34

@@ -38,3 +39,5 @@ export function runTSRuleTester(...args: Parameters<RuleTester['run']>) {
3839
}
3940

4041
export const test = (input: string) => `test('test', async () => { ${input} })`
42+
43+
export const javascript = dedent

0 commit comments

Comments
 (0)
Please sign in to comment.