Skip to content

Commit 07be616

Browse files
luxassveritem
andauthoredApr 13, 2024··
feat(no-focused-tests): add autofix (#424)
* feat(no-focused-tests): add autofix * fix(no-focused-tests): set fixable to true by default * chore: remove unused type import * docs(option): added an option for no-focused-tests rule --------- Co-authored-by: Verite Mugabo <mugaboverite@gmail.com>
1 parent b082ce6 commit 07be616

File tree

3 files changed

+189
-6
lines changed

3 files changed

+189
-6
lines changed
 

‎docs/rules/no-focused-tests.md

+32
Original file line numberDiff line numberDiff line change
@@ -31,3 +31,35 @@ test('it', () => {
3131
/* ... */
3232
})
3333
```
34+
35+
### Options
36+
37+
This rule have a `fixable` option that is enabled by default; It tell this plugin to fix tests for you. If you don't want this rule to auto fix your tests, you can disable it in your `eslint.config.js` file using the following configuration.
38+
39+
```ts
40+
import vitest from 'eslint-plugin-vitest'
41+
42+
export default [
43+
{
44+
files: ['**/*.ts', '**/*.js'], // or any other pattern
45+
plugins: {
46+
vitest,
47+
},
48+
rules: {
49+
...vitest.configs.recommended.all,
50+
'vitest/no-focused-tests': ['error', { 'fixable': false }]
51+
},
52+
settings: {
53+
vitest: {
54+
typecheck: true
55+
}
56+
},
57+
languageOptions: {
58+
parser: parser,
59+
globals: {
60+
...vitest.environments.env.globals
61+
}
62+
}
63+
}
64+
]
65+
```

‎src/rules/no-focused-tests.ts

+36-6
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,11 @@ import { createEslintRule } from '../utils'
33

44
export type MessageIds = 'noFocusedTests'
55
export const RULE_NAME = 'no-focused-tests'
6-
export type Options = []
6+
export type Options = [
7+
Partial<{
8+
fixable: boolean
9+
}>
10+
]
711

812
const isTestOrDescribe = (node: TSESTree.Expression) => {
913
return node.type === 'Identifier' && ['it', 'test', 'describe'].includes(node.name)
@@ -22,13 +26,29 @@ export default createEslintRule<Options, MessageIds>({
2226
recommended: 'strict'
2327
},
2428
fixable: 'code',
25-
schema: [],
29+
schema: [
30+
{
31+
type: 'object',
32+
properties: {
33+
fixable: {
34+
type: 'boolean',
35+
default: true
36+
}
37+
},
38+
additionalProperties: false
39+
}
40+
],
2641
messages: {
2742
noFocusedTests: 'Focused tests are not allowed.'
2843
}
2944
},
30-
defaultOptions: [],
45+
defaultOptions: [{ fixable: true }],
3146
create: (context) => {
47+
const config = context.options[0] ?? {
48+
fixable: true
49+
}
50+
const fixable = config.fixable
51+
3252
return {
3353
ExpressionStatement(node) {
3454
if (node.expression.type === 'CallExpression') {
@@ -40,7 +60,8 @@ export default createEslintRule<Options, MessageIds>({
4060
) {
4161
context.report({
4262
node: callee.property,
43-
messageId: 'noFocusedTests'
63+
messageId: 'noFocusedTests',
64+
fix: fixer => fixable ? fixer.removeRange([callee.property.range[0] - 1, callee.property.range[1]]) : null
4465
})
4566
}
4667

@@ -55,7 +76,8 @@ export default createEslintRule<Options, MessageIds>({
5576
) {
5677
context.report({
5778
node: tagCall.property,
58-
messageId: 'noFocusedTests'
79+
messageId: 'noFocusedTests',
80+
fix: fixer => fixable ? fixer.removeRange([tagCall.property.range[0] - 1, tagCall.property.range[1]]) : null
5981
})
6082
}
6183
}
@@ -73,9 +95,17 @@ export default createEslintRule<Options, MessageIds>({
7395
&& callee.property.type === 'Identifier'
7496
&& callee.property.name === 'each'
7597
) {
98+
const onlyCallee = callee.object.property
99+
76100
context.report({
77101
node: callee.object.property,
78-
messageId: 'noFocusedTests'
102+
messageId: 'noFocusedTests',
103+
fix: fixer => fixable
104+
? fixer.removeRange([
105+
onlyCallee.range[0] - 1,
106+
onlyCallee.range[1]
107+
])
108+
: null
79109
})
80110
}
81111
}

‎tests/no-focused-tests.test.ts

+121
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,9 @@ ruleTester.run(RULE_NAME, rule, {
66

77
invalid: [
88
{
9+
options: [{
10+
fixable: false
11+
}],
912
code: 'it.only("test", () => {});',
1013
errors: [
1114
{
@@ -19,6 +22,9 @@ ruleTester.run(RULE_NAME, rule, {
1922
output: 'it.only("test", () => {});'
2023
},
2124
{
25+
options: [{
26+
fixable: false
27+
}],
2228
code: 'describe.only("test", () => {});',
2329
errors: [
2430
{
@@ -32,6 +38,9 @@ ruleTester.run(RULE_NAME, rule, {
3238
output: 'describe.only("test", () => {});'
3339
},
3440
{
41+
options: [{
42+
fixable: false
43+
}],
3544
code: 'test.only("test", () => {});',
3645
errors: [
3746
{
@@ -45,6 +54,9 @@ ruleTester.run(RULE_NAME, rule, {
4554
output: 'test.only("test", () => {});'
4655
},
4756
{
57+
options: [{
58+
fixable: false
59+
}],
4860
code: 'it.only.each([])("test", () => {});',
4961
errors: [
5062
{
@@ -58,6 +70,9 @@ ruleTester.run(RULE_NAME, rule, {
5870
output: 'it.only.each([])("test", () => {});'
5971
},
6072
{
73+
options: [{
74+
fixable: false
75+
}],
6176
code: 'test.only.each``("test", () => {});',
6277
errors: [
6378
{
@@ -71,6 +86,9 @@ ruleTester.run(RULE_NAME, rule, {
7186
output: 'test.only.each``("test", () => {});'
7287
},
7388
{
89+
options: [{
90+
fixable: false
91+
}],
7492
code: 'it.only.each``("test", () => {});',
7593
errors: [
7694
{
@@ -85,3 +103,106 @@ ruleTester.run(RULE_NAME, rule, {
85103
}
86104
]
87105
})
106+
107+
ruleTester.run(RULE_NAME, rule, {
108+
valid: ['it("test", () => {});', 'describe("test group", () => {});'],
109+
110+
invalid: [
111+
{
112+
options: [{
113+
fixable: true
114+
}],
115+
code: 'it.only("test", () => {});',
116+
errors: [
117+
{
118+
column: 4,
119+
endColumn: 8,
120+
endLine: 1,
121+
line: 1,
122+
messageId: 'noFocusedTests'
123+
}
124+
],
125+
output: 'it("test", () => {});'
126+
},
127+
{
128+
options: [{
129+
fixable: true
130+
}],
131+
code: 'describe.only("test", () => {});',
132+
errors: [
133+
{
134+
column: 10,
135+
endColumn: 14,
136+
endLine: 1,
137+
line: 1,
138+
messageId: 'noFocusedTests'
139+
}
140+
],
141+
output: 'describe("test", () => {});'
142+
},
143+
{
144+
options: [{
145+
fixable: true
146+
}],
147+
code: 'test.only("test", () => {});',
148+
errors: [
149+
{
150+
column: 6,
151+
endColumn: 10,
152+
endLine: 1,
153+
line: 1,
154+
messageId: 'noFocusedTests'
155+
}
156+
],
157+
output: 'test("test", () => {});'
158+
},
159+
{
160+
options: [{
161+
fixable: true
162+
}],
163+
code: 'it.only.each([])("test", () => {});',
164+
errors: [
165+
{
166+
column: 4,
167+
endColumn: 8,
168+
endLine: 1,
169+
line: 1,
170+
messageId: 'noFocusedTests'
171+
}
172+
],
173+
output: 'it.each([])("test", () => {});'
174+
},
175+
{
176+
options: [{
177+
fixable: true
178+
}],
179+
code: 'test.only.each``("test", () => {});',
180+
errors: [
181+
{
182+
column: 6,
183+
endColumn: 10,
184+
endLine: 1,
185+
line: 1,
186+
messageId: 'noFocusedTests'
187+
}
188+
],
189+
output: 'test.each``("test", () => {});'
190+
},
191+
{
192+
options: [{
193+
fixable: true
194+
}],
195+
code: 'it.only.each``("test", () => {});',
196+
errors: [
197+
{
198+
column: 4,
199+
endColumn: 8,
200+
endLine: 1,
201+
line: 1,
202+
messageId: 'noFocusedTests'
203+
}
204+
],
205+
output: 'it.each``("test", () => {});'
206+
}
207+
]
208+
})

0 commit comments

Comments
 (0)
Please sign in to comment.