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
@@ -0,0 +1,1229 @@
1
+ // FreeLang v9: Pure HTTP Server (Express-free)
2
+ // Phase 4a: v9 순수 HTTP 서버 구현 (의존성 제거)
3
+ // Node.js http 모듈만 사용
4
+
5
+ import * as http from "http";
6
+ import * as url from "url";
7
+ import * as crypto from "crypto";
8
+ import * as fs from "fs";
9
+ import * as path from "path";
10
+ import * as os from "os";
11
+
12
+ // Global registry for hot-reload: tracks the currently active HTTP server
13
+ // across Interpreter instances (each createHttpServerModule() creates a new
14
+ // closure, but this registry persists in the Node.js process).
15
+ export const __activeServer: { server: http.Server | null } = { server: null };
16
+
17
+ // WebSocket (Node.js v25 native via stdlib-ws.ts)
18
+ // RFC 6455 핸드셰이크는 아래 server.on('upgrade') 에서 처리
19
+ const WS_OPEN = 1;
20
+ const WS_CLOSING = 2;
21
+
22
+ type CallFn = (name: string, args: any[]) => any;
23
+ type CallFunctionValue = (fnValue: any, args: any[]) => any;
24
+ type RouteHandler = (req: http.IncomingMessage, res: http.ServerResponse) => void;
25
+
26
+ interface Route {
27
+ method: string;
28
+ path: string;
29
+ pattern: RegExp;
30
+ params: string[];
31
+ handler: string | any; // string (function name) or function-value object
32
+ }
33
+
34
+ interface Request {
35
+ __fl_request: true;
36
+ method: string;
37
+ path: string;
38
+ query: Record<string, string | string[]>;
39
+ headers: Record<string, string | string[]>;
40
+ // v11.1: body is auto-parsed JSON (object) when Content-Type: application/json,
41
+ // otherwise raw string (or undefined)
42
+ body?: string | any;
43
+ params: Record<string, string>;
44
+ request_id?: string;
45
+ timestamp?: number;
46
+ }
47
+
48
+ /**
49
+ * Create pure HTTP server for FreeLang v9 (no Express)
50
+ */
51
+ export function createHttpServerModule(callFn: CallFn, callFunctionValue?: CallFunctionValue) {
52
+ const routes: Route[] = [];
53
+ // server_use 미들웨어: [{pattern, handler}] — 라우트 핸들러 실행 전에 순서대로 실행
54
+ // handler가 null/undefined 반환하면 통과, 응답 객체 반환하면 즉시 응답
55
+ const middlewares: Array<{ pattern: RegExp; handler: string | any }> = [];
56
+ let server: http.Server | null = null;
57
+ let requestCounter = 0;
58
+
59
+ // Phase 57: 비동기 응답 보류용 저장소
60
+ const pendingResponses = new Map<string, http.ServerResponse>();
61
+ let currentRequestId: string | null = null;
62
+
63
+ // WebSocket 공개 클라이언트 (터널 WS 프록시용)
64
+ const wsPublicMap = new Map<string, WebSocket>();
65
+ let upgradeHandler: string | null = null;
66
+ let wsClientMessageHandler: string | null = null;
67
+ let wsClientCloseHandler: string | null = null;
68
+ let wssPublic: WebSocketServer | null = null;
69
+
70
+ // Request ID 생성
71
+ function generateRequestId(): string {
72
+ const timestamp = Date.now();
73
+ const counter = ++requestCounter;
74
+ return `req_${timestamp}_${counter}`;
75
+ }
76
+
77
+ // Access log 출력
78
+ function logAccess(method: string, path: string, status: number, duration: number, requestId: string): void {
79
+ const icon = status >= 400 ? "❌" : "✅";
80
+ console.log(`${icon} [${requestId}] ${method} ${path} ${status} ${duration}ms`);
81
+ }
82
+
83
+ // URL 경로를 정규표현식으로 변환 (예: /users/:id → /users/(.+), /* → /.*))
84
+ function pathToRegex(path: string): [RegExp, string[]] {
85
+ const params: string[] = [];
86
+ const pattern = path
87
+ .replace(/\//g, "\\/")
88
+ .replace(/\*/g, ".*")
89
+ .replace(/:(\w+)/g, (_, param) => {
90
+ params.push(param);
91
+ return "([^\\/]+)";
92
+ });
93
+ return [new RegExp(`^${pattern}$`), params];
94
+ }
95
+
96
+ // 요청 파싱
97
+ function parseUrl(urlStr: string): { path: string; query: Record<string, string | string[]> } {
98
+ const parsed = url.parse(urlStr, true);
99
+ return {
100
+ path: parsed.pathname || "/",
101
+ query: parsed.query as Record<string, string | string[]>,
102
+ };
103
+ }
104
+
105
+ // 요청 본문 읽기 (multipart/form-data 포함)
106
+ async function readBody(req: http.IncomingMessage): Promise<string | any> {
107
+ return new Promise((resolve) => {
108
+ const chunks: Buffer[] = [];
109
+ req.on("data", (chunk: Buffer) => chunks.push(chunk));
110
+ req.on("end", () => {
111
+ const raw = Buffer.concat(chunks);
112
+ const ct = (req.headers["content-type"] || "").toString();
113
+
114
+ if (ct.includes("application/json")) {
115
+ try { resolve(JSON.parse(raw.toString())); return; } catch { /* fall through */ }
116
+ }
117
+
118
+ if (ct.includes("multipart/form-data")) {
119
+ try { resolve(parseMultipart(raw, ct)); return; } catch { /* fall through */ }
120
+ }
121
+
122
+ if (ct.includes("application/x-www-form-urlencoded")) {
123
+ try {
124
+ const params: Record<string, string> = {};
125
+ new url.URLSearchParams(raw.toString()).forEach((v, k) => { params[k] = v; });
126
+ resolve(params);
127
+ return;
128
+ } catch { /* fall through */ }
129
+ }
130
+
131
+ resolve(raw.toString());
132
+ });
133
+ });
134
+ }
135
+
136
+ // multipart/form-data 파서 — 파일은 /tmp/fl-uploads/ 에 저장
137
+ function parseMultipart(raw: Buffer, contentType: string): any {
138
+ const boundaryMatch = contentType.match(/boundary=([^\s;]+)/);
139
+ if (!boundaryMatch) return {};
140
+
141
+ const boundary = boundaryMatch[1];
142
+ const delimiter = Buffer.from("\r\n--" + boundary);
143
+ const fields: Record<string, string> = {};
144
+ const files: any[] = [];
145
+
146
+ const uploadDir = path.join(os.tmpdir(), "fl-uploads");
147
+ if (!fs.existsSync(uploadDir)) fs.mkdirSync(uploadDir, { recursive: true });
148
+
149
+ // 파트 분리
150
+ const start = Buffer.from("--" + boundary + "\r\n");
151
+ let pos = raw.indexOf(start);
152
+ if (pos < 0) return { fields, files };
153
+ pos += start.length;
154
+
155
+ while (pos < raw.length) {
156
+ const next = raw.indexOf(delimiter, pos);
157
+ const partEnd = next < 0 ? raw.length : next;
158
+ const part = raw.slice(pos, partEnd);
159
+
160
+ // 헤더/바디 분리 (\r\n\r\n)
161
+ const headerEnd = part.indexOf("\r\n\r\n");
162
+ if (headerEnd < 0) break;
163
+
164
+ const headerStr = part.slice(0, headerEnd).toString();
165
+ const bodyBuf = part.slice(headerEnd + 4);
166
+
167
+ // Content-Disposition 파싱
168
+ const dispMatch = headerStr.match(/Content-Disposition:[^\n]*?;\s*name="([^"]+)"(?:[^\n]*?filename="([^"]+)")?/i);
169
+ if (!dispMatch) { pos = partEnd + delimiter.length + 2; continue; }
170
+
171
+ const fieldName = dispMatch[1];
172
+ const fileName = dispMatch[2];
173
+
174
+ if (fileName) {
175
+ // 파일 파트
176
+ const ctMatch = headerStr.match(/Content-Type:\s*([^\r\n]+)/i);
177
+ const mimetype = ctMatch ? ctMatch[1].trim() : "application/octet-stream";
178
+ const ext = path.extname(fileName) || "";
179
+ const savedName = crypto.randomBytes(8).toString("hex") + ext;
180
+ const savedPath = path.join(uploadDir, savedName);
181
+ fs.writeFileSync(savedPath, bodyBuf);
182
+
183
+ const m = new Map<string, any>();
184
+ m.set("fieldname", fieldName);
185
+ m.set("originalname", fileName);
186
+ m.set("mimetype", mimetype);
187
+ m.set("size", bodyBuf.length);
188
+ m.set("path", savedPath);
189
+ m.set("filename", savedName);
190
+ files.push(m);
191
+ } else {
192
+ // 일반 필드
193
+ fields[fieldName] = bodyBuf.toString().replace(/\r\n$/, "");
194
+ }
195
+
196
+ if (next < 0) break;
197
+ pos = next + delimiter.length;
198
+ if (raw.slice(pos, pos + 2).toString() === "--") break; // 종료 경계
199
+ pos += 2; // \r\n 건너뜀
200
+ }
201
+
202
+ const result = new Map<string, any>();
203
+ const fieldsMap = new Map<string, string>();
204
+ Object.entries(fields).forEach(([k, v]) => fieldsMap.set(k, v));
205
+ result.set("fields", fieldsMap);
206
+ result.set("files", files);
207
+ return result;
208
+ }
209
+
210
+ // 응답 작성 (extraHeaders: 터널 응답 헤더 전달용)
211
+ function sendResponse(
212
+ res: http.ServerResponse,
213
+ status: number,
214
+ body: any,
215
+ contentType: string = "application/json",
216
+ extraHeaders?: Record<string, string>
217
+ ) {
218
+ const headersToWrite: Record<string, string | string[]> = { "Content-Type": contentType };
219
+ if (extraHeaders) {
220
+ // hop-by-hop 헤더 제외 (프록시에서 전달 불가)
221
+ const hopByHop = new Set([
222
+ 'connection', 'keep-alive', 'transfer-encoding', 'te',
223
+ 'trailer', 'proxy-authorization', 'proxy-authenticate',
224
+ 'upgrade', 'content-encoding'
225
+ ]);
226
+ for (const [k, v] of Object.entries(extraHeaders)) {
227
+ if (!hopByHop.has(k.toLowerCase())) {
228
+ headersToWrite[k] = v;
229
+ }
230
+ }
231
+ }
232
+ res.writeHead(status, headersToWrite);
233
+ if (typeof body === 'string') {
234
+ res.end(body);
235
+ } else if (Buffer.isBuffer(body)) {
236
+ res.end(body);
237
+ } else if (contentType.includes("json") && typeof body === 'object') {
238
+ // FreeLang Map 객체 자동 직렬화 (server_json {:key val} 패턴 지원)
239
+ res.end(JSON.stringify(body, (_k, v) =>
240
+ v instanceof Map ? Object.fromEntries(v) :
241
+ Array.isArray(v) ? v : v
242
+ ));
243
+ } else {
244
+ res.end(String(body ?? ""));
245
+ }
246
+ }
247
+
248
+ // v9 요청 객체 생성
249
+ function createFlRequest(
250
+ method: string,
251
+ path: string,
252
+ query: Record<string, any>,
253
+ headers: Record<string, any>,
254
+ body: string,
255
+ params: Record<string, string>,
256
+ requestId?: string
257
+ ): Request {
258
+ return {
259
+ __fl_request: true,
260
+ method,
261
+ path,
262
+ query,
263
+ headers,
264
+ body: body || undefined,
265
+ params,
266
+ request_id: requestId,
267
+ timestamp: Date.now(),
268
+ };
269
+ }
270
+
271
+ // ── Rate Limiting (IP 기반 슬라이딩 윈도우) ─────────────────────────────
272
+ const rlStore = new Map<string, { count: number; resetAt: number }>();
273
+ let rlMax = 100; // 윈도우당 최대 요청 수
274
+ let rlWindowMs = 60000; // 윈도우 크기 (ms)
275
+
276
+ function checkRateLimit(ip: string): boolean {
277
+ const now = Date.now();
278
+ let entry = rlStore.get(ip);
279
+ if (!entry || now > entry.resetAt) {
280
+ entry = { count: 1, resetAt: now + rlWindowMs };
281
+ rlStore.set(ip, entry);
282
+ return true;
283
+ }
284
+ entry.count++;
285
+ return entry.count <= rlMax;
286
+ }
287
+
288
+ // 오래된 항목 정리 (5분마다)
289
+ setInterval(() => {
290
+ const now = Date.now();
291
+ for (const [ip, e] of rlStore) {
292
+ if (now > e.resetAt + rlWindowMs) rlStore.delete(ip);
293
+ }
294
+ }, 300000).unref();
295
+
296
+ return {
297
+ // server_use path middlewareName — 경로 패턴 매칭 시 미들웨어 실행
298
+ // handler가 null/undefined 반환 → 다음 미들웨어/라우트 진행
299
+ // handler가 응답 객체 반환 → 즉시 응답 (라우트 실행 안 함)
300
+ "server_use": (path: string, handlerName: string | any): null => {
301
+ const [pattern] = pathToRegex(path);
302
+ middlewares.push({ pattern, handler: handlerName });
303
+ return null;
304
+ },
305
+
306
+ // server_rate_limit max window_ms → null (e.g. 100req/60s)
307
+ "server_rate_limit": (max: number, windowMs: number): null => {
308
+ rlMax = Math.max(1, Math.floor(max));
309
+ rlWindowMs = Math.max(1000, Math.floor(windowMs));
310
+ return null;
311
+ },
312
+
313
+ // server_get path handlerName -> null
314
+ "server_get": (path: string, handlerName: string | any): null => {
315
+ const [pattern, params] = pathToRegex(path);
316
+ routes.push({ method: "GET", path, pattern, params, handler: handlerName });
317
+ return null;
318
+ },
319
+
320
+ // server_post path handlerName -> null
321
+ "server_post": (path: string, handlerName: string | any): null => {
322
+ const [pattern, params] = pathToRegex(path);
323
+ routes.push({ method: "POST", path, pattern, params, handler: handlerName });
324
+ return null;
325
+ },
326
+
327
+ // server_put path handlerName -> null
328
+ "server_put": (path: string, handlerName: string | any): null => {
329
+ const [pattern, params] = pathToRegex(path);
330
+ routes.push({ method: "PUT", path, pattern, params, handler: handlerName });
331
+ return null;
332
+ },
333
+
334
+ // server_patch path handlerName -> null
335
+ "server_patch": (path: string, handlerName: string | any): null => {
336
+ const [pattern, params] = pathToRegex(path);
337
+ routes.push({ method: "PATCH", path, pattern, params, handler: handlerName });
338
+ return null;
339
+ },
340
+
341
+ // server_delete path handlerName -> null
342
+ "server_delete": (path: string, handlerName: string | any): null => {
343
+ const [pattern, params] = pathToRegex(path);
344
+ routes.push({ method: "DELETE", path, pattern, params, handler: handlerName });
345
+ return null;
346
+ },
347
+
348
+ // route method path handler → 메서드 문자열로 등록
349
+ // (route "GET" "/api/x" handler-fn)
350
+ // (route "POST" "/api/x" handler-fn)
351
+ "route": (method: string, path: string, handlerName: string | any): null => {
352
+ const m = String(method).toUpperCase();
353
+ const [pattern, params] = pathToRegex(path);
354
+ routes.push({ method: m, path, pattern, params, handler: handlerName });
355
+ return null;
356
+ },
357
+
358
+ // server_static dir [urlPrefix] -> null 정적 파일 서빙 (server-static "public" "/")
359
+ "server_static": (dir: string, urlPrefix: string = "/"): null => {
360
+ const mimeMap: Record<string, string> = {
361
+ ".html": "text/html; charset=utf-8",
362
+ ".htm": "text/html; charset=utf-8",
363
+ ".css": "text/css; charset=utf-8",
364
+ ".js": "application/javascript; charset=utf-8",
365
+ ".mjs": "application/javascript; charset=utf-8",
366
+ ".json": "application/json; charset=utf-8",
367
+ ".png": "image/png",
368
+ ".jpg": "image/jpeg",
369
+ ".jpeg": "image/jpeg",
370
+ ".gif": "image/gif",
371
+ ".svg": "image/svg+xml",
372
+ ".ico": "image/x-icon",
373
+ ".woff": "font/woff",
374
+ ".woff2":"font/woff2",
375
+ ".ttf": "font/ttf",
376
+ ".txt": "text/plain; charset=utf-8",
377
+ ".pdf": "application/pdf",
378
+ };
379
+ const absDir = path.resolve(dir);
380
+ const prefix = urlPrefix.endsWith("/") ? urlPrefix : urlPrefix + "/";
381
+
382
+ const handler = (_req: http.IncomingMessage, res: http.ServerResponse, reqPath: string) => {
383
+ // strip url prefix, default to index.html
384
+ let rel = reqPath.startsWith(prefix) ? reqPath.slice(prefix.length - 1) : reqPath;
385
+ if (rel === "" || rel === "/") rel = "/index.html";
386
+ // security: block path traversal
387
+ const filePath = path.join(absDir, rel);
388
+ if (!filePath.startsWith(absDir)) {
389
+ res.writeHead(403, { "Content-Type": "text/plain" });
390
+ res.end("Forbidden");
391
+ return true;
392
+ }
393
+ if (!fs.existsSync(filePath) || !fs.statSync(filePath).isFile()) return false;
394
+ const ext = path.extname(filePath).toLowerCase();
395
+ const mime = mimeMap[ext] || "application/octet-stream";
396
+ const content = fs.readFileSync(filePath);
397
+ res.writeHead(200, {
398
+ "Content-Type": mime,
399
+ "Content-Length": content.length,
400
+ "Cache-Control": "public, max-age=3600",
401
+ });
402
+ res.end(content);
403
+ return true;
404
+ };
405
+
406
+ // 와일드카드 패턴으로 GET 라우트 등록
407
+ const routePath = prefix === "/" ? "/*" : prefix + "*";
408
+ const [pattern, params] = pathToRegex(routePath);
409
+ routes.push({
410
+ method: "GET",
411
+ path: routePath,
412
+ pattern,
413
+ params,
414
+ handler: { __fl_static_handler: true, fn: handler } as any,
415
+ });
416
+ return null;
417
+ },
418
+
419
+ // server_all path handler → 모든 메서드 등록 (catch-all)
420
+ "server_all": (path: string, handlerName: string | any): null => {
421
+ const [pattern, params] = pathToRegex(path);
422
+ for (const m of ["GET","POST","PUT","PATCH","DELETE"]) {
423
+ routes.push({ method: m, path, pattern, params, handler: handlerName });
424
+ }
425
+ return null;
426
+ },
427
+
428
+ // server_start port|config -> string
429
+ // 선언형 API: (server_start {:port 40090 :routes [...] :middleware [...]})
430
+ // 기존 API: (server_start 40090)
431
+ "server_start": (portOrConfig: number | Record<string, any>): string => {
432
+ // 선언형 API: (server-start {:port 40090 :routes [...] :middleware [...]})
433
+ // :routes 안의 server-get/post 등은 평가 시 이미 routes[]에 등록됨 → 포트만 추출
434
+ const port: number = (portOrConfig !== null && typeof portOrConfig === "object")
435
+ ? ((portOrConfig as any)[":port"] ?? (portOrConfig as any)["port"] ?? 8080)
436
+ : portOrConfig as number;
437
+ // Hot-reload: close previous server before starting new one
438
+ if (__activeServer.server) {
439
+ try {
440
+ __activeServer.server.close();
441
+ } catch (_e) { /* ignore */ }
442
+ __activeServer.server = null;
443
+ }
444
+ server = http.createServer(async (req, res) => {
445
+ const requestStart = Date.now();
446
+ const requestId = generateRequestId();
447
+ currentRequestId = requestId; // Phase 57: 현재 요청 ID 저장
448
+ const method = req.method || "GET";
449
+ const { path, query } = parseUrl(req.url || "/");
450
+ const headers = req.headers;
451
+ const body = await readBody(req);
452
+
453
+ // CORS
454
+ res.setHeader("Access-Control-Allow-Origin", "*");
455
+ res.setHeader("Access-Control-Allow-Methods", "GET, POST, PUT, PATCH, DELETE, OPTIONS");
456
+ res.setHeader("Access-Control-Allow-Headers", "Content-Type, Authorization");
457
+ res.setHeader("X-Request-Id", requestId);
458
+ // 보안 헤더
459
+ res.setHeader("X-Content-Type-Options", "nosniff");
460
+ res.setHeader("X-Frame-Options", "SAMEORIGIN");
461
+ res.setHeader("X-XSS-Protection", "1; mode=block");
462
+ res.setHeader("Content-Security-Policy", "default-src 'self'; script-src 'self' 'unsafe-inline'; style-src 'self' 'unsafe-inline'");
463
+ res.setHeader("Referrer-Policy", "strict-origin-when-cross-origin");
464
+
465
+ if (method === "OPTIONS") {
466
+ res.writeHead(200);
467
+ res.end();
468
+ return;
469
+ }
470
+
471
+ // Rate Limiting
472
+ const clientIp = (req.headers["x-forwarded-for"] as string || req.socket.remoteAddress || "unknown").split(",")[0].trim();
473
+ if (!checkRateLimit(clientIp)) {
474
+ res.writeHead(429, { "Content-Type": "application/json", "Retry-After": String(Math.ceil(rlWindowMs / 1000)) });
475
+ res.end(JSON.stringify({ error: "Too Many Requests", retry_after: Math.ceil(rlWindowMs / 1000) }));
476
+ return;
477
+ }
478
+
479
+ // Dev mode: /__hot SSE endpoint for browser hot-reload
480
+ // Client connects, keeps open; when server restarts the connection
481
+ // drops and EventSource auto-reconnects → onopen fires → reload().
482
+ if (process.env.FL_DEV === "1" && path === "/__hot" && method === "GET") {
483
+ res.writeHead(200, {
484
+ "Content-Type": "text/event-stream",
485
+ "Cache-Control": "no-cache",
486
+ "Connection": "keep-alive",
487
+ "X-Accel-Buffering": "no",
488
+ });
489
+ res.write("retry: 400\n\n");
490
+ // Keep connection open until client disconnects or server restarts
491
+ return;
492
+ }
493
+
494
+ // 미들웨어 실행 — 경로 매칭 시 핸들러 호출, 응답 반환 시 즉시 응답
495
+ const baseReq = createFlRequest(method, path, query, headers, body, {}, requestId);
496
+ for (const mw of middlewares) {
497
+ if (!mw.pattern.exec(path)) continue;
498
+ try {
499
+ let mwResult;
500
+ if (typeof mw.handler === "string") {
501
+ mwResult = callFn(mw.handler, [baseReq]);
502
+ } else if (mw.handler?.kind === "function-value" && callFunctionValue) {
503
+ mwResult = callFunctionValue(mw.handler, [baseReq]);
504
+ }
505
+ if (mwResult instanceof Promise) mwResult = await mwResult;
506
+ // null/undefined → 통과, 응답 객체 → 즉시 전송
507
+ if (mwResult !== null && mwResult !== undefined) {
508
+ // server_status()/server_json() 반환 객체: { __fl_response, status, body, contentType }
509
+ const mwStatus = mwResult.status ?? (mwResult.__fl_status ?? 200);
510
+ const headersObj: Record<string, string> = {};
511
+ if (mwResult.__fl_headers) Object.assign(headersObj, mwResult.__fl_headers);
512
+ const mwBody = mwResult.__fl_response
513
+ ? (typeof mwResult.body === "object" ? JSON.stringify(mwResult.body) : String(mwResult.body ?? ""))
514
+ : (typeof mwResult === "object" ? JSON.stringify(mwResult) : String(mwResult));
515
+ const mwCT = mwResult.contentType ?? "application/json";
516
+ sendResponse(res, mwStatus, mwBody, mwCT, headersObj);
517
+ logAccess(method, path, mwStatus, Date.now() - requestStart, requestId);
518
+ return;
519
+ }
520
+ } catch (mwErr: any) {
521
+ sendResponse(res, 500, JSON.stringify({ error: mwErr.message ?? "middleware error" }));
522
+ return;
523
+ }
524
+ }
525
+
526
+ // 라우트 매칭
527
+ let matched = false;
528
+ for (const route of routes) {
529
+ if (route.method !== method) continue;
530
+
531
+ const match = route.pattern.exec(path);
532
+ if (!match) continue;
533
+
534
+ matched = true;
535
+ let status = 200;
536
+ try {
537
+ // 경로 파라미터 추출
538
+ const params: Record<string, string> = {};
539
+ for (let i = 0; i < route.params.length; i++) {
540
+ params[route.params[i]] = match[i + 1];
541
+ }
542
+
543
+ // v9 요청 객체 생성 (request_id 포함)
544
+ const flReq = createFlRequest(method, path, query, headers, body, params, requestId);
545
+
546
+ // 핸들러 호출 (Phase 57: Promise 지원 + 람다 함수 지원)
547
+ let rawResult;
548
+ if (typeof route.handler === 'string') {
549
+ // 함수 이름 (문자열) → callFn으로 호출
550
+ rawResult = callFn(route.handler, [flReq]);
551
+ } else if (route.handler && (route.handler as any).__fl_static_handler === true) {
552
+ // server_static 정적 파일 핸들러
553
+ const served = (route.handler as any).fn(req, res, path);
554
+ if (served) return; // 직접 응답 완료
555
+ // 파일 없으면 404
556
+ sendResponse(res, 404, { error: "Not found" });
557
+ return;
558
+ } else if (route.handler && route.handler.kind === 'function-value' && callFunctionValue) {
559
+ // v9 람다 함수 (function-value 객체) → callFunctionValue로 호출
560
+ rawResult = callFunctionValue(route.handler, [flReq]);
561
+ } else {
562
+ throw new Error(`Invalid handler: expected string or function-value, got ${typeof route.handler}`);
563
+ }
564
+ const result = (rawResult instanceof Promise) ? await rawResult : rawResult;
565
+
566
+ // 응답 처리 (응답 보류 중이면 skip)
567
+ if (pendingResponses.has(requestId)) {
568
+ pendingResponses.delete(requestId);
569
+ } else if (result && typeof result === "object" && result.__fl_wait_and_respond === true) {
570
+ // Phase 57+: 비동기 응답 대기 (Promise 처리)
571
+ const asyncResp = await result.promise;
572
+ if (!asyncResp) {
573
+ sendResponse(res, 504, { error: "Gateway Timeout" });
574
+ } else {
575
+ status = asyncResp.status ?? 200;
576
+ // encoding: 'base64'이면 binary body 디코딩
577
+ let respBody = asyncResp.body ?? "";
578
+ let contentType = asyncResp.contentType ?? "application/json";
579
+ const extraHeaders = asyncResp.headers ?? {};
580
+ if (asyncResp.encoding === 'base64' && typeof respBody === 'string') {
581
+ const buf = Buffer.from(respBody, 'base64');
582
+ // Content-Type은 에이전트가 준 것 우선
583
+ if (extraHeaders['content-type']) {
584
+ contentType = extraHeaders['content-type'] as string;
585
+ }
586
+ sendResponse(res, status, buf, contentType, extraHeaders);
587
+ } else {
588
+ if (extraHeaders['content-type']) {
589
+ contentType = extraHeaders['content-type'] as string;
590
+ }
591
+ sendResponse(res, status, respBody, contentType, extraHeaders);
592
+ }
593
+ }
594
+ } else {
595
+ if (result && typeof result === "object") {
596
+ const getField = (obj: any, key: string) =>
597
+ obj instanceof Map ? (obj.get(key) ?? obj.get(":" + key)) : obj[key];
598
+ const resStatus = getField(result, "status");
599
+ const resBody = getField(result, "body");
600
+ if (result.__fl_response === true || (resStatus !== undefined && resBody !== undefined)) {
601
+ status = resStatus ?? 200;
602
+ const resHeaders = getField(result, "headers") ?? {};
603
+ const headersObj = resHeaders instanceof Map ? Object.fromEntries(resHeaders) : resHeaders;
604
+ const contentType = headersObj["content-type"] ?? getField(result, "contentType") ?? "application/json";
605
+ sendResponse(res, status, resBody ?? "", contentType, headersObj);
606
+ } else {
607
+ // #49: 라우트 핸들러가 map을 직접 반환한 경우 — server-json 권장 힌트
608
+ const hint49 = `[FreeLang #49] 라우트 핸들러가 map을 직접 반환했습니다.\n 자동으로 JSON 직렬화하여 전송합니다.\n 명시적 응답: (server-json result) 사용을 권장합니다.`;
609
+ if (process.env.FL_V12 === "1") {
610
+ sendResponse(res, 400, { error: hint49 });
611
+ } else {
612
+ console.warn(`⚠️ ${hint49}`);
613
+ sendResponse(res, 200, result);
614
+ }
615
+ }
616
+ } else {
617
+ sendResponse(res, 200, result ?? "");
618
+ }
619
+ }
620
+
621
+ // Access log
622
+ const duration = Date.now() - requestStart;
623
+ logAccess(method, path, status, duration, requestId);
624
+ } catch (err: any) {
625
+ const status = 500;
626
+ sendResponse(res, status, { error: err.message });
627
+ const duration = Date.now() - requestStart;
628
+ logAccess(method, path, status, duration, requestId);
629
+ }
630
+ return;
631
+ }
632
+
633
+ // 404
634
+ if (!matched) {
635
+ const status = 404;
636
+ sendResponse(res, status, { error: "Not Found", path });
637
+ const duration = Date.now() - requestStart;
638
+ logAccess(method, path, status, duration, requestId);
639
+ }
640
+ });
641
+
642
+ // WebSocket upgrade 지원 (RFC 6455 직접 구현)
643
+ server.on('upgrade', (req, socket, head) => {
644
+ if (!upgradeHandler) {
645
+ socket.destroy();
646
+ return;
647
+ }
648
+
649
+ // RFC 6455 핸드셰이크
650
+ const key = (req.headers as any)['sec-websocket-key'];
651
+ if (!key) {
652
+ socket.destroy();
653
+ return;
654
+ }
655
+
656
+ const accept = crypto.createHash('sha1')
657
+ .update(key + '258EAFA5-E914-47DA-95CA-C5AB0DC85B11')
658
+ .digest('base64');
659
+
660
+ socket.write([
661
+ 'HTTP/1.1 101 Switching Protocols',
662
+ 'Upgrade: websocket',
663
+ 'Connection: Upgrade',
664
+ 'Sec-WebSocket-Accept: ' + accept,
665
+ '', ''
666
+ ].join('\r\n'));
667
+
668
+ const sessionId = 'wsc-' + crypto.randomBytes(8).toString('hex');
669
+ wsPublicMap.set(sessionId, socket as any);
670
+
671
+ let buf = Buffer.alloc(0);
672
+
673
+ socket.on('data', async (chunk: Buffer) => {
674
+ buf = Buffer.concat([buf, chunk]);
675
+
676
+ // RFC 6455 프레임 파싱 (간단한 구현)
677
+ while (buf.length >= 2) {
678
+ const fin = (buf[0] & 0x80) !== 0;
679
+ const opcode = buf[0] & 0x0f;
680
+ const masked = (buf[1] & 0x80) !== 0;
681
+ let payloadLen = buf[1] & 0x7f;
682
+ let hdrLen = 2;
683
+
684
+ if (payloadLen === 126) {
685
+ if (buf.length < 4) break;
686
+ payloadLen = buf.readUInt16BE(2);
687
+ hdrLen = 4;
688
+ } else if (payloadLen === 127) {
689
+ if (buf.length < 10) break;
690
+ payloadLen = Number(buf.readBigUInt64BE(2));
691
+ hdrLen = 10;
692
+ }
693
+
694
+ if (masked) hdrLen += 4;
695
+ if (buf.length < hdrLen + payloadLen) break;
696
+
697
+ let maskKey: Buffer | null = null;
698
+ if (masked) maskKey = buf.slice(hdrLen - 4, hdrLen);
699
+
700
+ const payload = Buffer.from(buf.slice(hdrLen, hdrLen + payloadLen));
701
+ if (maskKey) {
702
+ for (let i = 0; i < payload.length; i++) {
703
+ payload[i] ^= maskKey[i % 4];
704
+ }
705
+ }
706
+
707
+ if (opcode === 8) { // CLOSE
708
+ wsPublicMap.delete(sessionId);
709
+ if (wsClientCloseHandler) {
710
+ try { await callFn(wsClientCloseHandler, [sessionId, 1000]); } catch {}
711
+ }
712
+ socket.end();
713
+ return;
714
+ }
715
+
716
+ if (opcode === 9) { // PING
717
+ socket.write(Buffer.from([0x8a, 0x00]));
718
+ } else if (opcode === 0x01 || opcode === 0x02) { // TEXT or BINARY
719
+ if (wsClientMessageHandler) {
720
+ const isBinary = opcode === 0x02;
721
+ const data = isBinary ? payload.toString('base64') : payload.toString('utf8');
722
+ try {
723
+ await callFn(wsClientMessageHandler, [sessionId, data, isBinary]);
724
+ } catch {}
725
+ }
726
+ }
727
+
728
+ buf = buf.slice(hdrLen + payloadLen);
729
+ }
730
+ });
731
+
732
+ socket.on('close', async () => {
733
+ wsPublicMap.delete(sessionId);
734
+ if (wsClientCloseHandler) {
735
+ try { await callFn(wsClientCloseHandler, [sessionId, 1006]); } catch {}
736
+ }
737
+ });
738
+
739
+ socket.on('error', () => { wsPublicMap.delete(sessionId); });
740
+
741
+ // FreeLang upgrade 핸들러 호출
742
+ const _wsUrl = new URL(req.url || '/', 'http://localhost');
743
+ const _wsQuery: Record<string, string> = {};
744
+ _wsUrl.searchParams.forEach((v, k) => { _wsQuery[k] = v; });
745
+ const upgradeReq = {
746
+ __fl_request: true,
747
+ method: 'WS_UPGRADE',
748
+ path: req.url || '/',
749
+ headers: req.headers,
750
+ query: _wsQuery,
751
+ body: '',
752
+ params: {},
753
+ session_id: sessionId,
754
+ };
755
+ callFn(upgradeHandler!, [upgradeReq]);
756
+ });
757
+
758
+ server.on("error", (err: any) => {
759
+ if (err.code === "EADDRINUSE") {
760
+ console.warn(`[server] 포트 ${port} 이미 사용 중 — 서버 시작 건너뜀`);
761
+ } else {
762
+ console.error(`[server] 서버 오류: ${err.message}`);
763
+ }
764
+ });
765
+ server.listen(port);
766
+ __activeServer.server = server;
767
+ // Keep process alive while server is running
768
+ setInterval(() => {}, 10000).unref();
769
+ return `server listening on :${port}`;
770
+ },
771
+
772
+ // server_stop -> null
773
+ "server_stop": (): null => {
774
+ if (server) {
775
+ server.close();
776
+ server = null;
777
+ }
778
+ return null;
779
+ },
780
+
781
+ // server_json [status] obj -> response object
782
+ // (server_json data) → 200 JSON
783
+ // (server_json 201 data) → 201 JSON
784
+ "server_json": (statusOrBody: any, maybeBody?: any): Record<string, any> => {
785
+ const isStatus = typeof statusOrBody === "number" && statusOrBody >= 100 && statusOrBody < 600;
786
+ return {
787
+ __fl_response: true,
788
+ status: isStatus ? statusOrBody : 200,
789
+ contentType: "application/json",
790
+ body: isStatus ? maybeBody : statusOrBody,
791
+ };
792
+ },
793
+
794
+ // server_text text -> response object
795
+ "server_text": (body: string): Record<string, any> => {
796
+ return {
797
+ __fl_response: true,
798
+ status: 200,
799
+ contentType: "text/plain",
800
+ body,
801
+ };
802
+ },
803
+
804
+ // server_status code body -> response object
805
+ "server_status": (code: any, body: any): Record<string, any> => {
806
+ // #50: 인자 순서 역전 감지 — (server-status "msg" 200) 같은 실수 경고/에러
807
+ if (typeof code === "string" && typeof body === "number") {
808
+ const hint = `server-status 인자 순서: (server-status code body)\n 예: (server-status 404 "Not Found")\n → 받은 값: (server-status "${code}" ${body}) — 숫자와 문자열 역전`;
809
+ if (process.env.FL_V12 === "1") throw new Error(`[v12] ${hint}`);
810
+ console.warn(`⚠️ [FreeLang] ${hint}`);
811
+ return { __fl_response: true, status: Number(body), contentType: "application/json", body: code };
812
+ }
813
+ return {
814
+ __fl_response: true,
815
+ status: Number(code),
816
+ contentType: "application/json",
817
+ body,
818
+ };
819
+ },
820
+
821
+ // server_html body -> response object (text/html)
822
+ // In dev mode (FL_DEV=1), injects hot-reload script before </body>
823
+ // so browsers auto-refresh when the FreeLang source file changes.
824
+ "server_html": (body: string, statusOrUndef?: number): Record<string, any> => {
825
+ const status = (typeof statusOrUndef === "number") ? statusOrUndef : 200;
826
+ let finalBody = body;
827
+ if (process.env.FL_DEV === "1" && typeof finalBody === "string") {
828
+ const script = `<script>(function(){let w=false;function c(){const e=new EventSource('/__hot');e.onopen=function(){if(w)location.reload();w=true;};e.onerror=function(){e.close();setTimeout(c,400);};}c();})();</script>`;
829
+ if (finalBody.includes("</body>")) {
830
+ finalBody = finalBody.replace("</body>", script + "</body>");
831
+ } else {
832
+ finalBody = finalBody + script;
833
+ }
834
+ }
835
+ return {
836
+ __fl_response: true,
837
+ status,
838
+ contentType: "text/html; charset=utf-8",
839
+ body: finalBody,
840
+ };
841
+ },
842
+
843
+ // server_html_cookie cookie html -> response (Set-Cookie 헤더 포함 HTML 응답)
844
+ "server_html_cookie": (cookie: string, html: string): Record<string, any> => {
845
+ return {
846
+ __fl_response: true,
847
+ status: 200,
848
+ contentType: "text/html; charset=utf-8",
849
+ body: html,
850
+ headers: { "Set-Cookie": cookie },
851
+ };
852
+ },
853
+
854
+ // server_set_cookie name value opts -> cookie string (HttpOnly+Secure+SameSite 자동)
855
+ "server_set_cookie": (name: string, value: string, opts: any = {}): string => {
856
+ let cookie = `${encodeURIComponent(String(name))}=${encodeURIComponent(String(value))}`;
857
+ if (opts && opts["max_age"] !== undefined) cookie += `; Max-Age=${opts["max_age"]}`;
858
+ cookie += `; Path=${(opts && opts["path"]) || "/"}`;
859
+ if (opts && opts["domain"]) cookie += `; Domain=${opts["domain"]}`;
860
+ if (!opts || opts["http_only"] !== false) cookie += "; HttpOnly";
861
+ if (!opts || opts["secure"] !== false) cookie += "; Secure";
862
+ const ss = (opts && opts["same_site"]) || "Strict";
863
+ cookie += `; SameSite=${ss}`;
864
+ return cookie;
865
+ },
866
+
867
+ // server_redirect url -> response (302 리다이렉트)
868
+ "server_redirect": (url: string): Record<string, any> => {
869
+ return {
870
+ __fl_response: true,
871
+ status: 302,
872
+ contentType: "text/plain",
873
+ body: "",
874
+ headers: { "Location": url },
875
+ };
876
+ },
877
+
878
+ // v12: 짧은 응답 별칭 — res-json / res-html / res-status / res-redirect / res-text
879
+ "res-json": (statusOrBody: any, maybeBody?: any): Record<string, any> => {
880
+ const isStatus = typeof statusOrBody === "number" && statusOrBody >= 100 && statusOrBody < 600;
881
+ return { __fl_response: true, status: isStatus ? statusOrBody : 200,
882
+ contentType: "application/json", body: isStatus ? maybeBody : statusOrBody };
883
+ },
884
+ "res-html": (body: string, status = 200): Record<string, any> => ({
885
+ __fl_response: true, status, contentType: "text/html; charset=utf-8", body,
886
+ }),
887
+ "res-status": (code: number, body: any): Record<string, any> => ({
888
+ __fl_response: true, status: code, contentType: "application/json", body,
889
+ }),
890
+ "res-redirect": (url: string): Record<string, any> => ({
891
+ __fl_response: true, status: 302, contentType: "text/plain", body: "",
892
+ headers: { "Location": url },
893
+ }),
894
+ "res-text": (body: string, status = 200): Record<string, any> => ({
895
+ __fl_response: true, status, contentType: "text/plain; charset=utf-8", body,
896
+ }),
897
+
898
+ // server_redirect_cookie url cookie -> response (302 리다이렉트 + Set-Cookie)
899
+ "server_redirect_cookie": (url: string, cookie: string): Record<string, any> => {
900
+ return {
901
+ __fl_response: true,
902
+ status: 302,
903
+ contentType: "text/plain",
904
+ body: "",
905
+ headers: { "Location": url, "Set-Cookie": cookie },
906
+ };
907
+ },
908
+
909
+ // server_header response key value -> response (헤더 추가)
910
+ "server_header": (response: Record<string, any>, key: string, value: string): Record<string, any> => {
911
+ const existing = response.headers || {};
912
+ return { ...response, headers: { ...existing, [key]: value } };
913
+ },
914
+
915
+ // ── API 응답 헬퍼 ─────────────────────────────────────────────────
916
+ // api_ok data → 200 {:ok true :data ...}
917
+ // api_ok "msg" → 200 {:ok true :message ...}
918
+ "api_ok": (data: any): Record<string, any> => ({
919
+ __fl_response: true, status: 200, contentType: "application/json",
920
+ body: typeof data === "string" ? { ok: true, message: data } : { ok: true, data },
921
+ }),
922
+
923
+ // api_created data → 201 {:ok true :data ...}
924
+ "api_created": (data: any): Record<string, any> => ({
925
+ __fl_response: true, status: 201, contentType: "application/json",
926
+ body: { ok: true, data },
927
+ }),
928
+
929
+ // api_error message [code] → 4xx/5xx {:ok false :error ...}
930
+ // (api_error "Not found" 404)
931
+ // (api_error "Server error") → 500
932
+ "api_error": (message: string, code?: number): Record<string, any> => ({
933
+ __fl_response: true, status: code ?? 500, contentType: "application/json",
934
+ body: { ok: false, error: message },
935
+ }),
936
+
937
+ // api_not_found [message] → 404
938
+ "api_not_found": (message?: string): Record<string, any> => ({
939
+ __fl_response: true, status: 404, contentType: "application/json",
940
+ body: { ok: false, error: message ?? "Not Found" },
941
+ }),
942
+
943
+ // api_bad_request [message] → 400
944
+ "api_bad_request": (message?: string): Record<string, any> => ({
945
+ __fl_response: true, status: 400, contentType: "application/json",
946
+ body: { ok: false, error: message ?? "Bad Request" },
947
+ }),
948
+
949
+ // api_unauthorized [message] → 401
950
+ "api_unauthorized": (message?: string): Record<string, any> => ({
951
+ __fl_response: true, status: 401, contentType: "application/json",
952
+ body: { ok: false, error: message ?? "Unauthorized" },
953
+ }),
954
+
955
+ // api_forbidden [message] → 403
956
+ "api_forbidden": (message?: string): Record<string, any> => ({
957
+ __fl_response: true, status: 403, contentType: "application/json",
958
+ body: { ok: false, error: message ?? "Forbidden" },
959
+ }),
960
+
961
+ // ── CORS 헬퍼 ────────────────────────────────────────────────────
962
+ // server_cors response [origin] → response에 CORS 헤더 추가
963
+ // (server_cors (server_json data))
964
+ // (server_cors (server_json data) "https://app.example.com")
965
+ "server_cors": (response: Record<string, any>, origin?: string): Record<string, any> => {
966
+ const corsHeaders = {
967
+ "Access-Control-Allow-Origin": origin ?? "*",
968
+ "Access-Control-Allow-Methods": "GET, POST, PUT, PATCH, DELETE, OPTIONS",
969
+ "Access-Control-Allow-Headers": "Content-Type, Authorization",
970
+ };
971
+ return { ...response, headers: { ...(response.headers || {}), ...corsHeaders } };
972
+ },
973
+
974
+ // server_cors_all → 모든 라우트에 CORS 미들웨어 등록 (use와 함께)
975
+ "server_cors_middleware": (): any => {
976
+ return (req: any) => {
977
+ if (req.method === "OPTIONS") {
978
+ return {
979
+ __fl_response: true, status: 204, contentType: "text/plain", body: "",
980
+ headers: {
981
+ "Access-Control-Allow-Origin": "*",
982
+ "Access-Control-Allow-Methods": "GET, POST, PUT, PATCH, DELETE, OPTIONS",
983
+ "Access-Control-Allow-Headers": "Content-Type, Authorization",
984
+ },
985
+ };
986
+ }
987
+ return null; // 통과
988
+ };
989
+ },
990
+
991
+ // server_options response -> 204 No Content (CORS preflight 응답)
992
+ "server_options": (allowMethods = "GET, POST, PUT, PATCH, DELETE, OPTIONS"): Record<string, any> => {
993
+ return {
994
+ __fl_response: true,
995
+ status: 204,
996
+ contentType: "text/plain",
997
+ body: "",
998
+ headers: {
999
+ "Access-Control-Allow-Methods": allowMethods,
1000
+ "Access-Control-Allow-Headers": "Content-Type, Authorization",
1001
+ },
1002
+ };
1003
+ },
1004
+
1005
+ // server_req_cookie req name -> string | null (쿠키 값 읽기)
1006
+ "server_req_cookie": (req: Request, name: string): string | null => {
1007
+ const cookieHeader = req.headers["cookie"] as string;
1008
+ if (!cookieHeader) return null;
1009
+ const cookies = cookieHeader.split(";").map(c => c.trim());
1010
+ for (const cookie of cookies) {
1011
+ const [k, ...rest] = cookie.split("=");
1012
+ if (k.trim() === name) return rest.join("=").trim();
1013
+ }
1014
+ return null;
1015
+ },
1016
+
1017
+ // server_wait_respond promise -> response object (비동기 응답 대기)
1018
+ "server_wait_respond": (promise: Promise<any>): Record<string, any> => {
1019
+ return {
1020
+ __fl_wait_and_respond: true,
1021
+ promise,
1022
+ };
1023
+ },
1024
+
1025
+ // server_req_body req -> string OR object (Content-Type 자동 가드)
1026
+ // 자잘 마찰 #1 (2026-04-25): Content-Type=application/json이면 이미 객체이므로
1027
+ // json_parse 두 번 하면 [object Object] 에러. (server_req_body req)는 string 보장.
1028
+ "server_req_body": (req: Request): string => {
1029
+ const b = req.body;
1030
+ if (b === null || b === undefined) return "";
1031
+ // 이미 객체/배열로 파싱된 경우 → 다시 string으로 (사용자가 json_parse 안전)
1032
+ if (typeof b === "object") return JSON.stringify(b);
1033
+ return String(b);
1034
+ },
1035
+
1036
+ // server_req_json req -> parsed object (자동 파싱, Content-Type 무관)
1037
+ // 자잘 마찰 #1 (2026-04-25): 사용자 18세션에서 18군데 우회 헬퍼 작성
1038
+ // (if (string? raw) (json_parse raw) raw) 대신 한 줄
1039
+ "server_req_json": (req: Request): any => {
1040
+ const b = req.body;
1041
+ if (b === null || b === undefined) return null;
1042
+ if (typeof b === "object") return b; // 이미 파싱됨
1043
+ try { return JSON.parse(String(b)); } catch { return null; }
1044
+ },
1045
+
1046
+ // server_req_query req [key] -> object or string
1047
+ "server_req_query": (req: Request, key?: string): any => {
1048
+ if (key === undefined) {
1049
+ return req.query;
1050
+ }
1051
+ const value = req.query[key];
1052
+ if (Array.isArray(value)) return value[0];
1053
+ return value ?? null;
1054
+ },
1055
+
1056
+ // server_req_files req -> array of multipart files
1057
+ "server_req_files": (req: Request): any[] => {
1058
+ const b = req.body;
1059
+ if (b instanceof Map) return b.get("files") ?? [];
1060
+ return [];
1061
+ },
1062
+
1063
+ // server_req_fields req -> map of multipart text fields
1064
+ "server_req_fields": (req: Request): any => {
1065
+ const b = req.body;
1066
+ if (b instanceof Map) return b.get("fields") ?? new Map();
1067
+ return null;
1068
+ },
1069
+
1070
+ // server_req_header req name -> string
1071
+ "server_req_header": (req: Request, name: string): string | null => {
1072
+ const value = req.headers[name.toLowerCase()];
1073
+ if (Array.isArray(value)) return value[0];
1074
+ return value as string | null;
1075
+ },
1076
+
1077
+ // server_req_headers req -> object (전체 헤더 맵)
1078
+ "server_req_headers": (req: Request): Record<string, string> => {
1079
+ const result: Record<string, string> = {};
1080
+ for (const [k, v] of Object.entries(req.headers)) {
1081
+ result[k] = Array.isArray(v) ? v[0] : (v as string) ?? "";
1082
+ }
1083
+ return result;
1084
+ },
1085
+
1086
+ // server_req_param req name -> string
1087
+ "server_req_param": (req: Request, name: string): string | null => {
1088
+ return req.params[name] ?? null;
1089
+ },
1090
+
1091
+ // server_req_params req -> object (all URL params as an object)
1092
+ "server_req_params": (req: Request): Record<string, string> => {
1093
+ return req.params ?? {};
1094
+ },
1095
+
1096
+ // v12: 짧은 별칭 — req-param / req-query / req-body / req-header
1097
+ "req-param": (req: Request, name: string): string | null => req.params[name] ?? null,
1098
+ "req-query": (req: Request, key?: string): any => {
1099
+ if (key === undefined) return req.query ?? {};
1100
+ const v = (req.query ?? {})[key];
1101
+ return Array.isArray(v) ? v[0] : (v ?? null);
1102
+ },
1103
+ "req-body": (req: Request): any => {
1104
+ const b = req.body;
1105
+ if (b === null || b === undefined) return null;
1106
+ if (typeof b === "object") return b;
1107
+ if (typeof b === "string") { try { return JSON.parse(b); } catch { return b; } }
1108
+ return b;
1109
+ },
1110
+ "req-header": (req: Request, name: string): string | null => {
1111
+ const v = req.headers[name.toLowerCase()];
1112
+ return Array.isArray(v) ? v[0] : (v ?? null);
1113
+ },
1114
+
1115
+ // server_req_method req -> string
1116
+ "server_req_method": (req: Request): string => {
1117
+ return req.method;
1118
+ },
1119
+
1120
+ // server_req_path req -> string
1121
+ "server_req_path": (req: Request): string => {
1122
+ return req.path;
1123
+ },
1124
+
1125
+ // Phase 57: 비동기 응답 보류 함수들
1126
+ // server_req_id -> string | null (현재 요청 ID)
1127
+ "server_req_id": (): string | null => {
1128
+ return currentRequestId;
1129
+ },
1130
+
1131
+ // server_hold_response reqId -> null (응답 보류)
1132
+ "server_hold_response": (reqId: string): null => {
1133
+ // 이 함수는 특정 요청의 응답을 보류한다고 표시
1134
+ // 실제 구현은: 핸들러가 null을 반환하면 자동으로 응답을 보류
1135
+ // reqId로 응답 객체를 저장해야 하는데, 현재 인터페이스에서는 res를 얻을 수 없음
1136
+ // 대신 requestId와 매칭되는 응답 객체를 펜딩 상태로 표시
1137
+ if (currentRequestId === reqId) {
1138
+ pendingResponses.set(reqId, true as any); // 플래그만 저장
1139
+ }
1140
+ return null;
1141
+ },
1142
+
1143
+ // server_send_held reqId status body -> boolean (보류된 응답 전송)
1144
+ "server_send_held": (reqId: string, status: number, body: any): boolean => {
1145
+ const isPending = pendingResponses.has(reqId);
1146
+ if (isPending) {
1147
+ pendingResponses.delete(reqId);
1148
+ return true;
1149
+ }
1150
+ return false;
1151
+ },
1152
+
1153
+ // ── WebSocket 터널 프록시 함수들 ─────────────────────────────
1154
+ // server_on_upgrade fnName -> null (WS upgrade 핸들러 등록)
1155
+ "server_on_upgrade": (fnName: string): null => {
1156
+ upgradeHandler = fnName;
1157
+ return null;
1158
+ },
1159
+
1160
+ // server_on_ws_message fnName -> null (클라이언트 WS 메시지 핸들러)
1161
+ "server_on_ws_message": (fnName: string): null => {
1162
+ wsClientMessageHandler = fnName;
1163
+ return null;
1164
+ },
1165
+
1166
+ // server_on_ws_close fnName -> null (클라이언트 WS 종료 핸들러)
1167
+ "server_on_ws_close": (fnName: string): null => {
1168
+ wsClientCloseHandler = fnName;
1169
+ return null;
1170
+ },
1171
+
1172
+ // ws_send_to_client sessionId data [isBinary] -> boolean
1173
+ "ws_send_to_client": (sessionId: string, data: string, isBinary: boolean = false): boolean => {
1174
+ const socket = wsPublicMap.get(sessionId) as any;
1175
+ if (!socket || socket.destroyed) return false;
1176
+ try {
1177
+ // RFC 6455 프레임 빌더 (서버→클라이언트, 마스킹 없음)
1178
+ const payload = isBinary ? Buffer.from(data, 'base64') : Buffer.from(data);
1179
+ const opcode = isBinary ? 0x02 : 0x01;
1180
+ let frame: Buffer;
1181
+
1182
+ if (payload.length < 126) {
1183
+ const h = Buffer.alloc(2);
1184
+ h[0] = 0x80 | opcode;
1185
+ h[1] = payload.length;
1186
+ frame = Buffer.concat([h, payload]);
1187
+ } else if (payload.length < 65536) {
1188
+ const h = Buffer.alloc(4);
1189
+ h[0] = 0x80 | opcode;
1190
+ h[1] = 126;
1191
+ h.writeUInt16BE(payload.length, 2);
1192
+ frame = Buffer.concat([h, payload]);
1193
+ } else {
1194
+ const h = Buffer.alloc(10);
1195
+ h[0] = 0x80 | opcode;
1196
+ h[1] = 127;
1197
+ h.writeBigUInt64BE(BigInt(payload.length), 2);
1198
+ frame = Buffer.concat([h, payload]);
1199
+ }
1200
+
1201
+ socket.write(frame);
1202
+ return true;
1203
+ } catch {
1204
+ return false;
1205
+ }
1206
+ },
1207
+
1208
+ // ws_close_client sessionId [code] -> null
1209
+ "ws_close_client": (sessionId: string, code: number = 1000): null => {
1210
+ const socket = wsPublicMap.get(sessionId) as any;
1211
+ if (socket && !socket.destroyed) {
1212
+ // RFC 6455 CLOSE 프레임
1213
+ const b = Buffer.alloc(4);
1214
+ b[0] = 0x88;
1215
+ b[1] = 0x02;
1216
+ b.writeUInt16BE(code, 2);
1217
+ socket.write(b);
1218
+ socket.end();
1219
+ wsPublicMap.delete(sessionId);
1220
+ }
1221
+ return null;
1222
+ },
1223
+
1224
+ // server_req_session_id req -> string | null
1225
+ "server_req_session_id": (req: any): string | null => {
1226
+ return req?.session_id ?? null;
1227
+ },
1228
+ };
1229
+ }