-
-
Notifications
You must be signed in to change notification settings - Fork 3.1k
/
Copy pathno-rest-destructuring.rule.ts
115 lines (101 loc) · 3.21 KB
/
no-rest-destructuring.rule.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
import { AST_NODE_TYPES, ESLintUtils } from '@typescript-eslint/utils'
import { getDocsUrl } from '../../utils/get-docs-url'
import { ASTUtils } from '../../utils/ast-utils'
import { detectTanstackQueryImports } from '../../utils/detect-react-query-imports'
import { NoRestDestructuringUtils } from './no-rest-destructuring.utils'
import type { ExtraRuleDocs } from '../../types'
export const name = 'no-rest-destructuring'
const queryHooks = [
'useQuery',
'useQueries',
'useInfiniteQuery',
'useSuspenseQuery',
'useSuspenseQueries',
'useSuspenseInfiniteQuery',
]
const createRule = ESLintUtils.RuleCreator<ExtraRuleDocs>(getDocsUrl)
export const rule = createRule({
name,
meta: {
type: 'problem',
docs: {
description: 'Disallows rest destructuring in queries',
recommended: 'warn',
},
messages: {
objectRestDestructure: `Object rest destructuring on a query will observe all changes to the query, leading to excessive re-renders.`,
},
schema: [],
},
defaultOptions: [],
create: detectTanstackQueryImports((context, _, helpers) => {
const queryResultVariables = new Set<string>()
return {
CallExpression: (node) => {
if (
!ASTUtils.isIdentifierWithOneOfNames(node.callee, queryHooks) ||
node.parent.type !== AST_NODE_TYPES.VariableDeclarator ||
!helpers.isTanstackQueryImport(node.callee)
) {
return
}
const returnValue = node.parent.id
if (
node.callee.name !== 'useQueries' &&
node.callee.name !== 'useSuspenseQueries'
) {
if (NoRestDestructuringUtils.isObjectRestDestructuring(returnValue)) {
return context.report({
node: node.parent,
messageId: 'objectRestDestructure',
})
}
if (returnValue.type === AST_NODE_TYPES.Identifier) {
queryResultVariables.add(returnValue.name)
}
return
}
if (returnValue.type !== AST_NODE_TYPES.ArrayPattern) {
if (returnValue.type === AST_NODE_TYPES.Identifier) {
queryResultVariables.add(returnValue.name)
}
return
}
returnValue.elements.forEach((queryResult) => {
if (queryResult === null) {
return
}
if (NoRestDestructuringUtils.isObjectRestDestructuring(queryResult)) {
context.report({
node: queryResult,
messageId: 'objectRestDestructure',
})
}
})
},
VariableDeclarator: (node) => {
if (
node.init?.type === AST_NODE_TYPES.Identifier &&
queryResultVariables.has(node.init.name) &&
NoRestDestructuringUtils.isObjectRestDestructuring(node.id)
) {
context.report({
node,
messageId: 'objectRestDestructure',
})
}
},
SpreadElement: (node) => {
if (
node.argument.type === AST_NODE_TYPES.Identifier &&
queryResultVariables.has(node.argument.name)
) {
context.report({
node,
messageId: 'objectRestDestructure',
})
}
},
}
}),
})