@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.
Files changed (67) hide show
  1. package/.vscode/settings.json +0 -2
  2. package/build/configs/base.js +29 -7
  3. package/build/configs/base.js.map +1 -1
  4. package/build/plugin/index.d.ts +8 -0
  5. package/build/plugin/index.d.ts.map +1 -1
  6. package/build/plugin/index.js +12 -0
  7. package/build/plugin/index.js.map +1 -1
  8. package/build/plugin/rules/explicit-types.d.ts.map +1 -1
  9. package/build/plugin/rules/explicit-types.js +16 -2
  10. package/build/plugin/rules/explicit-types.js.map +1 -1
  11. package/build/plugin/rules/import/import-order.d.ts.map +1 -1
  12. package/build/plugin/rules/import/import-order.js +0 -9
  13. package/build/plugin/rules/import/import-order.js.map +1 -1
  14. package/build/plugin/rules/import/no-js-import.d.ts.map +1 -1
  15. package/build/plugin/rules/import/no-js-import.js +12 -15
  16. package/build/plugin/rules/import/no-js-import.js.map +1 -1
  17. package/build/plugin/rules/prefer-enum-over-string-union.d.ts +4 -0
  18. package/build/plugin/rules/prefer-enum-over-string-union.d.ts.map +1 -0
  19. package/build/plugin/rules/prefer-enum-over-string-union.js +290 -0
  20. package/build/plugin/rules/prefer-enum-over-string-union.js.map +1 -0
  21. package/build/plugin/rules/prefer-enum-over-string-union.spec.d.ts +2 -0
  22. package/build/plugin/rules/prefer-enum-over-string-union.spec.d.ts.map +1 -0
  23. package/build/plugin/rules/prefer-enum-over-string-union.spec.js +505 -0
  24. package/build/plugin/rules/prefer-enum-over-string-union.spec.js.map +1 -0
  25. package/build/plugin/rules/prefer-interface-over-duplicate-types.d.ts +4 -0
  26. package/build/plugin/rules/prefer-interface-over-duplicate-types.d.ts.map +1 -0
  27. package/build/plugin/rules/prefer-interface-over-duplicate-types.js +231 -0
  28. package/build/plugin/rules/prefer-interface-over-duplicate-types.js.map +1 -0
  29. package/build/plugin/rules/prefer-interface-over-duplicate-types.spec.d.ts +2 -0
  30. package/build/plugin/rules/prefer-interface-over-duplicate-types.spec.d.ts.map +1 -0
  31. package/build/plugin/rules/prefer-interface-over-duplicate-types.spec.js +324 -0
  32. package/build/plugin/rules/prefer-interface-over-duplicate-types.spec.js.map +1 -0
  33. package/build/plugin/rules/require-jsdoc-description.d.ts +4 -0
  34. package/build/plugin/rules/require-jsdoc-description.d.ts.map +1 -0
  35. package/build/plugin/rules/require-jsdoc-description.js +379 -0
  36. package/build/plugin/rules/require-jsdoc-description.js.map +1 -0
  37. package/build/plugin/rules/require-jsdoc-description.spec.d.ts +2 -0
  38. package/build/plugin/rules/require-jsdoc-description.spec.d.ts.map +1 -0
  39. package/build/plugin/rules/require-jsdoc-description.spec.js +452 -0
  40. package/build/plugin/rules/require-jsdoc-description.spec.js.map +1 -0
  41. package/build/plugin/rules/require-test-description-prefix.d.ts +4 -0
  42. package/build/plugin/rules/require-test-description-prefix.d.ts.map +1 -0
  43. package/build/plugin/rules/require-test-description-prefix.js +135 -0
  44. package/build/plugin/rules/require-test-description-prefix.js.map +1 -0
  45. package/build/plugin/rules/require-test-description-prefix.spec.d.ts +2 -0
  46. package/build/plugin/rules/require-test-description-prefix.spec.d.ts.map +1 -0
  47. package/build/plugin/rules/require-test-description-prefix.spec.js +371 -0
  48. package/build/plugin/rules/require-test-description-prefix.spec.js.map +1 -0
  49. package/build/scripts/eslintrc-audit.js.map +1 -1
  50. package/futdevpro-dynamo-eslint-1.14.7.tgz +0 -0
  51. package/package.json +1 -1
  52. package/samples/package.json.example +1 -1
  53. package/src/configs/base.ts +45 -23
  54. package/src/plugin/index.ts +12 -0
  55. package/src/plugin/rules/explicit-types.ts +19 -2
  56. package/src/plugin/rules/import/import-order.ts +0 -9
  57. package/src/plugin/rules/import/no-js-import.ts +19 -17
  58. package/src/plugin/rules/prefer-enum-over-string-union.spec.ts +583 -0
  59. package/src/plugin/rules/prefer-enum-over-string-union.ts +333 -0
  60. package/src/plugin/rules/prefer-interface-over-duplicate-types.spec.ts +374 -0
  61. package/src/plugin/rules/prefer-interface-over-duplicate-types.ts +276 -0
  62. package/src/plugin/rules/require-jsdoc-description.spec.ts +542 -0
  63. package/src/plugin/rules/require-jsdoc-description.ts +436 -0
  64. package/src/plugin/rules/require-test-description-prefix.spec.ts +459 -0
  65. package/src/plugin/rules/require-test-description-prefix.ts +153 -0
  66. package/src/scripts/eslintrc-audit.ts +8 -6
  67. 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((a, b) => b[1] - a[1]);
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