freelang-v4 4.3.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 (261) hide show
  1. package/README.md +548 -0
  2. package/dist/ast.d.ts +367 -0
  3. package/dist/ast.js +4 -0
  4. package/dist/ast.js.map +1 -0
  5. package/dist/async-basic.test.d.ts +1 -0
  6. package/dist/async-basic.test.js +88 -0
  7. package/dist/async-basic.test.js.map +1 -0
  8. package/dist/async-jest.test.d.ts +1 -0
  9. package/dist/async-jest.test.js +99 -0
  10. package/dist/async-jest.test.js.map +1 -0
  11. package/dist/channel-jest.test.d.ts +1 -0
  12. package/dist/channel-jest.test.js +148 -0
  13. package/dist/channel-jest.test.js.map +1 -0
  14. package/dist/checker-jest.test.d.ts +1 -0
  15. package/dist/checker-jest.test.js +160 -0
  16. package/dist/checker-jest.test.js.map +1 -0
  17. package/dist/checker.d.ts +149 -0
  18. package/dist/checker.js +1565 -0
  19. package/dist/checker.js.map +1 -0
  20. package/dist/checker.test.d.ts +1 -0
  21. package/dist/checker.test.js +217 -0
  22. package/dist/checker.test.js.map +1 -0
  23. package/dist/compiler-jest.test.d.ts +1 -0
  24. package/dist/compiler-jest.test.js +233 -0
  25. package/dist/compiler-jest.test.js.map +1 -0
  26. package/dist/compiler.d.ts +127 -0
  27. package/dist/compiler.js +1588 -0
  28. package/dist/compiler.js.map +1 -0
  29. package/dist/compiler.test.d.ts +1 -0
  30. package/dist/compiler.test.js +313 -0
  31. package/dist/compiler.test.js.map +1 -0
  32. package/dist/db-100m-full.d.ts +5 -0
  33. package/dist/db-100m-full.js +78 -0
  34. package/dist/db-100m-full.js.map +1 -0
  35. package/dist/db-100m-no-index.d.ts +12 -0
  36. package/dist/db-100m-no-index.js +119 -0
  37. package/dist/db-100m-no-index.js.map +1 -0
  38. package/dist/db-100m-real.d.ts +5 -0
  39. package/dist/db-100m-real.js +131 -0
  40. package/dist/db-100m-real.js.map +1 -0
  41. package/dist/db-100m-streaming.d.ts +15 -0
  42. package/dist/db-100m-streaming.js +164 -0
  43. package/dist/db-100m-streaming.js.map +1 -0
  44. package/dist/db-100m-test.d.ts +5 -0
  45. package/dist/db-100m-test.js +111 -0
  46. package/dist/db-100m-test.js.map +1 -0
  47. package/dist/db-jest.test.d.ts +1 -0
  48. package/dist/db-jest.test.js +182 -0
  49. package/dist/db-jest.test.js.map +1 -0
  50. package/dist/db-runtime.d.ts +24 -0
  51. package/dist/db-runtime.js +204 -0
  52. package/dist/db-runtime.js.map +1 -0
  53. package/dist/db.d.ts +249 -0
  54. package/dist/db.js +593 -0
  55. package/dist/db.js.map +1 -0
  56. package/dist/file-io-jest.test.d.ts +1 -0
  57. package/dist/file-io-jest.test.js +225 -0
  58. package/dist/file-io-jest.test.js.map +1 -0
  59. package/dist/for-of-jest.test.d.ts +1 -0
  60. package/dist/for-of-jest.test.js +230 -0
  61. package/dist/for-of-jest.test.js.map +1 -0
  62. package/dist/for-of.test.d.ts +1 -0
  63. package/dist/for-of.test.js +305 -0
  64. package/dist/for-of.test.js.map +1 -0
  65. package/dist/function-literal-jest.test.d.ts +1 -0
  66. package/dist/function-literal-jest.test.js +180 -0
  67. package/dist/function-literal-jest.test.js.map +1 -0
  68. package/dist/function-literal.test.d.ts +1 -0
  69. package/dist/function-literal.test.js +245 -0
  70. package/dist/function-literal.test.js.map +1 -0
  71. package/dist/generics-jest.test.d.ts +1 -0
  72. package/dist/generics-jest.test.js +93 -0
  73. package/dist/generics-jest.test.js.map +1 -0
  74. package/dist/ir-gen.d.ts +15 -0
  75. package/dist/ir-gen.js +400 -0
  76. package/dist/ir-gen.js.map +1 -0
  77. package/dist/ir.d.ts +114 -0
  78. package/dist/ir.js +5 -0
  79. package/dist/ir.js.map +1 -0
  80. package/dist/lexer.d.ts +110 -0
  81. package/dist/lexer.js +467 -0
  82. package/dist/lexer.js.map +1 -0
  83. package/dist/lexer.test.d.ts +1 -0
  84. package/dist/lexer.test.js +426 -0
  85. package/dist/lexer.test.js.map +1 -0
  86. package/dist/main.d.ts +2 -0
  87. package/dist/main.js +241 -0
  88. package/dist/main.js.map +1 -0
  89. package/dist/module-jest.test.d.ts +1 -0
  90. package/dist/module-jest.test.js +123 -0
  91. package/dist/module-jest.test.js.map +1 -0
  92. package/dist/parser.d.ts +56 -0
  93. package/dist/parser.js +1060 -0
  94. package/dist/parser.js.map +1 -0
  95. package/dist/parser.test.d.ts +1 -0
  96. package/dist/parser.test.js +461 -0
  97. package/dist/parser.test.js.map +1 -0
  98. package/dist/pattern-matching-jest.test.d.ts +1 -0
  99. package/dist/pattern-matching-jest.test.js +158 -0
  100. package/dist/pattern-matching-jest.test.js.map +1 -0
  101. package/dist/pkg/init.d.ts +1 -0
  102. package/dist/pkg/init.js +118 -0
  103. package/dist/pkg/init.js.map +1 -0
  104. package/dist/pkg/install.d.ts +1 -0
  105. package/dist/pkg/install.js +77 -0
  106. package/dist/pkg/install.js.map +1 -0
  107. package/dist/pkg/registry.d.ts +23 -0
  108. package/dist/pkg/registry.js +106 -0
  109. package/dist/pkg/registry.js.map +1 -0
  110. package/dist/pkg/run.d.ts +1 -0
  111. package/dist/pkg/run.js +76 -0
  112. package/dist/pkg/run.js.map +1 -0
  113. package/dist/pkg/toml.d.ts +5 -0
  114. package/dist/pkg/toml.js +117 -0
  115. package/dist/pkg/toml.js.map +1 -0
  116. package/dist/repl.d.ts +15 -0
  117. package/dist/repl.js +197 -0
  118. package/dist/repl.js.map +1 -0
  119. package/dist/runtime/bytecode.d.ts +92 -0
  120. package/dist/runtime/bytecode.js +253 -0
  121. package/dist/runtime/bytecode.js.map +1 -0
  122. package/dist/runtime/value.d.ts +102 -0
  123. package/dist/runtime/value.js +302 -0
  124. package/dist/runtime/value.js.map +1 -0
  125. package/dist/runtime/vm.d.ts +65 -0
  126. package/dist/runtime/vm.js +293 -0
  127. package/dist/runtime/vm.js.map +1 -0
  128. package/dist/struct-instance-jest.test.d.ts +1 -0
  129. package/dist/struct-instance-jest.test.js +209 -0
  130. package/dist/struct-instance-jest.test.js.map +1 -0
  131. package/dist/struct-instance.test.d.ts +1 -0
  132. package/dist/struct-instance.test.js +291 -0
  133. package/dist/struct-instance.test.js.map +1 -0
  134. package/dist/struct-jest.test.d.ts +1 -0
  135. package/dist/struct-jest.test.js +176 -0
  136. package/dist/struct-jest.test.js.map +1 -0
  137. package/dist/struct.test.d.ts +1 -0
  138. package/dist/struct.test.js +231 -0
  139. package/dist/struct.test.js.map +1 -0
  140. package/dist/trait-jest.test.d.ts +1 -0
  141. package/dist/trait-jest.test.js +120 -0
  142. package/dist/trait-jest.test.js.map +1 -0
  143. package/dist/vm-jest.test.d.ts +1 -0
  144. package/dist/vm-jest.test.js +569 -0
  145. package/dist/vm-jest.test.js.map +1 -0
  146. package/dist/vm.d.ts +81 -0
  147. package/dist/vm.js +1956 -0
  148. package/dist/vm.js.map +1 -0
  149. package/dist/vm.test.d.ts +1 -0
  150. package/dist/vm.test.js +337 -0
  151. package/dist/vm.test.js.map +1 -0
  152. package/dist/web-repl/sandbox.d.ts +11 -0
  153. package/dist/web-repl/sandbox.js +76 -0
  154. package/dist/web-repl/sandbox.js.map +1 -0
  155. package/dist/web-repl/server.d.ts +1 -0
  156. package/dist/web-repl/server.js +111 -0
  157. package/dist/web-repl/server.js.map +1 -0
  158. package/dist/while-loop-jest.test.d.ts +1 -0
  159. package/dist/while-loop-jest.test.js +201 -0
  160. package/dist/while-loop-jest.test.js.map +1 -0
  161. package/dist/while-loop.test.d.ts +1 -0
  162. package/dist/while-loop.test.js +262 -0
  163. package/dist/while-loop.test.js.map +1 -0
  164. package/docs/EXPERIENCE.md +787 -0
  165. package/docs/README.md +175 -0
  166. package/docs/V1_V2_V3_ANALYSIS.md +107 -0
  167. package/docs/_config.yml +36 -0
  168. package/docs/api-reference.md +459 -0
  169. package/docs/architecture.md +470 -0
  170. package/docs/benchmarks.md +295 -0
  171. package/docs/comparison.md +454 -0
  172. package/docs/index.md +335 -0
  173. package/docs/language-completeness.md +228 -0
  174. package/docs/learning-guide.md +651 -0
  175. package/package.json +65 -0
  176. package/src/api/deploy_key.fl +294 -0
  177. package/src/api/issue.fl +302 -0
  178. package/src/api/org.fl +356 -0
  179. package/src/api/repo.fl +394 -0
  180. package/src/api/team.fl +299 -0
  181. package/src/api/user.fl +385 -0
  182. package/src/api/webhook.fl +273 -0
  183. package/src/ast.ts +158 -0
  184. package/src/async-basic.test.ts +94 -0
  185. package/src/async-jest.test.ts +107 -0
  186. package/src/channel-jest.test.ts +158 -0
  187. package/src/checker-jest.test.ts +189 -0
  188. package/src/checker.test.ts +279 -0
  189. package/src/checker.ts +1861 -0
  190. package/src/commands/analyze.fl +227 -0
  191. package/src/commands/auth.fl +315 -0
  192. package/src/commands/batch.fl +349 -0
  193. package/src/commands/config.fl +199 -0
  194. package/src/commands/deploy_key.fl +352 -0
  195. package/src/commands/issue.fl +275 -0
  196. package/src/commands/main.fl +492 -0
  197. package/src/commands/org.fl +425 -0
  198. package/src/commands/repo.fl +581 -0
  199. package/src/commands/team.fl +244 -0
  200. package/src/commands/user.fl +423 -0
  201. package/src/commands/webhook.fl +400 -0
  202. package/src/compiler-jest.test.ts +275 -0
  203. package/src/compiler.test.ts +375 -0
  204. package/src/compiler.ts +1770 -0
  205. package/src/config.fl +175 -0
  206. package/src/core/batch.fl +355 -0
  207. package/src/core/cache.fl +284 -0
  208. package/src/core/ensure.fl +324 -0
  209. package/src/db-100m-full.ts +96 -0
  210. package/src/db-100m-no-index.ts +133 -0
  211. package/src/db-100m-real.ts +152 -0
  212. package/src/db-100m-streaming.ts +154 -0
  213. package/src/db-100m-test.ts +136 -0
  214. package/src/db-jest.test.ts +161 -0
  215. package/src/db-runtime.ts +242 -0
  216. package/src/db.ts +676 -0
  217. package/src/errors.fl +134 -0
  218. package/src/for-of-jest.test.ts +246 -0
  219. package/src/for-of.test.ts +308 -0
  220. package/src/function-literal-jest.test.ts +193 -0
  221. package/src/function-literal.test.ts +248 -0
  222. package/src/generics-jest.test.ts +104 -0
  223. package/src/http/client.fl +327 -0
  224. package/src/ir-gen.ts +459 -0
  225. package/src/ir.ts +80 -0
  226. package/src/lexer.test.ts +499 -0
  227. package/src/lexer.ts +522 -0
  228. package/src/main.ts +223 -0
  229. package/src/models.fl +162 -0
  230. package/src/module-jest.test.ts +145 -0
  231. package/src/parser.test.ts +542 -0
  232. package/src/parser.ts +1211 -0
  233. package/src/pattern-matching-jest.test.ts +170 -0
  234. package/src/pkg/init.ts +91 -0
  235. package/src/pkg/install.ts +56 -0
  236. package/src/pkg/registry.ts +103 -0
  237. package/src/pkg/run.ts +49 -0
  238. package/src/pkg/toml.ts +129 -0
  239. package/src/repl.ts +190 -0
  240. package/src/runtime/bytecode.ts +291 -0
  241. package/src/runtime/value.ts +322 -0
  242. package/src/runtime/vm.ts +354 -0
  243. package/src/self-host/bootstrap.fl +68 -0
  244. package/src/self-host/interpreter.fl +361 -0
  245. package/src/self-host/lexer-simple.fl +22 -0
  246. package/src/self-host/lexer.fl +305 -0
  247. package/src/self-host/parser.fl +580 -0
  248. package/src/struct-instance-jest.test.ts +221 -0
  249. package/src/struct-instance.test.ts +293 -0
  250. package/src/struct-jest.test.ts +187 -0
  251. package/src/struct.test.ts +234 -0
  252. package/src/trait-jest.test.ts +136 -0
  253. package/src/vm-jest.test.ts +754 -0
  254. package/src/vm.ts +1976 -0
  255. package/src/web-repl/public/index.html +50 -0
  256. package/src/web-repl/public/main.js +105 -0
  257. package/src/web-repl/public/style.css +225 -0
  258. package/src/web-repl/sandbox.ts +88 -0
  259. package/src/web-repl/server.ts +97 -0
  260. package/src/while-loop-jest.test.ts +218 -0
  261. package/src/while-loop.test.ts +267 -0
