@rcrsr/rill-cli 0.6.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 (171) hide show
  1. package/LICENSE +21 -0
  2. package/dist/check/config.d.ts +20 -0
  3. package/dist/check/config.d.ts.map +1 -0
  4. package/dist/check/config.js +151 -0
  5. package/dist/check/config.js.map +1 -0
  6. package/dist/check/fixer.d.ts +39 -0
  7. package/dist/check/fixer.d.ts.map +1 -0
  8. package/dist/check/fixer.js +119 -0
  9. package/dist/check/fixer.js.map +1 -0
  10. package/dist/check/index.d.ts +10 -0
  11. package/dist/check/index.d.ts.map +1 -0
  12. package/dist/check/index.js +21 -0
  13. package/dist/check/index.js.map +1 -0
  14. package/dist/check/rules/anti-patterns.d.ts +65 -0
  15. package/dist/check/rules/anti-patterns.d.ts.map +1 -0
  16. package/dist/check/rules/anti-patterns.js +481 -0
  17. package/dist/check/rules/anti-patterns.js.map +1 -0
  18. package/dist/check/rules/closures.d.ts +66 -0
  19. package/dist/check/rules/closures.d.ts.map +1 -0
  20. package/dist/check/rules/closures.js +370 -0
  21. package/dist/check/rules/closures.js.map +1 -0
  22. package/dist/check/rules/collections.d.ts +90 -0
  23. package/dist/check/rules/collections.d.ts.map +1 -0
  24. package/dist/check/rules/collections.js +373 -0
  25. package/dist/check/rules/collections.js.map +1 -0
  26. package/dist/check/rules/conditionals.d.ts +41 -0
  27. package/dist/check/rules/conditionals.d.ts.map +1 -0
  28. package/dist/check/rules/conditionals.js +134 -0
  29. package/dist/check/rules/conditionals.js.map +1 -0
  30. package/dist/check/rules/flow.d.ts +46 -0
  31. package/dist/check/rules/flow.d.ts.map +1 -0
  32. package/dist/check/rules/flow.js +206 -0
  33. package/dist/check/rules/flow.js.map +1 -0
  34. package/dist/check/rules/formatting.d.ts +143 -0
  35. package/dist/check/rules/formatting.d.ts.map +1 -0
  36. package/dist/check/rules/formatting.js +656 -0
  37. package/dist/check/rules/formatting.js.map +1 -0
  38. package/dist/check/rules/helpers.d.ts +26 -0
  39. package/dist/check/rules/helpers.d.ts.map +1 -0
  40. package/dist/check/rules/helpers.js +66 -0
  41. package/dist/check/rules/helpers.js.map +1 -0
  42. package/dist/check/rules/index.d.ts +21 -0
  43. package/dist/check/rules/index.d.ts.map +1 -0
  44. package/dist/check/rules/index.js +78 -0
  45. package/dist/check/rules/index.js.map +1 -0
  46. package/dist/check/rules/loops.d.ts +77 -0
  47. package/dist/check/rules/loops.d.ts.map +1 -0
  48. package/dist/check/rules/loops.js +310 -0
  49. package/dist/check/rules/loops.js.map +1 -0
  50. package/dist/check/rules/naming.d.ts +21 -0
  51. package/dist/check/rules/naming.d.ts.map +1 -0
  52. package/dist/check/rules/naming.js +174 -0
  53. package/dist/check/rules/naming.js.map +1 -0
  54. package/dist/check/rules/strings.d.ts +28 -0
  55. package/dist/check/rules/strings.d.ts.map +1 -0
  56. package/dist/check/rules/strings.js +79 -0
  57. package/dist/check/rules/strings.js.map +1 -0
  58. package/dist/check/rules/types.d.ts +41 -0
  59. package/dist/check/rules/types.d.ts.map +1 -0
  60. package/dist/check/rules/types.js +167 -0
  61. package/dist/check/rules/types.js.map +1 -0
  62. package/dist/check/types.d.ts +112 -0
  63. package/dist/check/types.d.ts.map +1 -0
  64. package/dist/check/types.js +6 -0
  65. package/dist/check/types.js.map +1 -0
  66. package/dist/check/validator.d.ts +18 -0
  67. package/dist/check/validator.d.ts.map +1 -0
  68. package/dist/check/validator.js +110 -0
  69. package/dist/check/validator.js.map +1 -0
  70. package/dist/check/visitor.d.ts +33 -0
  71. package/dist/check/visitor.d.ts.map +1 -0
  72. package/dist/check/visitor.js +259 -0
  73. package/dist/check/visitor.js.map +1 -0
  74. package/dist/cli-check.d.ts +43 -0
  75. package/dist/cli-check.d.ts.map +1 -0
  76. package/dist/cli-check.js +366 -0
  77. package/dist/cli-check.js.map +1 -0
  78. package/dist/cli-error-enrichment.d.ts +73 -0
  79. package/dist/cli-error-enrichment.d.ts.map +1 -0
  80. package/dist/cli-error-enrichment.js +205 -0
  81. package/dist/cli-error-enrichment.js.map +1 -0
  82. package/dist/cli-error-formatter.d.ts +45 -0
  83. package/dist/cli-error-formatter.d.ts.map +1 -0
  84. package/dist/cli-error-formatter.js +218 -0
  85. package/dist/cli-error-formatter.js.map +1 -0
  86. package/dist/cli-eval.d.ts +15 -0
  87. package/dist/cli-eval.d.ts.map +1 -0
  88. package/dist/cli-eval.js +116 -0
  89. package/dist/cli-eval.js.map +1 -0
  90. package/dist/cli-exec.d.ts +58 -0
  91. package/dist/cli-exec.d.ts.map +1 -0
  92. package/dist/cli-exec.js +326 -0
  93. package/dist/cli-exec.js.map +1 -0
  94. package/dist/cli-explain.d.ts +24 -0
  95. package/dist/cli-explain.d.ts.map +1 -0
  96. package/dist/cli-explain.js +68 -0
  97. package/dist/cli-explain.js.map +1 -0
  98. package/dist/cli-lsp-diagnostic.d.ts +35 -0
  99. package/dist/cli-lsp-diagnostic.d.ts.map +1 -0
  100. package/dist/cli-lsp-diagnostic.js +98 -0
  101. package/dist/cli-lsp-diagnostic.js.map +1 -0
  102. package/dist/cli-module-loader.d.ts +19 -0
  103. package/dist/cli-module-loader.d.ts.map +1 -0
  104. package/dist/cli-module-loader.js +83 -0
  105. package/dist/cli-module-loader.js.map +1 -0
  106. package/dist/cli-shared.d.ts +62 -0
  107. package/dist/cli-shared.d.ts.map +1 -0
  108. package/dist/cli-shared.js +158 -0
  109. package/dist/cli-shared.js.map +1 -0
  110. package/dist/cli.d.ts +13 -0
  111. package/dist/cli.d.ts.map +1 -0
  112. package/dist/cli.js +62 -0
  113. package/dist/cli.js.map +1 -0
  114. package/dist/test-internal-import.d.ts +2 -0
  115. package/dist/test-internal-import.d.ts.map +1 -0
  116. package/dist/test-internal-import.js +7 -0
  117. package/dist/test-internal-import.js.map +1 -0
  118. package/package.json +24 -0
  119. package/src/check/config.ts +202 -0
  120. package/src/check/fixer.ts +174 -0
  121. package/src/check/index.ts +39 -0
  122. package/src/check/rules/anti-patterns.ts +585 -0
  123. package/src/check/rules/closures.ts +445 -0
  124. package/src/check/rules/collections.ts +437 -0
  125. package/src/check/rules/conditionals.ts +155 -0
  126. package/src/check/rules/flow.ts +262 -0
  127. package/src/check/rules/formatting.ts +811 -0
  128. package/src/check/rules/helpers.ts +89 -0
  129. package/src/check/rules/index.ts +140 -0
  130. package/src/check/rules/loops.ts +372 -0
  131. package/src/check/rules/naming.ts +242 -0
  132. package/src/check/rules/strings.ts +104 -0
  133. package/src/check/rules/types.ts +214 -0
  134. package/src/check/types.ts +163 -0
  135. package/src/check/validator.ts +136 -0
  136. package/src/check/visitor.ts +338 -0
  137. package/src/cli-check.ts +456 -0
  138. package/src/cli-error-enrichment.ts +274 -0
  139. package/src/cli-error-formatter.ts +313 -0
  140. package/src/cli-eval.ts +145 -0
  141. package/src/cli-exec.ts +408 -0
  142. package/src/cli-explain.ts +76 -0
  143. package/src/cli-lsp-diagnostic.ts +132 -0
  144. package/src/cli-module-loader.ts +101 -0
  145. package/src/cli-shared.ts +187 -0
  146. package/tests/check/cli-check.test.ts +189 -0
  147. package/tests/check/config.test.ts +350 -0
  148. package/tests/check/fixer.test.ts +373 -0
  149. package/tests/check/format-diagnostics.test.ts +327 -0
  150. package/tests/check/rules/anti-patterns.test.ts +467 -0
  151. package/tests/check/rules/closures.test.ts +192 -0
  152. package/tests/check/rules/collections.test.ts +380 -0
  153. package/tests/check/rules/conditionals.test.ts +185 -0
  154. package/tests/check/rules/flow.test.ts +250 -0
  155. package/tests/check/rules/formatting.test.ts +755 -0
  156. package/tests/check/rules/loops.test.ts +334 -0
  157. package/tests/check/rules/naming.test.ts +336 -0
  158. package/tests/check/rules/strings.test.ts +129 -0
  159. package/tests/check/rules/types.test.ts +257 -0
  160. package/tests/check/validator.test.ts +444 -0
  161. package/tests/check/visitor.test.ts +171 -0
  162. package/tests/cli/check.test.ts +801 -0
  163. package/tests/cli/error-enrichment.test.ts +510 -0
  164. package/tests/cli/error-formatter.test.ts +631 -0
  165. package/tests/cli/eval.test.ts +85 -0
  166. package/tests/cli/exec.test.ts +537 -0
  167. package/tests/cli-explain.test.ts +249 -0
  168. package/tests/cli-lsp-diagnostic.test.ts +202 -0
  169. package/tests/cli-shared.test.ts +439 -0
  170. package/tsconfig.json +9 -0
  171. package/tsconfig.tsbuildinfo +1 -0
