Skip to content

Commit bf48dc5

Browse files
authored
fix(eslint-plugin): [no-unsafe-member-access] differentiate a types-error any from a true any (#9291)
* fix: support error type * fix: add "typed" * fix: Fixed the case of unsafeComputedMemberAccess as well. * test: update snapshot
1 parent 4598b0d commit bf48dc5

File tree

3 files changed

+61
-4
lines changed

3 files changed

+61
-4
lines changed

packages/eslint-plugin/src/rules/no-unsafe-member-access.ts

+10-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import type { TSESTree } from '@typescript-eslint/utils';
22
import { AST_NODE_TYPES } from '@typescript-eslint/utils';
33
import * as tsutils from 'ts-api-utils';
4+
import type * as ts from 'typescript';
45

56
import {
67
createRule,
@@ -15,6 +16,11 @@ const enum State {
1516
Safe = 2,
1617
}
1718

19+
function createDataType(type: ts.Type): '`error` typed' | '`any`' {
20+
const isErrorType = tsutils.isIntrinsicErrorType(type);
21+
return isErrorType ? '`error` typed' : '`any`';
22+
}
23+
1824
export default createRule({
1925
name: 'no-unsafe-member-access',
2026
meta: {
@@ -26,13 +32,13 @@ export default createRule({
2632
},
2733
messages: {
2834
unsafeMemberExpression:
29-
'Unsafe member access {{property}} on an `any` value.',
35+
'Unsafe member access {{property}} on an {{type}} value.',
3036
unsafeThisMemberExpression: [
3137
'Unsafe member access {{property}} on an `any` value. `this` is typed as `any`.',
3238
'You can try to fix this by turning on the `noImplicitThis` compiler option, or adding a `this` parameter to the function.',
3339
].join('\n'),
3440
unsafeComputedMemberAccess:
35-
'Computed name {{property}} resolves to an any value.',
41+
'Computed name {{property}} resolves to an {{type}} value.',
3642
},
3743
schema: [],
3844
},
@@ -92,6 +98,7 @@ export default createRule({
9298
messageId,
9399
data: {
94100
property: node.computed ? `[${propertyName}]` : `.${propertyName}`,
101+
type: createDataType(type),
95102
},
96103
});
97104
}
@@ -127,6 +134,7 @@ export default createRule({
127134
messageId: 'unsafeComputedMemberAccess',
128135
data: {
129136
property: `[${propertyName}]`,
137+
type: createDataType(type),
130138
},
131139
});
132140
}

packages/eslint-plugin/tests/docs-eslint-output-snapshots/no-unsafe-member-access.shot

+2-2
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

packages/eslint-plugin/tests/rules/no-unsafe-member-access.test.ts

+49
Original file line numberDiff line numberDiff line change
@@ -75,6 +75,7 @@ function foo(x: any) {
7575
column: 5,
7676
endColumn: 6,
7777
data: {
78+
type: '`any`',
7879
property: '.a',
7980
},
8081
},
@@ -93,6 +94,7 @@ function foo(x: any) {
9394
column: 5,
9495
endColumn: 6,
9596
data: {
97+
type: '`any`',
9698
property: '.a',
9799
},
98100
},
@@ -111,6 +113,7 @@ function foo(x: { a: any }) {
111113
column: 7,
112114
endColumn: 8,
113115
data: {
116+
type: '`any`',
114117
property: '.b',
115118
},
116119
},
@@ -129,6 +132,7 @@ function foo(x: any) {
129132
column: 5,
130133
endColumn: 8,
131134
data: {
135+
type: '`any`',
132136
property: "['a']",
133137
},
134138
},
@@ -147,13 +151,33 @@ function foo(x: any) {
147151
column: 5,
148152
endColumn: 8,
149153
data: {
154+
type: '`any`',
150155
property: "['a']",
151156
},
152157
},
153158
],
154159
},
155160
{
156161
code: `
162+
let value: NotKnown;
163+
164+
value.property;
165+
`,
166+
errors: [
167+
{
168+
messageId: 'unsafeMemberExpression',
169+
line: 4,
170+
column: 7,
171+
endColumn: 15,
172+
data: {
173+
type: '`error` typed',
174+
property: '.property',
175+
},
176+
},
177+
],
178+
},
179+
{
180+
code: `
157181
function foo(x: { a: number }, y: any) {
158182
x[y];
159183
}
@@ -166,6 +190,7 @@ function foo(x: { a: number }, y: any) {
166190
endColumn: 6,
167191
data: {
168192
property: '[y]',
193+
type: '`any`',
169194
},
170195
},
171196
],
@@ -184,6 +209,7 @@ function foo(x?: { a: number }, y: any) {
184209
endColumn: 8,
185210
data: {
186211
property: '[y]',
212+
type: '`any`',
187213
},
188214
},
189215
],
@@ -202,6 +228,7 @@ function foo(x: { a: number }, y: any) {
202228
endColumn: 12,
203229
data: {
204230
property: '[y += 1]',
231+
type: '`any`',
205232
},
206233
},
207234
],
@@ -220,6 +247,7 @@ function foo(x: { a: number }, y: any) {
220247
endColumn: 13,
221248
data: {
222249
property: '[1 as any]',
250+
type: '`any`',
223251
},
224252
},
225253
],
@@ -238,6 +266,7 @@ function foo(x: { a: number }, y: any) {
238266
endColumn: 8,
239267
data: {
240268
property: '[y()]',
269+
type: '`any`',
241270
},
242271
},
243272
],
@@ -256,6 +285,26 @@ function foo(x: string[], y: any) {
256285
endColumn: 6,
257286
data: {
258287
property: '[y]',
288+
type: '`any`',
289+
},
290+
},
291+
],
292+
},
293+
{
294+
code: `
295+
function foo(x: { a: number }, y: NotKnown) {
296+
x[y];
297+
}
298+
`,
299+
errors: [
300+
{
301+
messageId: 'unsafeComputedMemberAccess',
302+
line: 3,
303+
column: 5,
304+
endColumn: 6,
305+
data: {
306+
property: '[y]',
307+
type: '`error` typed',
259308
},
260309
},
261310
],

0 commit comments

Comments
 (0)