@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,252 @@
1
+ import { describe, expect, test } from 'bun:test';
2
+ import { parse } from '../parse';
3
+
4
+ describe('Parser - TypeScript Blocks', () => {
5
+ // ============================================================================
6
+ // Basic ts block syntax
7
+ // ============================================================================
8
+
9
+ test('simple ts block with no params', () => {
10
+ const ast = parse('let x = ts() { return 42 }');
11
+ expect(ast.body).toHaveLength(1);
12
+ expect(ast.body[0]).toMatchObject({
13
+ type: 'LetDeclaration',
14
+ name: 'x',
15
+ initializer: {
16
+ type: 'TsBlock',
17
+ params: [],
18
+ body: ' return 42 ',
19
+ },
20
+ });
21
+ });
22
+
23
+ test('ts block with one param', () => {
24
+ const ast = parse('let doubled = ts(x) { return x * 2 }');
25
+ expect(ast.body).toHaveLength(1);
26
+ expect(ast.body[0]).toMatchObject({
27
+ type: 'LetDeclaration',
28
+ name: 'doubled',
29
+ initializer: {
30
+ type: 'TsBlock',
31
+ params: ['x'],
32
+ body: ' return x * 2 ',
33
+ },
34
+ });
35
+ });
36
+
37
+ test('ts block with multiple params', () => {
38
+ const ast = parse('let sum = ts(a, b, c) { return a + b + c }');
39
+ expect(ast.body).toHaveLength(1);
40
+ const tsBlock = (ast.body[0] as any).initializer;
41
+ expect(tsBlock.type).toBe('TsBlock');
42
+ expect(tsBlock.params).toEqual(['a', 'b', 'c']);
43
+ expect(tsBlock.body).toContain('return a + b + c');
44
+ });
45
+
46
+ // ============================================================================
47
+ // Multi-line ts blocks
48
+ // ============================================================================
49
+
50
+ test('multi-line ts block', () => {
51
+ const ast = parse(`let result = ts(data) {
52
+ const parsed = JSON.parse(data)
53
+ const filtered = parsed.filter(x => x > 0)
54
+ return filtered.length
55
+ }`);
56
+ expect(ast.body).toHaveLength(1);
57
+ const tsBlock = (ast.body[0] as any).initializer;
58
+ expect(tsBlock.type).toBe('TsBlock');
59
+ expect(tsBlock.params).toEqual(['data']);
60
+ expect(tsBlock.body).toContain('JSON.parse(data)');
61
+ expect(tsBlock.body).toContain('filtered.length');
62
+ });
63
+
64
+ // ============================================================================
65
+ // Nested braces in ts block
66
+ // ============================================================================
67
+
68
+ test('ts block with nested braces (object literal)', () => {
69
+ const ast = parse('let obj = ts(name) { return { name: name, id: 123 } }');
70
+ expect(ast.body).toHaveLength(1);
71
+ const tsBlock = (ast.body[0] as any).initializer;
72
+ expect(tsBlock.type).toBe('TsBlock');
73
+ expect(tsBlock.body).toContain('{ name: name, id: 123 }');
74
+ });
75
+
76
+ test('ts block with nested braces (if statement)', () => {
77
+ const ast = parse('let result = ts(x) { if (x > 0) { return x } else { return -x } }');
78
+ expect(ast.body).toHaveLength(1);
79
+ const tsBlock = (ast.body[0] as any).initializer;
80
+ expect(tsBlock.type).toBe('TsBlock');
81
+ expect(tsBlock.body).toContain('if (x > 0)');
82
+ expect(tsBlock.body).toContain('return x');
83
+ expect(tsBlock.body).toContain('return -x');
84
+ });
85
+
86
+ test('ts block with arrow function', () => {
87
+ const ast = parse('let mapped = ts(items) { return items.map(x => x * 2) }');
88
+ expect(ast.body).toHaveLength(1);
89
+ const tsBlock = (ast.body[0] as any).initializer;
90
+ expect(tsBlock.type).toBe('TsBlock');
91
+ expect(tsBlock.body).toContain('items.map(x => x * 2)');
92
+ });
93
+
94
+ // ============================================================================
95
+ // Strings inside ts block
96
+ // ============================================================================
97
+
98
+ test('ts block with single-quoted string', () => {
99
+ const ast = parse(`let greeting = ts(name) { return 'Hello, ' + name }`);
100
+ expect(ast.body).toHaveLength(1);
101
+ const tsBlock = (ast.body[0] as any).initializer;
102
+ expect(tsBlock.type).toBe('TsBlock');
103
+ expect(tsBlock.body).toContain("'Hello, '");
104
+ });
105
+
106
+ test('ts block with double-quoted string', () => {
107
+ const ast = parse('let greeting = ts(name) { return "Hello, " + name }');
108
+ expect(ast.body).toHaveLength(1);
109
+ const tsBlock = (ast.body[0] as any).initializer;
110
+ expect(tsBlock.body).toContain('"Hello, "');
111
+ });
112
+
113
+ test('ts block with template literal', () => {
114
+ const ast = parse('let greeting = ts(name) { return `Hello, ${name}!` }');
115
+ expect(ast.body).toHaveLength(1);
116
+ const tsBlock = (ast.body[0] as any).initializer;
117
+ expect(tsBlock.body).toContain('`Hello, ${name}!`');
118
+ });
119
+
120
+ test('ts block with string containing braces', () => {
121
+ const ast = parse('let jsonStr = ts() { return "{ key: value }" }');
122
+ expect(ast.body).toHaveLength(1);
123
+ const tsBlock = (ast.body[0] as any).initializer;
124
+ expect(tsBlock.type).toBe('TsBlock');
125
+ // The braces inside the string should not confuse the parser
126
+ expect(tsBlock.body).toContain('"{ key: value }"');
127
+ });
128
+
129
+ // ============================================================================
130
+ // Comments inside ts block
131
+ // ============================================================================
132
+
133
+ test('ts block with line comment', () => {
134
+ const ast = parse(`let x = ts(n) {
135
+ // Double the value
136
+ return n * 2
137
+ }`);
138
+ expect(ast.body).toHaveLength(1);
139
+ const tsBlock = (ast.body[0] as any).initializer;
140
+ expect(tsBlock.type).toBe('TsBlock');
141
+ expect(tsBlock.body).toContain('// Double the value');
142
+ });
143
+
144
+ test('ts block with block comment', () => {
145
+ const ast = parse(`let x = ts(n) {
146
+ /* Multiply by two */
147
+ return n * 2
148
+ }`);
149
+ expect(ast.body).toHaveLength(1);
150
+ const tsBlock = (ast.body[0] as any).initializer;
151
+ expect(tsBlock.type).toBe('TsBlock');
152
+ expect(tsBlock.body).toContain('/* Multiply by two */');
153
+ });
154
+
155
+ // ============================================================================
156
+ // ts block in different contexts
157
+ // ============================================================================
158
+
159
+ test('ts block in const declaration', () => {
160
+ const ast = parse('const add = ts(a, b) { return a + b }');
161
+ expect(ast.body).toHaveLength(1);
162
+ expect(ast.body[0]).toMatchObject({
163
+ type: 'ConstDeclaration',
164
+ name: 'add',
165
+ initializer: {
166
+ type: 'TsBlock',
167
+ params: ['a', 'b'],
168
+ },
169
+ });
170
+ });
171
+
172
+ test('ts block as expression statement', () => {
173
+ const ast = parse('ts(x) { console.log(x) }');
174
+ expect(ast.body).toHaveLength(1);
175
+ expect(ast.body[0]).toMatchObject({
176
+ type: 'ExpressionStatement',
177
+ expression: {
178
+ type: 'TsBlock',
179
+ params: ['x'],
180
+ },
181
+ });
182
+ });
183
+
184
+ test('ts block in function body', () => {
185
+ const ast = parse(`
186
+ function double(n: text): text {
187
+ let result = ts(n) { return n * 2 }
188
+ return result
189
+ }
190
+ `);
191
+ expect(ast.body).toHaveLength(1);
192
+ const func = ast.body[0] as any;
193
+ expect(func.type).toBe('FunctionDeclaration');
194
+ const letDecl = func.body.body[0];
195
+ expect(letDecl.initializer.type).toBe('TsBlock');
196
+ });
197
+
198
+ // ============================================================================
199
+ // ts block with whitespace variations
200
+ // ============================================================================
201
+
202
+ test('ts block with extra whitespace', () => {
203
+ const ast = parse('let x = ts( a , b ) { return a + b }');
204
+ expect(ast.body).toHaveLength(1);
205
+ const tsBlock = (ast.body[0] as any).initializer;
206
+ expect(tsBlock.params).toEqual(['a', 'b']);
207
+ });
208
+
209
+ test('ts block with newline before body', () => {
210
+ const ast = parse(`let x = ts(a, b)
211
+ { return a + b }`);
212
+ expect(ast.body).toHaveLength(1);
213
+ const tsBlock = (ast.body[0] as any).initializer;
214
+ expect(tsBlock.type).toBe('TsBlock');
215
+ expect(tsBlock.params).toEqual(['a', 'b']);
216
+ });
217
+
218
+ // ============================================================================
219
+ // Edge cases
220
+ // ============================================================================
221
+
222
+ test('ts is not a reserved word by itself', () => {
223
+ // 'ts' followed by something other than '(' should be an identifier
224
+ const ast = parse('let ts = "typescript"');
225
+ expect(ast.body).toHaveLength(1);
226
+ expect(ast.body[0]).toMatchObject({
227
+ type: 'LetDeclaration',
228
+ name: 'ts',
229
+ initializer: {
230
+ type: 'StringLiteral',
231
+ value: 'typescript',
232
+ },
233
+ });
234
+ });
235
+
236
+ test('multiple ts blocks in same program', () => {
237
+ const ast = parse(`
238
+ let add = ts(a, b) { return a + b }
239
+ let mul = ts(a, b) { return a * b }
240
+ `);
241
+ expect(ast.body).toHaveLength(2);
242
+ expect((ast.body[0] as any).initializer.type).toBe('TsBlock');
243
+ expect((ast.body[1] as any).initializer.type).toBe('TsBlock');
244
+ });
245
+
246
+ test('ts block with async/await', () => {
247
+ const ast = parse('let data = ts(url) { return await fetch(url).then(r => r.json()) }');
248
+ expect(ast.body).toHaveLength(1);
249
+ const tsBlock = (ast.body[0] as any).initializer;
250
+ expect(tsBlock.body).toContain('await fetch(url)');
251
+ });
252
+ });
@@ -0,0 +1,254 @@
1
+ import { describe, expect, test } from 'bun:test';
2
+ import { parse } from '../parse';
3
+
4
+ describe('Parser - Type Annotations', () => {
5
+ // ============================================================================
6
+ // Let with type annotations
7
+ // ============================================================================
8
+
9
+ test('let with text type', () => {
10
+ const ast = parse('let x: text = "hello"');
11
+ expect(ast.body).toHaveLength(1);
12
+ expect(ast.body[0]).toMatchObject({
13
+ type: 'LetDeclaration',
14
+ name: 'x',
15
+ typeAnnotation: 'text',
16
+ initializer: {
17
+ type: 'StringLiteral',
18
+ value: 'hello',
19
+ },
20
+ });
21
+ });
22
+
23
+ test('let with json type', () => {
24
+ const ast = parse('let x: json = "{\\"key\\": \\"value\\"}"');
25
+ expect(ast.body).toHaveLength(1);
26
+ expect(ast.body[0]).toMatchObject({
27
+ type: 'LetDeclaration',
28
+ name: 'x',
29
+ typeAnnotation: 'json',
30
+ initializer: {
31
+ type: 'StringLiteral',
32
+ },
33
+ });
34
+ });
35
+
36
+ test('let with text type no initializer', () => {
37
+ const ast = parse('let x: text');
38
+ expect(ast.body).toHaveLength(1);
39
+ expect(ast.body[0]).toMatchObject({
40
+ type: 'LetDeclaration',
41
+ name: 'x',
42
+ typeAnnotation: 'text',
43
+ initializer: null,
44
+ });
45
+ });
46
+
47
+ test('let with json type no initializer', () => {
48
+ const ast = parse('let x: json');
49
+ expect(ast.body).toHaveLength(1);
50
+ expect(ast.body[0]).toMatchObject({
51
+ type: 'LetDeclaration',
52
+ name: 'x',
53
+ typeAnnotation: 'json',
54
+ initializer: null,
55
+ });
56
+ });
57
+
58
+ test('let without type annotation (null)', () => {
59
+ const ast = parse('let x = "hello"');
60
+ expect(ast.body).toHaveLength(1);
61
+ expect(ast.body[0]).toMatchObject({
62
+ type: 'LetDeclaration',
63
+ name: 'x',
64
+ typeAnnotation: null,
65
+ });
66
+ });
67
+
68
+ // ============================================================================
69
+ // Const with type annotations
70
+ // ============================================================================
71
+
72
+ test('const with text type', () => {
73
+ const ast = parse('const x: text = "hello"');
74
+ expect(ast.body).toHaveLength(1);
75
+ expect(ast.body[0]).toMatchObject({
76
+ type: 'ConstDeclaration',
77
+ name: 'x',
78
+ typeAnnotation: 'text',
79
+ initializer: {
80
+ type: 'StringLiteral',
81
+ value: 'hello',
82
+ },
83
+ });
84
+ });
85
+
86
+ test('const with json type', () => {
87
+ const ast = parse('const x: json = "[]"');
88
+ expect(ast.body).toHaveLength(1);
89
+ expect(ast.body[0]).toMatchObject({
90
+ type: 'ConstDeclaration',
91
+ name: 'x',
92
+ typeAnnotation: 'json',
93
+ initializer: {
94
+ type: 'StringLiteral',
95
+ value: '[]',
96
+ },
97
+ });
98
+ });
99
+
100
+ test('const without type annotation (null)', () => {
101
+ const ast = parse('const x = "hello"');
102
+ expect(ast.body).toHaveLength(1);
103
+ expect(ast.body[0]).toMatchObject({
104
+ type: 'ConstDeclaration',
105
+ name: 'x',
106
+ typeAnnotation: null,
107
+ });
108
+ });
109
+
110
+ // ============================================================================
111
+ // Prompt type annotation
112
+ // ============================================================================
113
+
114
+ test('let with prompt type', () => {
115
+ const ast = parse('let x: prompt = "What is your name?"');
116
+ expect(ast.body).toHaveLength(1);
117
+ expect(ast.body[0]).toMatchObject({
118
+ type: 'LetDeclaration',
119
+ name: 'x',
120
+ typeAnnotation: 'prompt',
121
+ initializer: {
122
+ type: 'StringLiteral',
123
+ value: 'What is your name?',
124
+ },
125
+ });
126
+ });
127
+
128
+ test('const with prompt type', () => {
129
+ const ast = parse('const SYSTEM_PROMPT: prompt = "You are a helpful assistant"');
130
+ expect(ast.body).toHaveLength(1);
131
+ expect(ast.body[0]).toMatchObject({
132
+ type: 'ConstDeclaration',
133
+ name: 'SYSTEM_PROMPT',
134
+ typeAnnotation: 'prompt',
135
+ initializer: {
136
+ type: 'StringLiteral',
137
+ value: 'You are a helpful assistant',
138
+ },
139
+ });
140
+ });
141
+
142
+ test('let with prompt type no initializer', () => {
143
+ const ast = parse('let x: prompt');
144
+ expect(ast.body).toHaveLength(1);
145
+ expect(ast.body[0]).toMatchObject({
146
+ type: 'LetDeclaration',
147
+ name: 'x',
148
+ typeAnnotation: 'prompt',
149
+ initializer: null,
150
+ });
151
+ });
152
+
153
+ // ============================================================================
154
+ // Array type annotations
155
+ // ============================================================================
156
+
157
+ test('let with text[] array type', () => {
158
+ const ast = parse('let items: text[] = ["a", "b"]');
159
+ expect(ast.body).toHaveLength(1);
160
+ expect(ast.body[0]).toMatchObject({
161
+ type: 'LetDeclaration',
162
+ name: 'items',
163
+ typeAnnotation: 'text[]',
164
+ initializer: {
165
+ type: 'ArrayLiteral',
166
+ elements: [
167
+ { type: 'StringLiteral', value: 'a' },
168
+ { type: 'StringLiteral', value: 'b' },
169
+ ],
170
+ },
171
+ });
172
+ });
173
+
174
+ test('let with boolean[] array type', () => {
175
+ const ast = parse('let flags: boolean[] = [true, false]');
176
+ expect(ast.body).toHaveLength(1);
177
+ expect(ast.body[0]).toMatchObject({
178
+ type: 'LetDeclaration',
179
+ name: 'flags',
180
+ typeAnnotation: 'boolean[]',
181
+ });
182
+ });
183
+
184
+ test('let with nested text[][] array type', () => {
185
+ const ast = parse('let matrix: text[][] = [["a"], ["b"]]');
186
+ expect(ast.body).toHaveLength(1);
187
+ expect(ast.body[0]).toMatchObject({
188
+ type: 'LetDeclaration',
189
+ name: 'matrix',
190
+ typeAnnotation: 'text[][]',
191
+ });
192
+ });
193
+
194
+ test('const with json[] array type', () => {
195
+ const ast = parse('const items: json[] = [{}, {}]');
196
+ expect(ast.body).toHaveLength(1);
197
+ expect(ast.body[0]).toMatchObject({
198
+ type: 'ConstDeclaration',
199
+ name: 'items',
200
+ typeAnnotation: 'json[]',
201
+ });
202
+ });
203
+
204
+ test('function with array parameter', () => {
205
+ const ast = parse('function process(items: text[]) { return items }');
206
+ expect(ast.body).toHaveLength(1);
207
+ expect(ast.body[0]).toMatchObject({
208
+ type: 'FunctionDeclaration',
209
+ name: 'process',
210
+ params: [{ name: 'items', typeAnnotation: 'text[]' }],
211
+ });
212
+ });
213
+
214
+ test('function with array return type', () => {
215
+ const ast = parse('function getItems(): text[] { return ["a"] }');
216
+ expect(ast.body).toHaveLength(1);
217
+ expect(ast.body[0]).toMatchObject({
218
+ type: 'FunctionDeclaration',
219
+ name: 'getItems',
220
+ returnType: 'text[]',
221
+ });
222
+ });
223
+
224
+ test('function with nested array return type', () => {
225
+ const ast = parse('function getMatrix(): boolean[][] { return [[true]] }');
226
+ expect(ast.body).toHaveLength(1);
227
+ expect(ast.body[0]).toMatchObject({
228
+ type: 'FunctionDeclaration',
229
+ name: 'getMatrix',
230
+ returnType: 'boolean[][]',
231
+ });
232
+ });
233
+
234
+ test('function with model parameter type', () => {
235
+ const ast = parse('function process(m: model): text { return "done" }');
236
+ expect(ast.body).toHaveLength(1);
237
+ expect(ast.body[0]).toMatchObject({
238
+ type: 'FunctionDeclaration',
239
+ name: 'process',
240
+ params: [{ name: 'm', typeAnnotation: 'model' }],
241
+ returnType: 'text',
242
+ });
243
+ });
244
+
245
+ test('const declaration with model type', () => {
246
+ const ast = parse('const myModel: model = undefined');
247
+ expect(ast.body).toHaveLength(1);
248
+ expect(ast.body[0]).toMatchObject({
249
+ type: 'ConstDeclaration',
250
+ name: 'myModel',
251
+ typeAnnotation: 'model',
252
+ });
253
+ });
254
+ });