@@ -94,21 +94,51 @@ export const isSupportedAccessor = (
94
94
) : node is AccessorNode =>
95
95
isIdentifier ( node , value ) || isStringNode ( node , value )
96
96
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 )
100
107
}
101
108
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
109
111
}
110
112
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
+ }
112
142
}
113
143
114
144
const resolvePossibleAliasedGlobal = (
@@ -166,12 +196,13 @@ function determinePlaywrightFnGroup(name: string): FnGroup {
166
196
export const modifiers = new Set ( [ 'not' , 'resolves' , 'rejects' ] )
167
197
168
198
const findModifiersAndMatcher = (
199
+ chain : Chain ,
169
200
members : KnownMemberExpressionProperty [ ] ,
170
- ) : ModifiersAndMatcher | string => {
201
+ stage : ExpectParseStage ,
202
+ ) : ModifiersAndMatcher | string | null => {
171
203
const modifiers : KnownMemberExpressionProperty [ ] = [ ]
172
204
173
205
for ( const member of members ) {
174
- // Otherwise, it should be a modifier
175
206
const name = getStringValue ( member )
176
207
177
208
if ( name === 'soft' || name === 'poll' ) {
@@ -187,6 +218,12 @@ const findModifiersAndMatcher = (
187
218
return 'modifier-unknown'
188
219
}
189
220
} 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
+
190
227
// Check if the member is being called, which means it is the matcher
191
228
// (and also the end of the entire "expect" call chain).
192
229
if (
@@ -205,6 +242,13 @@ const findModifiersAndMatcher = (
205
242
return 'modifier-unknown'
206
243
}
207
244
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
208
252
modifiers . push ( member )
209
253
}
210
254
@@ -263,10 +307,22 @@ export interface ParsedExpectFnCall
263
307
264
308
export type ParsedFnCall = ParsedGeneralFnCall | ParsedExpectFnCall
265
309
310
+ type ExpectParseStage = 'matchers' | 'modifiers'
311
+
266
312
const parseExpectCall = (
313
+ chain : Chain ,
267
314
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
+ }
270
326
271
327
if ( typeof modifiersAndMatcher === 'string' ) {
272
328
return modifiersAndMatcher
@@ -316,13 +372,10 @@ function parse(
316
372
context : Rule . RuleContext ,
317
373
node : ESTree . CallExpression ,
318
374
) : 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
324
377
325
- const [ first , ...rest ] = chain
378
+ const [ first , ...rest ] = chain . nodes
326
379
const resolved = resolveToPlaywrightFn ( context , first )
327
380
if ( ! resolved ) return null
328
381
@@ -355,14 +408,20 @@ function parse(
355
408
const group = determinePlaywrightFnGroup ( name )
356
409
357
410
if ( group === 'expect' ) {
411
+ let stage : ExpectParseStage = chain . isLeaf ( parsedFnCall . head . node )
412
+ ? 'matchers'
413
+ : 'modifiers'
414
+
358
415
// If using `test.expect` style, the `rest` array will start with `expect`
359
416
// and we need to remove it to ensure the chain accurately represents the
360
417
// `expect` call chain.
361
418
if ( isIdentifier ( rest [ 0 ] , 'expect' ) ) {
419
+ stage = chain . isLeaf ( rest [ 0 ] ) ? 'matchers' : 'modifiers'
362
420
parsedFnCall . members . shift ( )
363
421
}
364
422
365
- const result = parseExpectCall ( parsedFnCall )
423
+ const result = parseExpectCall ( chain , parsedFnCall , stage )
424
+ if ( ! result ) return null
366
425
367
426
// If the `expect` call chain is not valid, only report on the topmost node
368
427
// since all members in the chain are likely to get flagged for some reason
@@ -384,8 +443,8 @@ function parse(
384
443
385
444
// Check that every link in the chain except the last is a member expression
386
445
if (
387
- chain
388
- . slice ( 0 , chain . length - 1 )
446
+ chain . nodes
447
+ . slice ( 0 , chain . nodes . length - 1 )
389
448
. some ( ( n ) => getParent ( n ) ?. type !== 'MemberExpression' )
390
449
) {
391
450
return null
0 commit comments