freelang-editor 11.7.4

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 (1473) hide show
  1. package/.config/jest.config.js +47 -0
  2. package/.config/jest.fast.config.js +9 -0
  3. package/.config/openapi.yaml +457 -0
  4. package/.config/tsconfig.json +30 -0
  5. package/.github/CODE_OF_CONDUCT.md +176 -0
  6. package/.github/CONTRIBUTING.md +296 -0
  7. package/.github/DISCUSSIONS.md +179 -0
  8. package/.github/ISSUE_TEMPLATE/bug-report.md +44 -0
  9. package/.github/ISSUE_TEMPLATE/feature-request.md +39 -0
  10. package/.github/pull_request_template.md +87 -0
  11. package/.github/workflows/phase-3c-l2-proof.yml +101 -0
  12. package/.github/workflows/phase-c-full-slack.yml +409 -0
  13. package/.github/workflows/phase-c-full.yml +241 -0
  14. package/.github/workflows/phase-c-scan.yml +64 -0
  15. package/.gitmodules +3 -0
  16. package/.tmp.js +3 -0
  17. package/CHANGELOG.md +1182 -0
  18. package/CLAUDE.md +633 -0
  19. package/Dockerfile +3 -0
  20. package/LICENSE +21 -0
  21. package/MISTAKES-COVERAGE.json +115 -0
  22. package/Makefile +95 -0
  23. package/PHASE-F-ROADMAP.md +554 -0
  24. package/README.md +459 -0
  25. package/SELF_HOSTING_CHECKLIST.md +155 -0
  26. package/benchmark-freelang.fl +47 -0
  27. package/benchmark-results.json +37 -0
  28. package/benchmark-simple.fl +38 -0
  29. package/bin/freelang-migrate +310 -0
  30. package/bin/freelang-smart +258 -0
  31. package/bootstrap.js +1134 -0
  32. package/docker-compose.yml +3 -0
  33. package/docs/AI_FRIENDLINESS.md +181 -0
  34. package/docs/AI_LEARNING_PATH.md +542 -0
  35. package/docs/AI_MAINTENANCE.md +153 -0
  36. package/docs/AI_QUICKSTART.md +380 -0
  37. package/docs/AI_REFERENCE.md +559 -0
  38. package/docs/AI_RELIABILITY_GUIDE.md +597 -0
  39. package/docs/AKL_ARCHITECTURE.md +228 -0
  40. package/docs/AKL_P0_KIMDB.md +215 -0
  41. package/docs/API.md +380 -0
  42. package/docs/ARCHITECTURE.md +208 -0
  43. package/docs/CHANGELOG.md +1014 -0
  44. package/docs/CLAUDE.md +991 -0
  45. package/docs/CLAUDE_AI.md +210 -0
  46. package/docs/CODEGEN_IMPROVEMENTS.md +158 -0
  47. package/docs/CODE_OF_CONDUCT.md +337 -0
  48. package/docs/CONTRIBUTING.md +437 -0
  49. package/docs/CRON_AUTOMATION.md +97 -0
  50. package/docs/DEPLOYMENT.md +357 -0
  51. package/docs/DETERMINISM_GUIDE.md +174 -0
  52. package/docs/HUMAN_GUIDE.md +599 -0
  53. package/docs/INDEX.md +199 -0
  54. package/docs/INLINE_TEST_GUIDE.md +231 -0
  55. package/docs/LANGUAGE-FAULTS.md +583 -0
  56. package/docs/LANGUAGE_IMPROVEMENT_REQUESTS.md +187 -0
  57. package/docs/LEARNING.md +257 -0
  58. package/docs/MISTAKES-100.md +52 -0
  59. package/docs/MISTAKES.md +131 -0
  60. package/docs/NAMING_CONVENTIONS.md +158 -0
  61. package/docs/NIGHTLY-MONITORING.md +222 -0
  62. package/docs/OFFICIAL_LANGUAGE.md +249 -0
  63. package/docs/ONBOARDING_1HOUR.md +210 -0
  64. package/docs/PERFORMANCE.md +329 -0
  65. package/docs/PHASE-3C-L2-PROOF-PLAN.md +282 -0
  66. package/docs/PHASE-3C-L2-RESULTS.md +183 -0
  67. package/docs/PHASE-3D-AI-LIBRARY.md +170 -0
  68. package/docs/PHASE-3E-VM-OPTIMIZATION.md +192 -0
  69. package/docs/PHASE-C-CI.md +242 -0
  70. package/docs/PHASE-C-PR-CHECKLIST.md +267 -0
  71. package/docs/PHASE-C-ROADMAP.md +141 -0
  72. package/docs/PHASE-C-SCAN.md +205 -0
  73. package/docs/PHASE-G-ROADMAP.md +228 -0
  74. package/docs/PHASE-X-V11.5-ROADMAP.md +267 -0
  75. package/docs/PHASE4_IMPLEMENTATION_PLAN.md +290 -0
  76. package/docs/PHASE5-COMPLETION.md +186 -0
  77. package/docs/PHASE5-SUMMARY.md +82 -0
  78. package/docs/PLUGIN_GUIDE.md +241 -0
  79. package/docs/PROJECT_OVERVIEW_KO.md +66 -0
  80. package/docs/QUICKSTART.md +160 -0
  81. package/docs/README.md +192 -0
  82. package/docs/RESERVED.md +125 -0
  83. package/docs/ROADMAP.md +191 -0
  84. package/docs/SECURITY.md +518 -0
  85. package/docs/SETUP.md +127 -0
  86. package/docs/SLACK-SETUP.md +190 -0
  87. package/docs/STATE_OF_V11.md +74 -0
  88. package/docs/STDLIB_NAMING_AUDIT.md +154 -0
  89. package/docs/STDLIB_REFERENCE.md +2722 -0
  90. package/docs/STYLE_GUIDE.md +559 -0
  91. package/docs/TASK5-MAP-DESTRUCTURE-DESIGN.md +136 -0
  92. package/docs/TOOLS.md +357 -0
  93. package/docs/TYPE_SYSTEM_GUIDE.md +201 -0
  94. package/docs/UI-PATTERN.md +265 -0
  95. package/docs/V11.5-RULES.md +416 -0
  96. package/docs/V11.6-HARNESS-RULES.md +222 -0
  97. package/docs/V11.6-MULTI-AGENT-PLAN.md +265 -0
  98. package/docs/V11.6-STABILIZATION-PLAN.md +290 -0
  99. package/docs/Y4-2B_DRY_RUN.md +142 -0
  100. package/docs/_classifications.json +882 -0
  101. package/docs/api/stdlib.md +134 -0
  102. package/docs/blog/2026-04-17-v11-ai-first-evolution.md +223 -0
  103. package/docs/blog/2026-04-20-self-hosting-fixedpoint.md +269 -0
  104. package/docs/blog/2026-05-04-mistakes-are-language-faults.md +267 -0
  105. package/docs/blog/2026-05-04-mistakes-coverage-honest.md +170 -0
  106. package/docs/blog/2026-05-04-phase-g-followup-smart-wrapper.md +188 -0
  107. package/docs/blog/2026-05-04-phase-g-mistakes-100.md +269 -0
  108. package/docs/blog/2026-05-04-phase-x-v11.5-launch.md +224 -0
  109. package/docs/blog/2026-05-04-v11.5.0-real-language-change.md +252 -0
  110. package/docs/blog-demo/README.md +44 -0
  111. package/docs/examples/index.md +78 -0
  112. package/docs/examples/todo-app.md +99 -0
  113. package/docs/guide/ai-blocks.md +335 -0
  114. package/docs/guide/basics.md +235 -0
  115. package/docs/guide/frameworks.md +319 -0
  116. package/docs/homepage/README.md +164 -0
  117. package/docs/index.md +183 -0
  118. package/docs/v12-DESIGN.md +214 -0
  119. package/examples/README.md +26 -0
  120. package/examples/browser-test.html +149 -0
  121. package/examples/csv-tool/analyze.fl +222 -0
  122. package/examples/csv-tool/employees.csv +16 -0
  123. package/examples/csv-tool/products.csv +21 -0
  124. package/examples/factorial.fl +9 -0
  125. package/examples/fib30.fl +4 -0
  126. package/examples/hello.fl +2 -0
  127. package/examples/hello.fl.js +3 -0
  128. package/examples/learning-session/01-grade-stats.fl +49 -0
  129. package/examples/learning-session/02-class-stats.fl +52 -0
  130. package/examples/learning-session/03-cart.fl +61 -0
  131. package/examples/learning-session/04-word-freq.fl +57 -0
  132. package/examples/learning-session/05-todo.fl +69 -0
  133. package/examples/learning-session/06-library.fl +76 -0
  134. package/examples/learning-session/07-currency.fl +99 -0
  135. package/examples/learning-session/08-csv-grades.fl +104 -0
  136. package/examples/learning-session/09-calc-repl.fl +92 -0
  137. package/examples/learning-session/10-contacts.fl +109 -0
  138. package/examples/learning-session/11-markdown.fl +107 -0
  139. package/examples/learning-session/12-calendar.fl +87 -0
  140. package/examples/learning-session/13-statistics.fl +123 -0
  141. package/examples/learning-session/14-primes.fl +107 -0
  142. package/examples/learning-session/15-statistics.fl +123 -0
  143. package/examples/learning-session/16-calendar.fl +87 -0
  144. package/examples/learning-session/17-markdown.fl +107 -0
  145. package/examples/learning-session/18-contacts.fl +109 -0
  146. package/examples/learning-session/19-json-serializer.fl +86 -0
  147. package/examples/learning-session/20-template.fl +201 -0
  148. package/examples/learning-session/21-calc-parser.fl +148 -0
  149. package/examples/learning-session/22-table.fl +96 -0
  150. package/examples/patterns/01-map-filter-reduce.fl +50 -0
  151. package/examples/patterns/02-error-handling.fl +51 -0
  152. package/examples/patterns/03-type-validation.fl +57 -0
  153. package/examples/patterns/04-state-management.fl +44 -0
  154. package/examples/patterns/05-api-integration.fl +95 -0
  155. package/examples/patterns/06-database-operations.fl +113 -0
  156. package/examples/patterns/07-string-processing.fl +86 -0
  157. package/examples/patterns/08-data-transformation.fl +92 -0
  158. package/examples/patterns/09-decision-logic.fl +122 -0
  159. package/examples/patterns/10-agent-orchestration.fl +184 -0
  160. package/examples/patterns/README.md +344 -0
  161. package/examples/simple-server.fl +29 -0
  162. package/examples/tables-app.fl +219 -0
  163. package/examples/test-libs.fl +272 -0
  164. package/jest.config.js +4 -0
  165. package/lib/datetime.fl +123 -0
  166. package/lib/http-client.fl +129 -0
  167. package/lib/pipeline.fl +150 -0
  168. package/lib/query.fl +160 -0
  169. package/lib/result.fl +124 -0
  170. package/lib/template.fl +161 -0
  171. package/lib/validate.fl +136 -0
  172. package/package.json +58 -0
  173. package/scripts/ai-eval.js +318 -0
  174. package/scripts/ai-self-verify.js +200 -0
  175. package/scripts/ai-validate.js +302 -0
  176. package/scripts/bench.sh +71 -0
  177. package/scripts/benchmark.js +258 -0
  178. package/scripts/benchmark.sh +99 -0
  179. package/scripts/build-binary.sh +68 -0
  180. package/scripts/build-runtime.sh +46 -0
  181. package/scripts/build-stage1-final.sh +58 -0
  182. package/scripts/build-stage1.sh +119 -0
  183. package/scripts/build-standalone.sh +17 -0
  184. package/scripts/build.js +130 -0
  185. package/scripts/check-let-regressions.js +95 -0
  186. package/scripts/check-parens.py +117 -0
  187. package/scripts/check-ports.sh +49 -0
  188. package/scripts/check-syntax.js +30 -0
  189. package/scripts/cli-extras.js +135 -0
  190. package/scripts/cron-daily-verify.sh +101 -0
  191. package/scripts/deploy.sh +91 -0
  192. package/scripts/dev.sh +37 -0
  193. package/scripts/extract-fl-examples.js +29 -0
  194. package/scripts/extract-runtime-helpers.js +122 -0
  195. package/scripts/fl-compile.sh +14 -0
  196. package/scripts/fl-fixpoint.sh +28 -0
  197. package/scripts/fl-repl.sh +21 -0
  198. package/scripts/fl-run.sh +41 -0
  199. package/scripts/fl-serve.sh +25 -0
  200. package/scripts/fuzz-compiler.js +92 -0
  201. package/scripts/gen-ai-prompt.js +340 -0
  202. package/scripts/gen-fixtures.js +166 -0
  203. package/scripts/gen-mistakes-split.js +199 -0
  204. package/scripts/gen-stdlib-docs.js +69 -0
  205. package/scripts/lint-stdlib-aliases.js +166 -0
  206. package/scripts/measure-baseline.sh +43 -0
  207. package/scripts/property-test.js +885 -0
  208. package/scripts/push-to-gogs.sh +60 -0
  209. package/scripts/regression-scan.js +132 -0
  210. package/scripts/run-jest-shard.sh +85 -0
  211. package/scripts/safe-push.sh +37 -0
  212. package/scripts/scan-fl-tokens.js +170 -0
  213. package/scripts/scan-for-fl-tokens.sh +17 -0
  214. package/scripts/self-diff.sh +95 -0
  215. package/scripts/snapshot-v0.sh +35 -0
  216. package/scripts/test-l2-fixpoint.sh +54 -0
  217. package/scripts/verify-all.sh +130 -0
  218. package/scripts/verify-build-deterministic.sh +110 -0
  219. package/scripts/verify-c7-bootstrap.sh +150 -0
  220. package/scripts/verify-c8-fixed-point-simple.sh +121 -0
  221. package/scripts/verify-c8-fixed-point.sh +133 -0
  222. package/scripts/verify-c9-fuzzing.sh +214 -0
  223. package/scripts/verify-fixed-point-deep.sh +133 -0
  224. package/scripts/verify-fixed-point.sh +65 -0
  225. package/scripts/verify-l2-proof.sh +153 -0
  226. package/scripts/verify-l3-proof.sh +40 -0
  227. package/scripts/verify-mistakes-coverage.js +120 -0
  228. package/scripts/verify-self-host.sh +509 -0
  229. package/self/CHANGELOG.md +964 -0
  230. package/self/COMPILER_REDESIGN_FAILURES.md +59 -0
  231. package/self/INTEGRATION-CHECKLIST.md +113 -0
  232. package/self/README.md +34 -0
  233. package/self/all.fl +1353 -0
  234. package/self/ast.fl +219 -0
  235. package/self/ast.self.js +47 -0
  236. package/self/bench/_bi_json-parse.js +46 -0
  237. package/self/bench/_bi_json-str.js +60 -0
  238. package/self/bench/_bi_list-distinct.js +37 -0
  239. package/self/bench/_bi_list-filter-even.js +56 -0
  240. package/self/bench/_bi_list-first.js +29 -0
  241. package/self/bench/_bi_list-last.js +28 -0
  242. package/self/bench/_bi_list-map-inc.js +41 -0
  243. package/self/bench/_bi_list-range.js +22 -0
  244. package/self/bench/_bi_list-reduce-sum.js +50 -0
  245. package/self/bench/_bi_list-reverse.js +28 -0
  246. package/self/bench/_bi_list-sort.js +35 -0
  247. package/self/bench/_bi_list-take.js +36 -0
  248. package/self/bench/_bi_m-abs.js +24 -0
  249. package/self/bench/_bi_m-floor.js +22 -0
  250. package/self/bench/_bi_m-max.js +26 -0
  251. package/self/bench/_bi_m-min.js +26 -0
  252. package/self/bench/_bi_m-mod.js +21 -0
  253. package/self/bench/_bi_map-has.js +50 -0
  254. package/self/bench/_bi_map-keys.js +50 -0
  255. package/self/bench/_bi_map-values.js +52 -0
  256. package/self/bench/_bi_str-concat.js +36 -0
  257. package/self/bench/_bi_str-contains.js +43 -0
  258. package/self/bench/_bi_str-ends.js +45 -0
  259. package/self/bench/_bi_str-index.js +34 -0
  260. package/self/bench/_bi_str-lower.js +23 -0
  261. package/self/bench/_bi_str-repeat.js +26 -0
  262. package/self/bench/_bi_str-split.js +41 -0
  263. package/self/bench/_bi_str-starts.js +47 -0
  264. package/self/bench/_bi_str-trim.js +26 -0
  265. package/self/bench/_bi_str-upper.js +23 -0
  266. package/self/bench/_bi_t-list.js +24 -0
  267. package/self/bench/_bi_t-null.js +23 -0
  268. package/self/bench/_bi_t-number.js +23 -0
  269. package/self/bench/_bi_t-string.js +25 -0
  270. package/self/bench/_bi_t-typeof.js +25 -0
  271. package/self/bench/_c_parse_probe.fl +426 -0
  272. package/self/bench/_c_parse_probe.js +67 -0
  273. package/self/bench/_codegen_nodrv.fl +751 -0
  274. package/self/bench/_combined2.fl +423 -0
  275. package/self/bench/_combined_arr.fl +422 -0
  276. package/self/bench/_combined_astprobe.fl +429 -0
  277. package/self/bench/_combined_astprobe.js +69 -0
  278. package/self/bench/_combined_cg.fl +424 -0
  279. package/self/bench/_combined_cg.js +64 -0
  280. package/self/bench/_combined_lex.fl +421 -0
  281. package/self/bench/_combined_trace.fl +435 -0
  282. package/self/bench/_combined_trace.js +75 -0
  283. package/self/bench/_ext_abs-neg.js +2 -0
  284. package/self/bench/_ext_and.js +2 -0
  285. package/self/bench/_ext_cond-1.js +2 -0
  286. package/self/bench/_ext_cond-2.js +2 -0
  287. package/self/bench/_ext_do-seq.js +2 -0
  288. package/self/bench/_ext_fact20.js +3 -0
  289. package/self/bench/_ext_fib25.js +3 -0
  290. package/self/bench/_ext_length.js +2 -0
  291. package/self/bench/_ext_let-1d.js +18 -0
  292. package/self/bench/_ext_let-2d.js +2 -0
  293. package/self/bench/_ext_list-sum.js +3 -0
  294. package/self/bench/_ext_math.js +2 -0
  295. package/self/bench/_ext_or.js +2 -0
  296. package/self/bench/_ext_str.js +2 -0
  297. package/self/bench/_fact15.js +2 -0
  298. package/self/bench/_fb_fact15.js +2 -0
  299. package/self/bench/_fb_func-fact.js +2 -0
  300. package/self/bench/_fb_func-multi.js +2 -0
  301. package/self/bench/_fb_func-mutual.js +3 -0
  302. package/self/bench/_fb_func-simple.js +21 -0
  303. package/self/bench/_fib.js +2 -0
  304. package/self/bench/_fn_async-await-seq.js +40 -0
  305. package/self/bench/_fn_async-await.js +3 -0
  306. package/self/bench/_fn_async-print.js +26 -0
  307. package/self/bench/_fn_async-result.js +1 -0
  308. package/self/bench/_fn_call-form.js +24 -0
  309. package/self/bench/_fn_compose.js +37 -0
  310. package/self/bench/_fn_pipe.js +34 -0
  311. package/self/bench/_fn_thread-first-sexpr.js +31 -0
  312. package/self/bench/_fn_thread-first.js +25 -0
  313. package/self/bench/_fn_thread-last.js +31 -0
  314. package/self/bench/_generated.js +35 -0
  315. package/self/bench/_mt_match-default.js +51 -0
  316. package/self/bench/_mt_match-num.js +58 -0
  317. package/self/bench/_mt_match-str.js +56 -0
  318. package/self/bench/_mt_match-var.js +32 -0
  319. package/self/bench/_mt_match-wild.js +36 -0
  320. package/self/bench/_mt_struct-make.js +81 -0
  321. package/self/bench/_mt_struct-tag.js +81 -0
  322. package/self/bench/_probe.fl +1 -0
  323. package/self/bench/_probe2.fl +1 -0
  324. package/self/bench/_probe3.fl +3 -0
  325. package/self/bench/_probe_apply.fl +1 -0
  326. package/self/bench/_probe_args.fl +7 -0
  327. package/self/bench/_probe_arr.fl +7 -0
  328. package/self/bench/_probe_arr2.fl +5 -0
  329. package/self/bench/_probe_block.fl +7 -0
  330. package/self/bench/_probe_call.fl +1 -0
  331. package/self/bench/_probe_cg.fl +27 -0
  332. package/self/bench/_probe_compose.fl +13 -0
  333. package/self/bench/_probe_define.fl +1 -0
  334. package/self/bench/_probe_keys.fl +5 -0
  335. package/self/bench/_probe_length.fl +5 -0
  336. package/self/bench/_probe_lexparse.fl +4 -0
  337. package/self/bench/_probe_loop.fl +12 -0
  338. package/self/bench/_probe_map.fl +8 -0
  339. package/self/bench/_probe_map2.fl +8 -0
  340. package/self/bench/_probe_map3.fl +5 -0
  341. package/self/bench/_probe_mapkeys.fl +5 -0
  342. package/self/bench/_probe_mapv12.fl +1 -0
  343. package/self/bench/_probe_mapv12.js +2 -0
  344. package/self/bench/_probe_match.fl +1 -0
  345. package/self/bench/_probe_match2.fl +1 -0
  346. package/self/bench/_probe_neg.fl +7 -0
  347. package/self/bench/_probe_neg2.fl +3 -0
  348. package/self/bench/_probe_protocol.fl +7 -0
  349. package/self/bench/_probe_self_parse.fl +6 -0
  350. package/self/bench/_probe_struct.fl +1 -0
  351. package/self/bench/_probe_true.fl +3 -0
  352. package/self/bench/_probe_true2.fl +6 -0
  353. package/self/bench/_probe_v12_cg.fl +7 -0
  354. package/self/bench/_probe_v12_extract.fl +8 -0
  355. package/self/bench/_probe_v12_extract.js +7 -0
  356. package/self/bench/_probe_v12_internal.fl +21 -0
  357. package/self/bench/_probe_v12_map_ast.fl +12 -0
  358. package/self/bench/_rs_combo.js +3 -0
  359. package/self/bench/_rs_gcd.js +36 -0
  360. package/self/bench/_rs_grade.js +2 -0
  361. package/self/bench/_rs_mutual-fib.js +2 -0
  362. package/self/bench/_rs_range-sum.js +2 -0
  363. package/self/bench/_rs_sum-10k.js +2 -0
  364. package/self/bench/_rs_sum-5k.js +2 -0
  365. package/self/bench/_sf_loop-fact-10.js +73 -0
  366. package/self/bench/_sf_loop-sum-10k.js +87 -0
  367. package/self/bench/_sf_set-x.js +40 -0
  368. package/self/bench/_sf_throw-catch.js +1 -0
  369. package/self/bench/_sf_while-5.js +62 -0
  370. package/self/bench/_v12_combined.fl +1184 -0
  371. package/self/bench/_v12_extract_test.fl +14 -0
  372. package/self/bench/_v12_extract_test.js +7 -0
  373. package/self/bench/_v12_inspect.js +26 -0
  374. package/self/bench/_v12_inspect_mod.js +137 -0
  375. package/self/bench/_v12_makevartest.fl +8 -0
  376. package/self/bench/_v12_makevartest.js +6 -0
  377. package/self/bench/_v12_parse_probe.fl +9 -0
  378. package/self/bench/_v12_parse_trace.fl +18 -0
  379. package/self/bench/_v12_simple_fn.fl +2 -0
  380. package/self/bench/_v12_simple_fn.js +3 -0
  381. package/self/bench/_v12_trace.fl +2 -0
  382. package/self/bench/_v12_trace.js +3 -0
  383. package/self/bench/collection.js +13 -0
  384. package/self/bench/fib30.fl +4 -0
  385. package/self/bench/final-test.fl +12 -0
  386. package/self/bench/final-test.js +2 -0
  387. package/self/bench/fp1-v12.js +4 -0
  388. package/self/bench/fp1.fl +3 -0
  389. package/self/bench/fp1.fp.js +4 -0
  390. package/self/bench/fp2-v12.js +4 -0
  391. package/self/bench/fp2.fl +3 -0
  392. package/self/bench/fp2.fp.js +4 -0
  393. package/self/bench/fp3-v12.js +8 -0
  394. package/self/bench/fp3.fl +11 -0
  395. package/self/bench/fp3.fp.js +8 -0
  396. package/self/bench/fp4-v12.js +6 -0
  397. package/self/bench/fp4.fl +9 -0
  398. package/self/bench/fp4.fp.js +6 -0
  399. package/self/bench/hello-sample.fl +3 -0
  400. package/self/bench/hello-sample.js +4 -0
  401. package/self/bench/hello.fl +1 -0
  402. package/self/bench/json.js +9 -0
  403. package/self/bench/json1mb.fl +3 -0
  404. package/self/bench/math.js +8 -0
  405. package/self/bench/realworld.fl +17 -0
  406. package/self/bench/realworld.js +8 -0
  407. package/self/bench/regex.js +10 -0
  408. package/self/bench/sample.json +1 -0
  409. package/self/bench/string.js +7 -0
  410. package/self/bench/test-ffi.fl +3 -0
  411. package/self/bench/test-ffi.js +2 -0
  412. package/self/bench/test-hof-advanced.fl +7 -0
  413. package/self/bench/test-hof-advanced.js +2 -0
  414. package/self/bench/test-hof.fl +5 -0
  415. package/self/bench/test-hof.js +2 -0
  416. package/self/bench/test-output.txt +1 -0
  417. package/self/bench/test-phase15-integration.fl +16 -0
  418. package/self/bench/test-phase15-integration.js +3 -0
  419. package/self/bench/test-phase19-complete.fl +24 -0
  420. package/self/bench/test-phase19-complete.js +2 -0
  421. package/self/bench/test-stdlib-codegen.fl +9 -0
  422. package/self/bench/test-stdlib-codegen.js +2 -0
  423. package/self/bench/test-time.fl +8 -0
  424. package/self/bench/test-time.js +2 -0
  425. package/self/bench/time.js +9 -0
  426. package/self/bench/tiny-v12.js +2 -0
  427. package/self/bench/tiny.fl +1 -0
  428. package/self/builtins/core.fl +150 -0
  429. package/self/char-class.fl +45 -0
  430. package/self/codegen-core.fl +733 -0
  431. package/self/codegen-final.fl +135 -0
  432. package/self/codegen.fl +379 -0
  433. package/self/codegen.fl.bak +856 -0
  434. package/self/codegen.self.js +75 -0
  435. package/self/examples/agent-loop.fl +32 -0
  436. package/self/examples/ci-cd-example.fl +114 -0
  437. package/self/examples/cloud-real.fl +134 -0
  438. package/self/examples/docker-example.fl +67 -0
  439. package/self/examples/fullstack-app.fl +330 -0
  440. package/self/examples/k8s-example.fl +91 -0
  441. package/self/examples/microservices.fl +448 -0
  442. package/self/examples/rest-api.fl +253 -0
  443. package/self/examples/style-example.fl +133 -0
  444. package/self/examples/workflow-orchestration.fl +44 -0
  445. package/self/fixtures/eval/add.fl +1 -0
  446. package/self/fixtures/eval/and-sc.fl +1 -0
  447. package/self/fixtures/eval/bool-false.fl +1 -0
  448. package/self/fixtures/eval/bool-true.fl +1 -0
  449. package/self/fixtures/eval/closure.fl +1 -0
  450. package/self/fixtures/eval/cmp-eq.fl +1 -0
  451. package/self/fixtures/eval/cmp-lt.fl +1 -0
  452. package/self/fixtures/eval/cond-first.fl +1 -0
  453. package/self/fixtures/eval/define-fn3.fl +1 -0
  454. package/self/fixtures/eval/define-var.fl +1 -0
  455. package/self/fixtures/eval/defn-bare.fl +1 -0
  456. package/self/fixtures/eval/defn-call.fl +1 -0
  457. package/self/fixtures/eval/div.fl +1 -0
  458. package/self/fixtures/eval/filter-hof.fl +1 -0
  459. package/self/fixtures/eval/floor.fl +1 -0
  460. package/self/fixtures/eval/fn-apply.fl +1 -0
  461. package/self/fixtures/eval/get-idx.fl +1 -0
  462. package/self/fixtures/eval/get-key.fl +1 -0
  463. package/self/fixtures/eval/higher-order.fl +1 -0
  464. package/self/fixtures/eval/let-bare.fl +1 -0
  465. package/self/fixtures/eval/let-flat.fl +1 -0
  466. package/self/fixtures/eval/let-use.fl +1 -0
  467. package/self/fixtures/eval/list-first.fl +1 -0
  468. package/self/fixtures/eval/list-last.fl +1 -0
  469. package/self/fixtures/eval/list-len.fl +1 -0
  470. package/self/fixtures/eval/map-hof.fl +1 -0
  471. package/self/fixtures/eval/mod.fl +1 -0
  472. package/self/fixtures/eval/mul-nested.fl +1 -0
  473. package/self/fixtures/eval/not.fl +1 -0
  474. package/self/fixtures/eval/null-p.fl +1 -0
  475. package/self/fixtures/eval/or-sc.fl +1 -0
  476. package/self/fixtures/eval/pow.fl +1 -0
  477. package/self/fixtures/eval/recursion.fl +1 -0
  478. package/self/fixtures/eval/reduce-hof.fl +1 -0
  479. package/self/fixtures/eval/replace.fl +1 -0
  480. package/self/fixtures/eval/sqrt.fl +1 -0
  481. package/self/fixtures/eval/str-concat.fl +1 -0
  482. package/self/fixtures/eval/str-num.fl +1 -0
  483. package/self/fixtures/eval/sub.fl +1 -0
  484. package/self/fixtures/eval/substring.fl +1 -0
  485. package/self/fixtures/lex/array.fl +1 -0
  486. package/self/fixtures/lex/at-atom.fl +1 -0
  487. package/self/fixtures/lex/block.fl +1 -0
  488. package/self/fixtures/lex/comment.fl +2 -0
  489. package/self/fixtures/lex/deep-array.fl +1 -0
  490. package/self/fixtures/lex/deep-map.fl +1 -0
  491. package/self/fixtures/lex/deep-nest.fl +1 -0
  492. package/self/fixtures/lex/dollar-dot.fl +1 -0
  493. package/self/fixtures/lex/empty-list.fl +1 -0
  494. package/self/fixtures/lex/empty-str.fl +1 -0
  495. package/self/fixtures/lex/empty.fl +1 -0
  496. package/self/fixtures/lex/float.fl +1 -0
  497. package/self/fixtures/lex/hex.fl +1 -0
  498. package/self/fixtures/lex/kebab-sym.fl +1 -0
  499. package/self/fixtures/lex/keyword.fl +1 -0
  500. package/self/fixtures/lex/leading-zero.fl +1 -0
  501. package/self/fixtures/lex/long-num.fl +1 -0
  502. package/self/fixtures/lex/many-args.fl +1 -0
  503. package/self/fixtures/lex/map.fl +1 -0
  504. package/self/fixtures/lex/mixed.fl +1 -0
  505. package/self/fixtures/lex/neg-float.fl +1 -0
  506. package/self/fixtures/lex/negative.fl +1 -0
  507. package/self/fixtures/lex/nested.fl +1 -0
  508. package/self/fixtures/lex/null.fl +1 -0
  509. package/self/fixtures/lex/pipe-op.fl +1 -0
  510. package/self/fixtures/lex/route-like.fl +1 -0
  511. package/self/fixtures/lex/scientific.fl +1 -0
  512. package/self/fixtures/lex/single-num.fl +1 -0
  513. package/self/fixtures/lex/small-sexpr.fl +1 -0
  514. package/self/fixtures/lex/spaces.fl +1 -0
  515. package/self/fixtures/lex/str-quote.fl +1 -0
  516. package/self/fixtures/lex/str-tab.fl +1 -0
  517. package/self/fixtures/lex/string-esc.fl +1 -0
  518. package/self/fixtures/lex/string.fl +1 -0
  519. package/self/fixtures/lex/symbol.fl +1 -0
  520. package/self/fixtures/lex/tabs-nl.fl +3 -0
  521. package/self/fixtures/lex/true-false.fl +1 -0
  522. package/self/fixtures/lex/unicode.fl +1 -0
  523. package/self/fixtures/lex/variable.fl +1 -0
  524. package/self/fixtures/lex/zero.fl +1 -0
  525. package/self/fixtures/parse/and-or.fl +1 -0
  526. package/self/fixtures/parse/array-lit.fl +1 -0
  527. package/self/fixtures/parse/block-func.fl +1 -0
  528. package/self/fixtures/parse/compose.fl +1 -0
  529. package/self/fixtures/parse/cond.fl +1 -0
  530. package/self/fixtures/parse/define-fn.fl +1 -0
  531. package/self/fixtures/parse/define-val.fl +1 -0
  532. package/self/fixtures/parse/defn.fl +1 -0
  533. package/self/fixtures/parse/do-begin.fl +1 -0
  534. package/self/fixtures/parse/dotted.fl +1 -0
  535. package/self/fixtures/parse/empty-fn.fl +1 -0
  536. package/self/fixtures/parse/empty-list.fl +1 -0
  537. package/self/fixtures/parse/fn.fl +1 -0
  538. package/self/fixtures/parse/if-else.fl +1 -0
  539. package/self/fixtures/parse/if-only.fl +1 -0
  540. package/self/fixtures/parse/keyword-arg.fl +1 -0
  541. package/self/fixtures/parse/kw.fl +1 -0
  542. package/self/fixtures/parse/let-1d.fl +1 -0
  543. package/self/fixtures/parse/let-2d.fl +1 -0
  544. package/self/fixtures/parse/let-bare.fl +1 -0
  545. package/self/fixtures/parse/literal-num.fl +1 -0
  546. package/self/fixtures/parse/literal-str.fl +1 -0
  547. package/self/fixtures/parse/loop-recur.fl +1 -0
  548. package/self/fixtures/parse/map-lit.fl +1 -0
  549. package/self/fixtures/parse/match-simple.fl +1 -0
  550. package/self/fixtures/parse/multi-body.fl +1 -0
  551. package/self/fixtures/parse/nested-arr.fl +1 -0
  552. package/self/fixtures/parse/nested-block.fl +1 -0
  553. package/self/fixtures/parse/nested-map.fl +1 -0
  554. package/self/fixtures/parse/nested.fl +1 -0
  555. package/self/fixtures/parse/pipe.fl +1 -0
  556. package/self/fixtures/parse/quote.fl +1 -0
  557. package/self/fixtures/parse/set-bang.fl +1 -0
  558. package/self/fixtures/parse/sexpr.fl +1 -0
  559. package/self/fixtures/parse/str-interp.fl +1 -0
  560. package/self/fixtures/parse/sym.fl +1 -0
  561. package/self/fixtures/parse/thread-first.fl +1 -0
  562. package/self/fixtures/parse/thread-last.fl +1 -0
  563. package/self/fixtures/parse/try-catch.fl +1 -0
  564. package/self/fixtures/parse/var.fl +1 -0
  565. package/self/full-compiler-fixed.fl +1308 -0
  566. package/self/full-compiler-v1-bloated.fl +1298 -0
  567. package/self/full-compiler.fl +1301 -0
  568. package/self/interpreter.fl +351 -0
  569. package/self/lexer.fl +200 -0
  570. package/self/lexer.self.js +30 -0
  571. package/self/main.fl +22 -0
  572. package/self/parser.fl +241 -0
  573. package/self/parser.self.js +33 -0
  574. package/self/runtime/http-server.js +642 -0
  575. package/self/runtime/interpreter.js +30787 -0
  576. package/self/runtime/repl.js +186 -0
  577. package/self/runtime-helpers.ts +282 -0
  578. package/self/scope.fl +80 -0
  579. package/self/scope.self.js +1 -0
  580. package/self/self/bench/_bi_list-distinct.js +2 -0
  581. package/self/self/bench/_bi_list-filter-even.js +2 -0
  582. package/self/self/bench/_bi_list-first.js +2 -0
  583. package/self/self/bench/_bi_list-last.js +2 -0
  584. package/self/self/bench/_bi_list-map-inc.js +2 -0
  585. package/self/self/bench/_bi_list-range.js +2 -0
  586. package/self/self/bench/_bi_list-reduce-sum.js +2 -0
  587. package/self/self/bench/_bi_list-reverse.js +2 -0
  588. package/self/self/bench/_bi_list-sort.js +2 -0
  589. package/self/self/bench/_bi_list-take.js +2 -0
  590. package/self/self/bench/_bi_map-has.js +2 -0
  591. package/self/self/bench/_bi_map-keys.js +2 -0
  592. package/self/self/bench/_bi_map-values.js +2 -0
  593. package/self/self/bench/_bi_str-concat.js +2 -0
  594. package/self/self/bench/_bi_str-contains.js +2 -0
  595. package/self/self/bench/_bi_str-ends.js +2 -0
  596. package/self/self/bench/_bi_str-index.js +2 -0
  597. package/self/self/bench/_bi_str-lower.js +2 -0
  598. package/self/self/bench/_bi_str-repeat.js +2 -0
  599. package/self/self/bench/_bi_str-split.js +2 -0
  600. package/self/self/bench/_bi_str-starts.js +2 -0
  601. package/self/self/bench/_bi_str-trim.js +2 -0
  602. package/self/self/bench/_bi_str-upper.js +2 -0
  603. package/self/self/bench/_ext_abs-neg.js +2 -0
  604. package/self/self/bench/_ext_and.js +2 -0
  605. package/self/self/bench/_ext_cond-1.js +2 -0
  606. package/self/self/bench/_ext_cond-2.js +2 -0
  607. package/self/self/bench/_ext_do-seq.js +2 -0
  608. package/self/self/bench/_ext_fact20.js +3 -0
  609. package/self/self/bench/_ext_fib25.js +3 -0
  610. package/self/self/bench/_ext_length.js +2 -0
  611. package/self/self/bench/_ext_let-1d.js +2 -0
  612. package/self/self/bench/_ext_let-2d.js +2 -0
  613. package/self/self/bench/_ext_list-sum.js +3 -0
  614. package/self/self/bench/_ext_math.js +2 -0
  615. package/self/self/bench/_ext_or.js +2 -0
  616. package/self/self/bench/_ext_str.js +2 -0
  617. package/self/self/bench/_fact15.js +2 -0
  618. package/self/self/bench/_fb_fact15.js +2 -0
  619. package/self/self/bench/_fb_func-fact.js +2 -0
  620. package/self/self/bench/_fb_func-multi.js +2 -0
  621. package/self/self/bench/_fb_func-mutual.js +3 -0
  622. package/self/self/bench/_fb_func-simple.js +2 -0
  623. package/self/self/bench/_fib.js +2 -0
  624. package/self/self/bench/_fn_async-await-seq.js +2 -0
  625. package/self/self/bench/_fn_async-print.js +2 -0
  626. package/self/self/bench/_fn_call-form.js +2 -0
  627. package/self/self/bench/_fn_compose.js +2 -0
  628. package/self/self/bench/_fn_pipe.js +2 -0
  629. package/self/self/bench/_fn_thread-first-sexpr.js +2 -0
  630. package/self/self/bench/_fn_thread-first.js +2 -0
  631. package/self/self/bench/_fn_thread-last.js +2 -0
  632. package/self/self/bench/_generated.js +2 -0
  633. package/self/self/bench/_mt_match-default.js +2 -0
  634. package/self/self/bench/_mt_match-num.js +2 -0
  635. package/self/self/bench/_mt_match-str.js +2 -0
  636. package/self/self/bench/_mt_match-var.js +2 -0
  637. package/self/self/bench/_mt_match-wild.js +2 -0
  638. package/self/self/bench/_mt_struct-make.js +4 -0
  639. package/self/self/bench/_mt_struct-tag.js +4 -0
  640. package/self/self/bench/_rs_combo.js +3 -0
  641. package/self/self/bench/_rs_gcd.js +2 -0
  642. package/self/self/bench/_rs_grade.js +2 -0
  643. package/self/self/bench/_rs_mutual-fib.js +2 -0
  644. package/self/self/bench/_rs_range-sum.js +2 -0
  645. package/self/self/bench/_rs_sum-5k.js +2 -0
  646. package/self/self/bench/_sf_loop-fact-10.js +1 -0
  647. package/self/self/bench/_sf_loop-sum-10k.js +1 -0
  648. package/self/self/bench/_sf_set-x.js +3 -0
  649. package/self/self/bench/_sf_throw-catch.js +1 -0
  650. package/self/self/bench/_sf_while-5.js +3 -0
  651. package/self/stdlib/ai.fl +73 -0
  652. package/self/stdlib/assert.fl +30 -0
  653. package/self/stdlib/async.fl +48 -0
  654. package/self/stdlib/base64.fl +22 -0
  655. package/self/stdlib/binary.fl +30 -0
  656. package/self/stdlib/build.fl +69 -0
  657. package/self/stdlib/collection.fl +37 -0
  658. package/self/stdlib/color.fl +35 -0
  659. package/self/stdlib/crypto-hash.fl +29 -0
  660. package/self/stdlib/crypto.fl +56 -0
  661. package/self/stdlib/data.fl +69 -0
  662. package/self/stdlib/db.fl +53 -0
  663. package/self/stdlib/encoding.fl +49 -0
  664. package/self/stdlib/error.fl +36 -0
  665. package/self/stdlib/feed.fl +86 -0
  666. package/self/stdlib/file.fl +53 -0
  667. package/self/stdlib/format.fl +34 -0
  668. package/self/stdlib/graph.fl +72 -0
  669. package/self/stdlib/hash.fl +32 -0
  670. package/self/stdlib/heap.fl +76 -0
  671. package/self/stdlib/http.fl +47 -0
  672. package/self/stdlib/image.fl +45 -0
  673. package/self/stdlib/json.fl +11 -0
  674. package/self/stdlib/list-extra.fl +32 -0
  675. package/self/stdlib/markdown.fl +59 -0
  676. package/self/stdlib/math-extra.fl +34 -0
  677. package/self/stdlib/math.fl +13 -0
  678. package/self/stdlib/metadata.fl +66 -0
  679. package/self/stdlib/mongodb/bson.fl +21 -0
  680. package/self/stdlib/mongodb/schema.fl +95 -0
  681. package/self/stdlib/mongodb/wire.fl +54 -0
  682. package/self/stdlib/mongodb.fl +172 -0
  683. package/self/stdlib/path.fl +72 -0
  684. package/self/stdlib/plot.fl +56 -0
  685. package/self/stdlib/process.fl +50 -0
  686. package/self/stdlib/queue.fl +65 -0
  687. package/self/stdlib/regex.fl +11 -0
  688. package/self/stdlib/resource.fl +61 -0
  689. package/self/stdlib/search.fl +54 -0
  690. package/self/stdlib/set-extra.fl +35 -0
  691. package/self/stdlib/sort.fl +61 -0
  692. package/self/stdlib/stack.fl +54 -0
  693. package/self/stdlib/stats.fl +110 -0
  694. package/self/stdlib/stream.fl +56 -0
  695. package/self/stdlib/string-extended.fl +25 -0
  696. package/self/stdlib/string-extra.fl +40 -0
  697. package/self/stdlib/string.fl +21 -0
  698. package/self/stdlib/test-helpers.fl +51 -0
  699. package/self/stdlib/test-runner.fl +76 -0
  700. package/self/stdlib/time-extra.fl +27 -0
  701. package/self/stdlib/time.fl +31 -0
  702. package/self/stdlib/tree.fl +57 -0
  703. package/self/stdlib/types.fl +85 -0
  704. package/self/stdlib/url.fl +33 -0
  705. package/self/stdlib/uuid.fl +28 -0
  706. package/self/stdlib/validation.fl +42 -0
  707. package/self/stdlib/vector-math.fl +52 -0
  708. package/self/stdlib/ws.fl +52 -0
  709. package/self/tests/mongodb-phase3.fl +75 -0
  710. package/self/tests/mongodb-wire-phase2.fl +45 -0
  711. package/self/tests/test-ast.fl +36 -0
  712. package/self/tests/test-builtins.fl +340 -0
  713. package/self/tests/test-char-class.fl +29 -0
  714. package/self/tests/test-codegen-builtins.fl +240 -0
  715. package/self/tests/test-codegen-ext.fl +380 -0
  716. package/self/tests/test-codegen-ffi.fl +295 -0
  717. package/self/tests/test-codegen-fn.fl +177 -0
  718. package/self/tests/test-codegen-match.fl +161 -0
  719. package/self/tests/test-codegen-run.fl +252 -0
  720. package/self/tests/test-codegen-sf.fl +262 -0
  721. package/self/tests/test-codegen.fl +260 -0
  722. package/self/tests/test-core-fl.fl +63 -0
  723. package/self/tests/test-forward-decl.fl +5 -0
  724. package/self/tests/test-func-block.fl +4 -0
  725. package/self/tests/test-interp-user-fn.fl +296 -0
  726. package/self/tests/test-interp.fl +225 -0
  727. package/self/tests/test-lexer.fl +162 -0
  728. package/self/tests/test-list-debug.fl +21 -0
  729. package/self/tests/test-mutual-rec.fl +6 -0
  730. package/self/tests/test-parser-debug.fl +35 -0
  731. package/self/tests/test-parser-full-debug.fl +66 -0
  732. package/self/tests/test-parser-lex-debug.fl +107 -0
  733. package/self/tests/test-parser-lex-only.fl +40 -0
  734. package/self/tests/test-parser-simple-debug.fl +35 -0
  735. package/self/tests/test-parser.fl +262 -0
  736. package/self/tests/test-real-stdlib.fl +377 -0
  737. package/self/tests/test-scope.fl +74 -0
  738. package/self/tests/test-selfcompile.fl +325 -0
  739. package/self/tests/test-stdlib.fl +44 -0
  740. package/self/tests/test-tco.fl +16 -0
  741. package/self/tests/test-token.fl +4 -0
  742. package/self/token.fl +38 -0
  743. package/self/token.self.js +8 -0
  744. package/self/v12-driver.fl +16 -0
  745. package/self/verify.fl +39 -0
  746. package/self-evolve/1-observe-ast.fl +28 -0
  747. package/self-evolve/2-inspect-block.fl +35 -0
  748. package/self-evolve/3-full-ast-json.fl +10 -0
  749. package/self-evolve/4-trace-block-fields.fl +23 -0
  750. package/self-evolve/5-analyze-do-ast.fl +29 -0
  751. package/self-evolve/6-test-do-elim-simple.fl +14 -0
  752. package/self-evolve/debug-evaluate.fl +24 -0
  753. package/self-evolve/debug-gen1-const-fold.fl +57 -0
  754. package/self-evolve/debug-map.fl +15 -0
  755. package/self-evolve/debug-optimize-simple.fl +95 -0
  756. package/self-evolve/debug-recursive-map.fl +23 -0
  757. package/self-evolve/logs/gen-0.fl +1658 -0
  758. package/self-evolve/logs/gen-1.fl +1658 -0
  759. package/self-evolve/logs/gen-2.fl +1658 -0
  760. package/self-evolve/logs/speed-gen-0.fl +1423 -0
  761. package/self-evolve/run-semantic-test.js +130 -0
  762. package/self-evolve/test-basic.fl +19 -0
  763. package/self-evolve/test-fold-direct.fl +60 -0
  764. package/self-evolve/test-gen1-simple.fl +67 -0
  765. package/self-evolve/test-optimize-debug.fl +63 -0
  766. package/self-evolve/tmp-test.fl +1 -0
  767. package/self-evolve/v11-analyzer.fl +40 -0
  768. package/self-evolve/v11-benchmark.fl +45 -0
  769. package/self-evolve/v11-evolution-real.fl +45 -0
  770. package/self-evolve/v11-evolution.fl +32 -0
  771. package/self-evolve/v11-gen1-validate.fl +157 -0
  772. package/self-evolve/v11-optimizer-constant-fold.fl +124 -0
  773. package/self-evolve/v11-optimizer-dead-expr.fl +111 -0
  774. package/self-evolve/v11-optimizer-gen1-complete.fl +180 -0
  775. package/self-evolve/v11-optimizer-gen1.fl +68 -0
  776. package/self-evolve/v11-optimizer.fl +56 -0
  777. package/self-evolve/v11-semantic-test.fl +54 -0
  778. package/self-evolve/v11-speed-benchmark.fl +106 -0
  779. package/self-evolve/v11-speed-optimizer.fl +79 -0
  780. package/src/AGENT-DSL.md +527 -0
  781. package/src/CLI-DEPLOYMENT.md +204 -0
  782. package/src/CLI.md +361 -0
  783. package/src/EXPRESS-ADVANCED.md +294 -0
  784. package/src/EXPRESS-AUTH.md +365 -0
  785. package/src/EXPRESS-CACHE.md +409 -0
  786. package/src/EXPRESS-COMPLETE.md +569 -0
  787. package/src/EXPRESS-README.md +121 -0
  788. package/src/EXPRESS-TEST.md +492 -0
  789. package/src/EXPRESS-WEBSOCKET.md +391 -0
  790. package/src/LOGGING-GUIDE.md +354 -0
  791. package/src/STORAGE-GUIDE.md +416 -0
  792. package/src/__tests__/advanced.test.ts +573 -0
  793. package/src/__tests__/ai-library.test.ts +210 -0
  794. package/src/__tests__/build-determinism.test.ts +33 -0
  795. package/src/__tests__/builtins-advanced.test.ts +150 -0
  796. package/src/__tests__/builtins-sanity.test.ts +180 -0
  797. package/src/__tests__/codegen.let.test.ts +81 -0
  798. package/src/__tests__/core.test.ts +188 -0
  799. package/src/__tests__/coverage-boost.test.ts +626 -0
  800. package/src/__tests__/cron-scheduler.test.ts +269 -0
  801. package/src/__tests__/enterprise-blocks.test.ts +63 -0
  802. package/src/__tests__/errors.test.ts +313 -0
  803. package/src/__tests__/integration.test.ts +219 -0
  804. package/src/__tests__/interpreter.test.ts +170 -0
  805. package/src/__tests__/l2-proof.test.ts +78 -0
  806. package/src/__tests__/lexer-parser.test.ts +366 -0
  807. package/src/__tests__/lexer.test.ts +106 -0
  808. package/src/__tests__/mariadb-prepared-statement.test.ts +206 -0
  809. package/src/__tests__/migrate.test.ts +403 -0
  810. package/src/__tests__/mongodb-integration.test.ts.skip +207 -0
  811. package/src/__tests__/mongodb-phase4.test.ts +132 -0
  812. package/src/__tests__/p1-1-parallel-tasks.test.ts +179 -0
  813. package/src/__tests__/p1-2-compensation.test.ts +242 -0
  814. package/src/__tests__/p1-3-distributed.test.ts +160 -0
  815. package/src/__tests__/p1-4-observability.test.ts +161 -0
  816. package/src/__tests__/parser.test.ts +142 -0
  817. package/src/__tests__/phase151-self-evolve.test.ts +161 -0
  818. package/src/__tests__/rate-limiter.test.ts +274 -0
  819. package/src/__tests__/self-hosting.test.ts +174 -0
  820. package/src/__tests__/semantic-preservation.test.ts +207 -0
  821. package/src/__tests__/setup.ts +34 -0
  822. package/src/__tests__/special-forms-advanced.test.ts +106 -0
  823. package/src/__tests__/stdlib-crypto-rsa.test.ts +122 -0
  824. package/src/__tests__/stdlib-f4.test.ts +159 -0
  825. package/src/__tests__/stdlib-helpers.test.ts +128 -0
  826. package/src/__tests__/stdlib-modules.test.ts +161 -0
  827. package/src/__tests__/stdlib-mongodb.test.ts +331 -0
  828. package/src/__tests__/stdlib-new.test.ts +269 -0
  829. package/src/__tests__/stdlib-perf.test.ts +125 -0
  830. package/src/__tests__/stdlib-phase-b.test.ts +249 -0
  831. package/src/__tests__/stdlib.test.ts +163 -0
  832. package/src/__tests__/v12-alpha.test.ts +205 -0
  833. package/src/__tests__/vm-optin.test.ts +141 -0
  834. package/src/__tests__/web-integration.test.ts +1452 -0
  835. package/src/__tests__/weblibs.test.ts +210 -0
  836. package/src/_aliases.json +1123 -0
  837. package/src/_mongodb_helper.js +272 -0
  838. package/src/_stdlib-signatures.json +1 -0
  839. package/src/agent-chain.ts +71 -0
  840. package/src/agent-dsl.fl +168 -0
  841. package/src/agent-example-error-handling.fl +51 -0
  842. package/src/agent-example-sequential.fl +66 -0
  843. package/src/agent-example-state-tracking.fl +70 -0
  844. package/src/agent.ts +393 -0
  845. package/src/align.ts +349 -0
  846. package/src/analogy.ts +78 -0
  847. package/src/ast-helpers.ts +199 -0
  848. package/src/ast.ts +817 -0
  849. package/src/async-runtime.ts +263 -0
  850. package/src/belief.ts +86 -0
  851. package/src/benchmark-self.ts +275 -0
  852. package/src/benchmarks/bench-interpreter.ts +73 -0
  853. package/src/benchmarks/bench-runner.ts +85 -0
  854. package/src/benchmarks/bench-vm.ts +128 -0
  855. package/src/browser-debug-panel.ts +220 -0
  856. package/src/browser-entry.ts +79 -0
  857. package/src/browser-stubs/child-process-stubs.ts +6 -0
  858. package/src/browser-stubs/crypto-stubs.ts +15 -0
  859. package/src/browser-stubs/misc-stubs.ts +4 -0
  860. package/src/browser-stubs/node-stubs.ts +17 -0
  861. package/src/browser-stubs/path-stubs.ts +6 -0
  862. package/src/bytecode.ts +41 -0
  863. package/src/causal.ts +258 -0
  864. package/src/chain-agents.ts +86 -0
  865. package/src/checkpoint.ts +106 -0
  866. package/src/ci-runner.ts +279 -0
  867. package/src/cli.ts +2136 -0
  868. package/src/codegen-js.ts +847 -0
  869. package/src/cognitive.ts +95 -0
  870. package/src/compete.ts +60 -0
  871. package/src/compiler.ts +267 -0
  872. package/src/compose-reason.ts +118 -0
  873. package/src/consensus.ts +109 -0
  874. package/src/context-window.ts +139 -0
  875. package/src/cot.ts +241 -0
  876. package/src/counterfactual.ts +268 -0
  877. package/src/critique.ts +77 -0
  878. package/src/crossover.ts +218 -0
  879. package/src/curiosity.ts +305 -0
  880. package/src/debate.ts +64 -0
  881. package/src/debug-api.ts +92 -0
  882. package/src/debugger.ts +185 -0
  883. package/src/delegate.ts +78 -0
  884. package/src/doc-extractor.ts +255 -0
  885. package/src/doc-renderer.ts +131 -0
  886. package/src/echo-server-demo.fl +17 -0
  887. package/src/error-formatter.ts +369 -0
  888. package/src/error-system.ts +88 -0
  889. package/src/errors.ts +281 -0
  890. package/src/ethics-check.ts +408 -0
  891. package/src/eval-ai-blocks.ts +182 -0
  892. package/src/eval-ai-handlers.ts +132 -0
  893. package/src/eval-builtins-ai.ts +1659 -0
  894. package/src/eval-builtins.ts +5956 -0
  895. package/src/eval-call-function.ts +665 -0
  896. package/src/eval-infra-blocks.ts +450 -0
  897. package/src/eval-module-system.ts +286 -0
  898. package/src/eval-pattern-match.ts +289 -0
  899. package/src/eval-phase150.ts +87 -0
  900. package/src/eval-reasoning-sequence.ts +218 -0
  901. package/src/eval-special-forms.ts +2154 -0
  902. package/src/eval-style-blocks.ts +193 -0
  903. package/src/eval-type-classes.ts +148 -0
  904. package/src/evolve.ts +276 -0
  905. package/src/explain.ts +345 -0
  906. package/src/express-advanced.fl +78 -0
  907. package/src/express-auth.fl +118 -0
  908. package/src/express-cache.fl +118 -0
  909. package/src/express-chat.fl +81 -0
  910. package/src/express-example.fl +67 -0
  911. package/src/express-test.fl +331 -0
  912. package/src/express.fl +92 -0
  913. package/src/fitness.ts +238 -0
  914. package/src/fl-app-demo.fl +19 -0
  915. package/src/fl-files/fl-fmt.fl +29 -0
  916. package/src/fl-files/fl-lint.fl +61 -0
  917. package/src/fl-files/fl-test.fl +52 -0
  918. package/src/fl-http-demo.fl +19 -0
  919. package/src/fl-list-utils.fl +108 -0
  920. package/src/fl-math-lib.fl +14 -0
  921. package/src/fl-sdk.ts +154 -0
  922. package/src/fl-server-demo.fl +74 -0
  923. package/src/fl-str-utils.fl +29 -0
  924. package/src/fl-tutor.ts +126 -0
  925. package/src/formatter.ts +581 -0
  926. package/src/freelang-codegen.fl +904 -0
  927. package/src/freelang-interpreter.fl +483 -0
  928. package/src/freelang-lexer.fl +337 -0
  929. package/src/freelang-parser.fl +349 -0
  930. package/src/freelang-stdlib.fl +246 -0
  931. package/src/freelang-typechecker.fl +422 -0
  932. package/src/freelang-v9-complete.ts +474 -0
  933. package/src/generation.ts +201 -0
  934. package/src/gpt-mini-p3.fl +316 -0
  935. package/src/hot-reload.ts +235 -0
  936. package/src/http-server-runner.ts +89 -0
  937. package/src/hypothesis.ts +62 -0
  938. package/src/immutable.ts +140 -0
  939. package/src/interpreter-context.ts +87 -0
  940. package/src/interpreter-scope.ts +140 -0
  941. package/src/interpreter.ts +2351 -0
  942. package/src/lazy-seq.ts +134 -0
  943. package/src/learned-facts-store.ts +306 -0
  944. package/src/lexer.ts +359 -0
  945. package/src/lint-rules.ts +584 -0
  946. package/src/linter.ts +237 -0
  947. package/src/logger.ts +128 -0
  948. package/src/logging-deterministic.fl +138 -0
  949. package/src/lsp-server.ts +379 -0
  950. package/src/macro-expander.ts +195 -0
  951. package/src/maybe-chain.ts +108 -0
  952. package/src/maybe-type.ts +163 -0
  953. package/src/memory-system.ts +142 -0
  954. package/src/meta-reason.ts +93 -0
  955. package/src/multi-agent-hub.ts +247 -0
  956. package/src/multi-agent.ts +101 -0
  957. package/src/mutate.ts +226 -0
  958. package/src/negotiate.ts +55 -0
  959. package/src/optimizer.ts +277 -0
  960. package/src/orchestrate.ts +154 -0
  961. package/src/package-manager.ts +375 -0
  962. package/src/parser.ts +2722 -0
  963. package/src/peer-review.ts +53 -0
  964. package/src/predict.ts +365 -0
  965. package/src/profiler.ts +150 -0
  966. package/src/prompt-compiler.ts +123 -0
  967. package/src/protocol.ts +114 -0
  968. package/src/prune.ts +195 -0
  969. package/src/quality-loop.ts +105 -0
  970. package/src/rag.ts +81 -0
  971. package/src/reasoning-debugger.ts +122 -0
  972. package/src/refactor-self.ts +362 -0
  973. package/src/reflect.ts +186 -0
  974. package/src/repl.ts +323 -0
  975. package/src/result-type.ts +126 -0
  976. package/src/return-signal.ts +10 -0
  977. package/src/runtime-entry.ts +8 -0
  978. package/src/runtime-helpers.ts +234 -0
  979. package/src/self-evolution-hub.ts +431 -0
  980. package/src/self-improve.ts +107 -0
  981. package/src/source-map.ts +114 -0
  982. package/src/stdlib-agent.js +164 -0
  983. package/src/stdlib-agent.ts +225 -0
  984. package/src/stdlib-ai-native.ts +176 -0
  985. package/src/stdlib-ai-workflow.ts +308 -0
  986. package/src/stdlib-ai.ts +180 -0
  987. package/src/stdlib-async.ts +179 -0
  988. package/src/stdlib-audit.ts +94 -0
  989. package/src/stdlib-auth.ts +196 -0
  990. package/src/stdlib-bits.ts +86 -0
  991. package/src/stdlib-blog.ts +127 -0
  992. package/src/stdlib-browser.ts +239 -0
  993. package/src/stdlib-cache.ts +147 -0
  994. package/src/stdlib-capture-error.ts +183 -0
  995. package/src/stdlib-channel.ts +96 -0
  996. package/src/stdlib-checkpoint.js +109 -0
  997. package/src/stdlib-checkpoint.ts +97 -0
  998. package/src/stdlib-cloud.ts +317 -0
  999. package/src/stdlib-collection.ts +227 -0
  1000. package/src/stdlib-compile.ts +111 -0
  1001. package/src/stdlib-cron.ts +219 -0
  1002. package/src/stdlib-crypto-rsa.ts +82 -0
  1003. package/src/stdlib-crypto.js +203 -0
  1004. package/src/stdlib-crypto.ts +208 -0
  1005. package/src/stdlib-data.ts +614 -0
  1006. package/src/stdlib-db-query.ts +198 -0
  1007. package/src/stdlib-db.ts +185 -0
  1008. package/src/stdlib-distributed.ts +292 -0
  1009. package/src/stdlib-error.ts +90 -0
  1010. package/src/stdlib-fd.ts +130 -0
  1011. package/src/stdlib-feed.ts +171 -0
  1012. package/src/stdlib-file.ts +182 -0
  1013. package/src/stdlib-helpers.ts +273 -0
  1014. package/src/stdlib-http-macro.ts +178 -0
  1015. package/src/stdlib-http-server.ts +1229 -0
  1016. package/src/stdlib-http.ts +405 -0
  1017. package/src/stdlib-image.ts +92 -0
  1018. package/src/stdlib-kebab-aliases.ts +131 -0
  1019. package/src/stdlib-lazy-registry.ts +106 -0
  1020. package/src/stdlib-loader.ts +810 -0
  1021. package/src/stdlib-mail.ts +251 -0
  1022. package/src/stdlib-mariadb.ts +467 -0
  1023. package/src/stdlib-markdown.ts +227 -0
  1024. package/src/stdlib-matrix.ts +170 -0
  1025. package/src/stdlib-middleware.ts +221 -0
  1026. package/src/stdlib-module.ts +178 -0
  1027. package/src/stdlib-mongodb.ts +174 -0
  1028. package/src/stdlib-oci.ts +321 -0
  1029. package/src/stdlib-optional.ts +56 -0
  1030. package/src/stdlib-orm.ts +241 -0
  1031. package/src/stdlib-perf.ts +140 -0
  1032. package/src/stdlib-pg.ts +181 -0
  1033. package/src/stdlib-plot.ts +196 -0
  1034. package/src/stdlib-process.ts +120 -0
  1035. package/src/stdlib-property.ts +157 -0
  1036. package/src/stdlib-pubsub.ts +93 -0
  1037. package/src/stdlib-queue-helpers.ts +92 -0
  1038. package/src/stdlib-registry.ts +78 -0
  1039. package/src/stdlib-resource.ts +553 -0
  1040. package/src/stdlib-rest-crud.ts +146 -0
  1041. package/src/stdlib-service.ts +206 -0
  1042. package/src/stdlib-shell.ts +76 -0
  1043. package/src/stdlib-stats.ts +172 -0
  1044. package/src/stdlib-table.ts +200 -0
  1045. package/src/stdlib-test-enhanced.ts +76 -0
  1046. package/src/stdlib-test.ts +153 -0
  1047. package/src/stdlib-time.js +217 -0
  1048. package/src/stdlib-time.ts +282 -0
  1049. package/src/stdlib-timer.ts +134 -0
  1050. package/src/stdlib-totp.ts +110 -0
  1051. package/src/stdlib-type-predicates.ts +136 -0
  1052. package/src/stdlib-types.ts +107 -0
  1053. package/src/stdlib-validation.ts +248 -0
  1054. package/src/stdlib-verify.ts +181 -0
  1055. package/src/stdlib-webauthn.ts +192 -0
  1056. package/src/stdlib-workflow.js +715 -0
  1057. package/src/stdlib-workflow.ts +950 -0
  1058. package/src/stdlib-ws.ts +333 -0
  1059. package/src/stdlib-wsc.test.ts +122 -0
  1060. package/src/stdlib-wsc.ts +243 -0
  1061. package/src/storage-unified.fl +279 -0
  1062. package/src/streaming.ts +101 -0
  1063. package/src/struct-system.ts +104 -0
  1064. package/src/style-registry.ts +54 -0
  1065. package/src/swarm.ts +89 -0
  1066. package/src/tco.ts +31 -0
  1067. package/src/test-advanced-patterns.ts +211 -0
  1068. package/src/test-ast-debug.ts +20 -0
  1069. package/src/test-ast-helpers.ts +208 -0
  1070. package/src/test-async.ts +406 -0
  1071. package/src/test-bootstrap-self-compile.ts +449 -0
  1072. package/src/test-bootstrap-verification.ts +336 -0
  1073. package/src/test-composition.ts +206 -0
  1074. package/src/test-errors-phase6.ts +166 -0
  1075. package/src/test-extended-monads.ts +313 -0
  1076. package/src/test-field-parsing.ts +135 -0
  1077. package/src/test-first-class-functions.ts +257 -0
  1078. package/src/test-freelang-interpreter.ts +320 -0
  1079. package/src/test-freelang-lexer.ts +306 -0
  1080. package/src/test-freelang-parser.ts +268 -0
  1081. package/src/test-fullstack-core.ts +258 -0
  1082. package/src/test-fullstack-phase7-12.ts +305 -0
  1083. package/src/test-fullstack-practical.ts +338 -0
  1084. package/src/test-integration-stdlib.ts +195 -0
  1085. package/src/test-interpreter-phase6.ts +305 -0
  1086. package/src/test-lexer-comparison.ts +108 -0
  1087. package/src/test-lexer-phase6.ts +271 -0
  1088. package/src/test-modules.ts +325 -0
  1089. package/src/test-monad-laws.ts +383 -0
  1090. package/src/test-monads.ts +197 -0
  1091. package/src/test-p0-checkpoint.ts +304 -0
  1092. package/src/test-p0-conditional.ts +284 -0
  1093. package/src/test-p0-error-handling.ts +231 -0
  1094. package/src/test-p0-error-messages.ts +220 -0
  1095. package/src/test-p1-1-parallel-tasks.js +214 -0
  1096. package/src/test-parser-phase6.ts +222 -0
  1097. package/src/test-performance.ts +206 -0
  1098. package/src/test-phase10-data.ts +259 -0
  1099. package/src/test-phase10-file.ts +216 -0
  1100. package/src/test-phase100-stdlib-ai.ts +343 -0
  1101. package/src/test-phase101-memory.ts +309 -0
  1102. package/src/test-phase102-rag.ts +296 -0
  1103. package/src/test-phase103-multi-agent.ts +418 -0
  1104. package/src/test-phase104-try-reason.ts +459 -0
  1105. package/src/test-phase105-streaming.ts +287 -0
  1106. package/src/test-phase106-quality.ts +397 -0
  1107. package/src/test-phase107-tutor.ts +305 -0
  1108. package/src/test-phase108-debugger.ts +316 -0
  1109. package/src/test-phase109-prompt-compiler.ts +333 -0
  1110. package/src/test-phase11-12-complete.ts +275 -0
  1111. package/src/test-phase11-error.ts +192 -0
  1112. package/src/test-phase110-sdk.ts +320 -0
  1113. package/src/test-phase111-hypothesis.ts +380 -0
  1114. package/src/test-phase112-maybe-chain.ts +313 -0
  1115. package/src/test-phase113-debate.ts +364 -0
  1116. package/src/test-phase114-checkpoint.ts +348 -0
  1117. package/src/test-phase115-meta-reason.ts +277 -0
  1118. package/src/test-phase116-belief.ts +275 -0
  1119. package/src/test-phase117-analogy.ts +325 -0
  1120. package/src/test-phase118-critique.ts +308 -0
  1121. package/src/test-phase119-compose.ts +434 -0
  1122. package/src/test-phase12-http-shell.ts +120 -0
  1123. package/src/test-phase120-cognitive.ts +297 -0
  1124. package/src/test-phase121-consensus.ts +404 -0
  1125. package/src/test-phase122-delegate.ts +411 -0
  1126. package/src/test-phase123-vote.ts +339 -0
  1127. package/src/test-phase124-negotiate.ts +403 -0
  1128. package/src/test-phase125-swarm.ts +321 -0
  1129. package/src/test-phase126-orchestrate.ts +343 -0
  1130. package/src/test-phase127-peer-review.ts +279 -0
  1131. package/src/test-phase128-chain-agents.ts +456 -0
  1132. package/src/test-phase129-compete.ts +256 -0
  1133. package/src/test-phase13-data.ts +223 -0
  1134. package/src/test-phase130-hub.ts +390 -0
  1135. package/src/test-phase131-evolve.ts +536 -0
  1136. package/src/test-phase132-mutate.ts +268 -0
  1137. package/src/test-phase133-crossover.ts +289 -0
  1138. package/src/test-phase134-fitness.ts +306 -0
  1139. package/src/test-phase135-generation.ts +328 -0
  1140. package/src/test-phase136-prune.ts +228 -0
  1141. package/src/test-phase137-refactor-self.ts +354 -0
  1142. package/src/test-phase138-benchmark-self.ts +325 -0
  1143. package/src/test-phase139-version-self.ts +278 -0
  1144. package/src/test-phase14-collection.ts +254 -0
  1145. package/src/test-phase140-self-evolution.ts +410 -0
  1146. package/src/test-phase141-world-model.ts +387 -0
  1147. package/src/test-phase142-causal.ts +384 -0
  1148. package/src/test-phase143-counterfactual.ts +280 -0
  1149. package/src/test-phase144-predict.ts +312 -0
  1150. package/src/test-phase145-explain.ts +287 -0
  1151. package/src/test-phase146-align.ts +439 -0
  1152. package/src/test-phase147-ethics-check.ts +399 -0
  1153. package/src/test-phase148-curiosity.ts +247 -0
  1154. package/src/test-phase149-wisdom.ts +758 -0
  1155. package/src/test-phase15-agent.ts +320 -0
  1156. package/src/test-phase150-complete.ts +481 -0
  1157. package/src/test-phase16-time.ts +292 -0
  1158. package/src/test-phase17-crypto.ts +312 -0
  1159. package/src/test-phase18-integration.ts +429 -0
  1160. package/src/test-phase19-resource.ts +214 -0
  1161. package/src/test-phase20-server-db.ts +160 -0
  1162. package/src/test-phase21-ws-auth-cache-pubsub.ts +212 -0
  1163. package/src/test-phase22-process.ts +166 -0
  1164. package/src/test-phase23-selfhosting.ts +383 -0
  1165. package/src/test-phase24-codegen.ts +236 -0
  1166. package/src/test-phase25-bootstrap.ts +227 -0
  1167. package/src/test-phase26-map-filter-reduce.ts +282 -0
  1168. package/src/test-phase27-stdlib-codegen.ts +325 -0
  1169. package/src/test-phase28-loop-recur.ts +182 -0
  1170. package/src/test-phase29-import.ts +165 -0
  1171. package/src/test-phase3-web-server.ts +234 -0
  1172. package/src/test-phase30-selfcompile.ts +254 -0
  1173. package/src/test-phase31-gen2-lexer.ts +251 -0
  1174. package/src/test-phase33-gen3-bootstrap.ts +323 -0
  1175. package/src/test-phase34-parser-tco.ts +268 -0
  1176. package/src/test-phase35-error-messages.ts +280 -0
  1177. package/src/test-phase36-sourcemap.ts +240 -0
  1178. package/src/test-phase37-variadic.ts +228 -0
  1179. package/src/test-phase38-destructuring.ts +261 -0
  1180. package/src/test-phase39-do-block.ts +288 -0
  1181. package/src/test-phase40-const-fold.ts +249 -0
  1182. package/src/test-phase41-repl.ts +312 -0
  1183. package/src/test-phase42-watch.ts +314 -0
  1184. package/src/test-phase43-format.ts +377 -0
  1185. package/src/test-phase44-check.ts +505 -0
  1186. package/src/test-phase45-interpreter.ts +367 -0
  1187. package/src/test-phase46-match.ts +390 -0
  1188. package/src/test-phase47-file-io.ts +338 -0
  1189. package/src/test-phase48-types.ts +308 -0
  1190. package/src/test-phase49-stdlib.ts +365 -0
  1191. package/src/test-phase5-week1.ts +160 -0
  1192. package/src/test-phase50-fl-server.ts +159 -0
  1193. package/src/test-phase51-transformer.ts +152 -0
  1194. package/src/test-phase52-import.ts +161 -0
  1195. package/src/test-phase53-training.ts +122 -0
  1196. package/src/test-phase54-fl-utils.ts +122 -0
  1197. package/src/test-phase55-http-client.ts +182 -0
  1198. package/src/test-phase56-lexical-scope.ts +230 -0
  1199. package/src/test-phase58-module-refactor.ts +298 -0
  1200. package/src/test-phase59-errors.ts +226 -0
  1201. package/src/test-phase6-compile.ts +227 -0
  1202. package/src/test-phase60-types.ts +457 -0
  1203. package/src/test-phase61-tco.ts +209 -0
  1204. package/src/test-phase63-macros.ts +191 -0
  1205. package/src/test-phase64-protocols.ts +451 -0
  1206. package/src/test-phase65-patterns.ts +301 -0
  1207. package/src/test-phase66-structs.ts +215 -0
  1208. package/src/test-phase67-concurrency.ts +348 -0
  1209. package/src/test-phase68-pipeline.ts +209 -0
  1210. package/src/test-phase69-lazy.ts +237 -0
  1211. package/src/test-phase7-registry.ts +236 -0
  1212. package/src/test-phase70-immutable.ts +488 -0
  1213. package/src/test-phase71-ai-native.ts +252 -0
  1214. package/src/test-phase72-integration.ts +533 -0
  1215. package/src/test-phase73-formatter.ts +361 -0
  1216. package/src/test-phase74-linter.ts +565 -0
  1217. package/src/test-phase75-repl.ts +227 -0
  1218. package/src/test-phase76-testrunner.ts +439 -0
  1219. package/src/test-phase77-doc.ts +416 -0
  1220. package/src/test-phase78-debugger.ts +370 -0
  1221. package/src/test-phase79-watch.ts +224 -0
  1222. package/src/test-phase8-oci.ts +264 -0
  1223. package/src/test-phase80-ci.ts +282 -0
  1224. package/src/test-phase81-pkg.ts +383 -0
  1225. package/src/test-phase82-profiler.ts +336 -0
  1226. package/src/test-phase83-vm.ts +318 -0
  1227. package/src/test-phase84-optimizer.ts +424 -0
  1228. package/src/test-phase85-codegen.ts +380 -0
  1229. package/src/test-phase86-lsp.ts +533 -0
  1230. package/src/test-phase87-packages.ts +277 -0
  1231. package/src/test-phase88-selfhost.ts +412 -0
  1232. package/src/test-phase89-bench.ts +361 -0
  1233. package/src/test-phase9-flnext-v2.ts +372 -0
  1234. package/src/test-phase9-learn.ts +241 -0
  1235. package/src/test-phase9-reasoning.ts +312 -0
  1236. package/src/test-phase9-search.ts +212 -0
  1237. package/src/test-phase90-release.ts +365 -0
  1238. package/src/test-phase91-maybe.ts +257 -0
  1239. package/src/test-phase92-cot.ts +438 -0
  1240. package/src/test-phase93-tot.ts +462 -0
  1241. package/src/test-phase94-reflect.ts +498 -0
  1242. package/src/test-phase95-context.ts +268 -0
  1243. package/src/test-phase96-errors.ts +296 -0
  1244. package/src/test-phase97-tools.ts +344 -0
  1245. package/src/test-phase98-agent.ts +370 -0
  1246. package/src/test-phase99-self-improve.ts +394 -0
  1247. package/src/test-phase9a-websearch.ts +283 -0
  1248. package/src/test-phase9ab-integration.ts +140 -0
  1249. package/src/test-phase9b-persistence.ts +448 -0
  1250. package/src/test-phase9c-conditional.ts +251 -0
  1251. package/src/test-phase9c-extension.ts +270 -0
  1252. package/src/test-phase9c-feedback.ts +263 -0
  1253. package/src/test-phase9c-loop.ts +239 -0
  1254. package/src/test-selfhosting-sh1.ts +41 -0
  1255. package/src/test-selfhosting-sh2.ts +84 -0
  1256. package/src/test-selfhosting-sh3.ts +61 -0
  1257. package/src/test-selfhosting-sh4.ts +65 -0
  1258. package/src/test-type-classes-dispatch.ts +202 -0
  1259. package/src/test-type-classes.ts +320 -0
  1260. package/src/test-type-inference.ts +464 -0
  1261. package/src/test-typeclass-parsing-final.ts +218 -0
  1262. package/src/test-typeclass-parsing.ts +264 -0
  1263. package/src/todo-server-30116.ts +53 -0
  1264. package/src/token.ts +130 -0
  1265. package/src/tool-registry.ts +195 -0
  1266. package/src/tot.ts +258 -0
  1267. package/src/try-reason.ts +109 -0
  1268. package/src/type-check-static.ts +417 -0
  1269. package/src/type-checker.ts +245 -0
  1270. package/src/type-inference.ts +271 -0
  1271. package/src/type-system.ts +169 -0
  1272. package/src/version-self.ts +219 -0
  1273. package/src/vm-eligible.ts +106 -0
  1274. package/src/vm.ts +219 -0
  1275. package/src/vote.ts +127 -0
  1276. package/src/vpm/checksum-test.fl +193 -0
  1277. package/src/vpm/checksum.fl +219 -0
  1278. package/src/vpm/error-test.fl +211 -0
  1279. package/src/vpm/error.fl +218 -0
  1280. package/src/vpm/lock-file.fl +380 -0
  1281. package/src/vpm/logging-test.fl +194 -0
  1282. package/src/vpm/logging.fl +294 -0
  1283. package/src/vpm/resolver-test.fl +117 -0
  1284. package/src/vpm/resolver.fl +618 -0
  1285. package/src/vpm/semver-test.fl +180 -0
  1286. package/src/vpm/semver.fl +143 -0
  1287. package/src/vpm-cli.ts +1955 -0
  1288. package/src/web/app-router.ts +286 -0
  1289. package/src/web/fl-executor.ts +655 -0
  1290. package/src/web/image-optimizer.ts +206 -0
  1291. package/src/web/index.ts +14 -0
  1292. package/src/web/page-renderer.ts +287 -0
  1293. package/src/web/server.ts +553 -0
  1294. package/src/web-search-adapter.ts +348 -0
  1295. package/src/wisdom.ts +441 -0
  1296. package/src/world-model.ts +348 -0
  1297. package/stdlib/cache.fl +42 -0
  1298. package/stdlib/csv.fl +38 -0
  1299. package/stdlib/date-ext.fl +68 -0
  1300. package/stdlib/log.fl +44 -0
  1301. package/stdlib/math-ext.fl +57 -0
  1302. package/stdlib/number-ext.fl +47 -0
  1303. package/stdlib/queue.fl +57 -0
  1304. package/stdlib/string-ext.fl +45 -0
  1305. package/stdlib/validate.fl +123 -0
  1306. package/stdlib/web/components.fl +125 -0
  1307. package/stdlib/web/csrf.fl +61 -0
  1308. package/stdlib/web/format.fl +75 -0
  1309. package/stdlib/web/forms.fl +94 -0
  1310. package/stdlib/web/image.fl +77 -0
  1311. package/stdlib/web/metadata.fl +85 -0
  1312. package/stdlib/web/pagination.fl +62 -0
  1313. package/stdlib/web/state.fl +167 -0
  1314. package/stdlib/web/styles.fl +68 -0
  1315. package/stdlib/web/toast.fl +62 -0
  1316. package/stdlib/web/v9-stdlib-dom.fl +92 -0
  1317. package/stdlib/web/v9-stdlib-fetch.fl +90 -0
  1318. package/stdlib/web/v9-stdlib-storage.fl +70 -0
  1319. package/stdlib/web/v9-stdlib-ui.fl +115 -0
  1320. package/stdlib/web/ws-client.fl +125 -0
  1321. package/tests/.v11-backup/test-migrate-sample.fl +22 -0
  1322. package/tests/PHASE-C-VERIFICATION-REPORT.md +125 -0
  1323. package/tests/benchmark-v11.1.js +103 -0
  1324. package/tests/evidence/01-about.html +1 -0
  1325. package/tests/evidence/02-post-hello.html +1 -0
  1326. package/tests/evidence/02-post-world.html +1 -0
  1327. package/tests/evidence/03-home.html +1 -0
  1328. package/tests/evidence/04-dist/404.html +1 -0
  1329. package/tests/evidence/04-dist/about/index.html +1 -0
  1330. package/tests/evidence/04-dist/index.html +1 -0
  1331. package/tests/evidence/04-dist/post/hello/index.html +1 -0
  1332. package/tests/evidence/04-dist/post/world/index.html +1 -0
  1333. package/tests/evidence/05-dist/404.html +1 -0
  1334. package/tests/evidence/05-dist/about/index.html +1 -0
  1335. package/tests/evidence/05-dist/index.html +1 -0
  1336. package/tests/evidence/05-dist/post/hello/index.html +1 -0
  1337. package/tests/evidence/05-dist/post/world/index.html +1 -0
  1338. package/tests/evidence/06-dist/404.html +1 -0
  1339. package/tests/evidence/06-dist/about/index.html +1 -0
  1340. package/tests/evidence/06-dist/index.html +1 -0
  1341. package/tests/evidence/06-dist/post/hello/index.html +1 -0
  1342. package/tests/evidence/06-dist/post/world/index.html +1 -0
  1343. package/tests/evidence/07-markdown.html +1 -0
  1344. package/tests/evidence/08-rss.xml +25 -0
  1345. package/tests/evidence/09-robots.txt +4 -0
  1346. package/tests/evidence/09-sitemap.xml +12 -0
  1347. package/tests/evidence/10-jsonld.html +1 -0
  1348. package/tests/evidence/_results.txt +11 -0
  1349. package/tests/evidence/about.html +1 -0
  1350. package/tests/evidence/home.html +1 -0
  1351. package/tests/evidence/missing.html +42 -0
  1352. package/tests/evidence/post-hello.html +1 -0
  1353. package/tests/evidence/post-world.html +1 -0
  1354. package/tests/fixtures/basic-app/about/page.fl +1 -0
  1355. package/tests/fixtures/basic-app/api/echo/route.fl +5 -0
  1356. package/tests/fixtures/basic-app/layout.fl +1 -0
  1357. package/tests/fixtures/basic-app/not-found.fl +1 -0
  1358. package/tests/fixtures/basic-app/page.fl +1 -0
  1359. package/tests/fixtures/basic-app/post/[slug]/generate-static-params.fl +1 -0
  1360. package/tests/fixtures/basic-app/post/[slug]/page.fl +1 -0
  1361. package/tests/fixtures/stdlib-probes/blog.fl +6 -0
  1362. package/tests/fixtures/stdlib-probes/jsonld.fl +1 -0
  1363. package/tests/fixtures/stdlib-probes/map-probe.fl +3 -0
  1364. package/tests/fixtures/stdlib-probes/map-shape.fl +2 -0
  1365. package/tests/fixtures/stdlib-probes/md.fl +1 -0
  1366. package/tests/fixtures/stdlib-probes/robots.fl +1 -0
  1367. package/tests/fixtures/stdlib-probes/rss.fl +4 -0
  1368. package/tests/fixtures/stdlib-probes/sitemap.fl +1 -0
  1369. package/tests/l2/case-01-arithmetic.fl +6 -0
  1370. package/tests/l2/case-02-comparisons.fl +15 -0
  1371. package/tests/l2/case-03-logic.fl +15 -0
  1372. package/tests/l2/case-04-control-flow.fl +15 -0
  1373. package/tests/l2/case-05-functions.fl +13 -0
  1374. package/tests/l2/case-06-collections.fl +14 -0
  1375. package/tests/l2/case-07-pattern-matching.fl +14 -0
  1376. package/tests/l2/case-08-recursion.fl +19 -0
  1377. package/tests/l2/case-09-strings.fl +13 -0
  1378. package/tests/l2/case-10-loops.fl +17 -0
  1379. package/tests/l2/case-11-higher-order.fl +13 -0
  1380. package/tests/l2/case-12-edge-cases.fl +14 -0
  1381. package/tests/l2/case-13-ai-vector.fl +22 -0
  1382. package/tests/l2/case-14-ai-cosine.fl +25 -0
  1383. package/tests/l2/case-15-ai-template.fl +22 -0
  1384. package/tests/l2/case-16-ai-ranking.fl +20 -0
  1385. package/tests/l2/case-17-stdlib-extended.fl +32 -0
  1386. package/tests/l2-proof/01-arithmetic.bootstrap.js +141 -0
  1387. package/tests/l2-proof/01-arithmetic.fl +15 -0
  1388. package/tests/l2-proof/01-arithmetic.stage1.js +141 -0
  1389. package/tests/l2-proof/02-comparisons.bootstrap.js +141 -0
  1390. package/tests/l2-proof/02-comparisons.fl +17 -0
  1391. package/tests/l2-proof/03-logic.bootstrap.js +141 -0
  1392. package/tests/l2-proof/03-logic.fl +15 -0
  1393. package/tests/l2-proof/04-control-flow.bootstrap.js +146 -0
  1394. package/tests/l2-proof/04-control-flow.fl +24 -0
  1395. package/tests/l2-proof/05-functions.bootstrap.js +143 -0
  1396. package/tests/l2-proof/05-functions.fl +16 -0
  1397. package/tests/l2-proof/06-collections.bootstrap.js +146 -0
  1398. package/tests/l2-proof/06-collections.fl +27 -0
  1399. package/tests/l2-proof/07-pattern-matching.bootstrap.js +142 -0
  1400. package/tests/l2-proof/07-pattern-matching.fl +23 -0
  1401. package/tests/l2-proof/08-async-errors.bootstrap.js +142 -0
  1402. package/tests/l2-proof/08-async-errors.fl +17 -0
  1403. package/tests/l2-proof/09-strings.bootstrap.js +141 -0
  1404. package/tests/l2-proof/09-strings.fl +13 -0
  1405. package/tests/l2-proof/10-type-checks.bootstrap.js +141 -0
  1406. package/tests/l2-proof/10-type-checks.fl +17 -0
  1407. package/tests/l2-proof/11-recursion.bootstrap.js +143 -0
  1408. package/tests/l2-proof/11-recursion.fl +21 -0
  1409. package/tests/l2-proof/12-edge-cases.bootstrap.js +141 -0
  1410. package/tests/l2-proof/12-edge-cases.fl +17 -0
  1411. package/tests/parity/01-app-router-static.sh +18 -0
  1412. package/tests/parity/02-app-router-dynamic.sh +18 -0
  1413. package/tests/parity/03-layout-children.sh +17 -0
  1414. package/tests/parity/04-not-found.sh +22 -0
  1415. package/tests/parity/05-ssg.sh +21 -0
  1416. package/tests/parity/06-parallel-render.sh +21 -0
  1417. package/tests/parity/07-markdown.sh +16 -0
  1418. package/tests/parity/08-rss-atom.sh +18 -0
  1419. package/tests/parity/09-sitemap-robots.sh +23 -0
  1420. package/tests/parity/10-jsonld.sh +16 -0
  1421. package/tests/parity/11-blog-helpers.sh +28 -0
  1422. package/tests/parity/12-api-routes.sh +25 -0
  1423. package/tests/parity/_lib.sh +53 -0
  1424. package/tests/parity/run-all.sh +89 -0
  1425. package/tests/parity/score.sh +20 -0
  1426. package/tests/phase-c-fuzzing.fl +89 -0
  1427. package/tests/phase-c-property-testing.fl +137 -0
  1428. package/tests/phase-c-sha-verification.fl +117 -0
  1429. package/tests/phase-c-validation.fl +194 -0
  1430. package/tests/property-testing.sh +47 -0
  1431. package/tests/regen/let-in-expr-01.fl +5 -0
  1432. package/tests/regen/let-in-expr-02.fl +6 -0
  1433. package/tests/regen/let-in-expr-03.fl +4 -0
  1434. package/tests/regen/let-in-expr-04.fl +6 -0
  1435. package/tests/regen/let-in-expr-05.fl +5 -0
  1436. package/tests/test-ai-library.fl +36 -0
  1437. package/tests/test-andor.fl +2 -0
  1438. package/tests/test-bootstrap-match.fl +5 -0
  1439. package/tests/test-cg-final.fl +5 -0
  1440. package/tests/test-cli.fl +19 -0
  1441. package/tests/test-codegen.fl +4 -0
  1442. package/tests/test-cond-flat.fl +1 -0
  1443. package/tests/test-fl-exec-op.fl +1 -0
  1444. package/tests/test-fp.fl +10 -0
  1445. package/tests/test-funcs.fl +11 -0
  1446. package/tests/test-fuzz-crash.fl +3 -0
  1447. package/tests/test-http-get.fl +3 -0
  1448. package/tests/test-json-load.fl +32 -0
  1449. package/tests/test-let-flat.fl +2 -0
  1450. package/tests/test-let-order.fl +3 -0
  1451. package/tests/test-let-simple.fl +3 -0
  1452. package/tests/test-let-syntax.fl +2 -0
  1453. package/tests/test-let.fl +3 -0
  1454. package/tests/test-lex-debug.fl +75 -0
  1455. package/tests/test-loop-bug.fl +5 -0
  1456. package/tests/test-map-entries.fl +1 -0
  1457. package/tests/test-map-pattern.fl +3 -0
  1458. package/tests/test-match-syntax.fl +5 -0
  1459. package/tests/test-migrate-sample.fl +22 -0
  1460. package/tests/test-neg.fl +1 -0
  1461. package/tests/test-nested-let.fl +4 -0
  1462. package/tests/test-node-to-pattern.fl +8 -0
  1463. package/tests/test-quant-lib.fl +114 -0
  1464. package/tests/test-reduce.fl +10 -0
  1465. package/tests/test-server.fl +3 -0
  1466. package/tests/test-simple-map.fl +4 -0
  1467. package/tests/test-simple-match.fl +3 -0
  1468. package/tests/test-simple.fl +1 -0
  1469. package/tests/test-try-bootstrap.fl +7 -0
  1470. package/tests/test-try-catch.fl +3 -0
  1471. package/tests/test-watchdog-lib.fl +69 -0
  1472. package/tests/test-which-lex.fl +4 -0
  1473. package/tsconfig.json +39 -0
