kopytko-linter 0.1.0

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 (102) hide show
  1. package/README.md +190 -0
  2. package/dist/bin/kopytko-lint.d.ts +3 -0
  3. package/dist/bin/kopytko-lint.d.ts.map +1 -0
  4. package/dist/bin/kopytko-lint.js +118 -0
  5. package/dist/bin/kopytko-lint.js.map +1 -0
  6. package/dist/src/analysis/fsWrapper.d.ts +19 -0
  7. package/dist/src/analysis/fsWrapper.d.ts.map +1 -0
  8. package/dist/src/analysis/fsWrapper.js +62 -0
  9. package/dist/src/analysis/fsWrapper.js.map +1 -0
  10. package/dist/src/analysis/functionIndex.d.ts +19 -0
  11. package/dist/src/analysis/functionIndex.d.ts.map +1 -0
  12. package/dist/src/analysis/functionIndex.js +69 -0
  13. package/dist/src/analysis/functionIndex.js.map +1 -0
  14. package/dist/src/analysis/globMatcher.d.ts +8 -0
  15. package/dist/src/analysis/globMatcher.d.ts.map +1 -0
  16. package/dist/src/analysis/globMatcher.js +27 -0
  17. package/dist/src/analysis/globMatcher.js.map +1 -0
  18. package/dist/src/analysis/importParser.d.ts +22 -0
  19. package/dist/src/analysis/importParser.d.ts.map +1 -0
  20. package/dist/src/analysis/importParser.js +131 -0
  21. package/dist/src/analysis/importParser.js.map +1 -0
  22. package/dist/src/analysis/patternSiblings.d.ts +13 -0
  23. package/dist/src/analysis/patternSiblings.d.ts.map +1 -0
  24. package/dist/src/analysis/patternSiblings.js +91 -0
  25. package/dist/src/analysis/patternSiblings.js.map +1 -0
  26. package/dist/src/analysis/scopeAnalysis.d.ts +17 -0
  27. package/dist/src/analysis/scopeAnalysis.d.ts.map +1 -0
  28. package/dist/src/analysis/scopeAnalysis.js +180 -0
  29. package/dist/src/analysis/scopeAnalysis.js.map +1 -0
  30. package/dist/src/analysis/testUtils.d.ts +17 -0
  31. package/dist/src/analysis/testUtils.d.ts.map +1 -0
  32. package/dist/src/analysis/testUtils.js +124 -0
  33. package/dist/src/analysis/testUtils.js.map +1 -0
  34. package/dist/src/analysis/textUtils.d.ts +9 -0
  35. package/dist/src/analysis/textUtils.d.ts.map +1 -0
  36. package/dist/src/analysis/textUtils.js +45 -0
  37. package/dist/src/analysis/textUtils.js.map +1 -0
  38. package/dist/src/catalog/builtins.d.ts +26 -0
  39. package/dist/src/catalog/builtins.d.ts.map +1 -0
  40. package/dist/src/catalog/builtins.js +123 -0
  41. package/dist/src/catalog/builtins.js.map +1 -0
  42. package/dist/src/catalog/components.d.ts +8 -0
  43. package/dist/src/catalog/components.d.ts.map +1 -0
  44. package/dist/src/catalog/components.js +35 -0
  45. package/dist/src/catalog/components.js.map +1 -0
  46. package/dist/src/config.d.ts +23 -0
  47. package/dist/src/config.d.ts.map +1 -0
  48. package/dist/src/config.js +167 -0
  49. package/dist/src/config.js.map +1 -0
  50. package/dist/src/context.d.ts +22 -0
  51. package/dist/src/context.d.ts.map +1 -0
  52. package/dist/src/context.js +3 -0
  53. package/dist/src/context.js.map +1 -0
  54. package/dist/src/index.d.ts +28 -0
  55. package/dist/src/index.d.ts.map +1 -0
  56. package/dist/src/index.js +54 -0
  57. package/dist/src/index.js.map +1 -0
  58. package/dist/src/jsonc.d.ts +6 -0
  59. package/dist/src/jsonc.d.ts.map +1 -0
  60. package/dist/src/jsonc.js +46 -0
  61. package/dist/src/jsonc.js.map +1 -0
  62. package/dist/src/linter.d.ts +19 -0
  63. package/dist/src/linter.d.ts.map +1 -0
  64. package/dist/src/linter.js +274 -0
  65. package/dist/src/linter.js.map +1 -0
  66. package/dist/src/output/jsonFormatter.d.ts +3 -0
  67. package/dist/src/output/jsonFormatter.d.ts.map +1 -0
  68. package/dist/src/output/jsonFormatter.js +26 -0
  69. package/dist/src/output/jsonFormatter.js.map +1 -0
  70. package/dist/src/output/sarifFormatter.d.ts +3 -0
  71. package/dist/src/output/sarifFormatter.d.ts.map +1 -0
  72. package/dist/src/output/sarifFormatter.js +110 -0
  73. package/dist/src/output/sarifFormatter.js.map +1 -0
  74. package/dist/src/output/textFormatter.d.ts +3 -0
  75. package/dist/src/output/textFormatter.d.ts.map +1 -0
  76. package/dist/src/output/textFormatter.js +61 -0
  77. package/dist/src/output/textFormatter.js.map +1 -0
  78. package/dist/src/rules/identifierRules.d.ts +6 -0
  79. package/dist/src/rules/identifierRules.d.ts.map +1 -0
  80. package/dist/src/rules/identifierRules.js +322 -0
  81. package/dist/src/rules/identifierRules.js.map +1 -0
  82. package/dist/src/rules/importRules.d.ts +6 -0
  83. package/dist/src/rules/importRules.d.ts.map +1 -0
  84. package/dist/src/rules/importRules.js +164 -0
  85. package/dist/src/rules/importRules.js.map +1 -0
  86. package/dist/src/rules/index.d.ts +4 -0
  87. package/dist/src/rules/index.d.ts.map +1 -0
  88. package/dist/src/rules/index.js +21 -0
  89. package/dist/src/rules/index.js.map +1 -0
  90. package/dist/src/rules/syntaxRules.d.ts +6 -0
  91. package/dist/src/rules/syntaxRules.d.ts.map +1 -0
  92. package/dist/src/rules/syntaxRules.js +221 -0
  93. package/dist/src/rules/syntaxRules.js.map +1 -0
  94. package/dist/src/rules/testRules.d.ts +3 -0
  95. package/dist/src/rules/testRules.d.ts.map +1 -0
  96. package/dist/src/rules/testRules.js +79 -0
  97. package/dist/src/rules/testRules.js.map +1 -0
  98. package/dist/src/types.d.ts +54 -0
  99. package/dist/src/types.d.ts.map +1 -0
  100. package/dist/src/types.js +3 -0
  101. package/dist/src/types.js.map +1 -0
  102. package/package.json +52 -0