@@ -0,0 +1,305 @@
1
+ // ============================================================
2
+ // FreeLang v4 Self-Hosting Lexer (lexer.fl)
3
+ // Tree-walking interpreter - Tokenization phase
4
+ // ============================================================
5
+
6
+ // ============================================================
7
+ // Token Types (Enum)
8
+ // ============================================================
9
+
10
+ struct Token {
11
+ kind: str,
12
+ value: str,
13
+ line: i32,
14
+ }
15
+
16
+ // ============================================================
17
+ // Lexer (Struct)
18
+ // ============================================================
19
+
20
+ struct Lexer {
21
+ source: str,
22
+ pos: i32,
23
+ line: i32,
24
+ }
25
+
26
+ // ============================================================
27
+ // Helper Functions
28
+ // ============================================================
29
+
30
+ fn is_digit(c: str) -> bool {
31
+ var code = char_code(c)
32
+ return code >= 48 && code <= 57 // '0' = 48, '9' = 57
33
+ }
34
+
35
+ fn is_alpha(c: str) -> bool {
36
+ var code = char_code(c)
37
+ return (code >= 65 && code <= 90) || (code >= 97 && code <= 122) || code == 95 // A-Z, a-z, _
38
+ }
39
+
40
+ fn is_alphanumeric(c: str) -> bool {
41
+ return is_alpha(c) || is_digit(c)
42
+ }
43
+
44
+ fn is_whitespace(c: str) -> bool {
45
+ return c == " " || c == "\t" || c == "\n" || c == "\r"
46
+ }
47
+
48
+ fn char_code(c: str) -> i32 {
49
+ // ASCII code extraction - simplified for common chars
50
+ if c == " " { return 32 }
51
+ if c == "\t" { return 9 }
52
+ if c == "\n" { return 10 }
53
+ if c == "0" { return 48 }
54
+ if c == "1" { return 49 }
55
+ if c == "2" { return 50 }
56
+ if c == "3" { return 51 }
57
+ if c == "4" { return 52 }
58
+ if c == "5" { return 53 }
59
+ if c == "6" { return 54 }
60
+ if c == "7" { return 55 }
61
+ if c == "8" { return 56 }
62
+ if c == "9" { return 57 }
63
+ if c == "A" { return 65 }
64
+ if c == "Z" { return 90 }
65
+ if c == "a" { return 97 }
66
+ if c == "z" { return 122 }
67
+ if c == "_" { return 95 }
68
+ return 0
69
+ }
70
+
71
+ // ============================================================
72
+ // Lexer Methods
73
+ // ============================================================
74
+
75
+ fn lexer_new(source: str) -> Lexer {
76
+ return Lexer { source: source, pos: 0, line: 1 }
77
+ }
78
+
79
+ fn lexer_current_char(lexer: Lexer) -> str {
80
+ if lexer.pos >= length(lexer.source) {
81
+ return ""
82
+ }
83
+ return char_at(lexer.source, lexer.pos)
84
+ }
85
+
86
+ fn lexer_peek_char(lexer: Lexer, offset: i32) -> str {
87
+ var p = lexer.pos + offset
88
+ if p >= length(lexer.source) {
89
+ return ""
90
+ }
91
+ return char_at(lexer.source, p)
92
+ }
93
+
94
+ fn lexer_advance(lexer: Lexer) -> Lexer {
95
+ var c = lexer_current_char(lexer)
96
+ if c == "\n" {
97
+ return Lexer { source: lexer.source, pos: lexer.pos + 1, line: lexer.line + 1 }
98
+ }
99
+ return Lexer { source: lexer.source, pos: lexer.pos + 1, line: lexer.line }
100
+ }
101
+
102
+ fn lexer_skip_whitespace(lexer: Lexer) -> Lexer {
103
+ var c = lexer_current_char(lexer)
104
+ if is_whitespace(c) && c != "" {
105
+ return lexer_skip_whitespace(lexer_advance(lexer))
106
+ }
107
+ return lexer
108
+ }
109
+
110
+ fn lexer_skip_comment(lexer: Lexer) -> Lexer {
111
+ // Skip // comment to end of line
112
+ var c = lexer_current_char(lexer)
113
+ if c == "/" && lexer_peek_char(lexer, 1) == "/" {
114
+ var l2 = lexer_advance(lexer)
115
+ var l3 = lexer_advance(l2)
116
+ var l4 = l3
117
+ while lexer_current_char(l4) != "\n" && lexer_current_char(l4) != "" {
118
+ l4 = lexer_advance(l4)
119
+ }
120
+ return l4
121
+ }
122
+ return lexer
123
+ }
124
+
125
+ // ============================================================
126
+ // Number Lexing
127
+ // ============================================================
128
+
129
+ fn lex_number(lexer: Lexer) -> Token {
130
+ var start = lexer.pos
131
+ var l = lexer
132
+ var has_dot = false
133
+
134
+ while is_digit(lexer_current_char(l)) || (lexer_current_char(l) == "." && !has_dot) {
135
+ if lexer_current_char(l) == "." {
136
+ has_dot = true
137
+ }
138
+ l = lexer_advance(l)
139
+ }
140
+
141
+ var value = slice(lexer.source, start, l.pos)
142
+ var kind = if has_dot { "FLOAT" } else { "INT" }
143
+
144
+ return Token { kind: kind, value: value, line: lexer.line }
145
+ }
146
+
147
+ // ============================================================
148
+ // String Lexing
149
+ // ============================================================
150
+
151
+ fn lex_string(lexer: Lexer) -> Token {
152
+ var l = lexer_advance(lexer) // Skip opening "
153
+ var start = l.pos
154
+
155
+ while lexer_current_char(l) != "\"" && lexer_current_char(l) != "" {
156
+ if lexer_current_char(l) == "\\" {
157
+ l = lexer_advance(l)
158
+ }
159
+ l = lexer_advance(l)
160
+ }
161
+
162
+ var value = slice(lexer.source, start, l.pos)
163
+ var l2 = lexer_advance(l) // Skip closing "
164
+
165
+ return Token { kind: "STRING", value: value, line: lexer.line }
166
+ }
167
+
168
+ // ============================================================
169
+ // Identifier/Keyword Lexing
170
+ // ============================================================
171
+
172
+ fn lex_ident(lexer: Lexer) -> Token {
173
+ var start = lexer.pos
174
+ var l = lexer
175
+
176
+ while is_alphanumeric(lexer_current_char(l)) {
177
+ l = lexer_advance(l)
178
+ }
179
+
180
+ var value = slice(lexer.source, start, l.pos)
181
+ var kind = if is_keyword(value) { "KEYWORD" } else { "IDENT" }
182
+
183
+ return Token { kind: kind, value: value, line: lexer.line }
184
+ }
185
+
186
+ fn is_keyword(word: str) -> bool {
187
+ var keywords = ["var", "let", "const", "fn", "return", "if", "else", "while", "for", "match", "struct", "enum", "impl", "trait", "import", "export", "print", "println", "true", "false", "null"]
188
+ var i = 0
189
+ while i < length(keywords) {
190
+ if keywords[i] == word {
191
+ return true
192
+ }
193
+ i = i + 1
194
+ }
195
+ return false
196
+ }
197
+
198
+ // ============================================================
199
+ // Main Lexer
200
+ // ============================================================
201
+
202
+ fn lex(source: str) -> [Token] {
203
+ var tokens: [Token] = []
204
+ var lexer = lexer_new(source)
205
+
206
+ while lexer.pos < length(source) {
207
+ lexer = lexer_skip_whitespace(lexer)
208
+ lexer = lexer_skip_comment(lexer)
209
+
210
+ if lexer.pos >= length(source) {
211
+ break
212
+ }
213
+
214
+ var c = lexer_current_char(lexer)
215
+
216
+ if is_digit(c) {
217
+ var token = lex_number(lexer)
218
+ tokens = push(tokens, token)
219
+ // Update lexer position - need to advance to token end
220
+ lexer.pos = lexer.pos + length(token.value)
221
+ lexer = Lexer { source: lexer.source, pos: lexer.pos, line: lexer.line }
222
+ } else if c == "\"" {
223
+ var token = lex_string(lexer)
224
+ tokens = push(tokens, token)
225
+ lexer.pos = lexer.pos + length(token.value) + 2 // +2 for quotes
226
+ lexer = Lexer { source: lexer.source, pos: lexer.pos, line: lexer.line }
227
+ } else if is_alpha(c) {
228
+ var token = lex_ident(lexer)
229
+ tokens = push(tokens, token)
230
+ lexer.pos = lexer.pos + length(token.value)
231
+ lexer = Lexer { source: lexer.source, pos: lexer.pos, line: lexer.line }
232
+ } else if c == "+" {
233
+ var token = Token { kind: "PLUS", value: "+", line: lexer.line }
234
+ tokens = push(tokens, token)
235
+ lexer = lexer_advance(lexer)
236
+ } else if c == "-" {
237
+ var token = Token { kind: "MINUS", value: "-", line: lexer.line }
238
+ tokens = push(tokens, token)
239
+ lexer = lexer_advance(lexer)
240
+ } else if c == "*" {
241
+ var token = Token { kind: "STAR", value: "*", line: lexer.line }
242
+ tokens = push(tokens, token)
243
+ lexer = lexer_advance(lexer)
244
+ } else if c == "/" {
245
+ var token = Token { kind: "SLASH", value: "/", line: lexer.line }
246
+ tokens = push(tokens, token)
247
+ lexer = lexer_advance(lexer)
248
+ } else if c == "=" {
249
+ if lexer_peek_char(lexer, 1) == "=" {
250
+ var token = Token { kind: "EQ", value: "==", line: lexer.line }
251
+ tokens = push(tokens, token)
252
+ lexer = lexer_advance(lexer)
253
+ lexer = lexer_advance(lexer)
254
+ } else {
255
+ var token = Token { kind: "ASSIGN", value: "=", line: lexer.line }
256
+ tokens = push(tokens, token)
257
+ lexer = lexer_advance(lexer)
258
+ }
259
+ } else if c == "(" {
260
+ var token = Token { kind: "LPAREN", value: "(", line: lexer.line }
261
+ tokens = push(tokens, token)
262
+ lexer = lexer_advance(lexer)
263
+ } else if c == ")" {
264
+ var token = Token { kind: "RPAREN", value: ")", line: lexer.line }
265
+ tokens = push(tokens, token)
266
+ lexer = lexer_advance(lexer)
267
+ } else if c == "{" {
268
+ var token = Token { kind: "LBRACE", value: "{", line: lexer.line }
269
+ tokens = push(tokens, token)
270
+ lexer = lexer_advance(lexer)
271
+ } else if c == "}" {
272
+ var token = Token { kind: "RBRACE", value: "}", line: lexer.line }
273
+ tokens = push(tokens, token)
274
+ lexer = lexer_advance(lexer)
275
+ } else {
276
+ // Unknown character - skip
277
+ lexer = lexer_advance(lexer)
278
+ }
279
+ }
280
+
281
+ // Add EOF token
282
+ var eof = Token { kind: "EOF", value: "", line: lexer.line }
283
+ tokens = push(tokens, eof)
284
+
285
+ return tokens
286
+ }
287
+
288
+ // ============================================================
289
+ // Test Entry Point
290
+ // ============================================================
291
+
292
+ fn test_lexer() -> void {
293
+ var source = "var x = 42"
294
+ var tokens = lex(source)
295
+
296
+ println("Tokens:")
297
+ var i = 0
298
+ while i < length(tokens) {
299
+ var t = tokens[i]
300
+ println(t.kind + ": " + t.value)
301
+ i = i + 1
302
+ }
303
+ }
304
+
305
+ test_lexer()