@vibe-lang/runtime 0.2.5

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 (250) hide show
  1. package/package.json +46 -0
  2. package/src/ast/index.ts +375 -0
  3. package/src/ast.ts +2 -0
  4. package/src/debug/advanced-features.ts +482 -0
  5. package/src/debug/bun-inspector.ts +424 -0
  6. package/src/debug/handoff-manager.ts +283 -0
  7. package/src/debug/index.ts +150 -0
  8. package/src/debug/runner.ts +365 -0
  9. package/src/debug/server.ts +565 -0
  10. package/src/debug/stack-merger.ts +267 -0
  11. package/src/debug/state.ts +581 -0
  12. package/src/debug/test/advanced-features.test.ts +300 -0
  13. package/src/debug/test/e2e.test.ts +218 -0
  14. package/src/debug/test/handoff-manager.test.ts +256 -0
  15. package/src/debug/test/runner.test.ts +256 -0
  16. package/src/debug/test/stack-merger.test.ts +163 -0
  17. package/src/debug/test/state.test.ts +400 -0
  18. package/src/debug/test/ts-debug-integration.test.ts +374 -0
  19. package/src/debug/test/ts-import-tracker.test.ts +125 -0
  20. package/src/debug/test/ts-source-map.test.ts +169 -0
  21. package/src/debug/ts-import-tracker.ts +151 -0
  22. package/src/debug/ts-source-map.ts +171 -0
  23. package/src/errors/index.ts +124 -0
  24. package/src/index.ts +358 -0
  25. package/src/lexer/index.ts +348 -0
  26. package/src/lexer.ts +2 -0
  27. package/src/parser/index.ts +792 -0
  28. package/src/parser/parse.ts +45 -0
  29. package/src/parser/test/async.test.ts +248 -0
  30. package/src/parser/test/destructuring.test.ts +167 -0
  31. package/src/parser/test/do-expression.test.ts +486 -0
  32. package/src/parser/test/errors/do-expression.test.ts +95 -0
  33. package/src/parser/test/errors/error-locations.test.ts +230 -0
  34. package/src/parser/test/errors/invalid-expressions.test.ts +144 -0
  35. package/src/parser/test/errors/missing-tokens.test.ts +126 -0
  36. package/src/parser/test/errors/model-declaration.test.ts +185 -0
  37. package/src/parser/test/errors/nested-blocks.test.ts +226 -0
  38. package/src/parser/test/errors/unclosed-delimiters.test.ts +122 -0
  39. package/src/parser/test/errors/unexpected-tokens.test.ts +120 -0
  40. package/src/parser/test/import-export.test.ts +143 -0
  41. package/src/parser/test/literals.test.ts +404 -0
  42. package/src/parser/test/model-declaration.test.ts +161 -0
  43. package/src/parser/test/nested-blocks.test.ts +402 -0
  44. package/src/parser/test/parser.test.ts +743 -0
  45. package/src/parser/test/private.test.ts +136 -0
  46. package/src/parser/test/template-literal.test.ts +127 -0
  47. package/src/parser/test/tool-declaration.test.ts +302 -0
  48. package/src/parser/test/ts-block.test.ts +252 -0
  49. package/src/parser/test/type-annotations.test.ts +254 -0
  50. package/src/parser/visitor/helpers.ts +330 -0
  51. package/src/parser/visitor.ts +794 -0
  52. package/src/parser.ts +2 -0
  53. package/src/runtime/ai/cache-chunking.test.ts +69 -0
  54. package/src/runtime/ai/cache-chunking.ts +73 -0
  55. package/src/runtime/ai/client.ts +109 -0
  56. package/src/runtime/ai/context.ts +168 -0
  57. package/src/runtime/ai/formatters.ts +316 -0
  58. package/src/runtime/ai/index.ts +38 -0
  59. package/src/runtime/ai/language-ref.ts +38 -0
  60. package/src/runtime/ai/providers/anthropic.ts +253 -0
  61. package/src/runtime/ai/providers/google.ts +201 -0
  62. package/src/runtime/ai/providers/openai.ts +156 -0
  63. package/src/runtime/ai/retry.ts +100 -0
  64. package/src/runtime/ai/return-tools.ts +301 -0
  65. package/src/runtime/ai/test/client.test.ts +83 -0
  66. package/src/runtime/ai/test/formatters.test.ts +485 -0
  67. package/src/runtime/ai/test/retry.test.ts +137 -0
  68. package/src/runtime/ai/test/return-tools.test.ts +450 -0
  69. package/src/runtime/ai/test/tool-loop.test.ts +319 -0
  70. package/src/runtime/ai/test/tool-schema.test.ts +241 -0
  71. package/src/runtime/ai/tool-loop.ts +203 -0
  72. package/src/runtime/ai/tool-schema.ts +151 -0
  73. package/src/runtime/ai/types.ts +113 -0
  74. package/src/runtime/ai-logger.ts +255 -0
  75. package/src/runtime/ai-provider.ts +347 -0
  76. package/src/runtime/async/dependencies.ts +276 -0
  77. package/src/runtime/async/executor.ts +293 -0
  78. package/src/runtime/async/index.ts +43 -0
  79. package/src/runtime/async/scheduling.ts +163 -0
  80. package/src/runtime/async/test/dependencies.test.ts +284 -0
  81. package/src/runtime/async/test/executor.test.ts +388 -0
  82. package/src/runtime/context.ts +357 -0
  83. package/src/runtime/exec/ai.ts +139 -0
  84. package/src/runtime/exec/expressions.ts +475 -0
  85. package/src/runtime/exec/frames.ts +26 -0
  86. package/src/runtime/exec/functions.ts +305 -0
  87. package/src/runtime/exec/interpolation.ts +312 -0
  88. package/src/runtime/exec/statements.ts +604 -0
  89. package/src/runtime/exec/tools.ts +129 -0
  90. package/src/runtime/exec/typescript.ts +215 -0
  91. package/src/runtime/exec/variables.ts +279 -0
  92. package/src/runtime/index.ts +975 -0
  93. package/src/runtime/modules.ts +452 -0
  94. package/src/runtime/serialize.ts +103 -0
  95. package/src/runtime/state.ts +489 -0
  96. package/src/runtime/stdlib/core.ts +45 -0
  97. package/src/runtime/stdlib/directory.test.ts +156 -0
  98. package/src/runtime/stdlib/edit.test.ts +154 -0
  99. package/src/runtime/stdlib/fastEdit.test.ts +201 -0
  100. package/src/runtime/stdlib/glob.test.ts +106 -0
  101. package/src/runtime/stdlib/grep.test.ts +144 -0
  102. package/src/runtime/stdlib/index.ts +16 -0
  103. package/src/runtime/stdlib/readFile.test.ts +123 -0
  104. package/src/runtime/stdlib/tools/index.ts +707 -0
  105. package/src/runtime/stdlib/writeFile.test.ts +157 -0
  106. package/src/runtime/step.ts +969 -0
  107. package/src/runtime/test/ai-context.test.ts +1086 -0
  108. package/src/runtime/test/ai-result-object.test.ts +419 -0
  109. package/src/runtime/test/ai-tool-flow.test.ts +859 -0
  110. package/src/runtime/test/async-execution-order.test.ts +618 -0
  111. package/src/runtime/test/async-execution.test.ts +344 -0
  112. package/src/runtime/test/async-nested.test.ts +660 -0
  113. package/src/runtime/test/async-parallel-timing.test.ts +546 -0
  114. package/src/runtime/test/basic1.test.ts +154 -0
  115. package/src/runtime/test/binary-operators.test.ts +431 -0
  116. package/src/runtime/test/break-statement.test.ts +257 -0
  117. package/src/runtime/test/context-modes.test.ts +650 -0
  118. package/src/runtime/test/context.test.ts +466 -0
  119. package/src/runtime/test/core-functions.test.ts +228 -0
  120. package/src/runtime/test/e2e.test.ts +88 -0
  121. package/src/runtime/test/error-locations/error-locations.test.ts +80 -0
  122. package/src/runtime/test/error-locations/main-error.vibe +4 -0
  123. package/src/runtime/test/error-locations/main-import-error.vibe +3 -0
  124. package/src/runtime/test/error-locations/utils/helper.vibe +5 -0
  125. package/src/runtime/test/for-in.test.ts +312 -0
  126. package/src/runtime/test/helpers.ts +69 -0
  127. package/src/runtime/test/imports.test.ts +334 -0
  128. package/src/runtime/test/json-expressions.test.ts +232 -0
  129. package/src/runtime/test/literals.test.ts +372 -0
  130. package/src/runtime/test/logical-indexing.test.ts +478 -0
  131. package/src/runtime/test/member-methods.test.ts +324 -0
  132. package/src/runtime/test/model-config.test.ts +338 -0
  133. package/src/runtime/test/null-handling.test.ts +342 -0
  134. package/src/runtime/test/private-visibility.test.ts +332 -0
  135. package/src/runtime/test/runtime-state.test.ts +514 -0
  136. package/src/runtime/test/scoping.test.ts +370 -0
  137. package/src/runtime/test/string-interpolation.test.ts +354 -0
  138. package/src/runtime/test/template-literal.test.ts +181 -0
  139. package/src/runtime/test/tool-execution.test.ts +467 -0
  140. package/src/runtime/test/tool-schema-generation.test.ts +477 -0
  141. package/src/runtime/test/tostring.test.ts +210 -0
  142. package/src/runtime/test/ts-block.test.ts +594 -0
  143. package/src/runtime/test/ts-error-location.test.ts +231 -0
  144. package/src/runtime/test/types.test.ts +732 -0
  145. package/src/runtime/test/verbose-logger.test.ts +710 -0
  146. package/src/runtime/test/vibe-expression.test.ts +54 -0
  147. package/src/runtime/test/vibe-value-errors.test.ts +541 -0
  148. package/src/runtime/test/while.test.ts +232 -0
  149. package/src/runtime/tools/builtin.ts +30 -0
  150. package/src/runtime/tools/directory-tools.ts +70 -0
  151. package/src/runtime/tools/file-tools.ts +228 -0
  152. package/src/runtime/tools/index.ts +5 -0
  153. package/src/runtime/tools/registry.ts +48 -0
  154. package/src/runtime/tools/search-tools.ts +134 -0
  155. package/src/runtime/tools/security.ts +36 -0
  156. package/src/runtime/tools/system-tools.ts +312 -0
  157. package/src/runtime/tools/test/fixtures/base-types.ts +40 -0
  158. package/src/runtime/tools/test/fixtures/test-types.ts +132 -0
  159. package/src/runtime/tools/test/registry.test.ts +713 -0
  160. package/src/runtime/tools/test/security.test.ts +86 -0
  161. package/src/runtime/tools/test/system-tools.test.ts +679 -0
  162. package/src/runtime/tools/test/ts-schema.test.ts +357 -0
  163. package/src/runtime/tools/ts-schema.ts +341 -0
  164. package/src/runtime/tools/types.ts +89 -0
  165. package/src/runtime/tools/utility-tools.ts +198 -0
  166. package/src/runtime/ts-eval.ts +126 -0
  167. package/src/runtime/types.ts +797 -0
  168. package/src/runtime/validation.ts +160 -0
  169. package/src/runtime/verbose-logger.ts +459 -0
  170. package/src/runtime.ts +2 -0
  171. package/src/semantic/analyzer-context.ts +62 -0
  172. package/src/semantic/analyzer-validators.ts +575 -0
  173. package/src/semantic/analyzer-visitors.ts +534 -0
  174. package/src/semantic/analyzer.ts +83 -0
  175. package/src/semantic/index.ts +11 -0
  176. package/src/semantic/symbol-table.ts +58 -0
  177. package/src/semantic/test/async-validation.test.ts +301 -0
  178. package/src/semantic/test/compress-validation.test.ts +179 -0
  179. package/src/semantic/test/const-reassignment.test.ts +111 -0
  180. package/src/semantic/test/control-flow.test.ts +346 -0
  181. package/src/semantic/test/destructuring.test.ts +185 -0
  182. package/src/semantic/test/duplicate-declarations.test.ts +168 -0
  183. package/src/semantic/test/export-validation.test.ts +111 -0
  184. package/src/semantic/test/fixtures/math.ts +31 -0
  185. package/src/semantic/test/imports.test.ts +148 -0
  186. package/src/semantic/test/json-type.test.ts +68 -0
  187. package/src/semantic/test/literals.test.ts +127 -0
  188. package/src/semantic/test/model-validation.test.ts +179 -0
  189. package/src/semantic/test/prompt-validation.test.ts +343 -0
  190. package/src/semantic/test/scoping.test.ts +312 -0
  191. package/src/semantic/test/tool-validation.test.ts +306 -0
  192. package/src/semantic/test/ts-type-checking.test.ts +563 -0
  193. package/src/semantic/test/type-constraints.test.ts +111 -0
  194. package/src/semantic/test/type-inference.test.ts +87 -0
  195. package/src/semantic/test/type-validation.test.ts +552 -0
  196. package/src/semantic/test/undefined-variables.test.ts +163 -0
  197. package/src/semantic/ts-block-checker.ts +204 -0
  198. package/src/semantic/ts-signatures.ts +194 -0
  199. package/src/semantic/ts-types.ts +170 -0
  200. package/src/semantic/types.ts +58 -0
  201. package/tests/fixtures/conditional-logic.vibe +14 -0
  202. package/tests/fixtures/function-call.vibe +16 -0
  203. package/tests/fixtures/imports/cycle-detection/a.vibe +6 -0
  204. package/tests/fixtures/imports/cycle-detection/b.vibe +5 -0
  205. package/tests/fixtures/imports/cycle-detection/main.vibe +3 -0
  206. package/tests/fixtures/imports/module-isolation/main-b.vibe +8 -0
  207. package/tests/fixtures/imports/module-isolation/main.vibe +9 -0
  208. package/tests/fixtures/imports/module-isolation/moduleA.vibe +6 -0
  209. package/tests/fixtures/imports/module-isolation/moduleB.vibe +6 -0
  210. package/tests/fixtures/imports/nested-import/helper.vibe +6 -0
  211. package/tests/fixtures/imports/nested-import/main.vibe +3 -0
  212. package/tests/fixtures/imports/nested-import/utils.ts +3 -0
  213. package/tests/fixtures/imports/nested-isolation/file2.vibe +15 -0
  214. package/tests/fixtures/imports/nested-isolation/file3.vibe +10 -0
  215. package/tests/fixtures/imports/nested-isolation/main.vibe +21 -0
  216. package/tests/fixtures/imports/pure-cycle/a.vibe +5 -0
  217. package/tests/fixtures/imports/pure-cycle/b.vibe +5 -0
  218. package/tests/fixtures/imports/pure-cycle/main.vibe +3 -0
  219. package/tests/fixtures/imports/ts-boolean/checks.ts +14 -0
  220. package/tests/fixtures/imports/ts-boolean/main.vibe +10 -0
  221. package/tests/fixtures/imports/ts-boolean/type-mismatch.vibe +5 -0
  222. package/tests/fixtures/imports/ts-boolean/use-constant.vibe +18 -0
  223. package/tests/fixtures/imports/ts-error-handling/helpers.ts +42 -0
  224. package/tests/fixtures/imports/ts-error-handling/main.vibe +5 -0
  225. package/tests/fixtures/imports/ts-import/main.vibe +4 -0
  226. package/tests/fixtures/imports/ts-import/math.ts +9 -0
  227. package/tests/fixtures/imports/ts-variables/call-non-function.vibe +5 -0
  228. package/tests/fixtures/imports/ts-variables/data.ts +10 -0
  229. package/tests/fixtures/imports/ts-variables/import-json.vibe +5 -0
  230. package/tests/fixtures/imports/ts-variables/import-type-mismatch.vibe +5 -0
  231. package/tests/fixtures/imports/ts-variables/import-variable.vibe +5 -0
  232. package/tests/fixtures/imports/vibe-import/greet.vibe +5 -0
  233. package/tests/fixtures/imports/vibe-import/main.vibe +3 -0
  234. package/tests/fixtures/multiple-ai-calls.vibe +10 -0
  235. package/tests/fixtures/simple-greeting.vibe +6 -0
  236. package/tests/fixtures/template-literals.vibe +11 -0
  237. package/tests/integration/basic-ai/basic-ai.integration.test.ts +166 -0
  238. package/tests/integration/basic-ai/basic-ai.vibe +12 -0
  239. package/tests/integration/bug-fix/bug-fix.integration.test.ts +201 -0
  240. package/tests/integration/bug-fix/buggy-code.ts +22 -0
  241. package/tests/integration/bug-fix/fix-bug.vibe +21 -0
  242. package/tests/integration/compress/compress.integration.test.ts +206 -0
  243. package/tests/integration/destructuring/destructuring.integration.test.ts +92 -0
  244. package/tests/integration/hello-world-translator/hello-world-translator.integration.test.ts +61 -0
  245. package/tests/integration/line-annotator/context-modes.integration.test.ts +261 -0
  246. package/tests/integration/line-annotator/line-annotator.integration.test.ts +148 -0
  247. package/tests/integration/multi-feature/cumulative-sum.integration.test.ts +75 -0
  248. package/tests/integration/multi-feature/number-analyzer.integration.test.ts +191 -0
  249. package/tests/integration/multi-feature/number-analyzer.vibe +59 -0
  250. package/tests/integration/tool-calls/tool-calls.integration.test.ts +93 -0