@@ -0,0 +1,21 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ALL_RULE_GROUPS = void 0;
4
+ const importRules_1 = require("./importRules");
5
+ const identifierRules_1 = require("./identifierRules");
6
+ const syntaxRules_1 = require("./syntaxRules");
7
+ const testRules_1 = require("./testRules");
8
+ /** All rule groups as single functions that check multiple codes internally. */
9
+ exports.ALL_RULE_GROUPS = [
10
+ { code: 'import/*', defaultSeverity: 'error', fn: importRules_1.checkImports },
11
+ { code: 'identifier/undefined-function', defaultSeverity: 'error', fn: identifierRules_1.checkUndefinedCalls },
12
+ { code: 'identifier/undefined-variable', defaultSeverity: 'error', fn: identifierRules_1.checkUndefinedVariables },
13
+ { code: 'identifier/shadows-builtin', defaultSeverity: 'error', fn: identifierRules_1.checkShadowedBuiltins },
14
+ { code: 'identifier/unused-parameter', defaultSeverity: 'hint', fn: identifierRules_1.checkUnusedParameters },
15
+ { code: 'throw/*', defaultSeverity: 'warning', fn: syntaxRules_1.checkThrowStatements },
16
+ { code: 'createobject/unknown-component', defaultSeverity: 'warning', fn: syntaxRules_1.checkCreateObjectArgs },
17
+ { code: 'syntax/trailing-comma', defaultSeverity: 'error', fn: syntaxRules_1.checkTrailingCommaSyntaxErrors },
18
+ { code: 'syntax/flow-outside-loop', defaultSeverity: 'error', fn: syntaxRules_1.checkLoopFlowControl },
19
+ { code: 'test/*', defaultSeverity: 'warning', fn: testRules_1.checkTestFileStructure },
20
+ ];
21
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/rules/index.ts"],"names":[],"mappings":";;;AACA,+CAA6C;AAC7C,uDAA+H;AAC/H,+CAAkI;AAClI,2CAAqD;AAErD,gFAAgF;AACnE,QAAA,eAAe,GAAqB;IAC/C,EAAE,IAAI,EAAE,UAAU,EAAE,eAAe,EAAE,OAAO,EAAE,EAAE,EAAE,0BAAY,EAAE;IAChE,EAAE,IAAI,EAAE,+BAA+B,EAAE,eAAe,EAAE,OAAO,EAAE,EAAE,EAAE,qCAAmB,EAAE;IAC5F,EAAE,IAAI,EAAE,+BAA+B,EAAE,eAAe,EAAE,OAAO,EAAE,EAAE,EAAE,yCAAuB,EAAE;IAChG,EAAE,IAAI,EAAE,4BAA4B,EAAE,eAAe,EAAE,OAAO,EAAE,EAAE,EAAE,uCAAqB,EAAE;IAC3F,EAAE,IAAI,EAAE,6BAA6B,EAAE,eAAe,EAAE,MAAM,EAAE,EAAE,EAAE,uCAAqB,EAAE;IAC3F,EAAE,IAAI,EAAE,SAAS,EAAE,eAAe,EAAE,SAAS,EAAE,EAAE,EAAE,kCAAoB,EAAE;IACzE,EAAE,IAAI,EAAE,gCAAgC,EAAE,eAAe,EAAE,SAAS,EAAE,EAAE,EAAE,mCAAqB,EAAE;IACjG,EAAE,IAAI,EAAE,uBAAuB,EAAE,eAAe,EAAE,OAAO,EAAE,EAAE,EAAE,4CAA8B,EAAE;IAC/F,EAAE,IAAI,EAAE,0BAA0B,EAAE,eAAe,EAAE,OAAO,EAAE,EAAE,EAAE,kCAAoB,EAAE;IACxF,EAAE,IAAI,EAAE,QAAQ,EAAE,eAAe,EAAE,SAAS,EAAE,EAAE,EAAE,kCAAsB,EAAE;CAC3E,CAAC"}
@@ -0,0 +1,6 @@
1
+ import type { LintDiagnostic, RuleContext } from '../types';
2
+ export declare function checkThrowStatements(ctx: RuleContext): LintDiagnostic[];
3
+ export declare function checkCreateObjectArgs(ctx: RuleContext): LintDiagnostic[];
4
+ export declare function checkTrailingCommaSyntaxErrors(ctx: RuleContext): LintDiagnostic[];
5
+ export declare function checkLoopFlowControl(ctx: RuleContext): LintDiagnostic[];
6
+ //# sourceMappingURL=syntaxRules.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"syntaxRules.d.ts","sourceRoot":"","sources":["../../../src/rules/syntaxRules.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,WAAW,EAAE,MAAM,UAAU,CAAC;AAG5D,wBAAgB,oBAAoB,CAAC,GAAG,EAAE,WAAW,GAAG,cAAc,EAAE,CAoFvE;AAID,wBAAgB,qBAAqB,CAAC,GAAG,EAAE,WAAW,GAAG,cAAc,EAAE,CAmCxE;AAED,wBAAgB,8BAA8B,CAAC,GAAG,EAAE,WAAW,GAAG,cAAc,EAAE,CA2BjF;AAWD,wBAAgB,oBAAoB,CAAC,GAAG,EAAE,WAAW,GAAG,cAAc,EAAE,CAyDvE"}
@@ -0,0 +1,221 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.checkThrowStatements = checkThrowStatements;
4
+ exports.checkCreateObjectArgs = checkCreateObjectArgs;
5
+ exports.checkTrailingCommaSyntaxErrors = checkTrailingCommaSyntaxErrors;
6
+ exports.checkLoopFlowControl = checkLoopFlowControl;
7
+ const components_1 = require("../catalog/components");
8
+ function checkThrowStatements(ctx) {
9
+ const { lines, filePath, config } = ctx;
10
+ if (config['throw/invalid-value'] === 'off' && config['throw/missing-message'] === 'off')
11
+ return [];
12
+ const diagnostics = [];
13
+ for (let lineIdx = 0; lineIdx < lines.length; lineIdx++) {
14
+ const raw = lines[lineIdx];
15
+ if (/^\s*'/.test(raw) || /^\s*rem\b/i.test(raw))
16
+ continue;
17
+ const throwMatch = /^\s*throw\b\s+(.+?)(?:\s*'.*)?$/i.exec(raw);
18
+ if (!throwMatch)
19
+ continue;
20
+ let expr = throwMatch[1].trim();
21
+ if (expr.startsWith('(') && expr.endsWith(')')) {
22
+ let depth = 0;
23
+ let isOuterWrapped = true;
24
+ for (let i = 0; i < expr.length - 1; i++) {
25
+ if (expr[i] === '(')
26
+ depth++;
27
+ else if (expr[i] === ')') {
28
+ depth--;
29
+ if (depth === 0) {
30
+ isOuterWrapped = false;
31
+ break;
32
+ }
33
+ }
34
+ }
35
+ if (isOuterWrapped)
36
+ expr = expr.slice(1, -1).trim();
37
+ }
38
+ const throwKeywordStart = raw.search(/\bthrow\b/i);
39
+ const throwRange = {
40
+ line: lineIdx,
41
+ column: throwKeywordStart,
42
+ endLine: lineIdx,
43
+ endColumn: throwKeywordStart + 5,
44
+ filePath,
45
+ };
46
+ if (config['throw/invalid-value'] !== 'off') {
47
+ if (/^-?[\d.]/.test(expr)) {
48
+ diagnostics.push({
49
+ severity: config['throw/invalid-value'] ?? 'warning',
50
+ code: 'throw/invalid-value',
51
+ message: '`throw` requires a string or an associative array with a "message" field — numeric literals are not valid throw values.',
52
+ ...throwRange,
53
+ });
54
+ continue;
55
+ }
56
+ if (expr.startsWith('[')) {
57
+ diagnostics.push({
58
+ severity: config['throw/invalid-value'] ?? 'warning',
59
+ code: 'throw/invalid-value',
60
+ message: '`throw` requires a string or an associative array with a "message" field — array literals are not valid throw values.',
61
+ ...throwRange,
62
+ });
63
+ continue;
64
+ }
65
+ if (/^invalid$/i.test(expr)) {
66
+ diagnostics.push({
67
+ severity: config['throw/invalid-value'] ?? 'warning',
68
+ code: 'throw/invalid-value',
69
+ message: '`throw` requires a string or an associative array with a "message" field — `invalid` is not a valid throw value.',
70
+ ...throwRange,
71
+ });
72
+ continue;
73
+ }
74
+ }
75
+ if (config['throw/missing-message'] !== 'off') {
76
+ if (expr.startsWith('{') && expr.endsWith('}')) {
77
+ if (!/\bmessage\s*:/i.test(expr)) {
78
+ diagnostics.push({
79
+ severity: config['throw/missing-message'] ?? 'warning',
80
+ code: 'throw/missing-message',
81
+ message: 'Thrown associative array should include a "message" field (e.g. `{ message: "error description", number: -1 }`).',
82
+ ...throwRange,
83
+ });
84
+ }
85
+ }
86
+ }
87
+ }
88
+ return diagnostics;
89
+ }
90
+ const CREATE_OBJECT_ARG_RE = /\bCreateObject\s*\(\s*"([^"]*)"/gi;
91
+ function checkCreateObjectArgs(ctx) {
92
+ const { lines, filePath, config } = ctx;
93
+ if (config['createobject/unknown-component'] === 'off')
94
+ return [];
95
+ const diagnostics = [];
96
+ for (let lineIdx = 0; lineIdx < lines.length; lineIdx++) {
97
+ const raw = lines[lineIdx];
98
+ if (/^\s*'/.test(raw) || /^\s*rem\b/i.test(raw))
99
+ continue;
100
+ CREATE_OBJECT_ARG_RE.lastIndex = 0;
101
+ let match;
102
+ while ((match = CREATE_OBJECT_ARG_RE.exec(raw)) !== null) {
103
+ const componentName = match[1];
104
+ if (!componentName)
105
+ continue;
106
+ if (componentName.toLowerCase() === 'rosgnode')
107
+ continue;
108
+ if (!(0, components_1.findComponent)(componentName)) {
109
+ const argStart = raw.indexOf(`"${componentName}"`, match.index);
110
+ if (argStart < 0)
111
+ continue;
112
+ diagnostics.push({
113
+ severity: config['createobject/unknown-component'] ?? 'warning',
114
+ code: 'createobject/unknown-component',
115
+ message: `Unknown BrightScript component "${componentName}". Check the component name spelling.`,
116
+ line: lineIdx,
117
+ column: argStart,
118
+ endLine: lineIdx,
119
+ endColumn: argStart + componentName.length + 2,
120
+ filePath,
121
+ });
122
+ }
123
+ }
124
+ }
125
+ return diagnostics;
126
+ }
127
+ function checkTrailingCommaSyntaxErrors(ctx) {
128
+ const { lines, filePath, config } = ctx;
129
+ if (config['syntax/trailing-comma'] === 'off')
130
+ return [];
131
+ const diagnostics = [];
132
+ for (let lineIdx = 0; lineIdx < lines.length; lineIdx++) {
133
+ const raw = lines[lineIdx];
134
+ if (/^\s*'/.test(raw) || /^\s*rem\b/i.test(raw))
135
+ continue;
136
+ const returnCommaMatch = /^\s*return\b\s+.+,\s*$/.exec(raw);
137
+ if (returnCommaMatch) {
138
+ const commaPos = raw.lastIndexOf(',');
139
+ diagnostics.push({
140
+ severity: config['syntax/trailing-comma'] ?? 'error',
141
+ code: 'syntax/trailing-comma',
142
+ message: 'Trailing comma after return value is a syntax error — the code will not compile.',
143
+ line: lineIdx,
144
+ column: commaPos,
145
+ endLine: lineIdx,
146
+ endColumn: commaPos + 1,
147
+ filePath,
148
+ });
149
+ }
150
+ }
151
+ return diagnostics;
152
+ }
153
+ function popUntil(stack, target) {
154
+ for (let i = stack.length - 1; i >= 0; i--) {
155
+ if (stack[i] === target) {
156
+ stack.splice(i, 1);
157
+ return;
158
+ }
159
+ }
160
+ }
161
+ function checkLoopFlowControl(ctx) {
162
+ const { lines, filePath, config } = ctx;
163
+ if (config['syntax/flow-outside-loop'] === 'off')
164
+ return [];
165
+ const diagnostics = [];
166
+ const stack = [];
167
+ for (let i = 0; i < lines.length; i++) {
168
+ const raw = lines[i];
169
+ if (/^\s*'/.test(raw) || /^\s*rem\b/i.test(raw))
170
+ continue;
171
+ const trimmed = raw.replace(/'.*$/, '').trim();
172
+ if (!trimmed)
173
+ continue;
174
+ const lower = trimmed.toLowerCase();
175
+ if (/^for\b/i.test(lower)) {
176
+ stack.push('for');
177
+ }
178
+ else if (/^while\b/i.test(lower)) {
179
+ stack.push('while');
180
+ }
181
+ else if (/^(?:sub|function)\b/i.test(lower) && /\(/i.test(lower)) {
182
+ stack.push('other');
183
+ }
184
+ else if (/^(?:if)\b/i.test(lower) && /\bthen\s*$/i.test(lower)) {
185
+ stack.push('other');
186
+ }
187
+ else if (/^(?:try)\b/i.test(lower)) {
188
+ stack.push('other');
189
+ }
190
+ if (/^end\s*for\b/i.test(lower) || /^endfor\b/i.test(lower) || /^next\b/i.test(lower)) {
191
+ popUntil(stack, 'for');
192
+ }
193
+ else if (/^end\s*while\b/i.test(lower) || /^endwhile\b/i.test(lower)) {
194
+ popUntil(stack, 'while');
195
+ }
196
+ else if (/^end\s*(?:sub|function|if|try)\b/i.test(lower) || /^end(?:sub|function|if|try)\b/i.test(lower)) {
197
+ popUntil(stack, 'other');
198
+ }
199
+ const flowMatch = /^(continue|exit)\s+(for|while)\b/i.exec(lower);
200
+ if (flowMatch) {
201
+ const keyword = flowMatch[1].toLowerCase();
202
+ const loopType = flowMatch[2].toLowerCase();
203
+ const insideCorrectLoop = stack.includes(loopType);
204
+ if (!insideCorrectLoop) {
205
+ const col = raw.search(/\S/);
206
+ diagnostics.push({
207
+ severity: config['syntax/flow-outside-loop'] ?? 'error',
208
+ code: 'syntax/flow-outside-loop',
209
+ message: `\`${keyword} ${loopType}\` is only valid inside a \`${loopType}\` loop body.`,
210
+ line: i,
211
+ column: col,
212
+ endLine: i,
213
+ endColumn: col + trimmed.length,
214
+ filePath,
215
+ });
216
+ }
217
+ }
218
+ }
219
+ return diagnostics;
220
+ }
221
+ //# sourceMappingURL=syntaxRules.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"syntaxRules.js","sourceRoot":"","sources":["../../../src/rules/syntaxRules.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import type { LintDiagnostic, RuleContext } from '../types';
2
+ export declare function checkTestFileStructure(ctx: RuleContext): LintDiagnostic[];
3
+ //# sourceMappingURL=testRules.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testRules.d.ts","sourceRoot":"","sources":["../../../src/rules/testRules.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,WAAW,EAAE,MAAM,UAAU,CAAC;AAG5D,wBAAgB,sBAAsB,CAAC,GAAG,EAAE,WAAW,GAAG,cAAc,EAAE,CA8EzE"}
@@ -0,0 +1,79 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.checkTestFileStructure = checkTestFileStructure;
4
+ const functionIndex_1 = require("../analysis/functionIndex");
5
+ function checkTestFileStructure(ctx) {
6
+ const { lines, filePath, imports, config, lintContext } = ctx;
7
+ if (!lintContext.isTestFile(filePath))
8
+ return [];
9
+ if (config['test/missing-mock-annotation'] === 'off' && config['test/missing-return-ts'] === 'off')
10
+ return [];
11
+ const diagnostics = [];
12
+ let hasTestSuiteFunc = false;
13
+ let testSuiteFuncLine = -1;
14
+ let lastReturnTsLine = -1;
15
+ // Build set of known mocked function/method names from resolved @mock files
16
+ const mockedIdentifiers = new Set();
17
+ for (const imp of imports) {
18
+ if (!imp.isMock)
19
+ continue;
20
+ const resolved = lintContext.resolveImportPath(imp.importPath, imp.fromModule);
21
+ if (!resolved)
22
+ continue;
23
+ const text = lintContext.readFile(resolved);
24
+ if (text === null)
25
+ continue;
26
+ for (const fn of (0, functionIndex_1.parseFunctionDefs)(text, resolved)) {
27
+ mockedIdentifiers.add(fn.nameLower);
28
+ }
29
+ for (const method of (0, functionIndex_1.parseInnerMethodDefs)(text, resolved)) {
30
+ mockedIdentifiers.add(method.nameLower);
31
+ }
32
+ }
33
+ for (let i = 0; i < lines.length; i++) {
34
+ const line = lines[i];
35
+ if (/^\s*function\s+TestSuite__\w+/i.test(line)) {
36
+ hasTestSuiteFunc = true;
37
+ testSuiteFuncLine = i;
38
+ }
39
+ if (/^\s*return\s+ts\s*$/i.test(line)) {
40
+ lastReturnTsLine = i;
41
+ }
42
+ // Check mockFunction("X") — X should reference a function from a @mock'ed file
43
+ if (mockedIdentifiers.size > 0 && config['test/missing-mock-annotation'] !== 'off') {
44
+ const mockFuncRe = /mockFunction\s*\(\s*"([^"]+)"/g;
45
+ let mfMatch;
46
+ while ((mfMatch = mockFuncRe.exec(line)) !== null) {
47
+ const mockTarget = mfMatch[1];
48
+ const topLevel = mockTarget.includes('.') ? mockTarget.split('.')[0] : mockTarget;
49
+ if (!mockedIdentifiers.has(topLevel.toLowerCase())) {
50
+ const col = line.indexOf(mfMatch[0]);
51
+ diagnostics.push({
52
+ severity: config['test/missing-mock-annotation'] ?? 'warning',
53
+ code: 'test/missing-mock-annotation',
54
+ message: `"${topLevel}" is not defined in any \`@mock\`'ed file. Add a \`' @mock\` annotation for the file that defines "${topLevel}".`,
55
+ line: i,
56
+ column: col,
57
+ endLine: i,
58
+ endColumn: col + mfMatch[0].length + 1,
59
+ filePath,
60
+ });
61
+ }
62
+ }
63
+ }
64
+ }
65
+ if (config['test/missing-return-ts'] !== 'off' && hasTestSuiteFunc && lastReturnTsLine < 0) {
66
+ diagnostics.push({
67
+ severity: config['test/missing-return-ts'] ?? 'warning',
68
+ code: 'test/missing-return-ts',
69
+ message: 'Test suite function should end with `return ts` to return the suite object to the test runner.',
70
+ line: testSuiteFuncLine,
71
+ column: 0,
72
+ endLine: testSuiteFuncLine,
73
+ endColumn: Number.MAX_SAFE_INTEGER,
74
+ filePath,
75
+ });
76
+ }
77
+ return diagnostics;
78
+ }
79
+ //# sourceMappingURL=testRules.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"testRules.js","sourceRoot":"","sources":["../../../src/rules/testRules.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,54 @@
1
+ import type { LintContext } from './context';
2
+ export type LintSeverity = 'error' | 'warning' | 'info' | 'hint';
3
+ export interface LintDiagnostic {
4
+ code: string;
5
+ message: string;
6
+ severity: LintSeverity;
7
+ line: number;
8
+ column: number;
9
+ endLine?: number;
10
+ endColumn?: number;
11
+ filePath: string;
12
+ }
13
+ export interface LintResult {
14
+ diagnostics: LintDiagnostic[];
15
+ fileCount: number;
16
+ errorCount: number;
17
+ warningCount: number;
18
+ infoCount: number;
19
+ hintCount: number;
20
+ }
21
+ export interface GeneratedModuleConfig {
22
+ path: string;
23
+ functions: string[];
24
+ }
25
+ export interface KopytkoImport {
26
+ raw: string;
27
+ importPath: string;
28
+ fromModule?: string;
29
+ line: number;
30
+ isMock?: boolean;
31
+ }
32
+ export interface FunctionDefinition {
33
+ name: string;
34
+ nameLower: string;
35
+ line: number;
36
+ column: number;
37
+ filePath: string;
38
+ signature: string;
39
+ }
40
+ export interface RuleContext {
41
+ filePath: string;
42
+ lines: string[];
43
+ imports: KopytkoImport[];
44
+ config: RuleConfig;
45
+ lintContext: LintContext;
46
+ }
47
+ export type RuleConfig = Record<string, LintSeverity | 'off'>;
48
+ export type RuleFn = (ctx: RuleContext) => LintDiagnostic[];
49
+ export interface RuleDefinition {
50
+ code: string;
51
+ defaultSeverity: LintSeverity;
52
+ fn: RuleFn;
53
+ }
54
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,WAAW,CAAC;AAE7C,MAAM,MAAM,YAAY,GAAG,OAAO,GAAG,SAAS,GAAG,MAAM,GAAG,MAAM,CAAC;AAEjE,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;IAChB,QAAQ,EAAE,YAAY,CAAC;IACvB,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED,MAAM,WAAW,UAAU;IACzB,WAAW,EAAE,cAAc,EAAE,CAAC;IAC9B,SAAS,EAAE,MAAM,CAAC;IAClB,UAAU,EAAE,MAAM,CAAC;IACnB,YAAY,EAAE,MAAM,CAAC;IACrB,SAAS,EAAE,MAAM,CAAC;IAClB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,qBAAqB;IACpC,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,EAAE,CAAC;CACrB;AAED,MAAM,WAAW,aAAa;IAC5B,GAAG,EAAE,MAAM,CAAC;IACZ,UAAU,EAAE,MAAM,CAAC;IACnB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,CAAC,EAAE,OAAO,CAAC;CAClB;AAED,MAAM,WAAW,kBAAkB;IACjC,IAAI,EAAE,MAAM,CAAC;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,QAAQ,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,MAAM,WAAW,WAAW;IAC1B,QAAQ,EAAE,MAAM,CAAC;IACjB,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,OAAO,EAAE,aAAa,EAAE,CAAC;IACzB,MAAM,EAAE,UAAU,CAAC;IACnB,WAAW,EAAE,WAAW,CAAC;CAC1B;AAED,MAAM,MAAM,UAAU,GAAG,MAAM,CAAC,MAAM,EAAE,YAAY,GAAG,KAAK,CAAC,CAAC;AAE9D,MAAM,MAAM,MAAM,GAAG,CAAC,GAAG,EAAE,WAAW,KAAK,cAAc,EAAE,CAAC;AAE5D,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,eAAe,EAAE,YAAY,CAAC;IAC9B,EAAE,EAAE,MAAM,CAAC;CACZ"}
@@ -0,0 +1,3 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ //# sourceMappingURL=types.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.js","sourceRoot":"","sources":["../../src/types.ts"],"names":[],"mappings":""}
package/package.json ADDED
@@ -0,0 +1,52 @@
1
+ {
2
+ "name": "kopytko-linter",
3
+ "version": "0.1.0",
4
+ "description": "BrightScript linter for the Kopytko ecosystem. Use as a CLI tool or import as a library.",
5
+ "license": "MIT",
6
+ "author": "bchelkowski",
7
+ "repository": {
8
+ "type": "git",
9
+ "url": "git+https://github.com/bchelkowski/vscode-kopytko.git",
10
+ "directory": "packages/kopytko-linter"
11
+ },
12
+ "keywords": [
13
+ "brightscript",
14
+ "kopytko",
15
+ "roku",
16
+ "linter",
17
+ "diagnostics",
18
+ "static-analysis"
19
+ ],
20
+ "engines": {
21
+ "node": ">=24.0.0"
22
+ },
23
+ "main": "./dist/src/index.js",
24
+ "types": "./dist/src/index.d.ts",
25
+ "bin": {
26
+ "kopytko-lint": "dist/bin/kopytko-lint.js"
27
+ },
28
+ "files": [
29
+ "dist",
30
+ "README.md"
31
+ ],
32
+ "scripts": {
33
+ "build": "tsc -p tsconfig.json",
34
+ "test": "mocha",
35
+ "lint": "eslint \"./src/**/*.ts\" --ext .ts",
36
+ "clean": "node -e \"require('fs').rmSync('dist', {recursive: true, force: true})\"",
37
+ "prepublishOnly": "npm run clean && npm run build"
38
+ },
39
+ "devDependencies": {
40
+ "@types/mocha": "^10.0.10",
41
+ "@types/node": "^25.9.2",
42
+ "chai": "^6.2.2",
43
+ "mocha": "^11.7.6",
44
+ "source-map-support": "^0.5.21",
45
+ "tsx": "^4.20.3",
46
+ "typescript": "^6.0.3"
47
+ },
48
+ "overrides": {
49
+ "serialize-javascript": "^7.0.5",
50
+ "diff": "^8.0.3"
51
+ }
52
+ }