package/src/cli.ts ADDED
@@ -0,0 +1,2136 @@
1
+ #!/usr/bin/env node
2
+ // FreeLang v9 CLI
3
+ // 사용법:
4
+ // freelang run <file.fl> 파일 실행
5
+ // freelang run <file.fl> --watch 파일 변경시 자동 재실행
6
+ // freelang repl 대화형 REPL
7
+ // freelang check <file.fl> 문법 검사만 (실행 안 함)
8
+ // freelang fmt <file.fl> 파일 인플레이스 포맷
9
+ // freelang fmt --check <file.fl> 변경 필요 시 exit 1
10
+ // freelang fmt --stdin stdin → stdout 포맷
11
+
12
+ import * as fs from "fs";
13
+ import * as path from "path";
14
+ import * as readline from "readline";
15
+ import { lex } from "./lexer";
16
+ import { parse, ParserError } from "./parser";
17
+ import { interpret, Interpreter } from "./interpreter";
18
+ import { Block } from "./ast";
19
+ import { JSCodegen } from "./codegen-js"; // Phase 6: FL 컴파일러
20
+ import { formatFL } from "./formatter";
21
+ import { typeCheckSource, formatTypeIssues } from "./type-check-static";
22
+ import { DebugSession, setGlobalDebugSession } from "./debugger"; // Phase 78: 디버거
23
+ import { runWithWatch } from "./hot-reload"; // Phase 79: 워치 모드
24
+ import { extractDocs } from "./doc-extractor"; // Phase 77: 문서 추출기
25
+ import { renderMarkdown } from "./doc-renderer"; // Phase 77: 문서 렌더러
26
+ import { createDefaultPipeline, createFmtCheckStep, createLintStep, createTestStep } from "./ci-runner"; // Phase 80: CI
27
+ import { WebServer } from "./web"; // Phase 3: Web Server
28
+ import { fnMetaRegistry, FnMeta, EFFECT_CATALOG } from "./eval-special-forms"; // AI-Native Phase 1+2
29
+ import { propRegistry, runProp } from "./stdlib-property"; // AI-Native Phase 4
30
+
31
+ // ─────────────────────────────────────────
32
+ // 에러 포맷터: 소스 줄 강조
33
+ // ─────────────────────────────────────────
34
+
35
+ function formatCallStack(stack: Array<{fn: string; line: number; args?: any[]}>): string {
36
+ if (!stack || stack.length === 0) return "";
37
+ const frames = stack.slice().reverse().slice(0, 10);
38
+ return "\n\x1b[2m콜 스택:\x1b[0m\n" + frames.map((f, i) =>
39
+ ` \x1b[2m${i === 0 ? "→" : " "} ${(f as any).name ?? f.fn ?? "?"} (line ${f.line})\x1b[0m`
40
+ ).join("\n");
41
+ }
42
+
43
+ function formatError(err: any, source?: string, filePath?: string, callStack?: Array<{fn: string; line: number; args?: any[]}>): string {
44
+ const fileName = filePath ? path.basename(filePath) : "<stdin>";
45
+ const lines: string[] = [];
46
+
47
+ if (err instanceof ParserError) {
48
+ lines.push(`\n\x1b[31m파싱 오류\x1b[0m ${fileName}:${err.line}:${err.col}`);
49
+ if (source) {
50
+ const srcLines = source.split("\n");
51
+ const lineIdx = err.line - 1;
52
+ if (lineIdx >= 0 && lineIdx < srcLines.length) {
53
+ const lineNum = String(err.line).padStart(4, " ");
54
+ lines.push(` ${lineNum} │ ${srcLines[lineIdx]}`);
55
+ lines.push(` ${"─".repeat(err.col - 1)}^`);
56
+ }
57
+ }
58
+ lines.push(` ${err.message}`);
59
+ } else if (err instanceof Error) {
60
+ // "FreeLang line N: msg" 패턴에서 라인 번호 추출
61
+ const lineMatch = err.message.match(/^FreeLang line (\d+):\s*/);
62
+ const errLine = lineMatch ? parseInt(lineMatch[1]) : 0;
63
+ const cleanMsg = lineMatch ? err.message.slice(lineMatch[0].length) : err.message;
64
+
65
+ lines.push(`\n\x1b[31m실행 오류\x1b[0m ${fileName}${errLine ? `:${errLine}` : ""}`);
66
+
67
+ // 소스 컨텍스트: 에러 라인 ±1줄 표시
68
+ if (source && errLine > 0) {
69
+ const srcLines = source.split("\n");
70
+ const start = Math.max(0, errLine - 2);
71
+ const end = Math.min(srcLines.length - 1, errLine);
72
+ for (let i = start; i <= end; i++) {
73
+ const num = String(i + 1).padStart(4, " ");
74
+ const marker = (i + 1 === errLine) ? "\x1b[31m→\x1b[0m" : " ";
75
+ const lineColor = (i + 1 === errLine) ? `\x1b[31m${srcLines[i]}\x1b[0m` : `\x1b[2m${srcLines[i]}\x1b[0m`;
76
+ lines.push(` ${marker} ${num} │ ${lineColor}`);
77
+ }
78
+ lines.push("");
79
+ }
80
+
81
+ lines.push(` \x1b[31m✖\x1b[0m ${cleanMsg}`);
82
+ const stack = callStack ?? (err as any).__flCallStack;
83
+ if (stack && stack.length > 0) lines.push(formatCallStack(stack));
84
+ } else {
85
+ lines.push(`\n\x1b[31m오류\x1b[0m ${String(err)}`);
86
+ }
87
+
88
+ return lines.join("\n");
89
+ }
90
+
91
+ // ─────────────────────────────────────────
92
+ // 실행 엔진
93
+ // ─────────────────────────────────────────
94
+
95
+ function runSource(source: string, filePath?: string): { ok: boolean; value: any } {
96
+ const interp = new Interpreter();
97
+ if (filePath) interp.currentFilePath = path.resolve(filePath);
98
+ try {
99
+ const tokens = lex(source);
100
+ const ast = parse(tokens);
101
+ const ctx = interp.interpret(ast);
102
+ return { ok: true, value: ctx.lastValue };
103
+ } catch (err: any) {
104
+ const callStack: Array<{fn: string; line: number; args?: any[]}> = (interp as any).callStack ?? [];
105
+ console.error(formatError(err, source, filePath, callStack));
106
+ return { ok: false, value: null };
107
+ }
108
+ }
109
+
110
+ function checkSource(source: string, filePath?: string): boolean {
111
+ try {
112
+ const tokens = lex(source);
113
+ parse(tokens);
114
+ const fileName = filePath ? path.basename(filePath) : "<stdin>";
115
+ console.log(`\x1b[32m✓\x1b[0m ${fileName} 문법 이상 없음`);
116
+ return true;
117
+ } catch (err: any) {
118
+ console.error(formatError(err, source, filePath));
119
+ return false;
120
+ }
121
+ }
122
+
123
+ // ─────────────────────────────────────────
124
+ // run 커맨드
125
+ // ─────────────────────────────────────────
126
+
127
+ function flPidFile(absPath: string): string {
128
+ const safe = absPath.replace(/[^a-zA-Z0-9]/g, "_");
129
+ return `/tmp/fl_${safe}.pid`;
130
+ }
131
+
132
+ function cmdRun(filePath: string, watch: boolean, extraArgs: string[] = []): void {
133
+ const absPath = path.resolve(filePath);
134
+ const vmBench = extraArgs.includes("--vm-bench");
135
+
136
+ if (!fs.existsSync(absPath)) {
137
+ console.error(`\x1b[31m오류\x1b[0m 파일을 찾을 수 없습니다: ${filePath}`);
138
+ process.exit(1);
139
+ }
140
+
141
+ // PID 파일 기록 — fl stop으로 정확하게 종료 가능
142
+ // 형식: "pid\noriginalAbsPath" (경로 역산 불필요)
143
+ const pidFile = flPidFile(absPath);
144
+ try { fs.writeFileSync(pidFile, `${process.pid}\n${absPath}`); } catch {}
145
+ process.on("exit", () => { try { fs.unlinkSync(pidFile); } catch {} });
146
+
147
+ function execute(): void {
148
+ const source = fs.readFileSync(absPath, "utf-8");
149
+ // 추가 인수를 $__argv__로 인터프리터에 전달 (셀프 호스팅 지원)
150
+ let ctx: any;
151
+ try {
152
+ const tokens = lex(source);
153
+ const ast = parse(tokens);
154
+ const interp = new Interpreter();
155
+ interp.currentFilePath = absPath;
156
+ // ⚠️ 조건부 정의: extraArgs가 없으면 $__argv__ 변수 자체가 없음 (null이 아님!)
157
+ // freelang-interpreter.fl에서 (null? $__argv__)로 체크해야 하는 이유
158
+ if (extraArgs.length > 0) {
159
+ interp.context.variables.set("$__argv__", extraArgs);
160
+ }
161
+ ctx = interp.interpret(ast);
162
+ } catch (err: any) {
163
+ console.error(formatError(err, source, absPath));
164
+ if (!watch) process.exit(1);
165
+ return;
166
+ }
167
+ const value = ctx?.lastValue;
168
+ if (value !== null && value !== undefined) {
169
+ if (typeof value === "object") {
170
+ console.log(JSON.stringify(value, null, 2));
171
+ } else {
172
+ console.log(String(value));
173
+ }
174
+ }
175
+ }
176
+
177
+ if (watch) {
178
+ // Enable dev mode BEFORE first execute: server_html will inject the
179
+ // hot-reload client script, and /__hot SSE endpoint will be served.
180
+ process.env.FL_DEV = "1";
181
+ }
182
+
183
+ // Phase 3-E: VM opt-in 성능 벤치마크
184
+ if (vmBench) {
185
+ console.log("\n\x1b[36m[vm-bench] 성능 측정 시작...\x1b[0m");
186
+ const ITERATIONS = 100;
187
+ const source = fs.readFileSync(absPath, "utf-8");
188
+
189
+ // 기존 경로 (VM 미사용)
190
+ const t0 = performance.now();
191
+ for (let i = 0; i < ITERATIONS; i++) {
192
+ delete process.env.FL_VM;
193
+ try {
194
+ const tokens = lex(source);
195
+ const ast = parse(tokens);
196
+ const interp = new Interpreter();
197
+ interp.currentFilePath = absPath;
198
+ if (extraArgs.length > 0) {
199
+ interp.context.variables.set("$__argv__", extraArgs.filter(a => a !== "--vm-bench"));
200
+ }
201
+ interp.interpret(ast);
202
+ } catch (_e) { /* ignore */ }
203
+ }
204
+ const interpMs = performance.now() - t0;
205
+
206
+ // VM 경로
207
+ const t1 = performance.now();
208
+ for (let i = 0; i < ITERATIONS; i++) {
209
+ process.env.FL_VM = "1";
210
+ try {
211
+ const tokens = lex(source);
212
+ const ast = parse(tokens);
213
+ const interp = new Interpreter();
214
+ interp.currentFilePath = absPath;
215
+ if (extraArgs.length > 0) {
216
+ interp.context.variables.set("$__argv__", extraArgs.filter(a => a !== "--vm-bench"));
217
+ }
218
+ interp.interpret(ast);
219
+ } catch (_e) { /* ignore */ }
220
+ }
221
+ const vmMs = performance.now() - t1;
222
+
223
+ // 결과 출력
224
+ delete process.env.FL_VM;
225
+ const speedup = interpMs / vmMs;
226
+ console.log(`\x1b[36m[vm-bench]\x1b[0m interpreter: ${interpMs.toFixed(1)}ms (${ITERATIONS} iter)`);
227
+ console.log(`\x1b[36m[vm-bench]\x1b[0m vm: ${vmMs.toFixed(1)}ms (${ITERATIONS} iter)`);
228
+ console.log(`\x1b[36m[vm-bench]\x1b[0m speedup: ${speedup.toFixed(2)}x`);
229
+
230
+ if (speedup < 1.0) {
231
+ console.log(`\x1b[33m⚠️ VM이 느림 (산술 집약 코드가 아닐 수 있음)\x1b[0m`);
232
+ } else if (speedup >= 1.5) {
233
+ console.log(`\x1b[32m✓ 목표 달성 (1.5배 이상)\x1b[0m`);
234
+ } else {
235
+ console.log(`\x1b[2m○ 1.0x ~ 1.5x 범위\x1b[0m`);
236
+ }
237
+ console.log("");
238
+
239
+ if (!watch) return; // watch가 아니면 여기서 종료
240
+ }
241
+
242
+ // 개발 중 --watch 미사용 시 한 번만 힌트 출력
243
+ if (!watch && !vmBench && !process.env.FL_NO_HINT) {
244
+ const isServerFile = fs.readFileSync(absPath, "utf-8").includes("server_start");
245
+ if (isServerFile) {
246
+ process.stderr.write(`\x1b[2m💡 개발 중에는: freelang watch ${path.basename(absPath)}\x1b[0m\n`);
247
+ }
248
+ }
249
+
250
+ execute();
251
+
252
+ if (watch) {
253
+ console.log(`\x1b[2m watching ${path.basename(absPath)}... (dev mode: browser auto-reload enabled)\x1b[0m`);
254
+ // Use setInterval polling for reliability across filesystems
255
+ // (fs.watch silently breaks on Termux/Android, or after the editor
256
+ // rewrites the inode — polling survives re-executes and inode changes)
257
+ let lastMtime = 0;
258
+ let lastSize = -1;
259
+ try {
260
+ const st = fs.statSync(absPath);
261
+ lastMtime = st.mtimeMs;
262
+ lastSize = st.size;
263
+ } catch (_e) { /* ignore */ }
264
+ let debounce: NodeJS.Timeout | null = null;
265
+ setInterval(() => {
266
+ try {
267
+ const st = fs.statSync(absPath);
268
+ if (st.mtimeMs !== lastMtime || st.size !== lastSize) {
269
+ lastMtime = st.mtimeMs;
270
+ lastSize = st.size;
271
+ if (debounce) clearTimeout(debounce);
272
+ debounce = setTimeout(() => {
273
+ console.log(`\n\x1b[2m─── 변경 감지, 재실행 ───\x1b[0m`);
274
+ execute();
275
+ }, 100);
276
+ }
277
+ } catch (_e) { /* transient errors ignored */ }
278
+ }, 500);
279
+ }
280
+ }
281
+
282
+ // ─────────────────────────────────────────
283
+ // check 커맨드
284
+ // ─────────────────────────────────────────
285
+
286
+ function cmdProps(filePath: string, extraArgs: string[]): void {
287
+ const absPath = path.resolve(filePath);
288
+ if (!fs.existsSync(absPath)) {
289
+ console.error(`\x1b[31m오류\x1b[0m 파일을 찾을 수 없습니다: ${filePath}`);
290
+ process.exit(1);
291
+ }
292
+
293
+ // --samples N 오버라이드
294
+ const samplesIdx = extraArgs.indexOf("--samples");
295
+ const samplesOverride = samplesIdx >= 0 ? parseInt(extraArgs[samplesIdx + 1], 10) : undefined;
296
+
297
+ // 파일 실행 (defprop 등록)
298
+ const source = fs.readFileSync(absPath, "utf-8");
299
+ try {
300
+ const tokens = lex(source);
301
+ const ast = parse(tokens);
302
+ const interp = new Interpreter();
303
+ interp.currentFilePath = absPath;
304
+ // run-props를 자동 주입하지 않고 defprop 등록만 먼저 수행
305
+ interp.interpret(ast);
306
+ } catch (err: any) {
307
+ // 실행 에러는 무시하고 등록된 props만 실행
308
+ }
309
+
310
+ if (propRegistry.size === 0) {
311
+ console.log(`\x1b[33m[props]\x1b[0m ${path.basename(absPath)} — defprop 없음`);
312
+ console.log(`\x1b[2m 팁: (defprop my-prop {:fn "add" :args [:int :int] :check (fn [$a $b] (= (add $a $b) (add $b $a)))})\x1b[0m`);
313
+ return;
314
+ }
315
+
316
+ console.log(`\x1b[36m[props]\x1b[0m ${path.basename(absPath)} — ${propRegistry.size}개 property 실행\n`);
317
+
318
+ // interp 재생성 (함수 접근용)
319
+ let interp2: any;
320
+ try {
321
+ const tokens = lex(source);
322
+ const ast = parse(tokens);
323
+ interp2 = new Interpreter();
324
+ interp2.currentFilePath = absPath;
325
+ interp2.interpret(ast);
326
+ } catch { /* ignore */ }
327
+
328
+ let totalPassed = 0;
329
+ let totalFailed = 0;
330
+ let exitCode = 0;
331
+
332
+ for (const [, prop] of propRegistry) {
333
+ const p = samplesOverride ? { ...prop, samples: samplesOverride } : prop;
334
+ const result = runProp(
335
+ p,
336
+ (fnName, fnArgs) => {
337
+ const fnVal = interp2?.context?.variables?.get(fnName)
338
+ ?? interp2?.context?.variables?.get("$" + fnName);
339
+ if (!fnVal) throw new Error(`함수 없음: ${fnName}`);
340
+ return interp2.callFunctionValue(fnVal, fnArgs);
341
+ },
342
+ (checkFn, checkArgs) => {
343
+ if (!checkFn) return true;
344
+ return interp2.callFunctionValue(checkFn, checkArgs);
345
+ }
346
+ );
347
+
348
+ totalPassed += result.passed;
349
+ totalFailed += result.failed;
350
+
351
+ const ok = result.failed === 0;
352
+ const status = ok ? "\x1b[32m✓\x1b[0m" : "\x1b[31m✖\x1b[0m";
353
+ process.stdout.write(
354
+ ` ${status} \x1b[1m${result.name}\x1b[0m` +
355
+ ` \x1b[2m${result.fn} ${result.passed}/${result.samples} ${result.durationMs}ms\x1b[0m\n`
356
+ );
357
+ if (!ok && result.firstFailure) {
358
+ const f = result.firstFailure;
359
+ process.stdout.write(
360
+ ` \x1b[31m반례\x1b[0m: args=${JSON.stringify(f.args)}` +
361
+ (f.error ? ` error=${f.error}` : ` result=${JSON.stringify(f.result)}`) + "\n"
362
+ );
363
+ exitCode = 1;
364
+ }
365
+ }
366
+
367
+ const allOk = totalFailed === 0;
368
+ process.stdout.write(
369
+ `\n ${allOk ? "\x1b[32m[PROPS PASS]\x1b[0m" : "\x1b[31m[PROPS FAIL]\x1b[0m"}` +
370
+ ` ${propRegistry.size}개 property — ${totalPassed} passed / ${totalFailed} failed\n`
371
+ );
372
+ if (exitCode) process.exit(exitCode);
373
+ }
374
+
375
+ function cmdCheck(filePath: string): void {
376
+ const absPath = path.resolve(filePath);
377
+ if (!fs.existsSync(absPath)) {
378
+ console.error(`\x1b[31m오류\x1b[0m 파일을 찾을 수 없습니다: ${filePath}`);
379
+ process.exit(1);
380
+ }
381
+ const source = fs.readFileSync(absPath, "utf-8");
382
+ const ok = checkSource(source, absPath);
383
+ if (!ok) process.exit(1);
384
+
385
+ // AI-Native Phase 1+2: defn 메타 + effects 검사 (AST 정적 분석)
386
+ const { metaMissing, effectsWarn } = checkDefnMeta(source, absPath);
387
+ let hasWarnings = false;
388
+
389
+ if (metaMissing.length > 0) {
390
+ hasWarnings = true;
391
+ process.stderr.write(`\n\x1b[33m[meta-check]\x1b[0m ${path.basename(absPath)} — ${metaMissing.length}개 함수에 메타 없음\n`);
392
+ for (const w of metaMissing) {
393
+ process.stderr.write(` \x1b[33m⚠\x1b[0m line ${w.line}: \x1b[1m${w.name}\x1b[0m — :context/:returns 없음\n`);
394
+ }
395
+ process.stderr.write(`\x1b[2m 팁: (defn ${metaMissing[0].name} [...] {:context "..." :returns "..."} ...)\x1b[0m\n`);
396
+ }
397
+
398
+ const pureErrors = effectsWarn.filter((w: any) => w.isPure);
399
+ const effectsOnly = effectsWarn.filter((w: any) => !w.isPure);
400
+
401
+ if (pureErrors.length > 0) {
402
+ hasWarnings = true;
403
+ process.stderr.write(`\n\x1b[31m[pure-violation]\x1b[0m ${path.basename(absPath)} — ${pureErrors.length}개 ^pure 함수에서 side effect 발견\n`);
404
+ for (const w of pureErrors) {
405
+ const hint = w.undeclared.map((e: string) => `:${e}`).join(" ");
406
+ process.stderr.write(` \x1b[31m✖\x1b[0m line ${w.line}: \x1b[1m${w.name}\x1b[0m — \x1b[31m${hint}\x1b[0m (순수 함수 위반)\n`);
407
+ }
408
+ process.stderr.write(`\x1b[2m :effects [] 또는 ^pure 선언 시 side effect 호출 불가\x1b[0m\n`);
409
+ }
410
+
411
+ if (effectsOnly.length > 0) {
412
+ hasWarnings = true;
413
+ process.stderr.write(`\n\x1b[33m[effects-check]\x1b[0m ${path.basename(absPath)} — ${effectsOnly.length}개 함수에 미선언 effect\n`);
414
+ for (const w of effectsOnly) {
415
+ const hint = w.undeclared.map((e: string) => `:${e}`).join(" ");
416
+ process.stderr.write(` \x1b[33m⚠\x1b[0m line ${w.line}: \x1b[1m${w.name}\x1b[0m — 미선언 effect: \x1b[33m${hint}\x1b[0m\n`);
417
+ }
418
+ }
419
+
420
+ if (hasWarnings) process.stderr.write("\n");
421
+
422
+ // 정적 타입 검사 (arity + type hint)
423
+ const { errors: typeErrors, warnings: typeWarns, fnCount } = typeCheckSource(source);
424
+ if (typeErrors.length > 0 || typeWarns.length > 0) {
425
+ process.stderr.write(formatTypeIssues([...typeErrors, ...typeWarns], absPath) + "\n\n");
426
+ if (typeErrors.length > 0) process.exit(1);
427
+ } else if (fnCount > 0) {
428
+ console.log(`\x1b[36m[type-check]\x1b[0m ${path.basename(absPath)} 타입 이상 없음 (함수 ${fnCount}개 분석)`);
429
+ }
430
+ }
431
+
432
+ function checkDefnMeta(source: string, filePath?: string): {
433
+ metaMissing: Array<{name: string; line: number}>;
434
+ effectsWarn: Array<{name: string; line: number; undeclared: string[]}>;
435
+ } {
436
+ const metaMissing: Array<{name: string; line: number}> = [];
437
+ const effectsWarn: Array<{name: string; line: number; undeclared: string[]; isPure: boolean}> = [];
438
+ try {
439
+ const tokens = lex(source);
440
+ const ast = parse(tokens);
441
+ const META_KEYS = new Set(["returns", "context", "effects", "examples"]);
442
+
443
+ function collectCalls(node: any, found: Set<string>): void {
444
+ if (!node) return;
445
+ if (node.kind === "sexpr") {
446
+ if (node.op) found.add(node.op);
447
+ if (Array.isArray(node.args)) node.args.forEach((a: any) => collectCalls(a, found));
448
+ } else if (node.kind === "block" && node.fields instanceof Map) {
449
+ node.fields.forEach((v: any) => collectCalls(v, found));
450
+ }
451
+ }
452
+
453
+ function walkNodes(nodes: any[]): void {
454
+ for (const node of nodes) {
455
+ if (!node) continue;
456
+ if (node.kind === "sexpr" && (node.op === "defn" || node.op === "defun")) {
457
+ let argIdx = 0;
458
+ const args = node.args ?? [];
459
+ // ^pure 또는 ^type 힌트
460
+ let isPureHint = false;
461
+ if (args[argIdx]?.kind === "literal" && String((args[argIdx] as any).value ?? "").startsWith("^")) {
462
+ if (String((args[argIdx] as any).value) === "^pure") isPureHint = true;
463
+ argIdx++;
464
+ }
465
+ const nameNode = args[argIdx];
466
+ const name: string = nameNode?.kind === "variable" ? nameNode.name
467
+ : nameNode?.kind === "literal" ? String(nameNode.value) : "?";
468
+ let bodyArgs = args.slice(argIdx + 2);
469
+ // check if first body arg is a meta map
470
+ const first = bodyArgs[0] as any;
471
+ let metaMap: Map<string, any> | null = null;
472
+ if (first?.kind === "block" && first?.type === "Map" && first.fields instanceof Map) {
473
+ if ([...first.fields.keys()].some(k => META_KEYS.has(k))) {
474
+ metaMap = first.fields;
475
+ bodyArgs = bodyArgs.slice(1);
476
+ }
477
+ }
478
+ if (!metaMap && bodyArgs.length > 0) {
479
+ metaMissing.push({ name, line: node.line ?? 0 });
480
+ }
481
+ // AI-Native Phase 2+3: effects 정적 검사
482
+ const effectsDeclared = metaMap?.has("effects") || isPureHint;
483
+ if (effectsDeclared) {
484
+ let declared: string[] = [];
485
+ if (!isPureHint && metaMap?.has("effects")) {
486
+ const eNode = metaMap!.get("effects") as any;
487
+ if (eNode?.kind === "block" && eNode?.type === "Array") {
488
+ const items = eNode.fields?.get("items") as any[];
489
+ if (Array.isArray(items)) {
490
+ declared = items.map((it: any) =>
491
+ it?.kind === "literal" ? String(it.value).replace(/^:/, "") : "?");
492
+ }
493
+ }
494
+ }
495
+ const isPure = isPureHint || declared.length === 0;
496
+ const declaredSet = new Set(declared);
497
+ const calledOps = new Set<string>();
498
+ bodyArgs.forEach((b: any) => collectCalls(b, calledOps));
499
+ const undeclared: string[] = [];
500
+ calledOps.forEach(op => {
501
+ const eff = EFFECT_CATALOG.get(op);
502
+ if (eff && !declaredSet.has(eff)) undeclared.push(eff);
503
+ });
504
+ if (undeclared.length > 0) {
505
+ effectsWarn.push({ name, line: node.line ?? 0, undeclared: [...new Set(undeclared)], isPure });
506
+ }
507
+ }
508
+ walkNodes(bodyArgs);
509
+ } else if (node.kind === "sexpr" && node.args) {
510
+ walkNodes(node.args);
511
+ } else if (node.kind === "block" && node.fields instanceof Map) {
512
+ walkNodes([...node.fields.values()]);
513
+ }
514
+ }
515
+ }
516
+
517
+ walkNodes(ast as any[]);
518
+ } catch { /* parse errors already reported */ }
519
+ return { metaMissing, effectsWarn };
520
+ }
521
+
522
+ // ─────────────────────────────────────────
523
+ // compile 커맨드 (Phase 6)
524
+ // ─────────────────────────────────────────
525
+
526
+ function cmdCodegen(args: string[]): void {
527
+ // freelang codegen <file.fl> [--target typescript|sql|all]
528
+ const inputFile = args.find(a => !a.startsWith("-") && a.endsWith(".fl"));
529
+ const target = args.includes("--target") ? args[args.indexOf("--target") + 1] : "all";
530
+
531
+ if (!inputFile) {
532
+ console.error(`\x1b[31m오류\x1b[0m 입력 파일을 지정하세요: codegen <file.fl>`);
533
+ process.exit(1);
534
+ }
535
+
536
+ const absInput = path.resolve(inputFile);
537
+ if (!fs.existsSync(absInput)) {
538
+ console.error(`\x1b[31m오류\x1b[0m 파일을 찾을 수 없습니다: ${inputFile}`);
539
+ process.exit(1);
540
+ }
541
+
542
+ try {
543
+ const source = fs.readFileSync(absInput, "utf-8");
544
+ const tokens = lex(source);
545
+ const ast = parse(tokens);
546
+
547
+ const cg = new JSCodegen();
548
+
549
+ // 모든 블록에 대해 코드 생성
550
+ for (const node of ast) {
551
+ if (node.kind === "block") {
552
+ const blockType = (node as Block).type;
553
+ if (["SERVICE", "MODEL", "CONTROLLER"].includes(blockType)) {
554
+ if (target === "all" || (target === "typescript" && blockType !== "MODEL") || (target === "sql" && blockType === "MODEL")) {
555
+ const code = cg.generate([node]);
556
+ console.log(code);
557
+ console.log("\n" + "─".repeat(80) + "\n");
558
+ }
559
+ }
560
+ }
561
+ }
562
+ } catch (err) {
563
+ console.error(`\x1b[31m오류\x1b[0m ${formatError(err as Error, fs.readFileSync(absInput, "utf-8"), absInput)}`);
564
+ process.exit(1);
565
+ }
566
+ }
567
+
568
+ function cmdCompile(args: string[]): void {
569
+ // 옵션 파싱: compile input.fl -o output.js [--esm] [--runtime]
570
+ const outputIdx = args.indexOf("-o");
571
+ const inputFile = args.find(a => !a.startsWith("-") && a !== args[outputIdx + 1]);
572
+ const outputFile = outputIdx !== -1 ? args[outputIdx + 1] : null;
573
+ const useEsm = args.includes("--esm");
574
+ const withRuntime = args.includes("--runtime");
575
+
576
+ // 입력 파일 검증
577
+ if (!inputFile) {
578
+ console.error(`\x1b[31m오류\x1b[0m 입력 파일을 지정하세요: compile <file.fl> [-o <out.js>]`);
579
+ process.exit(1);
580
+ }
581
+
582
+ const absInput = path.resolve(inputFile);
583
+ if (!fs.existsSync(absInput)) {
584
+ console.error(`\x1b[31m오류\x1b[0m 파일을 찾을 수 없습니다: ${inputFile}`);
585
+ process.exit(1);
586
+ }
587
+
588
+ try {
589
+ // 파이프라인: lex → parse → JSCodegen.generate()
590
+ const source = fs.readFileSync(absInput, "utf-8");
591
+ const tokens = lex(source);
592
+ const ast = parse(tokens);
593
+
594
+ const cg = new JSCodegen();
595
+ const js = cg.generate(ast, {
596
+ module: useEsm ? "esm" : "commonjs",
597
+ runtime: withRuntime,
598
+ minify: false,
599
+ target: "node",
600
+ });
601
+
602
+ // 출력
603
+ if (outputFile) {
604
+ const absOutput = path.resolve(outputFile);
605
+ const dir = path.dirname(absOutput);
606
+ if (dir !== "." && !fs.existsSync(dir)) {
607
+ fs.mkdirSync(dir, { recursive: true });
608
+ }
609
+ fs.writeFileSync(absOutput, js, "utf-8");
610
+ console.log(`\x1b[32m✓\x1b[0m 컴파일 완료 ${path.basename(inputFile)} → ${outputFile}`);
611
+ } else {
612
+ // stdout 출력
613
+ process.stdout.write(js);
614
+ }
615
+ } catch (err: any) {
616
+ console.error(formatError(err, fs.readFileSync(absInput, "utf-8"), absInput));
617
+ process.exit(1);
618
+ }
619
+ }
620
+
621
+ // ─────────────────────────────────────────
622
+ // repl 커맨드
623
+ // ─────────────────────────────────────────
624
+
625
+ function cmdRepl(): void {
626
+ console.log(`FreeLang v11 REPL (\x1b[2m:q / (exit) / (quit) 종료 :help 도움말 :reset 세션 초기화\x1b[0m)`);
627
+ console.log(`─────────────────────────────────────────`);
628
+
629
+ // v11.3: ~/.fl_history 기반 영구 history
630
+ const historyPath = (() => {
631
+ try {
632
+ const os = require("os") as typeof import("os");
633
+ const path = require("path") as typeof import("path");
634
+ return path.join(os.homedir(), ".fl_history");
635
+ } catch { return null; }
636
+ })();
637
+
638
+ let initialHistory: string[] = [];
639
+ if (historyPath) {
640
+ try {
641
+ const fs = require("fs") as typeof import("fs");
642
+ if (fs.existsSync(historyPath)) {
643
+ initialHistory = fs.readFileSync(historyPath, "utf8")
644
+ .split("\n").filter(l => l.trim()).slice(-500).reverse();
645
+ }
646
+ } catch {}
647
+ }
648
+
649
+ const rl = readline.createInterface({
650
+ input: process.stdin,
651
+ output: process.stdout,
652
+ prompt: "\x1b[36mfl>\x1b[0m ",
653
+ terminal: true,
654
+ history: initialHistory,
655
+ historySize: 500,
656
+ } as any);
657
+
658
+ // 멀티라인: 여는 괄호/대괄호가 남아있으면 계속 입력 받음
659
+ let buffer = "";
660
+ // v11.3: 영속 Interpreter — define/defn 바인딩이 세션 내내 유지됨
661
+ let sessionInterp = new Interpreter();
662
+
663
+ function countBalance(s: string): number {
664
+ let balance = 0;
665
+ let inStr = false;
666
+ for (let i = 0; i < s.length; i++) {
667
+ const ch = s[i];
668
+ if (ch === '"') {
669
+ let backslashCount = 0;
670
+ let j = i - 1;
671
+ while (j >= 0 && s[j] === "\\") { backslashCount++; j--; }
672
+ if (backslashCount % 2 === 0) inStr = !inStr;
673
+ }
674
+ if (!inStr) {
675
+ if (ch === "(" || ch === "[" || ch === "{") balance++;
676
+ if (ch === ")" || ch === "]" || ch === "}") balance--;
677
+ }
678
+ }
679
+ return balance;
680
+ }
681
+
682
+ rl.prompt();
683
+
684
+ rl.on("line", (line: string) => {
685
+ // 커맨드 처리
686
+ const trimmed = line.trim();
687
+ if (trimmed === ":q" || trimmed === ":quit" || trimmed === ":exit" ||
688
+ trimmed === "(exit)" || trimmed === "(quit)" || trimmed === "exit" || trimmed === "quit") {
689
+ console.log("bye.");
690
+ rl.close();
691
+ process.exit(0);
692
+ }
693
+ // (exit) / (quit) 오타 감지 — 친절한 안내
694
+ if (trimmed === "(exit" || trimmed === "(quit") {
695
+ console.log(`\x1b[33m💡 종료하려면: :q 또는 (exit) 입력\x1b[0m`);
696
+ return;
697
+ }
698
+ if (trimmed === ":help") {
699
+ console.log([
700
+ " :q / :quit 종료",
701
+ " :clear 버퍼 초기화",
702
+ " :help 이 도움말",
703
+ " :ls 정의된 함수 목록",
704
+ " :stack callStack 출력 (최근 20개)",
705
+ " :locals 현재 변수 dump",
706
+ " :debug debugger ON/OFF toggle",
707
+ " :step step 모드 toggle",
708
+ "",
709
+ " 예제:",
710
+ " (+ 1 2)",
711
+ ' (println "Hello, World!")',
712
+ ' (let [[$x 42]] (println "x = {$x}"))',
713
+ " [FUNC add :params [$a $b] :body (+ $a $b)]",
714
+ " (add 3 5)",
715
+ ].join("\n"));
716
+ rl.prompt();
717
+ return;
718
+ }
719
+ if (trimmed === ":clear") {
720
+ buffer = "";
721
+ console.log(" 버퍼 초기화됨.");
722
+ rl.prompt();
723
+ return;
724
+ }
725
+ if (trimmed === ":reset") {
726
+ buffer = "";
727
+ sessionInterp = new Interpreter();
728
+ console.log(" 세션 초기화됨 (모든 변수/함수 제거).");
729
+ rl.prompt();
730
+ return;
731
+ }
732
+
733
+ // S1 (2026-04-25): 디버거 + introspection 명령
734
+ if (trimmed === ":ls") {
735
+ const fns = [...sessionInterp.context.functions.keys()];
736
+ console.log(fns.length === 0 ? "(함수 없음)" : fns.slice(0, 50).join(" "));
737
+ if (fns.length > 50) console.log(` ... 외 ${fns.length - 50}개`);
738
+ rl.prompt();
739
+ return;
740
+ }
741
+ if (trimmed === ":stack") {
742
+ const stack: Array<{fn: string; line: number; args?: any[]}> = (sessionInterp as any).callStack ?? [];
743
+ if (stack.length === 0) console.log(" (callStack 비어있음 — 호출 중일 때만 표시)");
744
+ else {
745
+ const tail = stack.slice(-20);
746
+ for (let i = 0; i < tail.length; i++) {
747
+ const argsStr = tail[i].args ? `(${tail[i].args!.join(", ")})` : "";
748
+ console.log(` #${stack.length - tail.length + i}: ${tail[i].fn}${argsStr} (line ${tail[i].line})`);
749
+ }
750
+ }
751
+ rl.prompt();
752
+ return;
753
+ }
754
+ if (trimmed === ":locals") {
755
+ const vars: Map<string, any> = (sessionInterp.context.variables as any).snapshot?.() ?? new Map();
756
+ if (vars.size === 0) console.log(" (변수 없음)");
757
+ else {
758
+ let count = 0;
759
+ for (const [k, v] of vars) {
760
+ if (count++ >= 30) { console.log(` ... ${vars.size - 30}개 더`); break; }
761
+ const valStr = typeof v === "function" ? "<function>"
762
+ : v?.kind === "function-value" ? "<fn-value>"
763
+ : (() => { try { return JSON.stringify(v)?.slice(0, 60); } catch { return "<unserializable>"; } })();
764
+ console.log(` ${k} = ${valStr}`);
765
+ }
766
+ }
767
+ rl.prompt();
768
+ return;
769
+ }
770
+ if (trimmed === ":debug") {
771
+ try {
772
+ const { getGlobalDebugSession } = require("./debugger");
773
+ const sess = getGlobalDebugSession();
774
+ sess.enabled = !sess.enabled;
775
+ console.log(` debugger: ${sess.enabled ? "ON" : "OFF"}`);
776
+ } catch (e: any) {
777
+ console.log(` debug 모듈 로드 실패: ${e.message}`);
778
+ }
779
+ rl.prompt();
780
+ return;
781
+ }
782
+ if (trimmed === ":step") {
783
+ try {
784
+ const { getGlobalDebugSession } = require("./debugger");
785
+ const sess = getGlobalDebugSession();
786
+ sess.enabled = true;
787
+ sess.stepMode = !sess.stepMode;
788
+ console.log(` step 모드: ${sess.stepMode ? "ON (모든 줄에서 break)" : "OFF"}`);
789
+ } catch (e: any) {
790
+ console.log(` debug 모듈 로드 실패: ${e.message}`);
791
+ }
792
+ rl.prompt();
793
+ return;
794
+ }
795
+
796
+ // 세미콜론 주석 줄 스킵
797
+ if (trimmed.startsWith(";")) {
798
+ rl.prompt();
799
+ return;
800
+ }
801
+
802
+ buffer += (buffer ? "\n" : "") + line;
803
+
804
+ const balance = countBalance(buffer);
805
+ if (balance > 0) {
806
+ // 아직 닫히지 않은 괄호 있음 — 계속 입력
807
+ process.stdout.write("\x1b[2m …\x1b[0m ");
808
+ return;
809
+ }
810
+
811
+ const source = buffer.trim();
812
+ buffer = "";
813
+
814
+ if (!source) {
815
+ rl.prompt();
816
+ return;
817
+ }
818
+
819
+ try {
820
+ const tokens = lex(source);
821
+ const ast = parse(tokens);
822
+ // v11.3: persistent interpreter 재사용
823
+ const ctx = sessionInterp.interpret(ast);
824
+ const val = ctx.lastValue;
825
+ if (val !== null && val !== undefined) {
826
+ if (typeof val === "object") {
827
+ console.log("\x1b[33m=>\x1b[0m", JSON.stringify(val, null, 2));
828
+ } else {
829
+ console.log("\x1b[33m=>\x1b[0m", String(val));
830
+ }
831
+ }
832
+ } catch (err: any) {
833
+ console.error(formatError(err, source));
834
+ }
835
+
836
+ // v11.3: history 저장
837
+ if (historyPath && source) {
838
+ try {
839
+ const fs = require("fs") as typeof import("fs");
840
+ fs.appendFileSync(historyPath, source.replace(/\n/g, " ") + "\n");
841
+ } catch {}
842
+ }
843
+
844
+ rl.prompt();
845
+ });
846
+
847
+ rl.on("close", () => {
848
+ process.exit(0);
849
+ });
850
+ }
851
+
852
+ // ─────────────────────────────────────────
853
+ // 진입점
854
+ // ─────────────────────────────────────────
855
+
856
+ // ─────────────────────────────────────────
857
+ // fmt 커맨드 (Phase 73)
858
+ // ─────────────────────────────────────────
859
+
860
+ // ─────────────────────────────────────────
861
+ // v11.4: help 커맨드 — stdlib 함수 시그니처 빠른 조회
862
+ // ─────────────────────────────────────────
863
+
864
+ // v11.10: 빌드 시점에 embed 된 stdlib 시그니처 (배포 후에도 fn-doc 동작)
865
+ // scripts/build.js 가 src/_stdlib-signatures.json 을 생성 → esbuild 가 bundle.
866
+ // dev 환경(ts-jest 등)에서 이 파일이 없을 수 있으므로 try/catch fallback.
867
+ // eslint-disable-next-line @typescript-eslint/no-var-requires
868
+ function loadEmbeddedSignatures(): { module: string; name: string; params: string; returns: string }[] {
869
+ try {
870
+ return require("./_stdlib-signatures.json");
871
+ } catch {
872
+ return [];
873
+ }
874
+ }
875
+
876
+ function cmdStdlibDoc(query: string): void {
877
+ interface Entry { module: string; name: string; params: string; ret: string; }
878
+ const signatures = loadEmbeddedSignatures();
879
+ const entries: Entry[] = signatures.map((s) => ({
880
+ module: s.module, name: s.name, params: s.params, ret: s.returns,
881
+ }));
882
+
883
+ if (entries.length === 0) {
884
+ console.error(JSON.stringify({ error: "stdlib_signatures_missing", hint: "run `npm run build` to regenerate" }));
885
+ process.exit(2);
886
+ }
887
+
888
+ const q = query.toLowerCase();
889
+ const exact = entries.filter((e) => e.name === query);
890
+ const partial = entries.filter((e) => e.name.toLowerCase().includes(q) && e.name !== query);
891
+
892
+ // AI-first: JSON output for programmatic parsing
893
+ if (exact.length === 0 && partial.length === 0) {
894
+ const near = entries
895
+ .map((e) => ({ e, d: levenshtein(e.name.toLowerCase(), q) }))
896
+ .filter((x) => x.d <= 3)
897
+ .sort((a, b) => a.d - b.d)
898
+ .slice(0, 5);
899
+ console.log(JSON.stringify({
900
+ query,
901
+ found: false,
902
+ suggestions: near.map(({ e }) => ({ name: e.name, module: e.module })),
903
+ }));
904
+ process.exit(1);
905
+ }
906
+
907
+ const trim20 = partial.slice(0, 20);
908
+ console.log(JSON.stringify({
909
+ query,
910
+ found: true,
911
+ exact: exact.map((e) => ({ name: e.name, module: e.module, params: e.params, returns: e.ret })),
912
+ partial: trim20.map((e) => ({ name: e.name, module: e.module, params: e.params, returns: e.ret })),
913
+ partial_truncated: partial.length > 20 ? partial.length - 20 : 0,
914
+ }));
915
+ }
916
+
917
+ function levenshtein(a: string, b: string): number {
918
+ if (a === b) return 0;
919
+ if (!a.length) return b.length;
920
+ if (!b.length) return a.length;
921
+ const prev = new Array(b.length + 1).fill(0).map((_, i) => i);
922
+ for (let i = 1; i <= a.length; i++) {
923
+ let next = [i];
924
+ for (let j = 1; j <= b.length; j++) {
925
+ next[j] = a[i - 1] === b[j - 1]
926
+ ? prev[j - 1]
927
+ : Math.min(prev[j - 1], prev[j], next[j - 1]) + 1;
928
+ }
929
+ for (let j = 0; j <= b.length; j++) prev[j] = next[j];
930
+ }
931
+ return prev[b.length];
932
+ }
933
+
934
+ function cmdFmt(args: string[]): void {
935
+ // --stdin 모드
936
+ if (args.includes("--stdin")) {
937
+ let src = "";
938
+ process.stdin.setEncoding("utf-8");
939
+ process.stdin.on("data", (chunk: string) => { src += chunk; });
940
+ process.stdin.on("end", () => {
941
+ try {
942
+ const formatted = formatFL(src);
943
+ process.stdout.write(formatted);
944
+ } catch (err: any) {
945
+ console.error(`\x1b[31m포맷 오류\x1b[0m ${err.message}`);
946
+ process.exit(1);
947
+ }
948
+ });
949
+ return;
950
+ }
951
+
952
+ // --check 모드
953
+ const checkMode = args.includes("--check");
954
+ const filePaths = args.filter((a) => !a.startsWith("--"));
955
+
956
+ if (filePaths.length === 0) {
957
+ console.error(`\x1b[31m오류\x1b[0m 파일 경로를 지정하세요`);
958
+ process.exit(1);
959
+ }
960
+
961
+ let needsChange = false;
962
+
963
+ for (const filePath of filePaths) {
964
+ const absPath = path.resolve(filePath);
965
+ if (!fs.existsSync(absPath)) {
966
+ console.error(`\x1b[31m오류\x1b[0m 파일을 찾을 수 없습니다: ${filePath}`);
967
+ process.exit(1);
968
+ }
969
+
970
+ const src = fs.readFileSync(absPath, "utf-8");
971
+ let formatted: string;
972
+ try {
973
+ formatted = formatFL(src);
974
+ } catch (err: any) {
975
+ console.error(`\x1b[31m포맷 오류\x1b[0m ${path.basename(absPath)}: ${err.message}`);
976
+ process.exit(1);
977
+ }
978
+
979
+ if (checkMode) {
980
+ if (src !== formatted) {
981
+ console.log(`\x1b[33m변경 필요\x1b[0m ${path.basename(absPath)}`);
982
+ needsChange = true;
983
+ } else {
984
+ console.log(`\x1b[32m이미 포맷됨\x1b[0m ${path.basename(absPath)}`);
985
+ }
986
+ } else {
987
+ if (src !== formatted) {
988
+ fs.writeFileSync(absPath, formatted, "utf-8");
989
+ console.log(`\x1b[32m포맷 완료\x1b[0m ${path.basename(absPath)}`);
990
+ } else {
991
+ console.log(`\x1b[2m변경 없음\x1b[0m ${path.basename(absPath)}`);
992
+ }
993
+ }
994
+ }
995
+
996
+ if (checkMode && needsChange) {
997
+ process.exit(1);
998
+ }
999
+ }
1000
+
1001
+ // ─────────────────────────────────────────
1002
+ // debug 커맨드 (Phase 78)
1003
+ // ─────────────────────────────────────────
1004
+
1005
+ function cmdDebug(filePath: string, stepMode: boolean): void {
1006
+ const absPath = path.resolve(filePath);
1007
+
1008
+ if (!fs.existsSync(absPath)) {
1009
+ console.error(`\x1b[31m오류\x1b[0m 파일을 찾을 수 없습니다: ${filePath}`);
1010
+ process.exit(1);
1011
+ }
1012
+
1013
+ // 디버그 세션 설정
1014
+ const session = new DebugSession();
1015
+ session.enabled = true;
1016
+ session.stepMode = stepMode;
1017
+ setGlobalDebugSession(session);
1018
+
1019
+ console.log(`\x1b[35m[FreeLang Debugger]\x1b[0m ${path.basename(absPath)}${stepMode ? " (step mode)" : ""}`);
1020
+ console.log(`\x1b[2m (break!) 위치에서 중단점 발생\x1b[0m`);
1021
+ console.log(`─────────────────────────────────────────`);
1022
+
1023
+ try {
1024
+ const source = fs.readFileSync(absPath, "utf-8");
1025
+ const tokens = lex(source);
1026
+ const ast = parse(tokens);
1027
+ const interp = new Interpreter();
1028
+ interp.currentFilePath = absPath;
1029
+ interp.debugSession = session;
1030
+ const ctx = interp.interpret(ast);
1031
+
1032
+ if (ctx.lastValue !== null && ctx.lastValue !== undefined) {
1033
+ if (typeof ctx.lastValue === "object") {
1034
+ console.log(JSON.stringify(ctx.lastValue, null, 2));
1035
+ } else {
1036
+ console.log(String(ctx.lastValue));
1037
+ }
1038
+ }
1039
+
1040
+ console.log(`\n\x1b[35m[디버그 완료]\x1b[0m 중단점 ${session.breakLog.length}회 도달`);
1041
+ } catch (err: any) {
1042
+ console.error(formatError(err, undefined, absPath));
1043
+ process.exit(1);
1044
+ }
1045
+ }
1046
+
1047
+ // ─────────────────────────────────────────
1048
+ // ci 커맨드 (Phase 80)
1049
+ // ─────────────────────────────────────────
1050
+
1051
+ async function cmdCi(ciArgs: string[]): Promise<void> {
1052
+ const noFailFast = ciArgs.includes("--no-fail-fast");
1053
+ const filePaths = ciArgs.filter((a) => !a.startsWith("--"));
1054
+
1055
+ let targetFiles: string[];
1056
+
1057
+ if (filePaths.length > 0) {
1058
+ // 특정 파일 지정
1059
+ targetFiles = filePaths.map((f) => path.resolve(f)).filter((f) => fs.existsSync(f));
1060
+ if (targetFiles.length === 0) {
1061
+ console.error(`\x1b[31m오류\x1b[0m 지정한 파일을 찾을 수 없습니다`);
1062
+ process.exit(1);
1063
+ }
1064
+ } else {
1065
+ // 현재 디렉토리의 .fl 파일 전체
1066
+ const cwd = process.cwd();
1067
+ targetFiles = fs.readdirSync(cwd)
1068
+ .filter((f) => f.endsWith(".fl"))
1069
+ .map((f) => path.join(cwd, f));
1070
+ }
1071
+
1072
+ console.log(`\x1b[36m[FreeLang CI]\x1b[0m 파일 ${targetFiles.length}개 fail-fast=${!noFailFast}`);
1073
+ console.log(`─────────────────────────────────────────`);
1074
+
1075
+ const pipeline = createDefaultPipeline(targetFiles, { failFast: !noFailFast });
1076
+ const summary = await pipeline.run();
1077
+
1078
+ console.log(`─────────────────────────────────────────`);
1079
+ const stepCount = summary.steps.length;
1080
+ const passCount = summary.steps.filter((s) => s.passed && !s.skipped).length;
1081
+ const skipCount = summary.steps.filter((s) => s.skipped).length;
1082
+
1083
+ if (summary.passed) {
1084
+ console.log(`\x1b[32m[CI PASS]\x1b[0m ${passCount}/${stepCount} steps (${summary.totalMs}ms)`);
1085
+ } else {
1086
+ console.log(`\x1b[31m[CI FAIL]\x1b[0m ${passCount}/${stepCount} steps (${summary.totalMs}ms, ${skipCount} skipped)`);
1087
+ process.exit(1);
1088
+ }
1089
+ }
1090
+
1091
+ // ─────────────────────────────────────────
1092
+ // doc 커맨드 (Phase 77)
1093
+ // ─────────────────────────────────────────
1094
+
1095
+ function cmdDoc(docArgs: string[]): void {
1096
+ // --dir 모드: 디렉토리 내 모든 .fl 파일 통합 문서화
1097
+ const dirIdx = docArgs.indexOf("--dir");
1098
+ if (dirIdx !== -1) {
1099
+ const dirPath = docArgs[dirIdx + 1];
1100
+ if (!dirPath) {
1101
+ console.error(`\x1b[31m오류\x1b[0m --dir 뒤에 디렉토리 경로를 지정하세요`);
1102
+ process.exit(1);
1103
+ }
1104
+ const absDir = path.resolve(dirPath);
1105
+ if (!fs.existsSync(absDir) || !fs.statSync(absDir).isDirectory()) {
1106
+ console.error(`\x1b[31m오류\x1b[0m 디렉토리를 찾을 수 없습니다: ${dirPath}`);
1107
+ process.exit(1);
1108
+ }
1109
+
1110
+ const flFiles = fs.readdirSync(absDir)
1111
+ .filter((f) => f.endsWith(".fl"))
1112
+ .map((f) => path.join(absDir, f));
1113
+
1114
+ if (flFiles.length === 0) {
1115
+ console.error(`\x1b[33m경고\x1b[0m .fl 파일이 없습니다: ${dirPath}`);
1116
+ return;
1117
+ }
1118
+
1119
+ const allEntries: import("./doc-extractor").DocEntry[] = [];
1120
+ for (const filePath of flFiles) {
1121
+ const src = fs.readFileSync(filePath, "utf-8");
1122
+ allEntries.push(...extractDocs(src));
1123
+ }
1124
+
1125
+ const title = path.basename(absDir) + " API 문서";
1126
+ const md = renderMarkdown(allEntries, title);
1127
+ const outIdx = docArgs.indexOf("-o");
1128
+ if (outIdx !== -1 && docArgs[outIdx + 1]) {
1129
+ const outPath = path.resolve(docArgs[outIdx + 1]);
1130
+ fs.writeFileSync(outPath, md, "utf-8");
1131
+ console.log(`\x1b[32m문서 저장됨\x1b[0m ${outPath} (${allEntries.length}개 항목)`);
1132
+ } else {
1133
+ process.stdout.write(md);
1134
+ }
1135
+ return;
1136
+ }
1137
+
1138
+ // 단일 파일 모드
1139
+ const filePaths = docArgs.filter((a) => !a.startsWith("-"));
1140
+ if (filePaths.length === 0) {
1141
+ console.error(`\x1b[31m오류\x1b[0m 파일 경로를 지정하세요`);
1142
+ process.exit(1);
1143
+ }
1144
+ const filePath = filePaths[0];
1145
+ const absPath = path.resolve(filePath);
1146
+ if (!fs.existsSync(absPath)) {
1147
+ console.error(`\x1b[31m오류\x1b[0m 파일을 찾을 수 없습니다: ${filePath}`);
1148
+ process.exit(1);
1149
+ }
1150
+
1151
+ const src = fs.readFileSync(absPath, "utf-8");
1152
+ const entries = extractDocs(src);
1153
+ const title = path.basename(absPath, ".fl") + " API 문서";
1154
+ const md = renderMarkdown(entries, title);
1155
+
1156
+ const outIdx = docArgs.indexOf("-o");
1157
+ if (outIdx !== -1 && docArgs[outIdx + 1]) {
1158
+ const outPath = path.resolve(docArgs[outIdx + 1]);
1159
+ fs.writeFileSync(outPath, md, "utf-8");
1160
+ console.log(`\x1b[32m문서 저장됨\x1b[0m ${outPath} (${entries.length}개 항목)`);
1161
+ } else {
1162
+ process.stdout.write(md);
1163
+ }
1164
+ }
1165
+
1166
+ // ─────────────────────────────────────────
1167
+ // build 커맨드 (Phase 8)
1168
+ // ─────────────────────────────────────────
1169
+
1170
+ function cmdBuild(buildArgs: string[]): void {
1171
+ const isOci = buildArgs.includes("--oci");
1172
+ const isStatic = buildArgs.includes("--static");
1173
+
1174
+ if (isStatic) {
1175
+ // Static HTML export — starts serve, GETs each route, saves HTML, stops serve.
1176
+ // Supports both (println ...) style pages and full [PAGE]/(page ...) blocks via app-router.
1177
+ // Usage: fl build --static [--app app/] [--out dist/] [--port 43099]
1178
+ const appIdx = buildArgs.indexOf("--app");
1179
+ const outIdx = buildArgs.indexOf("--out");
1180
+ const portIdx = buildArgs.indexOf("--port");
1181
+ const appDir = appIdx !== -1 ? buildArgs[appIdx + 1] : "app";
1182
+ const outDir = outIdx !== -1 ? buildArgs[outIdx + 1] : "dist";
1183
+ const port = portIdx !== -1 ? parseInt(buildArgs[portIdx + 1], 10) : 43099;
1184
+
1185
+ const absApp = path.resolve(appDir);
1186
+ const absOut = path.resolve(outDir);
1187
+ if (!fs.existsSync(absApp)) {
1188
+ console.error(`build.error event=app_not_found path=${appDir}`);
1189
+ process.exit(1);
1190
+ }
1191
+
1192
+ console.log(`build.start app=${appDir} out=${outDir} port=${port}`);
1193
+ fs.mkdirSync(absOut, { recursive: true });
1194
+
1195
+ // Walk app/ and collect page.fl routes.
1196
+ // Dynamic routes ([slug]): look for generate-static-params.fl in the same
1197
+ // dir. If present, run it and expand each param set into a concrete route.
1198
+ // The params file should `(println <json>)` a JSON array of objects:
1199
+ // [{"slug":"post-1"},{"slug":"post-2"}]
1200
+ const pages: { filePath: string; route: string }[] = [];
1201
+
1202
+ function expandDynamicParams(dir: string, paramName: string): any[] {
1203
+ const paramsFile = path.join(dir, "generate-static-params.fl");
1204
+ if (!fs.existsSync(paramsFile)) return [];
1205
+ try {
1206
+ const cwdBootstrap2 = path.resolve(process.cwd(), "bootstrap.js");
1207
+ const bs = fs.existsSync(cwdBootstrap2) ? cwdBootstrap2 : path.resolve(__dirname, "bootstrap.js");
1208
+ const { execSync } = require("child_process");
1209
+ const out = execSync(`node "${bs}" run "${paramsFile}"`, { encoding: "utf-8", stdio: ["ignore", "pipe", "pipe"] });
1210
+ // Find a JSON array in stdout
1211
+ const m = out.match(/\[[\s\S]*\]/);
1212
+ if (!m) return [];
1213
+ const parsed = JSON.parse(m[0]);
1214
+ return Array.isArray(parsed) ? parsed : [];
1215
+ } catch (err: any) {
1216
+ console.log(`build.params_error dir=${dir} err=${(err.message || String(err)).split("\n")[0]}`);
1217
+ return [];
1218
+ }
1219
+ }
1220
+
1221
+ function walk(dir: string, routeBase: string): void {
1222
+ const entries = fs.readdirSync(dir, { withFileTypes: true });
1223
+ for (const e of entries) {
1224
+ const full = path.join(dir, e.name);
1225
+ if (e.isDirectory()) {
1226
+ if (e.name.startsWith("[") && e.name.endsWith("]")) {
1227
+ const paramName = e.name.slice(1, -1);
1228
+ const params = expandDynamicParams(full, paramName);
1229
+ const pageFile = path.join(full, "page.fl");
1230
+ if (params.length === 0) {
1231
+ console.log(`build.skip reason=dynamic_no_params path=/${path.relative(absApp, full)} param=${paramName}`);
1232
+ continue;
1233
+ }
1234
+ if (!fs.existsSync(pageFile)) {
1235
+ console.log(`build.skip reason=dynamic_no_page path=/${path.relative(absApp, full)}`);
1236
+ continue;
1237
+ }
1238
+ for (const p of params) {
1239
+ const value = p && typeof p === "object" ? p[paramName] : null;
1240
+ if (!value) continue;
1241
+ pages.push({ filePath: pageFile, route: routeBase + "/" + String(value) });
1242
+ }
1243
+ // Also walk into for any non-page files (unlikely but possible)
1244
+ continue;
1245
+ }
1246
+ if (e.name === "api") continue;
1247
+ walk(full, routeBase + "/" + e.name);
1248
+ } else if (e.name === "page.fl") {
1249
+ pages.push({ filePath: full, route: routeBase || "/" });
1250
+ }
1251
+ }
1252
+ }
1253
+ walk(absApp, "");
1254
+
1255
+ // Special file: app/not-found.fl → dist/404.html (most CDNs serve this
1256
+ // for unmatched routes automatically). Generated via the same strategies.
1257
+ const notFoundFile = path.join(absApp, "not-found.fl");
1258
+ if (fs.existsSync(notFoundFile)) {
1259
+ pages.push({ filePath: notFoundFile, route: "/__404__" });
1260
+ }
1261
+
1262
+ if (pages.length === 0) {
1263
+ console.log(`build.error event=no_pages app=${appDir}`);
1264
+ return;
1265
+ }
1266
+
1267
+ // Start `serve` in the background, then HTTP GET each route.
1268
+ const { spawn } = require("child_process");
1269
+ const http = require("http");
1270
+ const cwdBootstrap = path.resolve(process.cwd(), "bootstrap.js");
1271
+ const bootstrap = fs.existsSync(cwdBootstrap)
1272
+ ? cwdBootstrap
1273
+ : path.resolve(__dirname, "bootstrap.js");
1274
+
1275
+ const serveProc = spawn(
1276
+ "node",
1277
+ [bootstrap, "serve", "--app", absApp, "--port", String(port)],
1278
+ { stdio: ["ignore", "pipe", "pipe"] },
1279
+ );
1280
+
1281
+ // Wait for server to be ready (poll /__probe or any known route with timeout)
1282
+ const waitForServer = async (): Promise<boolean> => {
1283
+ for (let i = 0; i < 30; i++) {
1284
+ const ready = await new Promise<boolean>((resolve) => {
1285
+ const req = http.get(
1286
+ { host: "localhost", port, path: "/", timeout: 500 },
1287
+ (res: any) => { res.destroy(); resolve(true); },
1288
+ );
1289
+ req.on("error", () => resolve(false));
1290
+ req.on("timeout", () => { req.destroy(); resolve(false); });
1291
+ });
1292
+ if (ready) return true;
1293
+ await new Promise((r) => setTimeout(r, 200));
1294
+ }
1295
+ return false;
1296
+ };
1297
+
1298
+ const fetchRoute = (route: string): Promise<string> =>
1299
+ new Promise((resolve, reject) => {
1300
+ const req = http.get(
1301
+ { host: "localhost", port, path: route, timeout: 5000 },
1302
+ (res: any) => {
1303
+ let buf = "";
1304
+ res.on("data", (c: any) => { buf += c.toString(); });
1305
+ res.on("end", () => {
1306
+ if (res.statusCode && res.statusCode >= 400) {
1307
+ reject(new Error(`HTTP ${res.statusCode}`));
1308
+ } else {
1309
+ resolve(buf);
1310
+ }
1311
+ });
1312
+ },
1313
+ );
1314
+ req.on("error", reject);
1315
+ req.on("timeout", () => { req.destroy(); reject(new Error("timeout")); });
1316
+ });
1317
+
1318
+ const runPage = (p: { filePath: string; route: string }): string | null => {
1319
+ try {
1320
+ const { execSync } = require("child_process");
1321
+ const out = execSync(`node "${bootstrap}" run "${p.filePath}"`, {
1322
+ encoding: "utf-8",
1323
+ stdio: ["ignore", "pipe", "pipe"],
1324
+ });
1325
+ const m = out.match(/<!DOCTYPE html[\s\S]*?<\/html>/i) || out.match(/<html[\s\S]*?<\/html>/i);
1326
+ return m ? m[0] : null;
1327
+ } catch { return null; }
1328
+ };
1329
+
1330
+ const isUseful = (html: string | null | undefined): boolean => {
1331
+ if (!html) return false;
1332
+ const t = html.trim();
1333
+ if (t.length < 50) return false;
1334
+ if (t.startsWith("Internal Server Error")) return false;
1335
+ return true;
1336
+ };
1337
+
1338
+ // Concurrency: --concurrency=N (default 8)
1339
+ const concIdx = buildArgs.indexOf("--concurrency");
1340
+ const concurrency = Math.max(1, concIdx !== -1 ? parseInt(buildArgs[concIdx + 1] || "8", 10) : 8);
1341
+
1342
+ (async () => {
1343
+ const ready = await waitForServer();
1344
+ const t0 = Date.now();
1345
+ let ok = 0;
1346
+ let fail = 0;
1347
+
1348
+ const renderOne = async (p: { filePath: string; route: string }): Promise<void> => {
1349
+ let html: string | null = null;
1350
+ if (ready) {
1351
+ try {
1352
+ const res = await fetchRoute(p.route);
1353
+ if (isUseful(res)) html = res;
1354
+ } catch { /* fall through */ }
1355
+ }
1356
+ if (!html) {
1357
+ const out = runPage(p);
1358
+ if (isUseful(out)) html = out;
1359
+ }
1360
+ if (html) {
1361
+ const outPath = p.route === "/__404__"
1362
+ ? path.join(absOut, "404.html")
1363
+ : path.join(absOut, p.route === "/" ? "index.html" : p.route.slice(1) + "/index.html");
1364
+ fs.mkdirSync(path.dirname(outPath), { recursive: true });
1365
+ fs.writeFileSync(outPath, html);
1366
+ console.log(`build.page route=${p.route === "/__404__" ? "/404" : p.route} ok=true file=${path.relative(process.cwd(), outPath)} bytes=${html.length}`);
1367
+ ok++;
1368
+ } else {
1369
+ console.log(`build.page route=${p.route} ok=false`);
1370
+ fail++;
1371
+ }
1372
+ };
1373
+
1374
+ // Render pages in parallel batches of size `concurrency`.
1375
+ for (let i = 0; i < pages.length; i += concurrency) {
1376
+ const batch = pages.slice(i, i + concurrency);
1377
+ await Promise.all(batch.map(renderOne));
1378
+ }
1379
+
1380
+ serveProc.kill();
1381
+ const ms = Date.now() - t0;
1382
+ console.log(`build.done ok=${ok} fail=${fail} out=${outDir} ms=${ms} concurrency=${concurrency}`);
1383
+ if (fail > 0) process.exit(1);
1384
+ process.exit(0);
1385
+ })();
1386
+ return;
1387
+ }
1388
+
1389
+ if (isOci) {
1390
+ // OCI 빌드 모드
1391
+ const fileIdx = buildArgs.indexOf("--oci") + 1;
1392
+ const appFile = buildArgs[fileIdx];
1393
+ const tagIdx = buildArgs.indexOf("--tag");
1394
+ const tag = tagIdx !== -1 ? buildArgs[tagIdx + 1] : "my-app:latest";
1395
+ const registryIdx = buildArgs.indexOf("--registry");
1396
+ const registry = registryIdx !== -1 ? buildArgs[registryIdx + 1] : undefined;
1397
+
1398
+ if (!appFile) {
1399
+ console.error(`\x1b[31m오류\x1b[0m app 파일을 지정하세요: fl build --oci <app.fl> --tag <tag>`);
1400
+ process.exit(1);
1401
+ }
1402
+
1403
+ const absPath = path.resolve(appFile);
1404
+ if (!fs.existsSync(absPath)) {
1405
+ console.error(`\x1b[31m오류\x1b[0m 파일을 찾을 수 없습니다: ${appFile}`);
1406
+ process.exit(1);
1407
+ }
1408
+
1409
+ console.log(`\x1b[36m[OCI Build]\x1b[0m ${path.basename(appFile)} → ${tag}`);
1410
+
1411
+ // v9-oci.fl 실행
1412
+ const ociScriptPath = path.resolve(__dirname, "../vpm/v9-oci.fl");
1413
+ if (!fs.existsSync(ociScriptPath)) {
1414
+ console.error(`\x1b[31m오류\x1b[0m v9-oci.fl을 찾을 수 없습니다`);
1415
+ process.exit(1);
1416
+ }
1417
+
1418
+ const { execSync } = require("child_process");
1419
+ try {
1420
+ const cmd = registry
1421
+ ? `node ${path.resolve(__dirname, "../src/cli.js")} run ${ociScriptPath} build ${appFile} ${tag} ${registry}`
1422
+ : `node ${path.resolve(__dirname, "../src/cli.js")} run ${ociScriptPath} build ${appFile} ${tag}`;
1423
+
1424
+ console.log(`\x1b[2m Command: ${cmd}\x1b[0m`);
1425
+ execSync(cmd, { stdio: "inherit" });
1426
+ console.log(`\x1b[32m[OK]\x1b[0m OCI 빌드 완료: ${tag}`);
1427
+ } catch (err: any) {
1428
+ console.error(`\x1b[31m[Error]\x1b[0m OCI 빌드 실패: ${err.message}`);
1429
+ process.exit(1);
1430
+ }
1431
+ } else {
1432
+ console.error(`\x1b[31m오류\x1b[0m --oci 플래그를 지정하세요`);
1433
+ console.log(`\n사용법:\n fl build --oci <app.fl> --tag <tag> [--registry <url>]`);
1434
+ process.exit(1);
1435
+ }
1436
+ }
1437
+
1438
+ // ─────────────────────────────────────────
1439
+ // registry 커맨드 (Phase 7)
1440
+ // ─────────────────────────────────────────
1441
+
1442
+ function cmdRegistry(registryArgs: string[]): void {
1443
+ const subCmd = registryArgs[0];
1444
+
1445
+ if (subCmd === "start") {
1446
+ // 레지스트리 서버 시작
1447
+ const portIdx = registryArgs.indexOf("--port");
1448
+ const port = portIdx !== -1 && registryArgs[portIdx + 1]
1449
+ ? parseInt(registryArgs[portIdx + 1], 10)
1450
+ : 4873;
1451
+
1452
+ if (isNaN(port) || port < 1024 || port > 65535) {
1453
+ console.error(`\x1b[31m오류\x1b[0m 유효하지 않은 포트: ${port}`);
1454
+ process.exit(1);
1455
+ }
1456
+
1457
+ console.log(`\x1b[36m[Registry]\x1b[0m v9 패키지 레지스트리 시작 (포트 ${port})`);
1458
+ console.log(`\x1b[36m[Registry]\x1b[0m http://localhost:${port}/`);
1459
+
1460
+ // 레지스트리 서버를 별도 프로세스로 실행
1461
+ const registryPath = path.resolve(__dirname, "../vpm/registry-server.fl");
1462
+ if (!fs.existsSync(registryPath)) {
1463
+ console.error(`\x1b[31m오류\x1b[0m registry-server.fl을 찾을 수 없습니다: ${registryPath}`);
1464
+ process.exit(1);
1465
+ }
1466
+
1467
+ // registry-server.fl을 v9-run으로 실행
1468
+ const { execSync } = require("child_process");
1469
+ try {
1470
+ process.env.REGISTRY_PORT = String(port);
1471
+ execSync(`node ${path.resolve(__dirname, "../src/cli.js")} run ${registryPath}`, {
1472
+ stdio: "inherit",
1473
+ env: { ...process.env, REGISTRY_PORT: String(port) }
1474
+ });
1475
+ } catch (err: any) {
1476
+ console.error(`\x1b[31m레지스트리 시작 오류:\x1b[0m ${err.message}`);
1477
+ process.exit(1);
1478
+ }
1479
+ } else if (subCmd === "status") {
1480
+ // 레지스트리 상태 확인
1481
+ const portIdx = registryArgs.indexOf("--port");
1482
+ const port = portIdx !== -1 && registryArgs[portIdx + 1]
1483
+ ? parseInt(registryArgs[portIdx + 1], 10)
1484
+ : 4873;
1485
+
1486
+ try {
1487
+ const http = require("http");
1488
+ const req = http.get(`http://localhost:${port}/-/all`, (res: any) => {
1489
+ if (res.statusCode === 200) {
1490
+ console.log(`\x1b[32m[OK]\x1b[0m 레지스트리 정상 운영 중 (포트 ${port})`);
1491
+ process.exit(0);
1492
+ } else {
1493
+ console.error(`\x1b[31m오류\x1b[0m 레지스트리 응답 이상 (상태: ${res.statusCode})`);
1494
+ process.exit(1);
1495
+ }
1496
+ });
1497
+ req.on("error", (err: any) => {
1498
+ console.error(`\x1b[31m오류\x1b[0m 레지스트리 연결 실패: ${err.message}`);
1499
+ process.exit(1);
1500
+ });
1501
+ req.setTimeout(5000, () => {
1502
+ req.destroy();
1503
+ console.error(`\x1b[31m오류\x1b[0m 레지스트리 타임아웃`);
1504
+ process.exit(1);
1505
+ });
1506
+ } catch (err: any) {
1507
+ console.error(`\x1b[31m오류\x1b[0m ${err.message}`);
1508
+ process.exit(1);
1509
+ }
1510
+ } else {
1511
+ console.error(`\x1b[31m알 수 없는 서브커맨드:\x1b[0m registry ${subCmd}`);
1512
+ console.log(`\n사용법:\n fl registry start [--port 4873]\n fl registry status [--port 4873]`);
1513
+ process.exit(1);
1514
+ }
1515
+ }
1516
+
1517
+ // Y5: 플러그인 설치 커맨드
1518
+ function cmdInstall(args: string[]): void {
1519
+ const pluginName = args[0];
1520
+ if (!pluginName) {
1521
+ console.error(`\x1b[31m오류\x1b[0m 플러그인 이름을 지정하세요: install <plugin-name>`);
1522
+ process.exit(1);
1523
+ }
1524
+
1525
+ const homeDir = require("os").homedir();
1526
+ const pluginsDir = path.resolve(homeDir, ".fl", "plugins");
1527
+
1528
+ // ~/.fl/plugins 디렉토리 생성
1529
+ if (!fs.existsSync(pluginsDir)) {
1530
+ fs.mkdirSync(pluginsDir, { recursive: true });
1531
+ console.log(`\x1b[36m[Y5]\x1b[0m 플러그인 디렉토리 생성: ${pluginsDir}`);
1532
+ }
1533
+
1534
+ // 현재 프로젝트에서 plugins/NAME.fl 찾기 또는 내장 stdlib에서 찾기
1535
+ const localPath = path.resolve(process.cwd(), "plugins", pluginName + ".fl");
1536
+ const stdlibPath = path.resolve(process.cwd(), "self/stdlib", pluginName + ".fl");
1537
+ const installedPath = path.resolve(pluginsDir, pluginName + ".fl");
1538
+
1539
+ let sourceFile: string | null = null;
1540
+
1541
+ if (fs.existsSync(localPath)) {
1542
+ sourceFile = localPath;
1543
+ console.log(`\x1b[36m[Y5]\x1b[0m 로컬 플러그인 찾음: ${localPath}`);
1544
+ } else if (fs.existsSync(stdlibPath)) {
1545
+ sourceFile = stdlibPath;
1546
+ console.log(`\x1b[36m[Y5]\x1b[0m 내장 stdlib 플러그인 찾음: ${stdlibPath}`);
1547
+ } else {
1548
+ // TODO: 원격 레지스트리에서 다운로드
1549
+ console.error(`\x1b[31m오류\x1b[0m 플러그인을 찾을 수 없습니다: ${pluginName}`);
1550
+ console.log(` 시도한 경로:`);
1551
+ console.log(` - ${localPath}`);
1552
+ console.log(` - ${stdlibPath}`);
1553
+ process.exit(1);
1554
+ }
1555
+
1556
+ // 플러그인 복사
1557
+ try {
1558
+ const content = fs.readFileSync(sourceFile, "utf-8");
1559
+ fs.writeFileSync(installedPath, content, "utf-8");
1560
+ console.log(`\x1b[32m✓\x1b[0m 플러그인 설치 완료: ${installedPath}`);
1561
+ console.log(`\x1b[2m 사용: (use ${pluginName})\x1b[0m`);
1562
+ } catch (err: any) {
1563
+ console.error(`\x1b[31m오류\x1b[0m 설치 실패: ${err.message}`);
1564
+ process.exit(1);
1565
+ }
1566
+ }
1567
+
1568
+ function cmdPublish(args: string[]): void {
1569
+ const filePath = args[0];
1570
+ if (!filePath) {
1571
+ console.error(`\x1b[31m오류\x1b[0m 플러그인 파일을 지정하세요: publish <plugin-file.fl>`);
1572
+ process.exit(1);
1573
+ }
1574
+
1575
+ if (!fs.existsSync(filePath)) {
1576
+ console.error(`\x1b[31m오류\x1b[0m 파일을 찾을 수 없습니다: ${filePath}`);
1577
+ process.exit(1);
1578
+ }
1579
+
1580
+ const content = fs.readFileSync(filePath, "utf-8");
1581
+ const nameMatch = content.match(/^;; plugin:\s*(.+)$/m);
1582
+ const versionMatch = content.match(/^;; version:\s*(.+)$/m);
1583
+
1584
+ if (!nameMatch) {
1585
+ console.error(`\x1b[31m오류\x1b[0m 플러그인 메타 블록이 없습니다. 파일 상단에 다음을 추가하세요:`);
1586
+ console.log(` ;; plugin: <name>`);
1587
+ console.log(` ;; version: 1.0.0`);
1588
+ console.log(` ;; depends:`);
1589
+ process.exit(1);
1590
+ }
1591
+
1592
+ const pluginName = nameMatch[1].trim();
1593
+ const version = versionMatch ? versionMatch[1].trim() : "1.0.0";
1594
+
1595
+ console.log(`\x1b[36m[Y5]\x1b[0m 배포 중: ${pluginName} (v${version})`);
1596
+
1597
+ try {
1598
+ const { execSync } = require("child_process");
1599
+ const tmpDir = path.resolve("/tmp", `fl-publish-${Date.now()}`);
1600
+ fs.mkdirSync(tmpDir, { recursive: true });
1601
+
1602
+ // 저장소 clone
1603
+ execSync("git clone https://gogs.dclub.kr/kim/fl-plugins.git .", {
1604
+ cwd: tmpDir,
1605
+ stdio: "pipe",
1606
+ });
1607
+
1608
+ // 플러그인 파일 복사
1609
+ fs.copyFileSync(filePath, path.resolve(tmpDir, `${pluginName}.fl`));
1610
+
1611
+ // git add, commit, push
1612
+ execSync(`git add ${pluginName}.fl`, { cwd: tmpDir, stdio: "pipe" });
1613
+ execSync(
1614
+ `git -c user.name="FreeLang CLI" -c user.email="cli@freelang.dev" commit -m "Add plugin: ${pluginName} v${version}"`,
1615
+ { cwd: tmpDir, stdio: "pipe" }
1616
+ );
1617
+ execSync("git push origin master", { cwd: tmpDir, stdio: "pipe" });
1618
+
1619
+ console.log(`\x1b[32m✓\x1b[0m 플러그인 '${pluginName}' 게시 완료`);
1620
+ console.log(`\x1b[2m 저장소: https://gogs.dclub.kr/kim/fl-plugins\x1b[0m`);
1621
+
1622
+ // 임시 디렉토리 정리
1623
+ fs.rmSync(tmpDir, { recursive: true, force: true });
1624
+ } catch (err: any) {
1625
+ console.error(`\x1b[31m오류\x1b[0m 배포 실패: ${err.message}`);
1626
+ process.exit(1);
1627
+ }
1628
+ }
1629
+
1630
+ function cmdServe(args: string[]): void {
1631
+ // Phase 3: freelang serve [appDir] [--app app] [--port 3000] [--mode ssr|isr|ssg]
1632
+ let appDir = "app";
1633
+ let port: number | null = null;
1634
+ let renderMode: "ssr" | "isr" | "ssg" = "ssr";
1635
+ let positionalIdx = 0;
1636
+
1637
+ for (let i = 0; i < args.length; i++) {
1638
+ if (args[i] === "--app" && args[i + 1]) {
1639
+ appDir = args[++i];
1640
+ } else if (args[i] === "--port" && args[i + 1]) {
1641
+ port = parseInt(args[++i], 10);
1642
+ } else if (args[i] === "--mode" && args[i + 1]) {
1643
+ const m = args[++i] as "ssr" | "isr" | "ssg";
1644
+ if (["ssr", "isr", "ssg"].includes(m)) {
1645
+ renderMode = m;
1646
+ }
1647
+ } else if (!args[i].startsWith("--")) {
1648
+ // 첫 번째 positional argument는 appDir
1649
+ if (positionalIdx === 0) {
1650
+ appDir = args[i];
1651
+ }
1652
+ positionalIdx++;
1653
+ }
1654
+ }
1655
+
1656
+ // 포트 우선순위: CLI --port > ENV PORT > fl.config.json > 기본값 3000
1657
+ if (port === null) {
1658
+ if (process.env.PORT) {
1659
+ port = parseInt(process.env.PORT, 10);
1660
+ } else {
1661
+ // fl.config.json 읽기 시도
1662
+ try {
1663
+ const fs = require("fs") as typeof import("fs");
1664
+ const path = require("path") as typeof import("path");
1665
+ const configPath = path.join(process.cwd(), "fl.config.json");
1666
+ if (fs.existsSync(configPath)) {
1667
+ const cfg = JSON.parse(fs.readFileSync(configPath, "utf-8")) as { port?: number };
1668
+ if (cfg.port) port = cfg.port;
1669
+ }
1670
+ } catch (_e) {
1671
+ // Silently ignore errors reading config file
1672
+ }
1673
+ if (port === null) port = 3000;
1674
+ }
1675
+ }
1676
+
1677
+ const server = new WebServer({ appDir, port, renderMode });
1678
+ const interp = new Interpreter();
1679
+ server.setInterpreter(interp);
1680
+
1681
+ server
1682
+ .start()
1683
+ .then((msg) => {
1684
+ console.log(msg);
1685
+ // Keep process alive
1686
+ setInterval(() => {}, 10000).unref();
1687
+ })
1688
+ .catch((err) => {
1689
+ console.error(`\x1b[31m서버 오류\x1b[0m ${err.message}`);
1690
+ process.exit(1);
1691
+ });
1692
+ }
1693
+
1694
+ const C = {
1695
+ bold: (s: string) => `\x1b[1m${s}\x1b[0m`,
1696
+ cyan: (s: string) => `\x1b[36m${s}\x1b[0m`,
1697
+ green: (s: string) => `\x1b[32m${s}\x1b[0m`,
1698
+ dim: (s: string) => `\x1b[2m${s}\x1b[0m`,
1699
+ red: (s: string) => `\x1b[31m${s}\x1b[0m`,
1700
+ yellow:(s: string) => `\x1b[33m${s}\x1b[0m`,
1701
+ };
1702
+
1703
+ const FL_VERSION = "11.1.1-dev";
1704
+
1705
+ function printUsage(errCmd?: string): void {
1706
+ const b = C.bold, c = C.cyan, g = C.green, d = C.dim;
1707
+ console.log([
1708
+ "",
1709
+ `${b("FreeLang")} ${c("v" + FL_VERSION)} ${d("AI-Native Lisp · Self-Hosting · 500+ stdlib")}`,
1710
+ "",
1711
+ `${b("사용법:")} freelang <커맨드> [옵션]`,
1712
+ "",
1713
+ `${c("── 실행 ───────────────────────────────────────────────")}`,
1714
+ ` ${g("run")} <file.fl> [-- args...] 파일 실행`,
1715
+ ` ${g("run")} <file.fl> --watch 저장 시 자동 재실행`,
1716
+ ` ${g("repl")} 대화형 REPL`,
1717
+ ` ${g("check")} <file.fl> 문법 검사 (실행 없음)`,
1718
+ "",
1719
+ `${c("── 개발 ───────────────────────────────────────────────")}`,
1720
+ ` ${g("watch")} <file.fl> 파일 변경 감지 재실행`,
1721
+ ` ${g("debug")} <file.fl> 디버그 모드 (break! 지원)`,
1722
+ ` ${g("debug")} <file.fl> --step 스텝 모드`,
1723
+ ` ${g("fmt")} <file.fl> 코드 포맷 (인플레이스)`,
1724
+ ` ${g("fmt")} --check <file.fl> 포맷 검사 (exit 1 if dirty)`,
1725
+ ` ${g("fmt")} --stdin stdin → stdout 포맷`,
1726
+ "",
1727
+ `${c("── 문서 / 탐색 ────────────────────────────────────────")}`,
1728
+ ` ${g("ls-fns")} [키워드] stdlib 함수 목록 (키워드 필터)`,
1729
+ ` ${g("fn-doc")} <이름> 함수 시그니처 + 설명`,
1730
+ ` ${g("doc")} <file.fl> [-o out.md] Markdown 문서 생성`,
1731
+ ` ${g("doc")} --dir <dir> 디렉토리 통합 문서화`,
1732
+ "",
1733
+ `${c("── 빌드 / 배포 ────────────────────────────────────────")}`,
1734
+ ` ${g("compile")} <file.fl> [-o out.js] JS로 컴파일 (AOT)`,
1735
+ ` ${g("build")} --oci <app.fl> --tag <tag> OCI 이미지 빌드`,
1736
+ ` ${g("serve")} [--port 3000] 웹 서버 시작`,
1737
+ ` ${g("ci")} [file.fl] CI 검사 실행`,
1738
+ "",
1739
+ `${c("── 프로세스 관리 ──────────────────────────────────────")}`,
1740
+ ` ${g("ps")} 실행 중인 fl 프로세스 목록`,
1741
+ ` ${g("stop")} [file.fl] 프로세스 종료 (없으면 전체)`,
1742
+ "",
1743
+ `${c("── 에러 분석 ──────────────────────────────────────────")}`,
1744
+ ` ${g("errors")} [N] 미발견 함수 TOP N 분석 (기본 20)`,
1745
+ ` ${g("errors-clear")} 에러 로그 초기화`,
1746
+ "",
1747
+ `${c("── 기타 ───────────────────────────────────────────────")}`,
1748
+ ` ${g("--version")} / ${g("-v")} 버전 출력`,
1749
+ ` ${g("--help")} / ${g("-h")} 이 도움말`,
1750
+ "",
1751
+ `${b("예제:")}`,
1752
+ ` freelang run app.fl`,
1753
+ ` freelang run app.fl --watch`,
1754
+ ` freelang run app.fl -- arg1 arg2`,
1755
+ ` freelang repl`,
1756
+ ` freelang check app.fl`,
1757
+ ` freelang ls-fns http`,
1758
+ ` freelang fn-doc json_parse`,
1759
+ ` freelang compile app.fl -o app.js`,
1760
+ ` freelang ps`,
1761
+ ` freelang stop app.fl`,
1762
+ ` freelang errors`,
1763
+ "",
1764
+ ].join("\n"));
1765
+
1766
+ if (errCmd) {
1767
+ // 유사 커맨드 제안
1768
+ const known = ["run","repl","check","watch","debug","fmt","ls-fns","fn-doc",
1769
+ "doc","compile","build","serve","ci","version","help"];
1770
+ const suggest = known.find(k =>
1771
+ k.startsWith(errCmd[0]) ||
1772
+ Math.abs(k.length - errCmd.length) <= 2 &&
1773
+ [...errCmd].filter((c,i) => k[i] === c).length >= Math.min(k.length,errCmd.length) - 2
1774
+ );
1775
+ console.error(`${C.red("오류")} 알 수 없는 커맨드: ${C.bold(errCmd)}${suggest ? `\n 혹시 ${C.green("freelang " + suggest)} ?` : ""}`);
1776
+ process.exit(1);
1777
+ }
1778
+ }
1779
+
1780
+ function printSubHelp(cmd: string): void {
1781
+ const g = C.green, d = C.dim, b = C.bold;
1782
+ const helps: Record<string, string[]> = {
1783
+ run: [
1784
+ `${b("freelang run")} — 파일 실행`,
1785
+ "",
1786
+ ` freelang run <file.fl> 실행`,
1787
+ ` freelang run <file.fl> --watch 저장 시 자동 재실행`,
1788
+ ` freelang run <file.fl> -- a b 스크립트에 인자 전달 ($__argv__)`,
1789
+ "",
1790
+ `${d("예제:")}`,
1791
+ ` freelang run app.fl`,
1792
+ ` freelang run server.fl --watch`,
1793
+ ` freelang run cli.fl -- input.txt`,
1794
+ ],
1795
+ repl: [
1796
+ `${b("freelang repl")} — 대화형 REPL`,
1797
+ "",
1798
+ ` (+ 1 2) 표현식 평가`,
1799
+ ` (defn f [x] ...) 함수 정의 (세션 유지)`,
1800
+ ` :stack 콜 스택 출력`,
1801
+ ` :env 바인딩 목록`,
1802
+ ` :exit / Ctrl+D 종료`,
1803
+ ],
1804
+ check: [
1805
+ `${b("freelang check")} — 문법 검사 (실행 없음)`,
1806
+ "",
1807
+ ` freelang check <file.fl>`,
1808
+ "",
1809
+ ` 파싱 오류 발생 시 라인:컬럼 + 괄호 힌트 표시`,
1810
+ ],
1811
+ fmt: [
1812
+ `${b("freelang fmt")} — 코드 포맷`,
1813
+ "",
1814
+ ` freelang fmt <file.fl> 인플레이스 포맷`,
1815
+ ` freelang fmt --check <file.fl> 포맷 필요 여부 검사 (exit 1)`,
1816
+ ` freelang fmt --stdin stdin → stdout`,
1817
+ "",
1818
+ `${d("CI 활용:")}`,
1819
+ ` freelang fmt --check src/*.fl`,
1820
+ ],
1821
+ compile: [
1822
+ `${b("freelang compile")} — JS로 AOT 컴파일`,
1823
+ "",
1824
+ ` freelang compile <file.fl> → file.fl.out.js`,
1825
+ ` freelang compile <file.fl> -o app.js → app.js`,
1826
+ "",
1827
+ ` 생성된 .js는 Node.js / Bun으로 직접 실행 가능`,
1828
+ ` 실행 시 FreeLang 불필요 (독립 배포)`,
1829
+ ],
1830
+ "ls-fns": [
1831
+ `${b("freelang ls-fns")} — stdlib 함수 탐색`,
1832
+ "",
1833
+ ` freelang ls-fns 전체 목록 (500+)`,
1834
+ ` freelang ls-fns http "http" 포함 함수만`,
1835
+ ` freelang ls-fns json json 관련`,
1836
+ ],
1837
+ "fn-doc": [
1838
+ `${b("freelang fn-doc")} — 함수 상세 문서`,
1839
+ "",
1840
+ ` freelang fn-doc json_parse`,
1841
+ ` freelang fn-doc str_split`,
1842
+ ],
1843
+ };
1844
+ const lines = helps[cmd];
1845
+ if (lines) {
1846
+ console.log("\n" + lines.join("\n") + "\n");
1847
+ } else {
1848
+ printUsage();
1849
+ }
1850
+ }
1851
+
1852
+
1853
+
1854
+ const args = process.argv.slice(2);
1855
+ const cmd = args[0];
1856
+
1857
+ switch (cmd) {
1858
+ case "run": {
1859
+ if (args[1] === "--help" || args[1] === "-h") { printSubHelp("run"); break; }
1860
+ const filePath = args[1];
1861
+ if (!filePath) { printSubHelp("run"); process.exit(1); }
1862
+ const watch = args.includes("--watch") || args.includes("-w");
1863
+ // `--` 구분자 뒤의 인수는 스크립트로 전달 ($__argv__)
1864
+ // ⚠️ --watch와 extraArgs 충돌: -- 없이 `fl run file.fl --watch arg1` 하면
1865
+ // arg1이 --watch 필터링과 함께 extraArgs로 들어감 (의도한 동작인지 불명확)
1866
+ // 셀프 호스팅 실행 시엔 항상 -- 구분자 사용 권장: fl run interp.fl -- target.fl
1867
+ const ddIdx = args.indexOf("--");
1868
+ const extraArgs = ddIdx >= 0
1869
+ ? args.slice(ddIdx + 1)
1870
+ : args.slice(2).filter((a) => a !== "--watch" && a !== "-w");
1871
+ cmdRun(filePath, watch, extraArgs);
1872
+ break;
1873
+ }
1874
+ case "check": {
1875
+ if (args[1] === "--help" || args[1] === "-h") { printSubHelp("check"); break; }
1876
+ const filePath = args[1];
1877
+ if (!filePath) { printSubHelp("check"); process.exit(1); }
1878
+ cmdCheck(filePath);
1879
+ break;
1880
+ }
1881
+ case "compile": {
1882
+ if (args[1] === "--help" || args[1] === "-h") { printSubHelp("compile"); break; }
1883
+ if (args.length < 2) { printSubHelp("compile"); process.exit(1); }
1884
+ cmdCompile(args.slice(1));
1885
+ break;
1886
+ }
1887
+ case "codegen": {
1888
+ if (args.length < 2) { printUsage(); process.exit(1); }
1889
+ cmdCodegen(args.slice(1));
1890
+ break;
1891
+ }
1892
+ case "fmt": {
1893
+ cmdFmt(args.slice(1));
1894
+ break;
1895
+ }
1896
+ case "repl":
1897
+ cmdRepl();
1898
+ break;
1899
+ case "fn-doc":
1900
+ case "fl-doc": {
1901
+ // v11.4: node bootstrap.js fn-doc <name> stdlib 함수 시그니처 검색
1902
+ const query = args[1];
1903
+ if (!query) {
1904
+ console.error("Usage: freelang fn-doc <name>");
1905
+ console.error(" (name can be exact or partial)");
1906
+ process.exit(1);
1907
+ }
1908
+ cmdStdlibDoc(query);
1909
+ break;
1910
+ }
1911
+ case "ls-fns":
1912
+ case "ls-fn": {
1913
+ // freelang ls-fns — 전체 함수 목록 (모듈별 그룹)
1914
+ // freelang ls-fns "http" — 이름에 "http" 포함된 것만
1915
+ // freelang ls-fns --new — 최근 추가 (미구현, 향후)
1916
+ const filter = args[1] ?? "";
1917
+ const signatures = loadEmbeddedSignatures();
1918
+ if (signatures.length === 0) {
1919
+ console.error("함수 목록을 불러올 수 없습니다. npm run build 실행 후 재시도하세요.");
1920
+ process.exit(1);
1921
+ }
1922
+ const q = filter.toLowerCase();
1923
+ const filtered = q ? signatures.filter(s => s.name.toLowerCase().includes(q) || s.module.toLowerCase().includes(q)) : signatures;
1924
+
1925
+ // 모듈별 그룹
1926
+ const byModule = new Map<string, typeof signatures>();
1927
+ for (const s of filtered) {
1928
+ if (!byModule.has(s.module)) byModule.set(s.module, []);
1929
+ byModule.get(s.module)!.push(s);
1930
+ }
1931
+
1932
+ const total = filtered.length;
1933
+ const header = q ? `\x1b[36m[ls-fns]\x1b[0m "${q}" 포함 — ${total}개\n` : `\x1b[36m[ls-fns]\x1b[0m 전체 ${total}개 함수\n`;
1934
+ process.stdout.write(header);
1935
+
1936
+ byModule.forEach((fns, mod) => {
1937
+ process.stdout.write(`\n\x1b[33m${mod}\x1b[0m (${fns.length})\n`);
1938
+ for (const f of fns) {
1939
+ const params = f.params ? ` \x1b[2m${f.params}\x1b[0m` : "";
1940
+ const ret = f.returns ? ` → \x1b[2m${f.returns}\x1b[0m` : "";
1941
+ process.stdout.write(` ${f.name}${params}${ret}\n`);
1942
+ }
1943
+ });
1944
+ process.stdout.write(`\n\x1b[2m💡 freelang fn-doc <이름> 으로 자세한 설명을 볼 수 있습니다\x1b[0m\n`);
1945
+ break;
1946
+ }
1947
+ case "debug": {
1948
+ const filePath = args[1];
1949
+ if (!filePath) { printUsage(); process.exit(1); }
1950
+ const stepMode = args.includes("--step");
1951
+ cmdDebug(filePath, stepMode);
1952
+ break;
1953
+ }
1954
+ case "watch": {
1955
+ // Phase 79: freelang watch <file.fl> [--no-clear]
1956
+ const filePath = args[1];
1957
+ if (!filePath) { printUsage(); process.exit(1); }
1958
+ const noClear = args.includes("--no-clear");
1959
+ console.log(`\x1b[36m[Watch Mode]\x1b[0m ${path.basename(filePath)} — 변경 감지 시 자동 재실행`);
1960
+ runWithWatch(filePath, {
1961
+ clearConsole: !noClear,
1962
+ debounceMs: 300,
1963
+ onError: (file, err) => {
1964
+ console.error(`\x1b[31m[ERROR]\x1b[0m ${path.basename(file)}: ${err.message}`);
1965
+ },
1966
+ });
1967
+ break;
1968
+ }
1969
+ case "props":
1970
+ case "prop": {
1971
+ // AI-Native Phase 4: freelang props <file.fl> [--samples N]
1972
+ const filePath = args[1];
1973
+ if (!filePath) {
1974
+ console.error("Usage: freelang props <file.fl> [--samples N]");
1975
+ process.exit(1);
1976
+ }
1977
+ cmdProps(filePath, args.slice(2));
1978
+ break;
1979
+ }
1980
+ case "ci": {
1981
+ // Phase 80: freelang ci [<file.fl>] [--no-fail-fast]
1982
+ cmdCi(args.slice(1)).catch((err) => {
1983
+ console.error(`\x1b[31m[CI 오류]\x1b[0m ${err.message}`);
1984
+ process.exit(1);
1985
+ });
1986
+ break;
1987
+ }
1988
+ case "doc": {
1989
+ // Phase 77: freelang doc <file.fl> [-o out.md] | --dir <dir>
1990
+ cmdDoc(args.slice(1));
1991
+ break;
1992
+ }
1993
+ case "build": {
1994
+ // Phase 8: freelang build --oci <app.fl> --tag <tag> [--registry <url>]
1995
+ cmdBuild(args.slice(1));
1996
+ break;
1997
+ }
1998
+ case "registry": {
1999
+ // Phase 7: freelang registry start [--port 4873] | status
2000
+ cmdRegistry(args.slice(1));
2001
+ break;
2002
+ }
2003
+ case "install": {
2004
+ // Y5: freelang install <plugin-name>
2005
+ cmdInstall(args.slice(1));
2006
+ break;
2007
+ }
2008
+ case "publish": {
2009
+ // Y5: freelang publish <plugin-file.fl>
2010
+ cmdPublish(args.slice(1));
2011
+ break;
2012
+ }
2013
+ case "serve": {
2014
+ // Phase 3: freelang serve [--app app] [--port 3000] [--mode ssr|isr|ssg]
2015
+ cmdServe(args.slice(1));
2016
+ break;
2017
+ }
2018
+ case "errors": {
2019
+ // fl errors [top N] — 누적된 unknown function 로그 빈도 분석
2020
+ const logPath = process.env.FL_ERROR_LOG ?? "/tmp/fl-unknown-functions.jsonl";
2021
+ const topN = parseInt(args[1] ?? "20", 10);
2022
+ try {
2023
+ const raw = fs.readFileSync(logPath, "utf-8").trim();
2024
+ if (!raw) { console.log("(기록된 에러 없음)"); break; }
2025
+ const freq: Record<string, number> = {};
2026
+ const examples: Record<string, string> = {};
2027
+ for (const line of raw.split("\n")) {
2028
+ try {
2029
+ const e = JSON.parse(line);
2030
+ freq[e.name] = (freq[e.name] ?? 0) + 1;
2031
+ if (!examples[e.name] && e.file) examples[e.name] = `${e.file}:${e.line ?? "?"}`;
2032
+ } catch { /* 손상된 줄 무시 */ }
2033
+ }
2034
+ const sorted = Object.entries(freq).sort((a, b) => b[1] - a[1]).slice(0, topN);
2035
+ console.log(`\n📊 FreeLang 미발견 함수 TOP ${topN} (로그: ${logPath})\n`);
2036
+ sorted.forEach(([name, count], i) => {
2037
+ const ex = examples[name] ? ` ← ${examples[name]}` : "";
2038
+ console.log(` ${String(i + 1).padStart(2)}. ${name.padEnd(30)} ${String(count).padStart(4)}회${ex}`);
2039
+ });
2040
+ console.log(`\n총 ${Object.keys(freq).length}종류 / ${raw.split("\n").length}건\n`);
2041
+ } catch (e: any) {
2042
+ if (e.code === "ENOENT") console.log(`(아직 기록 없음 — ${logPath})`);
2043
+ else console.error(e.message);
2044
+ }
2045
+ break;
2046
+ }
2047
+ case "errors-clear": {
2048
+ const logPath = process.env.FL_ERROR_LOG ?? "/tmp/fl-unknown-functions.jsonl";
2049
+ try {
2050
+ fs.unlinkSync(logPath);
2051
+ console.log(`✓ 로그 삭제: ${logPath}`);
2052
+ } catch { console.log("(로그 없음)"); }
2053
+ break;
2054
+ }
2055
+ case "stop": {
2056
+ // fl stop <file.fl> — PID 파일로 정확하게 프로세스 종료
2057
+ const target = args[1];
2058
+ if (!target) {
2059
+ // 인수 없으면 모든 fl PID 파일 종료
2060
+ const pidFiles = fs.readdirSync("/tmp").filter(f => f.startsWith("fl_") && f.endsWith(".pid"));
2061
+ if (pidFiles.length === 0) { console.log("(실행 중인 fl 프로세스 없음)"); break; }
2062
+ let killed = 0;
2063
+ for (const pf of pidFiles) {
2064
+ try {
2065
+ const content = fs.readFileSync(`/tmp/${pf}`, "utf-8").trim();
2066
+ const [pidStr, filePath] = content.split("\n");
2067
+ const pid = parseInt(pidStr, 10);
2068
+ process.kill(pid, "SIGTERM");
2069
+ fs.unlinkSync(`/tmp/${pf}`);
2070
+ console.log(`✓ 종료: PID ${pid} ${filePath ?? pf}`);
2071
+ killed++;
2072
+ } catch (e: any) {
2073
+ if (e.code === "ESRCH") {
2074
+ try { fs.unlinkSync(`/tmp/${pf}`); } catch {}
2075
+ console.log(`✓ 이미 종료됨: ${pf}`);
2076
+ } else {
2077
+ console.error(`✗ 실패: ${pf} — ${e.message}`);
2078
+ }
2079
+ }
2080
+ }
2081
+ console.log(`\n총 ${killed}개 종료`);
2082
+ } else {
2083
+ const absTarget = path.resolve(target);
2084
+ const pidFile = flPidFile(absTarget);
2085
+ try {
2086
+ const content = fs.readFileSync(pidFile, "utf-8").trim();
2087
+ const pid = parseInt(content.split("\n")[0], 10);
2088
+ process.kill(pid, "SIGTERM");
2089
+ try { fs.unlinkSync(pidFile); } catch {}
2090
+ console.log(`✓ 종료: ${target} (PID ${pid})`);
2091
+ } catch (e: any) {
2092
+ if (e.code === "ENOENT") console.log(`(PID 파일 없음 — ${target} 이 실행 중이지 않거나 이미 종료됨)`);
2093
+ else if (e.code === "ESRCH") { try { fs.unlinkSync(pidFile); } catch {} console.log(`(이미 종료됨)`); }
2094
+ else console.error(e.message);
2095
+ }
2096
+ }
2097
+ break;
2098
+ }
2099
+ case "ps": {
2100
+ // fl ps — 실행 중인 fl 프로세스 목록
2101
+ const pidFiles = fs.readdirSync("/tmp").filter(f => f.startsWith("fl_") && f.endsWith(".pid"));
2102
+ if (pidFiles.length === 0) { console.log("(실행 중인 fl 프로세스 없음)"); break; }
2103
+ console.log("\n실행 중인 FreeLang 프로세스:\n");
2104
+ for (const pf of pidFiles) {
2105
+ try {
2106
+ const content = fs.readFileSync(`/tmp/${pf}`, "utf-8").trim();
2107
+ const [pidStr, filePath] = content.split("\n");
2108
+ const pid = parseInt(pidStr, 10);
2109
+ process.kill(pid, 0); // 생존 확인
2110
+ const name = filePath ?? pf;
2111
+ console.log(` PID ${String(pid).padEnd(8)} ${name}`);
2112
+ } catch (e: any) {
2113
+ if (e.code === "ESRCH") { try { fs.unlinkSync(`/tmp/${pf}`); } catch {} } // 죽은 프로세스 청소
2114
+ }
2115
+ }
2116
+ console.log();
2117
+ break;
2118
+ }
2119
+ case "version":
2120
+ case "-v":
2121
+ case "--version":
2122
+ console.log(`FreeLang v${FL_VERSION}`);
2123
+ break;
2124
+ case "help":
2125
+ case "-h":
2126
+ case "--help":
2127
+ printUsage();
2128
+ break;
2129
+ default:
2130
+ if (cmd) {
2131
+ printUsage(cmd);
2132
+ } else {
2133
+ printUsage();
2134
+ }
2135
+ break;
2136
+ }