@@ -0,0 +1,89 @@
1
+ /**
2
+ * Shared Helper Functions
3
+ * Common utilities used across validation rules.
4
+ */
5
+
6
+ import type {
7
+ ExpressionNode,
8
+ PipeChainNode,
9
+ PostfixExprNode,
10
+ VariableNode,
11
+ } from '@rcrsr/rill';
12
+
13
+ /**
14
+ * Extract source line at location for context display.
15
+ * Splits source by newlines, retrieves the specified line (1-indexed), and trims it.
16
+ */
17
+ export function extractContextLine(line: number, source: string): string {
18
+ const lines = source.split('\n');
19
+ const sourceLine = lines[line - 1];
20
+ return sourceLine ? sourceLine.trim() : '';
21
+ }
22
+
23
+ /**
24
+ * Detect if expression is a bare $ (pipe variable) reference.
25
+ * Used by IMPLICIT_DOLLAR_* rules to detect replaceable patterns.
26
+ *
27
+ * Returns true only for single bare $, not $var or $.field or $[0].
28
+ * O(1) depth traversal (max 3 node levels): PipeChain -> ArithHead -> PostfixExpr -> Variable.
29
+ *
30
+ * Distinct from containsBareReference() in closures.ts:
31
+ * - isBareReference(): O(1) single-node check, answers "is this exact node a bare $?"
32
+ * - containsBareReference(): Recursive AST walker, answers "does this subtree contain any bare $?"
33
+ *
34
+ * @param expr - Expression node to check
35
+ * @returns true if expr is a bare $ reference, false otherwise
36
+ */
37
+ export function isBareReference(
38
+ expr: ExpressionNode | null | undefined
39
+ ): boolean {
40
+ // Defensive: handle null/undefined input
41
+ if (!expr) {
42
+ return false;
43
+ }
44
+
45
+ // Expression is PipeChain
46
+ if (expr.type !== 'PipeChain') {
47
+ return false;
48
+ }
49
+
50
+ const pipeChain = expr as PipeChainNode;
51
+
52
+ // Must have no pipe targets (just the head)
53
+ if (pipeChain.pipes.length > 0 || pipeChain.terminator !== null) {
54
+ return false;
55
+ }
56
+
57
+ const head = pipeChain.head;
58
+
59
+ // ArithHead can be BinaryExpr, UnaryExpr, or PostfixExpr
60
+ // For bare $, we need PostfixExpr
61
+ if (head.type !== 'PostfixExpr') {
62
+ return false;
63
+ }
64
+
65
+ const postfix = head as PostfixExprNode;
66
+
67
+ // Must have no method calls (just the primary)
68
+ if (postfix.methods.length > 0) {
69
+ return false;
70
+ }
71
+
72
+ const primary = postfix.primary;
73
+
74
+ // Primary must be a Variable
75
+ if (primary.type !== 'Variable') {
76
+ return false;
77
+ }
78
+
79
+ const variable = primary as VariableNode;
80
+
81
+ // Must be pipe variable ($) with no access chain, default value, or existence check
82
+ return (
83
+ variable.isPipeVar &&
84
+ variable.name === null &&
85
+ variable.accessChain.length === 0 &&
86
+ variable.defaultValue === null &&
87
+ variable.existenceCheck === null
88
+ );
89
+ }
@@ -0,0 +1,140 @@
1
+ /**
2
+ * Validation Rules Registry
3
+ * Barrel export for all validation rules.
4
+ */
5
+
6
+ import type { ValidationRule } from '../types.js';
7
+ import { NAMING_SNAKE_CASE } from './naming.js';
8
+ import { CAPTURE_INLINE_CHAIN, CAPTURE_BEFORE_BRANCH } from './flow.js';
9
+ import {
10
+ BREAK_IN_PARALLEL,
11
+ PREFER_MAP,
12
+ FOLD_INTERMEDIATES,
13
+ FILTER_NEGATION,
14
+ METHOD_SHORTHAND,
15
+ } from './collections.js';
16
+ import { LOOP_ACCUMULATOR, PREFER_DO_WHILE, USE_EACH } from './loops.js';
17
+ import { USE_DEFAULT_OPERATOR, CONDITION_TYPE } from './conditionals.js';
18
+ import {
19
+ CLOSURE_BARE_DOLLAR,
20
+ CLOSURE_BRACES,
21
+ CLOSURE_LATE_BINDING,
22
+ } from './closures.js';
23
+ import { UNNECESSARY_ASSERTION, VALIDATE_EXTERNAL } from './types.js';
24
+ import { USE_EMPTY_METHOD } from './strings.js';
25
+ import {
26
+ AVOID_REASSIGNMENT,
27
+ COMPLEX_CONDITION,
28
+ LOOP_OUTER_CAPTURE,
29
+ } from './anti-patterns.js';
30
+ import {
31
+ SPACING_OPERATOR,
32
+ SPACING_BRACES,
33
+ SPACING_BRACKETS,
34
+ SPACING_CLOSURE,
35
+ INDENT_CONTINUATION,
36
+ IMPLICIT_DOLLAR_METHOD,
37
+ IMPLICIT_DOLLAR_FUNCTION,
38
+ IMPLICIT_DOLLAR_CLOSURE,
39
+ THROWAWAY_CAPTURE,
40
+ } from './formatting.js';
41
+
42
+ // ============================================================
43
+ // RE-EXPORT INDIVIDUAL RULES
44
+ // ============================================================
45
+
46
+ export { NAMING_SNAKE_CASE } from './naming.js';
47
+ export { CAPTURE_INLINE_CHAIN, CAPTURE_BEFORE_BRANCH } from './flow.js';
48
+ export {
49
+ BREAK_IN_PARALLEL,
50
+ PREFER_MAP,
51
+ FOLD_INTERMEDIATES,
52
+ FILTER_NEGATION,
53
+ METHOD_SHORTHAND,
54
+ } from './collections.js';
55
+ export { LOOP_ACCUMULATOR, PREFER_DO_WHILE, USE_EACH } from './loops.js';
56
+ export { USE_DEFAULT_OPERATOR, CONDITION_TYPE } from './conditionals.js';
57
+ export {
58
+ CLOSURE_BARE_DOLLAR,
59
+ CLOSURE_BRACES,
60
+ CLOSURE_LATE_BINDING,
61
+ } from './closures.js';
62
+ export { UNNECESSARY_ASSERTION, VALIDATE_EXTERNAL } from './types.js';
63
+ export { USE_EMPTY_METHOD } from './strings.js';
64
+ export {
65
+ AVOID_REASSIGNMENT,
66
+ COMPLEX_CONDITION,
67
+ LOOP_OUTER_CAPTURE,
68
+ } from './anti-patterns.js';
69
+ export {
70
+ SPACING_OPERATOR,
71
+ SPACING_BRACES,
72
+ SPACING_BRACKETS,
73
+ SPACING_CLOSURE,
74
+ INDENT_CONTINUATION,
75
+ IMPLICIT_DOLLAR_METHOD,
76
+ IMPLICIT_DOLLAR_FUNCTION,
77
+ IMPLICIT_DOLLAR_CLOSURE,
78
+ THROWAWAY_CAPTURE,
79
+ } from './formatting.js';
80
+
81
+ // ============================================================
82
+ // RULE REGISTRY
83
+ // ============================================================
84
+
85
+ /**
86
+ * All registered validation rules.
87
+ * Rules are applied during AST traversal via the validator.
88
+ */
89
+ export const VALIDATION_RULES: ValidationRule[] = [
90
+ // Naming conventions
91
+ NAMING_SNAKE_CASE,
92
+
93
+ // Flow and capture
94
+ CAPTURE_INLINE_CHAIN,
95
+ CAPTURE_BEFORE_BRANCH,
96
+
97
+ // Collection operators
98
+ BREAK_IN_PARALLEL,
99
+ PREFER_MAP,
100
+ FOLD_INTERMEDIATES,
101
+ FILTER_NEGATION,
102
+ METHOD_SHORTHAND,
103
+
104
+ // Loop conventions
105
+ LOOP_ACCUMULATOR,
106
+ PREFER_DO_WHILE,
107
+ USE_EACH,
108
+
109
+ // Conditional conventions
110
+ USE_DEFAULT_OPERATOR,
111
+ CONDITION_TYPE,
112
+
113
+ // Closure conventions
114
+ CLOSURE_BARE_DOLLAR,
115
+ CLOSURE_BRACES,
116
+ CLOSURE_LATE_BINDING,
117
+
118
+ // Type safety
119
+ UNNECESSARY_ASSERTION,
120
+ VALIDATE_EXTERNAL,
121
+
122
+ // String handling
123
+ USE_EMPTY_METHOD,
124
+
125
+ // Anti-patterns
126
+ AVOID_REASSIGNMENT,
127
+ COMPLEX_CONDITION,
128
+ LOOP_OUTER_CAPTURE,
129
+
130
+ // Formatting
131
+ SPACING_OPERATOR,
132
+ SPACING_BRACES,
133
+ SPACING_BRACKETS,
134
+ SPACING_CLOSURE,
135
+ INDENT_CONTINUATION,
136
+ IMPLICIT_DOLLAR_METHOD,
137
+ IMPLICIT_DOLLAR_FUNCTION,
138
+ IMPLICIT_DOLLAR_CLOSURE,
139
+ THROWAWAY_CAPTURE,
140
+ ];
@@ -0,0 +1,372 @@
1
+ /**
2
+ * Loop Convention Rules
3
+ * Enforces conventions for while, do-while, and loop control flow.
4
+ */
5
+
6
+ import type {
7
+ ValidationRule,
8
+ Diagnostic,
9
+ ValidationContext,
10
+ } from '../types.js';
11
+ import type {
12
+ ASTNode,
13
+ WhileLoopNode,
14
+ DoWhileLoopNode,
15
+ PipeChainNode,
16
+ } from '@rcrsr/rill';
17
+ import { extractContextLine } from './helpers.js';
18
+
19
+ // ============================================================
20
+ // HELPER FUNCTIONS
21
+ // ============================================================
22
+
23
+ /**
24
+ * Collect all variable captures (=> $name) in the given AST node.
25
+ */
26
+ function collectCaptures(node: ASTNode, names: string[]): void {
27
+ switch (node.type) {
28
+ case 'Capture':
29
+ names.push(`$${node.name}`);
30
+ return;
31
+
32
+ case 'Block':
33
+ node.statements.forEach((stmt) => collectCaptures(stmt, names));
34
+ return;
35
+
36
+ case 'Statement':
37
+ collectCaptures(node.expression, names);
38
+ return;
39
+
40
+ case 'AnnotatedStatement':
41
+ collectCaptures(node.statement, names);
42
+ return;
43
+
44
+ case 'PipeChain':
45
+ node.pipes.forEach((pipe) => collectCaptures(pipe as ASTNode, names));
46
+ if (node.terminator && node.terminator.type === 'Capture')
47
+ collectCaptures(node.terminator, names);
48
+ return;
49
+
50
+ case 'PostfixExpr':
51
+ collectCaptures(node.primary, names);
52
+ node.methods.forEach((method) => collectCaptures(method, names));
53
+ return;
54
+
55
+ case 'BinaryExpr':
56
+ collectCaptures(node.left, names);
57
+ collectCaptures(node.right, names);
58
+ return;
59
+
60
+ case 'UnaryExpr':
61
+ collectCaptures(node.operand, names);
62
+ return;
63
+
64
+ case 'GroupedExpr':
65
+ collectCaptures(node.expression, names);
66
+ return;
67
+
68
+ case 'Conditional':
69
+ if (node.input) collectCaptures(node.input, names);
70
+ if (node.condition) collectCaptures(node.condition, names);
71
+ collectCaptures(node.thenBranch, names);
72
+ if (node.elseBranch) collectCaptures(node.elseBranch, names);
73
+ return;
74
+
75
+ case 'WhileLoop':
76
+ case 'DoWhileLoop':
77
+ collectCaptures(node.body, names);
78
+ return;
79
+
80
+ default:
81
+ return;
82
+ }
83
+ }
84
+
85
+ /**
86
+ * Collect all variable references ($name) in the given AST node.
87
+ */
88
+ function collectVariableReferences(node: ASTNode, names: string[]): void {
89
+ switch (node.type) {
90
+ case 'Variable':
91
+ // Add the variable name if it's not the pipe variable ($)
92
+ if (!node.isPipeVar && node.name) {
93
+ names.push(`$${node.name}`);
94
+ }
95
+ return;
96
+
97
+ case 'Block':
98
+ node.statements.forEach((stmt) => collectVariableReferences(stmt, names));
99
+ return;
100
+
101
+ case 'Statement':
102
+ collectVariableReferences(node.expression, names);
103
+ return;
104
+
105
+ case 'AnnotatedStatement':
106
+ collectVariableReferences(node.statement, names);
107
+ return;
108
+
109
+ case 'PipeChain':
110
+ collectVariableReferences(node.head, names);
111
+ node.pipes.forEach((pipe) =>
112
+ collectVariableReferences(pipe as ASTNode, names)
113
+ );
114
+ if (node.terminator)
115
+ collectVariableReferences(node.terminator as ASTNode, names);
116
+ return;
117
+
118
+ case 'PostfixExpr':
119
+ collectVariableReferences(node.primary, names);
120
+ node.methods.forEach((method) =>
121
+ collectVariableReferences(method, names)
122
+ );
123
+ return;
124
+
125
+ case 'BinaryExpr':
126
+ collectVariableReferences(node.left, names);
127
+ collectVariableReferences(node.right, names);
128
+ return;
129
+
130
+ case 'UnaryExpr':
131
+ collectVariableReferences(node.operand, names);
132
+ return;
133
+
134
+ case 'GroupedExpr':
135
+ collectVariableReferences(node.expression, names);
136
+ return;
137
+
138
+ case 'Conditional':
139
+ if (node.input) collectVariableReferences(node.input, names);
140
+ if (node.condition) collectVariableReferences(node.condition, names);
141
+ collectVariableReferences(node.thenBranch, names);
142
+ if (node.elseBranch) collectVariableReferences(node.elseBranch, names);
143
+ return;
144
+
145
+ case 'WhileLoop':
146
+ case 'DoWhileLoop':
147
+ collectVariableReferences(node.condition, names);
148
+ collectVariableReferences(node.body, names);
149
+ return;
150
+
151
+ default:
152
+ return;
153
+ }
154
+ }
155
+
156
+ /**
157
+ * Check if a loop body appears to be calling a retry function.
158
+ * Simple heuristic: looks for function calls like attemptOperation() or retry().
159
+ */
160
+ function callsRetryFunction(node: ASTNode): boolean {
161
+ if (node.type === 'Block') {
162
+ return node.statements.some((stmt) => callsRetryFunction(stmt));
163
+ }
164
+
165
+ if (node.type === 'Statement') {
166
+ return callsRetryFunction(node.expression);
167
+ }
168
+
169
+ if (node.type === 'PipeChain') {
170
+ const chain = node as PipeChainNode;
171
+ const head = chain.head;
172
+
173
+ // Check if head is a function call
174
+ if (head.type === 'PostfixExpr') {
175
+ const primary = head.primary;
176
+ if (primary.type === 'HostCall' || primary.type === 'ClosureCall') {
177
+ return true;
178
+ }
179
+ }
180
+ }
181
+
182
+ return false;
183
+ }
184
+
185
+ // ============================================================
186
+ // LOOP_ACCUMULATOR RULE
187
+ // ============================================================
188
+
189
+ /**
190
+ * Validates that variables captured in loop bodies aren't referenced in conditions.
191
+ *
192
+ * In while and do-while loops, $ serves as the accumulator across iterations.
193
+ * Variables captured inside the loop body exist only within that iteration, so
194
+ * referencing them in the loop condition is a logic error - the condition will
195
+ * always see undefined (or the outer scope variable if one exists).
196
+ *
197
+ * Error pattern (captured variable in condition):
198
+ * 0 -> ($x < 5) @ { # $x is undefined in condition
199
+ * $ => $x
200
+ * $x + 1
201
+ * }
202
+ *
203
+ * Correct pattern ($ as accumulator):
204
+ * 0 -> ($ < 5) @ { $ + 1 }
205
+ *
206
+ * Also correct (capture only used within iteration):
207
+ * 0 -> ($ < 5) @ {
208
+ * $ => $x
209
+ * log($x) # $x only used in body, not condition
210
+ * $x + 1
211
+ * }
212
+ *
213
+ * References:
214
+ * - docs/guide-conventions.md:151-171
215
+ */
216
+ export const LOOP_ACCUMULATOR: ValidationRule = {
217
+ code: 'LOOP_ACCUMULATOR',
218
+ category: 'loops',
219
+ severity: 'info',
220
+ nodeTypes: ['WhileLoop', 'DoWhileLoop'],
221
+
222
+ validate(node: ASTNode, context: ValidationContext): Diagnostic[] {
223
+ const loop = node as WhileLoopNode | DoWhileLoopNode;
224
+
225
+ // Collect all variable captures in loop body
226
+ const capturedNames: string[] = [];
227
+ collectCaptures(loop.body, capturedNames);
228
+
229
+ if (capturedNames.length === 0) {
230
+ return []; // No captures, no problem
231
+ }
232
+
233
+ // Collect all variable references in loop condition
234
+ const conditionRefs: string[] = [];
235
+ collectVariableReferences(loop.condition, conditionRefs);
236
+
237
+ // Find captures that are referenced in the condition
238
+ const capturedSet = new Set(capturedNames);
239
+ const problematicVars = conditionRefs.filter((ref) => capturedSet.has(ref));
240
+
241
+ if (problematicVars.length > 0) {
242
+ const vars = [...new Set(problematicVars)].join(', ');
243
+ return [
244
+ {
245
+ location: node.span.start,
246
+ severity: 'info',
247
+ code: 'LOOP_ACCUMULATOR',
248
+ message: `${vars} captured in loop body but referenced in condition; loop body variables reset each iteration`,
249
+ context: extractContextLine(node.span.start.line, context.source),
250
+ fix: null, // Complex fix - requires refactoring loop body
251
+ },
252
+ ];
253
+ }
254
+
255
+ return [];
256
+ },
257
+ };
258
+
259
+ // ============================================================
260
+ // PREFER_DO_WHILE RULE
261
+ // ============================================================
262
+
263
+ /**
264
+ * Suggests using do-while for retry patterns.
265
+ *
266
+ * Do-while is clearer for retry patterns where the body must run at least once:
267
+ *
268
+ * Good (do-while for retry):
269
+ * @ {
270
+ * attemptOperation()
271
+ * } ? (.contains("RETRY"))
272
+ *
273
+ * Less clear (while with separate first attempt):
274
+ * attemptOperation() => $result
275
+ * $result -> .contains("RETRY") @ {
276
+ * attemptOperation()
277
+ * }
278
+ *
279
+ * This is informational - helps guide users to the clearer pattern.
280
+ *
281
+ * References:
282
+ * - docs/guide-conventions.md:173-186
283
+ */
284
+ export const PREFER_DO_WHILE: ValidationRule = {
285
+ code: 'PREFER_DO_WHILE',
286
+ category: 'loops',
287
+ severity: 'info',
288
+ nodeTypes: ['WhileLoop'],
289
+
290
+ validate(node: ASTNode, context: ValidationContext): Diagnostic[] {
291
+ const loop = node as WhileLoopNode;
292
+
293
+ // Heuristic: if loop body appears to be calling a retry/attempt function,
294
+ // suggest do-while
295
+ if (callsRetryFunction(loop.body)) {
296
+ return [
297
+ {
298
+ location: node.span.start,
299
+ severity: 'info',
300
+ code: 'PREFER_DO_WHILE',
301
+ message:
302
+ 'Consider do-while for retry patterns where body runs at least once: @ { body } ? (condition)',
303
+ context: extractContextLine(node.span.start.line, context.source),
304
+ fix: null, // Complex fix - requires restructuring to do-while
305
+ },
306
+ ];
307
+ }
308
+
309
+ return [];
310
+ },
311
+ };
312
+
313
+ // ============================================================
314
+ // USE_EACH RULE
315
+ // ============================================================
316
+
317
+ /**
318
+ * Suggests using each for collection iteration instead of while loops.
319
+ *
320
+ * When iterating over a collection, each is clearer and more idiomatic:
321
+ *
322
+ * Good (each for collection):
323
+ * $items -> each { process($) }
324
+ *
325
+ * Less clear (while loop):
326
+ * 0 => $i
327
+ * ($i < $items.len) @ {
328
+ * $items[$i] -> process()
329
+ * $i + 1
330
+ * }
331
+ *
332
+ * This is informational - while loops work, but each is clearer for collections.
333
+ *
334
+ * References:
335
+ * - docs/guide-conventions.md:188-196
336
+ */
337
+ export const USE_EACH: ValidationRule = {
338
+ code: 'USE_EACH',
339
+ category: 'loops',
340
+ severity: 'info',
341
+ nodeTypes: ['WhileLoop'],
342
+
343
+ validate(node: ASTNode, context: ValidationContext): Diagnostic[] {
344
+ const loop = node as WhileLoopNode;
345
+
346
+ // Simple heuristic: if the condition or body appears to be doing array iteration
347
+ const conditionStr = JSON.stringify(loop.condition);
348
+ const bodyStr = JSON.stringify(loop.body);
349
+
350
+ // Look for patterns like:
351
+ // - field access to 'len' (array length checks)
352
+ // - bracket access patterns with BracketAccess nodes in body
353
+ const hasLenCheck = conditionStr.includes('"field":"len"');
354
+ const hasBracketAccess = bodyStr.includes('"accessKind":"bracket"');
355
+
356
+ if (hasLenCheck || hasBracketAccess) {
357
+ return [
358
+ {
359
+ location: node.span.start,
360
+ severity: 'info',
361
+ code: 'USE_EACH',
362
+ message:
363
+ "Use 'each' for collection iteration instead of while loops: collection -> each { body }",
364
+ context: extractContextLine(node.span.start.line, context.source),
365
+ fix: null, // Complex fix - requires restructuring to each
366
+ },
367
+ ];
368
+ }
369
+
370
+ return [];
371
+ },
372
+ };