summaryrefslogtreecommitdiff
path: root/src/lang/js-lang/builtin/sbuiltin.ts
diff options
context:
space:
mode:
authorKai Stevenson <kai@kaistevenson.com>2025-11-06 00:18:26 -0800
committerKai Stevenson <kai@kaistevenson.com>2025-11-06 20:28:00 -0800
commit490b9c94fba16f484be3bb58b8a4a4880b9396bc (patch)
treea94bd52ca129828fe284ee96651018613e42f6c6 /src/lang/js-lang/builtin/sbuiltin.ts
parentd8a969e231135978c4dd1fa67559101f506ad6f3 (diff)
implement recursion properly with closures
Diffstat (limited to 'src/lang/js-lang/builtin/sbuiltin.ts')
-rw-r--r--src/lang/js-lang/builtin/sbuiltin.ts19
1 files changed, 18 insertions, 1 deletions
diff --git a/src/lang/js-lang/builtin/sbuiltin.ts b/src/lang/js-lang/builtin/sbuiltin.ts
index e5bc6ab..663c2f9 100644
--- a/src/lang/js-lang/builtin/sbuiltin.ts
+++ b/src/lang/js-lang/builtin/sbuiltin.ts
@@ -1,4 +1,4 @@
-import { callFn, getEvaluatedChildren } from "../core/eval";
+import { _evaluate, callFn, getEvaluatedChildren } from "../core/eval";
import { ASTNode, FnPrim, StackFrame } from "../../ts-lang";
type SBUILTIN = (node: ASTNode, frame: StackFrame) => any;
@@ -40,7 +40,24 @@ export const V_SBUILTIN_Map: SBUILTIN = (node, frame) => {
return values.map((v, i) => callFn(fn, [v, i], frame));
};
+export const V_SBUILTIN_IfElse: SBUILTIN = (node, frame) => {
+ const children = node.children;
+
+ if (children.length !== 3) {
+ throw new Error(`Invalid args for "if": ${JSON.stringify(children)}`);
+ }
+
+ const cond = _evaluate(children[0], frame);
+
+ if (typeof cond !== "boolean") {
+ throw new Error(`Condition value ${JSON.stringify(cond)} is not a boolean`);
+ }
+
+ return cond ? _evaluate(children[1], frame) : _evaluate(children[2], frame);
+};
+
export const nameToSBUILTIN: Record<string, SBUILTIN> = {
call: V_SBUILTIN_Call,
map: V_SBUILTIN_Map,
+ "?": V_SBUILTIN_IfElse,
};