@futdevpro/dynamo-eslint 1.14.4 → 1.14.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.vscode/settings.json +0 -2
- package/build/configs/base.js +29 -7
- package/build/configs/base.js.map +1 -1
- package/build/plugin/index.d.ts +8 -0
- package/build/plugin/index.d.ts.map +1 -1
- package/build/plugin/index.js +12 -0
- package/build/plugin/index.js.map +1 -1
- package/build/plugin/rules/explicit-types.d.ts.map +1 -1
- package/build/plugin/rules/explicit-types.js +16 -2
- package/build/plugin/rules/explicit-types.js.map +1 -1
- package/build/plugin/rules/import/import-order.d.ts.map +1 -1
- package/build/plugin/rules/import/import-order.js +0 -9
- package/build/plugin/rules/import/import-order.js.map +1 -1
- package/build/plugin/rules/import/no-js-import.d.ts.map +1 -1
- package/build/plugin/rules/import/no-js-import.js +12 -15
- package/build/plugin/rules/import/no-js-import.js.map +1 -1
- package/build/plugin/rules/prefer-enum-over-string-union.d.ts +4 -0
- package/build/plugin/rules/prefer-enum-over-string-union.d.ts.map +1 -0
- package/build/plugin/rules/prefer-enum-over-string-union.js +290 -0
- package/build/plugin/rules/prefer-enum-over-string-union.js.map +1 -0
- package/build/plugin/rules/prefer-enum-over-string-union.spec.d.ts +2 -0
- package/build/plugin/rules/prefer-enum-over-string-union.spec.d.ts.map +1 -0
- package/build/plugin/rules/prefer-enum-over-string-union.spec.js +505 -0
- package/build/plugin/rules/prefer-enum-over-string-union.spec.js.map +1 -0
- package/build/plugin/rules/prefer-interface-over-duplicate-types.d.ts +4 -0
- package/build/plugin/rules/prefer-interface-over-duplicate-types.d.ts.map +1 -0
- package/build/plugin/rules/prefer-interface-over-duplicate-types.js +231 -0
- package/build/plugin/rules/prefer-interface-over-duplicate-types.js.map +1 -0
- package/build/plugin/rules/prefer-interface-over-duplicate-types.spec.d.ts +2 -0
- package/build/plugin/rules/prefer-interface-over-duplicate-types.spec.d.ts.map +1 -0
- package/build/plugin/rules/prefer-interface-over-duplicate-types.spec.js +324 -0
- package/build/plugin/rules/prefer-interface-over-duplicate-types.spec.js.map +1 -0
- package/build/plugin/rules/require-jsdoc-description.d.ts +4 -0
- package/build/plugin/rules/require-jsdoc-description.d.ts.map +1 -0
- package/build/plugin/rules/require-jsdoc-description.js +379 -0
- package/build/plugin/rules/require-jsdoc-description.js.map +1 -0
- package/build/plugin/rules/require-jsdoc-description.spec.d.ts +2 -0
- package/build/plugin/rules/require-jsdoc-description.spec.d.ts.map +1 -0
- package/build/plugin/rules/require-jsdoc-description.spec.js +452 -0
- package/build/plugin/rules/require-jsdoc-description.spec.js.map +1 -0
- package/build/plugin/rules/require-test-description-prefix.d.ts +4 -0
- package/build/plugin/rules/require-test-description-prefix.d.ts.map +1 -0
- package/build/plugin/rules/require-test-description-prefix.js +135 -0
- package/build/plugin/rules/require-test-description-prefix.js.map +1 -0
- package/build/plugin/rules/require-test-description-prefix.spec.d.ts +2 -0
- package/build/plugin/rules/require-test-description-prefix.spec.d.ts.map +1 -0
- package/build/plugin/rules/require-test-description-prefix.spec.js +371 -0
- package/build/plugin/rules/require-test-description-prefix.spec.js.map +1 -0
- package/build/scripts/eslintrc-audit.js.map +1 -1
- package/futdevpro-dynamo-eslint-1.14.7.tgz +0 -0
- package/package.json +1 -1
- package/samples/package.json.example +1 -1
- package/src/configs/base.ts +45 -23
- package/src/plugin/index.ts +12 -0
- package/src/plugin/rules/explicit-types.ts +19 -2
- package/src/plugin/rules/import/import-order.ts +0 -9
- package/src/plugin/rules/import/no-js-import.ts +19 -17
- package/src/plugin/rules/prefer-enum-over-string-union.spec.ts +583 -0
- package/src/plugin/rules/prefer-enum-over-string-union.ts +333 -0
- package/src/plugin/rules/prefer-interface-over-duplicate-types.spec.ts +374 -0
- package/src/plugin/rules/prefer-interface-over-duplicate-types.ts +276 -0
- package/src/plugin/rules/require-jsdoc-description.spec.ts +542 -0
- package/src/plugin/rules/require-jsdoc-description.ts +436 -0
- package/src/plugin/rules/require-test-description-prefix.spec.ts +459 -0
- package/src/plugin/rules/require-test-description-prefix.ts +153 -0
- package/src/scripts/eslintrc-audit.ts +8 -6
- package/futdevpro-dynamo-eslint-01.14.4.tgz +0 -0
|
@@ -0,0 +1,459 @@
|
|
|
1
|
+
import requireTestPrefixRule from './require-test-description-prefix';
|
|
2
|
+
|
|
3
|
+
describe('| require-test-description-prefix', () => {
|
|
4
|
+
it('| should be a valid ESLint rule', () => {
|
|
5
|
+
expect(requireTestPrefixRule.meta.type).toBe('suggestion');
|
|
6
|
+
expect(requireTestPrefixRule.meta.docs.description).toBe('Require "| " prefix in test descriptions');
|
|
7
|
+
expect(requireTestPrefixRule.meta.fixable).toBe('code');
|
|
8
|
+
});
|
|
9
|
+
|
|
10
|
+
it('| should not report when describe has correct prefix', () => {
|
|
11
|
+
let reportCalled = false;
|
|
12
|
+
const mockContext = {
|
|
13
|
+
report: (options: any) => {
|
|
14
|
+
reportCalled = true;
|
|
15
|
+
},
|
|
16
|
+
options: [{ prefix: '| ' }],
|
|
17
|
+
} as any;
|
|
18
|
+
|
|
19
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
20
|
+
|
|
21
|
+
const mockNode = {
|
|
22
|
+
type: 'CallExpression',
|
|
23
|
+
callee: { name: 'describe' },
|
|
24
|
+
arguments: [
|
|
25
|
+
{ type: 'Literal', value: '| my test suite' },
|
|
26
|
+
],
|
|
27
|
+
} as any;
|
|
28
|
+
|
|
29
|
+
rule.CallExpression(mockNode);
|
|
30
|
+
|
|
31
|
+
expect(reportCalled).toBe(false);
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it('| should not report when it has correct prefix', () => {
|
|
35
|
+
let reportCalled = false;
|
|
36
|
+
const mockContext = {
|
|
37
|
+
report: (options: any) => {
|
|
38
|
+
reportCalled = true;
|
|
39
|
+
},
|
|
40
|
+
options: [{ prefix: '| ' }],
|
|
41
|
+
} as any;
|
|
42
|
+
|
|
43
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
44
|
+
|
|
45
|
+
const mockNode = {
|
|
46
|
+
type: 'CallExpression',
|
|
47
|
+
callee: { name: 'it' },
|
|
48
|
+
arguments: [
|
|
49
|
+
{ type: 'Literal', value: '| should do something' },
|
|
50
|
+
],
|
|
51
|
+
} as any;
|
|
52
|
+
|
|
53
|
+
rule.CallExpression(mockNode);
|
|
54
|
+
|
|
55
|
+
expect(reportCalled).toBe(false);
|
|
56
|
+
});
|
|
57
|
+
|
|
58
|
+
it('| should not report when test has correct prefix', () => {
|
|
59
|
+
let reportCalled = false;
|
|
60
|
+
const mockContext = {
|
|
61
|
+
report: (options: any) => {
|
|
62
|
+
reportCalled = true;
|
|
63
|
+
},
|
|
64
|
+
options: [{ prefix: '| ' }],
|
|
65
|
+
} as any;
|
|
66
|
+
|
|
67
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
68
|
+
|
|
69
|
+
const mockNode = {
|
|
70
|
+
type: 'CallExpression',
|
|
71
|
+
callee: { name: 'test' },
|
|
72
|
+
arguments: [
|
|
73
|
+
{ type: 'Literal', value: '| validates input' },
|
|
74
|
+
],
|
|
75
|
+
} as any;
|
|
76
|
+
|
|
77
|
+
rule.CallExpression(mockNode);
|
|
78
|
+
|
|
79
|
+
expect(reportCalled).toBe(false);
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
it('| should report when describe is missing prefix', () => {
|
|
83
|
+
let reportCalled = false;
|
|
84
|
+
const mockContext = {
|
|
85
|
+
report: (options: any) => {
|
|
86
|
+
reportCalled = true;
|
|
87
|
+
expect(options.messageId).toBe('missingPrefix');
|
|
88
|
+
expect(options.data.prefix).toBe('| ');
|
|
89
|
+
expect(options.fix).toBeDefined();
|
|
90
|
+
},
|
|
91
|
+
options: [{ prefix: '| ' }],
|
|
92
|
+
} as any;
|
|
93
|
+
|
|
94
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
95
|
+
|
|
96
|
+
const mockNode = {
|
|
97
|
+
type: 'CallExpression',
|
|
98
|
+
callee: { name: 'describe' },
|
|
99
|
+
arguments: [
|
|
100
|
+
{ type: 'Literal', value: 'my test suite' },
|
|
101
|
+
],
|
|
102
|
+
} as any;
|
|
103
|
+
|
|
104
|
+
rule.CallExpression(mockNode);
|
|
105
|
+
|
|
106
|
+
expect(reportCalled).toBe(true);
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
it('| should report when it is missing prefix', () => {
|
|
110
|
+
let reportCalled = false;
|
|
111
|
+
const mockContext = {
|
|
112
|
+
report: (options: any) => {
|
|
113
|
+
reportCalled = true;
|
|
114
|
+
expect(options.messageId).toBe('missingPrefix');
|
|
115
|
+
expect(options.data.prefix).toBe('| ');
|
|
116
|
+
},
|
|
117
|
+
options: [{ prefix: '| ' }],
|
|
118
|
+
} as any;
|
|
119
|
+
|
|
120
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
121
|
+
|
|
122
|
+
const mockNode = {
|
|
123
|
+
type: 'CallExpression',
|
|
124
|
+
callee: { name: 'it' },
|
|
125
|
+
arguments: [
|
|
126
|
+
{ type: 'Literal', value: 'should do something' },
|
|
127
|
+
],
|
|
128
|
+
} as any;
|
|
129
|
+
|
|
130
|
+
rule.CallExpression(mockNode);
|
|
131
|
+
|
|
132
|
+
expect(reportCalled).toBe(true);
|
|
133
|
+
});
|
|
134
|
+
|
|
135
|
+
it('| should report when describe has wrong prefix', () => {
|
|
136
|
+
let reportCalled = false;
|
|
137
|
+
const mockContext = {
|
|
138
|
+
report: (options: any) => {
|
|
139
|
+
reportCalled = true;
|
|
140
|
+
expect(options.messageId).toBe('wrongPrefix');
|
|
141
|
+
expect(options.data.prefix).toBe('| ');
|
|
142
|
+
expect(options.data.found).toBe('|');
|
|
143
|
+
},
|
|
144
|
+
options: [{ prefix: '| ' }],
|
|
145
|
+
} as any;
|
|
146
|
+
|
|
147
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
148
|
+
|
|
149
|
+
const mockNode = {
|
|
150
|
+
type: 'CallExpression',
|
|
151
|
+
callee: { name: 'describe' },
|
|
152
|
+
arguments: [
|
|
153
|
+
{ type: 'Literal', value: '|my test suite' },
|
|
154
|
+
],
|
|
155
|
+
} as any;
|
|
156
|
+
|
|
157
|
+
rule.CallExpression(mockNode);
|
|
158
|
+
|
|
159
|
+
expect(reportCalled).toBe(true);
|
|
160
|
+
});
|
|
161
|
+
|
|
162
|
+
it('| should report when it has only pipe without space', () => {
|
|
163
|
+
let reportCalled = false;
|
|
164
|
+
const mockContext = {
|
|
165
|
+
report: (options: any) => {
|
|
166
|
+
reportCalled = true;
|
|
167
|
+
expect(options.messageId).toBe('wrongPrefix');
|
|
168
|
+
expect(options.data.found).toBe('|');
|
|
169
|
+
},
|
|
170
|
+
options: [{ prefix: '| ' }],
|
|
171
|
+
} as any;
|
|
172
|
+
|
|
173
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
174
|
+
|
|
175
|
+
const mockNode = {
|
|
176
|
+
type: 'CallExpression',
|
|
177
|
+
callee: { name: 'it' },
|
|
178
|
+
arguments: [
|
|
179
|
+
{ type: 'Literal', value: '|should do something' },
|
|
180
|
+
],
|
|
181
|
+
} as any;
|
|
182
|
+
|
|
183
|
+
rule.CallExpression(mockNode);
|
|
184
|
+
|
|
185
|
+
expect(reportCalled).toBe(true);
|
|
186
|
+
});
|
|
187
|
+
|
|
188
|
+
it('| should not report on non-test functions', () => {
|
|
189
|
+
let reportCalled = false;
|
|
190
|
+
const mockContext = {
|
|
191
|
+
report: (options: any) => {
|
|
192
|
+
reportCalled = true;
|
|
193
|
+
},
|
|
194
|
+
options: [{ prefix: '| ' }],
|
|
195
|
+
} as any;
|
|
196
|
+
|
|
197
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
198
|
+
|
|
199
|
+
const mockNode = {
|
|
200
|
+
type: 'CallExpression',
|
|
201
|
+
callee: { name: 'console' },
|
|
202
|
+
arguments: [
|
|
203
|
+
{ type: 'Literal', value: 'log message' },
|
|
204
|
+
],
|
|
205
|
+
} as any;
|
|
206
|
+
|
|
207
|
+
rule.CallExpression(mockNode);
|
|
208
|
+
|
|
209
|
+
expect(reportCalled).toBe(false);
|
|
210
|
+
});
|
|
211
|
+
|
|
212
|
+
it('| should not report when no arguments provided', () => {
|
|
213
|
+
let reportCalled = false;
|
|
214
|
+
const mockContext = {
|
|
215
|
+
report: (options: any) => {
|
|
216
|
+
reportCalled = true;
|
|
217
|
+
},
|
|
218
|
+
options: [{ prefix: '| ' }],
|
|
219
|
+
} as any;
|
|
220
|
+
|
|
221
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
222
|
+
|
|
223
|
+
const mockNode = {
|
|
224
|
+
type: 'CallExpression',
|
|
225
|
+
callee: { name: 'describe' },
|
|
226
|
+
arguments: [],
|
|
227
|
+
} as any;
|
|
228
|
+
|
|
229
|
+
rule.CallExpression(mockNode);
|
|
230
|
+
|
|
231
|
+
expect(reportCalled).toBe(false);
|
|
232
|
+
});
|
|
233
|
+
|
|
234
|
+
it('| should not report when first argument is not a string', () => {
|
|
235
|
+
let reportCalled = false;
|
|
236
|
+
const mockContext = {
|
|
237
|
+
report: (options: any) => {
|
|
238
|
+
reportCalled = true;
|
|
239
|
+
},
|
|
240
|
+
options: [{ prefix: '| ' }],
|
|
241
|
+
} as any;
|
|
242
|
+
|
|
243
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
244
|
+
|
|
245
|
+
const mockNode = {
|
|
246
|
+
type: 'CallExpression',
|
|
247
|
+
callee: { name: 'describe' },
|
|
248
|
+
arguments: [
|
|
249
|
+
{ type: 'Identifier', name: 'myVariable' },
|
|
250
|
+
],
|
|
251
|
+
} as any;
|
|
252
|
+
|
|
253
|
+
rule.CallExpression(mockNode);
|
|
254
|
+
|
|
255
|
+
expect(reportCalled).toBe(false);
|
|
256
|
+
});
|
|
257
|
+
|
|
258
|
+
it('| should handle beforeEach and afterEach', () => {
|
|
259
|
+
let reportCalled = false;
|
|
260
|
+
const mockContext = {
|
|
261
|
+
report: (options: any) => {
|
|
262
|
+
reportCalled = true;
|
|
263
|
+
expect(options.messageId).toBe('missingPrefix');
|
|
264
|
+
},
|
|
265
|
+
options: [{ prefix: '| ' }],
|
|
266
|
+
} as any;
|
|
267
|
+
|
|
268
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
269
|
+
|
|
270
|
+
const mockNode = {
|
|
271
|
+
type: 'CallExpression',
|
|
272
|
+
callee: { name: 'beforeEach' },
|
|
273
|
+
arguments: [
|
|
274
|
+
{ type: 'Literal', value: 'setup test data' },
|
|
275
|
+
],
|
|
276
|
+
} as any;
|
|
277
|
+
|
|
278
|
+
rule.CallExpression(mockNode);
|
|
279
|
+
|
|
280
|
+
expect(reportCalled).toBe(true);
|
|
281
|
+
});
|
|
282
|
+
|
|
283
|
+
it('| should handle beforeAll and afterAll', () => {
|
|
284
|
+
let reportCalled = false;
|
|
285
|
+
const mockContext = {
|
|
286
|
+
report: (options: any) => {
|
|
287
|
+
reportCalled = true;
|
|
288
|
+
expect(options.messageId).toBe('missingPrefix');
|
|
289
|
+
},
|
|
290
|
+
options: [{ prefix: '| ' }],
|
|
291
|
+
} as any;
|
|
292
|
+
|
|
293
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
294
|
+
|
|
295
|
+
const mockNode = {
|
|
296
|
+
type: 'CallExpression',
|
|
297
|
+
callee: { name: 'beforeAll' },
|
|
298
|
+
arguments: [
|
|
299
|
+
{ type: 'Literal', value: 'setup suite' },
|
|
300
|
+
],
|
|
301
|
+
} as any;
|
|
302
|
+
|
|
303
|
+
rule.CallExpression(mockNode);
|
|
304
|
+
|
|
305
|
+
expect(reportCalled).toBe(true);
|
|
306
|
+
});
|
|
307
|
+
|
|
308
|
+
it('| should handle custom prefix configuration', () => {
|
|
309
|
+
let reportCalled = false;
|
|
310
|
+
const mockContext = {
|
|
311
|
+
report: (options: any) => {
|
|
312
|
+
reportCalled = true;
|
|
313
|
+
expect(options.data.prefix).toBe('>>> ');
|
|
314
|
+
},
|
|
315
|
+
options: [{ prefix: '>>> ' }],
|
|
316
|
+
} as any;
|
|
317
|
+
|
|
318
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
319
|
+
|
|
320
|
+
const mockNode = {
|
|
321
|
+
type: 'CallExpression',
|
|
322
|
+
callee: { name: 'describe' },
|
|
323
|
+
arguments: [
|
|
324
|
+
{ type: 'Literal', value: 'my test suite' },
|
|
325
|
+
],
|
|
326
|
+
} as any;
|
|
327
|
+
|
|
328
|
+
rule.CallExpression(mockNode);
|
|
329
|
+
|
|
330
|
+
expect(reportCalled).toBe(true);
|
|
331
|
+
});
|
|
332
|
+
|
|
333
|
+
it('| should handle custom test functions configuration', () => {
|
|
334
|
+
let reportCalled = false;
|
|
335
|
+
const mockContext = {
|
|
336
|
+
report: (options: any) => {
|
|
337
|
+
reportCalled = true;
|
|
338
|
+
},
|
|
339
|
+
options: [{ testFunctions: [ 'myTest' ] }],
|
|
340
|
+
} as any;
|
|
341
|
+
|
|
342
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
343
|
+
|
|
344
|
+
const mockNode = {
|
|
345
|
+
type: 'CallExpression',
|
|
346
|
+
callee: { name: 'myTest' },
|
|
347
|
+
arguments: [
|
|
348
|
+
{ type: 'Literal', value: 'my test description' },
|
|
349
|
+
],
|
|
350
|
+
} as any;
|
|
351
|
+
|
|
352
|
+
rule.CallExpression(mockNode);
|
|
353
|
+
|
|
354
|
+
expect(reportCalled).toBe(true);
|
|
355
|
+
});
|
|
356
|
+
|
|
357
|
+
it('| should provide auto-fix functionality', () => {
|
|
358
|
+
const mockContext = {
|
|
359
|
+
report: (options: any) => {
|
|
360
|
+
expect(options.fix).toBeDefined();
|
|
361
|
+
expect(typeof options.fix).toBe('function');
|
|
362
|
+
|
|
363
|
+
// Test the fix function
|
|
364
|
+
const fixResult = options.fix({
|
|
365
|
+
replaceText: (node: any, text: string) => {
|
|
366
|
+
expect(text).toBe('\'| my test suite\'');
|
|
367
|
+
|
|
368
|
+
return [];
|
|
369
|
+
},
|
|
370
|
+
});
|
|
371
|
+
|
|
372
|
+
expect(Array.isArray(fixResult)).toBe(true);
|
|
373
|
+
},
|
|
374
|
+
options: [{ prefix: '| ' }],
|
|
375
|
+
} as any;
|
|
376
|
+
|
|
377
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
378
|
+
|
|
379
|
+
const mockNode = {
|
|
380
|
+
type: 'CallExpression',
|
|
381
|
+
callee: { name: 'describe' },
|
|
382
|
+
arguments: [
|
|
383
|
+
{ type: 'Literal', value: 'my test suite' },
|
|
384
|
+
],
|
|
385
|
+
} as any;
|
|
386
|
+
|
|
387
|
+
rule.CallExpression(mockNode);
|
|
388
|
+
});
|
|
389
|
+
|
|
390
|
+
it('| should handle empty string descriptions', () => {
|
|
391
|
+
let reportCalled = false;
|
|
392
|
+
const mockContext = {
|
|
393
|
+
report: (options: any) => {
|
|
394
|
+
reportCalled = true;
|
|
395
|
+
expect(options.messageId).toBe('missingPrefix');
|
|
396
|
+
},
|
|
397
|
+
options: [{ prefix: '| ' }],
|
|
398
|
+
} as any;
|
|
399
|
+
|
|
400
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
401
|
+
|
|
402
|
+
const mockNode = {
|
|
403
|
+
type: 'CallExpression',
|
|
404
|
+
callee: { name: 'describe' },
|
|
405
|
+
arguments: [
|
|
406
|
+
{ type: 'Literal', value: '' },
|
|
407
|
+
],
|
|
408
|
+
} as any;
|
|
409
|
+
|
|
410
|
+
rule.CallExpression(mockNode);
|
|
411
|
+
|
|
412
|
+
expect(reportCalled).toBe(true);
|
|
413
|
+
});
|
|
414
|
+
|
|
415
|
+
it('| should handle errors gracefully', () => {
|
|
416
|
+
const mockContext = {
|
|
417
|
+
report: (options: any) => {
|
|
418
|
+
// Should not throw
|
|
419
|
+
},
|
|
420
|
+
options: [{ prefix: '| ' }],
|
|
421
|
+
} as any;
|
|
422
|
+
|
|
423
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
424
|
+
|
|
425
|
+
// Test with malformed nodes
|
|
426
|
+
const malformedNode = {
|
|
427
|
+
type: 'CallExpression',
|
|
428
|
+
// Missing required properties
|
|
429
|
+
} as any;
|
|
430
|
+
|
|
431
|
+
expect(() => {
|
|
432
|
+
rule.CallExpression(malformedNode);
|
|
433
|
+
}).not.toThrow();
|
|
434
|
+
});
|
|
435
|
+
|
|
436
|
+
it('| should handle missing callee gracefully', () => {
|
|
437
|
+
let reportCalled = false;
|
|
438
|
+
const mockContext = {
|
|
439
|
+
report: (options: any) => {
|
|
440
|
+
reportCalled = true;
|
|
441
|
+
},
|
|
442
|
+
options: [{ prefix: '| ' }],
|
|
443
|
+
} as any;
|
|
444
|
+
|
|
445
|
+
const rule = requireTestPrefixRule.create(mockContext);
|
|
446
|
+
|
|
447
|
+
const mockNode = {
|
|
448
|
+
type: 'CallExpression',
|
|
449
|
+
// Missing callee
|
|
450
|
+
arguments: [
|
|
451
|
+
{ type: 'Literal', value: 'my test suite' },
|
|
452
|
+
],
|
|
453
|
+
} as any;
|
|
454
|
+
|
|
455
|
+
rule.CallExpression(mockNode);
|
|
456
|
+
|
|
457
|
+
expect(reportCalled).toBe(false);
|
|
458
|
+
});
|
|
459
|
+
});
|
|
@@ -0,0 +1,153 @@
|
|
|
1
|
+
import { Rule } from 'eslint';
|
|
2
|
+
|
|
3
|
+
interface RuleOptions {
|
|
4
|
+
prefix?: string;
|
|
5
|
+
testFunctions?: string[];
|
|
6
|
+
}
|
|
7
|
+
|
|
8
|
+
const rule: Rule.RuleModule = {
|
|
9
|
+
meta: {
|
|
10
|
+
type: 'suggestion',
|
|
11
|
+
docs: {
|
|
12
|
+
description: 'Require "| " prefix in test descriptions',
|
|
13
|
+
recommended: true,
|
|
14
|
+
},
|
|
15
|
+
schema: [
|
|
16
|
+
{
|
|
17
|
+
type: 'object',
|
|
18
|
+
properties: {
|
|
19
|
+
prefix: {
|
|
20
|
+
type: 'string',
|
|
21
|
+
default: '| ',
|
|
22
|
+
},
|
|
23
|
+
testFunctions: {
|
|
24
|
+
type: 'array',
|
|
25
|
+
items: {
|
|
26
|
+
type: 'string',
|
|
27
|
+
},
|
|
28
|
+
default: [ 'describe', 'it', 'test', 'beforeEach', 'afterEach', 'beforeAll', 'afterAll' ],
|
|
29
|
+
},
|
|
30
|
+
},
|
|
31
|
+
additionalProperties: false,
|
|
32
|
+
},
|
|
33
|
+
],
|
|
34
|
+
messages: {
|
|
35
|
+
missingPrefix: 'Test description must start with "{{prefix}}"',
|
|
36
|
+
wrongPrefix: 'Test description must start with "{{prefix}}" but found "{{found}}"',
|
|
37
|
+
},
|
|
38
|
+
fixable: 'code',
|
|
39
|
+
},
|
|
40
|
+
create(context) {
|
|
41
|
+
const options: RuleOptions = context.options[0] || {};
|
|
42
|
+
const prefix: string = options.prefix || '| ';
|
|
43
|
+
const testFunctions: string[] = options.testFunctions || [ 'describe', 'it', 'test', 'beforeEach', 'afterEach', 'beforeAll', 'afterAll' ];
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* Check if a CallExpression is a test function
|
|
47
|
+
*/
|
|
48
|
+
function isTestFunction(node: any): boolean {
|
|
49
|
+
try {
|
|
50
|
+
return node.type === 'CallExpression' &&
|
|
51
|
+
node.callee &&
|
|
52
|
+
node.callee.name &&
|
|
53
|
+
testFunctions.includes(node.callee.name);
|
|
54
|
+
} catch (error) {
|
|
55
|
+
console.error('[require-test-description-prefix] Error in isTestFunction:', error);
|
|
56
|
+
|
|
57
|
+
return false;
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
|
|
61
|
+
/**
|
|
62
|
+
* Check if a string has the correct prefix
|
|
63
|
+
*/
|
|
64
|
+
function hasCorrectPrefix(description: string): boolean {
|
|
65
|
+
try {
|
|
66
|
+
return typeof description === 'string' && description.startsWith(prefix);
|
|
67
|
+
} catch (error) {
|
|
68
|
+
console.error('[require-test-description-prefix] Error in hasCorrectPrefix:', error);
|
|
69
|
+
|
|
70
|
+
return false;
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
/**
|
|
75
|
+
* Get the actual prefix found in the string
|
|
76
|
+
*/
|
|
77
|
+
function getFoundPrefix(description: string): string {
|
|
78
|
+
try {
|
|
79
|
+
if (typeof description !== 'string' || description.length === 0) {
|
|
80
|
+
return '';
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
// Find the first non-alphanumeric character or end of string
|
|
84
|
+
const match = description.match(/^[^a-zA-Z0-9]*/);
|
|
85
|
+
|
|
86
|
+
return match ? match[0] : '';
|
|
87
|
+
} catch (error) {
|
|
88
|
+
console.error('[require-test-description-prefix] Error in getFoundPrefix:', error);
|
|
89
|
+
|
|
90
|
+
return '';
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
/**
|
|
95
|
+
* Check and report on test function calls
|
|
96
|
+
*/
|
|
97
|
+
function checkTestFunction(node: any): void {
|
|
98
|
+
try {
|
|
99
|
+
if (!isTestFunction(node)) {
|
|
100
|
+
return;
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
// Check if there are arguments
|
|
104
|
+
if (!node.arguments || node.arguments.length === 0) {
|
|
105
|
+
return;
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
const firstArg = node.arguments[0];
|
|
109
|
+
|
|
110
|
+
// Only check string literals
|
|
111
|
+
if (firstArg.type !== 'Literal' || typeof firstArg.value !== 'string') {
|
|
112
|
+
return;
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
const description = firstArg.value;
|
|
116
|
+
|
|
117
|
+
if (!hasCorrectPrefix(description)) {
|
|
118
|
+
const foundPrefix = getFoundPrefix(description);
|
|
119
|
+
|
|
120
|
+
context.report({
|
|
121
|
+
node: firstArg,
|
|
122
|
+
messageId: foundPrefix ? 'wrongPrefix' : 'missingPrefix',
|
|
123
|
+
data: {
|
|
124
|
+
prefix,
|
|
125
|
+
found: foundPrefix,
|
|
126
|
+
},
|
|
127
|
+
fix(fixer) {
|
|
128
|
+
try {
|
|
129
|
+
const newText = `${prefix}${description}`;
|
|
130
|
+
|
|
131
|
+
return fixer.replaceText(firstArg, `'${newText}'`);
|
|
132
|
+
} catch (fixError) {
|
|
133
|
+
console.error('[require-test-description-prefix] Error in fix function:', fixError);
|
|
134
|
+
|
|
135
|
+
return null;
|
|
136
|
+
}
|
|
137
|
+
},
|
|
138
|
+
});
|
|
139
|
+
}
|
|
140
|
+
} catch (error) {
|
|
141
|
+
console.error('[require-test-description-prefix] Error in checkTestFunction:', error);
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
return {
|
|
146
|
+
CallExpression(node: any) {
|
|
147
|
+
checkTestFunction(node);
|
|
148
|
+
},
|
|
149
|
+
};
|
|
150
|
+
},
|
|
151
|
+
};
|
|
152
|
+
|
|
153
|
+
export default rule;
|
|
@@ -46,7 +46,7 @@ type AnyJson = any;
|
|
|
46
46
|
function safeParse(jsonPath: string): AnyJson | null {
|
|
47
47
|
try {
|
|
48
48
|
const content = readFileSync(jsonPath, 'utf8');
|
|
49
|
-
|
|
49
|
+
|
|
50
50
|
return JSON.parse(content);
|
|
51
51
|
} catch {
|
|
52
52
|
// Return null for any parsing errors (malformed JSON, file not found, etc.)
|
|
@@ -73,11 +73,11 @@ function safeParse(jsonPath: string): AnyJson | null {
|
|
|
73
73
|
* // └─────────┴─────────────────────────┴───────┘
|
|
74
74
|
* ```
|
|
75
75
|
*/
|
|
76
|
-
async function main() {
|
|
76
|
+
async function main(): Promise<void> {
|
|
77
77
|
// Find all ESLint configuration files, excluding node_modules
|
|
78
|
-
const files = await fg([ '**/.eslintrc.json' ], { ignore: [ '**/node_modules/**' ] });
|
|
78
|
+
const files: string[] = await fg([ '**/.eslintrc.json' ], { ignore: [ '**/node_modules/**' ] });
|
|
79
79
|
const ruleCounts: Record<string, number> = {};
|
|
80
|
-
const total = files.length;
|
|
80
|
+
const total: number = files.length;
|
|
81
81
|
|
|
82
82
|
// Process each configuration file
|
|
83
83
|
for (const p of files) {
|
|
@@ -93,7 +93,9 @@ async function main() {
|
|
|
93
93
|
}
|
|
94
94
|
|
|
95
95
|
// Sort rules by usage count (most used first)
|
|
96
|
-
const entries = Object.entries(ruleCounts).sort(
|
|
96
|
+
const entries = Object.entries(ruleCounts).sort(
|
|
97
|
+
(a: [ string, number ], b: [ string, number ]): number => b[1] - a[1]
|
|
98
|
+
);
|
|
97
99
|
|
|
98
100
|
// Display results
|
|
99
101
|
console.log(`[dynamo-eslintrc-audit] analyzed ${total} files`);
|
|
@@ -101,7 +103,7 @@ async function main() {
|
|
|
101
103
|
}
|
|
102
104
|
|
|
103
105
|
// Execute the main function and handle any errors
|
|
104
|
-
main().catch((err) => {
|
|
106
|
+
main().catch((err: Error): void => {
|
|
105
107
|
console.error(err);
|
|
106
108
|
process.exit(1);
|
|
107
109
|
});
|
|
Binary file
|