@@ -0,0 +1,743 @@
1
+ import { describe, expect, test } from 'bun:test';
2
+ import { parse } from '../parse';
3
+
4
+ describe('Parser - Single Statements', () => {
5
+ // ============================================================================
6
+ // Let Declaration
7
+ // ============================================================================
8
+
9
+ test('let without initializer', () => {
10
+ const ast = parse('let x');
11
+ expect(ast.body).toHaveLength(1);
12
+ expect(ast.body[0]).toMatchObject({
13
+ type: 'LetDeclaration',
14
+ name: 'x',
15
+ initializer: null,
16
+ });
17
+ });
18
+
19
+ test('let with string initializer', () => {
20
+ const ast = parse('let x = "hello"');
21
+ expect(ast.body).toHaveLength(1);
22
+ expect(ast.body[0]).toMatchObject({
23
+ type: 'LetDeclaration',
24
+ name: 'x',
25
+ initializer: {
26
+ type: 'StringLiteral',
27
+ value: 'hello',
28
+ },
29
+ });
30
+ });
31
+
32
+ test('let with identifier initializer', () => {
33
+ const ast = parse('let x = y');
34
+ expect(ast.body).toHaveLength(1);
35
+ expect(ast.body[0]).toMatchObject({
36
+ type: 'LetDeclaration',
37
+ name: 'x',
38
+ initializer: {
39
+ type: 'Identifier',
40
+ name: 'y',
41
+ },
42
+ });
43
+ });
44
+
45
+ // ============================================================================
46
+ // Const Declaration
47
+ // ============================================================================
48
+
49
+ test('const with string initializer', () => {
50
+ const ast = parse('const x = "hello"');
51
+ expect(ast.body).toHaveLength(1);
52
+ expect(ast.body[0]).toMatchObject({
53
+ type: 'ConstDeclaration',
54
+ name: 'x',
55
+ initializer: {
56
+ type: 'StringLiteral',
57
+ value: 'hello',
58
+ },
59
+ });
60
+ });
61
+
62
+ // ============================================================================
63
+ // Function Declaration
64
+ // ============================================================================
65
+
66
+ test('function with no params', () => {
67
+ const ast = parse(`
68
+ function myFunc() {
69
+ return "hello"
70
+ }
71
+ `);
72
+ expect(ast.body).toHaveLength(1);
73
+ expect(ast.body[0]).toMatchObject({
74
+ type: 'FunctionDeclaration',
75
+ name: 'myFunc',
76
+ params: [],
77
+ body: {
78
+ type: 'BlockStatement',
79
+ body: [
80
+ {
81
+ type: 'ReturnStatement',
82
+ value: {
83
+ type: 'StringLiteral',
84
+ value: 'hello',
85
+ },
86
+ },
87
+ ],
88
+ },
89
+ });
90
+ });
91
+
92
+ test('function with one param', () => {
93
+ const ast = parse(`
94
+ function greet(name: text): text {
95
+ return name
96
+ }
97
+ `);
98
+ expect(ast.body).toHaveLength(1);
99
+ expect(ast.body[0]).toMatchObject({
100
+ type: 'FunctionDeclaration',
101
+ name: 'greet',
102
+ params: [{ name: 'name', typeAnnotation: 'text' }],
103
+ returnType: 'text',
104
+ body: {
105
+ type: 'BlockStatement',
106
+ body: [
107
+ {
108
+ type: 'ReturnStatement',
109
+ value: {
110
+ type: 'Identifier',
111
+ name: 'name',
112
+ },
113
+ },
114
+ ],
115
+ },
116
+ });
117
+ });
118
+
119
+ test('function with multiple params', () => {
120
+ const ast = parse(`
121
+ function add(a: text, b: text, c: text): text {
122
+ return a
123
+ }
124
+ `);
125
+ expect(ast.body).toHaveLength(1);
126
+ expect(ast.body[0]).toMatchObject({
127
+ type: 'FunctionDeclaration',
128
+ name: 'add',
129
+ params: [
130
+ { name: 'a', typeAnnotation: 'text' },
131
+ { name: 'b', typeAnnotation: 'text' },
132
+ { name: 'c', typeAnnotation: 'text' },
133
+ ],
134
+ returnType: 'text',
135
+ body: {
136
+ type: 'BlockStatement',
137
+ body: [
138
+ {
139
+ type: 'ReturnStatement',
140
+ value: {
141
+ type: 'Identifier',
142
+ name: 'a',
143
+ },
144
+ },
145
+ ],
146
+ },
147
+ });
148
+ });
149
+
150
+ // ============================================================================
151
+ // Return Statement
152
+ // ============================================================================
153
+
154
+ test('return without value', () => {
155
+ const ast = parse(`
156
+ function test() {
157
+ return
158
+ }
159
+ `);
160
+ expect(ast.body[0]).toMatchObject({
161
+ type: 'FunctionDeclaration',
162
+ name: 'test',
163
+ body: {
164
+ type: 'BlockStatement',
165
+ body: [
166
+ {
167
+ type: 'ReturnStatement',
168
+ value: null,
169
+ },
170
+ ],
171
+ },
172
+ });
173
+ });
174
+
175
+ test('return with value', () => {
176
+ const ast = parse(`
177
+ function test() {
178
+ return "hello"
179
+ }
180
+ `);
181
+ expect(ast.body[0]).toMatchObject({
182
+ type: 'FunctionDeclaration',
183
+ name: 'test',
184
+ body: {
185
+ type: 'BlockStatement',
186
+ body: [
187
+ {
188
+ type: 'ReturnStatement',
189
+ value: {
190
+ type: 'StringLiteral',
191
+ value: 'hello',
192
+ },
193
+ },
194
+ ],
195
+ },
196
+ });
197
+ });
198
+
199
+ // ============================================================================
200
+ // If Statement
201
+ // ============================================================================
202
+
203
+ test('if without else', () => {
204
+ const ast = parse(`
205
+ if x {
206
+ let y = "yes"
207
+ }
208
+ `);
209
+ expect(ast.body).toHaveLength(1);
210
+ expect(ast.body[0]).toMatchObject({
211
+ type: 'IfStatement',
212
+ condition: {
213
+ type: 'Identifier',
214
+ name: 'x',
215
+ },
216
+ consequent: {
217
+ type: 'BlockStatement',
218
+ body: [
219
+ {
220
+ type: 'LetDeclaration',
221
+ name: 'y',
222
+ initializer: {
223
+ type: 'StringLiteral',
224
+ value: 'yes',
225
+ },
226
+ },
227
+ ],
228
+ },
229
+ alternate: null,
230
+ });
231
+ });
232
+
233
+ test('if with else', () => {
234
+ const ast = parse(`
235
+ if x {
236
+ let y = "yes"
237
+ } else {
238
+ let n = "no"
239
+ }
240
+ `);
241
+ expect(ast.body).toHaveLength(1);
242
+ expect(ast.body[0]).toMatchObject({
243
+ type: 'IfStatement',
244
+ condition: {
245
+ type: 'Identifier',
246
+ name: 'x',
247
+ },
248
+ consequent: {
249
+ type: 'BlockStatement',
250
+ body: [
251
+ {
252
+ type: 'LetDeclaration',
253
+ name: 'y',
254
+ initializer: {
255
+ type: 'StringLiteral',
256
+ value: 'yes',
257
+ },
258
+ },
259
+ ],
260
+ },
261
+ alternate: {
262
+ type: 'BlockStatement',
263
+ body: [
264
+ {
265
+ type: 'LetDeclaration',
266
+ name: 'n',
267
+ initializer: {
268
+ type: 'StringLiteral',
269
+ value: 'no',
270
+ },
271
+ },
272
+ ],
273
+ },
274
+ });
275
+ });
276
+
277
+ test('if with else if', () => {
278
+ const ast = parse(`
279
+ if x {
280
+ let a = "a"
281
+ } else if y {
282
+ let b = "b"
283
+ }
284
+ `);
285
+ expect(ast.body).toHaveLength(1);
286
+ expect(ast.body[0]).toMatchObject({
287
+ type: 'IfStatement',
288
+ condition: {
289
+ type: 'Identifier',
290
+ name: 'x',
291
+ },
292
+ consequent: {
293
+ type: 'BlockStatement',
294
+ body: [
295
+ {
296
+ type: 'LetDeclaration',
297
+ name: 'a',
298
+ initializer: {
299
+ type: 'StringLiteral',
300
+ value: 'a',
301
+ },
302
+ },
303
+ ],
304
+ },
305
+ alternate: {
306
+ type: 'IfStatement',
307
+ condition: {
308
+ type: 'Identifier',
309
+ name: 'y',
310
+ },
311
+ consequent: {
312
+ type: 'BlockStatement',
313
+ body: [
314
+ {
315
+ type: 'LetDeclaration',
316
+ name: 'b',
317
+ initializer: {
318
+ type: 'StringLiteral',
319
+ value: 'b',
320
+ },
321
+ },
322
+ ],
323
+ },
324
+ alternate: null,
325
+ },
326
+ });
327
+ });
328
+
329
+ // ============================================================================
330
+ // While Loop
331
+ // ============================================================================
332
+
333
+ test('while statement', () => {
334
+ const ast = parse(`
335
+ while keepGoing {
336
+ let x = 1
337
+ }
338
+ `);
339
+ expect(ast.body[0]).toMatchObject({
340
+ type: 'WhileStatement',
341
+ condition: {
342
+ type: 'Identifier',
343
+ name: 'keepGoing',
344
+ },
345
+ body: {
346
+ type: 'BlockStatement',
347
+ body: [
348
+ {
349
+ type: 'LetDeclaration',
350
+ name: 'x',
351
+ },
352
+ ],
353
+ },
354
+ });
355
+ });
356
+
357
+ // ============================================================================
358
+ // Block Statement
359
+ // ============================================================================
360
+
361
+ test('empty block', () => {
362
+ const ast = parse('{ }');
363
+ expect(ast.body).toHaveLength(1);
364
+ expect(ast.body[0]).toMatchObject({
365
+ type: 'BlockStatement',
366
+ body: [],
367
+ });
368
+ });
369
+
370
+ test('block with statements', () => {
371
+ const ast = parse(`
372
+ {
373
+ let x = "a"
374
+ let y = "b"
375
+ }
376
+ `);
377
+ expect(ast.body).toHaveLength(1);
378
+ expect(ast.body[0]).toMatchObject({
379
+ type: 'BlockStatement',
380
+ body: [
381
+ {
382
+ type: 'LetDeclaration',
383
+ name: 'x',
384
+ initializer: {
385
+ type: 'StringLiteral',
386
+ value: 'a',
387
+ },
388
+ },
389
+ {
390
+ type: 'LetDeclaration',
391
+ name: 'y',
392
+ initializer: {
393
+ type: 'StringLiteral',
394
+ value: 'b',
395
+ },
396
+ },
397
+ ],
398
+ });
399
+ });
400
+
401
+ // ============================================================================
402
+ // Vibe Expression
403
+ // ============================================================================
404
+
405
+ test('vibe with string', () => {
406
+ const ast = parse('let x = vibe "what is 2+2?" myModel default');
407
+ expect(ast.body).toHaveLength(1);
408
+ expect(ast.body[0]).toMatchObject({
409
+ type: 'LetDeclaration',
410
+ name: 'x',
411
+ initializer: {
412
+ type: 'VibeExpression',
413
+ prompt: {
414
+ type: 'StringLiteral',
415
+ value: 'what is 2+2?',
416
+ },
417
+ model: {
418
+ type: 'Identifier',
419
+ name: 'myModel',
420
+ },
421
+ context: {
422
+ type: 'ContextSpecifier',
423
+ kind: 'default',
424
+ },
425
+ },
426
+ });
427
+ });
428
+
429
+ test('vibe with identifier', () => {
430
+ const ast = parse('let x = vibe message myModel local');
431
+ expect(ast.body).toHaveLength(1);
432
+ expect(ast.body[0]).toMatchObject({
433
+ type: 'LetDeclaration',
434
+ name: 'x',
435
+ initializer: {
436
+ type: 'VibeExpression',
437
+ prompt: {
438
+ type: 'Identifier',
439
+ name: 'message',
440
+ },
441
+ model: {
442
+ type: 'Identifier',
443
+ name: 'myModel',
444
+ },
445
+ context: {
446
+ type: 'ContextSpecifier',
447
+ kind: 'local',
448
+ },
449
+ },
450
+ });
451
+ });
452
+
453
+ // ============================================================================
454
+ // Vibe Expression (synonym for do)
455
+ // ============================================================================
456
+
457
+ test('vibe with string, model, and default context', () => {
458
+ const ast = parse('let x = vibe "generate a hello function" myModel default');
459
+ expect(ast.body).toHaveLength(1);
460
+ expect(ast.body[0]).toMatchObject({
461
+ type: 'LetDeclaration',
462
+ name: 'x',
463
+ initializer: {
464
+ type: 'VibeExpression',
465
+ prompt: {
466
+ type: 'StringLiteral',
467
+ value: 'generate a hello function',
468
+ },
469
+ model: {
470
+ type: 'Identifier',
471
+ name: 'myModel',
472
+ },
473
+ context: {
474
+ type: 'ContextSpecifier',
475
+ kind: 'default',
476
+ },
477
+ },
478
+ });
479
+ });
480
+
481
+ test('vibe with local context', () => {
482
+ const ast = parse('let x = vibe "generate code" coder local');
483
+ expect(ast.body).toHaveLength(1);
484
+ expect(ast.body[0]).toMatchObject({
485
+ type: 'LetDeclaration',
486
+ name: 'x',
487
+ initializer: {
488
+ type: 'VibeExpression',
489
+ prompt: {
490
+ type: 'StringLiteral',
491
+ value: 'generate code',
492
+ },
493
+ model: {
494
+ type: 'Identifier',
495
+ name: 'coder',
496
+ },
497
+ context: {
498
+ type: 'ContextSpecifier',
499
+ kind: 'local',
500
+ },
501
+ },
502
+ });
503
+ });
504
+
505
+ // ============================================================================
506
+ // Call Expression
507
+ // ============================================================================
508
+
509
+ test('call with no args', () => {
510
+ const ast = parse('myFunc()');
511
+ expect(ast.body).toHaveLength(1);
512
+ expect(ast.body[0]).toMatchObject({
513
+ type: 'ExpressionStatement',
514
+ expression: {
515
+ type: 'CallExpression',
516
+ callee: {
517
+ type: 'Identifier',
518
+ name: 'myFunc',
519
+ },
520
+ arguments: [],
521
+ },
522
+ });
523
+ });
524
+
525
+ test('call with one arg', () => {
526
+ const ast = parse('greet("world")');
527
+ expect(ast.body).toHaveLength(1);
528
+ expect(ast.body[0]).toMatchObject({
529
+ type: 'ExpressionStatement',
530
+ expression: {
531
+ type: 'CallExpression',
532
+ callee: {
533
+ type: 'Identifier',
534
+ name: 'greet',
535
+ },
536
+ arguments: [
537
+ {
538
+ type: 'StringLiteral',
539
+ value: 'world',
540
+ },
541
+ ],
542
+ },
543
+ });
544
+ });
545
+
546
+ test('call with multiple args', () => {
547
+ const ast = parse('combine("a", "b", "c")');
548
+ expect(ast.body).toHaveLength(1);
549
+ expect(ast.body[0]).toMatchObject({
550
+ type: 'ExpressionStatement',
551
+ expression: {
552
+ type: 'CallExpression',
553
+ callee: {
554
+ type: 'Identifier',
555
+ name: 'combine',
556
+ },
557
+ arguments: [
558
+ {
559
+ type: 'StringLiteral',
560
+ value: 'a',
561
+ },
562
+ {
563
+ type: 'StringLiteral',
564
+ value: 'b',
565
+ },
566
+ {
567
+ type: 'StringLiteral',
568
+ value: 'c',
569
+ },
570
+ ],
571
+ },
572
+ });
573
+ });
574
+
575
+ test('chained calls', () => {
576
+ const ast = parse('a()()');
577
+ expect(ast.body).toHaveLength(1);
578
+ expect(ast.body[0]).toMatchObject({
579
+ type: 'ExpressionStatement',
580
+ expression: {
581
+ type: 'CallExpression',
582
+ callee: {
583
+ type: 'CallExpression',
584
+ callee: {
585
+ type: 'Identifier',
586
+ name: 'a',
587
+ },
588
+ arguments: [],
589
+ },
590
+ arguments: [],
591
+ },
592
+ });
593
+ });
594
+
595
+ // ============================================================================
596
+ // String Literal
597
+ // ============================================================================
598
+
599
+ test('double quoted string', () => {
600
+ const ast = parse('let x = "hello"');
601
+ expect(ast.body[0]).toMatchObject({
602
+ type: 'LetDeclaration',
603
+ name: 'x',
604
+ initializer: {
605
+ type: 'StringLiteral',
606
+ value: 'hello',
607
+ },
608
+ });
609
+ });
610
+
611
+ test('single quoted string', () => {
612
+ const ast = parse("let x = 'hello'");
613
+ expect(ast.body[0]).toMatchObject({
614
+ type: 'LetDeclaration',
615
+ name: 'x',
616
+ initializer: {
617
+ type: 'StringLiteral',
618
+ value: 'hello',
619
+ },
620
+ });
621
+ });
622
+
623
+ test('string with escaped quote', () => {
624
+ const ast = parse('let x = "say \\"hello\\""');
625
+ expect(ast.body[0]).toMatchObject({
626
+ type: 'LetDeclaration',
627
+ name: 'x',
628
+ initializer: {
629
+ type: 'StringLiteral',
630
+ value: 'say "hello"',
631
+ },
632
+ });
633
+ });
634
+
635
+ // ============================================================================
636
+ // Boolean Literal
637
+ // ============================================================================
638
+
639
+ test('true literal', () => {
640
+ const ast = parse('let x = true');
641
+ expect(ast.body[0]).toMatchObject({
642
+ type: 'LetDeclaration',
643
+ name: 'x',
644
+ initializer: {
645
+ type: 'BooleanLiteral',
646
+ value: true,
647
+ },
648
+ });
649
+ });
650
+
651
+ test('false literal', () => {
652
+ const ast = parse('let x = false');
653
+ expect(ast.body[0]).toMatchObject({
654
+ type: 'LetDeclaration',
655
+ name: 'x',
656
+ initializer: {
657
+ type: 'BooleanLiteral',
658
+ value: false,
659
+ },
660
+ });
661
+ });
662
+
663
+ test('boolean as expression statement', () => {
664
+ const ast = parse('true');
665
+ expect(ast.body[0]).toMatchObject({
666
+ type: 'ExpressionStatement',
667
+ expression: {
668
+ type: 'BooleanLiteral',
669
+ value: true,
670
+ },
671
+ });
672
+ });
673
+
674
+ // ============================================================================
675
+ // Identifier
676
+ // ============================================================================
677
+
678
+ test('simple identifier', () => {
679
+ const ast = parse('x');
680
+ expect(ast.body[0]).toMatchObject({
681
+ type: 'ExpressionStatement',
682
+ expression: {
683
+ type: 'Identifier',
684
+ name: 'x',
685
+ },
686
+ });
687
+ });
688
+
689
+ test('identifier with underscore', () => {
690
+ const ast = parse('my_var');
691
+ expect(ast.body[0]).toMatchObject({
692
+ type: 'ExpressionStatement',
693
+ expression: {
694
+ type: 'Identifier',
695
+ name: 'my_var',
696
+ },
697
+ });
698
+ });
699
+
700
+ test('identifier with numbers', () => {
701
+ const ast = parse('var123');
702
+ expect(ast.body[0]).toMatchObject({
703
+ type: 'ExpressionStatement',
704
+ expression: {
705
+ type: 'Identifier',
706
+ name: 'var123',
707
+ },
708
+ });
709
+ });
710
+
711
+ // ============================================================================
712
+ // Comments
713
+ // ============================================================================
714
+
715
+ test('line comment is ignored', () => {
716
+ const ast = parse(`
717
+ // this is a comment
718
+ let x = "hello"
719
+ `);
720
+ expect(ast.body).toHaveLength(1);
721
+ expect(ast.body[0]).toMatchObject({
722
+ type: 'LetDeclaration',
723
+ name: 'x',
724
+ initializer: {
725
+ type: 'StringLiteral',
726
+ value: 'hello',
727
+ },
728
+ });
729
+ });
730
+
731
+ test('block comment is ignored', () => {
732
+ const ast = parse('/* block comment */ let x = "hello"');
733
+ expect(ast.body).toHaveLength(1);
734
+ expect(ast.body[0]).toMatchObject({
735
+ type: 'LetDeclaration',
736
+ name: 'x',
737
+ initializer: {
738
+ type: 'StringLiteral',
739
+ value: 'hello',
740
+ },
741
+ });
742
+ });
743
+ });