Added short aliases make-buffer / buffer? / buffer-append! / buffer->string / buffer-length on both OCaml and JS hosts, sharing the existing StringBuffer value type. buffer-append! auto-coerces non-strings via inspect. Rewrote the OCaml host inspect function to walk a single shared Buffer.t instead of allocating O(n) intermediate strings via String.concat at every recursion level. inspect underlies sx-serialize and error-path formatting, so this benefits the tightest serialization paths. Median improvements (bin/bench_inspect.exe, best-of-3 of 9-run min): tree-d8 (75KB): 5.31ms -> 1.30ms (-76%) tree-d10 (679KB): 81.89ms -> 16.02ms (-80%) dict-1000: 0.80ms -> 0.31ms (-61%) list-2000: 0.74ms -> 0.33ms (-55%) Tests: OCaml 4545 -> 4550. JS 2591 -> 2596. Zero regressions. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
7404 lines
339 KiB
JavaScript
7404 lines
339 KiB
JavaScript
/**
|
||
* sx-ref.js — Generated from reference SX evaluator specification.
|
||
*
|
||
* Bootstrap-compiled from shared/sx/ref/{eval,render,primitives}.sx
|
||
* Compare against hand-written sx.js for correctness verification.
|
||
*
|
||
* DO NOT EDIT — regenerate with: python bootstrap_js.py
|
||
*/
|
||
;(function(global) {
|
||
"use strict";
|
||
|
||
// =========================================================================
|
||
// Equality — used by transpiled code (= a b) → sxEq(a, b)
|
||
// =========================================================================
|
||
function sxEq(a, b) {
|
||
if (a === b) return true;
|
||
if (a && b && a._sym && b._sym) return a.name === b.name;
|
||
if (a && b && a._kw && b._kw) return a.name === b.name;
|
||
if (a && b && a._vector && b._vector) {
|
||
if (a.arr.length !== b.arr.length) return false;
|
||
for (var _i = 0; _i < a.arr.length; _i++) {
|
||
if (!sxEq(a.arr[_i], b.arr[_i])) return false;
|
||
}
|
||
return true;
|
||
}
|
||
if (Array.isArray(a) && Array.isArray(b)) {
|
||
if (a.length !== b.length) return false;
|
||
for (var _j = 0; _j < a.length; _j++) {
|
||
if (!sxEq(a[_j], b[_j])) return false;
|
||
}
|
||
return true;
|
||
}
|
||
if (a && b && a._rational && b._rational) return a._n === b._n && a._d === b._d;
|
||
if (a && a._rational && typeof b === "number") return b === a._n / a._d;
|
||
if (b && b._rational && typeof a === "number") return a === b._n / b._d;
|
||
return false;
|
||
}
|
||
|
||
// =========================================================================
|
||
// Types
|
||
// =========================================================================
|
||
|
||
var NIL = Object.freeze({ _nil: true, toString: function() { return "nil"; } });
|
||
var SX_VERSION = "2026-05-07T02:05:49Z";
|
||
|
||
function isNil(x) { return x === NIL || x === null || x === undefined; }
|
||
function isSxTruthy(x) { return x !== false && !isNil(x); }
|
||
|
||
function Symbol(name) { this.name = name; }
|
||
Symbol.prototype.toString = function() { return this.name; };
|
||
Symbol.prototype._sym = true;
|
||
|
||
function Keyword(name) { this.name = name; }
|
||
Keyword.prototype.toString = function() { return ":" + this.name; };
|
||
Keyword.prototype._kw = true;
|
||
|
||
function Lambda(params, body, closure, name) {
|
||
this.params = params;
|
||
this.body = body;
|
||
this.closure = closure || {};
|
||
this.name = name || null;
|
||
}
|
||
Lambda.prototype._lambda = true;
|
||
|
||
function Component(name, params, hasChildren, body, closure, affinity) {
|
||
this.name = name;
|
||
this.params = params;
|
||
this.hasChildren = hasChildren;
|
||
this.body = body;
|
||
this.closure = closure || {};
|
||
this.affinity = affinity || "auto";
|
||
}
|
||
Component.prototype._component = true;
|
||
|
||
function Island(name, params, hasChildren, body, closure) {
|
||
this.name = name;
|
||
this.params = params;
|
||
this.hasChildren = hasChildren;
|
||
this.body = body;
|
||
this.closure = closure || {};
|
||
}
|
||
Island.prototype._island = true;
|
||
|
||
function Macro(params, restParam, body, closure, name) {
|
||
this.params = params;
|
||
this.restParam = restParam;
|
||
this.body = body;
|
||
this.closure = closure || {};
|
||
this.name = name || null;
|
||
}
|
||
Macro.prototype._macro = true;
|
||
|
||
function Thunk(expr, env) { this.expr = expr; this.env = env; }
|
||
Thunk.prototype._thunk = true;
|
||
|
||
function RawHTML(html) { this.html = html; }
|
||
RawHTML.prototype._raw = true;
|
||
|
||
function SxSpread(attrs) { this.attrs = attrs || {}; }
|
||
SxSpread.prototype._spread = true;
|
||
|
||
function SxVector(arr) { this.arr = arr || []; }
|
||
SxVector.prototype._vector = true;
|
||
|
||
var _paramUidCounter = 0;
|
||
function SxParameter(defaultVal, converter) {
|
||
this._uid = ++_paramUidCounter;
|
||
this._default = defaultVal;
|
||
this._converter = converter || null;
|
||
}
|
||
SxParameter.prototype._parameter = true;
|
||
function parameter_p(x) { return x != null && x._parameter === true; }
|
||
function parameterUid(p) { return p._uid; }
|
||
function parameterDefault(p) { return p._default; }
|
||
|
||
function SxCallccContinuation(capturedKont, windersLen) { this._captured = capturedKont; this._winders_len = windersLen !== undefined ? windersLen : 0; }
|
||
SxCallccContinuation.prototype._callcc = true;
|
||
function makeCallccContinuation(kont, windersLen) { return new SxCallccContinuation(kont, windersLen !== undefined ? windersLen : 0); }
|
||
function callccContinuation_p(x) { return x != null && x._callcc === true; }
|
||
function callccContinuationData(x) { return x._captured; }
|
||
function callccContinuationWindersLen(x) { return x._winders_len !== undefined ? x._winders_len : 0; }
|
||
|
||
function evalError_p(v) {
|
||
return v != null && typeof v === "object" && v["__eval_error__"] === true;
|
||
}
|
||
|
||
function sxApplyCek(f, args) {
|
||
try {
|
||
return typeof f === "function" ? f.apply(null, args) : f;
|
||
} catch (e) {
|
||
if (e && e._perform_request) throw e;
|
||
if (e && e._cek_suspend) throw e;
|
||
return {"__eval_error__": true, "message": e && e.message ? e.message : String(e)};
|
||
}
|
||
}
|
||
|
||
var _JIT_SKIP_SENTINEL = {"__jit_skip": true};
|
||
function jitTryCall(f, args) { return _JIT_SKIP_SENTINEL; }
|
||
function jitSkip_p(v) { return v === _JIT_SKIP_SENTINEL || (v != null && v["__jit_skip"] === true); }
|
||
|
||
var _scopeStacks = {};
|
||
|
||
function isSym(x) { return x != null && x._sym === true; }
|
||
function isKw(x) { return x != null && x._kw === true; }
|
||
|
||
function merge() {
|
||
var out = {};
|
||
for (var i = 0; i < arguments.length; i++) {
|
||
var d = arguments[i];
|
||
if (d) for (var k in d) out[k] = d[k];
|
||
}
|
||
return out;
|
||
}
|
||
|
||
function sxOr() {
|
||
for (var i = 0; i < arguments.length; i++) {
|
||
if (isSxTruthy(arguments[i])) return arguments[i];
|
||
}
|
||
return arguments.length ? arguments[arguments.length - 1] : false;
|
||
}
|
||
|
||
// =========================================================================
|
||
// Platform interface — JS implementation
|
||
// =========================================================================
|
||
|
||
function typeOf(x) {
|
||
if (isNil(x)) return "nil";
|
||
if (typeof x === "number") return "number";
|
||
if (typeof x === "string") return "string";
|
||
if (typeof x === "boolean") return "boolean";
|
||
if (x._sym) return "symbol";
|
||
if (x._kw) return "keyword";
|
||
if (x._thunk) return "thunk";
|
||
if (x._lambda) return "lambda";
|
||
if (x._component) return "component";
|
||
if (x._island) return "island";
|
||
if (x._spread) return "spread";
|
||
if (x._macro) return "macro";
|
||
if (x._raw) return "raw-html";
|
||
if (x._sx_expr) return "sx-expr";
|
||
if (x._char) return "char";
|
||
if (x._eof) return "eof-object";
|
||
if (x._port) return x._kind === "input" ? "input-port" : "output-port";
|
||
if (x._vector) return "vector";
|
||
if (x._string_buffer) return "string-buffer";
|
||
if (x._hash_table) return "hash-table";
|
||
if (x._sxset) return "set";
|
||
if (x._regexp) return "regexp";
|
||
if (x._bytevector) return "bytevector";
|
||
if (x._rational) return "rational";
|
||
if (x._adtv) return x._type;
|
||
if (typeof Node !== "undefined" && x instanceof Node) return "dom-node";
|
||
if (Array.isArray(x)) return "list";
|
||
if (typeof x === "object") return "dict";
|
||
return "unknown";
|
||
}
|
||
|
||
// AdtValue — native algebraic data type instance (Step 6 mirror of OCaml Step 5).
|
||
// Constructed by define-type. Carries _adt:true plus _adtv:true tag so type-of
|
||
// returns the type name rather than "dict". dict? remains true (shim approach)
|
||
// so spec-level match-pattern in evaluator.sx works without changes.
|
||
function makeAdtValue(typeName, ctorName, fields) {
|
||
return {
|
||
_adtv: true,
|
||
_adt: true,
|
||
_type: typeName,
|
||
_ctor: ctorName,
|
||
_fields: fields
|
||
};
|
||
}
|
||
function isAdtValue(x) {
|
||
return x !== null && typeof x === "object" && x._adtv === true;
|
||
}
|
||
|
||
function symbolName(s) { return s.name; }
|
||
function keywordName(k) { return k.name; }
|
||
function makeSymbol(n) { return new Symbol(n); }
|
||
function makeKeyword(n) { return new Keyword(n); }
|
||
|
||
function makeLambda(params, body, env) { return new Lambda(params, body, env); }
|
||
function makeComponent(name, params, hasChildren, body, env, affinity) {
|
||
return new Component(name, params, hasChildren, body, env, affinity);
|
||
}
|
||
function makeMacro(params, restParam, body, env, name) {
|
||
return new Macro(params, restParam, body, env, name);
|
||
}
|
||
function makeThunk(expr, env) { return new Thunk(expr, env); }
|
||
|
||
function makeSpread(attrs) { return new SxSpread(attrs || {}); }
|
||
function isSpread(x) { return x != null && x._spread === true; }
|
||
function spreadAttrs(s) { return s && s._spread ? s.attrs : {}; }
|
||
|
||
function scopePush(name, value) {
|
||
if (!_scopeStacks[name]) _scopeStacks[name] = [];
|
||
_scopeStacks[name].push({value: value !== undefined ? value : NIL, emitted: [], dedup: false});
|
||
}
|
||
function scopePop(name) {
|
||
if (_scopeStacks[name] && _scopeStacks[name].length) _scopeStacks[name].pop();
|
||
}
|
||
// Aliases — provide-push!/provide-pop! map to scope-push!/scope-pop!
|
||
var providePush = scopePush;
|
||
var providePop = scopePop;
|
||
|
||
function sxContext(name) {
|
||
if (_scopeStacks[name] && _scopeStacks[name].length) {
|
||
return _scopeStacks[name][_scopeStacks[name].length - 1].value;
|
||
}
|
||
if (arguments.length > 1) return arguments[1];
|
||
throw new Error("No provider for: " + name);
|
||
}
|
||
function sxEmit(name, value) {
|
||
if (_scopeStacks[name] && _scopeStacks[name].length) {
|
||
var entry = _scopeStacks[name][_scopeStacks[name].length - 1];
|
||
if (entry.dedup && entry.emitted.indexOf(value) !== -1) return NIL;
|
||
entry.emitted.push(value);
|
||
}
|
||
return NIL;
|
||
}
|
||
function sxEmitted(name) {
|
||
if (_scopeStacks[name] && _scopeStacks[name].length) {
|
||
return _scopeStacks[name][_scopeStacks[name].length - 1].emitted.slice();
|
||
}
|
||
return [];
|
||
}
|
||
function sxCollect(bucket, value) {
|
||
if (!_scopeStacks[bucket] || !_scopeStacks[bucket].length) {
|
||
if (!_scopeStacks[bucket]) _scopeStacks[bucket] = [];
|
||
_scopeStacks[bucket].push({value: NIL, emitted: [], dedup: true});
|
||
}
|
||
var entry = _scopeStacks[bucket][_scopeStacks[bucket].length - 1];
|
||
if (entry.emitted.indexOf(value) === -1) entry.emitted.push(value);
|
||
}
|
||
function sxCollected(bucket) {
|
||
return sxEmitted(bucket);
|
||
}
|
||
function sxClearCollected(bucket) {
|
||
if (_scopeStacks[bucket] && _scopeStacks[bucket].length) {
|
||
_scopeStacks[bucket][_scopeStacks[bucket].length - 1].emitted = [];
|
||
}
|
||
}
|
||
|
||
function lambdaParams(f) { return f.params; }
|
||
function lambdaBody(f) { return f.body; }
|
||
function lambdaClosure(f) { return f.closure; }
|
||
function lambdaName(f) { return f.name; }
|
||
function setLambdaName(f, n) { f.name = n; }
|
||
|
||
function componentParams(c) { return c.params; }
|
||
function componentBody(c) { return c.body; }
|
||
function componentClosure(c) { return c.closure; }
|
||
function componentHasChildren(c) { return c.hasChildren; }
|
||
function componentName(c) { return c.name; }
|
||
function componentFile(c) { return (c && c.file) ? c.file : NIL; }
|
||
function componentAffinity(c) { return c.affinity || "auto"; }
|
||
function componentParamTypes(c) { return (c && c._paramTypes) ? c._paramTypes : NIL; }
|
||
function componentSetParamTypes_b(c, t) { if (c) c._paramTypes = t; return NIL; }
|
||
|
||
function macroParams(m) { return m.params; }
|
||
function macroRestParam(m) { return m.restParam; }
|
||
function macroBody(m) { return m.body; }
|
||
function macroClosure(m) { return m.closure; }
|
||
|
||
function isThunk(x) { return x != null && x._thunk === true; }
|
||
function thunkExpr(t) { return t.expr; }
|
||
function thunkEnv(t) { return t.env; }
|
||
|
||
function isCallable(x) { return typeof x === "function" || (x != null && x._lambda === true); }
|
||
function isLambda(x) { return x != null && x._lambda === true; }
|
||
function isComponent(x) { return x != null && x._component === true; }
|
||
function isIsland(x) { return x != null && x._island === true; }
|
||
function isMacro(x) { return x != null && x._macro === true; }
|
||
function isIdentical(a, b) { return a === b; }
|
||
|
||
// Island platform
|
||
function makeIsland(name, params, hasChildren, body, env) {
|
||
return new Island(name, params, hasChildren, body, env);
|
||
}
|
||
|
||
// JSON / dict helpers for island state serialization
|
||
function jsonSerialize(obj) {
|
||
return JSON.stringify(obj);
|
||
}
|
||
function isEmptyDict(d) {
|
||
if (!d || typeof d !== "object") return true;
|
||
for (var k in d) if (d.hasOwnProperty(k)) return false;
|
||
return true;
|
||
}
|
||
|
||
function envHas(env, name) { return name in env; }
|
||
function envGet(env, name) { return env[name]; }
|
||
function envBind(env, name, val) {
|
||
// Direct property set — creates or overwrites on THIS env only.
|
||
// Used by let, define, defcomp, lambda param binding.
|
||
env[name] = val;
|
||
}
|
||
function envSet(env, name, val) {
|
||
// Walk prototype chain to find where the variable is defined (for set!)
|
||
var obj = env;
|
||
while (obj !== null && obj !== Object.prototype) {
|
||
if (obj.hasOwnProperty(name)) { obj[name] = val; return; }
|
||
obj = Object.getPrototypeOf(obj);
|
||
}
|
||
// Not found in any parent scope — set on the immediate env
|
||
env[name] = val;
|
||
}
|
||
function envExtend(env) { return Object.create(env); }
|
||
function envMerge(base, overlay) {
|
||
// Same env or overlay is descendant of base — just extend, no copy.
|
||
// This prevents set! inside lambdas from modifying shadow copies.
|
||
if (base === overlay) return Object.create(base);
|
||
var p = overlay;
|
||
for (var d = 0; p && p !== Object.prototype && d < 100; d++) {
|
||
if (p === base) return Object.create(base);
|
||
p = Object.getPrototypeOf(p);
|
||
}
|
||
// General case: extend base, copy ONLY overlay properties that don't
|
||
// exist in the base chain (avoids shadowing closure bindings).
|
||
var child = Object.create(base);
|
||
if (overlay) {
|
||
for (var k in overlay) {
|
||
if (overlay.hasOwnProperty(k) && !(k in base)) child[k] = overlay[k];
|
||
}
|
||
}
|
||
return child;
|
||
}
|
||
|
||
function dictSet(d, k, v) { d[k] = v; return v; }
|
||
function dictGet(d, k) { var v = d[k]; return v !== undefined ? v : NIL; }
|
||
|
||
// Render-expression detection — lets the evaluator delegate to the active adapter.
|
||
// Matches HTML tags, SVG tags, <>, raw!, ~components, html: prefix, custom elements.
|
||
// Placeholder — overridden by transpiled version from render.sx
|
||
function isRenderExpr(expr) { return false; }
|
||
|
||
// Last error continuation — saved when a raise goes unhandled, for post-mortem inspection.
|
||
var _lastErrorKont_ = null;
|
||
|
||
// hostError — throw a host-level error that propagates out of cekRun.
|
||
function hostError(msg) { throw new Error(typeof msg === "string" ? msg : inspect(msg)); }
|
||
|
||
// hostWarn — emit a host-level warning to console (no-op if console missing).
|
||
function hostWarn(msg) {
|
||
var m = typeof msg === "string" ? msg : inspect(msg);
|
||
if (typeof console !== "undefined" && console.warn) console.warn(m);
|
||
return NIL;
|
||
}
|
||
|
||
// Render dispatch — call the active adapter's render function.
|
||
// Set by each adapter when loaded; defaults to identity (no rendering).
|
||
var _renderExprFn = null;
|
||
|
||
// Render mode flag — set by render-to-html/aser, checked by eval-list.
|
||
// When false, render expressions fall through to evalCall.
|
||
var _renderMode = false;
|
||
function renderActiveP() { return _renderMode; }
|
||
function setRenderActiveB(val) { _renderMode = !!val; }
|
||
|
||
function renderExpr(expr, env) {
|
||
if (_renderExprFn) return _renderExprFn(expr, env);
|
||
// No adapter loaded — fall through to evalCall
|
||
return evalCall(first(expr), rest(expr), env);
|
||
}
|
||
|
||
function stripPrefix(s, prefix) {
|
||
return s.indexOf(prefix) === 0 ? s.slice(prefix.length) : s;
|
||
}
|
||
|
||
function error(msg) { throw new Error(msg); }
|
||
function inspect(x) {
|
||
if (x !== null && typeof x === "object" && x._adtv === true) {
|
||
var fs = x._fields || [];
|
||
if (fs.length === 0) return "(" + x._ctor + ")";
|
||
var parts = [];
|
||
for (var i = 0; i < fs.length; i++) parts.push(inspect(fs[i]));
|
||
return "(" + x._ctor + " " + parts.join(" ") + ")";
|
||
}
|
||
return JSON.stringify(x);
|
||
}
|
||
function debugLog() { console.error.apply(console, ["[sx-debug]"].concat(Array.prototype.slice.call(arguments))); }
|
||
|
||
|
||
|
||
// =========================================================================
|
||
// Primitives
|
||
// =========================================================================
|
||
|
||
var PRIMITIVES = {};
|
||
|
||
// core.arithmetic
|
||
function _ratMake(n, d) {
|
||
if (d === 0) throw new Error("division by zero");
|
||
var r = new SxRational(n, d);
|
||
return r._d === 1 ? r._n : r;
|
||
}
|
||
function _ratN(x) { return x && x._rational ? x._n : x; }
|
||
function _ratD(x) { return x && x._rational ? x._d : 1; }
|
||
function _hasFloat(args) {
|
||
for (var i = 0; i < args.length; i++) {
|
||
var x = args[i];
|
||
if (typeof x === "number" && !Number.isInteger(x)) return true;
|
||
}
|
||
return false;
|
||
}
|
||
function _ratToFloat(x) { return x && x._rational ? x._n / x._d : x; }
|
||
PRIMITIVES["+"] = function() {
|
||
var hasRat = false;
|
||
for (var i = 0; i < arguments.length; i++) if (arguments[i] && arguments[i]._rational) { hasRat = true; break; }
|
||
if (!hasRat) { var s = 0; for (var i = 0; i < arguments.length; i++) s += arguments[i]; return s; }
|
||
if (_hasFloat(arguments)) { var s = 0; for (var i = 0; i < arguments.length; i++) s += _ratToFloat(arguments[i]); return s; }
|
||
var an = 0, ad = 1;
|
||
for (var i = 0; i < arguments.length; i++) {
|
||
var bn = _ratN(arguments[i]), bd = _ratD(arguments[i]);
|
||
an = an * bd + bn * ad; ad = ad * bd;
|
||
}
|
||
return _ratMake(an, ad);
|
||
};
|
||
PRIMITIVES["-"] = function() {
|
||
if (arguments.length === 0) return 0;
|
||
var hasRat = false;
|
||
for (var i = 0; i < arguments.length; i++) if (arguments[i] && arguments[i]._rational) { hasRat = true; break; }
|
||
if (!hasRat) return arguments.length === 1 ? -arguments[0] : arguments[0] - arguments[1];
|
||
if (_hasFloat(arguments)) {
|
||
if (arguments.length === 1) return -_ratToFloat(arguments[0]);
|
||
var s = _ratToFloat(arguments[0]);
|
||
for (var i = 1; i < arguments.length; i++) s -= _ratToFloat(arguments[i]);
|
||
return s;
|
||
}
|
||
if (arguments.length === 1) { var x = arguments[0]; return x._rational ? _ratMake(-x._n, x._d) : -x; }
|
||
var an = _ratN(arguments[0]), ad = _ratD(arguments[0]);
|
||
for (var i = 1; i < arguments.length; i++) {
|
||
var bn = _ratN(arguments[i]), bd = _ratD(arguments[i]);
|
||
an = an * bd - bn * ad; ad = ad * bd;
|
||
}
|
||
return _ratMake(an, ad);
|
||
};
|
||
PRIMITIVES["*"] = function() {
|
||
var hasRat = false;
|
||
for (var i = 0; i < arguments.length; i++) if (arguments[i] && arguments[i]._rational) { hasRat = true; break; }
|
||
if (!hasRat) { var s = 1; for (var i = 0; i < arguments.length; i++) s *= arguments[i]; return s; }
|
||
if (_hasFloat(arguments)) { var s = 1; for (var i = 0; i < arguments.length; i++) s *= _ratToFloat(arguments[i]); return s; }
|
||
var an = 1, ad = 1;
|
||
for (var i = 0; i < arguments.length; i++) { an *= _ratN(arguments[i]); ad *= _ratD(arguments[i]); }
|
||
return _ratMake(an, ad);
|
||
};
|
||
PRIMITIVES["/"] = function(a, b) {
|
||
var aRat = a && a._rational, bRat = b && b._rational;
|
||
if (!aRat && !bRat) return a / b;
|
||
if (typeof a === "number" && !Number.isInteger(a) || typeof b === "number" && !Number.isInteger(b))
|
||
return _ratToFloat(a) / _ratToFloat(b);
|
||
return _ratMake(_ratN(a) * _ratD(b), _ratD(a) * _ratN(b));
|
||
};
|
||
PRIMITIVES["mod"] = function(a, b) { return a % b; };
|
||
PRIMITIVES["inc"] = function(n) { return n + 1; };
|
||
PRIMITIVES["dec"] = function(n) { return n - 1; };
|
||
PRIMITIVES["abs"] = Math.abs;
|
||
PRIMITIVES["floor"] = Math.floor;
|
||
PRIMITIVES["ceil"] = Math.ceil;
|
||
PRIMITIVES["round"] = function(x, n) {
|
||
if (n === undefined || n === 0) return Math.round(x);
|
||
var f = Math.pow(10, n); return Math.round(x * f) / f;
|
||
};
|
||
PRIMITIVES["truncate"] = Math.trunc;
|
||
PRIMITIVES["remainder"] = function(a, b) { return a % b; };
|
||
PRIMITIVES["modulo"] = function(a, b) { var r = a % b; return (r !== 0 && (r < 0) !== (b < 0)) ? r + b : r; };
|
||
PRIMITIVES["min"] = Math.min;
|
||
PRIMITIVES["max"] = Math.max;
|
||
PRIMITIVES["sqrt"] = Math.sqrt;
|
||
PRIMITIVES["pow"] = Math.pow;
|
||
PRIMITIVES["clamp"] = function(x, lo, hi) { return Math.max(lo, Math.min(hi, x)); };
|
||
PRIMITIVES["random-int"] = function(lo, hi) { return Math.floor(Math.random() * (hi - lo + 1)) + lo; };
|
||
PRIMITIVES["exact->inexact"] = function(x) {
|
||
if (x && x._rational) return x._n / x._d;
|
||
return x;
|
||
};
|
||
PRIMITIVES["inexact->exact"] = Math.round;
|
||
PRIMITIVES["parse-number"] = function(s) { var n = Number(s); return isNaN(n) ? null : n; };
|
||
|
||
|
||
// core.comparison
|
||
function _ratCmp(a, b) {
|
||
return _ratN(a) * _ratD(b) - _ratN(b) * _ratD(a);
|
||
}
|
||
PRIMITIVES["="] = sxEq;
|
||
PRIMITIVES["!="] = function(a, b) { return !sxEq(a, b); };
|
||
PRIMITIVES["<"] = function(a, b) {
|
||
if ((a && a._rational) || (b && b._rational)) return _ratCmp(a, b) < 0;
|
||
return a < b;
|
||
};
|
||
PRIMITIVES[">"] = function(a, b) {
|
||
if ((a && a._rational) || (b && b._rational)) return _ratCmp(a, b) > 0;
|
||
return a > b;
|
||
};
|
||
PRIMITIVES["<="] = function(a, b) {
|
||
if ((a && a._rational) || (b && b._rational)) return _ratCmp(a, b) <= 0;
|
||
return a <= b;
|
||
};
|
||
PRIMITIVES[">="] = function(a, b) {
|
||
if ((a && a._rational) || (b && b._rational)) return _ratCmp(a, b) >= 0;
|
||
return a >= b;
|
||
};
|
||
|
||
|
||
// core.logic
|
||
PRIMITIVES["not"] = function(x) { return !isSxTruthy(x); };
|
||
|
||
|
||
// core.predicates
|
||
PRIMITIVES["nil?"] = isNil;
|
||
PRIMITIVES["number?"] = function(x) { return typeof x === "number" || (x != null && x._rational === true); };
|
||
PRIMITIVES["integer?"] = function(x) { return typeof x === "number" && Number.isInteger(x); };
|
||
PRIMITIVES["float?"] = function(x) { return typeof x === "number" && !Number.isInteger(x); };
|
||
PRIMITIVES["exact?"] = function(x) { return (typeof x === "number" && Number.isInteger(x)) || (x != null && x._rational === true); };
|
||
PRIMITIVES["inexact?"] = function(x) { return typeof x === "number" && !Number.isInteger(x); };
|
||
PRIMITIVES["string?"] = function(x) { return typeof x === "string"; };
|
||
PRIMITIVES["list?"] = Array.isArray;
|
||
PRIMITIVES["dict?"] = function(x) { return x !== null && typeof x === "object" && !Array.isArray(x) && !x._sym && !x._kw && !x._string_buffer && !x._vector && !x._hash_table && !x._rational; };
|
||
PRIMITIVES["empty?"] = function(c) { return isNil(c) || (Array.isArray(c) ? c.length === 0 : typeof c === "string" ? c.length === 0 : Object.keys(c).length === 0); };
|
||
PRIMITIVES["contains?"] = function(c, k) {
|
||
if (typeof c === "string") return c.indexOf(String(k)) !== -1;
|
||
if (Array.isArray(c)) return c.indexOf(k) !== -1;
|
||
return k in c;
|
||
};
|
||
PRIMITIVES["odd?"] = function(n) { return n % 2 !== 0; };
|
||
PRIMITIVES["even?"] = function(n) { return n % 2 === 0; };
|
||
PRIMITIVES["zero?"] = function(n) { return n === 0; };
|
||
PRIMITIVES["boolean?"] = function(x) { return x === true || x === false; };
|
||
PRIMITIVES["symbol?"] = function(x) { return x != null && x._sym === true; };
|
||
PRIMITIVES["keyword?"] = function(x) { return x != null && x._kw === true; };
|
||
PRIMITIVES["adt?"] = function(x) { return x !== null && typeof x === "object" && x._adtv === true; };
|
||
PRIMITIVES["component-affinity"] = componentAffinity;
|
||
|
||
|
||
// core.strings
|
||
PRIMITIVES["str"] = function() {
|
||
var p = [];
|
||
for (var i = 0; i < arguments.length; i++) {
|
||
var v = arguments[i]; if (isNil(v)) continue; p.push(String(v));
|
||
}
|
||
return p.join("");
|
||
};
|
||
PRIMITIVES["upper"] = function(s) { return String(s).toUpperCase(); };
|
||
PRIMITIVES["lower"] = function(s) { return String(s).toLowerCase(); };
|
||
PRIMITIVES["trim"] = function(s) { return String(s).trim(); };
|
||
PRIMITIVES["split"] = function(s, sep) { return String(s).split(sep || " "); };
|
||
PRIMITIVES["join"] = function(sep, coll) { return coll.join(sep); };
|
||
PRIMITIVES["replace"] = function(s, old, nw) { return s.split(old).join(nw); };
|
||
PRIMITIVES["index-of"] = function(s, needle, from) {
|
||
if (Array.isArray(s)) {
|
||
var _start = from || 0;
|
||
for (var _i = _start; _i < s.length; _i++) {
|
||
var _a = s[_i];
|
||
if (_a === needle) return _i;
|
||
if (_a != null && needle != null && typeof _a === "object" && typeof needle === "object") {
|
||
if ((_a._sym && needle._sym || _a._kw && needle._kw) && _a.name === needle.name) return _i;
|
||
}
|
||
}
|
||
return NIL;
|
||
}
|
||
return String(s).indexOf(needle, from || 0);
|
||
};
|
||
PRIMITIVES["starts-with?"] = function(s, p) { return String(s).indexOf(p) === 0; };
|
||
PRIMITIVES["ends-with?"] = function(s, p) { var str = String(s); return str.indexOf(p, str.length - p.length) !== -1; };
|
||
PRIMITIVES["slice"] = function(c, a, b) { if (!c || typeof c.slice !== "function") { console.error("[sx-debug] slice called on non-sliceable:", typeof c, c, "a=", a, "b=", b, new Error().stack); return []; } return b !== undefined ? c.slice(a, b) : c.slice(a); };
|
||
PRIMITIVES["substring"] = function(s, a, b) { return String(s).substring(a, b); };
|
||
PRIMITIVES["char-from-code"] = function(n) { return String.fromCharCode(n); };
|
||
PRIMITIVES["char-code"] = function(s) { return String(s).charCodeAt(0); };
|
||
var charCode = PRIMITIVES["char-code"];
|
||
function makeChar(n) { return {_char: true, codepoint: n}; }
|
||
PRIMITIVES["make-char"] = makeChar;
|
||
var isChar = function(v) { return v != null && typeof v === "object" && v._char === true; };
|
||
PRIMITIVES["char?"] = isChar;
|
||
var charToInteger = function(c) { return c.codepoint; };
|
||
PRIMITIVES["char->integer"] = charToInteger;
|
||
var charUpcase = function(c) { return makeChar(String.fromCharCode(c.codepoint).toUpperCase().charCodeAt(0)); };
|
||
PRIMITIVES["char-upcase"] = charUpcase;
|
||
var charDowncase = function(c) { return makeChar(String.fromCharCode(c.codepoint).toLowerCase().charCodeAt(0)); };
|
||
PRIMITIVES["char-downcase"] = charDowncase;
|
||
PRIMITIVES["char=?"] = function(a, b) { return a.codepoint === b.codepoint; };
|
||
PRIMITIVES["char<?"] = function(a, b) { return a.codepoint < b.codepoint; };
|
||
PRIMITIVES["char>?"] = function(a, b) { return a.codepoint > b.codepoint; };
|
||
PRIMITIVES["char<=?"] = function(a, b) { return a.codepoint <= b.codepoint; };
|
||
PRIMITIVES["char>=?"] = function(a, b) { return a.codepoint >= b.codepoint; };
|
||
PRIMITIVES["char-ci=?"] = function(a, b) { return charDowncase(a).codepoint === charDowncase(b).codepoint; };
|
||
PRIMITIVES["char-ci<?"] = function(a, b) { return charDowncase(a).codepoint < charDowncase(b).codepoint; };
|
||
PRIMITIVES["char-ci>?"] = function(a, b) { return charDowncase(a).codepoint > charDowncase(b).codepoint; };
|
||
PRIMITIVES["char-ci<=?"] = function(a, b) { return charDowncase(a).codepoint <= charDowncase(b).codepoint; };
|
||
PRIMITIVES["char-ci>=?"] = function(a, b) { return charDowncase(a).codepoint >= charDowncase(b).codepoint; };
|
||
PRIMITIVES["char-alphabetic?"] = function(c) { var n = c.codepoint; return (n >= 65 && n <= 90) || (n >= 97 && n <= 122); };
|
||
PRIMITIVES["char-numeric?"] = function(c) { var n = c.codepoint; return n >= 48 && n <= 57; };
|
||
PRIMITIVES["char-whitespace?"] = function(c) { var n = c.codepoint; return n === 32 || n === 9 || n === 10 || n === 13; };
|
||
PRIMITIVES["char-upper-case?"] = function(c) { var n = c.codepoint; return n >= 65 && n <= 90; };
|
||
PRIMITIVES["char-lower-case?"] = function(c) { var n = c.codepoint; return n >= 97 && n <= 122; };
|
||
PRIMITIVES["string->list"] = function(s) {
|
||
var chars = []; var str = String(s);
|
||
for (var i = 0; i < str.length; i++) chars.push(makeChar(str.charCodeAt(i)));
|
||
return chars;
|
||
};
|
||
PRIMITIVES["list->string"] = function(chars) {
|
||
return chars.map(function(c) { return String.fromCharCode(c.codepoint); }).join('');
|
||
};
|
||
// Phase 14: string ports + eof-object
|
||
var _eof = {_eof: true};
|
||
PRIMITIVES["eof-object"] = function() { return _eof; };
|
||
PRIMITIVES["eof-object?"] = function(v) { return v != null && v._eof === true; };
|
||
var isEofObject = PRIMITIVES["eof-object?"];
|
||
PRIMITIVES["open-input-string"] = function(s) {
|
||
return {_port: true, _kind: "input", _source: String(s), _pos: 0, _closed: false};
|
||
};
|
||
PRIMITIVES["open-output-string"] = function() {
|
||
return {_port: true, _kind: "output", _buffer: "", _closed: false};
|
||
};
|
||
PRIMITIVES["get-output-string"] = function(p) {
|
||
if (!p || p._kind !== "output") throw new Error("get-output-string: expected output port");
|
||
return p._buffer;
|
||
};
|
||
PRIMITIVES["port?"] = function(v) { return v != null && v._port === true; };
|
||
PRIMITIVES["input-port?"] = function(v) { return v != null && v._port === true && v._kind === "input"; };
|
||
PRIMITIVES["output-port?"] = function(v) { return v != null && v._port === true && v._kind === "output"; };
|
||
PRIMITIVES["close-port"] = function(p) {
|
||
if (p && p._port) p._closed = true;
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["read-char"] = function(p) {
|
||
if (p === undefined || p === NIL || p == null) {
|
||
return _eof; // no stdin in this env
|
||
}
|
||
if (!p._port || p._kind !== "input") throw new Error("read-char: expected input port");
|
||
if (p._closed || p._pos >= p._source.length) return _eof;
|
||
var cp = p._source.charCodeAt(p._pos);
|
||
p._pos++;
|
||
return makeChar(cp);
|
||
};
|
||
PRIMITIVES["peek-char"] = function(p) {
|
||
if (p === undefined || p === NIL || p == null) return _eof;
|
||
if (!p._port || p._kind !== "input") throw new Error("peek-char: expected input port");
|
||
if (p._closed || p._pos >= p._source.length) return _eof;
|
||
return makeChar(p._source.charCodeAt(p._pos));
|
||
};
|
||
PRIMITIVES["read-line"] = function(p) {
|
||
if (p === undefined || p === NIL || p == null) return _eof;
|
||
if (!p._port || p._kind !== "input") throw new Error("read-line: expected input port");
|
||
if (p._closed || p._pos >= p._source.length) return _eof;
|
||
var start = p._pos;
|
||
while (p._pos < p._source.length && p._source[p._pos] !== '\n') p._pos++;
|
||
var line = p._source.slice(start, p._pos);
|
||
if (p._pos < p._source.length) p._pos++; // skip
|
||
|
||
return line;
|
||
};
|
||
PRIMITIVES["write-char"] = function(c, p) {
|
||
if (!p || !p._port || p._kind !== "output") throw new Error("write-char: expected char and output port");
|
||
if (!p._closed) p._buffer += String.fromCharCode(c.codepoint);
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["write-string"] = function(s, p) {
|
||
if (!p || !p._port || p._kind !== "output") throw new Error("write-string: expected string and output port");
|
||
if (!p._closed) p._buffer += String(s);
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["char-ready?"] = function(p) {
|
||
if (p === undefined || p === NIL || p == null) return false;
|
||
if (!p._port || p._kind !== "input") return false;
|
||
return !p._closed && p._pos < p._source.length;
|
||
};
|
||
// read/write/display
|
||
var _sxBs92 = String.fromCharCode(92);
|
||
function sxReadNormalize(src) {
|
||
var out = "", i = 0, n = src.length;
|
||
while (i < n) {
|
||
if (src[i] === '"') {
|
||
out += '"'; i++;
|
||
while (i < n) {
|
||
if (src[i] === _sxBs92 && i+1 < n) { out += src[i]; out += src[i+1]; i += 2; continue; }
|
||
if (src[i] === '"') { out += src[i++]; break; }
|
||
out += src[i++];
|
||
}
|
||
} else if (src[i] === '#' && i+1 < n && (src[i+1] === 't' || src[i+1] === 'f')) {
|
||
var nc2 = i+2 < n ? src[i+2] : '';
|
||
if (!nc2 || !/[a-zA-Z0-9_]/.test(nc2)) {
|
||
out += (src[i+1] === 't') ? 'true' : 'false';
|
||
i += 2;
|
||
} else { out += src[i++]; }
|
||
} else { out += src[i++]; }
|
||
}
|
||
return out;
|
||
}
|
||
function sxReadConvert(v) {
|
||
if (Array.isArray(v) && v.length === 0) return NIL;
|
||
if (Array.isArray(v)) return v.map(sxReadConvert);
|
||
return v;
|
||
}
|
||
PRIMITIVES["read"] = function() {
|
||
var p = arguments.length > 0 && arguments[0] && arguments[0]._port ? arguments[0] : null;
|
||
if (!p || p._kind !== "input" || p._closed) return _eof;
|
||
if (!p._forms) {
|
||
var sxP = PRIMITIVES["sx-parse"];
|
||
var src = sxReadNormalize(p._source.slice(p._pos || 0));
|
||
p._forms = sxP ? (sxP(src) || []) : [];
|
||
p._form_idx = 0;
|
||
}
|
||
if (p._form_idx >= p._forms.length) return _eof;
|
||
return sxReadConvert(p._forms[p._form_idx++]);
|
||
};
|
||
var _sxBs = String.fromCharCode(92);
|
||
var _sxDq = String.fromCharCode(34);
|
||
function sxWriteVal(v, mode) {
|
||
if (v === null || v === undefined || v === NIL) return "()";
|
||
if (v && v._eof) return "#!eof";
|
||
if (typeof v === "boolean") return v ? "#t" : "#f";
|
||
if (typeof v === "number") return String(v);
|
||
if (v && v._rational) return v._n + "/" + v._d;
|
||
if (typeof v === "string") {
|
||
if (mode === "display") return v;
|
||
return _sxDq + v.split("").map(function(c) {
|
||
var n = c.charCodeAt(0);
|
||
if (n === 34) return _sxBs + _sxDq;
|
||
if (n === 92) return _sxBs + _sxBs;
|
||
if (n === 10) return _sxBs + "n";
|
||
if (n === 13) return _sxBs + "r";
|
||
if (n === 9) return _sxBs + "t";
|
||
return c;
|
||
}).join("") + _sxDq;
|
||
}
|
||
if (v && v._char) {
|
||
if (mode === "display") return String.fromCodePoint(v.codepoint);
|
||
var cp = v.codepoint;
|
||
if (cp === 32) return "#" + _sxBs + "space";
|
||
if (cp === 10) return "#" + _sxBs + "newline";
|
||
if (cp === 9) return "#" + _sxBs + "tab";
|
||
return "#" + _sxBs + String.fromCodePoint(cp);
|
||
}
|
||
if (v && v._sym) return v.name;
|
||
if (v && v._kw) return ":" + v.name;
|
||
if (Array.isArray(v)) return "(" + v.map(function(x){ return sxWriteVal(x, mode); }).join(" ") + ")";
|
||
return String(v);
|
||
}
|
||
PRIMITIVES["write"] = function() {
|
||
var val = arguments[0], port = arguments[1];
|
||
var s = sxWriteVal(val, "write");
|
||
if (port && port._port && port._kind === "output" && !port._closed) port._buffer += s;
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["display"] = function() {
|
||
var val = arguments[0], port = arguments[1];
|
||
var s = sxWriteVal(val, "display");
|
||
if (port && port._port && port._kind === "output" && !port._closed) port._buffer += s;
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["newline"] = function() {
|
||
var port = arguments[0];
|
||
if (port && port._port && port._kind === "output" && !port._closed) port._buffer += String.fromCharCode(10);
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["write-to-string"] = function(val) { return sxWriteVal(val, "write"); };
|
||
PRIMITIVES["display-to-string"] = function(val) { return sxWriteVal(val, "display"); };
|
||
PRIMITIVES["current-input-port"] = function() { return NIL; };
|
||
PRIMITIVES["current-output-port"] = function() { return NIL; };
|
||
PRIMITIVES["current-error-port"] = function() { return NIL; };
|
||
PRIMITIVES["string-length"] = function(s) { return String(s).length; };
|
||
var stringLength = PRIMITIVES["string-length"];
|
||
PRIMITIVES["string-contains?"] = function(s, sub) { return String(s).indexOf(String(sub)) !== -1; };
|
||
PRIMITIVES["concat"] = function() {
|
||
var out = [];
|
||
for (var i = 0; i < arguments.length; i++) if (!isNil(arguments[i])) out = out.concat(arguments[i]);
|
||
return out;
|
||
};
|
||
|
||
|
||
// core.collections
|
||
PRIMITIVES["list"] = function() { return Array.prototype.slice.call(arguments); };
|
||
PRIMITIVES["dict"] = function() {
|
||
var d = {};
|
||
for (var i = 0; i < arguments.length - 1; i += 2) d[arguments[i]] = arguments[i + 1];
|
||
return d;
|
||
};
|
||
PRIMITIVES["range"] = function(a, b, step) {
|
||
var r = []; step = step || 1;
|
||
for (var i = a; step > 0 ? i < b : i > b; i += step) r.push(i);
|
||
return r;
|
||
};
|
||
PRIMITIVES["get"] = function(c, k, def) { var v = (c && c[k]); return v !== undefined ? v : (def !== undefined ? def : NIL); };
|
||
PRIMITIVES["len"] = function(c) { return Array.isArray(c) ? c.length : typeof c === "string" ? c.length : Object.keys(c).length; };
|
||
PRIMITIVES["first"] = function(c) { return c && c.length > 0 ? c[0] : NIL; };
|
||
PRIMITIVES["last"] = function(c) { return c && c.length > 0 ? c[c.length - 1] : NIL; };
|
||
PRIMITIVES["rest"] = function(c) { if (!c || c._nil) return []; if (typeof c.slice !== "function") return []; return c.slice(1); };
|
||
PRIMITIVES["nth"] = function(c, n) { return c && n >= 0 && n < c.length ? c[n] : NIL; };
|
||
PRIMITIVES["cons"] = function(x, c) { return [x].concat(c || []); };
|
||
PRIMITIVES["append"] = function(c, x) { return (c || []).concat(Array.isArray(x) ? x : [x]); };
|
||
PRIMITIVES["append!"] = function(arr, x) { arr.push(x); return arr; };
|
||
PRIMITIVES["chunk-every"] = function(c, n) {
|
||
var r = []; for (var i = 0; i < c.length; i += n) r.push(c.slice(i, i + n)); return r;
|
||
};
|
||
PRIMITIVES["zip-pairs"] = function(c) {
|
||
var r = []; for (var i = 0; i < c.length - 1; i++) r.push([c[i], c[i + 1]]); return r;
|
||
};
|
||
PRIMITIVES["reverse"] = function(c) { return Array.isArray(c) ? c.slice().reverse() : String(c).split("").reverse().join(""); };
|
||
PRIMITIVES["flatten"] = function(c) {
|
||
var out = [];
|
||
function walk(a) { for (var i = 0; i < a.length; i++) Array.isArray(a[i]) ? walk(a[i]) : out.push(a[i]); }
|
||
walk(c || []); return out;
|
||
};
|
||
|
||
|
||
// core.dict
|
||
PRIMITIVES["keys"] = function(d) { return Object.keys(d || {}); };
|
||
PRIMITIVES["vals"] = function(d) { var r = []; for (var k in d) r.push(d[k]); return r; };
|
||
PRIMITIVES["merge"] = function() {
|
||
var out = {};
|
||
for (var i = 0; i < arguments.length; i++) { var d = arguments[i]; if (d && !isNil(d)) for (var k in d) out[k] = d[k]; }
|
||
return out;
|
||
};
|
||
PRIMITIVES["assoc"] = function(d) {
|
||
var out = {}; if (d && !isNil(d)) for (var k in d) out[k] = d[k];
|
||
for (var i = 1; i < arguments.length - 1; i += 2) out[arguments[i]] = arguments[i + 1];
|
||
return out;
|
||
};
|
||
PRIMITIVES["dissoc"] = function(d) {
|
||
var out = {}; for (var k in d) out[k] = d[k];
|
||
for (var i = 1; i < arguments.length; i++) delete out[arguments[i]];
|
||
return out;
|
||
};
|
||
PRIMITIVES["dict-set!"] = function(d, k, v) { d[k] = v; return v; };
|
||
PRIMITIVES["has-key?"] = function(d, k) { return d !== null && d !== undefined && k in d; };
|
||
PRIMITIVES["into"] = function(target, coll) {
|
||
if (target === "list") return Array.isArray(coll) ? coll.slice() : Object.entries(coll).map(function(e) { return [e[0], e[1]]; });
|
||
if (target === "dict") { var r = {}; for (var i = 0; i < coll.length; i++) { var p = coll[i]; if (Array.isArray(p) && p.length >= 2) r[p[0]] = p[1]; } return r; }
|
||
if (Array.isArray(target)) return Array.isArray(coll) ? coll.slice() : Object.entries(coll);
|
||
var r = {}; for (var i = 0; i < coll.length; i++) { var p = coll[i]; if (Array.isArray(p) && p.length >= 2) r[p[0]] = p[1]; }
|
||
return r;
|
||
};
|
||
|
||
|
||
// core.vectors — R7RS mutable fixed-size arrays
|
||
PRIMITIVES["make-vector"] = function(n, fill) {
|
||
var arr = new Array(n);
|
||
var f = (fill !== undefined) ? fill : NIL;
|
||
for (var i = 0; i < n; i++) arr[i] = f;
|
||
return new SxVector(arr);
|
||
};
|
||
PRIMITIVES["vector"] = function() {
|
||
return new SxVector(Array.prototype.slice.call(arguments));
|
||
};
|
||
PRIMITIVES["vector?"] = function(x) { return x != null && x._vector === true; };
|
||
PRIMITIVES["vector-length"] = function(v) { return v.arr.length; };
|
||
PRIMITIVES["vector-ref"] = function(v, i) {
|
||
if (i < 0 || i >= v.arr.length) throw new Error("vector-ref: index " + i + " out of bounds (length " + v.arr.length + ")");
|
||
return v.arr[i];
|
||
};
|
||
PRIMITIVES["vector-set!"] = function(v, i, val) {
|
||
if (i < 0 || i >= v.arr.length) throw new Error("vector-set!: index " + i + " out of bounds (length " + v.arr.length + ")");
|
||
v.arr[i] = val; return NIL;
|
||
};
|
||
PRIMITIVES["vector->list"] = function(v) { return v.arr.slice(); };
|
||
PRIMITIVES["list->vector"] = function(l) { return new SxVector(l.slice()); };
|
||
PRIMITIVES["vector-fill!"] = function(v, val) {
|
||
for (var i = 0; i < v.arr.length; i++) v.arr[i] = val; return NIL;
|
||
};
|
||
PRIMITIVES["vector-copy"] = function(v, start, end) {
|
||
var s = (start !== undefined) ? start : 0;
|
||
var e = (end !== undefined) ? Math.min(end, v.arr.length) : v.arr.length;
|
||
return new SxVector(v.arr.slice(s, e));
|
||
};
|
||
|
||
// String buffers — O(1) amortised append via array+join
|
||
function SxStringBuffer() { this.parts = []; this.len = 0; this._string_buffer = true; }
|
||
PRIMITIVES["make-string-buffer"] = function() { return new SxStringBuffer(); };
|
||
PRIMITIVES["string-buffer?"] = function(x) { return x instanceof SxStringBuffer; };
|
||
PRIMITIVES["string-buffer-append!"] = function(buf, s) {
|
||
buf.parts.push(String(s)); buf.len += String(s).length; return NIL;
|
||
};
|
||
PRIMITIVES["string-buffer->string"] = function(buf) { return buf.parts.join(""); };
|
||
PRIMITIVES["string-buffer-length"] = function(buf) { return buf.len; };
|
||
|
||
// Short aliases — terser names; append accepts any value
|
||
PRIMITIVES["make-buffer"] = function() { return new SxStringBuffer(); };
|
||
PRIMITIVES["buffer?"] = function(x) { return x instanceof SxStringBuffer; };
|
||
PRIMITIVES["buffer-append!"] = function(buf, v) {
|
||
var s;
|
||
if (v === null || v === undefined || v === NIL) s = "";
|
||
else if (typeof v === "string") s = v;
|
||
else if (typeof v === "boolean") s = v ? "true" : "false";
|
||
else if (typeof v === "number") s = String(v);
|
||
else if (v && typeof v === "object" && typeof v.name === "string" && v.constructor && v.constructor.name === "Symbol") s = v.name;
|
||
else s = (typeof inspect === "function") ? inspect(v) : String(v);
|
||
buf.parts.push(s); buf.len += s.length; return NIL;
|
||
};
|
||
PRIMITIVES["buffer->string"] = function(buf) { return buf.parts.join(""); };
|
||
PRIMITIVES["buffer-length"] = function(buf) { return buf.len; };
|
||
|
||
|
||
// stdlib.format
|
||
PRIMITIVES["format-decimal"] = function(v, p) { return Number(v).toFixed(p || 2); };
|
||
PRIMITIVES["parse-int"] = function(v, d) { var n = parseInt(v, 10); return isNaN(n) ? (d || 0) : n; };
|
||
PRIMITIVES["format-date"] = function(s, fmt) {
|
||
if (!s) return "";
|
||
try {
|
||
var d = new Date(s);
|
||
if (isNaN(d.getTime())) return String(s);
|
||
var months = ["January","February","March","April","May","June","July","August","September","October","November","December"];
|
||
var short_months = ["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"];
|
||
return fmt.replace(/%-d/g, d.getDate()).replace(/%d/g, ("0"+d.getDate()).slice(-2))
|
||
.replace(/%B/g, months[d.getMonth()]).replace(/%b/g, short_months[d.getMonth()])
|
||
.replace(/%Y/g, d.getFullYear()).replace(/%m/g, ("0"+(d.getMonth()+1)).slice(-2))
|
||
.replace(/%H/g, ("0"+d.getHours()).slice(-2)).replace(/%M/g, ("0"+d.getMinutes()).slice(-2));
|
||
} catch (e) { return String(s); }
|
||
};
|
||
PRIMITIVES["parse-datetime"] = function(s) { return s ? String(s) : NIL; };
|
||
|
||
|
||
// stdlib.text
|
||
PRIMITIVES["pluralize"] = function(n, s, p) {
|
||
if (s || (p && p !== "s")) return n == 1 ? (s || "") : (p || "s");
|
||
return n == 1 ? "" : "s";
|
||
};
|
||
PRIMITIVES["escape"] = function(s) {
|
||
return String(s).replace(/&/g,"&").replace(/</g,"<").replace(/>/g,">").replace(/"/g,""").replace(/'/g,"'");
|
||
};
|
||
PRIMITIVES["strip-tags"] = function(s) { return String(s).replace(/<[^>]+>/g, ""); };
|
||
|
||
|
||
// stdlib.debug
|
||
PRIMITIVES["assert"] = function(cond, msg) {
|
||
if (!isSxTruthy(cond)) throw new Error("Assertion error: " + (msg || "Assertion failed"));
|
||
return true;
|
||
};
|
||
|
||
|
||
// stdlib.spread — spread + collect + scope primitives
|
||
PRIMITIVES["make-spread"] = makeSpread;
|
||
PRIMITIVES["spread?"] = isSpread;
|
||
PRIMITIVES["spread-attrs"] = spreadAttrs;
|
||
PRIMITIVES["collect!"] = sxCollect;
|
||
PRIMITIVES["collected"] = sxCollected;
|
||
PRIMITIVES["clear-collected!"] = sxClearCollected;
|
||
// scope — unified render-time dynamic scope
|
||
PRIMITIVES["scope-push!"] = scopePush;
|
||
PRIMITIVES["scope-pop!"] = scopePop;
|
||
// provide-push!/provide-pop! — aliases for scope-push!/scope-pop!
|
||
PRIMITIVES["provide-push!"] = providePush;
|
||
PRIMITIVES["provide-pop!"] = providePop;
|
||
PRIMITIVES["context"] = sxContext;
|
||
PRIMITIVES["emit!"] = sxEmit;
|
||
PRIMITIVES["emitted"] = sxEmitted;
|
||
// Aliases for aser adapter (avoids CEK special form conflict on server)
|
||
var scopeEmit = sxEmit;
|
||
function scopePeek(name) {
|
||
if (_scopeStacks[name] && _scopeStacks[name].length) {
|
||
return _scopeStacks[name][_scopeStacks[name].length - 1].value;
|
||
}
|
||
return NIL;
|
||
}
|
||
PRIMITIVES["scope-emit!"] = scopeEmit;
|
||
PRIMITIVES["scope-peek"] = scopePeek;
|
||
PRIMITIVES["scope-emitted"] = sxEmitted;
|
||
PRIMITIVES["scope-collected"] = sxCollected;
|
||
PRIMITIVES["scope-clear-collected!"] = sxClearCollected;
|
||
|
||
// ---- VM stack primitives ----
|
||
// The VM spec (vm.sx) requires these array-like operations.
|
||
// In JS, a plain Array serves as the stack.
|
||
PRIMITIVES["make-vm-stack"] = function(size) {
|
||
var a = new Array(size);
|
||
for (var i = 0; i < size; i++) a[i] = NIL;
|
||
return a;
|
||
};
|
||
PRIMITIVES["vm-stack-get"] = function(stack, idx) { return stack[idx]; };
|
||
PRIMITIVES["vm-stack-set!"] = function(stack, idx, value) { stack[idx] = value; return NIL; };
|
||
PRIMITIVES["vm-stack-length"] = function(stack) { return stack.length; };
|
||
PRIMITIVES["vm-stack-copy!"] = function(src, dst, count) {
|
||
for (var i = 0; i < count; i++) dst[i] = src[i];
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["get-primitive"] = function(name) {
|
||
if (name in PRIMITIVES) return PRIMITIVES[name];
|
||
throw new Error("VM undefined: " + name);
|
||
};
|
||
PRIMITIVES["call-primitive"] = function(name, args) {
|
||
if (!(name in PRIMITIVES)) throw new Error("VM undefined: " + name);
|
||
var fn = PRIMITIVES[name];
|
||
return fn.apply(null, Array.isArray(args) ? args : []);
|
||
};
|
||
PRIMITIVES["primitive?"] = function(name) {
|
||
return name in PRIMITIVES;
|
||
};
|
||
PRIMITIVES["set-nth!"] = function(lst, idx, val) {
|
||
lst[idx] = val;
|
||
return NIL;
|
||
};
|
||
|
||
PRIMITIVES["env-parent"] = function(env) {
|
||
if (env && Object.getPrototypeOf(env) !== Object.prototype &&
|
||
Object.getPrototypeOf(env) !== null)
|
||
return Object.getPrototypeOf(env);
|
||
return NIL;
|
||
};
|
||
|
||
|
||
// stdlib.bitwise
|
||
PRIMITIVES["bitwise-and"] = function(a, b) { return (a & b) | 0; };
|
||
PRIMITIVES["bitwise-or"] = function(a, b) { return (a | b) | 0; };
|
||
PRIMITIVES["bitwise-xor"] = function(a, b) { return (a ^ b) | 0; };
|
||
PRIMITIVES["bitwise-not"] = function(a) { return ~a; };
|
||
PRIMITIVES["arithmetic-shift"] = function(a, count) {
|
||
return count >= 0 ? (a << count) | 0 : a >> (-count);
|
||
};
|
||
PRIMITIVES["bit-count"] = function(a) {
|
||
var n = Math.abs(a) >>> 0;
|
||
n = n - ((n >> 1) & 0x55555555);
|
||
n = (n & 0x33333333) + ((n >> 2) & 0x33333333);
|
||
return (((n + (n >> 4)) & 0x0f0f0f0f) * 0x01010101) >>> 24;
|
||
};
|
||
PRIMITIVES["integer-length"] = function(a) {
|
||
if (a === 0) return 0;
|
||
return 32 - Math.clz32(Math.abs(a));
|
||
};
|
||
|
||
|
||
// stdlib.math
|
||
PRIMITIVES["sin"] = Math.sin;
|
||
PRIMITIVES["cos"] = Math.cos;
|
||
PRIMITIVES["tan"] = Math.tan;
|
||
PRIMITIVES["asin"] = Math.asin;
|
||
PRIMITIVES["acos"] = Math.acos;
|
||
PRIMITIVES["atan"] = function(y, x) { return arguments.length >= 2 ? Math.atan2(y, x) : Math.atan(y); };
|
||
PRIMITIVES["exp"] = Math.exp;
|
||
PRIMITIVES["log"] = Math.log;
|
||
PRIMITIVES["expt"] = Math.pow;
|
||
PRIMITIVES["quotient"] = function(a, b) { return Math.trunc(a / b); };
|
||
PRIMITIVES["gcd"] = function(a, b) {
|
||
a = Math.abs(a); b = Math.abs(b);
|
||
while (b) { var t = b; b = a % b; a = t; }
|
||
return a;
|
||
};
|
||
PRIMITIVES["lcm"] = function(a, b) {
|
||
var g = PRIMITIVES["gcd"](Math.abs(a), Math.abs(b));
|
||
return g === 0 ? 0 : Math.abs(a / g * b);
|
||
};
|
||
PRIMITIVES["number->string"] = function(n, r) {
|
||
if (n && n._rational) return n._n + "/" + n._d;
|
||
if (r === undefined || r === null) return String(n);
|
||
return Math.floor(n).toString(r);
|
||
};
|
||
PRIMITIVES["string->number"] = function(s, r) {
|
||
s = String(s);
|
||
if (r !== undefined && r !== null) {
|
||
var radix = r | 0;
|
||
var valid = "0123456789abcdefghijklmnopqrstuvwxyz".slice(0, radix);
|
||
var norm = s.toLowerCase();
|
||
var start = norm[0] === '-' ? 1 : 0;
|
||
if (norm.length <= start) return NIL;
|
||
for (var i = start; i < norm.length; i++) {
|
||
if (valid.indexOf(norm[i]) === -1) return NIL;
|
||
}
|
||
return parseInt(s, radix);
|
||
}
|
||
if (s === '') return NIL;
|
||
var n = Number(s);
|
||
return isNaN(n) ? NIL : n;
|
||
};
|
||
|
||
|
||
// stdlib.rational
|
||
function SxRational(n, d) {
|
||
function gcd(a, b) { while (b) { var t=b; b=a%b; a=t; } return a; }
|
||
if (d === 0) throw new Error("make-rational: denominator cannot be zero");
|
||
var sign = (d < 0) ? -1 : 1;
|
||
var g = gcd(Math.abs(n), Math.abs(d));
|
||
this._n = sign * n / g;
|
||
this._d = sign * d / g;
|
||
this._rational = true;
|
||
}
|
||
SxRational.prototype.toString = function() { return this._n + "/" + this._d; };
|
||
PRIMITIVES["make-rational"] = function(n, d) {
|
||
var r = new SxRational(Math.trunc(n), Math.trunc(d));
|
||
if (r._d === 1) return r._n;
|
||
return r;
|
||
};
|
||
PRIMITIVES["rational?"] = function(v) { return v instanceof SxRational; };
|
||
PRIMITIVES["numerator"] = function(r) { return r instanceof SxRational ? r._n : r; };
|
||
PRIMITIVES["denominator"] = function(r) { return r instanceof SxRational ? r._d : 1; };
|
||
var makeRational = PRIMITIVES["make-rational"];
|
||
|
||
|
||
// stdlib.hash-table
|
||
function SxHashTable() { this.data = new Map(); this._hash_table = true; }
|
||
PRIMITIVES["make-hash-table"] = function() { return new SxHashTable(); };
|
||
PRIMITIVES["hash-table?"] = function(x) { return x instanceof SxHashTable; };
|
||
PRIMITIVES["hash-table-set!"] = function(ht, k, v) { ht.data.set(k, v); return null; };
|
||
PRIMITIVES["hash-table-ref"] = function(ht, k, dflt) {
|
||
if (ht.data.has(k)) return ht.data.get(k);
|
||
if (arguments.length > 2) return dflt;
|
||
throw new Error("hash-table-ref: key not found");
|
||
};
|
||
PRIMITIVES["hash-table-delete!"] = function(ht, k) { ht.data.delete(k); return null; };
|
||
PRIMITIVES["hash-table-size"] = function(ht) { return ht.data.size; };
|
||
PRIMITIVES["hash-table-keys"] = function(ht) { return Array.from(ht.data.keys()); };
|
||
PRIMITIVES["hash-table-values"] = function(ht) { return Array.from(ht.data.values()); };
|
||
PRIMITIVES["hash-table->alist"] = function(ht) {
|
||
var result = [];
|
||
ht.data.forEach(function(v, k) { result.push([k, v]); });
|
||
return result;
|
||
};
|
||
PRIMITIVES["hash-table-for-each"] = function(ht, fn) {
|
||
ht.data.forEach(function(v, k) { apply(fn, [k, v]); });
|
||
return null;
|
||
};
|
||
PRIMITIVES["hash-table-merge!"] = function(dst, src) {
|
||
src.data.forEach(function(v, k) { dst.data.set(k, v); });
|
||
return null;
|
||
};
|
||
|
||
|
||
// stdlib.regexp — native JS RegExp wrappers
|
||
function SxRegexp(source, flags) {
|
||
this._regexp = true;
|
||
this.source = source;
|
||
this.flags = flags || "";
|
||
}
|
||
function sxRxCompile(rx) {
|
||
if (!rx._compiled) {
|
||
var jsFlags = "";
|
||
if (rx.flags.indexOf("i") >= 0) jsFlags += "i";
|
||
if (rx.flags.indexOf("m") >= 0) jsFlags += "m";
|
||
if (rx.flags.indexOf("s") >= 0) jsFlags += "s";
|
||
rx._compiled = new RegExp(rx.source, jsFlags);
|
||
}
|
||
return rx._compiled;
|
||
}
|
||
function sxRxMatchDict(m, input) {
|
||
if (!m) return NIL;
|
||
var groups = [];
|
||
for (var i = 1; i < m.length; i++) groups.push(m[i] !== undefined ? m[i] : "");
|
||
return {"match": m[0], "start": m.index, "end": m.index + m[0].length,
|
||
"input": input, "groups": groups};
|
||
}
|
||
PRIMITIVES["make-regexp"] = function(src, flags) {
|
||
return new SxRegexp(src, flags || "");
|
||
};
|
||
PRIMITIVES["regexp?"] = function(v) { return v instanceof SxRegexp; };
|
||
PRIMITIVES["regexp-source"] = function(rx) { return rx.source; };
|
||
PRIMITIVES["regexp-flags"] = function(rx) { return rx.flags; };
|
||
PRIMITIVES["regexp-match"] = function(rx, s) {
|
||
var re = new RegExp(sxRxCompile(rx).source,
|
||
sxRxCompile(rx).flags.replace("g",""));
|
||
var m = s.match(re);
|
||
return sxRxMatchDict(m, s);
|
||
};
|
||
PRIMITIVES["regexp-match-all"] = function(rx, s) {
|
||
var compiled = sxRxCompile(rx);
|
||
var re = new RegExp(compiled.source, "g" + compiled.flags.replace("g",""));
|
||
var results = [], m;
|
||
while ((m = re.exec(s)) !== null) {
|
||
results.push(sxRxMatchDict(m, s));
|
||
if (m[0].length === 0) re.lastIndex++;
|
||
}
|
||
return results;
|
||
};
|
||
PRIMITIVES["regexp-replace"] = function(rx, s, replacement) {
|
||
var compiled = sxRxCompile(rx);
|
||
var re = new RegExp(compiled.source, compiled.flags.replace("g",""));
|
||
return s.replace(re, replacement);
|
||
};
|
||
PRIMITIVES["regexp-replace-all"] = function(rx, s, replacement) {
|
||
var compiled = sxRxCompile(rx);
|
||
var re = new RegExp(compiled.source, "g" + compiled.flags.replace("g",""));
|
||
return s.replace(re, replacement);
|
||
};
|
||
PRIMITIVES["regexp-split"] = function(rx, s) {
|
||
var re = sxRxCompile(rx);
|
||
return s.split(re);
|
||
};
|
||
|
||
|
||
// stdlib.sets — structural sets keyed by write-to-string serialization
|
||
function SxSet() { this.data = new Map(); this._sxset = true; }
|
||
SxSet.prototype._type = "set";
|
||
function sxSetKey(v) { return sxWriteVal(v, "write"); }
|
||
function sxSetSeed(s, lst) {
|
||
if (Array.isArray(lst)) lst.forEach(function(v) { s.data.set(sxSetKey(v), v); });
|
||
return s;
|
||
}
|
||
PRIMITIVES["make-set"] = function() {
|
||
var s = new SxSet();
|
||
if (arguments.length > 0 && Array.isArray(arguments[0])) sxSetSeed(s, arguments[0]);
|
||
return s;
|
||
};
|
||
PRIMITIVES["set?"] = function(v) { return v instanceof SxSet; };
|
||
PRIMITIVES["set-add!"] = function(s, v) { s.data.set(sxSetKey(v), v); return NIL; };
|
||
PRIMITIVES["set-member?"] = function(s, v) { return s.data.has(sxSetKey(v)); };
|
||
PRIMITIVES["set-remove!"] = function(s, v) { s.data.delete(sxSetKey(v)); return NIL; };
|
||
PRIMITIVES["set-size"] = function(s) { return s.data.size; };
|
||
PRIMITIVES["set->list"] = function(s) { return Array.from(s.data.values()); };
|
||
PRIMITIVES["list->set"] = function(lst) {
|
||
var s = new SxSet();
|
||
if (Array.isArray(lst)) lst.forEach(function(v) { s.data.set(sxSetKey(v), v); });
|
||
return s;
|
||
};
|
||
PRIMITIVES["set-union"] = function(a, b) {
|
||
var s = new SxSet();
|
||
a.data.forEach(function(v, k) { s.data.set(k, v); });
|
||
b.data.forEach(function(v, k) { s.data.set(k, v); });
|
||
return s;
|
||
};
|
||
PRIMITIVES["set-intersection"] = function(a, b) {
|
||
var s = new SxSet();
|
||
a.data.forEach(function(v, k) { if (b.data.has(k)) s.data.set(k, v); });
|
||
return s;
|
||
};
|
||
PRIMITIVES["set-difference"] = function(a, b) {
|
||
var s = new SxSet();
|
||
a.data.forEach(function(v, k) { if (!b.data.has(k)) s.data.set(k, v); });
|
||
return s;
|
||
};
|
||
PRIMITIVES["set-for-each"] = function(s, fn) {
|
||
s.data.forEach(function(v) { apply(fn, [v]); });
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["set-map"] = function(s, fn) {
|
||
var out = new SxSet();
|
||
s.data.forEach(function(v) {
|
||
var r = apply(fn, [v]);
|
||
out.data.set(sxSetKey(r), r);
|
||
});
|
||
return out;
|
||
};
|
||
|
||
|
||
// stdlib.bytevectors — R7RS bytevector type backed by Uint8Array
|
||
function SxBytevector(size_or_buf) {
|
||
if (size_or_buf instanceof Uint8Array) {
|
||
this.data = size_or_buf;
|
||
} else {
|
||
this.data = new Uint8Array(typeof size_or_buf === "number" ? size_or_buf : 0);
|
||
}
|
||
this._bytevector = true;
|
||
}
|
||
SxBytevector.prototype._type = "bytevector";
|
||
PRIMITIVES["make-bytevector"] = function(n, fill) {
|
||
var bv = new SxBytevector(n);
|
||
if (fill !== undefined) bv.data.fill(fill & 0xff);
|
||
return bv;
|
||
};
|
||
PRIMITIVES["bytevector?"] = function(v) { return v instanceof SxBytevector; };
|
||
PRIMITIVES["bytevector-length"] = function(bv) { return bv.data.length; };
|
||
PRIMITIVES["bytevector-u8-ref"] = function(bv, i) { return bv.data[i]; };
|
||
PRIMITIVES["bytevector-u8-set!"] = function(bv, i, byte) { bv.data[i] = byte & 0xff; return NIL; };
|
||
PRIMITIVES["bytevector-copy"] = function(bv, start, end_) {
|
||
var s = start === undefined ? 0 : start;
|
||
var e = end_ === undefined ? bv.data.length : end_;
|
||
return new SxBytevector(bv.data.slice(s, e));
|
||
};
|
||
PRIMITIVES["bytevector-copy!"] = function(dst, at, src, start, end_) {
|
||
var s = start === undefined ? 0 : start;
|
||
var e = end_ === undefined ? src.data.length : end_;
|
||
dst.data.set(src.data.subarray(s, e), at);
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["bytevector-append"] = function() {
|
||
var total = 0;
|
||
for (var i = 0; i < arguments.length; i++) total += arguments[i].data.length;
|
||
var result = new Uint8Array(total);
|
||
var pos = 0;
|
||
for (var i = 0; i < arguments.length; i++) {
|
||
result.set(arguments[i].data, pos);
|
||
pos += arguments[i].data.length;
|
||
}
|
||
return new SxBytevector(result);
|
||
};
|
||
PRIMITIVES["utf8->string"] = function(bv, start, end_) {
|
||
var s = start === undefined ? 0 : start;
|
||
var e = end_ === undefined ? bv.data.length : end_;
|
||
var dec = new TextDecoder("utf-8");
|
||
return dec.decode(bv.data.subarray(s, e));
|
||
};
|
||
PRIMITIVES["string->utf8"] = function(str, start, end_) {
|
||
var enc = new TextEncoder();
|
||
var full = enc.encode(str);
|
||
var s = start === undefined ? 0 : start;
|
||
var e = end_ === undefined ? full.length : end_;
|
||
return new SxBytevector(full.slice(s, e));
|
||
};
|
||
PRIMITIVES["bytevector->list"] = function(bv) {
|
||
var out = [];
|
||
for (var i = 0; i < bv.data.length; i++) out.push(bv.data[i]);
|
||
return out;
|
||
};
|
||
PRIMITIVES["list->bytevector"] = function(lst) {
|
||
if (!Array.isArray(lst)) lst = [];
|
||
var b = new Uint8Array(lst.length);
|
||
for (var i = 0; i < lst.length; i++) b[i] = lst[i] & 0xff;
|
||
return new SxBytevector(b);
|
||
};
|
||
|
||
|
||
function isPrimitive(name) { return name in PRIMITIVES; }
|
||
function getPrimitive(name) { return PRIMITIVES[name]; }
|
||
|
||
// Higher-order helpers used by the transpiled code
|
||
function map(fn, coll) { return coll.map(fn); }
|
||
function mapIndexed(fn, coll) { return coll.map(function(item, i) { return fn(i, item); }); }
|
||
function filter(fn, coll) { return coll.filter(function(x) { return isSxTruthy(fn(x)); }); }
|
||
function reduce(fn, init, coll) {
|
||
var acc = init;
|
||
for (var i = 0; i < coll.length; i++) acc = fn(acc, coll[i]);
|
||
return acc;
|
||
}
|
||
function some(fn, coll) {
|
||
for (var i = 0; i < coll.length; i++) { var r = fn(coll[i]); if (isSxTruthy(r)) return r; }
|
||
return NIL;
|
||
}
|
||
function forEach(fn, coll) { for (var i = 0; i < coll.length; i++) fn(coll[i]); return NIL; }
|
||
function isEvery(fn, coll) {
|
||
for (var i = 0; i < coll.length; i++) { if (!isSxTruthy(fn(coll[i]))) return false; }
|
||
return true;
|
||
}
|
||
function mapDict(fn, d) { var r = {}; for (var k in d) r[k] = fn(k, d[k]); return r; }
|
||
|
||
// Predicate aliases used by transpiled code
|
||
// Both naming conventions: isX (from js-renames) and x_p (from js-mangle of x?)
|
||
var isNumber = PRIMITIVES["number?"]; var number_p = isNumber;
|
||
var isString = PRIMITIVES["string?"]; var string_p = isString;
|
||
var isBoolean = PRIMITIVES["boolean?"]; var boolean_p = isBoolean;
|
||
var isDict = PRIMITIVES["dict?"];
|
||
var isList = PRIMITIVES["list?"]; var list_p = isList;
|
||
var isKeyword = PRIMITIVES["keyword?"]; var keyword_p = isKeyword;
|
||
var isSymbol = PRIMITIVES["symbol?"]; var symbol_p = isSymbol;
|
||
|
||
// List primitives used directly by transpiled code
|
||
var len = PRIMITIVES["len"];
|
||
var first = PRIMITIVES["first"];
|
||
var last = PRIMITIVES["last"];
|
||
var rest = PRIMITIVES["rest"];
|
||
var nth = PRIMITIVES["nth"];
|
||
var cons = PRIMITIVES["cons"];
|
||
var append = PRIMITIVES["append"];
|
||
var isEmpty = PRIMITIVES["empty?"];
|
||
var contains = PRIMITIVES["contains?"];
|
||
var startsWith = PRIMITIVES["starts-with?"];
|
||
var slice = PRIMITIVES["slice"];
|
||
var concat = PRIMITIVES["concat"];
|
||
var str = PRIMITIVES["str"];
|
||
var join = PRIMITIVES["join"];
|
||
var keys = PRIMITIVES["keys"];
|
||
var get = PRIMITIVES["get"];
|
||
var assoc = PRIMITIVES["assoc"];
|
||
var range = PRIMITIVES["range"];
|
||
var floor = PRIMITIVES["floor"];
|
||
var pow = PRIMITIVES["pow"];
|
||
var mod = PRIMITIVES["mod"];
|
||
var indexOf_ = PRIMITIVES["index-of"];
|
||
var hasKey = PRIMITIVES["has-key?"];
|
||
var vectorToList = PRIMITIVES["vector->list"];
|
||
var listToVector = PRIMITIVES["list->vector"];
|
||
var isVector = PRIMITIVES["vector?"];
|
||
var vectorLength = PRIMITIVES["vector-length"];
|
||
var vectorRef = PRIMITIVES["vector-ref"];
|
||
var reverse = PRIMITIVES["reverse"];
|
||
var stringToSymbol = PRIMITIVES["string->symbol"];
|
||
var symbolToString = PRIMITIVES["symbol->string"];
|
||
function zip(a, b) { var r = []; for (var i = 0; i < Math.min(a.length, b.length); i++) r.push([a[i], b[i]]); return r; }
|
||
function append_b(arr, x) { arr.push(x); return arr; }
|
||
var apply = function(f, args) {
|
||
if (isLambda(f)) return trampoline(callLambda(f, args, lambdaClosure(f)));
|
||
return f.apply(null, args);
|
||
};
|
||
PRIMITIVES["apply"] = apply;
|
||
|
||
// Additional primitive aliases used by adapter/engine transpiled code
|
||
var split = PRIMITIVES["split"];
|
||
var trim = PRIMITIVES["trim"];
|
||
var upper = PRIMITIVES["upper"];
|
||
var lower = PRIMITIVES["lower"];
|
||
var replace_ = function(s, old, nw) { return s.split(old).join(nw); };
|
||
var endsWith = PRIMITIVES["ends-with?"];
|
||
var parseInt_ = PRIMITIVES["parse-int"];
|
||
var dict_fn = PRIMITIVES["dict"];
|
||
|
||
// HTML rendering helpers
|
||
// escape-html and escape-attr are now library functions defined in render.sx
|
||
function rawHtmlContent(r) { return r.html; }
|
||
function makeRawHtml(s) { return { _raw: true, html: s }; }
|
||
function makeSxExpr(s) { return { _sx_expr: true, source: s }; }
|
||
function sxExprSource(x) { return x && x.source ? x.source : String(x); }
|
||
|
||
// Placeholders — overridden by transpiled spec from parser.sx / adapter-sx.sx
|
||
function serialize(val) { return String(val); }
|
||
function isSpecialForm(n) { return false; }
|
||
function isHoForm(n) { return false; }
|
||
|
||
// -----------------------------------------------------------------------
|
||
// Host FFI — the irreducible web platform primitives
|
||
// All DOM/browser operations are built on these in web/lib/dom.sx
|
||
// -----------------------------------------------------------------------
|
||
PRIMITIVES["host-global"] = function(name) {
|
||
if (typeof globalThis !== "undefined" && name in globalThis) return globalThis[name];
|
||
if (typeof window !== "undefined" && name in window) return window[name];
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["host-get"] = function(obj, prop) {
|
||
if (obj == null || obj === NIL) return NIL;
|
||
var v = obj[prop];
|
||
return v === undefined || v === null ? NIL : v;
|
||
};
|
||
PRIMITIVES["host-set!"] = function(obj, prop, val) {
|
||
if (obj != null && obj !== NIL) obj[prop] = val === NIL ? null : val;
|
||
};
|
||
PRIMITIVES["host-call"] = function() {
|
||
var obj = arguments[0], method = arguments[1];
|
||
var args = [];
|
||
for (var i = 2; i < arguments.length; i++) {
|
||
var a = arguments[i];
|
||
args.push(a === NIL ? null : a);
|
||
}
|
||
if (obj == null || obj === NIL) {
|
||
// Global function call
|
||
var fn = typeof globalThis !== "undefined" ? globalThis[method] : window[method];
|
||
if (typeof fn === "function") return fn.apply(null, args);
|
||
return NIL;
|
||
}
|
||
if (typeof obj[method] === "function") {
|
||
try { return obj[method].apply(obj, args); }
|
||
catch(e) { return NIL; }
|
||
}
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["host-new"] = function() {
|
||
var name = arguments[0];
|
||
var args = Array.prototype.slice.call(arguments, 1).map(function(a) { return a === NIL ? null : a; });
|
||
var Ctor = typeof globalThis !== "undefined" ? globalThis[name] : window[name];
|
||
if (typeof Ctor !== "function") return NIL;
|
||
// Support 0-4 args (covers all practical cases)
|
||
switch (args.length) {
|
||
case 0: return new Ctor();
|
||
case 1: return new Ctor(args[0]);
|
||
case 2: return new Ctor(args[0], args[1]);
|
||
case 3: return new Ctor(args[0], args[1], args[2]);
|
||
default: return new Ctor(args[0], args[1], args[2], args[3]);
|
||
}
|
||
};
|
||
PRIMITIVES["host-callback"] = function(fn) {
|
||
// Wrap SX function/lambda as a native JS callback
|
||
if (typeof fn === "function") return fn;
|
||
if (fn && fn._type === "lambda") {
|
||
return function() {
|
||
var a = Array.prototype.slice.call(arguments);
|
||
return cekCall(fn, a);
|
||
};
|
||
}
|
||
return function() {};
|
||
};
|
||
PRIMITIVES["host-typeof"] = function(obj) {
|
||
if (obj == null || obj === NIL) return "nil";
|
||
if (obj instanceof Element) return "element";
|
||
if (obj instanceof Text) return "text";
|
||
if (obj instanceof DocumentFragment) return "fragment";
|
||
if (obj instanceof Document) return "document";
|
||
if (obj instanceof Event) return "event";
|
||
if (obj instanceof Promise) return "promise";
|
||
if (obj instanceof AbortController) return "abort-controller";
|
||
return typeof obj;
|
||
};
|
||
PRIMITIVES["host-await"] = function(promise, callback) {
|
||
if (promise && typeof promise.then === "function") {
|
||
var cb = typeof callback === "function" ? callback :
|
||
(callback && callback._type === "lambda") ?
|
||
function(v) { return cekCall(callback, [v]); } : function() {};
|
||
promise.then(cb);
|
||
}
|
||
};
|
||
// Aliases for transpiled dom.sx / browser.sx code (transpiler mangles host-* names)
|
||
var hostGlobal = PRIMITIVES["host-global"];
|
||
var hostGet = PRIMITIVES["host-get"];
|
||
var hostSet = PRIMITIVES["host-set!"];
|
||
var hostCall = PRIMITIVES["host-call"];
|
||
var hostNew = PRIMITIVES["host-new"];
|
||
var hostCallback = PRIMITIVES["host-callback"];
|
||
var hostTypeof = PRIMITIVES["host-typeof"];
|
||
var hostAwait = PRIMITIVES["host-await"];
|
||
|
||
// processBindings and evalCond — now specced in render.sx, bootstrapped above
|
||
|
||
function isDefinitionForm(name) {
|
||
return name === "define" || name === "defcomp" || name === "defmacro" ||
|
||
name === "defstyle" || name === "defhandler" ||
|
||
name === "deftype" || name === "defeffect";
|
||
}
|
||
|
||
function indexOf_(s, ch) {
|
||
return typeof s === "string" ? s.indexOf(ch) : -1;
|
||
}
|
||
|
||
function dictHas(d, k) { return d != null && k in d; }
|
||
function dictDelete(d, k) { delete d[k]; }
|
||
|
||
function forEachIndexed(fn, coll) {
|
||
for (var i = 0; i < coll.length; i++) fn(i, coll[i]);
|
||
return NIL;
|
||
}
|
||
|
||
// =========================================================================
|
||
// Performance overrides — evaluator hot path
|
||
// =========================================================================
|
||
|
||
// Override parseKeywordArgs: imperative loop instead of reduce+assoc
|
||
parseKeywordArgs = function(rawArgs, env) {
|
||
var kwargs = {};
|
||
var children = [];
|
||
for (var i = 0; i < rawArgs.length; i++) {
|
||
var arg = rawArgs[i];
|
||
if (arg && arg._kw && (i + 1) < rawArgs.length) {
|
||
kwargs[arg.name] = trampoline(evalExpr(rawArgs[i + 1], env));
|
||
i++;
|
||
} else {
|
||
children.push(trampoline(evalExpr(arg, env)));
|
||
}
|
||
}
|
||
return [kwargs, children];
|
||
};
|
||
|
||
// Override callComponent: use prototype chain env, imperative kwarg binding
|
||
callComponent = function(comp, rawArgs, env) {
|
||
var kwargs = {};
|
||
var children = [];
|
||
for (var i = 0; i < rawArgs.length; i++) {
|
||
var arg = rawArgs[i];
|
||
if (arg && arg._kw && (i + 1) < rawArgs.length) {
|
||
kwargs[arg.name] = trampoline(evalExpr(rawArgs[i + 1], env));
|
||
i++;
|
||
} else {
|
||
children.push(trampoline(evalExpr(arg, env)));
|
||
}
|
||
}
|
||
var local = Object.create(componentClosure(comp));
|
||
for (var k in env) if (env.hasOwnProperty(k)) local[k] = env[k];
|
||
var params = componentParams(comp);
|
||
for (var j = 0; j < params.length; j++) {
|
||
var p = params[j];
|
||
local[p] = p in kwargs ? kwargs[p] : NIL;
|
||
}
|
||
if (componentHasChildren(comp)) {
|
||
local["children"] = children;
|
||
}
|
||
return makeThunk(componentBody(comp), local);
|
||
};
|
||
|
||
// =========================================================================
|
||
// Platform: deps module — component dependency analysis
|
||
// =========================================================================
|
||
|
||
function componentDeps(c) {
|
||
return c.deps ? c.deps.slice() : [];
|
||
}
|
||
|
||
function componentSetDeps(c, deps) {
|
||
c.deps = deps;
|
||
}
|
||
|
||
function componentCssClasses(c) {
|
||
return c.cssClasses ? c.cssClasses.slice() : [];
|
||
}
|
||
|
||
function envComponents(env) {
|
||
var names = [];
|
||
for (var k in env) {
|
||
var v = env[k];
|
||
if (v && (v._component || v._macro)) names.push(k);
|
||
}
|
||
return names;
|
||
}
|
||
|
||
function regexFindAll(pattern, source) {
|
||
var re = new RegExp(pattern, "g");
|
||
var results = [];
|
||
var m;
|
||
while ((m = re.exec(source)) !== null) {
|
||
if (m[1] !== undefined) results.push(m[1]);
|
||
else results.push(m[0]);
|
||
}
|
||
return results;
|
||
}
|
||
|
||
function scanCssClasses(source) {
|
||
var classes = {};
|
||
var result = [];
|
||
var m;
|
||
var re1 = /:class\s+"([^"]*)"/g;
|
||
while ((m = re1.exec(source)) !== null) {
|
||
var parts = m[1].split(/\s+/);
|
||
for (var i = 0; i < parts.length; i++) {
|
||
if (parts[i] && !classes[parts[i]]) {
|
||
classes[parts[i]] = true;
|
||
result.push(parts[i]);
|
||
}
|
||
}
|
||
}
|
||
var re2 = /:class\s+\(str\s+((?:"[^"]*"\s*)+)\)/g;
|
||
while ((m = re2.exec(source)) !== null) {
|
||
var re3 = /"([^"]*)"/g;
|
||
var m2;
|
||
while ((m2 = re3.exec(m[1])) !== null) {
|
||
var parts2 = m2[1].split(/\s+/);
|
||
for (var j = 0; j < parts2.length; j++) {
|
||
if (parts2[j] && !classes[parts2[j]]) {
|
||
classes[parts2[j]] = true;
|
||
result.push(parts2[j]);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
var re4 = /;;\s*@css\s+(.+)/g;
|
||
while ((m = re4.exec(source)) !== null) {
|
||
var parts3 = m[1].split(/\s+/);
|
||
for (var k = 0; k < parts3.length; k++) {
|
||
if (parts3[k] && !classes[parts3[k]]) {
|
||
classes[parts3[k]] = true;
|
||
result.push(parts3[k]);
|
||
}
|
||
}
|
||
}
|
||
return result;
|
||
}
|
||
|
||
function componentIoRefs(c) {
|
||
return c.ioRefs ? c.ioRefs.slice() : [];
|
||
}
|
||
|
||
function componentSetIoRefs(c, refs) {
|
||
c.ioRefs = refs;
|
||
}
|
||
|
||
|
||
// =========================================================================
|
||
// Platform interface — Parser
|
||
// =========================================================================
|
||
// Character classification derived from the grammar:
|
||
// ident-start → [a-zA-Z_~*+\-><=/!?&]
|
||
// ident-char → ident-start + [0-9.:\/\#,]
|
||
|
||
var _identStartRe = /[a-zA-Z_~*+\-><=/!?&]/;
|
||
var _identCharRe = /[a-zA-Z0-9_~*+\-><=/!?.:&/#,]/;
|
||
|
||
function isIdentStart(ch) { return _identStartRe.test(ch); }
|
||
function isIdentChar(ch) { return _identCharRe.test(ch); }
|
||
function parseNumber(s) { return Number(s); }
|
||
function escapeString(s) {
|
||
return s.replace(/\\/g, "\\\\").replace(/"/g, '\\"').replace(/\n/g, "\\n").replace(/\t/g, "\\t");
|
||
}
|
||
function sxExprSource(e) { return typeof e === "string" ? e : (e && e.source ? e.source : String(e)); }
|
||
var charFromCode = PRIMITIVES["char-from-code"];
|
||
var makeChar = PRIMITIVES["make-char"];
|
||
var charToInteger = PRIMITIVES["char->integer"];
|
||
var isChar = PRIMITIVES["char?"];
|
||
var _readerMacros = {};
|
||
function readerMacroGet(name) { return _readerMacros[name] || false; }
|
||
function readerMacroSet(name, fn) { _readerMacros[name] = fn; }
|
||
PRIMITIVES["reader-macro-get"] = readerMacroGet;
|
||
PRIMITIVES["reader-macro-set!"] = readerMacroSet;
|
||
|
||
|
||
// String/number utilities needed by transpiled spec code (content-hash etc)
|
||
PRIMITIVES["char-code-at"] = function(s, i) { return s.charCodeAt(i); };
|
||
var charCodeAt = PRIMITIVES["char-code-at"];
|
||
PRIMITIVES["to-hex"] = function(n) { return (n >>> 0).toString(16); };
|
||
var toHex = PRIMITIVES["to-hex"];
|
||
|
||
// =========================================================================
|
||
// Platform: CEK module — explicit CEK machine
|
||
// =========================================================================
|
||
|
||
// Continuation type — callable as JS function so isCallable/apply work.
|
||
// CEK is the canonical evaluator; continuations are always available.
|
||
function Continuation(fn) {
|
||
var c = function(value) { return fn(value !== undefined ? value : NIL); };
|
||
c.fn = fn;
|
||
c._continuation = true;
|
||
c.call = function(value) { return fn(value !== undefined ? value : NIL); };
|
||
return c;
|
||
}
|
||
PRIMITIVES["continuation?"] = function(x) { return x != null && x._continuation === true; };
|
||
|
||
// Standalone aliases for primitives used by cek.sx / frames.sx
|
||
var inc = PRIMITIVES["inc"];
|
||
var dec = PRIMITIVES["dec"];
|
||
var zip_pairs = PRIMITIVES["zip-pairs"];
|
||
|
||
var continuation_p = PRIMITIVES["continuation?"];
|
||
|
||
function makeCekContinuation(captured, restKont) {
|
||
var c = new Continuation(function(v) { return v !== undefined ? v : NIL; });
|
||
c._cek_data = {"captured": captured, "rest-kont": restKont};
|
||
return c;
|
||
}
|
||
function continuationData(c) {
|
||
return (c && c._cek_data) ? c._cek_data : {};
|
||
}
|
||
|
||
|
||
// === Transpiled from evaluator (frames + eval + CEK) ===
|
||
|
||
// make-cek-state
|
||
var makeCekState = function(control, env, kont) { return {"control": control, "env": env, "kont": kont, "value": NIL, "phase": "eval"}; };
|
||
PRIMITIVES["make-cek-state"] = makeCekState;
|
||
|
||
// make-cek-value
|
||
var makeCekValue = function(value, env, kont) { return {"control": NIL, "env": env, "kont": kont, "value": value, "phase": "continue"}; };
|
||
PRIMITIVES["make-cek-value"] = makeCekValue;
|
||
|
||
// make-cek-suspended
|
||
var makeCekSuspended = function(request, env, kont) { return {"env": env, "kont": kont, "phase": "io-suspended", "request": request}; };
|
||
PRIMITIVES["make-cek-suspended"] = makeCekSuspended;
|
||
|
||
// cek-terminal?
|
||
var cekTerminal_p = function(state) { return (isSxTruthy(sxEq(get(state, "phase"), "continue")) && isEmpty(get(state, "kont"))); };
|
||
PRIMITIVES["cek-terminal?"] = cekTerminal_p;
|
||
|
||
// cek-suspended?
|
||
var cekSuspended_p = function(state) { return sxEq(get(state, "phase"), "io-suspended"); };
|
||
PRIMITIVES["cek-suspended?"] = cekSuspended_p;
|
||
|
||
// cek-control
|
||
var cekControl = function(s) { return get(s, "control"); };
|
||
PRIMITIVES["cek-control"] = cekControl;
|
||
|
||
// cek-env
|
||
var cekEnv = function(s) { return get(s, "env"); };
|
||
PRIMITIVES["cek-env"] = cekEnv;
|
||
|
||
// cek-kont
|
||
var cekKont = function(s) { return get(s, "kont"); };
|
||
PRIMITIVES["cek-kont"] = cekKont;
|
||
|
||
// cek-phase
|
||
var cekPhase = function(s) { return get(s, "phase"); };
|
||
PRIMITIVES["cek-phase"] = cekPhase;
|
||
|
||
// cek-io-request
|
||
var cekIoRequest = function(s) { return get(s, "request"); };
|
||
PRIMITIVES["cek-io-request"] = cekIoRequest;
|
||
|
||
// cek-value
|
||
var cekValue = function(s) { return get(s, "value"); };
|
||
PRIMITIVES["cek-value"] = cekValue;
|
||
|
||
// make-if-frame
|
||
var makeIfFrame = function(thenExpr, elseExpr, env) { return {"else": elseExpr, "env": env, "type": "if", "then": thenExpr}; };
|
||
PRIMITIVES["make-if-frame"] = makeIfFrame;
|
||
|
||
// make-when-frame
|
||
var makeWhenFrame = function(bodyExprs, env) { return {"body": bodyExprs, "env": env, "type": "when"}; };
|
||
PRIMITIVES["make-when-frame"] = makeWhenFrame;
|
||
|
||
// make-begin-frame
|
||
var makeBeginFrame = function(remaining, env) { return {"env": env, "type": "begin", "remaining": remaining}; };
|
||
PRIMITIVES["make-begin-frame"] = makeBeginFrame;
|
||
|
||
// make-let-frame
|
||
var makeLetFrame = function(name, remaining, body, local) { return {"body": body, "env": local, "type": "let", "remaining": remaining, "name": name}; };
|
||
PRIMITIVES["make-let-frame"] = makeLetFrame;
|
||
|
||
// make-define-frame
|
||
var makeDefineFrame = function(name, env, hasEffects, effectList) { return {"env": env, "effect-list": effectList, "has-effects": hasEffects, "type": "define", "name": name}; };
|
||
PRIMITIVES["make-define-frame"] = makeDefineFrame;
|
||
|
||
// make-define-foreign-frame
|
||
var makeDefineForeignFrame = function(name, spec, env) { return {"spec": spec, "env": env, "type": "define-foreign", "name": name}; };
|
||
PRIMITIVES["make-define-foreign-frame"] = makeDefineForeignFrame;
|
||
|
||
// make-set-frame
|
||
var makeSetFrame = function(name, env) { return {"env": env, "type": "set", "name": name}; };
|
||
PRIMITIVES["make-set-frame"] = makeSetFrame;
|
||
|
||
// make-arg-frame
|
||
var makeArgFrame = function(f, evaled, remaining, env, rawArgs, headName) { return {"env": env, "head-name": sxOr(headName, NIL), "evaled": evaled, "type": "arg", "f": f, "remaining": remaining, "raw-args": rawArgs}; };
|
||
PRIMITIVES["make-arg-frame"] = makeArgFrame;
|
||
|
||
// make-call-frame
|
||
var makeCallFrame = function(f, args, env) { return {"args": args, "env": env, "type": "call", "f": f}; };
|
||
PRIMITIVES["make-call-frame"] = makeCallFrame;
|
||
|
||
// make-cond-frame
|
||
var makeCondFrame = function(remaining, env, scheme_p) { return {"scheme": scheme_p, "env": env, "type": "cond", "remaining": remaining}; };
|
||
PRIMITIVES["make-cond-frame"] = makeCondFrame;
|
||
|
||
// make-cond-arrow-frame
|
||
var makeCondArrowFrame = function(testValue, env) { return {"env": env, "match-val": testValue, "type": "cond-arrow"}; };
|
||
PRIMITIVES["make-cond-arrow-frame"] = makeCondArrowFrame;
|
||
|
||
// make-case-frame
|
||
var makeCaseFrame = function(matchVal, remaining, env) { return {"match-val": matchVal, "env": env, "type": "case", "remaining": remaining}; };
|
||
PRIMITIVES["make-case-frame"] = makeCaseFrame;
|
||
|
||
// make-thread-frame
|
||
var makeThreadFrame = function(remaining, env, mode, name) { return {"env": env, "type": "thread", "extra": mode, "remaining": remaining, "name": name}; };
|
||
PRIMITIVES["make-thread-frame"] = makeThreadFrame;
|
||
|
||
// thread-insert-arg
|
||
var threadInsertArg = function(form, value, fenv) { return (isSxTruthy(sxEq(typeOf(form), "list")) ? evalExpr(cons(first(form), cons([new Symbol("quote"), value], rest(form))), fenv) : evalExpr([form, [new Symbol("quote"), value]], fenv)); };
|
||
PRIMITIVES["thread-insert-arg"] = threadInsertArg;
|
||
|
||
// thread-insert-arg-last
|
||
var threadInsertArgLast = function(form, value, fenv) { return (isSxTruthy(sxEq(typeOf(form), "list")) ? evalExpr(append(form, [[new Symbol("quote"), value]]), fenv) : evalExpr([form, [new Symbol("quote"), value]], fenv)); };
|
||
PRIMITIVES["thread-insert-arg-last"] = threadInsertArgLast;
|
||
|
||
// make-map-frame
|
||
var makeMapFrame = function(f, remaining, results, env) { return {"indexed": false, "env": env, "results": results, "type": "map", "f": f, "remaining": remaining}; };
|
||
PRIMITIVES["make-map-frame"] = makeMapFrame;
|
||
|
||
// make-map-indexed-frame
|
||
var makeMapIndexedFrame = function(f, remaining, results, env) { return {"indexed": true, "env": env, "results": results, "type": "map", "f": f, "remaining": remaining}; };
|
||
PRIMITIVES["make-map-indexed-frame"] = makeMapIndexedFrame;
|
||
|
||
// make-multi-map-frame
|
||
var makeMultiMapFrame = function(f, remainingLists, results, env) { return {"env": env, "results": results, "type": "multi-map", "f": f, "remaining": remainingLists}; };
|
||
PRIMITIVES["make-multi-map-frame"] = makeMultiMapFrame;
|
||
|
||
// make-filter-frame
|
||
var makeFilterFrame = function(f, remaining, results, currentItem, env) { return {"current-item": currentItem, "env": env, "results": results, "type": "filter", "f": f, "remaining": remaining}; };
|
||
PRIMITIVES["make-filter-frame"] = makeFilterFrame;
|
||
|
||
// make-reduce-frame
|
||
var makeReduceFrame = function(f, remaining, env) { return {"env": env, "type": "reduce", "f": f, "remaining": remaining}; };
|
||
PRIMITIVES["make-reduce-frame"] = makeReduceFrame;
|
||
|
||
// make-for-each-frame
|
||
var makeForEachFrame = function(f, remaining, env) { return {"env": env, "type": "for-each", "f": f, "remaining": remaining}; };
|
||
PRIMITIVES["make-for-each-frame"] = makeForEachFrame;
|
||
|
||
// make-some-frame
|
||
var makeSomeFrame = function(f, remaining, env) { return {"env": env, "type": "some", "f": f, "remaining": remaining}; };
|
||
PRIMITIVES["make-some-frame"] = makeSomeFrame;
|
||
|
||
// make-every-frame
|
||
var makeEveryFrame = function(f, remaining, env) { return {"env": env, "type": "every", "f": f, "remaining": remaining}; };
|
||
PRIMITIVES["make-every-frame"] = makeEveryFrame;
|
||
|
||
// make-scope-frame
|
||
var makeScopeFrame = function(name, remaining, env) { return {"env": env, "type": "scope", "remaining": remaining, "name": name}; };
|
||
PRIMITIVES["make-scope-frame"] = makeScopeFrame;
|
||
|
||
// make-provide-frame
|
||
var makeProvideFrame = function(name, value, remaining, env) { return {"subscribers": [], "env": env, "value": value, "type": "provide", "remaining": remaining, "name": name}; };
|
||
PRIMITIVES["make-provide-frame"] = makeProvideFrame;
|
||
|
||
// make-bind-frame
|
||
var makeBindFrame = function(body, env, prevTracking) { return {"body": body, "env": env, "type": "bind", "prev-tracking": prevTracking}; };
|
||
PRIMITIVES["make-bind-frame"] = makeBindFrame;
|
||
|
||
// make-provide-set-frame
|
||
var makeProvideSetFrame = function(name, env) { return {"env": env, "type": "provide-set", "name": name}; };
|
||
PRIMITIVES["make-provide-set-frame"] = makeProvideSetFrame;
|
||
|
||
// make-scope-acc-frame
|
||
var makeScopeAccFrame = function(name, value, remaining, env) { return {"env": env, "value": sxOr(value, NIL), "type": "scope-acc", "remaining": remaining, "emitted": [], "name": name}; };
|
||
PRIMITIVES["make-scope-acc-frame"] = makeScopeAccFrame;
|
||
|
||
// make-reset-frame
|
||
var makeResetFrame = function(env) { return {"env": env, "type": "reset"}; };
|
||
PRIMITIVES["make-reset-frame"] = makeResetFrame;
|
||
|
||
// make-dict-frame
|
||
var makeDictFrame = function(remaining, results, env) { return {"env": env, "results": results, "type": "dict", "remaining": remaining}; };
|
||
PRIMITIVES["make-dict-frame"] = makeDictFrame;
|
||
|
||
// make-and-frame
|
||
var makeAndFrame = function(remaining, env) { return {"env": env, "type": "and", "remaining": remaining}; };
|
||
PRIMITIVES["make-and-frame"] = makeAndFrame;
|
||
|
||
// make-or-frame
|
||
var makeOrFrame = function(remaining, env) { return {"env": env, "type": "or", "remaining": remaining}; };
|
||
PRIMITIVES["make-or-frame"] = makeOrFrame;
|
||
|
||
// make-dynamic-wind-frame
|
||
var makeDynamicWindFrame = function(phase, bodyThunk, afterThunk, env) { return {"env": env, "phase": phase, "after-thunk": afterThunk, "type": "dynamic-wind", "body-thunk": bodyThunk}; };
|
||
PRIMITIVES["make-dynamic-wind-frame"] = makeDynamicWindFrame;
|
||
|
||
// make-reactive-reset-frame
|
||
var makeReactiveResetFrame = function(env, updateFn, firstRender_p) { return {"first-render": firstRender_p, "update-fn": updateFn, "env": env, "type": "reactive-reset"}; };
|
||
PRIMITIVES["make-reactive-reset-frame"] = makeReactiveResetFrame;
|
||
|
||
// make-callcc-frame
|
||
var makeCallccFrame = function(env) { return {"env": env, "type": "callcc"}; };
|
||
PRIMITIVES["make-callcc-frame"] = makeCallccFrame;
|
||
|
||
// make-wind-after-frame
|
||
var makeWindAfterFrame = function(afterThunk, windersLen, env) { return {"winders-len": windersLen, "env": env, "after-thunk": afterThunk, "type": "wind-after"}; };
|
||
PRIMITIVES["make-wind-after-frame"] = makeWindAfterFrame;
|
||
|
||
// make-wind-return-frame
|
||
var makeWindReturnFrame = function(bodyResult, env) { return {"body-result": bodyResult, "env": env, "type": "wind-return"}; };
|
||
PRIMITIVES["make-wind-return-frame"] = makeWindReturnFrame;
|
||
|
||
// make-deref-frame
|
||
var makeDerefFrame = function(env) { return {"env": env, "type": "deref"}; };
|
||
PRIMITIVES["make-deref-frame"] = makeDerefFrame;
|
||
|
||
// make-ho-setup-frame
|
||
var makeHoSetupFrame = function(hoType, remainingArgs, evaledArgs, env) { return {"ho-type": hoType, "env": env, "evaled": evaledArgs, "type": "ho-setup", "remaining": remainingArgs}; };
|
||
PRIMITIVES["make-ho-setup-frame"] = makeHoSetupFrame;
|
||
|
||
// make-comp-trace-frame
|
||
var makeCompTraceFrame = function(name, file) { return {"env": file, "type": "comp-trace", "name": name}; };
|
||
PRIMITIVES["make-comp-trace-frame"] = makeCompTraceFrame;
|
||
|
||
// kont-collect-comp-trace
|
||
var kontCollectCompTrace = function(kont) { return (isSxTruthy(isEmpty(kont)) ? [] : (function() {
|
||
var frame = first(kont);
|
||
return (isSxTruthy(sxEq(frameType(frame), "comp-trace")) ? cons({"file": get(frame, "file"), "name": get(frame, "name")}, kontCollectCompTrace(rest(kont))) : kontCollectCompTrace(rest(kont)));
|
||
})()); };
|
||
PRIMITIVES["kont-collect-comp-trace"] = kontCollectCompTrace;
|
||
|
||
// make-handler-frame
|
||
var makeHandlerFrame = function(handlers, remaining, env) { return {"env": env, "type": "handler", "f": handlers, "remaining": remaining}; };
|
||
PRIMITIVES["make-handler-frame"] = makeHandlerFrame;
|
||
|
||
// make-restart-frame
|
||
var makeRestartFrame = function(restarts, remaining, env) { return {"env": env, "type": "restart", "f": restarts, "remaining": remaining}; };
|
||
PRIMITIVES["make-restart-frame"] = makeRestartFrame;
|
||
|
||
// make-signal-return-frame
|
||
var makeSignalReturnFrame = function(env, savedKont) { return {"env": env, "type": "signal-return", "f": savedKont}; };
|
||
PRIMITIVES["make-signal-return-frame"] = makeSignalReturnFrame;
|
||
|
||
// make-raise-eval-frame
|
||
var makeRaiseEvalFrame = function(env, continuable_p) { return {"scheme": continuable_p, "env": env, "type": "raise-eval"}; };
|
||
PRIMITIVES["make-raise-eval-frame"] = makeRaiseEvalFrame;
|
||
|
||
// make-raise-guard-frame
|
||
var makeRaiseGuardFrame = function(env, savedKont) { return {"env": env, "type": "raise-guard", "remaining": savedKont}; };
|
||
PRIMITIVES["make-raise-guard-frame"] = makeRaiseGuardFrame;
|
||
|
||
// make-perform-frame
|
||
var makePerformFrame = function(env) { return {"env": env, "type": "perform"}; };
|
||
PRIMITIVES["make-perform-frame"] = makePerformFrame;
|
||
|
||
// make-vm-resume-frame
|
||
var makeVmResumeFrame = function(resumeFn, env) { return {"env": env, "type": "vm-resume", "f": resumeFn}; };
|
||
PRIMITIVES["make-vm-resume-frame"] = makeVmResumeFrame;
|
||
|
||
// make-import-frame
|
||
var makeImportFrame = function(importSet, remainingSets, env) { return {"args": importSet, "env": env, "type": "import", "remaining": remainingSets}; };
|
||
PRIMITIVES["make-import-frame"] = makeImportFrame;
|
||
|
||
// make-parameterize-frame
|
||
var makeParameterizeFrame = function(remaining, currentParam, results, body, env) { return {"env": env, "body": body, "results": results, "type": "parameterize", "f": currentParam, "remaining": remaining}; };
|
||
PRIMITIVES["make-parameterize-frame"] = makeParameterizeFrame;
|
||
|
||
// find-matching-handler
|
||
var findMatchingHandler = function(handlers, condition) { return (isSxTruthy(isEmpty(handlers)) ? NIL : (function() {
|
||
var pair = first(handlers);
|
||
return (function() {
|
||
var pred = first(pair);
|
||
var handlerFn = nth(pair, 1);
|
||
return (isSxTruthy(cekCall(pred, [condition])) ? handlerFn : findMatchingHandler(rest(handlers), condition));
|
||
})();
|
||
})()); };
|
||
PRIMITIVES["find-matching-handler"] = findMatchingHandler;
|
||
|
||
// kont-find-handler
|
||
var kontFindHandler = function(kont, condition) { return (isSxTruthy(isEmpty(kont)) ? NIL : (function() {
|
||
var frame = first(kont);
|
||
return (isSxTruthy(sxEq(frameType(frame), "handler")) ? (function() {
|
||
var match = findMatchingHandler(get(frame, "f"), condition);
|
||
return (isSxTruthy(isNil(match)) ? kontFindHandler(rest(kont), condition) : match);
|
||
})() : kontFindHandler(rest(kont), condition));
|
||
})()); };
|
||
PRIMITIVES["kont-find-handler"] = kontFindHandler;
|
||
|
||
// kont-unwind-to-handler
|
||
var kontUnwindToHandler = function(kont, condition) { return (isSxTruthy(isEmpty(kont)) ? {"handler": NIL, "kont": kont} : (function() {
|
||
var frame = first(kont);
|
||
var restK = rest(kont);
|
||
return (isSxTruthy(sxEq(frameType(frame), "handler")) ? (function() {
|
||
var match = findMatchingHandler(get(frame, "f"), condition);
|
||
return (isSxTruthy(isNil(match)) ? kontUnwindToHandler(restK, condition) : {"handler": match, "kont": kont});
|
||
})() : (isSxTruthy(sxEq(frameType(frame), "wind-after")) ? ((isSxTruthy((len(_winders_) > get(frame, "winders-len"))) ? (_winders_ = rest(_winders_)) : NIL), cekCall(get(frame, "after-thunk"), []), kontUnwindToHandler(restK, condition)) : kontUnwindToHandler(restK, condition)));
|
||
})()); };
|
||
PRIMITIVES["kont-unwind-to-handler"] = kontUnwindToHandler;
|
||
|
||
// wind-escape-to
|
||
var windEscapeTo = function(targetLen) { return (isSxTruthy((len(_winders_) > targetLen)) ? (function() {
|
||
var afterThunk = first(_winders_);
|
||
_winders_ = rest(_winders_);
|
||
cekCall(afterThunk, []);
|
||
return windEscapeTo(targetLen);
|
||
})() : NIL); };
|
||
PRIMITIVES["wind-escape-to"] = windEscapeTo;
|
||
|
||
// find-named-restart
|
||
var findNamedRestart = function(restarts, name) { return (isSxTruthy(isEmpty(restarts)) ? NIL : (function() {
|
||
var entry = first(restarts);
|
||
return (isSxTruthy(sxEq(first(entry), name)) ? entry : findNamedRestart(rest(restarts), name));
|
||
})()); };
|
||
PRIMITIVES["find-named-restart"] = findNamedRestart;
|
||
|
||
// kont-find-restart
|
||
var kontFindRestart = function(kont, name) { return (isSxTruthy(isEmpty(kont)) ? NIL : (function() {
|
||
var frame = first(kont);
|
||
return (isSxTruthy(sxEq(frameType(frame), "restart")) ? (function() {
|
||
var match = findNamedRestart(get(frame, "f"), name);
|
||
return (isSxTruthy(isNil(match)) ? kontFindRestart(rest(kont), name) : [match, frame, rest(kont)]);
|
||
})() : kontFindRestart(rest(kont), name));
|
||
})()); };
|
||
PRIMITIVES["kont-find-restart"] = kontFindRestart;
|
||
|
||
// frame-type
|
||
var frameType = function(f) { return get(f, "type"); };
|
||
PRIMITIVES["frame-type"] = frameType;
|
||
|
||
// kont-push
|
||
var kontPush = function(frame, kont) { return cons(frame, kont); };
|
||
PRIMITIVES["kont-push"] = kontPush;
|
||
|
||
// kont-top
|
||
var kontTop = function(kont) { return first(kont); };
|
||
PRIMITIVES["kont-top"] = kontTop;
|
||
|
||
// kont-pop
|
||
var kontPop = function(kont) { return rest(kont); };
|
||
PRIMITIVES["kont-pop"] = kontPop;
|
||
|
||
// kont-empty?
|
||
var kontEmpty_p = function(kont) { return isEmpty(kont); };
|
||
PRIMITIVES["kont-empty?"] = kontEmpty_p;
|
||
|
||
// kont-capture-to-reset
|
||
var kontCaptureToReset = function(kont) { var scan = function(k, captured) { return (isSxTruthy(isEmpty(k)) ? error("shift without enclosing reset") : (function() {
|
||
var frame = first(k);
|
||
return (isSxTruthy(sxOr(sxEq(frameType(frame), "reset"), sxEq(frameType(frame), "reactive-reset"))) ? [captured, rest(k)] : scan(rest(k), append(captured, [frame])));
|
||
})()); };
|
||
PRIMITIVES["scan"] = scan;
|
||
return scan(kont, []); };
|
||
PRIMITIVES["kont-capture-to-reset"] = kontCaptureToReset;
|
||
|
||
// kont-push-provides
|
||
var kontPushProvides = function(pairs, env, kont) { return (isSxTruthy(isEmpty(pairs)) ? kont : (function() {
|
||
var pair = first(pairs);
|
||
return kontPushProvides(rest(pairs), env, cons(makeProvideFrame(first(pair), nth(pair, 1), [], env), kont));
|
||
})()); };
|
||
PRIMITIVES["kont-push-provides"] = kontPushProvides;
|
||
|
||
// kont-find-provide
|
||
var kontFindProvide = function(kont, name) { return (isSxTruthy(isEmpty(kont)) ? NIL : (function() {
|
||
var frame = first(kont);
|
||
return (isSxTruthy((isSxTruthy(sxEq(frameType(frame), "provide")) && sxEq(get(frame, "name"), name))) ? frame : kontFindProvide(rest(kont), name));
|
||
})()); };
|
||
PRIMITIVES["kont-find-provide"] = kontFindProvide;
|
||
|
||
// kont-find-scope-acc
|
||
var kontFindScopeAcc = function(kont, name) { return (isSxTruthy(isEmpty(kont)) ? NIL : (function() {
|
||
var frame = first(kont);
|
||
return (isSxTruthy((isSxTruthy(sxEq(frameType(frame), "scope-acc")) && sxEq(get(frame, "name"), name))) ? frame : kontFindScopeAcc(rest(kont), name));
|
||
})()); };
|
||
PRIMITIVES["kont-find-scope-acc"] = kontFindScopeAcc;
|
||
|
||
// has-reactive-reset-frame?
|
||
var hasReactiveResetFrame_p = function(kont) { return (isSxTruthy(isEmpty(kont)) ? false : (isSxTruthy(sxEq(frameType(first(kont)), "reactive-reset")) ? true : hasReactiveResetFrame_p(rest(kont)))); };
|
||
PRIMITIVES["has-reactive-reset-frame?"] = hasReactiveResetFrame_p;
|
||
|
||
// kont-capture-to-reactive-reset
|
||
var kontCaptureToReactiveReset = function(kont) { var scan = function(k, captured) { return (isSxTruthy(isEmpty(k)) ? error("reactive deref without enclosing reactive-reset") : (function() {
|
||
var frame = first(k);
|
||
return (isSxTruthy(sxEq(frameType(frame), "reactive-reset")) ? [captured, frame, rest(k)] : scan(rest(k), append(captured, [frame])));
|
||
})()); };
|
||
PRIMITIVES["scan"] = scan;
|
||
return scan(kont, []); };
|
||
PRIMITIVES["kont-capture-to-reactive-reset"] = kontCaptureToReactiveReset;
|
||
|
||
// *custom-special-forms*
|
||
var _customSpecialForms = {};
|
||
PRIMITIVES["*custom-special-forms*"] = _customSpecialForms;
|
||
|
||
// register-special-form!
|
||
var registerSpecialForm = function(name, handler) { return dictSet(_customSpecialForms, name, handler); };
|
||
PRIMITIVES["register-special-form!"] = registerSpecialForm;
|
||
|
||
// *render-check*
|
||
var _renderCheck = NIL;
|
||
PRIMITIVES["*render-check*"] = _renderCheck;
|
||
|
||
// *render-fn*
|
||
var _renderFn = NIL;
|
||
PRIMITIVES["*render-fn*"] = _renderFn;
|
||
|
||
// *bind-tracking*
|
||
var _bindTracking_ = NIL;
|
||
PRIMITIVES["*bind-tracking*"] = _bindTracking_;
|
||
|
||
// *provide-batch-depth*
|
||
var _provideBatchDepth_ = 0;
|
||
PRIMITIVES["*provide-batch-depth*"] = _provideBatchDepth_;
|
||
|
||
// *provide-batch-queue*
|
||
var _provideBatchQueue_ = [];
|
||
PRIMITIVES["*provide-batch-queue*"] = _provideBatchQueue_;
|
||
|
||
// *provide-subscribers*
|
||
var _provideSubscribers_ = {};
|
||
PRIMITIVES["*provide-subscribers*"] = _provideSubscribers_;
|
||
|
||
// *winders*
|
||
var _winders_ = [];
|
||
PRIMITIVES["*winders*"] = _winders_;
|
||
|
||
// *library-registry*
|
||
var _libraryRegistry_ = {};
|
||
PRIMITIVES["*library-registry*"] = _libraryRegistry_;
|
||
|
||
// library-name-key
|
||
var libraryNameKey = function(spec) { return join(".", map(function(s) { return (isSxTruthy(symbol_p(s)) ? symbolName(s) : (String(s))); }, spec)); };
|
||
PRIMITIVES["library-name-key"] = libraryNameKey;
|
||
|
||
// library-loaded?
|
||
var libraryLoaded_p = function(spec) { return dictHas(_libraryRegistry_, libraryNameKey(spec)); };
|
||
PRIMITIVES["library-loaded?"] = libraryLoaded_p;
|
||
|
||
// library-exports
|
||
var libraryExports = function(spec) { return get(get(_libraryRegistry_, libraryNameKey(spec)), "exports"); };
|
||
PRIMITIVES["library-exports"] = libraryExports;
|
||
|
||
// register-library
|
||
var registerLibrary = function(spec, exports) { return dictSet(_libraryRegistry_, libraryNameKey(spec), {"exports": exports}); };
|
||
PRIMITIVES["register-library"] = registerLibrary;
|
||
|
||
// *io-registry*
|
||
var _ioRegistry_ = {};
|
||
PRIMITIVES["*io-registry*"] = _ioRegistry_;
|
||
|
||
// io-register!
|
||
var ioRegister_b = function(name, spec) { return dictSet(_ioRegistry_, name, spec); };
|
||
PRIMITIVES["io-register!"] = ioRegister_b;
|
||
|
||
// io-registered?
|
||
var ioRegistered_p = function(name) { return dictHas(_ioRegistry_, name); };
|
||
PRIMITIVES["io-registered?"] = ioRegistered_p;
|
||
|
||
// io-lookup
|
||
var ioLookup = function(name) { return get(_ioRegistry_, name); };
|
||
PRIMITIVES["io-lookup"] = ioLookup;
|
||
|
||
// io-names
|
||
var ioNames = function() { return keys(_ioRegistry_); };
|
||
PRIMITIVES["io-names"] = ioNames;
|
||
|
||
// *foreign-registry*
|
||
var _foreignRegistry_ = {};
|
||
PRIMITIVES["*foreign-registry*"] = _foreignRegistry_;
|
||
|
||
// foreign-register!
|
||
var foreignRegister_b = function(name, spec) { return dictSet(_foreignRegistry_, name, spec); };
|
||
PRIMITIVES["foreign-register!"] = foreignRegister_b;
|
||
|
||
// foreign-registered?
|
||
var foreignRegistered_p = function(name) { return dictHas(_foreignRegistry_, name); };
|
||
PRIMITIVES["foreign-registered?"] = foreignRegistered_p;
|
||
|
||
// foreign-lookup
|
||
var foreignLookup = function(name) { return get(_foreignRegistry_, name); };
|
||
PRIMITIVES["foreign-lookup"] = foreignLookup;
|
||
|
||
// foreign-names
|
||
var foreignNames = function() { return keys(_foreignRegistry_); };
|
||
PRIMITIVES["foreign-names"] = foreignNames;
|
||
|
||
// foreign-parse-params
|
||
var foreignParseParams = function(paramList) { return (function() {
|
||
var result = [];
|
||
var i = 0;
|
||
var items = (isSxTruthy(isList(paramList)) ? paramList : []);
|
||
return foreignParseParamsLoop(items, result);
|
||
})(); };
|
||
PRIMITIVES["foreign-parse-params"] = foreignParseParams;
|
||
|
||
// foreign-parse-kwargs!
|
||
var foreignParseKwargs_b = function(spec, remaining) { return (isSxTruthy((isSxTruthy(!isSxTruthy(isEmpty(remaining))) && isSxTruthy((len(remaining) >= 2)) && keyword_p(first(remaining)))) ? (dictSet(spec, keywordName(first(remaining)), (function() {
|
||
var v = nth(remaining, 1);
|
||
return (isSxTruthy(keyword_p(v)) ? keywordName(v) : v);
|
||
})()), foreignParseKwargs_b(spec, rest(rest(remaining)))) : NIL); };
|
||
PRIMITIVES["foreign-parse-kwargs!"] = foreignParseKwargs_b;
|
||
|
||
// foreign-resolve-binding
|
||
var foreignResolveBinding = function(bindingStr) { return (function() {
|
||
var parts = split(bindingStr, ".");
|
||
return (isSxTruthy((len(parts) <= 1)) ? {"method": bindingStr, "object": NIL} : (function() {
|
||
var method = last(parts);
|
||
var obj = join(".", reverse(rest(reverse(parts))));
|
||
return {"method": method, "object": obj};
|
||
})());
|
||
})(); };
|
||
PRIMITIVES["foreign-resolve-binding"] = foreignResolveBinding;
|
||
|
||
// foreign-check-args
|
||
var foreignCheckArgs = function(name, params, args) { if (isSxTruthy((isSxTruthy(!isSxTruthy(isEmpty(params))) && (len(args) < len(params))))) {
|
||
error((String("foreign ") + String(name) + String(": expected ") + String(len(params)) + String(" args, got ") + String(len(args))));
|
||
}
|
||
return forEach(function(i) { return (function() {
|
||
var spec = nth(params, i);
|
||
var val = nth(args, i);
|
||
var expected = get(spec, "type");
|
||
return (isSxTruthy((isSxTruthy(!isSxTruthy(sxEq(expected, "any"))) && !isSxTruthy(valueMatchesType_p(val, expected)))) ? error((String("foreign ") + String(name) + String(": arg '") + String(get(spec, "name")) + String("' expected ") + String(expected) + String(", got ") + String(typeOf(val)))) : NIL);
|
||
})(); }, range(0, min(len(params), len(args)))); };
|
||
PRIMITIVES["foreign-check-args"] = foreignCheckArgs;
|
||
|
||
// foreign-build-lambda
|
||
var foreignBuildLambda = function(spec) { return (function() {
|
||
var name = get(spec, "name");
|
||
var mode = (isSxTruthy(dictHas(spec, "returns")) ? (function() {
|
||
var r = get(spec, "returns");
|
||
return (isSxTruthy(sxEq(r, "promise")) ? "async" : "sync");
|
||
})() : "sync");
|
||
return (isSxTruthy(sxEq(mode, "async")) ? [new Symbol("fn"), [new Symbol("&rest"), new Symbol("__ffi-args__")], [new Symbol("perform"), [new Symbol("foreign-dispatch"), [new Symbol("quote"), name], new Symbol("__ffi-args__")]]] : [new Symbol("fn"), [new Symbol("&rest"), new Symbol("__ffi-args__")], [new Symbol("foreign-dispatch"), [new Symbol("quote"), name], new Symbol("__ffi-args__")]]);
|
||
})(); };
|
||
PRIMITIVES["foreign-build-lambda"] = foreignBuildLambda;
|
||
|
||
// sf-define-foreign
|
||
var sfDefineForeign = function(args, env) { return (function() {
|
||
var name = (isSxTruthy(symbol_p(first(args))) ? symbolName(first(args)) : first(args));
|
||
var paramList = nth(args, 1);
|
||
var spec = {};
|
||
spec["name"] = name;
|
||
spec["params"] = foreignParseParams(paramList);
|
||
foreignParseKwargs_b(spec, rest(rest(args)));
|
||
foreignRegister_b(name, spec);
|
||
return spec;
|
||
})(); };
|
||
PRIMITIVES["sf-define-foreign"] = sfDefineForeign;
|
||
|
||
// step-sf-define-foreign
|
||
var stepSfDefineForeign = function(args, env, kont) { return (function() {
|
||
var spec = sfDefineForeign(args, env);
|
||
var name = (isSxTruthy(symbol_p(first(args))) ? symbolName(first(args)) : first(args));
|
||
var lambdaExpr = foreignBuildLambda(spec);
|
||
return makeCekState(lambdaExpr, env, kontPush(makeDefineForeignFrame(name, spec, env), kont));
|
||
})(); };
|
||
PRIMITIVES["step-sf-define-foreign"] = stepSfDefineForeign;
|
||
|
||
// foreign-dispatch
|
||
var foreignDispatch = function(name, args) { return (function() {
|
||
var spec = foreignLookup(name);
|
||
if (isSxTruthy(isNil(spec))) {
|
||
error((String("foreign-dispatch: unknown foreign function '") + String(name) + String("'")));
|
||
}
|
||
return (function() {
|
||
var params = get(spec, "params");
|
||
var binding = get(spec, "js");
|
||
foreignCheckArgs(name, (isSxTruthy(isNil(params)) ? [] : params), args);
|
||
return (isSxTruthy(isNil(binding)) ? error((String("foreign ") + String(name) + String(": no binding for current platform"))) : (function() {
|
||
var resolved = foreignResolveBinding(binding);
|
||
var objName = get(resolved, "object");
|
||
var method = get(resolved, "method");
|
||
return (isSxTruthy(isPrimitive("host-call")) ? (isSxTruthy(isNil(objName)) ? apply(getPrimitive("host-call"), concat([NIL, method], args)) : (function() {
|
||
var obj = (getPrimitive("host-global"))(objName);
|
||
return apply(getPrimitive("host-call"), concat([obj, method], args));
|
||
})()) : error((String("foreign ") + String(name) + String(": host-call not available on this platform"))));
|
||
})());
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["foreign-dispatch"] = foreignDispatch;
|
||
|
||
// foreign-parse-params-loop
|
||
var foreignParseParamsLoop = function(items, acc) { return (isSxTruthy(isEmpty(items)) ? acc : (function() {
|
||
var item = first(items);
|
||
var restItems = rest(items);
|
||
return (isSxTruthy((isSxTruthy(!isSxTruthy(isEmpty(restItems))) && isSxTruthy(keyword_p(first(restItems))) && isSxTruthy(sxEq(keywordName(first(restItems)), "as")) && (len(restItems) >= 2))) ? foreignParseParamsLoop(rest(rest(restItems)), append(acc, [{"type": (function() {
|
||
var t = nth(restItems, 1);
|
||
return (isSxTruthy(keyword_p(t)) ? keywordName(t) : (String(t)));
|
||
})(), "name": (isSxTruthy(symbol_p(item)) ? symbolName(item) : (String(item)))}])) : foreignParseParamsLoop(restItems, append(acc, [{"type": "any", "name": (isSxTruthy(symbol_p(item)) ? symbolName(item) : (String(item)))}])));
|
||
})()); };
|
||
PRIMITIVES["foreign-parse-params-loop"] = foreignParseParamsLoop;
|
||
|
||
// step-sf-io
|
||
var stepSfIo = function(args, env, kont) { return (function() {
|
||
var name = first(args);
|
||
var ioArgs = rest(args);
|
||
if (isSxTruthy(!isSxTruthy(ioRegistered_p(name)))) {
|
||
error((String("io: unknown operation '") + String(name) + String("' — not in *io-registry*")));
|
||
}
|
||
return makeCekState(cons(new Symbol("perform"), [{"args": ioArgs, "op": name}]), env, kont);
|
||
})(); };
|
||
PRIMITIVES["step-sf-io"] = stepSfIo;
|
||
|
||
// trampoline
|
||
var trampoline = function(val) { return (function() {
|
||
var result = val;
|
||
return (isSxTruthy(isThunk(result)) ? trampoline(evalExpr(thunkExpr(result), thunkEnv(result))) : result);
|
||
})(); };
|
||
PRIMITIVES["trampoline"] = trampoline;
|
||
|
||
// *strict*
|
||
var _strict_ = false;
|
||
PRIMITIVES["*strict*"] = _strict_;
|
||
|
||
// set-strict!
|
||
var setStrict_b = function(val) { return (_strict_ = val); };
|
||
PRIMITIVES["set-strict!"] = setStrict_b;
|
||
|
||
// *prim-param-types*
|
||
var _primParamTypes_ = NIL;
|
||
PRIMITIVES["*prim-param-types*"] = _primParamTypes_;
|
||
|
||
// set-prim-param-types!
|
||
var setPrimParamTypes_b = function(types) { return (_primParamTypes_ = types); };
|
||
PRIMITIVES["set-prim-param-types!"] = setPrimParamTypes_b;
|
||
|
||
// value-matches-type?
|
||
var valueMatchesType_p = function(val, expectedType) { return (function() { var _m = expectedType; if (_m == "any") return true; if (_m == "number") return isNumber(val); if (_m == "string") return isString(val); if (_m == "boolean") return boolean_p(val); if (_m == "nil") return isNil(val); if (_m == "list") return isList(val); if (_m == "dict") return isDict(val); if (_m == "lambda") return isLambda(val); if (_m == "symbol") return sxEq(typeOf(val), "symbol"); if (_m == "keyword") return sxEq(typeOf(val), "keyword"); return (isSxTruthy((isSxTruthy(isString(expectedType)) && endsWith(expectedType, "?"))) ? sxOr(isNil(val), valueMatchesType_p(val, slice(expectedType, 0, (stringLength(expectedType) - 1)))) : true); })(); };
|
||
PRIMITIVES["value-matches-type?"] = valueMatchesType_p;
|
||
|
||
// strict-check-args
|
||
var strictCheckArgs = function(name, args) { return (isSxTruthy((isSxTruthy(_strict_) && _primParamTypes_)) ? (function() {
|
||
var spec = get(_primParamTypes_, name);
|
||
return (isSxTruthy(spec) ? (function() {
|
||
var positional = get(spec, "positional");
|
||
var restType = get(spec, "rest-type");
|
||
if (isSxTruthy(positional)) {
|
||
{ var _c = mapIndexed(function(i, p) { return [i, p]; }, positional); for (var _i = 0; _i < _c.length; _i++) { var pair = _c[_i]; (function() {
|
||
var idx = first(pair);
|
||
var param = nth(pair, 1);
|
||
var pName = first(param);
|
||
var pType = nth(param, 1);
|
||
return (isSxTruthy((idx < len(args))) ? (function() {
|
||
var val = nth(args, idx);
|
||
return (isSxTruthy(!isSxTruthy(valueMatchesType_p(val, pType))) ? error((String("Type error: ") + String(name) + String(" expected ") + String(pType) + String(" for param ") + String(pName) + String(", got ") + String(typeOf(val)) + String(" (") + String((String(val))) + String(")"))) : NIL);
|
||
})() : NIL);
|
||
})(); } }
|
||
}
|
||
return (isSxTruthy((isSxTruthy(restType) && (len(args) > len(sxOr(positional, []))))) ? forEach(function(pair) { return (function() {
|
||
var idx = first(pair);
|
||
var val = nth(pair, 1);
|
||
return (isSxTruthy(!isSxTruthy(valueMatchesType_p(val, restType))) ? error((String("Type error: ") + String(name) + String(" expected ") + String(restType) + String(" for rest arg ") + String(idx) + String(", got ") + String(typeOf(val)) + String(" (") + String((String(val))) + String(")"))) : NIL);
|
||
})(); }, mapIndexed(function(i, v) { return [i, v]; }, slice(args, len(sxOr(positional, []))))) : NIL);
|
||
})() : NIL);
|
||
})() : NIL); };
|
||
PRIMITIVES["strict-check-args"] = strictCheckArgs;
|
||
|
||
// eval-expr
|
||
var evalExpr = function(expr, env) { return NIL; };
|
||
PRIMITIVES["eval-expr"] = evalExpr;
|
||
|
||
// bind-lambda-params
|
||
var bindLambdaParams = function(params, args, local) { return (function() {
|
||
var restIdx = indexOf_(params, "&rest");
|
||
return (isSxTruthy((isSxTruthy(isNumber(restIdx)) && (restIdx < len(params)))) ? (function() {
|
||
var positional = slice(params, 0, restIdx);
|
||
var restName = nth(params, (restIdx + 1));
|
||
return (forEachIndexed(function(i, p) { return envBind(local, p, (isSxTruthy((i < len(args))) ? nth(args, i) : NIL)); }, positional), envBind(local, restName, (isSxTruthy((len(args) > restIdx)) ? slice(args, restIdx) : [])), true);
|
||
})() : false);
|
||
})(); };
|
||
PRIMITIVES["bind-lambda-params"] = bindLambdaParams;
|
||
|
||
// call-lambda
|
||
var callLambda = function(f, args, callerEnv) { return (function() {
|
||
var params = lambdaParams(f);
|
||
var local = envMerge(lambdaClosure(f), callerEnv);
|
||
if (isSxTruthy(!isSxTruthy(bindLambdaParams(params, args, local)))) {
|
||
if (isSxTruthy((len(args) > len(params)))) {
|
||
error((String(sxOr(lambdaName(f), "lambda")) + String(" expects ") + String(len(params)) + String(" args, got ") + String(len(args))));
|
||
}
|
||
{ var _c = zip(params, args); for (var _i = 0; _i < _c.length; _i++) { var pair = _c[_i]; envBind(local, first(pair), nth(pair, 1)); } }
|
||
{ var _c = slice(params, len(args)); for (var _i = 0; _i < _c.length; _i++) { var p = _c[_i]; envBind(local, p, NIL); } }
|
||
}
|
||
return makeThunk(lambdaBody(f), local);
|
||
})(); };
|
||
PRIMITIVES["call-lambda"] = callLambda;
|
||
|
||
// call-component
|
||
var callComponent = function(comp, rawArgs, env) { return (function() {
|
||
var parsed = parseKeywordArgs(rawArgs, env);
|
||
var kwargs = first(parsed);
|
||
var children = nth(parsed, 1);
|
||
var local = envMerge(componentClosure(comp), env);
|
||
{ var _c = componentParams(comp); for (var _i = 0; _i < _c.length; _i++) { var p = _c[_i]; envBind(local, p, sxOr(dictGet(kwargs, p), NIL)); } }
|
||
if (isSxTruthy(componentHasChildren(comp))) {
|
||
envBind(local, "children", children);
|
||
}
|
||
return makeThunk(componentBody(comp), local);
|
||
})(); };
|
||
PRIMITIVES["call-component"] = callComponent;
|
||
|
||
// parse-keyword-args
|
||
var parseKeywordArgs = function(rawArgs, env) { return (function() {
|
||
var kwargs = {};
|
||
var children = [];
|
||
var i = 0;
|
||
reduce(function(state, arg) { return (function() {
|
||
var idx = get(state, "i");
|
||
var skip = get(state, "skip");
|
||
return (isSxTruthy(skip) ? assoc(state, "skip", false, "i", (idx + 1)) : (isSxTruthy((isSxTruthy(sxEq(typeOf(arg), "keyword")) && ((idx + 1) < len(rawArgs)))) ? (dictSet(kwargs, keywordName(arg), trampoline(evalExpr(nth(rawArgs, (idx + 1)), env))), assoc(state, "skip", true, "i", (idx + 1))) : (append_b(children, trampoline(evalExpr(arg, env))), assoc(state, "i", (idx + 1)))));
|
||
})(); }, {["i"]: 0, ["skip"]: false}, rawArgs);
|
||
return [kwargs, children];
|
||
})(); };
|
||
PRIMITIVES["parse-keyword-args"] = parseKeywordArgs;
|
||
|
||
// cond-scheme?
|
||
var condScheme_p = function(clauses) { return isEvery(function(c) { return (isSxTruthy(sxEq(typeOf(c), "list")) && sxOr(sxEq(len(c), 2), (isSxTruthy(sxEq(len(c), 3)) && isSxTruthy(sxEq(typeOf(nth(c, 1)), "symbol")) && sxEq(symbolName(nth(c, 1)), "=>")))); }, clauses); };
|
||
PRIMITIVES["cond-scheme?"] = condScheme_p;
|
||
|
||
// is-else-clause?
|
||
var isElseClause = function(test) { return sxOr((isSxTruthy(sxEq(typeOf(test), "keyword")) && sxEq(keywordName(test), "else")), (isSxTruthy(sxEq(typeOf(test), "symbol")) && sxOr(sxEq(symbolName(test), "else"), sxEq(symbolName(test), ":else")))); };
|
||
PRIMITIVES["is-else-clause?"] = isElseClause;
|
||
|
||
// sf-named-let
|
||
var sfNamedLet = function(args, env) { return (function() {
|
||
var loopName = symbolName(first(args));
|
||
var bindings = nth(args, 1);
|
||
var body = slice(args, 2);
|
||
var params = [];
|
||
var inits = [];
|
||
(isSxTruthy((isSxTruthy(sxEq(typeOf(first(bindings)), "list")) && sxEq(len(first(bindings)), 2))) ? forEach(function(binding) { params.push((isSxTruthy(sxEq(typeOf(first(binding)), "symbol")) ? symbolName(first(binding)) : first(binding)));
|
||
return append_b(inits, nth(binding, 1)); }, bindings) : reduce(function(acc, pairIdx) { return (append_b(params, (isSxTruthy(sxEq(typeOf(nth(bindings, (pairIdx * 2))), "symbol")) ? symbolName(nth(bindings, (pairIdx * 2))) : nth(bindings, (pairIdx * 2)))), append_b(inits, nth(bindings, ((pairIdx * 2) + 1)))); }, NIL, range(0, (len(bindings) / 2))));
|
||
return (function() {
|
||
var loopBody = (isSxTruthy(sxEq(len(body), 1)) ? first(body) : cons(makeSymbol("begin"), body));
|
||
var loopFn = makeLambda(params, loopBody, env);
|
||
loopFn.name = loopName;
|
||
envBind(lambdaClosure(loopFn), loopName, loopFn);
|
||
return (function() {
|
||
var initVals = map(function(e) { return trampoline(evalExpr(e, env)); }, inits);
|
||
return cekCall(loopFn, initVals);
|
||
})();
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sf-named-let"] = sfNamedLet;
|
||
|
||
// sf-lambda
|
||
var sfLambda = function(args, env) { return (function() {
|
||
var paramsExpr = first(args);
|
||
var bodyExprs = rest(args);
|
||
var body = (isSxTruthy(sxEq(len(bodyExprs), 1)) ? first(bodyExprs) : cons(makeSymbol("begin"), bodyExprs));
|
||
var paramNames = map(function(p) { return (isSxTruthy(sxEq(typeOf(p), "symbol")) ? symbolName(p) : (isSxTruthy((isSxTruthy(sxEq(typeOf(p), "list")) && isSxTruthy(sxEq(len(p), 3)) && isSxTruthy(sxEq(typeOf(nth(p, 1)), "keyword")) && sxEq(keywordName(nth(p, 1)), "as"))) ? symbolName(first(p)) : p)); }, paramsExpr);
|
||
return makeLambda(paramNames, body, env);
|
||
})(); };
|
||
PRIMITIVES["sf-lambda"] = sfLambda;
|
||
|
||
// sf-defcomp
|
||
var sfDefcomp = function(args, env) { return (function() {
|
||
var nameSym = first(args);
|
||
var paramsRaw = nth(args, 1);
|
||
var body = last(args);
|
||
var compName = stripPrefix(symbolName(nameSym), "~");
|
||
var parsed = parseCompParams(paramsRaw);
|
||
var params = first(parsed);
|
||
var hasChildren = nth(parsed, 1);
|
||
var paramTypes = nth(parsed, 2);
|
||
var affinity = defcompKwarg(args, "affinity", "auto");
|
||
return (function() {
|
||
var comp = makeComponent(compName, params, hasChildren, body, env, affinity);
|
||
var effects = defcompKwarg(args, "effects", NIL);
|
||
if (isSxTruthy((isSxTruthy(!isSxTruthy(isNil(paramTypes))) && !isSxTruthy(isEmpty(keys(paramTypes)))))) {
|
||
componentSetParamTypes_b(comp, paramTypes);
|
||
}
|
||
if (isSxTruthy(!isSxTruthy(isNil(effects)))) {
|
||
(function() {
|
||
var effectList = (isSxTruthy(sxEq(typeOf(effects), "list")) ? map(function(e) { return (isSxTruthy(sxEq(typeOf(e), "symbol")) ? symbolName(e) : (String(e))); }, effects) : [(String(effects))]);
|
||
var effectAnns = (isSxTruthy(envHas(env, "*effect-annotations*")) ? envGet(env, "*effect-annotations*") : {});
|
||
effectAnns[symbolName(nameSym)] = effectList;
|
||
return envBind(env, "*effect-annotations*", effectAnns);
|
||
})();
|
||
}
|
||
if (isSxTruthy(envHas(env, "*current-file*"))) {
|
||
componentSetFile_b(comp, envGet(env, "*current-file*"));
|
||
}
|
||
envBind(env, symbolName(nameSym), comp);
|
||
return comp;
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sf-defcomp"] = sfDefcomp;
|
||
|
||
// defcomp-kwarg
|
||
var defcompKwarg = function(args, key, default_) { return (function() {
|
||
var end = (len(args) - 1);
|
||
var result = default_;
|
||
{ var _c = range(2, end, 1); for (var _i = 0; _i < _c.length; _i++) { var i = _c[_i]; if (isSxTruthy((isSxTruthy(sxEq(typeOf(nth(args, i)), "keyword")) && isSxTruthy(sxEq(keywordName(nth(args, i)), key)) && ((i + 1) < end)))) {
|
||
(function() {
|
||
var val = nth(args, (i + 1));
|
||
return (result = (isSxTruthy(sxEq(typeOf(val), "keyword")) ? keywordName(val) : val));
|
||
})();
|
||
} } }
|
||
return result;
|
||
})(); };
|
||
PRIMITIVES["defcomp-kwarg"] = defcompKwarg;
|
||
|
||
// parse-comp-params
|
||
var parseCompParams = function(paramsExpr) { return (function() {
|
||
var params = [];
|
||
var paramTypes = {};
|
||
var hasChildren = false;
|
||
var inKey = false;
|
||
{ var _c = paramsExpr; for (var _i = 0; _i < _c.length; _i++) { var p = _c[_i]; (isSxTruthy((isSxTruthy(sxEq(typeOf(p), "list")) && isSxTruthy(sxEq(len(p), 3)) && isSxTruthy(sxEq(typeOf(first(p)), "symbol")) && isSxTruthy(sxEq(typeOf(nth(p, 1)), "keyword")) && sxEq(keywordName(nth(p, 1)), "as"))) ? (function() {
|
||
var name = symbolName(first(p));
|
||
var ptype = nth(p, 2);
|
||
return (function() {
|
||
var typeVal = (isSxTruthy(sxEq(typeOf(ptype), "symbol")) ? symbolName(ptype) : ptype);
|
||
return (isSxTruthy(!isSxTruthy(hasChildren)) ? (append_b(params, name), dictSet(paramTypes, name, typeVal)) : NIL);
|
||
})();
|
||
})() : (isSxTruthy(sxEq(typeOf(p), "symbol")) ? (function() {
|
||
var name = symbolName(p);
|
||
return (isSxTruthy(sxEq(name, "&key")) ? (inKey = true) : (isSxTruthy(sxEq(name, "&rest")) ? (hasChildren = true) : (isSxTruthy(sxEq(name, "&children")) ? (hasChildren = true) : (isSxTruthy(hasChildren) ? NIL : (isSxTruthy(inKey) ? append_b(params, name) : append_b(params, name))))));
|
||
})() : NIL)); } }
|
||
return [params, hasChildren, paramTypes];
|
||
})(); };
|
||
PRIMITIVES["parse-comp-params"] = parseCompParams;
|
||
|
||
// sf-defisland
|
||
var sfDefisland = function(args, env) { return (function() {
|
||
var nameSym = first(args);
|
||
var paramsRaw = nth(args, 1);
|
||
var bodyExprs = slice(args, 2);
|
||
var body = (isSxTruthy(sxEq(len(bodyExprs), 1)) ? first(bodyExprs) : cons(makeSymbol("begin"), bodyExprs));
|
||
var compName = stripPrefix(symbolName(nameSym), "~");
|
||
var parsed = parseCompParams(paramsRaw);
|
||
var params = first(parsed);
|
||
var hasChildren = nth(parsed, 1);
|
||
return (function() {
|
||
var island = makeIsland(compName, params, hasChildren, body, env);
|
||
if (isSxTruthy(envHas(env, "*current-file*"))) {
|
||
componentSetFile_b(island, envGet(env, "*current-file*"));
|
||
}
|
||
envBind(env, symbolName(nameSym), island);
|
||
return island;
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sf-defisland"] = sfDefisland;
|
||
|
||
// defio-parse-kwargs!
|
||
var defioParseKwargs_b = function(spec, remaining) { return (isSxTruthy((isSxTruthy(!isSxTruthy(isEmpty(remaining))) && isSxTruthy((len(remaining) >= 2)) && keyword_p(first(remaining)))) ? (dictSet(spec, keywordName(first(remaining)), nth(remaining, 1)), defioParseKwargs_b(spec, rest(rest(remaining)))) : NIL); };
|
||
PRIMITIVES["defio-parse-kwargs!"] = defioParseKwargs_b;
|
||
|
||
// sf-defio
|
||
var sfDefio = function(args, env) { return (function() {
|
||
var name = first(args);
|
||
var spec = {};
|
||
spec["name"] = name;
|
||
defioParseKwargs_b(spec, rest(args));
|
||
ioRegister_b(name, spec);
|
||
return spec;
|
||
})(); };
|
||
PRIMITIVES["sf-defio"] = sfDefio;
|
||
|
||
// sf-defmacro
|
||
var sfDefmacro = function(args, env) { return (function() {
|
||
var nameSym = first(args);
|
||
var paramsRaw = nth(args, 1);
|
||
var body = nth(args, 2);
|
||
var parsed = parseMacroParams(paramsRaw);
|
||
var params = first(parsed);
|
||
var restParam = nth(parsed, 1);
|
||
return (function() {
|
||
var mac = makeMacro(params, restParam, body, env, symbolName(nameSym));
|
||
envBind(env, symbolName(nameSym), mac);
|
||
return mac;
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sf-defmacro"] = sfDefmacro;
|
||
|
||
// parse-macro-params
|
||
var parseMacroParams = function(paramsExpr) { return (function() {
|
||
var params = [];
|
||
var restParam = NIL;
|
||
reduce(function(state, p) { return (isSxTruthy((isSxTruthy(sxEq(typeOf(p), "symbol")) && sxEq(symbolName(p), "&rest"))) ? assoc(state, "in-rest", true) : (isSxTruthy(get(state, "in-rest")) ? ((restParam = (isSxTruthy(sxEq(typeOf(p), "symbol")) ? symbolName(p) : p)), state) : (append_b(params, (isSxTruthy(sxEq(typeOf(p), "symbol")) ? symbolName(p) : p)), state))); }, {["in-rest"]: false}, paramsExpr);
|
||
return [params, restParam];
|
||
})(); };
|
||
PRIMITIVES["parse-macro-params"] = parseMacroParams;
|
||
|
||
// qq-expand
|
||
var qqExpand = function(template, env) { return (isSxTruthy(!isSxTruthy(sxEq(typeOf(template), "list"))) ? template : (isSxTruthy(isEmpty(template)) ? [] : (function() {
|
||
var head = first(template);
|
||
return (isSxTruthy((isSxTruthy(sxEq(typeOf(head), "symbol")) && sxEq(symbolName(head), "unquote"))) ? trampoline(evalExpr(nth(template, 1), env)) : reduce(function(result, item) { return (isSxTruthy((isSxTruthy(sxEq(typeOf(item), "list")) && isSxTruthy(sxEq(len(item), 2)) && isSxTruthy(sxEq(typeOf(first(item)), "symbol")) && sxEq(symbolName(first(item)), "splice-unquote"))) ? (function() {
|
||
var spliced = trampoline(evalExpr(nth(item, 1), env));
|
||
return (isSxTruthy(sxEq(typeOf(spliced), "list")) ? concat(result, spliced) : (isSxTruthy(isNil(spliced)) ? result : concat(result, [spliced])));
|
||
})() : concat(result, [qqExpand(item, env)])); }, [], template));
|
||
})())); };
|
||
PRIMITIVES["qq-expand"] = qqExpand;
|
||
|
||
// sf-letrec
|
||
var sfLetrec = function(args, env) { return (function() {
|
||
var bindings = first(args);
|
||
var body = rest(args);
|
||
var local = envExtend(env);
|
||
var names = [];
|
||
var valExprs = [];
|
||
(isSxTruthy((isSxTruthy(sxEq(typeOf(first(bindings)), "list")) && sxEq(len(first(bindings)), 2))) ? forEach(function(binding) { return (function() {
|
||
var vname = (isSxTruthy(sxEq(typeOf(first(binding)), "symbol")) ? symbolName(first(binding)) : first(binding));
|
||
names.push(vname);
|
||
valExprs.push(nth(binding, 1));
|
||
return envBind(local, vname, NIL);
|
||
})(); }, bindings) : reduce(function(acc, pairIdx) { return (function() {
|
||
var vname = (isSxTruthy(sxEq(typeOf(nth(bindings, (pairIdx * 2))), "symbol")) ? symbolName(nth(bindings, (pairIdx * 2))) : nth(bindings, (pairIdx * 2)));
|
||
var valExpr = nth(bindings, ((pairIdx * 2) + 1));
|
||
names.push(vname);
|
||
valExprs.push(valExpr);
|
||
return envBind(local, vname, NIL);
|
||
})(); }, NIL, range(0, (len(bindings) / 2))));
|
||
(function() {
|
||
var values = map(function(e) { return trampoline(evalExpr(e, local)); }, valExprs);
|
||
{ var _c = zip(names, values); for (var _i = 0; _i < _c.length; _i++) { var pair = _c[_i]; envBind(local, first(pair), nth(pair, 1)); } }
|
||
return forEach(function(val) { return (isSxTruthy(isLambda(val)) ? forEach(function(n) { return envBind(lambdaClosure(val), n, envGet(local, n)); }, names) : NIL); }, values);
|
||
})();
|
||
{ var _c = slice(body, 0, (len(body) - 1)); for (var _i = 0; _i < _c.length; _i++) { var e = _c[_i]; trampoline(evalExpr(e, local)); } }
|
||
return makeThunk(last(body), local);
|
||
})(); };
|
||
PRIMITIVES["sf-letrec"] = sfLetrec;
|
||
|
||
// call-with-values
|
||
var callWithValues = function(producer, consumer) { return (function() {
|
||
var result = apply(producer, []);
|
||
return (isSxTruthy((isSxTruthy(isDict(result)) && get(result, "_values", false))) ? apply(consumer, get(result, "_list")) : apply(consumer, [result]));
|
||
})(); };
|
||
PRIMITIVES["call-with-values"] = callWithValues;
|
||
|
||
// sf-let-values
|
||
var sfLetValues = function(args, env) { return (function() {
|
||
var clauses = first(args);
|
||
var body = rest(args);
|
||
var local = envExtend(env);
|
||
{ var _c = clauses; for (var _i = 0; _i < _c.length; _i++) { var clause = _c[_i]; (function() {
|
||
var names = first(clause);
|
||
var valExpr = nth(clause, 1);
|
||
return (function() {
|
||
var result = trampoline(evalExpr(valExpr, local));
|
||
return (function() {
|
||
var vs = (isSxTruthy((isSxTruthy(isDict(result)) && get(result, "_values", false))) ? get(result, "_list") : [result]);
|
||
return forEachIndexed(function(idx, name) { return envBind(local, symbolName(name), nth(vs, idx)); }, names);
|
||
})();
|
||
})();
|
||
})(); } }
|
||
return (function() {
|
||
var lastVal = NIL;
|
||
{ var _c = body; for (var _i = 0; _i < _c.length; _i++) { var e = _c[_i]; lastVal = trampoline(evalExpr(e, local)); } }
|
||
return lastVal;
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sf-let-values"] = sfLetValues;
|
||
|
||
// sf-define-values
|
||
var sfDefineValues = function(args, env) { return (function() {
|
||
var names = first(args);
|
||
var valExpr = nth(args, 1);
|
||
return (function() {
|
||
var result = trampoline(evalExpr(valExpr, env));
|
||
return (function() {
|
||
var vs = (isSxTruthy((isSxTruthy(isDict(result)) && get(result, "_values", false))) ? get(result, "_list") : [result]);
|
||
forEachIndexed(function(idx, name) { return envBind(env, symbolName(name), nth(vs, idx)); }, names);
|
||
return NIL;
|
||
})();
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sf-define-values"] = sfDefineValues;
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("define-values", sfDefineValues);
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("let-values", sfLetValues);
|
||
|
||
// step-sf-letrec
|
||
var stepSfLetrec = function(args, env, kont) { return (function() {
|
||
var thk = sfLetrec(args, env);
|
||
return makeCekState(thunkExpr(thk), thunkEnv(thk), kont);
|
||
})(); };
|
||
PRIMITIVES["step-sf-letrec"] = stepSfLetrec;
|
||
|
||
// step-sf-dynamic-wind
|
||
var stepSfDynamicWind = function(args, env, kont) { return (function() {
|
||
var before = trampoline(evalExpr(first(args), env));
|
||
var body = trampoline(evalExpr(nth(args, 1), env));
|
||
var after = trampoline(evalExpr(nth(args, 2), env));
|
||
return (cekCall(before, []), (function() {
|
||
var windersLen = len(_winders_);
|
||
_winders_ = cons(after, _winders_);
|
||
return continueWithCall(body, [], env, [], kontPush(makeWindAfterFrame(after, windersLen, env), kont));
|
||
})());
|
||
})(); };
|
||
PRIMITIVES["step-sf-dynamic-wind"] = stepSfDynamicWind;
|
||
|
||
// sf-scope
|
||
var sfScope = function(args, env) { return (function() {
|
||
var name = trampoline(evalExpr(first(args), env));
|
||
var rest = slice(args, 1);
|
||
var val = NIL;
|
||
var bodyExprs = NIL;
|
||
(isSxTruthy((isSxTruthy((len(rest) >= 2)) && isSxTruthy(sxEq(typeOf(first(rest)), "keyword")) && sxEq(keywordName(first(rest)), "value"))) ? ((val = trampoline(evalExpr(nth(rest, 1), env))), (bodyExprs = slice(rest, 2))) : (bodyExprs = rest));
|
||
scopePush(name, val);
|
||
return (function() {
|
||
var result = NIL;
|
||
{ var _c = bodyExprs; for (var _i = 0; _i < _c.length; _i++) { var e = _c[_i]; result = trampoline(evalExpr(e, env)); } }
|
||
scopePop(name);
|
||
return result;
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sf-scope"] = sfScope;
|
||
|
||
// sf-provide
|
||
var sfProvide = function(args, env) { return (function() {
|
||
var name = trampoline(evalExpr(first(args), env));
|
||
var val = trampoline(evalExpr(nth(args, 1), env));
|
||
var bodyExprs = slice(args, 2);
|
||
var result = NIL;
|
||
scopePush(name, val);
|
||
{ var _c = bodyExprs; for (var _i = 0; _i < _c.length; _i++) { var e = _c[_i]; result = trampoline(evalExpr(e, env)); } }
|
||
scopePop(name);
|
||
return result;
|
||
})(); };
|
||
PRIMITIVES["sf-provide"] = sfProvide;
|
||
|
||
// expand-macro
|
||
var expandMacro = function(mac, rawArgs, env) { return (function() {
|
||
var body = macroBody(mac);
|
||
return (isSxTruthy((isSxTruthy(symbol_p(body)) && sxEq(symbolName(body), "__syntax-rules-body__"))) ? (function() {
|
||
var closure = macroClosure(mac);
|
||
return syntaxRulesExpand(envGet(closure, "__sr-literals"), envGet(closure, "__sr-rules"), rawArgs);
|
||
})() : (function() {
|
||
var local = envMerge(macroClosure(mac), env);
|
||
{ var _c = mapIndexed(function(i, p) { return [p, i]; }, macroParams(mac)); for (var _i = 0; _i < _c.length; _i++) { var pair = _c[_i]; envBind(local, first(pair), (isSxTruthy((nth(pair, 1) < len(rawArgs))) ? nth(rawArgs, nth(pair, 1)) : NIL)); } }
|
||
if (isSxTruthy(macroRestParam(mac))) {
|
||
envBind(local, macroRestParam(mac), slice(rawArgs, len(macroParams(mac))));
|
||
}
|
||
return trampoline(evalExpr(macroBody(mac), local));
|
||
})());
|
||
})(); };
|
||
PRIMITIVES["expand-macro"] = expandMacro;
|
||
|
||
// cek-step-loop
|
||
var cekStepLoop = function(state) { return (isSxTruthy(sxOr(cekTerminal_p(state), cekSuspended_p(state))) ? state : cekStepLoop(cekStep(state))); };
|
||
PRIMITIVES["cek-step-loop"] = cekStepLoop;
|
||
|
||
// cek-run
|
||
var cekRun = function(state) { return (function() {
|
||
var final_ = cekStepLoop(state);
|
||
return (isSxTruthy(cekSuspended_p(final_)) ? error("IO suspension in non-IO context") : cekValue(final_));
|
||
})(); };
|
||
PRIMITIVES["cek-run"] = cekRun;
|
||
|
||
// cek-resume
|
||
var cekResume = function(suspendedState, result) { return cekStepLoop(makeCekValue(result, cekEnv(suspendedState), cekKont(suspendedState))); };
|
||
PRIMITIVES["cek-resume"] = cekResume;
|
||
|
||
// cek-step
|
||
var cekStep = function(state) { return (isSxTruthy(sxEq(cekPhase(state), "eval")) ? stepEval(state) : stepContinue(state)); };
|
||
PRIMITIVES["cek-step"] = cekStep;
|
||
|
||
// step-eval
|
||
var stepEval = function(state) { return (function() {
|
||
var expr = cekControl(state);
|
||
var env = cekEnv(state);
|
||
var kont = cekKont(state);
|
||
return (function() { var _m = typeOf(expr); if (_m == "number") return makeCekValue(expr, env, kont); if (_m == "string") return makeCekValue(expr, env, kont); if (_m == "boolean") return makeCekValue(expr, env, kont); if (_m == "nil") return makeCekValue(NIL, env, kont); if (_m == "symbol") return (function() {
|
||
var name = symbolName(expr);
|
||
return (function() {
|
||
var val = (isSxTruthy(envHas(env, name)) ? envGet(env, name) : (isSxTruthy(isPrimitive(name)) ? getPrimitive(name) : (isSxTruthy(sxEq(name, "true")) ? true : (isSxTruthy(sxEq(name, "false")) ? false : (isSxTruthy(sxEq(name, "nil")) ? NIL : error((String("Undefined symbol: ") + String(name))))))));
|
||
if (isSxTruthy((isSxTruthy(isNil(val)) && startsWith(name, "~")))) {
|
||
debugLog("Component not found:", name);
|
||
}
|
||
return makeCekValue(val, env, kont);
|
||
})();
|
||
})(); if (_m == "keyword") return makeCekValue(keywordName(expr), env, kont); if (_m == "dict") return (function() {
|
||
var ks = keys(expr);
|
||
return (isSxTruthy(isEmpty(ks)) ? makeCekValue({}, env, kont) : (function() {
|
||
var firstKey = first(ks);
|
||
var remainingEntries = [];
|
||
{ var _c = rest(ks); for (var _i = 0; _i < _c.length; _i++) { var k = _c[_i]; remainingEntries.push([k, get(expr, k)]); } }
|
||
return makeCekState(get(expr, firstKey), env, kontPush(makeDictFrame(remainingEntries, [[firstKey]], env), kont));
|
||
})());
|
||
})(); if (_m == "list") return (isSxTruthy(isEmpty(expr)) ? makeCekValue([], env, kont) : stepEvalList(expr, env, kont)); return makeCekValue(expr, env, kont); })();
|
||
})(); };
|
||
PRIMITIVES["step-eval"] = stepEval;
|
||
|
||
// step-sf-raise
|
||
var stepSfRaise = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeRaiseEvalFrame(env, false), kont)); };
|
||
PRIMITIVES["step-sf-raise"] = stepSfRaise;
|
||
|
||
// step-sf-guard
|
||
var stepSfGuard = function(args, env, kont) { return (function() {
|
||
var varClauses = first(args);
|
||
var body = rest(args);
|
||
var var_ = first(varClauses);
|
||
var clauses = rest(varClauses);
|
||
var sentinel = makeSymbol("__guard-reraise__");
|
||
return stepEvalList([new Symbol("let"), [[new Symbol("__guard-result"), cons(new Symbol("call/cc"), [cons(new Symbol("fn"), cons([new Symbol("__guard-k")], [cons(new Symbol("handler-bind"), cons([[cons(new Symbol("fn"), cons([new Symbol("_")], [true])), cons(new Symbol("fn"), cons([var_], [[new Symbol("__guard-k"), cons(new Symbol("cond"), append(clauses, [[new Symbol("else"), [new Symbol("list"), [new Symbol("quote"), sentinel], var_]]]))]]))]], [[new Symbol("__guard-k"), cons(new Symbol("begin"), body)]]))]))])]], [new Symbol("if"), [new Symbol("and"), [new Symbol("list?"), new Symbol("__guard-result")], [new Symbol("="), [new Symbol("len"), new Symbol("__guard-result")], 2], [new Symbol("="), [new Symbol("first"), new Symbol("__guard-result")], [new Symbol("quote"), sentinel]]], [new Symbol("raise"), [new Symbol("nth"), new Symbol("__guard-result"), 1]], new Symbol("__guard-result")]], env, kont);
|
||
})(); };
|
||
PRIMITIVES["step-sf-guard"] = stepSfGuard;
|
||
|
||
// step-sf-callcc
|
||
var stepSfCallcc = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeCallccFrame(env), kont)); };
|
||
PRIMITIVES["step-sf-callcc"] = stepSfCallcc;
|
||
|
||
// step-sf-case
|
||
var stepSfCase = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeCaseFrame(NIL, rest(args), env), kont)); };
|
||
PRIMITIVES["step-sf-case"] = stepSfCase;
|
||
|
||
// step-sf-let-match
|
||
var stepSfLetMatch = function(args, env, kont) { return (function() {
|
||
var pattern = first(args);
|
||
var expr = nth(args, 1);
|
||
var body = rest(rest(args));
|
||
return stepSfMatch([expr, [pattern, cons(new Symbol("begin"), body)]], env, kont);
|
||
})(); };
|
||
PRIMITIVES["step-sf-let-match"] = stepSfLetMatch;
|
||
|
||
// step-eval-list
|
||
var stepEvalList = function(expr, env, kont) { return (function() {
|
||
var head = first(expr);
|
||
var args = rest(expr);
|
||
return (isSxTruthy(!isSxTruthy(sxOr(sxEq(typeOf(head), "symbol"), sxEq(typeOf(head), "lambda"), sxEq(typeOf(head), "list")))) ? (isSxTruthy(isEmpty(expr)) ? makeCekValue([], env, kont) : makeCekState(first(expr), env, kontPush(makeMapFrame(NIL, rest(expr), [], env), kont))) : (isSxTruthy(sxEq(typeOf(head), "symbol")) ? (function() {
|
||
var name = symbolName(head);
|
||
return (function() { var _m = name; if (_m == "if") return stepSfIf(args, env, kont); if (_m == "when") return stepSfWhen(args, env, kont); if (_m == "cond") return stepSfCond(args, env, kont); if (_m == "case") return stepSfCase(args, env, kont); if (_m == "and") return stepSfAnd(args, env, kont); if (_m == "or") return stepSfOr(args, env, kont); if (_m == "let") return stepSfLet(args, env, kont); if (_m == "let*") return stepSfLet(args, env, kont); if (_m == "lambda") return stepSfLambda(args, env, kont); if (_m == "fn") return stepSfLambda(args, env, kont); if (_m == "define") return stepSfDefine(args, env, kont); if (_m == "defcomp") return makeCekValue(sfDefcomp(args, env), env, kont); if (_m == "defisland") return makeCekValue(sfDefisland(args, env), env, kont); if (_m == "defmacro") return makeCekValue(sfDefmacro(args, env), env, kont); if (_m == "defio") return makeCekValue(sfDefio(args, env), env, kont); if (_m == "define-foreign") return stepSfDefineForeign(args, env, kont); if (_m == "io") return stepSfIo(args, env, kont); if (_m == "begin") return stepSfBegin(args, env, kont); if (_m == "do") return (isSxTruthy((isSxTruthy(!isSxTruthy(isEmpty(args))) && isSxTruthy(isList(first(args))) && isSxTruthy(!isSxTruthy(isEmpty(first(args)))) && isList(first(first(args))))) ? (function() {
|
||
var bindings = first(args);
|
||
var testClause = nth(args, 1);
|
||
var body = rest(rest(args));
|
||
var vars = map(function(b) { return first(b); }, bindings);
|
||
var inits = map(function(b) { return nth(b, 1); }, bindings);
|
||
var steps = map(function(b) { return (isSxTruthy((len(b) > 2)) ? nth(b, 2) : first(b)); }, bindings);
|
||
var test = first(testClause);
|
||
var result = rest(testClause);
|
||
return stepEvalList(cons(new Symbol("let"), cons(new Symbol("__do-loop"), cons(map(function(b) { return [first(b), nth(b, 1)]; }, bindings), [cons(new Symbol("if"), cons(test, cons((isSxTruthy(isEmpty(result)) ? NIL : cons(new Symbol("begin"), result)), [cons(new Symbol("begin"), append(body, [cons(new Symbol("__do-loop"), steps)]))])))]))), env, kont);
|
||
})() : stepSfBegin(args, env, kont)); if (_m == "guard") return stepSfGuard(args, env, kont); if (_m == "quote") return makeCekValue((isSxTruthy(isEmpty(args)) ? NIL : first(args)), env, kont); if (_m == "quasiquote") return makeCekValue(qqExpand(first(args), env), env, kont); if (_m == "->") return stepSfThreadFirst(args, env, kont); if (_m == "->>") return stepSfThreadLast(args, env, kont); if (_m == "|>") return stepSfThreadLast(args, env, kont); if (_m == "as->") return stepSfThreadAs(args, env, kont); if (_m == "set!") return stepSfSet(args, env, kont); if (_m == "letrec") return stepSfLetrec(args, env, kont); if (_m == "reset") return stepSfReset(args, env, kont); if (_m == "shift") return stepSfShift(args, env, kont); if (_m == "deref") return stepSfDeref(args, env, kont); if (_m == "scope") return stepSfScope(args, env, kont); if (_m == "provide") return stepSfProvide(args, env, kont); if (_m == "peek") return stepSfPeek(args, env, kont); if (_m == "provide!") return stepSfProvide_b(args, env, kont); if (_m == "context") return stepSfContext(args, env, kont); if (_m == "bind") return stepSfBind(args, env, kont); if (_m == "emit!") return stepSfEmit(args, env, kont); if (_m == "emitted") return stepSfEmitted(args, env, kont); if (_m == "handler-bind") return stepSfHandlerBind(args, env, kont); if (_m == "restart-case") return stepSfRestartCase(args, env, kont); if (_m == "signal-condition") return stepSfSignal(args, env, kont); if (_m == "invoke-restart") return stepSfInvokeRestart(args, env, kont); if (_m == "match") return stepSfMatch(args, env, kont); if (_m == "let-match") return stepSfLetMatch(args, env, kont); if (_m == "dynamic-wind") return stepSfDynamicWind(args, env, kont); if (_m == "map") return stepHoMap(args, env, kont); if (_m == "map-indexed") return stepHoMapIndexed(args, env, kont); if (_m == "filter") return stepHoFilter(args, env, kont); if (_m == "reduce") return stepHoReduce(args, env, kont); if (_m == "some") return stepHoSome(args, env, kont); if (_m == "every?") return stepHoEvery(args, env, kont); if (_m == "for-each") return stepHoForEach(args, env, kont); if (_m == "raise") return stepSfRaise(args, env, kont); if (_m == "raise-continuable") return makeCekState(first(args), env, kontPush(makeRaiseEvalFrame(env, true), kont)); if (_m == "call/cc") return stepSfCallcc(args, env, kont); if (_m == "call-with-current-continuation") return stepSfCallcc(args, env, kont); if (_m == "perform") return stepSfPerform(args, env, kont); if (_m == "define-library") return stepSfDefineLibrary(args, env, kont); if (_m == "import") return stepSfImport(args, env, kont); if (_m == "define-record-type") return makeCekValue(sfDefineRecordType(args, env), env, kont); if (_m == "define-protocol") return makeCekValue(sfDefineProtocol(args, env), env, kont); if (_m == "implement") return makeCekValue(sfImplement(args, env), env, kont); if (_m == "parameterize") return stepSfParameterize(args, env, kont); if (_m == "syntax-rules") return makeCekValue(sfSyntaxRules(args, env), env, kont); if (_m == "define-syntax") return stepSfDefine(args, env, kont); return (isSxTruthy((isSxTruthy(dictHas(_customSpecialForms, name)) && !isSxTruthy(envHas(env, name)))) ? makeCekValue((get(_customSpecialForms, name))(args, env), env, kont) : (isSxTruthy((isSxTruthy(envHas(env, name)) && isMacro(envGet(env, name)))) ? (function() {
|
||
var mac = envGet(env, name);
|
||
return makeCekState(expandMacro(mac, args, env), env, kont);
|
||
})() : (isSxTruthy((isSxTruthy(_renderCheck) && isSxTruthy(!isSxTruthy(envHas(env, name))) && _renderCheck(expr, env))) ? makeCekValue(_renderFn(expr, env), env, kont) : stepEvalCall(head, args, env, kont)))); })();
|
||
})() : stepEvalCall(head, args, env, kont)));
|
||
})(); };
|
||
PRIMITIVES["step-eval-list"] = stepEvalList;
|
||
|
||
// sf-define-type
|
||
var sfDefineType = function(args, env) { return (function() {
|
||
var typeSym = first(args);
|
||
var ctorSpecs = rest(args);
|
||
return (function() {
|
||
var typeName = symbolName(typeSym);
|
||
var ctorNames = map(function(spec) { return symbolName(first(spec)); }, ctorSpecs);
|
||
if (isSxTruthy(!isSxTruthy(envHas(env, "*adt-registry*")))) {
|
||
envBind(env, "*adt-registry*", {});
|
||
}
|
||
envGet(env, "*adt-registry*")[typeName] = ctorNames;
|
||
envBind(env, (String(typeName) + String("?")), function(v) { return (isSxTruthy(isDict(v)) && isSxTruthy(get(v, "_adt")) && sxEq(get(v, "_type"), typeName)); });
|
||
{ var _c = ctorSpecs; for (var _i = 0; _i < _c.length; _i++) { var spec = _c[_i]; (function() {
|
||
var cn = symbolName(first(spec));
|
||
var fieldNames = map(function(f) { return symbolName(f); }, rest(spec));
|
||
var arity = len(rest(spec));
|
||
envBind(env, cn, function() { var ctorArgs = Array.prototype.slice.call(arguments, 0); return (isSxTruthy(!isSxTruthy(sxEq(len(ctorArgs), arity))) ? error((String(cn) + String(": expected ") + String(arity) + String(" args, got ") + String(len(ctorArgs)))) : {"_ctor": cn, "_type": typeName, "_adt": true, "_fields": ctorArgs}); });
|
||
envBind(env, (String(cn) + String("?")), function(v) { return (isSxTruthy(isDict(v)) && isSxTruthy(get(v, "_adt")) && sxEq(get(v, "_ctor"), cn)); });
|
||
return forEachIndexed(function(idx, fieldName) { return envBind(env, (String(cn) + String("-") + String(fieldName)), function(v) { return nth(get(v, "_fields"), idx); }); }, fieldNames);
|
||
})(); } }
|
||
return NIL;
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sf-define-type"] = sfDefineType;
|
||
|
||
// sf-delay
|
||
var sfDelay = function(args, env) { return (function() {
|
||
var thunk = makeLambda([], first(args), env);
|
||
return {"forced": false, "value": NIL, "thunk": thunk, "_promise": true};
|
||
})(); };
|
||
PRIMITIVES["sf-delay"] = sfDelay;
|
||
|
||
// sf-delay-force
|
||
var sfDelayForce = function(args, env) { return (function() {
|
||
var thunk = makeLambda([], first(args), env);
|
||
return {"_iterative": true, "forced": false, "value": NIL, "thunk": thunk, "_promise": true};
|
||
})(); };
|
||
PRIMITIVES["sf-delay-force"] = sfDelayForce;
|
||
|
||
// promise?
|
||
var promise_p = function(v) { return (isSxTruthy(isDict(v)) && get(v, "_promise", false)); };
|
||
PRIMITIVES["promise?"] = promise_p;
|
||
|
||
// make-promise
|
||
var makePromise = function(v) { return {"forced": true, "value": v, "_promise": true}; };
|
||
PRIMITIVES["make-promise"] = makePromise;
|
||
|
||
// force
|
||
var force = function(p) { return (isSxTruthy(!isSxTruthy(promise_p(p))) ? p : (isSxTruthy(get(p, "forced", false)) ? get(p, "value", NIL) : (function() {
|
||
var result = apply(get(p, "thunk", NIL), []);
|
||
return (function() {
|
||
var final_ = (isSxTruthy((isSxTruthy(get(p, "_iterative", false)) && promise_p(result))) ? force(result) : result);
|
||
p["forced"] = true;
|
||
p["value"] = final_;
|
||
return final_;
|
||
})();
|
||
})())); };
|
||
PRIMITIVES["force"] = force;
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("delay", sfDelay);
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("delay-force", sfDelayForce);
|
||
|
||
// values
|
||
var values = function() { var vs = Array.prototype.slice.call(arguments, 0); return (isSxTruthy(sxEq(len(vs), 1)) ? first(vs) : {"_values": true, "_list": vs}); };
|
||
PRIMITIVES["values"] = values;
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("define-type", sfDefineType);
|
||
|
||
// kont-extract-provides
|
||
var kontExtractProvides = function(kont) { return (isSxTruthy(isEmpty(kont)) ? [] : (function() {
|
||
var frame = first(kont);
|
||
var restFrames = kontExtractProvides(rest(kont));
|
||
return (isSxTruthy(sxEq(frameType(frame), "provide")) ? cons({"subscribers": [], "env": get(frame, "env"), "value": get(frame, "value"), "type": "provide", "remaining": [], "name": get(frame, "name")}, restFrames) : restFrames);
|
||
})()); };
|
||
PRIMITIVES["kont-extract-provides"] = kontExtractProvides;
|
||
|
||
// fire-provide-subscribers
|
||
var fireProvideSubscribers = function(frame, kont) { return (function() {
|
||
var subs = get(frame, "subscribers");
|
||
return (isSxTruthy(!isSxTruthy(isEmpty(subs))) ? (isSxTruthy((_provideBatchDepth_ > 0)) ? forEach(function(sub) { return (isSxTruthy(!isSxTruthy(contains(_provideBatchQueue_, sub))) ? append_b(_provideBatchQueue_, sub) : NIL); }, subs) : forEach(function(sub) { return cekCall(sub, [kont]); }, subs)) : NIL);
|
||
})(); };
|
||
PRIMITIVES["fire-provide-subscribers"] = fireProvideSubscribers;
|
||
|
||
// fire-provide-subscribers
|
||
var fireProvideSubscribers = function(name) { return (function() {
|
||
var subs = get(_provideSubscribers_, name);
|
||
return (isSxTruthy((isSxTruthy(subs) && !isSxTruthy(isEmpty(subs)))) ? (isSxTruthy((_provideBatchDepth_ > 0)) ? forEach(function(sub) { return (isSxTruthy(!isSxTruthy(contains(_provideBatchQueue_, sub))) ? append_b(_provideBatchQueue_, sub) : NIL); }, subs) : forEach(function(sub) { return cekCall(sub, [NIL]); }, subs)) : NIL);
|
||
})(); };
|
||
PRIMITIVES["fire-provide-subscribers"] = fireProvideSubscribers;
|
||
|
||
// batch-begin!
|
||
var batchBegin_b = function() { return (_provideBatchDepth_ = (_provideBatchDepth_ + 1)); };
|
||
PRIMITIVES["batch-begin!"] = batchBegin_b;
|
||
|
||
// batch-end!
|
||
var batchEnd_b = function() { _provideBatchDepth_ = (_provideBatchDepth_ - 1);
|
||
return (isSxTruthy(sxEq(_provideBatchDepth_, 0)) ? (function() {
|
||
var queue = _provideBatchQueue_;
|
||
_provideBatchQueue_ = [];
|
||
return forEach(function(sub) { return cekCall(sub, [NIL]); }, queue);
|
||
})() : NIL); };
|
||
PRIMITIVES["batch-end!"] = batchEnd_b;
|
||
|
||
// step-sf-bind
|
||
var stepSfBind = function(args, env, kont) { return (function() {
|
||
var body = first(args);
|
||
var prev = _bindTracking_;
|
||
_bindTracking_ = [];
|
||
return makeCekState(body, env, kontPush(makeBindFrame(body, env, prev), kont));
|
||
})(); };
|
||
PRIMITIVES["step-sf-bind"] = stepSfBind;
|
||
|
||
// step-sf-parameterize
|
||
var stepSfParameterize = function(args, env, kont) { return (function() {
|
||
var bindings = first(args);
|
||
var body = rest(args);
|
||
return (isSxTruthy(sxOr(isNil(bindings), isEmpty(bindings))) ? stepSfBegin(body, env, kont) : (function() {
|
||
var firstPair = first(bindings);
|
||
return makeCekState(first(firstPair), env, kontPush(makeParameterizeFrame(bindings, NIL, [], body, env), kont));
|
||
})());
|
||
})(); };
|
||
PRIMITIVES["step-sf-parameterize"] = stepSfParameterize;
|
||
|
||
// syntax-rules-match
|
||
var syntaxRulesMatch = function(pattern, form, literals) { return (isSxTruthy((isSxTruthy(symbol_p(pattern)) && sxEq(symbolName(pattern), "_"))) ? {} : (isSxTruthy((isSxTruthy(symbol_p(pattern)) && contains(literals, symbolName(pattern)))) ? (isSxTruthy((isSxTruthy(symbol_p(form)) && sxEq(symbolName(pattern), symbolName(form)))) ? {} : NIL) : (isSxTruthy(symbol_p(pattern)) ? (function() {
|
||
var d = {};
|
||
d[symbolName(pattern)] = form;
|
||
return d;
|
||
})() : (isSxTruthy((isSxTruthy(isList(pattern)) && isEmpty(pattern))) ? (isSxTruthy((isSxTruthy(isList(form)) && isEmpty(form))) ? {} : NIL) : (isSxTruthy((isSxTruthy(isList(pattern)) && isList(form))) ? syntaxRulesMatchList(pattern, 0, form, 0, literals) : (isSxTruthy(sxEq(pattern, form)) ? {} : NIL)))))); };
|
||
PRIMITIVES["syntax-rules-match"] = syntaxRulesMatch;
|
||
|
||
// syntax-rules-match-list
|
||
var syntaxRulesMatchList = function(pattern, pi, form, fi, literals) { return (function() {
|
||
var plen = len(pattern);
|
||
var flen = len(form);
|
||
return (isSxTruthy((isSxTruthy((pi >= plen)) && (fi >= flen))) ? {} : (isSxTruthy((pi >= plen)) ? NIL : (isSxTruthy((isSxTruthy(((pi + 1) < plen)) && isSxTruthy(symbol_p(nth(pattern, (pi + 1)))) && sxEq(symbolName(nth(pattern, (pi + 1))), "..."))) ? (function() {
|
||
var subPat = nth(pattern, pi);
|
||
var restPatCount = (plen - (pi + 2));
|
||
var available = (flen - fi);
|
||
var nEllipsis = ((flen - fi) - (plen - (pi + 2)));
|
||
return (isSxTruthy((nEllipsis < 0)) ? NIL : (function() {
|
||
var ellipsisForms = slice(form, fi, (fi + nEllipsis));
|
||
var subBindings = map(function(f) { return syntaxRulesMatch(subPat, f, literals); }, slice(form, fi, (fi + nEllipsis)));
|
||
return (isSxTruthy(contains(subBindings, NIL)) ? NIL : (function() {
|
||
var restResult = syntaxRulesMatchList(pattern, (pi + 2), form, (fi + nEllipsis), literals);
|
||
return (isSxTruthy(isNil(restResult)) ? NIL : (function() {
|
||
var merged = {};
|
||
{ var _c = subBindings; for (var _i = 0; _i < _c.length; _i++) { var b = _c[_i]; { var _c = keys(b); for (var _i = 0; _i < _c.length; _i++) { var key = _c[_i]; (function() {
|
||
var existing = dictGet(merged, key);
|
||
return (isSxTruthy(isNil(existing)) ? dictSet(merged, key, [get(b, key)]) : dictSet(merged, key, append(existing, [get(b, key)])));
|
||
})(); } } } }
|
||
{ var _c = keys(restResult); for (var _i = 0; _i < _c.length; _i++) { var key = _c[_i]; merged[key] = get(restResult, key); } }
|
||
return merged;
|
||
})());
|
||
})());
|
||
})());
|
||
})() : (isSxTruthy((fi >= flen)) ? NIL : (function() {
|
||
var subResult = syntaxRulesMatch(nth(pattern, pi), nth(form, fi), literals);
|
||
return (isSxTruthy(isNil(subResult)) ? NIL : (function() {
|
||
var restResult = syntaxRulesMatchList(pattern, (pi + 1), form, (fi + 1), literals);
|
||
return (isSxTruthy(isNil(restResult)) ? NIL : (forEach(function(key) { return dictSet(restResult, key, get(subResult, key)); }, keys(subResult)), restResult));
|
||
})());
|
||
})()))));
|
||
})(); };
|
||
PRIMITIVES["syntax-rules-match-list"] = syntaxRulesMatchList;
|
||
|
||
// syntax-rules-find-var
|
||
var syntaxRulesFindVar = function(template, bindings) { return (isSxTruthy((isSxTruthy(symbol_p(template)) && isSxTruthy(dictHas(bindings, symbolName(template))) && isList(get(bindings, symbolName(template))))) ? symbolName(template) : (isSxTruthy(isList(template)) ? reduce(function(found, t) { return (isSxTruthy(isNil(found)) ? syntaxRulesFindVar(t, bindings) : found); }, NIL, template) : NIL)); };
|
||
PRIMITIVES["syntax-rules-find-var"] = syntaxRulesFindVar;
|
||
|
||
// syntax-rules-find-all-vars
|
||
var syntaxRulesFindAllVars = function(template, bindings) { return (isSxTruthy((isSxTruthy(symbol_p(template)) && isSxTruthy(dictHas(bindings, symbolName(template))) && isList(get(bindings, symbolName(template))))) ? [symbolName(template)] : (isSxTruthy(isList(template)) ? reduce(function(acc, t) { return append(acc, syntaxRulesFindAllVars(t, bindings)); }, [], template) : [])); };
|
||
PRIMITIVES["syntax-rules-find-all-vars"] = syntaxRulesFindAllVars;
|
||
|
||
// syntax-rules-instantiate
|
||
var syntaxRulesInstantiate = function(template, bindings) { return (isSxTruthy((isSxTruthy(symbol_p(template)) && dictHas(bindings, symbolName(template)))) ? get(bindings, symbolName(template)) : (isSxTruthy(!isSxTruthy(isList(template))) ? template : (isSxTruthy(isEmpty(template)) ? template : syntaxRulesInstantiateList(template, 0, bindings)))); };
|
||
PRIMITIVES["syntax-rules-instantiate"] = syntaxRulesInstantiate;
|
||
|
||
// syntax-rules-instantiate-list
|
||
var syntaxRulesInstantiateList = function(template, i, bindings) { return (isSxTruthy((i >= len(template))) ? [] : (function() {
|
||
var elem = nth(template, i);
|
||
var hasEllipsis = (isSxTruthy(((i + 1) < len(template))) && isSxTruthy(symbol_p(nth(template, (i + 1)))) && sxEq(symbolName(nth(template, (i + 1))), "..."));
|
||
return (isSxTruthy(hasEllipsis) ? (function() {
|
||
var allVars = syntaxRulesFindAllVars(elem, bindings);
|
||
return (isSxTruthy(isEmpty(allVars)) ? syntaxRulesInstantiateList(template, (i + 2), bindings) : (function() {
|
||
var count = len(get(bindings, first(allVars)));
|
||
var expanded = map(function(idx) { return (function() {
|
||
var b = {};
|
||
{ var _c = keys(bindings); for (var _i = 0; _i < _c.length; _i++) { var key = _c[_i]; b[key] = get(bindings, key); } }
|
||
{ var _c = allVars; for (var _i = 0; _i < _c.length; _i++) { var varName = _c[_i]; b[varName] = nth(get(bindings, varName), idx); } }
|
||
return syntaxRulesInstantiate(elem, b);
|
||
})(); }, range(count));
|
||
var restResult = syntaxRulesInstantiateList(template, (i + 2), bindings);
|
||
return append(expanded, restResult);
|
||
})());
|
||
})() : cons(syntaxRulesInstantiate(elem, bindings), syntaxRulesInstantiateList(template, (i + 1), bindings)));
|
||
})()); };
|
||
PRIMITIVES["syntax-rules-instantiate-list"] = syntaxRulesInstantiateList;
|
||
|
||
// syntax-rules-expand
|
||
var syntaxRulesExpand = function(literals, rules, form) { return (function() {
|
||
var fullForm = cons(makeSymbol("_"), form);
|
||
return syntaxRulesTryRules(literals, rules, fullForm);
|
||
})(); };
|
||
PRIMITIVES["syntax-rules-expand"] = syntaxRulesExpand;
|
||
|
||
// syntax-rules-try-rules
|
||
var syntaxRulesTryRules = function(literals, rules, fullForm) { return (isSxTruthy(isEmpty(rules)) ? error((String("syntax-rules: no pattern matched for ") + String(inspect(fullForm)))) : (function() {
|
||
var rule = first(rules);
|
||
var pattern = first(rule);
|
||
var template = nth(rule, 1);
|
||
return (function() {
|
||
var bindings = syntaxRulesMatch(pattern, fullForm, literals);
|
||
return (isSxTruthy(!isSxTruthy(isNil(bindings))) ? syntaxRulesInstantiate(template, bindings) : syntaxRulesTryRules(literals, rest(rules), fullForm));
|
||
})();
|
||
})()); };
|
||
PRIMITIVES["syntax-rules-try-rules"] = syntaxRulesTryRules;
|
||
|
||
// sf-syntax-rules
|
||
var sfSyntaxRules = function(args, env) { return (function() {
|
||
var literals = (isSxTruthy(isList(first(args))) ? map(function(s) { return (isSxTruthy(symbol_p(s)) ? symbolName(s) : (String(s))); }, first(args)) : []);
|
||
var rules = rest(args);
|
||
return (function() {
|
||
var closure = envExtend(env);
|
||
envBind(closure, "__sr-literals", literals);
|
||
envBind(closure, "__sr-rules", rules);
|
||
return makeMacro([], "__sr-form", new Symbol("__syntax-rules-body__"), closure, "syntax-rules");
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sf-syntax-rules"] = sfSyntaxRules;
|
||
|
||
// step-sf-define-library
|
||
var stepSfDefineLibrary = function(args, env, kont) { return (function() {
|
||
var libSpec = first(args);
|
||
var decls = rest(args);
|
||
return (function() {
|
||
var libEnv = envExtend(env);
|
||
var exports = [];
|
||
var bodyForms = [];
|
||
{ var _c = decls; for (var _i = 0; _i < _c.length; _i++) { var decl = _c[_i]; if (isSxTruthy((isSxTruthy(isList(decl)) && isSxTruthy(!isSxTruthy(isEmpty(decl))) && symbol_p(first(decl))))) {
|
||
(function() {
|
||
var kind = symbolName(first(decl));
|
||
return (isSxTruthy(sxEq(kind, "export")) ? (exports = append(exports, map(function(s) { return (isSxTruthy(symbol_p(s)) ? symbolName(s) : (String(s))); }, rest(decl)))) : (isSxTruthy(sxEq(kind, "import")) ? forEach(function(importSet) { return bindImportSet(importSet, libEnv); }, rest(decl)) : (isSxTruthy(sxEq(kind, "begin")) ? (bodyForms = append(bodyForms, rest(decl))) : NIL)));
|
||
})();
|
||
} } }
|
||
{ var _c = bodyForms; for (var _i = 0; _i < _c.length; _i++) { var form = _c[_i]; evalExpr(form, libEnv); } }
|
||
return (function() {
|
||
var exportDict = {};
|
||
{ var _c = exports; for (var _i = 0; _i < _c.length; _i++) { var name = _c[_i]; if (isSxTruthy(envHas(libEnv, name))) {
|
||
exportDict[name] = envGet(libEnv, name);
|
||
} } }
|
||
registerLibrary(libSpec, exportDict);
|
||
return makeCekValue(NIL, env, kont);
|
||
})();
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["step-sf-define-library"] = stepSfDefineLibrary;
|
||
|
||
// bind-import-set
|
||
var bindImportSet = function(importSet, env) { return (function() {
|
||
var head = (isSxTruthy((isSxTruthy(isList(importSet)) && isSxTruthy(!isSxTruthy(isEmpty(importSet))) && symbol_p(first(importSet)))) ? symbolName(first(importSet)) : NIL);
|
||
return (function() {
|
||
var libSpec = (isSxTruthy(sxOr(sxEq(head, "only"), sxEq(head, "except"), sxEq(head, "prefix"), sxEq(head, "rename"))) ? nth(importSet, 1) : importSet);
|
||
return (function() {
|
||
var exports = libraryExports(libSpec);
|
||
return (isSxTruthy(sxEq(head, "only")) ? forEach(function(s) { return (function() {
|
||
var id = (isSxTruthy(symbol_p(s)) ? symbolName(s) : (String(s)));
|
||
return (isSxTruthy(dictHas(exports, id)) ? envBind(env, id, get(exports, id)) : NIL);
|
||
})(); }, rest(rest(importSet))) : (isSxTruthy(sxEq(head, "prefix")) ? (function() {
|
||
var pfx = (String(nth(importSet, 2)));
|
||
return forEach(function(key) { return envBind(env, (String(pfx) + String(key)), get(exports, key)); }, keys(exports));
|
||
})() : forEach(function(key) { return envBind(env, key, get(exports, key)); }, keys(exports))));
|
||
})();
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["bind-import-set"] = bindImportSet;
|
||
|
||
// step-sf-import
|
||
var stepSfImport = function(args, env, kont) { return (isSxTruthy(isEmpty(args)) ? makeCekValue(NIL, env, kont) : (function() {
|
||
var importSet = first(args);
|
||
var restSets = rest(args);
|
||
return (function() {
|
||
var libSpec = (function() {
|
||
var head = (isSxTruthy((isSxTruthy(isList(importSet)) && isSxTruthy(!isSxTruthy(isEmpty(importSet))) && symbol_p(first(importSet)))) ? symbolName(first(importSet)) : NIL);
|
||
return (isSxTruthy(sxOr(sxEq(head, "only"), sxEq(head, "except"), sxEq(head, "prefix"), sxEq(head, "rename"))) ? nth(importSet, 1) : importSet);
|
||
})();
|
||
return (isSxTruthy(libraryLoaded_p(libSpec)) ? (bindImportSet(importSet, env), (isSxTruthy(isEmpty(restSets)) ? makeCekValue(NIL, env, kont) : stepSfImport(restSets, env, kont))) : makeCekSuspended({"library": libSpec, "op": "import"}, env, kontPush(makeImportFrame(importSet, restSets, env), kont)));
|
||
})();
|
||
})()); };
|
||
PRIMITIVES["step-sf-import"] = stepSfImport;
|
||
|
||
// step-sf-perform
|
||
var stepSfPerform = function(args, env, kont) { return (isSxTruthy(isEmpty(args)) ? error("perform requires an IO request argument") : makeCekState(first(args), env, kontPush(makePerformFrame(env), kont))); };
|
||
PRIMITIVES["step-sf-perform"] = stepSfPerform;
|
||
|
||
// *protocol-registry*
|
||
var _protocolRegistry_ = {};
|
||
PRIMITIVES["*protocol-registry*"] = _protocolRegistry_;
|
||
|
||
// sf-define-record-type
|
||
var sfDefineRecordType = function(args, env) { return (function() {
|
||
var typeSym = first(args);
|
||
var ctorSpec = nth(args, 1);
|
||
var predSym = nth(args, 2);
|
||
var fieldSpecs = slice(args, 3);
|
||
return (function() {
|
||
var rawName = symbolName(typeSym);
|
||
return (function() {
|
||
var typeName = (isSxTruthy((isSxTruthy(startsWith(rawName, "<")) && endsWith(rawName, ">"))) ? slice(rawName, 1, (len(rawName) - 1)) : rawName);
|
||
var ctorName = symbolName(first(ctorSpec));
|
||
var ctorParams = map(function(s) { return symbolName(s); }, rest(ctorSpec));
|
||
var predName = symbolName(predSym);
|
||
var fieldNames = map(function(fs) { return symbolName(first(fs)); }, fieldSpecs);
|
||
return (function() {
|
||
var rtdUid = makeRtd(typeName, fieldNames, ctorParams);
|
||
envBind(env, ctorName, makeRecordConstructor(rtdUid));
|
||
envBind(env, predName, makeRecordPredicate(rtdUid));
|
||
forEachIndexed(function(idx, fs) { return (function() {
|
||
var accessorName = symbolName(nth(fs, 1));
|
||
envBind(env, accessorName, makeRecordAccessor(idx));
|
||
return (isSxTruthy((len(fs) >= 3)) ? (function() {
|
||
var mutatorName = symbolName(nth(fs, 2));
|
||
return envBind(env, mutatorName, makeRecordMutator(idx));
|
||
})() : NIL);
|
||
})(); }, fieldSpecs);
|
||
return NIL;
|
||
})();
|
||
})();
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sf-define-record-type"] = sfDefineRecordType;
|
||
|
||
// sf-define-protocol
|
||
var sfDefineProtocol = function(args, env) { return (function() {
|
||
var protoName = symbolName(first(args));
|
||
var methodSpecs = rest(args);
|
||
envBind(env, "*protocol-registry*", _protocolRegistry_);
|
||
envBind(env, "satisfies?", function(pname, val) { return satisfies_p(pname, val); });
|
||
_protocolRegistry_[protoName] = {"impls": {}, "methods": map(function(spec) { return {"arity": len(spec), "name": symbolName(first(spec))}; }, methodSpecs), "name": protoName};
|
||
{ var _c = methodSpecs; for (var _i = 0; _i < _c.length; _i++) { var spec = _c[_i]; (function() {
|
||
var methodName = symbolName(first(spec));
|
||
var params = rest(spec);
|
||
var pname = protoName;
|
||
return (function() {
|
||
var selfSym = first(params);
|
||
var lookupExpr = [new Symbol("get"), [new Symbol("get"), [new Symbol("get"), [new Symbol("get"), new Symbol("*protocol-registry*"), pname], "impls"], [new Symbol("type-of"), selfSym]], methodName];
|
||
return envBind(env, methodName, evalExpr([new Symbol("fn"), params, [new Symbol("let"), [[new Symbol("_impl"), lookupExpr]], [new Symbol("if"), [new Symbol("nil?"), new Symbol("_impl")], [new Symbol("error"), (String(pname) + String(".") + String(methodName) + String(": not implemented for this type"))], cons(new Symbol("_impl"), params)]]], env));
|
||
})();
|
||
})(); } }
|
||
return NIL;
|
||
})(); };
|
||
PRIMITIVES["sf-define-protocol"] = sfDefineProtocol;
|
||
|
||
// check-match-exhaustiveness
|
||
var checkMatchExhaustiveness = function(clauses) { return (function() {
|
||
var warnings = [];
|
||
var patterns = map(first, clauses);
|
||
var hasWildcard = some(function(p) { return sxOr(sxEq(p, new Symbol("_")), (isSxTruthy(symbol_p(p)) && isSxTruthy(!isSxTruthy(sxEq(p, true))) && !isSxTruthy(sxEq(p, false)))); }, patterns);
|
||
var hasElse = some(function(p) { return sxEq(p, "else"); }, patterns);
|
||
var hasTrue = some(function(p) { return sxEq(p, true); }, patterns);
|
||
var hasFalse = some(function(p) { return sxEq(p, false); }, patterns);
|
||
var hasNil = some(function(p) { return sxEq(p, NIL); }, patterns);
|
||
var hasPredicate = some(function(p) { return (isSxTruthy(isList(p)) && sxEq(first(p), new Symbol("?"))); }, patterns);
|
||
if (isSxTruthy((isSxTruthy(!isSxTruthy(hasWildcard)) && !isSxTruthy(hasElse)))) {
|
||
warnings = append(warnings, ["match may be non-exhaustive (no wildcard or :else pattern)"]);
|
||
}
|
||
if (isSxTruthy((isSxTruthy(sxOr(hasTrue, hasFalse)) && isSxTruthy(!isSxTruthy((isSxTruthy(hasTrue) && hasFalse))) && isSxTruthy(!isSxTruthy(hasWildcard)) && !isSxTruthy(hasElse)))) {
|
||
warnings = append(warnings, [(isSxTruthy(hasTrue) ? "match on boolean missing false case" : "match on boolean missing true case")]);
|
||
}
|
||
if (isSxTruthy((isSxTruthy(hasNil) && isSxTruthy(!isSxTruthy(hasWildcard)) && isSxTruthy(!isSxTruthy(hasElse)) && sxEq(len(patterns), 1)))) {
|
||
warnings = append(warnings, ["match checks nil but has no non-nil pattern"]);
|
||
}
|
||
return warnings;
|
||
})(); };
|
||
PRIMITIVES["check-match-exhaustiveness"] = checkMatchExhaustiveness;
|
||
|
||
// sf-implement
|
||
var sfImplement = function(args, env) { return (function() {
|
||
var protoName = symbolName(first(args));
|
||
var rawTypeName = symbolName(nth(args, 1));
|
||
var typeName = slice(rawTypeName, 1, (len(rawTypeName) - 1));
|
||
var methodDefs = rest(rest(args));
|
||
return (function() {
|
||
var proto = get(_protocolRegistry_, protoName);
|
||
return (isSxTruthy(isNil(proto)) ? error((String("Unknown protocol: ") + String(protoName))) : (function() {
|
||
var impls = get(proto, "impls");
|
||
var typeImpls = sxOr(get(impls, typeName), {});
|
||
{ var _c = methodDefs; for (var _i = 0; _i < _c.length; _i++) { var methodDef = _c[_i]; (function() {
|
||
var mname = symbolName(first(methodDef));
|
||
var protoMethod = first(filter(function(m) { return sxEq(get(m, "name"), mname); }, get(proto, "methods")));
|
||
return (isSxTruthy(isNil(protoMethod)) ? error((String("Unknown method ") + String(mname) + String(" in protocol ") + String(protoName))) : (function() {
|
||
var arity = get(protoMethod, "arity");
|
||
var params = slice(methodDef, 1, arity);
|
||
var body = (isSxTruthy(sxEq(len(methodDef), (arity + 1))) ? nth(methodDef, arity) : cons(new Symbol("begin"), slice(methodDef, arity)));
|
||
return dictSet(typeImpls, mname, evalExpr([new Symbol("fn"), params, body], env));
|
||
})());
|
||
})(); } }
|
||
impls[typeName] = typeImpls;
|
||
return NIL;
|
||
})());
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sf-implement"] = sfImplement;
|
||
|
||
// satisfies?
|
||
var satisfies_p = function(protoName, value) { return (isSxTruthy(!isSxTruthy(record_p(value))) ? false : (function() {
|
||
var proto = get(_protocolRegistry_, (isSxTruthy(symbol_p(protoName)) ? symbolName(protoName) : protoName));
|
||
return (isSxTruthy(isNil(proto)) ? false : !isSxTruthy(isNil(get(get(proto, "impls"), typeOf(value)))));
|
||
})()); };
|
||
PRIMITIVES["satisfies?"] = satisfies_p;
|
||
|
||
// check-match-exhaustiveness
|
||
var checkMatchExhaustiveness = function(clauses) { return (function() {
|
||
var warnings = [];
|
||
var patterns = map(function(c) { return first(c); }, clauses);
|
||
var hasWildcard = some(function(p) { return (isSxTruthy(symbol_p(p)) && isSxTruthy(!isSxTruthy(sxEq(p, true))) && !isSxTruthy(sxEq(p, false))); }, patterns);
|
||
var hasElse = some(function(p) { return sxEq(p, "else"); }, patterns);
|
||
var hasTrue = some(function(p) { return sxEq(p, true); }, patterns);
|
||
var hasFalse = some(function(p) { return sxEq(p, false); }, patterns);
|
||
if (isSxTruthy((isSxTruthy(!isSxTruthy(hasWildcard)) && !isSxTruthy(hasElse)))) {
|
||
warnings = append(warnings, ["match may be non-exhaustive (no wildcard or :else pattern)"]);
|
||
}
|
||
if (isSxTruthy((isSxTruthy(sxOr(hasTrue, hasFalse)) && isSxTruthy(!isSxTruthy((isSxTruthy(hasTrue) && hasFalse))) && isSxTruthy(!isSxTruthy(hasWildcard)) && !isSxTruthy(hasElse)))) {
|
||
warnings = append(warnings, [(isSxTruthy(hasTrue) ? "match on boolean missing false case" : "match on boolean missing true case")]);
|
||
}
|
||
return warnings;
|
||
})(); };
|
||
PRIMITIVES["check-match-exhaustiveness"] = checkMatchExhaustiveness;
|
||
|
||
// match-find-clause
|
||
var matchFindClause = function(val, clauses, env) { return (isSxTruthy(isEmpty(clauses)) ? NIL : (function() {
|
||
var clause = first(clauses);
|
||
var pattern = first(clause);
|
||
var body = nth(clause, 1);
|
||
var local = envExtend(env);
|
||
return (isSxTruthy(matchPattern(pattern, val, local)) ? [local, body] : matchFindClause(val, rest(clauses), env));
|
||
})()); };
|
||
PRIMITIVES["match-find-clause"] = matchFindClause;
|
||
|
||
// match-pattern
|
||
var matchPattern = function(pattern, value, env) { return (isSxTruthy(sxEq(pattern, new Symbol("_"))) ? true : (isSxTruthy((isSxTruthy(isList(pattern)) && isSxTruthy(sxEq(len(pattern), 2)) && sxEq(first(pattern), new Symbol("?")))) ? (function() {
|
||
var pred = evalExpr(nth(pattern, 1), env);
|
||
return cekCall(pred, [value]);
|
||
})() : (isSxTruthy((isSxTruthy(isList(pattern)) && isSxTruthy(!isSxTruthy(isEmpty(pattern))) && sxEq(first(pattern), new Symbol("quote")))) ? sxEq(value, nth(pattern, 1)) : (isSxTruthy(symbol_p(pattern)) ? (envBind(env, symbolName(pattern), value), true) : (isSxTruthy((isSxTruthy(isList(pattern)) && isSxTruthy(!isSxTruthy(isEmpty(pattern))) && isSxTruthy(symbol_p(first(pattern))) && isSxTruthy(isDict(value)) && get(value, "_adt"))) ? (function() {
|
||
var ctorName = symbolName(first(pattern));
|
||
var fieldPatterns = rest(pattern);
|
||
var fields = get(value, "_fields");
|
||
return (isSxTruthy(sxEq(get(value, "_ctor"), ctorName)) && isSxTruthy(sxEq(len(fieldPatterns), len(fields))) && isEvery(function(pair) { return matchPattern(first(pair), nth(pair, 1), env); }, zip(fieldPatterns, fields)));
|
||
})() : (isSxTruthy((isSxTruthy(isDict(pattern)) && isDict(value))) ? isEvery(function(k) { return matchPattern(get(pattern, k), get(value, k), env); }, keys(pattern)) : (isSxTruthy((isSxTruthy(isList(pattern)) && isSxTruthy(isList(value)) && contains(pattern, new Symbol("&rest")))) ? (function() {
|
||
var restIdx = indexOf_(pattern, new Symbol("&rest"));
|
||
return (isSxTruthy((len(value) >= restIdx)) && isSxTruthy(isEvery(function(pair) { return matchPattern(first(pair), nth(pair, 1), env); }, zip(slice(pattern, 0, restIdx), slice(value, 0, restIdx)))) && (function() {
|
||
var restName = nth(pattern, (restIdx + 1));
|
||
envBind(env, symbolName(restName), slice(value, restIdx));
|
||
return true;
|
||
})());
|
||
})() : (isSxTruthy((isSxTruthy(isList(pattern)) && isList(value))) ? (isSxTruthy(!isSxTruthy(sxEq(len(pattern), len(value)))) ? false : (function() {
|
||
var pairs = zip(pattern, value);
|
||
return isEvery(function(pair) { return matchPattern(first(pair), nth(pair, 1), env); }, pairs);
|
||
})()) : sxEq(pattern, value))))))))); };
|
||
PRIMITIVES["match-pattern"] = matchPattern;
|
||
|
||
// match-clause-is-else?
|
||
var matchClauseIsElse_p = function(clause) { return (function() {
|
||
var p = first(clause);
|
||
return sxOr(sxEq(p, new Symbol("_")), sxEq(p, new Symbol("else")), sxEq(p, "else"));
|
||
})(); };
|
||
PRIMITIVES["match-clause-is-else?"] = matchClauseIsElse_p;
|
||
|
||
// match-clause-ctor-name
|
||
var matchClauseCtorName = function(clause) { return (function() {
|
||
var p = first(clause);
|
||
return (isSxTruthy((isSxTruthy(isList(p)) && isSxTruthy(!isSxTruthy(isEmpty(p))) && symbol_p(first(p)))) ? symbolName(first(p)) : (isSxTruthy((isSxTruthy(symbol_p(p)) && isSxTruthy(!isSxTruthy(sxEq(p, new Symbol("_")))) && !isSxTruthy(sxEq(p, new Symbol("else"))))) ? NIL : NIL));
|
||
})(); };
|
||
PRIMITIVES["match-clause-ctor-name"] = matchClauseCtorName;
|
||
|
||
// match-warn-non-exhaustive
|
||
var matchWarnNonExhaustive = function(env, typeName, registered, clauseCtors) { return (function() {
|
||
var missing = filter(function(c) { return !isSxTruthy(contains(clauseCtors, c)); }, registered);
|
||
if (isSxTruthy(!isSxTruthy(isEmpty(missing)))) {
|
||
if (isSxTruthy(!isSxTruthy(envHas(env, "*adt-warned*")))) {
|
||
envBind(env, "*adt-warned*", {});
|
||
}
|
||
(function() {
|
||
var warned = envGet(env, "*adt-warned*");
|
||
var key = (String(typeName) + String("|") + String(join(",", missing)));
|
||
return (isSxTruthy(!isSxTruthy(get(warned, key))) ? (dictSet(warned, key, true), hostWarn((String("[sx] match: non-exhaustive — ") + String(typeName) + String(": missing ") + String(join(", ", missing))))) : NIL);
|
||
})();
|
||
}
|
||
return NIL;
|
||
})(); };
|
||
PRIMITIVES["match-warn-non-exhaustive"] = matchWarnNonExhaustive;
|
||
|
||
// match-check-exhaustiveness
|
||
var matchCheckExhaustiveness = function(val, clauses, env) { return (isSxTruthy((isSxTruthy(isDict(val)) && get(val, "_adt"))) ? (function() {
|
||
var typeName = get(val, "_type");
|
||
return (isSxTruthy((isSxTruthy(envHas(env, "*adt-registry*")) && typeName)) ? (function() {
|
||
var registered = get(envGet(env, "*adt-registry*"), typeName);
|
||
return (isSxTruthy((isSxTruthy(registered) && !isSxTruthy(some(matchClauseIsElse_p, clauses)))) ? (function() {
|
||
var clauseCtors = filter(function(n) { return !isSxTruthy(isNil(n)); }, map(matchClauseCtorName, clauses));
|
||
return matchWarnNonExhaustive(env, typeName, registered, clauseCtors);
|
||
})() : NIL);
|
||
})() : NIL);
|
||
})() : NIL); };
|
||
PRIMITIVES["match-check-exhaustiveness"] = matchCheckExhaustiveness;
|
||
|
||
// step-sf-match
|
||
var stepSfMatch = function(args, env, kont) { return (function() {
|
||
var val = trampoline(evalExpr(first(args), env));
|
||
var clauses = rest(args);
|
||
return (matchCheckExhaustiveness(val, clauses, env), (function() {
|
||
var result = matchFindClause(val, clauses, env);
|
||
return (isSxTruthy(isNil(result)) ? makeCekValue((String("match: no clause matched ") + String(inspect(val))), env, kontPush(makeRaiseEvalFrame(env, false), kont)) : makeCekState(nth(result, 1), first(result), kont));
|
||
})());
|
||
})(); };
|
||
PRIMITIVES["step-sf-match"] = stepSfMatch;
|
||
|
||
// step-sf-handler-bind
|
||
var stepSfHandlerBind = function(args, env, kont) { return (function() {
|
||
var handlerSpecs = first(args);
|
||
var body = rest(args);
|
||
var handlers = map(function(spec) { return [trampoline(evalExpr(first(spec), env)), trampoline(evalExpr(nth(spec, 1), env))]; }, handlerSpecs);
|
||
return (isSxTruthy(isEmpty(body)) ? makeCekValue(NIL, env, kont) : makeCekState(first(body), env, kontPush(makeHandlerFrame(handlers, rest(body), env), kont)));
|
||
})(); };
|
||
PRIMITIVES["step-sf-handler-bind"] = stepSfHandlerBind;
|
||
|
||
// step-sf-restart-case
|
||
var stepSfRestartCase = function(args, env, kont) { return (function() {
|
||
var body = first(args);
|
||
var restartSpecs = rest(args);
|
||
var restarts = map(function(spec) { return [(isSxTruthy(symbol_p(first(spec))) ? symbolName(first(spec)) : first(spec)), nth(spec, 1), nth(spec, 2)]; }, restartSpecs);
|
||
return makeCekState(body, env, kontPush(makeRestartFrame(restarts, [], env), kont));
|
||
})(); };
|
||
PRIMITIVES["step-sf-restart-case"] = stepSfRestartCase;
|
||
|
||
// step-sf-signal
|
||
var stepSfSignal = function(args, env, kont) { return (function() {
|
||
var condition = trampoline(evalExpr(first(args), env));
|
||
var handlerFn = kontFindHandler(kont, condition);
|
||
return (isSxTruthy(isNil(handlerFn)) ? error((String("Unhandled condition: ") + String(inspect(condition)))) : continueWithCall(handlerFn, [condition], env, [condition], kontPush(makeSignalReturnFrame(env, kont), kont)));
|
||
})(); };
|
||
PRIMITIVES["step-sf-signal"] = stepSfSignal;
|
||
|
||
// step-sf-invoke-restart
|
||
var stepSfInvokeRestart = function(args, env, kont) { return (function() {
|
||
var restartName = (function() {
|
||
var rn = (isSxTruthy(symbol_p(first(args))) ? symbolName(first(args)) : trampoline(evalExpr(first(args), env)));
|
||
return (isSxTruthy(symbol_p(rn)) ? symbolName(rn) : rn);
|
||
})();
|
||
var restartArg = (isSxTruthy((len(args) >= 2)) ? trampoline(evalExpr(nth(args, 1), env)) : NIL);
|
||
var found = kontFindRestart(kont, restartName);
|
||
return (isSxTruthy(isNil(found)) ? error((String("No restart named: ") + String(inspect(restartName)))) : (function() {
|
||
var entry = first(found);
|
||
var restartFrame = nth(found, 1);
|
||
var restKont = nth(found, 2);
|
||
return (function() {
|
||
var params = nth(entry, 1);
|
||
var body = nth(entry, 2);
|
||
var restartEnv = envExtend(get(restartFrame, "env"));
|
||
if (isSxTruthy(!isSxTruthy(isEmpty(params)))) {
|
||
envBind(restartEnv, first(params), restartArg);
|
||
}
|
||
return makeCekState(body, restartEnv, restKont);
|
||
})();
|
||
})());
|
||
})(); };
|
||
PRIMITIVES["step-sf-invoke-restart"] = stepSfInvokeRestart;
|
||
|
||
// step-sf-if
|
||
var stepSfIf = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeIfFrame(nth(args, 1), (isSxTruthy((len(args) > 2)) ? nth(args, 2) : NIL), env), kont)); };
|
||
PRIMITIVES["step-sf-if"] = stepSfIf;
|
||
|
||
// step-sf-when
|
||
var stepSfWhen = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeWhenFrame(rest(args), env), kont)); };
|
||
PRIMITIVES["step-sf-when"] = stepSfWhen;
|
||
|
||
// step-sf-begin
|
||
var stepSfBegin = function(args, env, kont) { return (isSxTruthy(isEmpty(args)) ? makeCekValue(NIL, env, kont) : (isSxTruthy(sxEq(len(args), 1)) ? makeCekState(first(args), env, kont) : makeCekState(first(args), env, kontPush(makeBeginFrame(rest(args), env), kont)))); };
|
||
PRIMITIVES["step-sf-begin"] = stepSfBegin;
|
||
|
||
// step-sf-let
|
||
var stepSfLet = function(args, env, kont) { return (isSxTruthy(sxEq(typeOf(first(args)), "symbol")) ? makeCekValue(sfNamedLet(args, env), env, kont) : (function() {
|
||
var bindings = first(args);
|
||
var body = rest(args);
|
||
var local = envExtend(env);
|
||
return (isSxTruthy(isEmpty(bindings)) ? stepSfBegin(body, local, kont) : (function() {
|
||
var firstBinding = (isSxTruthy((isSxTruthy(sxEq(typeOf(first(bindings)), "list")) && sxEq(len(first(bindings)), 2))) ? first(bindings) : [first(bindings), nth(bindings, 1)]);
|
||
var restBindings = (isSxTruthy((isSxTruthy(sxEq(typeOf(first(bindings)), "list")) && sxEq(len(first(bindings)), 2))) ? rest(bindings) : (function() {
|
||
var pairs = [];
|
||
reduce(function(acc, i) { return append_b(pairs, [nth(bindings, (i * 2)), nth(bindings, ((i * 2) + 1))]); }, NIL, range(1, (len(bindings) / 2)));
|
||
return pairs;
|
||
})());
|
||
return (function() {
|
||
var vname = (isSxTruthy(sxEq(typeOf(first(firstBinding)), "symbol")) ? symbolName(first(firstBinding)) : first(firstBinding));
|
||
return makeCekState(nth(firstBinding, 1), local, kontPush(makeLetFrame(vname, restBindings, body, local), kont));
|
||
})();
|
||
})());
|
||
})()); };
|
||
PRIMITIVES["step-sf-let"] = stepSfLet;
|
||
|
||
// step-sf-define
|
||
var stepSfDefine = function(args, env, kont) { return (function() {
|
||
var resolvedArgs = (isSxTruthy(sxEq(typeOf(first(args)), "list")) ? (function() {
|
||
var fnName = first(first(args));
|
||
var params = rest(first(args));
|
||
var bodyParts = rest(args);
|
||
return [fnName, concat([makeSymbol("fn")], [params], bodyParts)];
|
||
})() : args);
|
||
return (function() {
|
||
var nameSym = first(resolvedArgs);
|
||
var hasEffects = (isSxTruthy((len(resolvedArgs) >= 4)) && isSxTruthy(sxEq(typeOf(nth(resolvedArgs, 1)), "keyword")) && sxEq(keywordName(nth(resolvedArgs, 1)), "effects"));
|
||
var valIdx = (isSxTruthy((isSxTruthy((len(resolvedArgs) >= 4)) && isSxTruthy(sxEq(typeOf(nth(resolvedArgs, 1)), "keyword")) && sxEq(keywordName(nth(resolvedArgs, 1)), "effects"))) ? 3 : 1);
|
||
var effectList = (isSxTruthy((isSxTruthy((len(resolvedArgs) >= 4)) && isSxTruthy(sxEq(typeOf(nth(resolvedArgs, 1)), "keyword")) && sxEq(keywordName(nth(resolvedArgs, 1)), "effects"))) ? nth(resolvedArgs, 2) : NIL);
|
||
return makeCekState(nth(resolvedArgs, valIdx), env, kontPush(makeDefineFrame(symbolName(nameSym), env, hasEffects, effectList), kont));
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["step-sf-define"] = stepSfDefine;
|
||
|
||
// step-sf-set!
|
||
var stepSfSet = function(args, env, kont) { return makeCekState(nth(args, 1), env, kontPush(makeSetFrame(symbolName(first(args)), env), kont)); };
|
||
PRIMITIVES["step-sf-set!"] = stepSfSet;
|
||
|
||
// step-sf-and
|
||
var stepSfAnd = function(args, env, kont) { return (isSxTruthy(isEmpty(args)) ? makeCekValue(true, env, kont) : makeCekState(first(args), env, kontPush(makeAndFrame(rest(args), env), kont))); };
|
||
PRIMITIVES["step-sf-and"] = stepSfAnd;
|
||
|
||
// step-sf-or
|
||
var stepSfOr = function(args, env, kont) { return (isSxTruthy(isEmpty(args)) ? makeCekValue(false, env, kont) : makeCekState(first(args), env, kontPush(makeOrFrame(rest(args), env), kont))); };
|
||
PRIMITIVES["step-sf-or"] = stepSfOr;
|
||
|
||
// step-sf-cond
|
||
var stepSfCond = function(args, env, kont) { return (function() {
|
||
var scheme_p = condScheme_p(args);
|
||
return (isSxTruthy(scheme_p) ? (isSxTruthy(isEmpty(args)) ? makeCekValue(NIL, env, kont) : (function() {
|
||
var clause = first(args);
|
||
var test = first(clause);
|
||
return (isSxTruthy(isElseClause(test)) ? makeCekState(nth(clause, 1), env, kont) : makeCekState(test, env, kontPush(makeCondFrame(args, env, true), kont)));
|
||
})()) : (isSxTruthy((len(args) < 2)) ? makeCekValue(NIL, env, kont) : (function() {
|
||
var test = first(args);
|
||
return (isSxTruthy(isElseClause(test)) ? makeCekState(nth(args, 1), env, kont) : makeCekState(test, env, kontPush(makeCondFrame(args, env, false), kont)));
|
||
})()));
|
||
})(); };
|
||
PRIMITIVES["step-sf-cond"] = stepSfCond;
|
||
|
||
// step-sf-thread-first
|
||
var stepSfThreadFirst = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeThreadFrame(rest(args), env, "first", NIL), kont)); };
|
||
PRIMITIVES["step-sf-thread-first"] = stepSfThreadFirst;
|
||
|
||
// step-sf-thread-last
|
||
var stepSfThreadLast = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeThreadFrame(rest(args), env, "last", NIL), kont)); };
|
||
PRIMITIVES["step-sf-thread-last"] = stepSfThreadLast;
|
||
|
||
// step-sf-thread-as
|
||
var stepSfThreadAs = function(args, env, kont) { return (function() {
|
||
var init = first(args);
|
||
var name = nth(args, 1);
|
||
var forms = rest(rest(args));
|
||
return makeCekState(init, env, kontPush(makeThreadFrame(forms, env, "as", name), kont));
|
||
})(); };
|
||
PRIMITIVES["step-sf-thread-as"] = stepSfThreadAs;
|
||
|
||
// step-sf-lambda
|
||
var stepSfLambda = function(args, env, kont) { return makeCekValue(sfLambda(args, env), env, kont); };
|
||
PRIMITIVES["step-sf-lambda"] = stepSfLambda;
|
||
|
||
// step-sf-scope
|
||
var stepSfScope = function(args, env, kont) { return (function() {
|
||
var name = trampoline(evalExpr(first(args), env));
|
||
var restArgs = slice(args, 1);
|
||
var val = NIL;
|
||
var body = NIL;
|
||
(isSxTruthy((isSxTruthy((len(restArgs) >= 2)) && isSxTruthy(sxEq(typeOf(first(restArgs)), "keyword")) && sxEq(keywordName(first(restArgs)), "value"))) ? ((val = trampoline(evalExpr(nth(restArgs, 1), env))), (body = slice(restArgs, 2))) : (body = restArgs));
|
||
return (isSxTruthy(isEmpty(body)) ? makeCekValue(NIL, env, kont) : makeCekState(first(body), env, kontPush(makeScopeAccFrame(name, val, rest(body), env), kont)));
|
||
})(); };
|
||
PRIMITIVES["step-sf-scope"] = stepSfScope;
|
||
|
||
// step-sf-provide
|
||
var stepSfProvide = function(args, env, kont) { return (function() {
|
||
var name = trampoline(evalExpr(first(args), env));
|
||
var val = trampoline(evalExpr(nth(args, 1), env));
|
||
var body = slice(args, 2);
|
||
scopePush(name, val);
|
||
return (isSxTruthy(isEmpty(body)) ? (scopePop(name), makeCekValue(NIL, env, kont)) : makeCekState(first(body), env, kontPush(makeProvideFrame(name, val, rest(body), env), kont)));
|
||
})(); };
|
||
PRIMITIVES["step-sf-provide"] = stepSfProvide;
|
||
|
||
// step-sf-context
|
||
var stepSfContext = function(args, env, kont) { return (function() {
|
||
var name = trampoline(evalExpr(first(args), env));
|
||
var defaultVal = (isSxTruthy((len(args) >= 2)) ? trampoline(evalExpr(nth(args, 1), env)) : NIL);
|
||
var frame = kontFindProvide(kont, name);
|
||
if (isSxTruthy(_bindTracking_)) {
|
||
if (isSxTruthy(!isSxTruthy(contains(_bindTracking_, name)))) {
|
||
_bindTracking_.push(name);
|
||
}
|
||
}
|
||
return makeCekValue((function() {
|
||
var sv = scopePeek(name);
|
||
return (isSxTruthy(isNil(sv)) ? (isSxTruthy(frame) ? get(frame, "value") : defaultVal) : sv);
|
||
})(), env, kont);
|
||
})(); };
|
||
PRIMITIVES["step-sf-context"] = stepSfContext;
|
||
|
||
// step-sf-peek
|
||
var stepSfPeek = function(args, env, kont) { return (function() {
|
||
var name = trampoline(evalExpr(first(args), env));
|
||
var defaultVal = (isSxTruthy((len(args) >= 2)) ? trampoline(evalExpr(nth(args, 1), env)) : NIL);
|
||
var frame = kontFindProvide(kont, name);
|
||
return makeCekValue((isSxTruthy(frame) ? get(frame, "value") : (isSxTruthy(envHas(env, "peek")) ? apply(envGet(env, "peek"), [name, defaultVal]) : defaultVal)), env, kont);
|
||
})(); };
|
||
PRIMITIVES["step-sf-peek"] = stepSfPeek;
|
||
|
||
// step-sf-provide!
|
||
var stepSfProvide_b = function(args, env, kont) { return (function() {
|
||
var name = trampoline(evalExpr(first(args), env));
|
||
return makeCekState(nth(args, 1), env, kontPush(makeProvideSetFrame(name, env), kont));
|
||
})(); };
|
||
PRIMITIVES["step-sf-provide!"] = stepSfProvide_b;
|
||
|
||
// step-sf-emit
|
||
var stepSfEmit = function(args, env, kont) { return (function() {
|
||
var name = trampoline(evalExpr(first(args), env));
|
||
var val = trampoline(evalExpr(nth(args, 1), env));
|
||
var frame = kontFindScopeAcc(kont, name);
|
||
return (isSxTruthy(frame) ? (dictSet(frame, "emitted", append(get(frame, "emitted"), [val])), makeCekValue(NIL, env, kont)) : ((isSxTruthy(envHas(env, "scope-emit!")) ? apply(envGet(env, "scope-emit!"), [name, val]) : NIL), makeCekValue(NIL, env, kont)));
|
||
})(); };
|
||
PRIMITIVES["step-sf-emit"] = stepSfEmit;
|
||
|
||
// step-sf-emitted
|
||
var stepSfEmitted = function(args, env, kont) { return (function() {
|
||
var name = trampoline(evalExpr(first(args), env));
|
||
var frame = kontFindScopeAcc(kont, name);
|
||
return makeCekValue((isSxTruthy(frame) ? get(frame, "emitted") : (isSxTruthy(envHas(env, "emitted")) ? apply(envGet(env, "emitted"), [name]) : [])), env, kont);
|
||
})(); };
|
||
PRIMITIVES["step-sf-emitted"] = stepSfEmitted;
|
||
|
||
// step-sf-reset
|
||
var stepSfReset = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeResetFrame(env), kont)); };
|
||
PRIMITIVES["step-sf-reset"] = stepSfReset;
|
||
|
||
// step-sf-shift
|
||
var stepSfShift = function(args, env, kont) { return (function() {
|
||
var kName = symbolName(first(args));
|
||
var body = nth(args, 1);
|
||
var capturedResult = kontCaptureToReset(kont);
|
||
var captured = first(capturedResult);
|
||
var restKont = nth(capturedResult, 1);
|
||
return (function() {
|
||
var k = makeCekContinuation(captured, restKont);
|
||
return (function() {
|
||
var shiftEnv = envExtend(env);
|
||
envBind(shiftEnv, kName, k);
|
||
return makeCekState(body, shiftEnv, restKont);
|
||
})();
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["step-sf-shift"] = stepSfShift;
|
||
|
||
// step-sf-deref
|
||
var stepSfDeref = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeDerefFrame(env), kont)); };
|
||
PRIMITIVES["step-sf-deref"] = stepSfDeref;
|
||
|
||
// cek-call
|
||
var cekCall = function(f, args) { return (function() {
|
||
var a = (isSxTruthy(isNil(args)) ? [] : args);
|
||
return (isSxTruthy(isNil(f)) ? NIL : (isSxTruthy(sxOr(isLambda(f), isCallable(f))) ? cekRun(continueWithCall(f, a, makeEnv(), a, [])) : NIL));
|
||
})(); };
|
||
PRIMITIVES["cek-call"] = cekCall;
|
||
|
||
// reactive-shift-deref
|
||
var reactiveShiftDeref = function(sig, env, kont) { return (function() {
|
||
var scanResult = kontCaptureToReactiveReset(kont);
|
||
var capturedFrames = first(scanResult);
|
||
var resetFrame = nth(scanResult, 1);
|
||
var remainingKont = nth(scanResult, 2);
|
||
var updateFn = get(resetFrame, "update-fn");
|
||
return (function() {
|
||
var subDisposers = [];
|
||
return (function() {
|
||
var subscriber = function() { { var _c = subDisposers; for (var _i = 0; _i < _c.length; _i++) { var d = _c[_i]; cekCall(d, NIL); } }
|
||
subDisposers = [];
|
||
return (function() {
|
||
var newReset = makeReactiveResetFrame(env, updateFn, false);
|
||
var newKont = concat(capturedFrames, [newReset], remainingKont);
|
||
return withIslandScope(function(d) { return append_b(subDisposers, d); }, function() { return cekRun(makeCekValue(signalValue(sig), env, newKont)); });
|
||
})(); };
|
||
signalAddSub(sig, subscriber);
|
||
registerInScope(function() { signalRemoveSub(sig, subscriber);
|
||
return forEach(function(d) { return cekCall(d, NIL); }, subDisposers); });
|
||
return (function() {
|
||
var initialKont = concat(capturedFrames, [resetFrame], remainingKont);
|
||
return makeCekValue(signalValue(sig), env, initialKont);
|
||
})();
|
||
})();
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["reactive-shift-deref"] = reactiveShiftDeref;
|
||
|
||
// step-eval-call
|
||
var stepEvalCall = function(head, args, env, kont) { return (function() {
|
||
var hname = (isSxTruthy(sxEq(typeOf(head), "symbol")) ? symbolName(head) : NIL);
|
||
return makeCekState(head, env, kontPush(makeArgFrame(NIL, [], args, env, args, hname), kont));
|
||
})(); };
|
||
PRIMITIVES["step-eval-call"] = stepEvalCall;
|
||
|
||
// ho-form-name?
|
||
var hoFormName_p = function(name) { return sxOr(sxEq(name, "map"), sxEq(name, "map-indexed"), sxEq(name, "filter"), sxEq(name, "reduce"), sxEq(name, "some"), sxEq(name, "every?"), sxEq(name, "for-each")); };
|
||
PRIMITIVES["ho-form-name?"] = hoFormName_p;
|
||
|
||
// ho-fn?
|
||
var hoFn_p = function(v) { return sxOr(isCallable(v), isLambda(v)); };
|
||
PRIMITIVES["ho-fn?"] = hoFn_p;
|
||
|
||
// ho-swap-args
|
||
var hoSwapArgs = function(hoType, evaled) { return (isSxTruthy(sxEq(hoType, "reduce")) ? (function() {
|
||
var a = first(evaled);
|
||
var b = nth(evaled, 1);
|
||
return (isSxTruthy((isSxTruthy(!isSxTruthy(hoFn_p(a))) && hoFn_p(b))) ? [b, nth(evaled, 2), a] : evaled);
|
||
})() : (function() {
|
||
var a = first(evaled);
|
||
var b = nth(evaled, 1);
|
||
return (isSxTruthy((isSxTruthy(!isSxTruthy(hoFn_p(a))) && hoFn_p(b))) ? [b, a] : evaled);
|
||
})()); };
|
||
PRIMITIVES["ho-swap-args"] = hoSwapArgs;
|
||
|
||
// seq-to-list
|
||
var seqToList = function(x) { return (isSxTruthy(sxEq(x, NIL)) ? [] : (isSxTruthy(isList(x)) ? x : (isSxTruthy(isVector(x)) ? vectorToList(x) : (isSxTruthy(isString(x)) ? (function() {
|
||
var n = len(x);
|
||
var loop = function(i, acc) { return (isSxTruthy((i < 0)) ? acc : loop((i - 1), cons(slice(x, i, (i + 1)), acc))); };
|
||
PRIMITIVES["loop"] = loop;
|
||
return loop((n - 1), []);
|
||
})() : x)))); };
|
||
PRIMITIVES["seq-to-list"] = seqToList;
|
||
|
||
// ho-setup-dispatch
|
||
var hoSetupDispatch = function(hoType, evaled, env, kont) { return (function() {
|
||
var ordered = hoSwapArgs(hoType, evaled);
|
||
return (function() {
|
||
var f = first(ordered);
|
||
return (function() { var _m = hoType; if (_m == "map") return (isSxTruthy((len(ordered) > 2)) ? (function() {
|
||
var colls = rest(ordered);
|
||
return (isSxTruthy(some(function(c) { return isEmpty(c); }, colls)) ? makeCekValue([], env, kont) : (function() {
|
||
var heads = map(function(c) { return first(c); }, colls);
|
||
var tails = map(function(c) { return rest(c); }, colls);
|
||
return continueWithCall(f, heads, env, [], kontPush(makeMultiMapFrame(f, tails, [], env), kont));
|
||
})());
|
||
})() : (function() {
|
||
var coll = seqToList(nth(ordered, 1));
|
||
return (isSxTruthy(isEmpty(coll)) ? makeCekValue([], env, kont) : continueWithCall(f, [first(coll)], env, [], kontPush(makeMapFrame(f, rest(coll), [], env), kont)));
|
||
})()); if (_m == "map-indexed") return (function() {
|
||
var coll = seqToList(nth(ordered, 1));
|
||
return (isSxTruthy(isEmpty(coll)) ? makeCekValue([], env, kont) : continueWithCall(f, [0, first(coll)], env, [], kontPush(makeMapIndexedFrame(f, rest(coll), [], env), kont)));
|
||
})(); if (_m == "filter") return (function() {
|
||
var coll = seqToList(nth(ordered, 1));
|
||
return (isSxTruthy(isEmpty(coll)) ? makeCekValue([], env, kont) : continueWithCall(f, [first(coll)], env, [], kontPush(makeFilterFrame(f, rest(coll), [], first(coll), env), kont)));
|
||
})(); if (_m == "reduce") return (function() {
|
||
var init = nth(ordered, 1);
|
||
var coll = seqToList(nth(ordered, 2));
|
||
return (isSxTruthy(isEmpty(coll)) ? makeCekValue(init, env, kont) : continueWithCall(f, [init, first(coll)], env, [], kontPush(makeReduceFrame(f, rest(coll), env), kont)));
|
||
})(); if (_m == "some") return (function() {
|
||
var coll = seqToList(nth(ordered, 1));
|
||
return (isSxTruthy(isEmpty(coll)) ? makeCekValue(false, env, kont) : continueWithCall(f, [first(coll)], env, [], kontPush(makeSomeFrame(f, rest(coll), env), kont)));
|
||
})(); if (_m == "every") return (function() {
|
||
var coll = seqToList(nth(ordered, 1));
|
||
return (isSxTruthy(isEmpty(coll)) ? makeCekValue(true, env, kont) : continueWithCall(f, [first(coll)], env, [], kontPush(makeEveryFrame(f, rest(coll), env), kont)));
|
||
})(); if (_m == "for-each") return (function() {
|
||
var coll = seqToList(nth(ordered, 1));
|
||
return (isSxTruthy(isEmpty(coll)) ? makeCekValue(NIL, env, kont) : continueWithCall(f, [first(coll)], env, [], kontPush(makeForEachFrame(f, rest(coll), env), kont)));
|
||
})(); return error((String("Unknown HO type: ") + String(hoType))); })();
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["ho-setup-dispatch"] = hoSetupDispatch;
|
||
|
||
// sequence-to-list
|
||
var sequenceToList = function(s) { return seqToList(s); };
|
||
PRIMITIVES["sequence-to-list"] = sequenceToList;
|
||
|
||
// sequence-to-vector
|
||
var sequenceToVector = function(s) { return listToVector(seqToList(s)); };
|
||
PRIMITIVES["sequence-to-vector"] = sequenceToVector;
|
||
|
||
// sequence-length
|
||
var sequenceLength = function(s) { return (isSxTruthy(sxEq(s, NIL)) ? 0 : (isSxTruthy(isList(s)) ? len(s) : (isSxTruthy(isVector(s)) ? vectorLength(s) : (isSxTruthy(isString(s)) ? len(s) : len(seqToList(s)))))); };
|
||
PRIMITIVES["sequence-length"] = sequenceLength;
|
||
|
||
// sequence-ref
|
||
var sequenceRef = function(s, i) { return (isSxTruthy(sxOr(sxEq(s, NIL), isList(s))) ? nth(s, i) : (isSxTruthy(isVector(s)) ? vectorRef(s, i) : (isSxTruthy(isString(s)) ? slice(s, i, (i + 1)) : nth(seqToList(s), i)))); };
|
||
PRIMITIVES["sequence-ref"] = sequenceRef;
|
||
|
||
// sequence-append
|
||
var sequenceAppend = function(s1, s2) { return (isSxTruthy((isSxTruthy(isVector(s1)) && isVector(s2))) ? listToVector(concat(vectorToList(s1), vectorToList(s2))) : (isSxTruthy((isSxTruthy(isString(s1)) && isString(s2))) ? (String(s1) + String(s2)) : concat(seqToList(s1), seqToList(s2)))); };
|
||
PRIMITIVES["sequence-append"] = sequenceAppend;
|
||
|
||
// build-range
|
||
var buildRange = function(i, end, step, acc) { return (isSxTruthy((isSxTruthy((step > 0)) ? (i >= end) : (i <= end))) ? reverse(acc) : buildRange((i + step), end, step, cons(i, acc))); };
|
||
PRIMITIVES["build-range"] = buildRange;
|
||
|
||
// in-range
|
||
var inRange = function(a) { var rest = Array.prototype.slice.call(arguments, 1); return (function() {
|
||
var end = (isSxTruthy(isEmpty(rest)) ? a : first(rest));
|
||
var step = (isSxTruthy((len(rest) >= 2)) ? nth(rest, 1) : 1);
|
||
var realStart = (isSxTruthy(isEmpty(rest)) ? 0 : a);
|
||
return (isSxTruthy(sxEq(step, 0)) ? error("in-range: step cannot be zero") : buildRange(realStart, end, step, []));
|
||
})(); };
|
||
PRIMITIVES["in-range"] = inRange;
|
||
|
||
// step-ho-map
|
||
var stepHoMap = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeHoSetupFrame("map", rest(args), [], env), kont)); };
|
||
PRIMITIVES["step-ho-map"] = stepHoMap;
|
||
|
||
// step-ho-map-indexed
|
||
var stepHoMapIndexed = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeHoSetupFrame("map-indexed", rest(args), [], env), kont)); };
|
||
PRIMITIVES["step-ho-map-indexed"] = stepHoMapIndexed;
|
||
|
||
// step-ho-filter
|
||
var stepHoFilter = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeHoSetupFrame("filter", rest(args), [], env), kont)); };
|
||
PRIMITIVES["step-ho-filter"] = stepHoFilter;
|
||
|
||
// step-ho-reduce
|
||
var stepHoReduce = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeHoSetupFrame("reduce", rest(args), [], env), kont)); };
|
||
PRIMITIVES["step-ho-reduce"] = stepHoReduce;
|
||
|
||
// step-ho-some
|
||
var stepHoSome = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeHoSetupFrame("some", rest(args), [], env), kont)); };
|
||
PRIMITIVES["step-ho-some"] = stepHoSome;
|
||
|
||
// step-ho-every
|
||
var stepHoEvery = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeHoSetupFrame("every", rest(args), [], env), kont)); };
|
||
PRIMITIVES["step-ho-every"] = stepHoEvery;
|
||
|
||
// step-ho-for-each
|
||
var stepHoForEach = function(args, env, kont) { return makeCekState(first(args), env, kontPush(makeHoSetupFrame("for-each", rest(args), [], env), kont)); };
|
||
PRIMITIVES["step-ho-for-each"] = stepHoForEach;
|
||
|
||
// step-continue
|
||
var stepContinue = function(state) { return (function() {
|
||
var value = cekValue(state);
|
||
var env = cekEnv(state);
|
||
var kont = cekKont(state);
|
||
return (isSxTruthy(kontEmpty_p(kont)) ? state : (function() {
|
||
var frame = kontTop(kont);
|
||
var restK = kontPop(kont);
|
||
var ft = frameType(frame);
|
||
return (function() { var _m = ft; if (_m == "if") return (isSxTruthy((isSxTruthy(value) && !isSxTruthy(isNil(value)))) ? makeCekState(get(frame, "then"), get(frame, "env"), restK) : (isSxTruthy(isNil(get(frame, "else"))) ? makeCekValue(NIL, env, restK) : makeCekState(get(frame, "else"), get(frame, "env"), restK))); if (_m == "when") return (isSxTruthy((isSxTruthy(value) && !isSxTruthy(isNil(value)))) ? (function() {
|
||
var body = get(frame, "body");
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(isEmpty(body)) ? makeCekValue(NIL, fenv, restK) : (isSxTruthy(sxEq(len(body), 1)) ? makeCekState(first(body), fenv, restK) : makeCekState(first(body), fenv, kontPush(makeBeginFrame(rest(body), fenv), restK))));
|
||
})() : makeCekValue(NIL, env, restK)); if (_m == "begin") return (function() {
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(isEmpty(remaining)) ? makeCekValue(value, fenv, restK) : (isSxTruthy(sxEq(len(remaining), 1)) ? makeCekState(first(remaining), fenv, restK) : makeCekState(first(remaining), fenv, kontPush(makeBeginFrame(rest(remaining), fenv), restK))));
|
||
})(); if (_m == "let") return (function() {
|
||
var name = get(frame, "name");
|
||
var remaining = get(frame, "remaining");
|
||
var body = get(frame, "body");
|
||
var local = get(frame, "env");
|
||
envBind(local, name, value);
|
||
return (isSxTruthy(isEmpty(remaining)) ? stepSfBegin(body, local, restK) : (function() {
|
||
var nextBinding = first(remaining);
|
||
var vname = (isSxTruthy(sxEq(typeOf(first(nextBinding)), "symbol")) ? symbolName(first(nextBinding)) : first(nextBinding));
|
||
return makeCekState(nth(nextBinding, 1), local, kontPush(makeLetFrame(vname, rest(remaining), body, local), restK));
|
||
})());
|
||
})(); if (_m == "define") return (function() {
|
||
var name = get(frame, "name");
|
||
var fenv = get(frame, "env");
|
||
var hasEffects = get(frame, "has-effects");
|
||
var effectList = get(frame, "effect-list");
|
||
if (isSxTruthy((isSxTruthy(isLambda(value)) && isNil(lambdaName(value))))) {
|
||
value.name = name;
|
||
}
|
||
envBind(fenv, name, value);
|
||
if (isSxTruthy(hasEffects)) {
|
||
(function() {
|
||
var effectNames = map(function(e) { return (isSxTruthy(sxEq(typeOf(e), "symbol")) ? symbolName(e) : e); }, effectList);
|
||
var effectAnns = (isSxTruthy(envHas(fenv, "*effect-annotations*")) ? envGet(fenv, "*effect-annotations*") : {});
|
||
effectAnns[name] = effectNames;
|
||
return envBind(fenv, "*effect-annotations*", effectAnns);
|
||
})();
|
||
}
|
||
return makeCekValue(value, fenv, restK);
|
||
})(); if (_m == "define-foreign") return (function() {
|
||
var name = get(frame, "name");
|
||
var fenv = get(frame, "env");
|
||
if (isSxTruthy((isSxTruthy(isLambda(value)) && isNil(lambdaName(value))))) {
|
||
value.name = name;
|
||
}
|
||
envBind(fenv, name, value);
|
||
return makeCekValue(value, fenv, restK);
|
||
})(); if (_m == "set") return (function() {
|
||
var name = get(frame, "name");
|
||
var fenv = get(frame, "env");
|
||
envSet(fenv, name, value);
|
||
return makeCekValue(value, env, restK);
|
||
})(); if (_m == "and") return (isSxTruthy(!isSxTruthy(value)) ? makeCekValue(value, env, restK) : (function() {
|
||
var remaining = get(frame, "remaining");
|
||
return (isSxTruthy(isEmpty(remaining)) ? makeCekValue(value, env, restK) : makeCekState(first(remaining), get(frame, "env"), (isSxTruthy(sxEq(len(remaining), 1)) ? restK : kontPush(makeAndFrame(rest(remaining), get(frame, "env")), restK))));
|
||
})()); if (_m == "or") return (isSxTruthy(value) ? makeCekValue(value, env, restK) : (function() {
|
||
var remaining = get(frame, "remaining");
|
||
return (isSxTruthy(isEmpty(remaining)) ? makeCekValue(false, env, restK) : makeCekState(first(remaining), get(frame, "env"), (isSxTruthy(sxEq(len(remaining), 1)) ? restK : kontPush(makeOrFrame(rest(remaining), get(frame, "env")), restK))));
|
||
})()); if (_m == "cond") return (function() {
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
var scheme_p = get(frame, "scheme");
|
||
return (isSxTruthy(scheme_p) ? (isSxTruthy(value) ? (function() {
|
||
var clause = first(remaining);
|
||
return (isSxTruthy((isSxTruthy((len(clause) > 2)) && isSxTruthy(sxEq(typeOf(nth(clause, 1)), "symbol")) && sxEq(symbolName(nth(clause, 1)), "=>"))) ? makeCekState(nth(clause, 2), fenv, kontPush(makeCondArrowFrame(value, fenv), restK)) : makeCekState(nth(clause, 1), fenv, restK));
|
||
})() : (function() {
|
||
var nextClauses = rest(remaining);
|
||
return (isSxTruthy(isEmpty(nextClauses)) ? makeCekValue(NIL, fenv, restK) : (function() {
|
||
var nextClause = first(nextClauses);
|
||
var nextTest = first(nextClause);
|
||
return (isSxTruthy(isElseClause(nextTest)) ? makeCekState(nth(nextClause, 1), fenv, restK) : makeCekState(nextTest, fenv, kontPush(makeCondFrame(nextClauses, fenv, true), restK)));
|
||
})());
|
||
})()) : (isSxTruthy(value) ? makeCekState(nth(remaining, 1), fenv, restK) : (function() {
|
||
var next = slice(remaining, 2, len(remaining));
|
||
return (isSxTruthy((len(next) < 2)) ? makeCekValue(NIL, fenv, restK) : (function() {
|
||
var nextTest = first(next);
|
||
return (isSxTruthy(isElseClause(nextTest)) ? makeCekState(nth(next, 1), fenv, restK) : makeCekState(nextTest, fenv, kontPush(makeCondFrame(next, fenv, false), restK)));
|
||
})());
|
||
})()));
|
||
})(); if (_m == "case") return (function() {
|
||
var matchVal = get(frame, "match-val");
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(isNil(matchVal)) ? sfCaseStepLoop(value, remaining, fenv, restK) : sfCaseStepLoop(matchVal, remaining, fenv, restK));
|
||
})(); if (_m == "thread") return (function() {
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
var mode = get(frame, "extra");
|
||
var bindName = get(frame, "name");
|
||
return (isSxTruthy(isEmpty(remaining)) ? makeCekValue(value, fenv, restK) : (function() {
|
||
var form = first(remaining);
|
||
var restForms = rest(remaining);
|
||
var newKont = (isSxTruthy(isEmpty(rest(remaining))) ? restK : kontPush(makeThreadFrame(rest(remaining), fenv, mode, bindName), restK));
|
||
return (isSxTruthy(sxEq(mode, "as")) ? (function() {
|
||
var newEnv = envExtend(fenv);
|
||
envBind(newEnv, symbolName(bindName), value);
|
||
return makeCekState(form, newEnv, newKont);
|
||
})() : (isSxTruthy((isSxTruthy(sxEq(typeOf(form), "list")) && isSxTruthy(!isSxTruthy(isEmpty(form))) && isSxTruthy(sxEq(typeOf(first(form)), "symbol")) && hoFormName_p(symbolName(first(form))))) ? makeCekState(cons(first(form), cons([new Symbol("quote"), value], rest(form))), fenv, newKont) : (isSxTruthy(sxEq(mode, "last")) ? (function() {
|
||
var result = threadInsertArgLast(form, value, fenv);
|
||
return (isSxTruthy(isEmpty(restForms)) ? makeCekValue(result, fenv, restK) : makeCekValue(result, fenv, kontPush(makeThreadFrame(restForms, fenv, mode, bindName), restK)));
|
||
})() : (function() {
|
||
var result = threadInsertArg(form, value, fenv);
|
||
return (isSxTruthy(isEmpty(restForms)) ? makeCekValue(result, fenv, restK) : makeCekValue(result, fenv, kontPush(makeThreadFrame(restForms, fenv, mode, bindName), restK)));
|
||
})())));
|
||
})());
|
||
})(); if (_m == "arg") return (function() {
|
||
var f = get(frame, "f");
|
||
var evaled = get(frame, "evaled");
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
var rawArgs = get(frame, "raw-args");
|
||
var hname = get(frame, "head-name");
|
||
return (isSxTruthy(isNil(f)) ? ((isSxTruthy((isSxTruthy(_strict_) && hname)) ? strictCheckArgs(hname, []) : NIL), (isSxTruthy(isEmpty(remaining)) ? continueWithCall(value, [], fenv, rawArgs, restK) : makeCekState(first(remaining), fenv, kontPush(makeArgFrame(value, [], rest(remaining), fenv, rawArgs, hname), restK)))) : (function() {
|
||
var newEvaled = append(evaled, [value]);
|
||
return (isSxTruthy(isEmpty(remaining)) ? ((isSxTruthy((isSxTruthy(_strict_) && hname)) ? strictCheckArgs(hname, newEvaled) : NIL), continueWithCall(f, newEvaled, fenv, rawArgs, restK)) : makeCekState(first(remaining), fenv, kontPush(makeArgFrame(f, newEvaled, rest(remaining), fenv, rawArgs, hname), restK)));
|
||
})());
|
||
})(); if (_m == "dict") return (function() {
|
||
var remaining = get(frame, "remaining");
|
||
var results = get(frame, "results");
|
||
var fenv = get(frame, "env");
|
||
return (function() {
|
||
var lastResult = last(results);
|
||
var completed = append(slice(results, 0, (len(results) - 1)), [[first(lastResult), value]]);
|
||
return (isSxTruthy(isEmpty(remaining)) ? (function() {
|
||
var d = {};
|
||
{ var _c = completed; for (var _i = 0; _i < _c.length; _i++) { var pair = _c[_i]; d[first(pair)] = nth(pair, 1); } }
|
||
return makeCekValue(d, fenv, restK);
|
||
})() : (function() {
|
||
var nextEntry = first(remaining);
|
||
return makeCekState(nth(nextEntry, 1), fenv, kontPush(makeDictFrame(rest(remaining), append(completed, [[first(nextEntry)]]), fenv), restK));
|
||
})());
|
||
})();
|
||
})(); if (_m == "ho-setup") return (function() {
|
||
var hoType = get(frame, "ho-type");
|
||
var remaining = get(frame, "remaining");
|
||
var evaled = append(get(frame, "evaled"), [value]);
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(isEmpty(remaining)) ? hoSetupDispatch(hoType, evaled, fenv, restK) : makeCekState(first(remaining), fenv, kontPush(makeHoSetupFrame(hoType, rest(remaining), evaled, fenv), restK)));
|
||
})(); if (_m == "reset") return makeCekValue(value, env, restK); if (_m == "deref") return (function() {
|
||
var val = value;
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(!isSxTruthy(isSignal(val))) ? makeCekValue(val, fenv, restK) : (isSxTruthy(hasReactiveResetFrame_p(restK)) ? reactiveShiftDeref(val, fenv, restK) : ((function() {
|
||
var ctx = sxContext("sx-reactive", NIL);
|
||
return (isSxTruthy(ctx) ? (function() {
|
||
var depList = get(ctx, "deps");
|
||
var notifyFn = get(ctx, "notify");
|
||
return (isSxTruthy(!isSxTruthy(contains(depList, val))) ? (append_b(depList, val), signalAddSub(val, notifyFn)) : NIL);
|
||
})() : NIL);
|
||
})(), makeCekValue(signalValue(val), fenv, restK))));
|
||
})(); if (_m == "reactive-reset") return (function() {
|
||
var updateFn = get(frame, "update-fn");
|
||
var first_p = get(frame, "first-render");
|
||
if (isSxTruthy((isSxTruthy(updateFn) && !isSxTruthy(first_p)))) {
|
||
cekCall(updateFn, [value]);
|
||
}
|
||
return makeCekValue(value, env, restK);
|
||
})(); if (_m == "scope") return (function() {
|
||
var name = get(frame, "name");
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(isEmpty(remaining)) ? (scopePop(name), makeCekValue(value, fenv, restK)) : makeCekState(first(remaining), fenv, kontPush(makeScopeFrame(name, rest(remaining), fenv), restK)));
|
||
})(); if (_m == "provide") return (function() {
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(isEmpty(remaining)) ? (scopePop(get(frame, "name")), makeCekValue(value, fenv, restK)) : (function() {
|
||
var newFrame = makeProvideFrame(get(frame, "name"), get(frame, "value"), rest(remaining), fenv);
|
||
newFrame["subscribers"] = get(frame, "subscribers");
|
||
return makeCekState(first(remaining), fenv, kontPush(newFrame, restK));
|
||
})());
|
||
})(); if (_m == "bind") return (function() {
|
||
var tracked = _bindTracking_;
|
||
var body = get(frame, "body");
|
||
var fenv = get(frame, "env");
|
||
var prev = get(frame, "prev-tracking");
|
||
_bindTracking_ = prev;
|
||
(function() {
|
||
var subscriber = function(fireKont) { return cekRun(makeCekState(body, fenv, [])); };
|
||
return forEach(function(name) { return (function() {
|
||
var existing = get(_provideSubscribers_, name);
|
||
return dictSet(_provideSubscribers_, name, append((isSxTruthy(existing) ? existing : []), [subscriber]));
|
||
})(); }, tracked);
|
||
})();
|
||
return makeCekValue(value, fenv, restK);
|
||
})(); if (_m == "provide-set") return (function() {
|
||
var name = get(frame, "name");
|
||
var fenv = get(frame, "env");
|
||
var target = kontFindProvide(restK, name);
|
||
return (function() {
|
||
var oldVal = (isSxTruthy(target) ? get(target, "value") : scopePeek(name));
|
||
if (isSxTruthy(target)) {
|
||
target["value"] = value;
|
||
}
|
||
scopePop(name);
|
||
scopePush(name, value);
|
||
if (isSxTruthy(!isSxTruthy(sxEq(oldVal, value)))) {
|
||
fireProvideSubscribers(name);
|
||
}
|
||
return makeCekValue(value, fenv, restK);
|
||
})();
|
||
})(); if (_m == "scope-acc") return (function() {
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(isEmpty(remaining)) ? makeCekValue(value, fenv, restK) : makeCekState(first(remaining), fenv, kontPush((function() {
|
||
var newFrame = makeScopeAccFrame(get(frame, "name"), get(frame, "value"), rest(remaining), fenv);
|
||
newFrame["emitted"] = get(frame, "emitted");
|
||
return newFrame;
|
||
})(), restK)));
|
||
})(); if (_m == "map") return (function() {
|
||
var f = get(frame, "f");
|
||
var remaining = get(frame, "remaining");
|
||
var results = get(frame, "results");
|
||
var indexed = get(frame, "indexed");
|
||
var fenv = get(frame, "env");
|
||
return (function() {
|
||
var newResults = append(results, [value]);
|
||
return (isSxTruthy(isEmpty(remaining)) ? makeCekValue(newResults, fenv, restK) : (function() {
|
||
var callArgs = (isSxTruthy(indexed) ? [len(newResults), first(remaining)] : [first(remaining)]);
|
||
var nextFrame = (isSxTruthy(indexed) ? makeMapIndexedFrame(f, rest(remaining), newResults, fenv) : makeMapFrame(f, rest(remaining), newResults, fenv));
|
||
return continueWithCall(f, callArgs, fenv, [], kontPush(nextFrame, restK));
|
||
})());
|
||
})();
|
||
})(); if (_m == "filter") return (function() {
|
||
var f = get(frame, "f");
|
||
var remaining = get(frame, "remaining");
|
||
var results = get(frame, "results");
|
||
var currentItem = get(frame, "current-item");
|
||
var fenv = get(frame, "env");
|
||
return (function() {
|
||
var newResults = (isSxTruthy(value) ? append(results, [currentItem]) : results);
|
||
return (isSxTruthy(isEmpty(remaining)) ? makeCekValue(newResults, fenv, restK) : continueWithCall(f, [first(remaining)], fenv, [], kontPush(makeFilterFrame(f, rest(remaining), newResults, first(remaining), fenv), restK)));
|
||
})();
|
||
})(); if (_m == "reduce") return (function() {
|
||
var f = get(frame, "f");
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(isEmpty(remaining)) ? makeCekValue(value, fenv, restK) : continueWithCall(f, [value, first(remaining)], fenv, [], kontPush(makeReduceFrame(f, rest(remaining), fenv), restK)));
|
||
})(); if (_m == "for-each") return (function() {
|
||
var f = get(frame, "f");
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(isEmpty(remaining)) ? makeCekValue(NIL, fenv, restK) : continueWithCall(f, [first(remaining)], fenv, [], kontPush(makeForEachFrame(f, rest(remaining), fenv), restK)));
|
||
})(); if (_m == "some") return (function() {
|
||
var f = get(frame, "f");
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(value) ? makeCekValue(value, fenv, restK) : (isSxTruthy(isEmpty(remaining)) ? makeCekValue(false, fenv, restK) : continueWithCall(f, [first(remaining)], fenv, [], kontPush(makeSomeFrame(f, rest(remaining), fenv), restK))));
|
||
})(); if (_m == "every") return (function() {
|
||
var f = get(frame, "f");
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(!isSxTruthy(value)) ? makeCekValue(false, fenv, restK) : (isSxTruthy(isEmpty(remaining)) ? makeCekValue(true, fenv, restK) : continueWithCall(f, [first(remaining)], fenv, [], kontPush(makeEveryFrame(f, rest(remaining), fenv), restK))));
|
||
})(); if (_m == "handler") return (function() {
|
||
var remaining = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(isEmpty(remaining)) ? makeCekValue(value, fenv, restK) : makeCekState(first(remaining), fenv, kontPush(makeHandlerFrame(get(frame, "f"), rest(remaining), fenv), restK)));
|
||
})(); if (_m == "restart") return makeCekValue(value, env, restK); if (_m == "signal-return") return (function() {
|
||
var savedKont = get(frame, "saved-kont");
|
||
return makeCekValue(value, get(frame, "env"), savedKont);
|
||
})(); if (_m == "comp-trace") return makeCekValue(value, env, restK); if (_m == "cond-arrow") return (function() {
|
||
var testValue = get(frame, "match-val");
|
||
var fenv = get(frame, "env");
|
||
return continueWithCall(value, [testValue], fenv, [testValue], restK);
|
||
})(); if (_m == "wind-after") return (function() {
|
||
var afterThunk = get(frame, "after-thunk");
|
||
var windersLen = get(frame, "winders-len");
|
||
var bodyResult = value;
|
||
var fenv = get(frame, "env");
|
||
return ((isSxTruthy((len(_winders_) > windersLen)) ? (_winders_ = rest(_winders_)) : NIL), continueWithCall(afterThunk, [], fenv, [], kontPush(makeWindReturnFrame(bodyResult, fenv), restK)));
|
||
})(); if (_m == "wind-return") return makeCekValue(get(frame, "body-result"), get(frame, "env"), restK); if (_m == "raise-eval") return (function() {
|
||
var condition = value;
|
||
var fenv = get(frame, "env");
|
||
var continuable_p = get(frame, "scheme");
|
||
var unwindResult = kontUnwindToHandler(restK, condition);
|
||
var handlerFn = get(unwindResult, "handler");
|
||
var unwoundK = get(unwindResult, "kont");
|
||
return (isSxTruthy(isNil(handlerFn)) ? ((_lastErrorKont_ = unwoundK), hostError((String("Unhandled exception: ") + String(inspect(condition))))) : continueWithCall(handlerFn, [condition], fenv, [condition], (isSxTruthy(continuable_p) ? kontPush(makeSignalReturnFrame(fenv, unwoundK), unwoundK) : kontPush(makeRaiseGuardFrame(fenv, unwoundK), unwoundK))));
|
||
})(); if (_m == "raise-guard") return ((_lastErrorKont_ = restK), hostError("exception handler returned from non-continuable raise")); if (_m == "multi-map") return (function() {
|
||
var f = get(frame, "f");
|
||
var remaining = get(frame, "remaining");
|
||
var newResults = append(get(frame, "results"), [value]);
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(some(function(c) { return isEmpty(c); }, remaining)) ? makeCekValue(newResults, fenv, restK) : (function() {
|
||
var heads = map(function(c) { return first(c); }, remaining);
|
||
var tails = map(function(c) { return rest(c); }, remaining);
|
||
return continueWithCall(f, heads, fenv, [], kontPush(makeMultiMapFrame(f, tails, newResults, fenv), restK));
|
||
})());
|
||
})(); if (_m == "callcc") return (function() {
|
||
var k = makeCallccContinuation(restK, len(_winders_));
|
||
return continueWithCall(value, [k], get(frame, "env"), [k], restK);
|
||
})(); if (_m == "vm-resume") return (function() {
|
||
var resumeFn = get(frame, "f");
|
||
return (function() {
|
||
var result = apply(resumeFn, [value]);
|
||
return (isSxTruthy((isSxTruthy(isDict(result)) && get(result, "__vm_suspended"))) ? makeCekSuspended(get(result, "request"), get(frame, "env"), kontPush(makeVmResumeFrame(get(result, "resume"), get(frame, "env")), restK)) : makeCekValue(result, get(frame, "env"), restK));
|
||
})();
|
||
})(); if (_m == "perform") return makeCekSuspended(value, get(frame, "env"), restK); if (_m == "import") return (function() {
|
||
var importSet = get(frame, "args");
|
||
var remainingSets = get(frame, "remaining");
|
||
var fenv = get(frame, "env");
|
||
return (bindImportSet(importSet, fenv), (isSxTruthy(isEmpty(remainingSets)) ? makeCekValue(NIL, fenv, restK) : stepSfImport(remainingSets, fenv, restK)));
|
||
})(); if (_m == "parameterize") return (function() {
|
||
var remaining = get(frame, "remaining");
|
||
var currentParam = get(frame, "f");
|
||
var results = get(frame, "results");
|
||
var body = get(frame, "body");
|
||
var fenv = get(frame, "env");
|
||
return (isSxTruthy(isNil(currentParam)) ? (function() {
|
||
var paramObj = value;
|
||
var valExpr = nth(first(remaining), 1);
|
||
return makeCekState(valExpr, fenv, kontPush(makeParameterizeFrame(remaining, paramObj, results, body, fenv), restK));
|
||
})() : (function() {
|
||
var convertedVal = value;
|
||
var newResults = append(results, [[parameterUid(currentParam), convertedVal]]);
|
||
var restBindings = rest(remaining);
|
||
return (isSxTruthy(isEmpty(restBindings)) ? (function() {
|
||
var bodyExpr = (isSxTruthy(sxEq(len(body), 1)) ? first(body) : cons(new Symbol("begin"), body));
|
||
var provideKont = kontPushProvides(newResults, fenv, restK);
|
||
return makeCekState(bodyExpr, fenv, provideKont);
|
||
})() : makeCekState(first(first(restBindings)), fenv, kontPush(makeParameterizeFrame(restBindings, NIL, newResults, body, fenv), restK)));
|
||
})());
|
||
})(); return ((_lastErrorKont_ = restK), error((String("Unknown frame type: ") + String(ft)))); })();
|
||
})());
|
||
})(); };
|
||
PRIMITIVES["step-continue"] = stepContinue;
|
||
|
||
// continue-with-call
|
||
var continueWithCall = function(f, args, env, rawArgs, kont) { return (isSxTruthy(parameter_p(f)) ? (function() {
|
||
var uid = parameterUid(f);
|
||
var frame = kontFindProvide(kont, uid);
|
||
return makeCekValue((isSxTruthy(frame) ? get(frame, "value") : parameterDefault(f)), env, kont);
|
||
})() : (isSxTruthy(callccContinuation_p(f)) ? (function() {
|
||
var arg = (isSxTruthy(isEmpty(args)) ? NIL : first(args));
|
||
var captured = callccContinuationData(f);
|
||
var wLen = callccContinuationWindersLen(f);
|
||
return (windEscapeTo(wLen), makeCekValue(arg, env, captured));
|
||
})() : (isSxTruthy(continuation_p(f)) ? (function() {
|
||
var arg = (isSxTruthy(isEmpty(args)) ? NIL : first(args));
|
||
var contData = continuationData(f);
|
||
return (function() {
|
||
var captured = get(contData, "captured");
|
||
return (function() {
|
||
var result = cekRun(makeCekValue(arg, env, captured));
|
||
return makeCekValue(result, env, kont);
|
||
})();
|
||
})();
|
||
})() : (isSxTruthy((isSxTruthy(isCallable(f)) && isSxTruthy(!isSxTruthy(isLambda(f))) && isSxTruthy(!isSxTruthy(isComponent(f))) && !isSxTruthy(isIsland(f)))) ? (function() {
|
||
var result = sxApplyCek(f, args);
|
||
return (isSxTruthy(evalError_p(result)) ? makeCekValue(get(result, "message"), env, kontPush(makeRaiseEvalFrame(env, false), kont)) : (isSxTruthy((isSxTruthy(isDict(result)) && get(result, "__vm_suspended"))) ? makeCekSuspended(get(result, "request"), env, kontPush(makeVmResumeFrame(get(result, "resume"), env), kont)) : makeCekValue(result, env, kont)));
|
||
})() : (isSxTruthy(isLambda(f)) ? (function() {
|
||
var params = lambdaParams(f);
|
||
var local = envMerge(lambdaClosure(f), env);
|
||
if (isSxTruthy(!isSxTruthy(bindLambdaParams(params, args, local)))) {
|
||
if (isSxTruthy((len(args) > len(params)))) {
|
||
error((String(sxOr(lambdaName(f), "lambda")) + String(" expects ") + String(len(params)) + String(" args, got ") + String(len(args))));
|
||
}
|
||
{ var _c = zip(params, args); for (var _i = 0; _i < _c.length; _i++) { var pair = _c[_i]; envBind(local, first(pair), nth(pair, 1)); } }
|
||
{ var _c = slice(params, len(args)); for (var _i = 0; _i < _c.length; _i++) { var p = _c[_i]; envBind(local, p, NIL); } }
|
||
}
|
||
return (function() {
|
||
var jitResult = jitTryCall(f, args);
|
||
return (isSxTruthy(jitSkip_p(jitResult)) ? makeCekState(lambdaBody(f), local, kont) : (isSxTruthy((isSxTruthy(isDict(jitResult)) && get(jitResult, "__vm_suspended"))) ? makeCekSuspended(get(jitResult, "request"), env, kontPush(makeVmResumeFrame(get(jitResult, "resume"), env), kont)) : makeCekValue(jitResult, local, kont)));
|
||
})();
|
||
})() : (isSxTruthy(sxOr(isComponent(f), isIsland(f))) ? (function() {
|
||
var parsed = parseKeywordArgs(rawArgs, env);
|
||
var kwargs = first(parsed);
|
||
var children = nth(parsed, 1);
|
||
var local = envMerge(componentClosure(f), env);
|
||
{ var _c = componentParams(f); for (var _i = 0; _i < _c.length; _i++) { var p = _c[_i]; envBind(local, p, sxOr(dictGet(kwargs, p), NIL)); } }
|
||
if (isSxTruthy(componentHasChildren(f))) {
|
||
envBind(local, "children", children);
|
||
}
|
||
return makeCekState(componentBody(f), local, kontPush(makeCompTraceFrame(componentName(f), componentFile(f)), kont));
|
||
})() : error((String("Not callable: ") + String(inspect(f)))))))))); };
|
||
PRIMITIVES["continue-with-call"] = continueWithCall;
|
||
|
||
// sf-case-step-loop
|
||
var sfCaseStepLoop = function(matchVal, clauses, env, kont) { return (isSxTruthy((len(clauses) < 2)) ? makeCekValue(NIL, env, kont) : (function() {
|
||
var test = first(clauses);
|
||
var body = nth(clauses, 1);
|
||
return (isSxTruthy(isElseClause(test)) ? makeCekState(body, env, kont) : (function() {
|
||
var testVal = trampoline(evalExpr(test, env));
|
||
return (isSxTruthy(sxEq(matchVal, testVal)) ? makeCekState(body, env, kont) : sfCaseStepLoop(matchVal, slice(clauses, 2), env, kont));
|
||
})());
|
||
})()); };
|
||
PRIMITIVES["sf-case-step-loop"] = sfCaseStepLoop;
|
||
|
||
// eval-expr-cek
|
||
var evalExprCek = function(expr, env) { return cekRun(makeCekState(expr, env, [])); };
|
||
PRIMITIVES["eval-expr-cek"] = evalExprCek;
|
||
|
||
// trampoline-cek
|
||
var trampolineCek = function(val) { return (isSxTruthy(isThunk(val)) ? evalExprCek(thunkExpr(val), thunkEnv(val)) : val); };
|
||
PRIMITIVES["trampoline-cek"] = trampolineCek;
|
||
|
||
// make-coroutine
|
||
var makeCoroutine = function(thunk) { return {"suspension": NIL, "thunk": thunk, "type": "coroutine", "state": "ready"}; };
|
||
PRIMITIVES["make-coroutine"] = makeCoroutine;
|
||
|
||
// eval-expr
|
||
var evalExpr = function(expr, env) { return cekRun(makeCekState(expr, env, [])); };
|
||
PRIMITIVES["eval-expr"] = evalExpr;
|
||
|
||
// trampoline
|
||
var trampoline = function(val) { return (isSxTruthy(isThunk(val)) ? evalExpr(thunkExpr(val), thunkEnv(val)) : val); };
|
||
PRIMITIVES["trampoline"] = trampoline;
|
||
|
||
// *gensym-counter*
|
||
var _gensymCounter_ = 0;
|
||
PRIMITIVES["*gensym-counter*"] = _gensymCounter_;
|
||
|
||
// gensym
|
||
var gensym = function() { var args = Array.prototype.slice.call(arguments, 0); return (function() {
|
||
var prefix = (isSxTruthy(isEmpty(args)) ? "g" : (String(first(args))));
|
||
return ((_gensymCounter_ = (_gensymCounter_ + 1)), makeSymbol((String(prefix) + String(_gensymCounter_))));
|
||
})(); };
|
||
PRIMITIVES["gensym"] = gensym;
|
||
|
||
// string->symbol
|
||
var stringToSymbol = function(s) { return makeSymbol(s); };
|
||
PRIMITIVES["string->symbol"] = stringToSymbol;
|
||
|
||
// symbol->string
|
||
var symbolToString = function(sym) { return symbolName(sym); };
|
||
PRIMITIVES["symbol->string"] = symbolToString;
|
||
|
||
// intern
|
||
var intern = function(s) { return makeSymbol(s); };
|
||
PRIMITIVES["intern"] = intern;
|
||
|
||
// symbol-interned?
|
||
var symbolInterned_p = function(sym) { return true; };
|
||
PRIMITIVES["symbol-interned?"] = symbolInterned_p;
|
||
|
||
// integer->char
|
||
var integerToChar = makeChar;
|
||
PRIMITIVES["integer->char"] = integerToChar;
|
||
|
||
|
||
// === Transpiled from freeze (serializable state boundaries) ===
|
||
|
||
|
||
|
||
// === Transpiled from content (content-addressed computation) ===
|
||
|
||
|
||
|
||
// === Transpiled from render (core) ===
|
||
|
||
|
||
|
||
// === Transpiled from web-forms (defstyle, deftype, defeffect, defrelation) ===
|
||
|
||
// parse-key-params
|
||
var parseKeyParams = function(paramsExpr) { return (function() {
|
||
var params = [];
|
||
{ var _c = paramsExpr; for (var _i = 0; _i < _c.length; _i++) { var p = _c[_i]; if (isSxTruthy(sxEq(typeOf(p), "symbol"))) {
|
||
(function() {
|
||
var name = symbolName(p);
|
||
return (isSxTruthy(!isSxTruthy(sxEq(name, "&key"))) ? append_b(params, name) : NIL);
|
||
})();
|
||
} } }
|
||
return params;
|
||
})(); };
|
||
PRIMITIVES["parse-key-params"] = parseKeyParams;
|
||
|
||
// parse-handler-args
|
||
var parseHandlerArgs = function(args) { return (function() {
|
||
var opts = {};
|
||
var params = [];
|
||
var body = NIL;
|
||
var i = 0;
|
||
var n = len(args);
|
||
var done = false;
|
||
{ var _c = range(0, n); for (var _i = 0; _i < _c.length; _i++) { var idx = _c[_i]; if (isSxTruthy((isSxTruthy(!isSxTruthy(done)) && sxEq(idx, i)))) {
|
||
(function() {
|
||
var arg = nth(args, idx);
|
||
return (isSxTruthy(sxEq(typeOf(arg), "keyword")) ? ((isSxTruthy(((idx + 1) < n)) ? (function() {
|
||
var val = nth(args, (idx + 1));
|
||
return dictSet(opts, keywordName(arg), (isSxTruthy(sxEq(typeOf(val), "keyword")) ? keywordName(val) : val));
|
||
})() : NIL), (i = (idx + 2))) : (isSxTruthy(sxEq(typeOf(arg), "list")) ? ((params = parseKeyParams(arg)), (isSxTruthy(((idx + 1) < n)) ? (body = nth(args, (idx + 1))) : NIL), (done = true)) : ((body = arg), (done = true))));
|
||
})();
|
||
} } }
|
||
return {["opts"]: opts, ["params"]: params, ["body"]: body};
|
||
})(); };
|
||
PRIMITIVES["parse-handler-args"] = parseHandlerArgs;
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("defhandler", function(args, env) { return (function() {
|
||
var nameSym = first(args);
|
||
var name = symbolName(first(args));
|
||
var parsed = parseHandlerArgs(rest(args));
|
||
var opts = get(parsed, "opts");
|
||
var params = get(parsed, "params");
|
||
var body = get(parsed, "body");
|
||
return (function() {
|
||
var hdef = {["__type"]: "handler", ["name"]: name, ["params"]: params, ["body"]: body, ["closure"]: env, ["path"]: sxOr(get(opts, "path"), NIL), ["method"]: sxOr(get(opts, "method"), "get"), ["csrf"]: (function() {
|
||
var v = get(opts, "csrf");
|
||
return (isSxTruthy(isNil(v)) ? true : v);
|
||
})(), ["returns"]: sxOr(get(opts, "returns"), "element")};
|
||
envBind(env, (String("handler:") + String(name)), hdef);
|
||
return hdef;
|
||
})();
|
||
})(); });
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("defquery", function(args, env) { return (function() {
|
||
var name = symbolName(first(args));
|
||
var paramsRaw = nth(args, 1);
|
||
var params = parseKeyParams(paramsRaw);
|
||
var hasDoc = (isSxTruthy((len(args) >= 4)) && sxEq(typeOf(nth(args, 2)), "string"));
|
||
var doc = (isSxTruthy(hasDoc) ? nth(args, 2) : "");
|
||
var body = (isSxTruthy(hasDoc) ? nth(args, 3) : nth(args, 2));
|
||
return (function() {
|
||
var qdef = {["__type"]: "query", ["name"]: name, ["params"]: params, ["doc"]: doc, ["body"]: body, ["closure"]: env};
|
||
envBind(env, (String("query:") + String(name)), qdef);
|
||
return qdef;
|
||
})();
|
||
})(); });
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("defaction", function(args, env) { return (function() {
|
||
var name = symbolName(first(args));
|
||
var paramsRaw = nth(args, 1);
|
||
var params = parseKeyParams(paramsRaw);
|
||
var hasDoc = (isSxTruthy((len(args) >= 4)) && sxEq(typeOf(nth(args, 2)), "string"));
|
||
var doc = (isSxTruthy(hasDoc) ? nth(args, 2) : "");
|
||
var body = (isSxTruthy(hasDoc) ? nth(args, 3) : nth(args, 2));
|
||
return (function() {
|
||
var adef = {["__type"]: "action", ["name"]: name, ["params"]: params, ["doc"]: doc, ["body"]: body, ["closure"]: env};
|
||
envBind(env, (String("action:") + String(name)), adef);
|
||
return adef;
|
||
})();
|
||
})(); });
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("defpage", function(args, env) { return (function() {
|
||
var name = symbolName(first(args));
|
||
var slots = {};
|
||
var n = len(args);
|
||
{ var _c = range(0, ((n - 1) / 2)); for (var _i = 0; _i < _c.length; _i++) { var idx = _c[_i]; (function() {
|
||
var kIdx = (1 + (idx * 2));
|
||
var vIdx = (2 + (idx * 2));
|
||
return (isSxTruthy((isSxTruthy((kIdx < n)) && isSxTruthy((vIdx < n)) && sxEq(typeOf(nth(args, kIdx)), "keyword"))) ? dictSet(slots, keywordName(nth(args, kIdx)), nth(args, vIdx)) : NIL);
|
||
})(); } }
|
||
return (function() {
|
||
var pdef = {["__type"]: "page", ["name"]: name, ["path"]: sxOr(get(slots, "path"), ""), ["auth"]: sxOr(get(slots, "auth"), "public"), ["layout"]: get(slots, "layout"), ["data"]: get(slots, "data"), ["content"]: get(slots, "content"), ["filter"]: get(slots, "filter"), ["aside"]: get(slots, "aside"), ["menu"]: get(slots, "menu"), ["stream"]: get(slots, "stream"), ["fallback"]: get(slots, "fallback"), ["shell"]: get(slots, "shell"), ["closure"]: env};
|
||
envBind(env, (String("page:") + String(name)), pdef);
|
||
return pdef;
|
||
})();
|
||
})(); });
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("defrelation", function(args, env) { return (function() {
|
||
var name = symbolName(first(args));
|
||
var slots = {};
|
||
var n = len(args);
|
||
{ var _c = range(0, ((n - 1) / 2)); for (var _i = 0; _i < _c.length; _i++) { var idx = _c[_i]; (function() {
|
||
var kIdx = (1 + (idx * 2));
|
||
var vIdx = (2 + (idx * 2));
|
||
return (isSxTruthy((isSxTruthy((kIdx < n)) && isSxTruthy((vIdx < n)) && sxEq(typeOf(nth(args, kIdx)), "keyword"))) ? dictSet(slots, keywordName(nth(args, kIdx)), nth(args, vIdx)) : NIL);
|
||
})(); } }
|
||
return (function() {
|
||
var rdef = {["__type"]: "relation", ["name"]: name, ["slots"]: slots, ["closure"]: env};
|
||
envBind(env, (String("relation:") + String(name)), rdef);
|
||
return rdef;
|
||
})();
|
||
})(); });
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("defstyle", function(args, env) { return (function() {
|
||
var nameSym = first(args);
|
||
var value = trampoline(evalExpr(nth(args, 1), env));
|
||
envBind(env, symbolName(nameSym), value);
|
||
return value;
|
||
})(); });
|
||
|
||
// normalize-type-body
|
||
var normalizeTypeBody = function(body) { return (isSxTruthy(isNil(body)) ? "nil" : (isSxTruthy(sxEq(typeOf(body), "symbol")) ? symbolName(body) : (isSxTruthy(sxEq(typeOf(body), "string")) ? body : (isSxTruthy(sxEq(typeOf(body), "keyword")) ? keywordName(body) : (isSxTruthy(sxEq(typeOf(body), "dict")) ? mapDict(function(k, v) { return normalizeTypeBody(v); }, body) : (isSxTruthy(sxEq(typeOf(body), "list")) ? (isSxTruthy(isEmpty(body)) ? "any" : (function() {
|
||
var headName = (isSxTruthy(sxEq(typeOf(first(body)), "symbol")) ? symbolName(first(body)) : (String(first(body))));
|
||
return (isSxTruthy(sxEq(headName, "union")) ? cons("or", map(normalizeTypeBody, rest(body))) : cons(headName, map(normalizeTypeBody, rest(body))));
|
||
})()) : (String(body)))))))); };
|
||
PRIMITIVES["normalize-type-body"] = normalizeTypeBody;
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("deftype", function(args, env) { return (function() {
|
||
var nameOrForm = first(args);
|
||
var bodyExpr = nth(args, 1);
|
||
var typeName = NIL;
|
||
var typeParams = [];
|
||
(isSxTruthy(sxEq(typeOf(nameOrForm), "symbol")) ? (typeName = symbolName(nameOrForm)) : (isSxTruthy(sxEq(typeOf(nameOrForm), "list")) ? ((typeName = symbolName(first(nameOrForm))), (typeParams = map(function(p) { return (isSxTruthy(sxEq(typeOf(p), "symbol")) ? symbolName(p) : (String(p))); }, rest(nameOrForm)))) : NIL));
|
||
return (function() {
|
||
var body = normalizeTypeBody(bodyExpr);
|
||
var registry = (isSxTruthy(envHas(env, "*type-registry*")) ? envGet(env, "*type-registry*") : {});
|
||
registry[typeName] = {"body": body, "params": typeParams, "name": typeName};
|
||
envBind(env, "*type-registry*", registry);
|
||
return NIL;
|
||
})();
|
||
})(); });
|
||
|
||
// (register-special-form! ...)
|
||
registerSpecialForm("defeffect", function(args, env) { return (function() {
|
||
var effectName = (isSxTruthy(sxEq(typeOf(first(args)), "symbol")) ? symbolName(first(args)) : (String(first(args))));
|
||
var registry = (isSxTruthy(envHas(env, "*effect-registry*")) ? envGet(env, "*effect-registry*") : []);
|
||
if (isSxTruthy(!isSxTruthy(contains(registry, effectName)))) {
|
||
registry.push(effectName);
|
||
}
|
||
envBind(env, "*effect-registry*", registry);
|
||
return NIL;
|
||
})(); });
|
||
|
||
// WEB_FORM_NAMES
|
||
var WEB_FORM_NAMES = ["defhandler", "defpage", "defquery", "defaction", "defrelation", "defstyle", "deftype", "defeffect"];
|
||
PRIMITIVES["WEB_FORM_NAMES"] = WEB_FORM_NAMES;
|
||
|
||
|
||
// === Transpiled from parser ===
|
||
|
||
// sx-parse
|
||
var sxParse = function(source) { return (function() {
|
||
var pos = 0;
|
||
var lenSrc = len(source);
|
||
var skipComment = function() { while(true) { if (isSxTruthy((isSxTruthy((pos < lenSrc)) && !isSxTruthy(sxEq(nth(source, pos), "\n"))))) { pos = (pos + 1);
|
||
continue; } else { return NIL; } } };
|
||
PRIMITIVES["skip-comment"] = skipComment;
|
||
var skipWs = function() { while(true) { if (isSxTruthy((pos < lenSrc))) { { var ch = nth(source, pos);
|
||
if (isSxTruthy(sxOr(sxEq(ch, " "), sxEq(ch, "\t"), sxEq(ch, "\n"), sxEq(ch, "\r")))) { pos = (pos + 1);
|
||
continue; } else if (isSxTruthy(sxEq(ch, ";"))) { pos = (pos + 1);
|
||
skipComment();
|
||
continue; } else { return NIL; } } } else { return NIL; } } };
|
||
PRIMITIVES["skip-ws"] = skipWs;
|
||
var hexDigitValue = function(ch) { return indexOf_("0123456789abcdef", lower(ch)); };
|
||
PRIMITIVES["hex-digit-value"] = hexDigitValue;
|
||
var readString = function() { pos = (pos + 1);
|
||
return (function() {
|
||
var buf = "";
|
||
var readStrLoop = function() { while(true) { if (isSxTruthy((pos >= lenSrc))) { return error("Unterminated string"); } else { { var ch = nth(source, pos);
|
||
if (isSxTruthy(sxEq(ch, "\""))) { pos = (pos + 1);
|
||
return NIL; } else if (isSxTruthy(sxEq(ch, "\\"))) { pos = (pos + 1);
|
||
{ var esc = nth(source, pos);
|
||
if (isSxTruthy(sxEq(esc, "u"))) { pos = (pos + 1);
|
||
{ var d0 = hexDigitValue(nth(source, pos));
|
||
var _ = (pos = (pos + 1));
|
||
var d1 = hexDigitValue(nth(source, pos));
|
||
var _ = (pos = (pos + 1));
|
||
var d2 = hexDigitValue(nth(source, pos));
|
||
var _ = (pos = (pos + 1));
|
||
var d3 = hexDigitValue(nth(source, pos));
|
||
var _ = (pos = (pos + 1));
|
||
buf = (String(buf) + String(charFromCode(((((d0 * 4096) + (d1 * 256)) + (d2 * 16)) + d3))));
|
||
continue; } } else { buf = (String(buf) + String((isSxTruthy(sxEq(esc, "n")) ? "\n" : (isSxTruthy(sxEq(esc, "t")) ? "\t" : (isSxTruthy(sxEq(esc, "r")) ? "\r" : esc)))));
|
||
pos = (pos + 1);
|
||
continue; } } } else { buf = (String(buf) + String(ch));
|
||
pos = (pos + 1);
|
||
continue; } } } } };
|
||
PRIMITIVES["read-str-loop"] = readStrLoop;
|
||
readStrLoop();
|
||
return buf;
|
||
})(); };
|
||
PRIMITIVES["read-string"] = readString;
|
||
var readIdent = function() { return (function() {
|
||
var start = pos;
|
||
var readIdentLoop = function() { while(true) { if (isSxTruthy((isSxTruthy((pos < lenSrc)) && isIdentChar(nth(source, pos))))) { pos = (pos + 1);
|
||
continue; } else { return NIL; } } };
|
||
PRIMITIVES["read-ident-loop"] = readIdentLoop;
|
||
readIdentLoop();
|
||
return slice(source, start, pos);
|
||
})(); };
|
||
PRIMITIVES["read-ident"] = readIdent;
|
||
var readKeyword = function() { pos = (pos + 1);
|
||
return makeKeyword(readIdent()); };
|
||
PRIMITIVES["read-keyword"] = readKeyword;
|
||
var readNumber = function() { return (function() {
|
||
var start = pos;
|
||
if (isSxTruthy((isSxTruthy((pos < lenSrc)) && sxEq(nth(source, pos), "-")))) {
|
||
pos = (pos + 1);
|
||
}
|
||
var readDigits = function() { while(true) { if (isSxTruthy((isSxTruthy((pos < lenSrc)) && (function() {
|
||
var c = nth(source, pos);
|
||
return (isSxTruthy((c >= "0")) && (c <= "9"));
|
||
})()))) { pos = (pos + 1);
|
||
continue; } else { return NIL; } } };
|
||
PRIMITIVES["read-digits"] = readDigits;
|
||
readDigits();
|
||
return (isSxTruthy((isSxTruthy((pos < lenSrc)) && isSxTruthy(sxEq(nth(source, pos), "/")) && isSxTruthy(((pos + 1) < lenSrc)) && (function() {
|
||
var nc = nth(source, (pos + 1));
|
||
return (isSxTruthy((nc >= "0")) && (nc <= "9"));
|
||
})())) ? (function() {
|
||
var numer = parseNumber(slice(source, start, pos));
|
||
pos = (pos + 1);
|
||
return (function() {
|
||
var denomStart = pos;
|
||
readDigits();
|
||
return makeRational(numer, parseNumber(slice(source, denomStart, pos)));
|
||
})();
|
||
})() : ((isSxTruthy((isSxTruthy((pos < lenSrc)) && sxEq(nth(source, pos), "."))) ? ((pos = (pos + 1)), readDigits()) : NIL), (isSxTruthy((isSxTruthy((pos < lenSrc)) && sxOr(sxEq(nth(source, pos), "e"), sxEq(nth(source, pos), "E")))) ? ((pos = (pos + 1)), (isSxTruthy((isSxTruthy((pos < lenSrc)) && sxOr(sxEq(nth(source, pos), "+"), sxEq(nth(source, pos), "-")))) ? (pos = (pos + 1)) : NIL), readDigits()) : NIL), parseNumber(slice(source, start, pos))));
|
||
})(); };
|
||
PRIMITIVES["read-number"] = readNumber;
|
||
var readSymbol = function() { return (function() {
|
||
var name = readIdent();
|
||
return (isSxTruthy(sxEq(name, "true")) ? true : (isSxTruthy(sxEq(name, "false")) ? false : (isSxTruthy(sxEq(name, "nil")) ? NIL : makeSymbol(name))));
|
||
})(); };
|
||
PRIMITIVES["read-symbol"] = readSymbol;
|
||
var readList = function(closeCh) { return (function() {
|
||
var items = [];
|
||
var readListLoop = function() { while(true) { skipWs();
|
||
if (isSxTruthy((pos >= lenSrc))) { return error("Unterminated list"); } else { if (isSxTruthy(sxEq(nth(source, pos), closeCh))) { pos = (pos + 1);
|
||
return NIL; } else { items.push(readExpr());
|
||
continue; } } } };
|
||
PRIMITIVES["read-list-loop"] = readListLoop;
|
||
readListLoop();
|
||
return items;
|
||
})(); };
|
||
PRIMITIVES["read-list"] = readList;
|
||
var readMap = function() { return (function() {
|
||
var result = {};
|
||
var readMapLoop = function() { while(true) { skipWs();
|
||
if (isSxTruthy((pos >= lenSrc))) { return error("Unterminated map"); } else { if (isSxTruthy(sxEq(nth(source, pos), "}"))) { pos = (pos + 1);
|
||
return NIL; } else { { var keyExpr = readExpr();
|
||
var keyStr = (isSxTruthy(sxEq(typeOf(keyExpr), "keyword")) ? keywordName(keyExpr) : (String(keyExpr)));
|
||
var valExpr = readExpr();
|
||
result[keyStr] = valExpr;
|
||
continue; } } } } };
|
||
PRIMITIVES["read-map-loop"] = readMapLoop;
|
||
readMapLoop();
|
||
return result;
|
||
})(); };
|
||
PRIMITIVES["read-map"] = readMap;
|
||
var readRawString = function() { return (function() {
|
||
var buf = "";
|
||
var rawLoop = function() { while(true) { if (isSxTruthy((pos >= lenSrc))) { return error("Unterminated raw string"); } else { { var ch = nth(source, pos);
|
||
if (isSxTruthy(sxEq(ch, "|"))) { pos = (pos + 1);
|
||
return NIL; } else { buf = (String(buf) + String(ch));
|
||
pos = (pos + 1);
|
||
continue; } } } } };
|
||
PRIMITIVES["raw-loop"] = rawLoop;
|
||
rawLoop();
|
||
return buf;
|
||
})(); };
|
||
PRIMITIVES["read-raw-string"] = readRawString;
|
||
var readCharLiteral = function() { return (isSxTruthy((pos >= lenSrc)) ? error("Unexpected end of input after #\\") : (function() {
|
||
var firstCh = nth(source, pos);
|
||
return (isSxTruthy(isIdentStart(firstCh)) ? (function() {
|
||
var charStart = pos;
|
||
var readCharNameLoop = function() { while(true) { if (isSxTruthy((isSxTruthy((pos < lenSrc)) && isIdentChar(nth(source, pos))))) { pos = (pos + 1);
|
||
continue; } else { return NIL; } } };
|
||
PRIMITIVES["read-char-name-loop"] = readCharNameLoop;
|
||
readCharNameLoop();
|
||
return (function() {
|
||
var charName = slice(source, charStart, pos);
|
||
return makeChar((isSxTruthy(sxEq(charName, "space")) ? 32 : (isSxTruthy(sxEq(charName, "newline")) ? 10 : (isSxTruthy(sxEq(charName, "tab")) ? 9 : (isSxTruthy(sxEq(charName, "nul")) ? 0 : (isSxTruthy(sxEq(charName, "null")) ? 0 : (isSxTruthy(sxEq(charName, "return")) ? 13 : (isSxTruthy(sxEq(charName, "escape")) ? 27 : (isSxTruthy(sxEq(charName, "delete")) ? 127 : (isSxTruthy(sxEq(charName, "backspace")) ? 8 : (isSxTruthy(sxEq(charName, "altmode")) ? 27 : (isSxTruthy(sxEq(charName, "rubout")) ? 127 : charCode(firstCh)))))))))))));
|
||
})();
|
||
})() : ((pos = (pos + 1)), makeChar(charCode(firstCh))));
|
||
})()); };
|
||
PRIMITIVES["read-char-literal"] = readCharLiteral;
|
||
var readExpr = function() { while(true) { skipWs();
|
||
if (isSxTruthy((pos >= lenSrc))) { return error("Unexpected end of input"); } else { { var ch = nth(source, pos);
|
||
if (isSxTruthy(sxEq(ch, "("))) { pos = (pos + 1);
|
||
return readList(")"); } else if (isSxTruthy(sxEq(ch, "["))) { pos = (pos + 1);
|
||
return readList("]"); } else if (isSxTruthy(sxEq(ch, "{"))) { pos = (pos + 1);
|
||
return readMap(); } else if (isSxTruthy(sxEq(ch, "\""))) { return readString(); } else if (isSxTruthy(sxEq(ch, ":"))) { return readKeyword(); } else if (isSxTruthy(sxEq(ch, "'"))) { pos = (pos + 1);
|
||
return [makeSymbol("quote"), readExpr()]; } else if (isSxTruthy(sxEq(ch, "`"))) { pos = (pos + 1);
|
||
return [makeSymbol("quasiquote"), readExpr()]; } else if (isSxTruthy(sxEq(ch, ","))) { pos = (pos + 1);
|
||
if (isSxTruthy((isSxTruthy((pos < lenSrc)) && sxEq(nth(source, pos), "@")))) { pos = (pos + 1);
|
||
return [makeSymbol("splice-unquote"), readExpr()]; } else { return [makeSymbol("unquote"), readExpr()]; } } else if (isSxTruthy(sxEq(ch, "#"))) { pos = (pos + 1);
|
||
if (isSxTruthy((pos >= lenSrc))) { return error("Unexpected end of input after #"); } else { { var dispatchCh = nth(source, pos);
|
||
if (isSxTruthy(sxEq(dispatchCh, ";"))) { pos = (pos + 1);
|
||
readExpr();
|
||
continue; } else if (isSxTruthy(sxEq(dispatchCh, "|"))) { pos = (pos + 1);
|
||
return readRawString(); } else if (isSxTruthy(sxEq(dispatchCh, "'"))) { pos = (pos + 1);
|
||
return [makeSymbol("quote"), readExpr()]; } else if (isSxTruthy(sxEq(dispatchCh, "\\"))) { pos = (pos + 1);
|
||
return readCharLiteral(); } else if (isSxTruthy(isIdentStart(dispatchCh))) { { var macroName = readIdent();
|
||
{ var handler = readerMacroGet(macroName);
|
||
if (isSxTruthy(handler)) { return handler(readExpr()); } else { return error((String("Unknown reader macro: #") + String(macroName))); } } } } else { return error((String("Unknown reader macro: #") + String(dispatchCh))); } } } } else if (isSxTruthy(sxOr((isSxTruthy((ch >= "0")) && (ch <= "9")), (isSxTruthy(sxEq(ch, "-")) && isSxTruthy(((pos + 1) < lenSrc)) && (function() {
|
||
var nextCh = nth(source, (pos + 1));
|
||
return (isSxTruthy((nextCh >= "0")) && (nextCh <= "9"));
|
||
})())))) { return readNumber(); } else if (isSxTruthy((isSxTruthy(sxEq(ch, ".")) && isSxTruthy(((pos + 2) < lenSrc)) && isSxTruthy(sxEq(nth(source, (pos + 1)), ".")) && sxEq(nth(source, (pos + 2)), ".")))) { pos = (pos + 3);
|
||
return makeSymbol("..."); } else if (isSxTruthy(isIdentStart(ch))) { return readSymbol(); } else { return error((String("Unexpected character: ") + String(ch))); } } } } };
|
||
PRIMITIVES["read-expr"] = readExpr;
|
||
return (function() {
|
||
var exprs = [];
|
||
var parseLoop = function() { while(true) { skipWs();
|
||
if (isSxTruthy((pos < lenSrc))) { exprs.push(readExpr());
|
||
continue; } else { return NIL; } } };
|
||
PRIMITIVES["parse-loop"] = parseLoop;
|
||
parseLoop();
|
||
return exprs;
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sx-parse"] = sxParse;
|
||
|
||
// sx-serialize
|
||
var sxSerialize = function(val) { return (function() { var _m = typeOf(val); if (_m == "nil") return "nil"; if (_m == "boolean") return (isSxTruthy(val) ? "true" : "false"); if (_m == "number") return (String(val)); if (_m == "rational") return (String(numerator(val)) + String("/") + String(denominator(val))); if (_m == "string") return (String("\"") + String(escapeString(val)) + String("\"")); if (_m == "symbol") return symbolName(val); if (_m == "keyword") return (String(":") + String(keywordName(val))); if (_m == "list") return (String("(") + String(join(" ", map(sxSerialize, val))) + String(")")); if (_m == "dict") return sxSerializeDict(val); if (_m == "sx-expr") return sxExprSource(val); if (_m == "spread") return (String("(make-spread ") + String(sxSerializeDict(spreadAttrs(val))) + String(")")); if (_m == "char") return (function() {
|
||
var n = charToInteger(val);
|
||
return (String("#\\") + String((isSxTruthy(sxEq(n, 32)) ? "space" : (isSxTruthy(sxEq(n, 10)) ? "newline" : (isSxTruthy(sxEq(n, 9)) ? "tab" : (isSxTruthy(sxEq(n, 13)) ? "return" : (isSxTruthy(sxEq(n, 0)) ? "nul" : (isSxTruthy(sxEq(n, 27)) ? "escape" : (isSxTruthy(sxEq(n, 127)) ? "delete" : (isSxTruthy(sxEq(n, 8)) ? "backspace" : charFromCode(n)))))))))));
|
||
})(); return (String(val)); })(); };
|
||
PRIMITIVES["sx-serialize"] = sxSerialize;
|
||
|
||
// sx-serialize-dict
|
||
var sxSerializeDict = function(d) { return (String("{") + String(join(" ", reduce(function(acc, key) { return concat(acc, [(String(":") + String(key)), sxSerialize(dictGet(d, key))]); }, [], keys(d)))) + String("}")); };
|
||
PRIMITIVES["sx-serialize-dict"] = sxSerializeDict;
|
||
|
||
// serialize
|
||
var serialize = sxSerialize;
|
||
PRIMITIVES["serialize"] = serialize;
|
||
|
||
|
||
// === Transpiled from adapter-html ===
|
||
|
||
|
||
|
||
// === Transpiled from adapter-sx ===
|
||
|
||
|
||
|
||
// === Transpiled from lib/dom (DOM library) ===
|
||
|
||
// dom-visible?
|
||
var domVisible_p = function(el) { return (isSxTruthy(el) ? !isSxTruthy(sxEq(hostGet(hostGet(el, "style"), "display"), "none")) : false); };
|
||
PRIMITIVES["dom-visible?"] = domVisible_p;
|
||
|
||
|
||
// === Transpiled from lib/browser (browser API library) ===
|
||
|
||
// json-stringify
|
||
var jsonStringify = function(v) { return hostCall(hostGlobal("JSON"), "stringify", v); };
|
||
PRIMITIVES["json-stringify"] = jsonStringify;
|
||
|
||
|
||
// === Transpiled from adapter-dom ===
|
||
|
||
|
||
|
||
// === Transpiled from engine ===
|
||
|
||
|
||
|
||
// === Transpiled from orchestration ===
|
||
|
||
|
||
|
||
// === Transpiled from boot ===
|
||
|
||
// HEAD_HOIST_SELECTOR
|
||
var HEAD_HOIST_SELECTOR = "meta, title, link[rel='canonical'], script[type='application/ld+json']";
|
||
PRIMITIVES["HEAD_HOIST_SELECTOR"] = HEAD_HOIST_SELECTOR;
|
||
|
||
// hoist-head-elements-full
|
||
var hoistHeadElementsFull = function(root) { return (function() {
|
||
var els = domQueryAll(root, HEAD_HOIST_SELECTOR);
|
||
return forEach(function(el) { return (function() {
|
||
var tag = lower(domTagName(el));
|
||
return (isSxTruthy(sxEq(tag, "title")) ? (setDocumentTitle(domTextContent(el)), domRemoveChild(domParent(el), el)) : (isSxTruthy(sxEq(tag, "meta")) ? ((function() {
|
||
var name = domGetAttr(el, "name");
|
||
var prop = domGetAttr(el, "property");
|
||
if (isSxTruthy(name)) {
|
||
removeHeadElement((String("meta[name=\"") + String(name) + String("\"]")));
|
||
}
|
||
return (isSxTruthy(prop) ? removeHeadElement((String("meta[property=\"") + String(prop) + String("\"]"))) : NIL);
|
||
})(), domRemoveChild(domParent(el), el), domAppendToHead(el)) : (isSxTruthy((isSxTruthy(sxEq(tag, "link")) && sxEq(domGetAttr(el, "rel"), "canonical"))) ? (removeHeadElement("link[rel=\"canonical\"]"), domRemoveChild(domParent(el), el), domAppendToHead(el)) : (domRemoveChild(domParent(el), el), domAppendToHead(el)))));
|
||
})(); }, els);
|
||
})(); };
|
||
PRIMITIVES["hoist-head-elements-full"] = hoistHeadElementsFull;
|
||
|
||
// sx-mount
|
||
var sxMount = function(target, source, extraEnv) { return (function() {
|
||
var el = resolveMountTarget(target);
|
||
return (isSxTruthy(el) ? ((isSxTruthy(isEmpty(domChildList(el))) ? (function() {
|
||
var node = sxRenderWithEnv(source, extraEnv);
|
||
domSetTextContent(el, "");
|
||
domAppend(el, node);
|
||
return hoistHeadElementsFull(el);
|
||
})() : NIL), processElements(el), sxHydrateElements(el), sxHydrateIslands(el), runPostRenderHooks()) : NIL);
|
||
})(); };
|
||
PRIMITIVES["sx-mount"] = sxMount;
|
||
|
||
// resolve-suspense
|
||
var resolveSuspense = function(id, sx) { processSxScripts(NIL);
|
||
return (function() {
|
||
var el = domQuery((String("[data-suspense=\"") + String(id) + String("\"]")));
|
||
return (isSxTruthy(el) ? (function() {
|
||
var exprs = parse(sx);
|
||
var env = getRenderEnv(NIL);
|
||
domSetTextContent(el, "");
|
||
{ var _c = exprs; for (var _i = 0; _i < _c.length; _i++) { var expr = _c[_i]; domAppend(el, renderToDom(expr, env, NIL)); } }
|
||
processElements(el);
|
||
sxHydrateElements(el);
|
||
sxHydrateIslands(el);
|
||
runPostRenderHooks();
|
||
return domDispatch(el, "sx:resolved", {"id": id});
|
||
})() : logWarn((String("resolveSuspense: no element for id=") + String(id))));
|
||
})(); };
|
||
PRIMITIVES["resolve-suspense"] = resolveSuspense;
|
||
|
||
// sx-hydrate-elements
|
||
var sxHydrateElements = function(root) { return (function() {
|
||
var els = domQueryAll(sxOr(root, domBody()), "[data-sx]");
|
||
return forEach(function(el) { return (isSxTruthy(!isSxTruthy(isProcessed(el, "hydrated"))) ? (markProcessed(el, "hydrated"), sxUpdateElement(el, NIL)) : NIL); }, els);
|
||
})(); };
|
||
PRIMITIVES["sx-hydrate-elements"] = sxHydrateElements;
|
||
|
||
// sx-update-element
|
||
var sxUpdateElement = function(el, newEnv) { return (function() {
|
||
var target = resolveMountTarget(el);
|
||
return (isSxTruthy(target) ? (function() {
|
||
var source = domGetAttr(target, "data-sx");
|
||
return (isSxTruthy(source) ? (function() {
|
||
var baseEnv = parseEnvAttr(target);
|
||
var env = mergeEnvs(baseEnv, newEnv);
|
||
return (function() {
|
||
var node = sxRenderWithEnv(source, env);
|
||
domSetTextContent(target, "");
|
||
domAppend(target, node);
|
||
return (isSxTruthy(newEnv) ? storeEnvAttr(target, baseEnv, newEnv) : NIL);
|
||
})();
|
||
})() : NIL);
|
||
})() : NIL);
|
||
})(); };
|
||
PRIMITIVES["sx-update-element"] = sxUpdateElement;
|
||
|
||
// sx-render-component
|
||
var sxRenderComponent = function(name, kwargs, extraEnv) { return (function() {
|
||
var fullName = (isSxTruthy(startsWith(name, "~")) ? name : (String("~") + String(name)));
|
||
return (function() {
|
||
var env = getRenderEnv(extraEnv);
|
||
var comp = envGet(env, fullName);
|
||
return (isSxTruthy(!isSxTruthy(isComponent(comp))) ? error((String("Unknown component: ") + String(fullName))) : (function() {
|
||
var callExpr = [makeSymbol(fullName)];
|
||
{ var _c = keys(kwargs); for (var _i = 0; _i < _c.length; _i++) { var k = _c[_i]; callExpr.push(makeKeyword(toKebab(k)));
|
||
callExpr.push(dictGet(kwargs, k)); } }
|
||
return renderToDom(callExpr, env, NIL);
|
||
})());
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["sx-render-component"] = sxRenderComponent;
|
||
|
||
// process-sx-scripts
|
||
var processSxScripts = function(root) { return (function() {
|
||
var scripts = querySxScripts(root);
|
||
return forEach(function(s) { return (isSxTruthy(!isSxTruthy(isProcessed(s, "script"))) ? (markProcessed(s, "script"), (function() {
|
||
var text = domTextContent(s);
|
||
return (isSxTruthy(domHasAttr(s, "data-components")) ? processComponentScript(s, text) : (isSxTruthy(sxOr(isNil(text), isEmpty(trim(text)))) ? NIL : (isSxTruthy(domHasAttr(s, "data-init")) ? (function() {
|
||
var exprs = sxParse(text);
|
||
return forEach(function(expr) { return cekEval(expr); }, exprs);
|
||
})() : (isSxTruthy(domHasAttr(s, "data-mount")) ? (function() {
|
||
var mountSel = domGetAttr(s, "data-mount");
|
||
var target = domQuery(mountSel);
|
||
return (isSxTruthy(target) ? sxMount(target, text, NIL) : NIL);
|
||
})() : sxLoadComponents(text)))));
|
||
})()) : NIL); }, scripts);
|
||
})(); };
|
||
PRIMITIVES["process-sx-scripts"] = processSxScripts;
|
||
|
||
// process-component-script
|
||
var processComponentScript = function(script, text) { return (function() {
|
||
var hash = domGetAttr(script, "data-hash");
|
||
return (isSxTruthy(isNil(hash)) ? (isSxTruthy((isSxTruthy(text) && !isSxTruthy(isEmpty(trim(text))))) ? sxLoadComponents(text) : NIL) : (function() {
|
||
var hasInline = (isSxTruthy(text) && !isSxTruthy(isEmpty(trim(text))));
|
||
(function() {
|
||
var cachedHash = localStorageGet("sx-components-hash");
|
||
return (isSxTruthy(sxEq(cachedHash, hash)) ? (isSxTruthy(hasInline) ? (localStorageSet("sx-components-hash", hash), localStorageSet("sx-components-src", text), sxLoadComponents(text), logInfo("components: downloaded (cookie stale)")) : (function() {
|
||
var cached = localStorageGet("sx-components-src");
|
||
return (isSxTruthy(cached) ? (sxLoadComponents(cached), logInfo((String("components: cached (") + String(hash) + String(")")))) : (clearSxCompCookie(), browserReload()));
|
||
})()) : (isSxTruthy(hasInline) ? (localStorageSet("sx-components-hash", hash), localStorageSet("sx-components-src", text), sxLoadComponents(text), logInfo((String("components: downloaded (") + String(hash) + String(")")))) : (localStorageRemove("sx-components-hash"), localStorageRemove("sx-components-src"), clearSxCompCookie(), browserReload())));
|
||
})();
|
||
return setSxCompCookie(hash);
|
||
})());
|
||
})(); };
|
||
PRIMITIVES["process-component-script"] = processComponentScript;
|
||
|
||
// _page-routes
|
||
var _pageRoutes = [];
|
||
PRIMITIVES["_page-routes"] = _pageRoutes;
|
||
|
||
// process-page-scripts
|
||
var processPageScripts = function() { return (function() {
|
||
var scripts = queryPageScripts();
|
||
logInfo((String("pages: found ") + String(len(scripts)) + String(" script tags")));
|
||
{ var _c = scripts; for (var _i = 0; _i < _c.length; _i++) { var s = _c[_i]; if (isSxTruthy(!isSxTruthy(isProcessed(s, "pages")))) {
|
||
markProcessed(s, "pages");
|
||
(function() {
|
||
var text = domTextContent(s);
|
||
logInfo((String("pages: script text length=") + String((isSxTruthy(text) ? len(text) : 0))));
|
||
return (isSxTruthy((isSxTruthy(text) && !isSxTruthy(isEmpty(trim(text))))) ? (function() {
|
||
var pages = parse(text);
|
||
logInfo((String("pages: parsed ") + String(len(pages)) + String(" entries")));
|
||
return forEach(function(page) { return append_b(_pageRoutes, merge(page, {"parsed": parseRoutePattern(get(page, "path"))})); }, pages);
|
||
})() : logWarn("pages: script tag is empty"));
|
||
})();
|
||
} } }
|
||
return logInfo((String("pages: ") + String(len(_pageRoutes)) + String(" routes loaded")));
|
||
})(); };
|
||
PRIMITIVES["process-page-scripts"] = processPageScripts;
|
||
|
||
// sx-hydrate-islands
|
||
var sxHydrateIslands = function(root) { return (function() {
|
||
var els = domQueryAll(sxOr(root, domBody()), "[data-sx-island]");
|
||
preloadIslandDefs();
|
||
logInfo((String("sx-hydrate-islands: ") + String(len(els)) + String(" island(s) in ") + String((isSxTruthy(root) ? "subtree" : "document"))));
|
||
return forEach(function(el) { return (isSxTruthy(isProcessed(el, "island-hydrated")) ? logInfo((String(" skip (already hydrated): ") + String(domGetAttr(el, "data-sx-island")))) : (logInfo((String(" hydrating: ") + String(domGetAttr(el, "data-sx-island")))), markProcessed(el, "island-hydrated"), hydrateIsland(el))); }, els);
|
||
})(); };
|
||
PRIMITIVES["sx-hydrate-islands"] = sxHydrateIslands;
|
||
|
||
// hydrate-island
|
||
var hydrateIsland = function(el) { return (function() {
|
||
var name = domGetAttr(el, "data-sx-island");
|
||
var stateSx = sxOr(domGetAttr(el, "data-sx-state"), "{}");
|
||
return (function() {
|
||
var compName = (String("~") + String(name));
|
||
var env = getRenderEnv(NIL);
|
||
return (function() {
|
||
var comp = envGet(globalEnv(), compName);
|
||
return (isSxTruthy(!isSxTruthy(sxOr(isComponent(comp), isIsland(comp)))) ? logWarn((String("hydrate-island: unknown island ") + String(compName))) : (function() {
|
||
var kwargs = sxOr(first(sxParse(stateSx)), {});
|
||
var disposers = [];
|
||
var local = envMerge(componentClosure(comp), env);
|
||
{ var _c = componentParams(comp); for (var _i = 0; _i < _c.length; _i++) { var p = _c[_i]; envBind(local, p, (isSxTruthy(dictHas(kwargs, p)) ? dictGet(kwargs, p) : NIL)); } }
|
||
return (function() {
|
||
var cursor = {["parent"]: el, ["index"]: 0};
|
||
hostCall(el, "replaceChildren");
|
||
scopePush("sx-hydrating", NIL);
|
||
cekTry(function() { return withIslandScope(function(disposable) { return append_b(disposers, disposable); }, function() { return (function() {
|
||
var bodyDom = renderToDom(componentBody(comp), local, NIL);
|
||
return (isSxTruthy(bodyDom) ? domAppend(el, bodyDom) : NIL);
|
||
})(); }); }, function(err) { scopePop("sx-hydrating");
|
||
logWarn((String("hydrate fallback: ") + String(compName) + String(" — ") + String(err)));
|
||
return (function() {
|
||
var fallback = cekTry(function() { return withIslandScope(function(d) { return append_b(disposers, d); }, function() { return renderToDom(componentBody(comp), local, NIL); }); }, function(err2) { return (function() {
|
||
var e = domCreateElement("div", NIL);
|
||
domSetTextContent(e, (String("Island error: ") + String(compName) + String("\n") + String(err2)));
|
||
return e;
|
||
})(); });
|
||
hostCall(el, "replaceChildren", fallback);
|
||
return NIL;
|
||
})(); });
|
||
scopePop("sx-hydrating");
|
||
domSetData(el, "sx-disposers", disposers);
|
||
setTimeout_(function() { return processElements(el); }, 0);
|
||
return logInfo((String("hydrated island: ~") + String(compName) + String(" (") + String(len(disposers)) + String(" disposers)")));
|
||
})();
|
||
})());
|
||
})();
|
||
})();
|
||
})(); };
|
||
PRIMITIVES["hydrate-island"] = hydrateIsland;
|
||
|
||
// dispose-island
|
||
var disposeIsland = function(el) { (function() {
|
||
var disposers = domGetData(el, "sx-disposers");
|
||
return (isSxTruthy(disposers) ? (forEach(function(d) { return (isSxTruthy(isCallable(d)) ? d() : NIL); }, disposers), domSetData(el, "sx-disposers", NIL)) : NIL);
|
||
})();
|
||
return clearProcessed(el, "island-hydrated"); };
|
||
PRIMITIVES["dispose-island"] = disposeIsland;
|
||
|
||
// dispose-islands-in
|
||
var disposeIslandsIn = function(root) { return (isSxTruthy(root) ? (function() {
|
||
var islands = domQueryAll(root, "[data-sx-island]");
|
||
return (isSxTruthy((isSxTruthy(islands) && !isSxTruthy(isEmpty(islands)))) ? (function() {
|
||
var toDispose = filter(function(el) { return !isSxTruthy(isProcessed(el, "island-hydrated")); }, islands);
|
||
return (isSxTruthy(!isSxTruthy(isEmpty(toDispose))) ? (logInfo((String("disposing ") + String(len(toDispose)) + String(" island(s)"))), forEach(disposeIsland, toDispose)) : NIL);
|
||
})() : NIL);
|
||
})() : NIL); };
|
||
PRIMITIVES["dispose-islands-in"] = disposeIslandsIn;
|
||
|
||
// force-dispose-islands-in
|
||
var forceDisposeIslandsIn = function(root) { return (isSxTruthy(root) ? (function() {
|
||
var islands = domQueryAll(root, "[data-sx-island]");
|
||
return (isSxTruthy((isSxTruthy(islands) && !isSxTruthy(isEmpty(islands)))) ? (logInfo((String("force-disposing ") + String(len(islands)) + String(" island(s)"))), forEach(disposeIsland, islands)) : NIL);
|
||
})() : NIL); };
|
||
PRIMITIVES["force-dispose-islands-in"] = forceDisposeIslandsIn;
|
||
|
||
// *pre-render-hooks*
|
||
var _preRenderHooks_ = [];
|
||
PRIMITIVES["*pre-render-hooks*"] = _preRenderHooks_;
|
||
|
||
// *post-render-hooks*
|
||
var _postRenderHooks_ = [];
|
||
PRIMITIVES["*post-render-hooks*"] = _postRenderHooks_;
|
||
|
||
// register-pre-render-hook
|
||
var registerPreRenderHook = function(hookFn) { return append_b(_preRenderHooks_, hookFn); };
|
||
PRIMITIVES["register-pre-render-hook"] = registerPreRenderHook;
|
||
|
||
// register-post-render-hook
|
||
var registerPostRenderHook = function(hookFn) { return append_b(_postRenderHooks_, hookFn); };
|
||
PRIMITIVES["register-post-render-hook"] = registerPostRenderHook;
|
||
|
||
// run-pre-render-hooks
|
||
var runPreRenderHooks = function() { return forEach(function(hook) { return cekCall(hook, NIL); }, _preRenderHooks_); };
|
||
PRIMITIVES["run-pre-render-hooks"] = runPreRenderHooks;
|
||
|
||
// run-post-render-hooks
|
||
var runPostRenderHooks = function() { logInfo((String("run-post-render-hooks: ") + String(len(_postRenderHooks_)) + String(" hooks")));
|
||
{ var _c = _postRenderHooks_; for (var _i = 0; _i < _c.length; _i++) { var hook = _c[_i]; logInfo((String(" hook type: ") + String(typeOf(hook)) + String(" callable: ") + String(isCallable(hook)) + String(" lambda: ") + String(isLambda(hook))));
|
||
cekCall(hook, NIL); } }
|
||
return flushCollectedStyles(); };
|
||
PRIMITIVES["run-post-render-hooks"] = runPostRenderHooks;
|
||
|
||
// boot-init
|
||
var bootInit = function() { return (logInfo((String("sx-browser ") + String(SX_VERSION))), processPageScripts(), processSxScripts(NIL), sxHydrateElements(NIL), sxHydrateIslands(NIL), runPostRenderHooks(), flushCollectedStyles(), setTimeout_(function() { return processElements(NIL); }, 0), domSetAttr(hostGet(domDocument(), "documentElement"), "data-sx-ready", "true"), domDispatch(domDocument(), "sx:ready", NIL), logInfo("sx:ready")); };
|
||
PRIMITIVES["boot-init"] = bootInit;
|
||
|
||
|
||
// === Transpiled from deps (component dependency analysis) ===
|
||
|
||
|
||
|
||
// === Transpiled from page-helpers (pure data transformation helpers) ===
|
||
|
||
|
||
|
||
// === Transpiled from router (client-side route matching) ===
|
||
|
||
|
||
|
||
// === Transpiled from signals (reactive signal runtime) ===
|
||
|
||
|
||
|
||
// === Transpiled from signals-web (stores, events, resources) ===
|
||
|
||
|
||
|
||
// =========================================================================
|
||
// Post-transpilation fixups
|
||
// =========================================================================
|
||
// The reference spec's call-lambda only handles Lambda objects, but HO forms
|
||
// (map, reduce, etc.) may receive native primitives. Wrap to handle both.
|
||
var _rawCallLambda = callLambda;
|
||
callLambda = function(f, args, callerEnv) {
|
||
if (typeof f === "function") return f.apply(null, args);
|
||
return _rawCallLambda(f, args, callerEnv);
|
||
};
|
||
|
||
// -----------------------------------------------------------------------
|
||
// Core primitives that require native JS (cannot be expressed via FFI)
|
||
// -----------------------------------------------------------------------
|
||
PRIMITIVES["error"] = function(msg) { throw new Error(msg); };
|
||
PRIMITIVES["host-error"] = function(msg) { throw new Error(typeof msg === "string" ? msg : inspect(msg)); };
|
||
PRIMITIVES["host-warn"] = function(msg) {
|
||
var m = typeof msg === "string" ? msg : inspect(msg);
|
||
if (typeof console !== "undefined" && console.warn) console.warn(m);
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["try-catch"] = function(tryFn, catchFn) {
|
||
try {
|
||
return cekRun(continueWithCall(tryFn, [], makeEnv(), [], []));
|
||
} catch(e) {
|
||
var msg = e && e.message ? e.message : String(e);
|
||
return cekRun(continueWithCall(catchFn, [msg], makeEnv(), [msg], []));
|
||
}
|
||
};
|
||
PRIMITIVES["without-io-hook"] = function(thunk) {
|
||
return cekRun(continueWithCall(thunk, [], makeEnv(), [], []));
|
||
};
|
||
PRIMITIVES["sort"] = function(lst) {
|
||
if (!Array.isArray(lst)) return lst;
|
||
return lst.slice().sort(function(a, b) {
|
||
if (a < b) return -1; if (a > b) return 1; return 0;
|
||
});
|
||
};
|
||
|
||
// Aliases for VM bytecode compatibility
|
||
PRIMITIVES["length"] = PRIMITIVES["len"];
|
||
// FFI library functions — defined in dom.sx/browser.sx but not transpiled.
|
||
// Registered here so runtime-evaluated SX code (data-init, islands) can use them.
|
||
PRIMITIVES["prevent-default"] = preventDefault_;
|
||
PRIMITIVES["stop-propagation"] = stopPropagation_;
|
||
PRIMITIVES["event-modifier-key?"] = eventModifierKey_p;
|
||
PRIMITIVES["element-value"] = elementValue;
|
||
PRIMITIVES["error-message"] = errorMessage;
|
||
PRIMITIVES["schedule-idle"] = scheduleIdle;
|
||
PRIMITIVES["console-log"] = function() {
|
||
var args = Array.prototype.slice.call(arguments);
|
||
console.log.apply(console, ["[sx]"].concat(args));
|
||
return args.length > 0 ? args[0] : NIL;
|
||
};
|
||
PRIMITIVES["set-cookie"] = function(name, value, days) {
|
||
var d = days || 365;
|
||
var expires = new Date(Date.now() + d * 864e5).toUTCString();
|
||
document.cookie = name + "=" + encodeURIComponent(value) + ";expires=" + expires + ";path=/;SameSite=Lax";
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["get-cookie"] = function(name) {
|
||
var m = document.cookie.match(new RegExp("(?:^|;\\s*)" + name + "=([^;]*)"));
|
||
return m ? decodeURIComponent(m[1]) : NIL;
|
||
};
|
||
|
||
// dom.sx / browser.sx library functions — not transpiled, registered from
|
||
// native platform implementations so runtime-eval'd SX code can use them.
|
||
if (typeof domBody === "function") PRIMITIVES["dom-body"] = domBody;
|
||
if (typeof domQuery === "function") PRIMITIVES["dom-query"] = domQuery;
|
||
if (typeof domQueryAll === "function") PRIMITIVES["dom-query-all"] = domQueryAll;
|
||
if (typeof domQueryById === "function") PRIMITIVES["dom-query-by-id"] = domQueryById;
|
||
if (typeof domSetAttr === "function") PRIMITIVES["dom-set-attr"] = domSetAttr;
|
||
if (typeof domGetAttr === "function") PRIMITIVES["dom-get-attr"] = domGetAttr;
|
||
if (typeof domRemoveAttr === "function") PRIMITIVES["dom-remove-attr"] = domRemoveAttr;
|
||
if (typeof domHasAttr === "function") PRIMITIVES["dom-has-attr?"] = domHasAttr;
|
||
if (typeof domAddClass === "function") PRIMITIVES["dom-add-class"] = domAddClass;
|
||
if (typeof domRemoveClass === "function") PRIMITIVES["dom-remove-class"] = domRemoveClass;
|
||
if (typeof domHasClass === "function") PRIMITIVES["dom-has-class?"] = domHasClass;
|
||
if (typeof domClosest === "function") PRIMITIVES["dom-closest"] = domClosest;
|
||
if (typeof domMatches === "function") PRIMITIVES["dom-matches?"] = domMatches;
|
||
if (typeof domOuterHtml === "function") PRIMITIVES["dom-outer-html"] = domOuterHtml;
|
||
if (typeof domInnerHtml === "function") PRIMITIVES["dom-inner-html"] = domInnerHtml;
|
||
if (typeof domTextContent === "function") PRIMITIVES["dom-text-content"] = domTextContent;
|
||
if (typeof domCreateElement === "function") PRIMITIVES["dom-create-element"] = domCreateElement;
|
||
if (typeof domAppend === "function") PRIMITIVES["dom-append"] = domAppend;
|
||
if (typeof domAppendToHead === "function") PRIMITIVES["dom-append-to-head"] = domAppendToHead;
|
||
if (typeof jsonParse === "function") PRIMITIVES["json-parse"] = jsonParse;
|
||
if (typeof nowMs === "function") PRIMITIVES["now-ms"] = nowMs;
|
||
PRIMITIVES["log-info"] = logInfo;
|
||
PRIMITIVES["log-warn"] = logWarn;
|
||
PRIMITIVES["dom-listen"] = domListen;
|
||
PRIMITIVES["dom-dispatch"] = domDispatch;
|
||
PRIMITIVES["event-detail"] = eventDetail;
|
||
PRIMITIVES["create-text-node"] = createTextNode;
|
||
PRIMITIVES["dom-set-text-content"] = domSetTextContent;
|
||
PRIMITIVES["dom-focus"] = domFocus;
|
||
PRIMITIVES["dom-tag-name"] = domTagName;
|
||
PRIMITIVES["dom-get-prop"] = domGetProp;
|
||
PRIMITIVES["dom-set-prop"] = domSetProp;
|
||
if (typeof reactiveText === "function") PRIMITIVES["reactive-text"] = reactiveText;
|
||
PRIMITIVES["set-interval"] = setInterval_;
|
||
PRIMITIVES["clear-interval"] = clearInterval_;
|
||
PRIMITIVES["promise-then"] = promiseThen;
|
||
PRIMITIVES["promise-delayed"] = promiseDelayed;
|
||
PRIMITIVES["local-storage-get"] = function(key) {
|
||
try { var v = localStorage.getItem(key); return v === null ? NIL : v; }
|
||
catch (e) { return NIL; }
|
||
};
|
||
PRIMITIVES["local-storage-set"] = function(key, val) {
|
||
try { localStorage.setItem(key, val); } catch (e) {}
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["local-storage-remove"] = function(key) {
|
||
try { localStorage.removeItem(key); } catch (e) {}
|
||
return NIL;
|
||
};
|
||
if (typeof sxParse === "function") PRIMITIVES["sx-parse"] = sxParse;
|
||
PRIMITIVES["cek-try"] = function(thunkFn, handlerFn) {
|
||
try {
|
||
var result = _wrapSxFn(thunkFn)();
|
||
if (!handlerFn || handlerFn === NIL) return [makeSymbol("ok"), result];
|
||
return result;
|
||
} catch (e) {
|
||
var msg = (e && e.message) ? e.message : String(e);
|
||
if (handlerFn && handlerFn !== NIL) return _wrapSxFn(handlerFn)(msg);
|
||
return [makeSymbol("error"), msg];
|
||
}
|
||
};
|
||
// Named stores — global mutable registry (mirrors OCaml sx_primitives.ml)
|
||
var _storeRegistry = {};
|
||
function defStore(name, initFn) {
|
||
if (!_storeRegistry.hasOwnProperty(name)) {
|
||
_storeRegistry[name] = _wrapSxFn(initFn)();
|
||
}
|
||
return _storeRegistry[name];
|
||
}
|
||
function useStore(name) {
|
||
if (!_storeRegistry.hasOwnProperty(name)) throw new Error("Store not found: " + name);
|
||
return _storeRegistry[name];
|
||
}
|
||
function clearStores() { _storeRegistry = {}; return NIL; }
|
||
PRIMITIVES["def-store"] = defStore;
|
||
PRIMITIVES["use-store"] = useStore;
|
||
PRIMITIVES["clear-stores"] = clearStores;
|
||
|
||
// -----------------------------------------------------------------------
|
||
// define-type override — produces native AdtValue instances (Step 6).
|
||
// The transpiled sfDefineType from evaluator.sx creates plain dict
|
||
// instances. We override here to construct AdtValue via makeAdtValue so
|
||
// type-of returns the type name and adt? can distinguish from dicts.
|
||
// dict? still returns true for AdtValue (shim) so spec-level match-pattern
|
||
// continues to work without changes.
|
||
// -----------------------------------------------------------------------
|
||
var _sfDefineTypeAdt = function(args, env) {
|
||
var typeSym = first(args);
|
||
var ctorSpecs = rest(args);
|
||
var typeName = symbolName(typeSym);
|
||
var ctorNames = map(function(spec) { return symbolName(first(spec)); }, ctorSpecs);
|
||
if (!isSxTruthy(envHas(env, "*adt-registry*"))) {
|
||
envBind(env, "*adt-registry*", {});
|
||
}
|
||
envGet(env, "*adt-registry*")[typeName] = ctorNames;
|
||
envBind(env, typeName + "?", function(v) { return isAdtValue(v) && v._type === typeName; });
|
||
for (var _i = 0; _i < ctorSpecs.length; _i++) {
|
||
(function(spec) {
|
||
var cn = symbolName(first(spec));
|
||
var fieldNames = map(function(f) { return symbolName(f); }, rest(spec));
|
||
var arity = fieldNames.length;
|
||
envBind(env, cn, function() {
|
||
var ctorArgs = Array.prototype.slice.call(arguments, 0);
|
||
if (ctorArgs.length !== arity) {
|
||
throw new Error(cn + ": expected " + arity + " args, got " + ctorArgs.length);
|
||
}
|
||
return makeAdtValue(typeName, cn, ctorArgs);
|
||
});
|
||
envBind(env, cn + "?", function(v) { return isAdtValue(v) && v._ctor === cn; });
|
||
for (var _j = 0; _j < fieldNames.length; _j++) {
|
||
(function(idx, fieldName) {
|
||
envBind(env, cn + "-" + fieldName, function(v) {
|
||
if (!isAdtValue(v)) throw new Error(cn + "-" + fieldName + ": not an ADT");
|
||
if (idx >= v._fields.length) throw new Error(cn + "-" + fieldName + ": index out of bounds");
|
||
return v._fields[idx];
|
||
});
|
||
})(_j, fieldNames[_j]);
|
||
}
|
||
})(ctorSpecs[_i]);
|
||
}
|
||
return NIL;
|
||
};
|
||
PRIMITIVES["sf-define-type"] = _sfDefineTypeAdt;
|
||
registerSpecialForm("define-type", _sfDefineTypeAdt);
|
||
PRIMITIVES["make-adt-value"] = makeAdtValue;
|
||
PRIMITIVES["adt-value?"] = isAdtValue;
|
||
|
||
// Platform deps functions (native JS, not transpiled — need explicit registration)
|
||
PRIMITIVES["component-deps"] = componentDeps;
|
||
PRIMITIVES["component-set-deps!"] = componentSetDeps;
|
||
PRIMITIVES["component-css-classes"] = componentCssClasses;
|
||
PRIMITIVES["regex-find-all"] = regexFindAll;
|
||
PRIMITIVES["scan-css-classes"] = scanCssClasses;
|
||
|
||
// Override recursive cekRun with iterative loop (avoids stack overflow)
|
||
cekRun = function(state) {
|
||
while (!cekTerminal_p(state) && !cekSuspended_p(state)) { state = cekStep(state); }
|
||
if (cekSuspended_p(state)) { throw new Error("IO suspension in non-IO context"); }
|
||
return cekValue(state);
|
||
};
|
||
|
||
// CEK is the canonical evaluator — override evalExpr to use it.
|
||
// The tree-walk evaluator (evalExpr from eval.sx) is superseded.
|
||
var _treeWalkEvalExpr = evalExpr;
|
||
evalExpr = function(expr, env) {
|
||
return cekRun(makeCekState(expr, env, []));
|
||
};
|
||
|
||
// CEK never produces thunks — trampoline resolves any legacy thunks
|
||
var _treeWalkTrampoline = trampoline;
|
||
trampoline = function(val) {
|
||
if (isThunk(val)) return evalExpr(thunkExpr(val), thunkEnv(val));
|
||
return val;
|
||
};
|
||
|
||
// Platform functions — defined in platform_js.py, not in .sx spec files.
|
||
// Spec defines self-register via js-emit-define; these are the platform interface.
|
||
PRIMITIVES["type-of"] = typeOf;
|
||
PRIMITIVES["inspect"] = inspect;
|
||
PRIMITIVES["symbol-name"] = symbolName;
|
||
PRIMITIVES["keyword-name"] = keywordName;
|
||
PRIMITIVES["callable?"] = isCallable;
|
||
PRIMITIVES["lambda?"] = isLambda;
|
||
PRIMITIVES["lambda-name"] = lambdaName;
|
||
PRIMITIVES["component?"] = isComponent;
|
||
PRIMITIVES["island?"] = isIsland;
|
||
PRIMITIVES["parameter?"] = parameter_p;
|
||
PRIMITIVES["parameter-uid"] = parameterUid;
|
||
PRIMITIVES["parameter-default"] = parameterDefault;
|
||
PRIMITIVES["make-parameter"] = function(defaultVal, converter) {
|
||
var p = new SxParameter(defaultVal, converter || null);
|
||
return p;
|
||
};
|
||
PRIMITIVES["make-symbol"] = function(n) { return new Symbol(n); };
|
||
PRIMITIVES["is-html-tag?"] = function(n) { return HTML_TAGS.indexOf(n) >= 0; };
|
||
function makeEnv() { return merge(componentEnv, PRIMITIVES); }
|
||
PRIMITIVES["make-env"] = makeEnv;
|
||
|
||
|
||
// === stdlib.sx (eval'd at runtime, not transpiled) ===
|
||
(function() {
|
||
var src = ";; ==========================================================================\n;; stdlib.sx — Pure SX standard library functions\n;;\n;; Loaded by test runners after primitives. These functions are implemented\n;; in SX and require no host-specific code.\n;;\n;; IMPORTANT: SX let/when bodies evaluate only the LAST expression.\n;; Multi-step bodies must be wrapped in (do expr1 expr2 ...).\n;; ==========================================================================\n\n;; --------------------------------------------------------------------------\n;; format — CL-style string formatting\n;;\n;; Directives:\n;; ~a display (no quotes) ~s write (with quotes)\n;; ~d decimal ~x hex ~o octal ~b binary\n;; ~f fixed-point (6dp) ~% newline\n;; ~& fresh line ~~ literal tilde\n;; ~t tab\n;;\n;; Signature: (format template arg...) -> string\n;; --------------------------------------------------------------------------\n\n(define\n (format template &rest args)\n (let\n ((buf (make-string-buffer)) (n (string-length template)))\n (define\n (consume-arg args)\n (if\n (nil? args)\n (list \"\" nil)\n (list (display-to-string (first args)) (rest args))))\n (define\n (consume-num args radix)\n (if\n (nil? args)\n (list \"\" nil)\n (list (number->string (first args) radix) (rest args))))\n (define\n (loop i args)\n (cond\n ((>= i n) (string-buffer->string buf))\n ((and (= (substring template i (+ i 1)) \"~\") (< (+ i 1) n))\n (let\n ((dir (substring template (+ i 1) (+ i 2))))\n (cond\n ((= dir \"a\")\n (let\n ((p (consume-arg args)))\n (do\n (string-buffer-append! buf (first p))\n (loop (+ i 2) (first (rest p))))))\n ((= dir \"s\")\n (if\n (nil? args)\n (loop (+ i 2) args)\n (do\n (string-buffer-append!\n buf\n (write-to-string (first args)))\n (loop (+ i 2) (rest args)))))\n ((= dir \"d\")\n (let\n ((p (consume-num args 10)))\n (do\n (string-buffer-append! buf (first p))\n (loop (+ i 2) (first (rest p))))))\n ((= dir \"x\")\n (let\n ((p (consume-num args 16)))\n (do\n (string-buffer-append! buf (first p))\n (loop (+ i 2) (first (rest p))))))\n ((= dir \"o\")\n (let\n ((p (consume-num args 8)))\n (do\n (string-buffer-append! buf (first p))\n (loop (+ i 2) (first (rest p))))))\n ((= dir \"b\")\n (let\n ((p (consume-num args 2)))\n (do\n (string-buffer-append! buf (first p))\n (loop (+ i 2) (first (rest p))))))\n ((= dir \"f\")\n (if\n (nil? args)\n (loop (+ i 2) args)\n (do\n (string-buffer-append!\n buf\n (format-decimal (first args) 6))\n (loop (+ i 2) (rest args)))))\n ((= dir \"%\")\n (do\n (string-buffer-append! buf \"\\n\")\n (loop (+ i 2) args)))\n ((= dir \"&\")\n (do\n (let\n ((so-far (string-buffer->string buf)))\n (when\n (or\n (= (string-length so-far) 0)\n (not\n (=\n (substring\n so-far\n (- (string-length so-far) 1)\n (string-length so-far))\n \"\\n\")))\n (string-buffer-append! buf \"\\n\")))\n (loop (+ i 2) args)))\n ((= dir \"~\")\n (do\n (string-buffer-append! buf \"~\")\n (loop (+ i 2) args)))\n ((= dir \"t\")\n (do\n (string-buffer-append! buf \"\\t\")\n (loop (+ i 2) args)))\n (else\n (do\n (string-buffer-append! buf \"~\")\n (loop (+ i 1) args))))))\n (else\n (do\n (string-buffer-append!\n buf\n (substring template i (+ i 1)))\n (loop (+ i 1) args)))))\n (loop 0 args)))\n";
|
||
var forms = sxParse(src);
|
||
var tmpEnv = merge({}, PRIMITIVES);
|
||
for (var i = 0; i < forms.length; i++) {
|
||
trampoline(evalExpr(forms[i], tmpEnv));
|
||
}
|
||
for (var k in tmpEnv) {
|
||
if (!PRIMITIVES[k]) PRIMITIVES[k] = tmpEnv[k];
|
||
}
|
||
})();
|
||
|
||
|
||
// =========================================================================
|
||
// Platform interface — DOM adapter (browser-only)
|
||
// =========================================================================
|
||
|
||
var _hasDom = typeof document !== "undefined";
|
||
|
||
// Register DOM adapter as the render dispatch target for the evaluator.
|
||
_renderExprFn = function(expr, env) { return renderToDom(expr, env, null); };
|
||
_renderMode = true; // Browser always evaluates in render context.
|
||
|
||
// Wire CEK render hooks — evaluator checks _renderCheck/_renderFn instead of
|
||
// the old renderActiveP()/isRenderExpr()/renderExpr() triple.
|
||
_renderCheck = function(expr, env) { return _renderMode && isRenderExpr(expr); };
|
||
_renderFn = function(expr, env) { return renderToDom(expr, env, null); };
|
||
|
||
var SVG_NS = "http://www.w3.org/2000/svg";
|
||
var MATH_NS = "http://www.w3.org/1998/Math/MathML";
|
||
|
||
function domCreateElement(tag, ns) {
|
||
if (!_hasDom) return null;
|
||
if (ns && ns !== NIL) return document.createElementNS(ns, tag);
|
||
return document.createElement(tag);
|
||
}
|
||
|
||
function createTextNode(s) {
|
||
return _hasDom ? document.createTextNode(s) : null;
|
||
}
|
||
|
||
function createComment(s) {
|
||
return _hasDom ? document.createComment(s || "") : null;
|
||
}
|
||
|
||
function createFragment() {
|
||
return _hasDom ? document.createDocumentFragment() : null;
|
||
}
|
||
|
||
function domAppend(parent, child) {
|
||
if (parent && child && !child._spread) parent.appendChild(child);
|
||
}
|
||
|
||
function domPrepend(parent, child) {
|
||
if (parent && child) parent.insertBefore(child, parent.firstChild);
|
||
}
|
||
|
||
function domSetAttr(el, name, val) {
|
||
if (el && el.setAttribute) el.setAttribute(name, val);
|
||
}
|
||
|
||
function domGetAttr(el, name) {
|
||
if (!el || !el.getAttribute) return NIL;
|
||
var v = el.getAttribute(name);
|
||
return v === null ? NIL : v;
|
||
}
|
||
|
||
function domRemoveAttr(el, name) {
|
||
if (el && el.removeAttribute) el.removeAttribute(name);
|
||
}
|
||
|
||
function domHasAttr(el, name) {
|
||
return !!(el && el.hasAttribute && el.hasAttribute(name));
|
||
}
|
||
|
||
function domParseHtml(html) {
|
||
if (!_hasDom) return null;
|
||
var tpl = document.createElement("template");
|
||
tpl.innerHTML = html;
|
||
return tpl.content;
|
||
}
|
||
|
||
function domClone(node) {
|
||
return node && node.cloneNode ? node.cloneNode(true) : node;
|
||
}
|
||
|
||
function domParent(el) { return el ? el.parentNode : null; }
|
||
function domId(el) { return el && el.id ? el.id : NIL; }
|
||
function domNodeType(el) { return el ? el.nodeType : 0; }
|
||
function domNodeName(el) { return el ? el.nodeName : ""; }
|
||
function domTextContent(el) { return el ? el.textContent || el.nodeValue || "" : ""; }
|
||
function domSetTextContent(el, s) { if (el) { if (el.nodeType === 3 || el.nodeType === 8) el.nodeValue = s; else el.textContent = s; } }
|
||
function domIsFragment(el) { return el ? el.nodeType === 11 : false; }
|
||
function domIsChildOf(child, parent) { return !!(parent && child && child.parentNode === parent); }
|
||
function domIsActiveElement(el) { return _hasDom && el === document.activeElement; }
|
||
function domIsInputElement(el) {
|
||
if (!el || !el.tagName) return false;
|
||
var t = el.tagName;
|
||
return t === "INPUT" || t === "TEXTAREA" || t === "SELECT";
|
||
}
|
||
function domFirstChild(el) { return el ? el.firstChild : null; }
|
||
function domNextSibling(el) { return el ? el.nextSibling : null; }
|
||
|
||
function domChildList(el) {
|
||
if (!el || !el.childNodes) return [];
|
||
return Array.prototype.slice.call(el.childNodes);
|
||
}
|
||
|
||
function domAttrList(el) {
|
||
if (!el || !el.attributes) return [];
|
||
var r = [];
|
||
for (var i = 0; i < el.attributes.length; i++) {
|
||
r.push([el.attributes[i].name, el.attributes[i].value]);
|
||
}
|
||
return r;
|
||
}
|
||
|
||
function domInsertBefore(parent, node, ref) {
|
||
if (parent && node) parent.insertBefore(node, ref || null);
|
||
}
|
||
|
||
function domInsertAfter(ref, node) {
|
||
if (ref && ref.parentNode && node && !node._spread) {
|
||
ref.parentNode.insertBefore(node, ref.nextSibling);
|
||
}
|
||
}
|
||
|
||
function domRemoveChild(parent, child) {
|
||
if (parent && child && child.parentNode === parent) parent.removeChild(child);
|
||
}
|
||
|
||
function domReplaceChild(parent, newChild, oldChild) {
|
||
if (parent && newChild && oldChild) parent.replaceChild(newChild, oldChild);
|
||
}
|
||
|
||
function domSetInnerHtml(el, html) {
|
||
if (el) el.innerHTML = html;
|
||
}
|
||
|
||
function domInsertAdjacentHtml(el, pos, html) {
|
||
if (el && el.insertAdjacentHTML) el.insertAdjacentHTML(pos, html);
|
||
}
|
||
|
||
function domGetStyle(el, prop) {
|
||
return el && el.style ? el.style[prop] || "" : "";
|
||
}
|
||
|
||
function domSetStyle(el, prop, val) {
|
||
if (el && el.style) el.style[prop] = val;
|
||
}
|
||
|
||
function domGetProp(el, name) { return el ? el[name] : NIL; }
|
||
function domSetProp(el, name, val) { if (el) el[name] = val; }
|
||
// Call a method on an object with correct this binding: (dom-call-method obj "methodName" arg1 arg2 ...)
|
||
function domCallMethod() {
|
||
var obj = arguments[0], method = arguments[1];
|
||
var args = Array.prototype.slice.call(arguments, 2);
|
||
if (obj && typeof obj[method] === 'function') {
|
||
try { return obj[method].apply(obj, args); }
|
||
catch(e) { console.error("[sx] dom-call-method error:", e); return NIL; }
|
||
}
|
||
return NIL;
|
||
}
|
||
// Post a message to an iframe's contentWindow without exposing the cross-origin
|
||
// Window object to the SX evaluator (which would trigger _thunk access errors).
|
||
function domPostMessage(iframe, msg, origin) {
|
||
try {
|
||
if (iframe && iframe.contentWindow) {
|
||
iframe.contentWindow.postMessage(msg, origin || '*');
|
||
}
|
||
} catch(e) { console.error("[sx] domPostMessage error:", e); }
|
||
return NIL;
|
||
}
|
||
|
||
function domAddClass(el, cls) {
|
||
if (el && el.classList) el.classList.add(cls);
|
||
}
|
||
|
||
function domRemoveClass(el, cls) {
|
||
if (el && el.classList) el.classList.remove(cls);
|
||
}
|
||
|
||
function domDispatch(el, name, detail) {
|
||
if (!_hasDom || !el || typeof el.dispatchEvent !== "function") return false;
|
||
var evt = new CustomEvent(name, { bubbles: true, cancelable: true, detail: detail || {} });
|
||
return el.dispatchEvent(evt);
|
||
}
|
||
|
||
function domListen(el, name, handler) {
|
||
if (!_hasDom || !el) return function() {};
|
||
// Wrap SX lambdas from runtime-evaluated island code into native fns
|
||
// If lambda takes 0 params, call without event arg (convenience for on-click handlers)
|
||
var wrapped = isLambda(handler)
|
||
? (lambdaParams(handler).length === 0
|
||
? function(e) { try { var r = cekCall(handler, NIL); if (globalThis._driveAsync) globalThis._driveAsync(r); } catch(err) { console.error("[sx-ref] domListen handler error:", name, err); } }
|
||
: function(e) { try { var r = cekCall(handler, [e]); if (globalThis._driveAsync) globalThis._driveAsync(r); } catch(err) { console.error("[sx-ref] domListen handler error:", name, err); } })
|
||
: handler;
|
||
if (name === "click") logInfo("domListen: click on <" + (el.tagName||"?").toLowerCase() + "> text=" + (el.textContent||"").substring(0,20) + " isLambda=" + isLambda(handler));
|
||
var passiveEvents = { touchstart: 1, touchmove: 1, wheel: 1, scroll: 1 };
|
||
var opts = passiveEvents[name] ? { passive: true } : undefined;
|
||
el.addEventListener(name, wrapped, opts);
|
||
return function() { el.removeEventListener(name, wrapped, opts); };
|
||
}
|
||
|
||
function eventDetail(e) {
|
||
return (e && e.detail != null) ? e.detail : NIL;
|
||
}
|
||
|
||
function domQuery(sel) {
|
||
return _hasDom ? document.querySelector(sel) : null;
|
||
}
|
||
|
||
function domEnsureElement(sel) {
|
||
if (!_hasDom) return null;
|
||
var el = document.querySelector(sel);
|
||
if (el) return el;
|
||
// Parse #id selector → create div with that id, append to body
|
||
if (sel.charAt(0) === '#') {
|
||
el = document.createElement('div');
|
||
el.id = sel.slice(1);
|
||
document.body.appendChild(el);
|
||
return el;
|
||
}
|
||
return null;
|
||
}
|
||
|
||
function domQueryAll(root, sel) {
|
||
if (!root || !root.querySelectorAll) return [];
|
||
return Array.prototype.slice.call(root.querySelectorAll(sel));
|
||
}
|
||
|
||
function domTagName(el) { return el && el.tagName ? el.tagName : ""; }
|
||
|
||
// Island DOM helpers
|
||
function domRemove(node) {
|
||
if (node && node.parentNode) node.parentNode.removeChild(node);
|
||
}
|
||
function domChildNodes(el) {
|
||
if (!el || !el.childNodes) return [];
|
||
return Array.prototype.slice.call(el.childNodes);
|
||
}
|
||
function domRemoveChildrenAfter(marker) {
|
||
if (!marker || !marker.parentNode) return;
|
||
var parent = marker.parentNode;
|
||
while (marker.nextSibling) parent.removeChild(marker.nextSibling);
|
||
}
|
||
function domSetData(el, key, val) {
|
||
if (el) { if (!el._sxData) el._sxData = {}; el._sxData[key] = val; }
|
||
}
|
||
function domGetData(el, key) {
|
||
return (el && el._sxData) ? (el._sxData[key] != null ? el._sxData[key] : NIL) : NIL;
|
||
}
|
||
function domInnerHtml(el) {
|
||
return (el && el.innerHTML != null) ? el.innerHTML : "";
|
||
}
|
||
function jsonParse(s) {
|
||
try { return JSON.parse(s); } catch(e) { return {}; }
|
||
}
|
||
|
||
// renderDomComponent and renderDomElement are transpiled from
|
||
// adapter-dom.sx — no imperative overrides needed.
|
||
|
||
|
||
// =========================================================================
|
||
// Platform interface — Engine pure logic (browser + node compatible)
|
||
// =========================================================================
|
||
|
||
function browserLocationHref() {
|
||
return typeof location !== "undefined" ? location.href : "";
|
||
}
|
||
|
||
function browserSameOrigin(url) {
|
||
try { return new URL(url, location.href).origin === location.origin; }
|
||
catch (e) { return true; }
|
||
}
|
||
|
||
function browserPushState(url) {
|
||
if (typeof history !== "undefined") {
|
||
try { history.pushState({ sxUrl: url, scrollY: typeof window !== "undefined" ? window.scrollY : 0 }, "", url); }
|
||
catch (e) {}
|
||
}
|
||
}
|
||
|
||
function browserReplaceState(url) {
|
||
if (typeof history !== "undefined") {
|
||
try { history.replaceState({ sxUrl: url, scrollY: typeof window !== "undefined" ? window.scrollY : 0 }, "", url); }
|
||
catch (e) {}
|
||
}
|
||
}
|
||
|
||
function nowMs() { return (typeof performance !== "undefined") ? performance.now() : Date.now(); }
|
||
|
||
function parseHeaderValue(s) {
|
||
if (!s) return null;
|
||
try {
|
||
if (s.charAt(0) === "{" && s.charAt(1) === ":") return parse(s);
|
||
return JSON.parse(s);
|
||
} catch (e) { return null; }
|
||
}
|
||
|
||
|
||
// =========================================================================
|
||
// Platform interface — Orchestration (browser-only)
|
||
// =========================================================================
|
||
|
||
// --- Stubs for define-library functions not transpiled by extract_defines ---
|
||
// These are defined in orchestration.sx's define-library and called from
|
||
// boot.sx top-level defines. The JS bootstrapper only transpiles top-level
|
||
// defines, so we provide stubs here for functions that need a JS identity.
|
||
|
||
function flushCollectedStyles() { return NIL; }
|
||
function processElements(root) { return NIL; }
|
||
|
||
// --- Browser/Network ---
|
||
|
||
function browserNavigate(url) {
|
||
if (typeof location !== "undefined") location.assign(url);
|
||
}
|
||
|
||
function browserReload() {
|
||
if (typeof location !== "undefined") location.reload();
|
||
}
|
||
|
||
function browserScrollTo(x, y) {
|
||
if (typeof window !== "undefined") window.scrollTo(x, y);
|
||
}
|
||
|
||
function browserMediaMatches(query) {
|
||
if (typeof window === "undefined") return false;
|
||
return window.matchMedia(query).matches;
|
||
}
|
||
|
||
function browserConfirm(msg) {
|
||
if (typeof window === "undefined") return false;
|
||
return window.confirm(msg);
|
||
}
|
||
|
||
function browserPrompt(msg) {
|
||
if (typeof window === "undefined") return NIL;
|
||
var r = window.prompt(msg);
|
||
return r === null ? NIL : r;
|
||
}
|
||
|
||
function csrfToken() {
|
||
if (!_hasDom) return NIL;
|
||
var m = document.querySelector('meta[name="csrf-token"]');
|
||
return m ? m.getAttribute("content") : NIL;
|
||
}
|
||
|
||
function isCrossOrigin(url) {
|
||
try {
|
||
var h = new URL(url, location.href).hostname;
|
||
return h !== location.hostname &&
|
||
(h.indexOf(".rose-ash.com") >= 0 || h.indexOf(".localhost") >= 0);
|
||
} catch (e) { return false; }
|
||
}
|
||
|
||
// --- Promises ---
|
||
|
||
function promiseResolve(val) { return Promise.resolve(val); }
|
||
|
||
function promiseThen(p, onResolve, onReject) {
|
||
if (!p || !p.then) return p;
|
||
return onReject ? p.then(onResolve, onReject) : p.then(onResolve);
|
||
}
|
||
|
||
function promiseCatch(p, fn) { return p && p.catch ? p.catch(fn) : p; }
|
||
|
||
function promiseDelayed(ms, value) {
|
||
return new Promise(function(resolve) {
|
||
setTimeout(function() { resolve(value); }, ms);
|
||
});
|
||
}
|
||
|
||
// --- Abort controllers ---
|
||
|
||
var _controllers = typeof WeakMap !== "undefined" ? new WeakMap() : null;
|
||
|
||
function abortPrevious(el) {
|
||
if (_controllers) {
|
||
var prev = _controllers.get(el);
|
||
if (prev) prev.abort();
|
||
}
|
||
}
|
||
|
||
function trackController(el, ctrl) {
|
||
if (_controllers) _controllers.set(el, ctrl);
|
||
}
|
||
|
||
var _targetControllers = typeof WeakMap !== "undefined" ? new WeakMap() : null;
|
||
|
||
function abortPreviousTarget(el) {
|
||
if (_targetControllers) {
|
||
var prev = _targetControllers.get(el);
|
||
if (prev) prev.abort();
|
||
}
|
||
}
|
||
|
||
function trackControllerTarget(el, ctrl) {
|
||
if (_targetControllers) _targetControllers.set(el, ctrl);
|
||
}
|
||
|
||
function newAbortController() {
|
||
return typeof AbortController !== "undefined" ? new AbortController() : { signal: null, abort: function() {} };
|
||
}
|
||
|
||
function controllerSignal(ctrl) { return ctrl ? ctrl.signal : null; }
|
||
|
||
function isAbortError(err) { return err && err.name === "AbortError"; }
|
||
|
||
// --- Timers ---
|
||
|
||
function _wrapSxFn(fn) {
|
||
if (fn && fn._lambda) {
|
||
return function() { return trampoline(callLambda(fn, [], lambdaClosure(fn))); };
|
||
}
|
||
return fn;
|
||
}
|
||
function setTimeout_(fn, ms) { return setTimeout(_wrapSxFn(fn), ms || 0); }
|
||
function setInterval_(fn, ms) { return setInterval(_wrapSxFn(fn), ms || 1000); }
|
||
function clearTimeout_(id) { clearTimeout(id); }
|
||
function clearInterval_(id) { clearInterval(id); }
|
||
function requestAnimationFrame_(fn) {
|
||
var cb = _wrapSxFn(fn);
|
||
if (typeof requestAnimationFrame !== "undefined") requestAnimationFrame(cb);
|
||
else setTimeout(cb, 16);
|
||
}
|
||
|
||
// --- Fetch ---
|
||
|
||
function fetchRequest(config, successFn, errorFn) {
|
||
var opts = { method: config.method, headers: config.headers };
|
||
if (config.signal) opts.signal = config.signal;
|
||
if (config.body && config.method !== "GET") opts.body = config.body;
|
||
if (config["cross-origin"]) opts.credentials = "include";
|
||
|
||
var p = (config.preloaded && config.preloaded !== NIL)
|
||
? Promise.resolve({
|
||
ok: true, status: 200,
|
||
headers: new Headers({ "Content-Type": config.preloaded["content-type"] || "" }),
|
||
text: function() { return Promise.resolve(config.preloaded.text); }
|
||
})
|
||
: fetch(config.url, opts);
|
||
|
||
return p.then(function(resp) {
|
||
return resp.text().then(function(text) {
|
||
var getHeader = function(name) {
|
||
var v = resp.headers.get(name);
|
||
return v === null ? NIL : v;
|
||
};
|
||
return successFn(resp.ok, resp.status, getHeader, text);
|
||
});
|
||
}).catch(function(err) {
|
||
return errorFn(err);
|
||
});
|
||
}
|
||
|
||
function fetchLocation(headerVal) {
|
||
if (!_hasDom) return;
|
||
var locUrl = headerVal;
|
||
try { var obj = JSON.parse(headerVal); locUrl = obj.path || obj; } catch (e) {}
|
||
fetch(locUrl, { headers: { "SX-Request": "true" } }).then(function(r) {
|
||
return r.text().then(function(t) {
|
||
var main = document.getElementById("main-panel");
|
||
if (main) {
|
||
main.innerHTML = t;
|
||
postSwap(main);
|
||
try { history.pushState({ sxUrl: locUrl }, "", locUrl); } catch (e) {}
|
||
}
|
||
});
|
||
});
|
||
}
|
||
|
||
function fetchAndRestore(main, url, headers, scrollY) {
|
||
var opts = { headers: headers };
|
||
try {
|
||
var h = new URL(url, location.href).hostname;
|
||
if (h !== location.hostname &&
|
||
(h.indexOf(".rose-ash.com") >= 0 || h.indexOf(".localhost") >= 0)) {
|
||
opts.credentials = "include";
|
||
}
|
||
} catch (e) {}
|
||
|
||
fetch(url, opts).then(function(resp) {
|
||
return resp.text().then(function(text) {
|
||
text = stripComponentScripts(text);
|
||
text = extractResponseCss(text);
|
||
text = text.trim();
|
||
if (text.charAt(0) === "(") {
|
||
try {
|
||
var dom = sxRender(text);
|
||
var container = document.createElement("div");
|
||
container.appendChild(dom);
|
||
processOobSwaps(container, function(t, oob, s) {
|
||
swapDomNodes(t, oob, s);
|
||
sxHydrate(t);
|
||
processElements(t);
|
||
});
|
||
var newMain = container.querySelector("#main-panel");
|
||
morphChildren(main, newMain || container);
|
||
postSwap(main);
|
||
if (typeof window !== "undefined") window.scrollTo(0, scrollY || 0);
|
||
} catch (err) {
|
||
console.error("sx-ref popstate error:", err);
|
||
location.reload();
|
||
}
|
||
} else {
|
||
var parser = new DOMParser();
|
||
var doc = parser.parseFromString(text, "text/html");
|
||
var newMain = doc.getElementById("main-panel");
|
||
if (newMain) {
|
||
morphChildren(main, newMain);
|
||
postSwap(main);
|
||
if (typeof window !== "undefined") window.scrollTo(0, scrollY || 0);
|
||
} else {
|
||
location.reload();
|
||
}
|
||
}
|
||
});
|
||
}).catch(function(err) {
|
||
logWarn("sx:popstate fetch error " + url + " — " + (err && err.message ? err.message : err));
|
||
location.reload();
|
||
});
|
||
}
|
||
|
||
function fetchStreaming(target, url, headers) {
|
||
// Streaming fetch for multi-stream pages.
|
||
// First chunk = OOB SX swap (shell with skeletons).
|
||
// Subsequent chunks = __sxResolve script tags filling suspense slots.
|
||
var opts = { headers: headers };
|
||
try {
|
||
var h = new URL(url, location.href).hostname;
|
||
if (h !== location.hostname &&
|
||
(h.indexOf(".rose-ash.com") >= 0 || h.indexOf(".localhost") >= 0)) {
|
||
opts.credentials = "include";
|
||
}
|
||
} catch (e) {}
|
||
|
||
fetch(url, opts).then(function(resp) {
|
||
if (!resp.ok || !resp.body) {
|
||
// Fallback: non-streaming
|
||
return resp.text().then(function(text) {
|
||
text = stripComponentScripts(text);
|
||
text = extractResponseCss(text);
|
||
text = text.trim();
|
||
if (text.charAt(0) === "(") {
|
||
var dom = sxRender(text);
|
||
var container = document.createElement("div");
|
||
container.appendChild(dom);
|
||
processOobSwaps(container, function(t, oob, s) {
|
||
swapDomNodes(t, oob, s);
|
||
sxHydrate(t);
|
||
processElements(t);
|
||
});
|
||
var newMain = container.querySelector("#main-panel");
|
||
morphChildren(target, newMain || container);
|
||
postSwap(target);
|
||
}
|
||
});
|
||
}
|
||
|
||
var reader = resp.body.getReader();
|
||
var decoder = new TextDecoder();
|
||
var buffer = "";
|
||
var initialSwapDone = false;
|
||
// Regex to match __sxResolve script tags
|
||
var RESOLVE_START = "<script>window.__sxResolve&&window.__sxResolve(";
|
||
var RESOLVE_END = ")</script>";
|
||
|
||
function processResolveScripts() {
|
||
// Strip and load any extra component defs before resolve scripts
|
||
buffer = stripSxScripts(buffer);
|
||
var idx;
|
||
while ((idx = buffer.indexOf(RESOLVE_START)) >= 0) {
|
||
var endIdx = buffer.indexOf(RESOLVE_END, idx);
|
||
if (endIdx < 0) break; // incomplete, wait for more data
|
||
var argsStr = buffer.substring(idx + RESOLVE_START.length, endIdx);
|
||
buffer = buffer.substring(endIdx + RESOLVE_END.length);
|
||
// argsStr is: "stream-id","sx source"
|
||
var commaIdx = argsStr.indexOf(",");
|
||
if (commaIdx >= 0) {
|
||
try {
|
||
var id = JSON.parse(argsStr.substring(0, commaIdx));
|
||
var sx = JSON.parse(argsStr.substring(commaIdx + 1));
|
||
if (typeof Sx !== "undefined" && Sx.resolveSuspense) {
|
||
Sx.resolveSuspense(id, sx);
|
||
}
|
||
} catch (e) {
|
||
console.error("[sx-ref] resolve parse error:", e);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
function pump() {
|
||
return reader.read().then(function(result) {
|
||
buffer += decoder.decode(result.value || new Uint8Array(), { stream: !result.done });
|
||
|
||
if (!initialSwapDone) {
|
||
// Look for the first resolve script — everything before it is OOB content
|
||
var scriptIdx = buffer.indexOf("<script>window.__sxResolve");
|
||
// If we found a script tag, or the stream is done, process OOB
|
||
var oobEnd = scriptIdx >= 0 ? scriptIdx : (result.done ? buffer.length : -1);
|
||
if (oobEnd >= 0) {
|
||
var oobContent = buffer.substring(0, oobEnd);
|
||
buffer = buffer.substring(oobEnd);
|
||
initialSwapDone = true;
|
||
|
||
// Process OOB SX content (same as fetchAndRestore)
|
||
oobContent = stripComponentScripts(oobContent);
|
||
// Also strip bare <script type="text/sx"> (extra defs from resolve chunks)
|
||
oobContent = stripSxScripts(oobContent);
|
||
oobContent = extractResponseCss(oobContent);
|
||
oobContent = oobContent.trim();
|
||
if (oobContent.charAt(0) === "(") {
|
||
try {
|
||
var dom = sxRender(oobContent);
|
||
var container = document.createElement("div");
|
||
container.appendChild(dom);
|
||
processOobSwaps(container, function(t, oob, s) {
|
||
swapDomNodes(t, oob, s);
|
||
sxHydrate(t);
|
||
processElements(t);
|
||
});
|
||
var newMain = container.querySelector("#main-panel");
|
||
morphChildren(target, newMain || container);
|
||
postSwap(target);
|
||
// Dispatch clientRoute so nav links update active state
|
||
domDispatch(target, "sx:clientRoute",
|
||
{ pathname: new URL(url, location.href).pathname });
|
||
} catch (err) {
|
||
console.error("[sx-ref] streaming OOB swap error:", err);
|
||
}
|
||
}
|
||
// Process any resolve scripts already in buffer
|
||
processResolveScripts();
|
||
}
|
||
} else {
|
||
// Process resolve scripts as they arrive
|
||
processResolveScripts();
|
||
}
|
||
|
||
if (!result.done) return pump();
|
||
});
|
||
}
|
||
|
||
return pump();
|
||
}).catch(function(err) {
|
||
console.error("[sx-ref] streaming fetch error:", err);
|
||
location.reload();
|
||
});
|
||
}
|
||
|
||
function fetchPreload(url, headers, cache) {
|
||
fetch(url, { headers: headers }).then(function(resp) {
|
||
if (!resp.ok) return;
|
||
var ct = resp.headers.get("Content-Type") || "";
|
||
return resp.text().then(function(text) {
|
||
preloadCacheSet(cache, url, text, ct);
|
||
});
|
||
}).catch(function(err) {
|
||
logInfo("sx:preload error " + url + " — " + (err && err.message ? err.message : err));
|
||
});
|
||
}
|
||
|
||
// --- Request body building ---
|
||
|
||
function buildRequestBody(el, method, url) {
|
||
if (!_hasDom) return { body: null, url: url, "content-type": NIL };
|
||
var body = null;
|
||
var ct = NIL;
|
||
var finalUrl = url;
|
||
var isJson = el.getAttribute("sx-encoding") === "json";
|
||
|
||
if (method !== "GET") {
|
||
var form = el.closest("form") || (el.tagName === "FORM" ? el : null);
|
||
if (form) {
|
||
if (isJson) {
|
||
var fd = new FormData(form);
|
||
var obj = {};
|
||
fd.forEach(function(v, k) {
|
||
if (obj[k] !== undefined) {
|
||
if (!Array.isArray(obj[k])) obj[k] = [obj[k]];
|
||
obj[k].push(v);
|
||
} else { obj[k] = v; }
|
||
});
|
||
body = JSON.stringify(obj);
|
||
ct = "application/json";
|
||
} else {
|
||
body = new URLSearchParams(new FormData(form));
|
||
ct = "application/x-www-form-urlencoded";
|
||
}
|
||
}
|
||
}
|
||
|
||
// sx-params
|
||
var paramsSpec = el.getAttribute("sx-params");
|
||
if (paramsSpec && body instanceof URLSearchParams) {
|
||
if (paramsSpec === "none") {
|
||
body = new URLSearchParams();
|
||
} else if (paramsSpec.indexOf("not ") === 0) {
|
||
paramsSpec.substring(4).split(",").forEach(function(k) { body.delete(k.trim()); });
|
||
} else if (paramsSpec !== "*") {
|
||
var allowed = paramsSpec.split(",").map(function(s) { return s.trim(); });
|
||
var filtered = new URLSearchParams();
|
||
allowed.forEach(function(k) {
|
||
body.getAll(k).forEach(function(v) { filtered.append(k, v); });
|
||
});
|
||
body = filtered;
|
||
}
|
||
}
|
||
|
||
// sx-include
|
||
var includeSel = el.getAttribute("sx-include");
|
||
if (includeSel && method !== "GET") {
|
||
if (!body) body = new URLSearchParams();
|
||
document.querySelectorAll(includeSel).forEach(function(inp) {
|
||
if (inp.name) body.append(inp.name, inp.value);
|
||
});
|
||
}
|
||
|
||
// sx-vals
|
||
var valsAttr = el.getAttribute("sx-vals");
|
||
if (valsAttr) {
|
||
try {
|
||
var vals = valsAttr.charAt(0) === "{" && valsAttr.charAt(1) === ":" ? parse(valsAttr) : JSON.parse(valsAttr);
|
||
if (method === "GET") {
|
||
for (var vk in vals) finalUrl += (finalUrl.indexOf("?") >= 0 ? "&" : "?") + encodeURIComponent(vk) + "=" + encodeURIComponent(vals[vk]);
|
||
} else if (body instanceof URLSearchParams) {
|
||
for (var vk2 in vals) body.append(vk2, vals[vk2]);
|
||
} else if (!body) {
|
||
body = new URLSearchParams();
|
||
for (var vk3 in vals) body.append(vk3, vals[vk3]);
|
||
ct = "application/x-www-form-urlencoded";
|
||
}
|
||
} catch (e) {}
|
||
}
|
||
|
||
// GET form data → URL
|
||
if (method === "GET") {
|
||
var form2 = el.closest("form") || (el.tagName === "FORM" ? el : null);
|
||
if (form2) {
|
||
var qs = new URLSearchParams(new FormData(form2)).toString();
|
||
if (qs) finalUrl += (finalUrl.indexOf("?") >= 0 ? "&" : "?") + qs;
|
||
}
|
||
if ((el.tagName === "INPUT" || el.tagName === "SELECT" || el.tagName === "TEXTAREA") && el.name) {
|
||
finalUrl += (finalUrl.indexOf("?") >= 0 ? "&" : "?") + encodeURIComponent(el.name) + "=" + encodeURIComponent(el.value);
|
||
}
|
||
}
|
||
|
||
return { body: body, url: finalUrl, "content-type": ct };
|
||
}
|
||
|
||
// --- Loading state ---
|
||
|
||
function showIndicator(el) {
|
||
if (!_hasDom) return NIL;
|
||
var sel = el.getAttribute("sx-indicator");
|
||
var ind = sel ? (document.querySelector(sel) || el.closest(sel)) : null;
|
||
if (ind) { ind.classList.add("sx-request"); ind.style.display = ""; }
|
||
return ind || NIL;
|
||
}
|
||
|
||
function disableElements(el) {
|
||
if (!_hasDom) return [];
|
||
var sel = el.getAttribute("sx-disabled-elt");
|
||
if (!sel) return [];
|
||
var elts = Array.prototype.slice.call(document.querySelectorAll(sel));
|
||
elts.forEach(function(e) { e.disabled = true; });
|
||
return elts;
|
||
}
|
||
|
||
function clearLoadingState(el, indicator, disabledElts) {
|
||
el.classList.remove("sx-request");
|
||
el.removeAttribute("aria-busy");
|
||
if (indicator && !isNil(indicator)) {
|
||
indicator.classList.remove("sx-request");
|
||
indicator.style.display = "none";
|
||
}
|
||
if (disabledElts) {
|
||
for (var i = 0; i < disabledElts.length; i++) disabledElts[i].disabled = false;
|
||
}
|
||
}
|
||
|
||
// --- DOM extras ---
|
||
|
||
function domQueryById(id) {
|
||
return _hasDom ? document.getElementById(id) : null;
|
||
}
|
||
|
||
function domMatches(el, sel) {
|
||
return el && el.matches ? el.matches(sel) : false;
|
||
}
|
||
|
||
function domClosest(el, sel) {
|
||
return el && el.closest ? el.closest(sel) : null;
|
||
}
|
||
|
||
function domDocument() {
|
||
return _hasDom ? document : null;
|
||
}
|
||
|
||
function domBody() {
|
||
return _hasDom ? document.body : null;
|
||
}
|
||
|
||
function domHasClass(el, cls) {
|
||
return el && el.classList ? el.classList.contains(cls) : false;
|
||
}
|
||
|
||
function domAppendToHead(el) {
|
||
if (_hasDom && document.head) document.head.appendChild(el);
|
||
}
|
||
|
||
function domParseHtmlDocument(text) {
|
||
if (!_hasDom) return null;
|
||
return new DOMParser().parseFromString(text, "text/html");
|
||
}
|
||
|
||
function domOuterHtml(el) {
|
||
return el ? el.outerHTML : "";
|
||
}
|
||
|
||
function domBodyInnerHtml(doc) {
|
||
return doc && doc.body ? doc.body.innerHTML : "";
|
||
}
|
||
|
||
// --- Events ---
|
||
|
||
function preventDefault_(e) { if (e && e.preventDefault) e.preventDefault(); }
|
||
function stopPropagation_(e) { if (e && e.stopPropagation) e.stopPropagation(); }
|
||
function eventModifierKey_p(e) { return !!(e && (e.ctrlKey || e.metaKey || e.shiftKey || e.altKey)); }
|
||
function domFocus(el) { if (el && el.focus) el.focus(); }
|
||
function tryCatch(tryFn, catchFn) {
|
||
var t = _wrapSxFn(tryFn);
|
||
var c = catchFn && catchFn._lambda
|
||
? function(e) { return trampoline(callLambda(catchFn, [e], lambdaClosure(catchFn))); }
|
||
: catchFn;
|
||
try { return t(); } catch (e) { return c(e); }
|
||
}
|
||
function cekTry(thunkFn, handlerFn) {
|
||
try {
|
||
var result = _wrapSxFn(thunkFn)();
|
||
if (!handlerFn || handlerFn === NIL) return [makeSymbol("ok"), result];
|
||
return result;
|
||
} catch (e) {
|
||
var msg = (e && e.message) ? e.message : String(e);
|
||
if (handlerFn && handlerFn !== NIL) return _wrapSxFn(handlerFn)(msg);
|
||
return [makeSymbol("error"), msg];
|
||
}
|
||
}
|
||
function errorMessage(e) {
|
||
return e && e.message ? e.message : String(e);
|
||
}
|
||
function scheduleIdle(fn) {
|
||
var cb = _wrapSxFn(fn);
|
||
if (typeof cb !== "function") {
|
||
console.error("[sx-ref] scheduleIdle: callback not callable, fn type:", typeof fn, "fn:", fn, "_lambda:", fn && fn._lambda);
|
||
return;
|
||
}
|
||
if (typeof requestIdleCallback !== "undefined") requestIdleCallback(cb);
|
||
else setTimeout(cb, 0);
|
||
}
|
||
function elementValue(el) { return el && el.value !== undefined ? el.value : NIL; }
|
||
|
||
function domAddListener(el, event, fn, opts) {
|
||
if (!el || !el.addEventListener) return;
|
||
var o = {};
|
||
if (opts && !isNil(opts)) {
|
||
if (opts.once || opts["once"]) o.once = true;
|
||
}
|
||
el.addEventListener(event, function(e) {
|
||
try { fn(e); } catch (err) { logInfo("EVENT ERROR: " + event + " " + (err && err.message ? err.message : err)); console.error("[sx-ref] event handler error:", event, err); }
|
||
}, o);
|
||
}
|
||
|
||
// --- Validation ---
|
||
|
||
function validateForRequest(el) {
|
||
if (!_hasDom) return true;
|
||
var attr = el.getAttribute("sx-validate");
|
||
if (attr === null) {
|
||
var vForm = el.closest("[sx-validate]");
|
||
if (vForm) attr = vForm.getAttribute("sx-validate");
|
||
}
|
||
if (attr === null) return true; // no validation configured
|
||
var form = el.tagName === "FORM" ? el : el.closest("form");
|
||
if (form && !form.reportValidity()) return false;
|
||
if (attr && attr !== "true" && attr !== "") {
|
||
var fn = window[attr];
|
||
if (typeof fn === "function" && !fn(el)) return false;
|
||
}
|
||
return true;
|
||
}
|
||
|
||
// --- View Transitions ---
|
||
|
||
function withTransition(enabled, fn) {
|
||
if (enabled && _hasDom && document.startViewTransition) {
|
||
document.startViewTransition(fn);
|
||
} else {
|
||
fn();
|
||
}
|
||
}
|
||
|
||
// --- IntersectionObserver ---
|
||
|
||
function observeIntersection(el, fn, once, delay) {
|
||
if (!_hasDom || !("IntersectionObserver" in window)) { fn(); return; }
|
||
var fired = false;
|
||
var d = isNil(delay) ? 0 : delay;
|
||
var obs = new IntersectionObserver(function(entries) {
|
||
entries.forEach(function(entry) {
|
||
if (!entry.isIntersecting) return;
|
||
if (once && fired) return;
|
||
fired = true;
|
||
if (once) obs.unobserve(el);
|
||
if (d) setTimeout(fn, d); else fn();
|
||
});
|
||
});
|
||
obs.observe(el);
|
||
}
|
||
|
||
// --- EventSource ---
|
||
|
||
function eventSourceConnect(url, el) {
|
||
var source = new EventSource(url);
|
||
source.addEventListener("error", function() { domDispatch(el, "sx:sseError", {}); });
|
||
source.addEventListener("open", function() { domDispatch(el, "sx:sseOpen", {}); });
|
||
if (typeof MutationObserver !== "undefined") {
|
||
var obs = new MutationObserver(function() {
|
||
if (!document.body.contains(el)) { source.close(); obs.disconnect(); }
|
||
});
|
||
obs.observe(document.body, { childList: true, subtree: true });
|
||
}
|
||
return source;
|
||
}
|
||
|
||
function eventSourceListen(source, event, fn) {
|
||
source.addEventListener(event, function(e) { fn(e.data); });
|
||
}
|
||
|
||
// --- Boost bindings ---
|
||
|
||
function bindBoostLink(el, _href) {
|
||
el.addEventListener("click", function(e) {
|
||
if (e.ctrlKey || e.metaKey || e.shiftKey || e.altKey) return;
|
||
e.preventDefault();
|
||
// Re-read href from element at click time (not closed-over value)
|
||
var liveHref = el.getAttribute("href") || _href;
|
||
console.log("[sx-debug] bindBoostLink click:", liveHref, "el:", el.tagName, el.textContent.slice(0,30));
|
||
executeRequest(el, { method: "GET", url: liveHref }).then(function() {
|
||
console.log("[sx-debug] boost fetch OK, pushState:", liveHref);
|
||
try { history.pushState({ sxUrl: liveHref, scrollY: window.scrollY }, "", liveHref); } catch (err) {}
|
||
}).catch(function(err) {
|
||
console.error("[sx-debug] boost fetch ERROR:", err);
|
||
});
|
||
});
|
||
}
|
||
|
||
function bindBoostForm(form, _method, _action) {
|
||
form.addEventListener("submit", function(e) {
|
||
e.preventDefault();
|
||
// Re-read from element at submit time
|
||
var liveMethod = (form.getAttribute("method") || _method || "GET").toUpperCase();
|
||
var liveAction = form.getAttribute("action") || _action || location.href;
|
||
executeRequest(form, { method: liveMethod, url: liveAction }).then(function() {
|
||
try { history.pushState({ sxUrl: liveAction, scrollY: window.scrollY }, "", liveAction); } catch (err) {}
|
||
}).catch(function(err) {
|
||
logWarn("sx:boost form error " + liveMethod + " " + liveAction + " — " + (err && err.message ? err.message : err));
|
||
});
|
||
});
|
||
}
|
||
|
||
// --- Client-side route bindings ---
|
||
|
||
function bindClientRouteClick(link, _href, fallbackFn) {
|
||
link.addEventListener("click", function(e) {
|
||
if (e.ctrlKey || e.metaKey || e.shiftKey || e.altKey) return;
|
||
e.preventDefault();
|
||
// Re-read href from element at click time (not closed-over value)
|
||
var liveHref = link.getAttribute("href") || _href;
|
||
var pathname = urlPathname(liveHref);
|
||
console.log("[sx-debug] bindClientRouteClick:", pathname, "el:", link.tagName, link.textContent.slice(0,30));
|
||
// Find target selector: sx-boost ancestor, explicit sx-target, or #main-panel
|
||
var boostEl = link.closest("[sx-boost]");
|
||
var targetSel = boostEl ? boostEl.getAttribute("sx-boost") : null;
|
||
if (!targetSel || targetSel === "true") {
|
||
targetSel = link.getAttribute("sx-target") || "#main-panel";
|
||
}
|
||
console.log("[sx-debug] targetSel:", targetSel, "trying client route...");
|
||
if (tryClientRoute(pathname, targetSel)) {
|
||
console.log("[sx-debug] client route SUCCESS, pushState:", liveHref);
|
||
try { history.pushState({ sxUrl: liveHref, scrollY: window.scrollY }, "", liveHref); } catch (err) {}
|
||
if (typeof window !== "undefined") window.scrollTo(0, 0);
|
||
} else {
|
||
console.log("[sx-debug] client route FAILED, server fetch:", liveHref);
|
||
executeRequest(link, { method: "GET", url: liveHref }).then(function() {
|
||
console.log("[sx-debug] server fetch OK, pushState:", liveHref);
|
||
try { history.pushState({ sxUrl: liveHref, scrollY: window.scrollY }, "", liveHref); } catch (err) {}
|
||
}).catch(function(err) {
|
||
console.error("[sx-debug] server fetch ERROR:", err);
|
||
});
|
||
}
|
||
});
|
||
}
|
||
|
||
function tryEvalContent(source, env) {
|
||
try {
|
||
var merged = merge(componentEnv);
|
||
if (env && !isNil(env)) {
|
||
var ks = Object.keys(env);
|
||
for (var i = 0; i < ks.length; i++) merged[ks[i]] = env[ks[i]];
|
||
}
|
||
return sxRenderWithEnv(source, merged);
|
||
} catch (e) {
|
||
logInfo("sx:route eval miss: " + (e && e.message ? e.message : e));
|
||
return NIL;
|
||
}
|
||
}
|
||
|
||
// Async eval with callback — used for pages with IO deps.
|
||
// Calls callback(rendered) when done, callback(null) on failure.
|
||
function tryAsyncEvalContent(source, env, callback) {
|
||
var merged = merge(componentEnv);
|
||
if (env && !isNil(env)) {
|
||
var ks = Object.keys(env);
|
||
for (var i = 0; i < ks.length; i++) merged[ks[i]] = env[ks[i]];
|
||
}
|
||
try {
|
||
var result = asyncSxRenderWithEnv(source, merged);
|
||
if (isPromise(result)) {
|
||
result.then(function(rendered) {
|
||
callback(rendered);
|
||
}).catch(function(e) {
|
||
logWarn("sx:async eval miss: " + (e && e.message ? e.message : e));
|
||
callback(null);
|
||
});
|
||
} else {
|
||
callback(result);
|
||
}
|
||
} catch (e) {
|
||
logInfo("sx:async eval miss: " + (e && e.message ? e.message : e));
|
||
callback(null);
|
||
}
|
||
}
|
||
|
||
function resolvePageData(pageName, params, callback) {
|
||
// Platform implementation: fetch page data via HTTP from /sx/data/ endpoint.
|
||
// The spec only knows about resolve-page-data(name, params, callback) —
|
||
// this function provides the concrete transport.
|
||
var url = "/sx/data/" + encodeURIComponent(pageName);
|
||
if (params && !isNil(params)) {
|
||
var qs = [];
|
||
var ks = Object.keys(params);
|
||
for (var i = 0; i < ks.length; i++) {
|
||
var v = params[ks[i]];
|
||
if (v !== null && v !== undefined && v !== NIL) {
|
||
qs.push(encodeURIComponent(ks[i]) + "=" + encodeURIComponent(v));
|
||
}
|
||
}
|
||
if (qs.length) url += "?" + qs.join("&");
|
||
}
|
||
var headers = { "SX-Request": "true" };
|
||
fetch(url, { headers: headers }).then(function(resp) {
|
||
if (!resp.ok) {
|
||
logWarn("sx:data resolve failed " + resp.status + " for " + pageName);
|
||
return;
|
||
}
|
||
return resp.text().then(function(text) {
|
||
try {
|
||
var exprs = parse(text);
|
||
var data = exprs.length === 1 ? exprs[0] : {};
|
||
callback(data || {});
|
||
} catch (e) {
|
||
logWarn("sx:data parse error for " + pageName + ": " + (e && e.message ? e.message : e));
|
||
}
|
||
});
|
||
}).catch(function(err) {
|
||
logWarn("sx:data resolve error for " + pageName + ": " + (err && err.message ? err.message : err));
|
||
});
|
||
}
|
||
|
||
function parseSxData(text) {
|
||
// Parse SX text into a data value. Returns the first parsed expression,
|
||
// or NIL on error. Used by cache update directives.
|
||
try {
|
||
var exprs = parse(text);
|
||
return exprs.length >= 1 ? exprs[0] : NIL;
|
||
} catch (e) {
|
||
logWarn("sx:cache parse error: " + (e && e.message ? e.message : e));
|
||
return NIL;
|
||
}
|
||
}
|
||
|
||
function swPostMessage(msg) {
|
||
// Send a message to the active service worker (if registered).
|
||
// Used to notify SW of cache invalidation.
|
||
if (typeof navigator !== "undefined" && navigator.serviceWorker &&
|
||
navigator.serviceWorker.controller) {
|
||
navigator.serviceWorker.controller.postMessage(msg);
|
||
}
|
||
}
|
||
|
||
function urlPathname(href) {
|
||
try {
|
||
return new URL(href, location.href).pathname;
|
||
} catch (e) {
|
||
// Fallback: strip query/hash
|
||
var idx = href.indexOf("?");
|
||
if (idx >= 0) href = href.substring(0, idx);
|
||
idx = href.indexOf("#");
|
||
if (idx >= 0) href = href.substring(0, idx);
|
||
return href;
|
||
}
|
||
}
|
||
|
||
// --- Preload binding ---
|
||
|
||
function bindPreload(el, events, debounceMs, fn) {
|
||
var timer = null;
|
||
events.forEach(function(evt) {
|
||
el.addEventListener(evt, function() {
|
||
if (debounceMs) {
|
||
clearTimeout(timer);
|
||
timer = setTimeout(fn, debounceMs);
|
||
} else {
|
||
fn();
|
||
}
|
||
}, { passive: true });
|
||
});
|
||
}
|
||
|
||
// --- Processing markers ---
|
||
|
||
var PROCESSED = "_sxBound";
|
||
|
||
function markProcessed(el, key) { el[PROCESSED + key] = true; }
|
||
function isProcessed(el, key) { return !!el[PROCESSED + key]; }
|
||
function clearProcessed(el, key) { delete el[PROCESSED + key]; }
|
||
|
||
// --- Script cloning ---
|
||
|
||
function createScriptClone(dead) {
|
||
var live = document.createElement("script");
|
||
for (var i = 0; i < dead.attributes.length; i++)
|
||
live.setAttribute(dead.attributes[i].name, dead.attributes[i].value);
|
||
live.textContent = dead.textContent;
|
||
return live;
|
||
}
|
||
|
||
// --- SX API references ---
|
||
|
||
function sxRender(source) {
|
||
var SxObj = typeof Sx !== "undefined" ? Sx : null;
|
||
if (SxObj && SxObj.render) return SxObj.render(source);
|
||
throw new Error("No SX renderer available");
|
||
}
|
||
|
||
function sxProcessScripts(root) {
|
||
var SxObj = typeof Sx !== "undefined" ? Sx : null;
|
||
var r = (root && root !== NIL) ? root : undefined;
|
||
if (SxObj && SxObj.processScripts) SxObj.processScripts(r);
|
||
}
|
||
|
||
function sxHydrate(root) {
|
||
var SxObj = typeof Sx !== "undefined" ? Sx : null;
|
||
var r = (root && root !== NIL) ? root : undefined;
|
||
if (SxObj && SxObj.hydrate) SxObj.hydrate(r);
|
||
}
|
||
|
||
function loadedComponentNames() {
|
||
var SxObj = typeof Sx !== "undefined" ? Sx : null;
|
||
if (!SxObj) return [];
|
||
var env = SxObj.componentEnv || (SxObj.getEnv ? SxObj.getEnv() : {});
|
||
return Object.keys(env).filter(function(k) { return k.charAt(0) === "~"; });
|
||
}
|
||
|
||
// --- Response processing ---
|
||
|
||
function stripComponentScripts(text) {
|
||
var SxObj = typeof Sx !== "undefined" ? Sx : null;
|
||
return text.replace(/<script[^>]*type="text\/sx"[^>]*data-components[^>]*>([\s\S]*?)<\/script>/gi,
|
||
function(_, defs) { if (SxObj && SxObj.loadComponents) SxObj.loadComponents(defs); return ""; });
|
||
}
|
||
|
||
function stripSxScripts(text) {
|
||
// Strip <script type="text/sx">...</script> (without data-components or data-init).
|
||
// These contain extra component defs from streaming resolve chunks.
|
||
// data-init scripts are preserved for process-sx-scripts to evaluate as side effects.
|
||
var SxObj = typeof Sx !== "undefined" ? Sx : null;
|
||
return text.replace(/<script[^>]*type="text\/sx"[^>]*>[\s\S]*?<\/script>/gi,
|
||
function(match) {
|
||
if (/data-init/.test(match)) return match; // preserve data-init scripts
|
||
var m = match.match(/<script[^>]*>([\s\S]*?)<\/script>/i);
|
||
if (m && SxObj && SxObj.loadComponents) SxObj.loadComponents(m[1]);
|
||
return "";
|
||
});
|
||
}
|
||
|
||
function extractResponseCss(text) {
|
||
if (!_hasDom) return text;
|
||
var target = document.getElementById("sx-css");
|
||
if (!target) return text;
|
||
return text.replace(/<style[^>]*data-sx-css[^>]*>([\s\S]*?)<\/style>/gi,
|
||
function(_, css) { target.textContent += css; return ""; });
|
||
}
|
||
|
||
function selectFromContainer(container, sel) {
|
||
var frag = document.createDocumentFragment();
|
||
sel.split(",").forEach(function(s) {
|
||
container.querySelectorAll(s.trim()).forEach(function(m) { frag.appendChild(m); });
|
||
});
|
||
return frag;
|
||
}
|
||
|
||
function childrenToFragment(container) {
|
||
var frag = document.createDocumentFragment();
|
||
while (container.firstChild) frag.appendChild(container.firstChild);
|
||
return frag;
|
||
}
|
||
|
||
function selectHtmlFromDoc(doc, sel) {
|
||
var parts = sel.split(",").map(function(s) { return s.trim(); });
|
||
var frags = [];
|
||
parts.forEach(function(s) {
|
||
doc.querySelectorAll(s).forEach(function(m) { frags.push(m.outerHTML); });
|
||
});
|
||
return frags.join("");
|
||
}
|
||
|
||
// --- Parsing ---
|
||
|
||
function tryParseJson(s) {
|
||
if (!s) return NIL;
|
||
try { return JSON.parse(s); } catch (e) { return NIL; }
|
||
}
|
||
|
||
|
||
// =========================================================================
|
||
// Platform interface — Boot (mount, hydrate, scripts, cookies)
|
||
// =========================================================================
|
||
|
||
function preloadIslandDefs() { return NIL; }
|
||
|
||
function resolveMountTarget(target) {
|
||
if (typeof target === "string") return _hasDom ? document.querySelector(target) : null;
|
||
return target;
|
||
}
|
||
|
||
function sxRenderWithEnv(source, extraEnv) {
|
||
var env = extraEnv ? merge(componentEnv, extraEnv) : componentEnv;
|
||
var exprs = parse(source);
|
||
if (!_hasDom) return null;
|
||
var frag = document.createDocumentFragment();
|
||
for (var i = 0; i < exprs.length; i++) {
|
||
var node = renderToDom(exprs[i], env, null);
|
||
if (node && !node._spread) frag.appendChild(node);
|
||
}
|
||
return frag;
|
||
}
|
||
|
||
function getRenderEnv(extraEnv) {
|
||
return extraEnv ? merge(componentEnv, PRIMITIVES, extraEnv) : merge(componentEnv, PRIMITIVES);
|
||
}
|
||
|
||
function mergeEnvs(base, newEnv) {
|
||
return newEnv ? merge(componentEnv, base, newEnv) : merge(componentEnv, base);
|
||
}
|
||
|
||
function sxLoadComponents(text) {
|
||
try {
|
||
var exprs = parse(text);
|
||
for (var i = 0; i < exprs.length; i++) trampoline(evalExpr(exprs[i], componentEnv));
|
||
} catch (err) {
|
||
logParseError("loadComponents", text, err);
|
||
throw err;
|
||
}
|
||
}
|
||
|
||
function setDocumentTitle(s) {
|
||
if (_hasDom) document.title = s || "";
|
||
}
|
||
|
||
function removeHeadElement(sel) {
|
||
if (!_hasDom) return;
|
||
var old = document.head.querySelector(sel);
|
||
if (old) old.parentNode.removeChild(old);
|
||
}
|
||
|
||
function querySxScripts(root) {
|
||
if (!_hasDom) return [];
|
||
var r = (root && root !== NIL) ? root : document;
|
||
return Array.prototype.slice.call(
|
||
r.querySelectorAll('script[type="text/sx"]'));
|
||
}
|
||
|
||
function queryPageScripts() {
|
||
if (!_hasDom) return [];
|
||
return Array.prototype.slice.call(
|
||
document.querySelectorAll('script[type="text/sx-pages"]'));
|
||
}
|
||
|
||
// --- localStorage ---
|
||
|
||
function localStorageGet(key) {
|
||
try { var v = localStorage.getItem(key); return v === null ? NIL : v; }
|
||
catch (e) { return NIL; }
|
||
}
|
||
|
||
function localStorageSet(key, val) {
|
||
try { localStorage.setItem(key, val); } catch (e) {}
|
||
}
|
||
|
||
function localStorageRemove(key) {
|
||
try { localStorage.removeItem(key); } catch (e) {}
|
||
}
|
||
// localStorage primitives registered in CEK_FIXUPS_JS for ordering
|
||
|
||
// --- Cookies ---
|
||
|
||
function setSxCompCookie(hash) {
|
||
if (_hasDom) document.cookie = "sx-comp-hash=" + hash + ";path=/;max-age=31536000;SameSite=Lax";
|
||
}
|
||
|
||
function clearSxCompCookie() {
|
||
if (_hasDom) document.cookie = "sx-comp-hash=;path=/;max-age=0;SameSite=Lax";
|
||
}
|
||
|
||
// --- Env helpers ---
|
||
|
||
function parseEnvAttr(el) {
|
||
var attr = el && el.getAttribute ? el.getAttribute("data-sx-env") : null;
|
||
if (!attr) return {};
|
||
try { return JSON.parse(attr); } catch (e) { return {}; }
|
||
}
|
||
|
||
function storeEnvAttr(el, base, newEnv) {
|
||
var merged = merge(base, newEnv);
|
||
if (el && el.setAttribute) el.setAttribute("data-sx-env", JSON.stringify(merged));
|
||
}
|
||
|
||
function toKebab(s) { return s.replace(/_/g, "-"); }
|
||
|
||
// --- Logging ---
|
||
|
||
function logInfo(msg) {
|
||
if (typeof console !== "undefined") console.log("[sx-ref] " + msg);
|
||
}
|
||
|
||
function logWarn(msg) {
|
||
if (typeof console !== "undefined") console.warn("[sx-ref] " + msg);
|
||
}
|
||
|
||
function logParseError(label, text, err) {
|
||
if (typeof console === "undefined") return;
|
||
var msg = err && err.message ? err.message : String(err);
|
||
var colMatch = msg.match(/col (\d+)/);
|
||
var lineMatch = msg.match(/line (\d+)/);
|
||
if (colMatch && text) {
|
||
var errLine = lineMatch ? parseInt(lineMatch[1]) : 1;
|
||
var errCol = parseInt(colMatch[1]);
|
||
var lines = text.split("\n");
|
||
var pos = 0;
|
||
for (var i = 0; i < errLine - 1 && i < lines.length; i++) pos += lines[i].length + 1;
|
||
pos += errCol;
|
||
var ws = 80;
|
||
var start = Math.max(0, pos - ws);
|
||
var end = Math.min(text.length, pos + ws);
|
||
console.error("[sx-ref] " + label + ":", msg,
|
||
"\n around error (pos ~" + pos + "):",
|
||
"\n \u00ab" + text.substring(start, pos) + "\u26d4" + text.substring(pos, end) + "\u00bb");
|
||
} else {
|
||
console.error("[sx-ref] " + label + ":", msg);
|
||
}
|
||
}
|
||
|
||
|
||
|
||
// =========================================================================
|
||
// Async IO: Promise-aware rendering for client-side IO primitives
|
||
// =========================================================================
|
||
//
|
||
// IO primitives (query, current-user, etc.) return Promises on the client.
|
||
// asyncRenderToDom walks the component tree; when it encounters an IO
|
||
// primitive, it awaits the Promise and continues rendering.
|
||
//
|
||
// The sync evaluator/renderer is untouched. This is a separate async path
|
||
// used only when a page's component tree contains IO references.
|
||
|
||
var IO_PRIMITIVES = {};
|
||
|
||
function registerIoPrimitive(name, fn) {
|
||
IO_PRIMITIVES[name] = fn;
|
||
}
|
||
|
||
function isPromise(x) {
|
||
return x != null && typeof x === "object" && typeof x.then === "function";
|
||
}
|
||
|
||
// Async trampoline: resolves thunks, awaits Promises
|
||
function asyncTrampoline(val) {
|
||
if (isPromise(val)) return val.then(asyncTrampoline);
|
||
if (isThunk(val)) return asyncTrampoline(evalExpr(thunkExpr(val), thunkEnv(val)));
|
||
return val;
|
||
}
|
||
|
||
// Async eval: like trampoline(evalExpr(...)) but handles IO primitives
|
||
function asyncEval(expr, env) {
|
||
// Intercept IO primitive calls at the AST level
|
||
if (Array.isArray(expr) && expr.length > 0) {
|
||
var head = expr[0];
|
||
if (head && head._sym) {
|
||
var name = head.name;
|
||
if (IO_PRIMITIVES[name]) {
|
||
// Evaluate args, then call the IO primitive
|
||
return asyncEvalIoCall(name, expr.slice(1), env);
|
||
}
|
||
}
|
||
}
|
||
// Non-IO: use sync eval, but result might be a thunk
|
||
var result = evalExpr(expr, env);
|
||
return asyncTrampoline(result);
|
||
}
|
||
|
||
function asyncEvalIoCall(name, rawArgs, env) {
|
||
// Parse keyword args and positional args, evaluating each (may be async)
|
||
var kwargs = {};
|
||
var args = [];
|
||
var promises = [];
|
||
var i = 0;
|
||
while (i < rawArgs.length) {
|
||
var arg = rawArgs[i];
|
||
if (arg && arg._kw && (i + 1) < rawArgs.length) {
|
||
var kName = arg.name;
|
||
var kVal = asyncEval(rawArgs[i + 1], env);
|
||
if (isPromise(kVal)) {
|
||
(function(k) { promises.push(kVal.then(function(v) { kwargs[k] = v; })); })(kName);
|
||
} else {
|
||
kwargs[kName] = kVal;
|
||
}
|
||
i += 2;
|
||
} else {
|
||
var aVal = asyncEval(arg, env);
|
||
if (isPromise(aVal)) {
|
||
(function(idx) { promises.push(aVal.then(function(v) { args[idx] = v; })); })(args.length);
|
||
args.push(null); // placeholder
|
||
} else {
|
||
args.push(aVal);
|
||
}
|
||
i++;
|
||
}
|
||
}
|
||
var ioFn = IO_PRIMITIVES[name];
|
||
if (promises.length > 0) {
|
||
return Promise.all(promises).then(function() { return ioFn(args, kwargs); });
|
||
}
|
||
return ioFn(args, kwargs);
|
||
}
|
||
|
||
// Async render-to-dom: returns Promise<Node> or Node
|
||
function asyncRenderToDom(expr, env, ns) {
|
||
// Literals
|
||
if (expr === NIL || expr === null || expr === undefined) return null;
|
||
if (expr === true || expr === false) return null;
|
||
if (typeof expr === "string") return document.createTextNode(expr);
|
||
if (typeof expr === "number") return document.createTextNode(String(expr));
|
||
|
||
// Symbol -> async eval then render
|
||
if (expr && expr._sym) {
|
||
var val = asyncEval(expr, env);
|
||
if (isPromise(val)) return val.then(function(v) { return asyncRenderToDom(v, env, ns); });
|
||
return asyncRenderToDom(val, env, ns);
|
||
}
|
||
|
||
// Keyword
|
||
if (expr && expr._kw) return document.createTextNode(expr.name);
|
||
|
||
// DocumentFragment / DOM nodes pass through
|
||
if (expr instanceof DocumentFragment || (expr && expr.nodeType)) return expr;
|
||
|
||
// Dict -> skip
|
||
if (expr && typeof expr === "object" && !Array.isArray(expr)) return null;
|
||
|
||
// List
|
||
if (!Array.isArray(expr) || expr.length === 0) return null;
|
||
|
||
var head = expr[0];
|
||
if (!head) return null;
|
||
|
||
// Symbol head
|
||
if (head._sym) {
|
||
var hname = head.name;
|
||
|
||
// IO primitive
|
||
if (IO_PRIMITIVES[hname]) {
|
||
var ioResult = asyncEval(expr, env);
|
||
if (isPromise(ioResult)) return ioResult.then(function(v) { return asyncRenderToDom(v, env, ns); });
|
||
return asyncRenderToDom(ioResult, env, ns);
|
||
}
|
||
|
||
// Fragment
|
||
if (hname === "<>") return asyncRenderChildren(expr.slice(1), env, ns);
|
||
|
||
// raw!
|
||
if (hname === "raw!") {
|
||
return asyncEvalRaw(expr.slice(1), env);
|
||
}
|
||
|
||
// Special forms that need async handling
|
||
if (hname === "if") return asyncRenderIf(expr, env, ns);
|
||
if (hname === "when") return asyncRenderWhen(expr, env, ns);
|
||
if (hname === "cond") return asyncRenderCond(expr, env, ns);
|
||
if (hname === "case") return asyncRenderCase(expr, env, ns);
|
||
if (hname === "let" || hname === "let*") return asyncRenderLet(expr, env, ns);
|
||
if (hname === "begin" || hname === "do") return asyncRenderChildren(expr.slice(1), env, ns);
|
||
if (hname === "map") return asyncRenderMap(expr, env, ns);
|
||
if (hname === "map-indexed") return asyncRenderMapIndexed(expr, env, ns);
|
||
if (hname === "for-each") return asyncRenderMap(expr, env, ns);
|
||
|
||
// define/defcomp/defmacro and custom special forms — eval for side effects
|
||
if (hname === "define" || hname === "defcomp" || hname === "defmacro" ||
|
||
hname === "defstyle" || hname === "defhandler" ||
|
||
hname === "deftype" || hname === "defeffect" ||
|
||
(typeof _customSpecialForms !== "undefined" && _customSpecialForms[hname])) {
|
||
trampoline(evalExpr(expr, env));
|
||
return null;
|
||
}
|
||
|
||
// quote
|
||
if (hname === "quote") return null;
|
||
|
||
// lambda/fn
|
||
if (hname === "lambda" || hname === "fn") {
|
||
trampoline(evalExpr(expr, env));
|
||
return null;
|
||
}
|
||
|
||
// and/or — eval and render result
|
||
if (hname === "and" || hname === "or" || hname === "->") {
|
||
var aoResult = asyncEval(expr, env);
|
||
if (isPromise(aoResult)) return aoResult.then(function(v) { return asyncRenderToDom(v, env, ns); });
|
||
return asyncRenderToDom(aoResult, env, ns);
|
||
}
|
||
|
||
// set!
|
||
if (hname === "set!") {
|
||
asyncEval(expr, env);
|
||
return null;
|
||
}
|
||
|
||
// Component or Island
|
||
if (hname.charAt(0) === "~") {
|
||
var comp = env[hname];
|
||
if (comp && comp._island) return renderDomIsland(comp, expr.slice(1), env, ns);
|
||
if (comp && comp._component) return asyncRenderComponent(comp, expr.slice(1), env, ns);
|
||
if (comp && comp._macro) {
|
||
var expanded = trampoline(expandMacro(comp, expr.slice(1), env));
|
||
return asyncRenderToDom(expanded, env, ns);
|
||
}
|
||
}
|
||
|
||
// Macro
|
||
if (env[hname] && env[hname]._macro) {
|
||
var mac = env[hname];
|
||
var expanded = trampoline(expandMacro(mac, expr.slice(1), env));
|
||
return asyncRenderToDom(expanded, env, ns);
|
||
}
|
||
|
||
// HTML tag
|
||
if (typeof renderDomElement === "function" && contains(HTML_TAGS, hname)) {
|
||
return asyncRenderElement(hname, expr.slice(1), env, ns);
|
||
}
|
||
|
||
// html: prefix
|
||
if (hname.indexOf("html:") === 0) {
|
||
return asyncRenderElement(hname.slice(5), expr.slice(1), env, ns);
|
||
}
|
||
|
||
// Custom element
|
||
if (hname.indexOf("-") >= 0 && expr.length > 1 && expr[1] && expr[1]._kw) {
|
||
return asyncRenderElement(hname, expr.slice(1), env, ns);
|
||
}
|
||
|
||
// SVG context
|
||
if (ns) return asyncRenderElement(hname, expr.slice(1), env, ns);
|
||
|
||
// Fallback: eval and render
|
||
var fResult = asyncEval(expr, env);
|
||
if (isPromise(fResult)) return fResult.then(function(v) { return asyncRenderToDom(v, env, ns); });
|
||
return asyncRenderToDom(fResult, env, ns);
|
||
}
|
||
|
||
// Non-symbol head: eval call
|
||
var cResult = asyncEval(expr, env);
|
||
if (isPromise(cResult)) return cResult.then(function(v) { return asyncRenderToDom(v, env, ns); });
|
||
return asyncRenderToDom(cResult, env, ns);
|
||
}
|
||
|
||
function asyncRenderChildren(exprs, env, ns) {
|
||
var frag = document.createDocumentFragment();
|
||
var pending = [];
|
||
for (var i = 0; i < exprs.length; i++) {
|
||
var result = asyncRenderToDom(exprs[i], env, ns);
|
||
if (isPromise(result)) {
|
||
// Insert placeholder, replace when resolved
|
||
var placeholder = document.createComment("async");
|
||
frag.appendChild(placeholder);
|
||
(function(ph) {
|
||
pending.push(result.then(function(node) {
|
||
if (node) ph.parentNode.replaceChild(node, ph);
|
||
else ph.parentNode.removeChild(ph);
|
||
}));
|
||
})(placeholder);
|
||
} else if (result && !result._spread) {
|
||
frag.appendChild(result);
|
||
}
|
||
}
|
||
if (pending.length > 0) {
|
||
return Promise.all(pending).then(function() { return frag; });
|
||
}
|
||
return frag;
|
||
}
|
||
|
||
function asyncRenderElement(tag, args, env, ns) {
|
||
var newNs = tag === "svg" ? SVG_NS : tag === "math" ? MATH_NS : ns;
|
||
var el = domCreateElement(tag, newNs);
|
||
var pending = [];
|
||
var isVoid = contains(VOID_ELEMENTS, tag);
|
||
for (var i = 0; i < args.length; i++) {
|
||
var arg = args[i];
|
||
if (arg && arg._kw && (i + 1) < args.length) {
|
||
var attrName = arg.name;
|
||
var attrVal = asyncEval(args[i + 1], env);
|
||
i++;
|
||
if (isPromise(attrVal)) {
|
||
(function(an, av) {
|
||
pending.push(av.then(function(v) {
|
||
if (!isNil(v) && v !== false) {
|
||
if (contains(BOOLEAN_ATTRS, an)) { if (isSxTruthy(v)) el.setAttribute(an, ""); }
|
||
else if (v === true) el.setAttribute(an, "");
|
||
else el.setAttribute(an, String(v));
|
||
}
|
||
}));
|
||
})(attrName, attrVal);
|
||
} else {
|
||
if (!isNil(attrVal) && attrVal !== false) {
|
||
if (contains(BOOLEAN_ATTRS, attrName)) {
|
||
if (isSxTruthy(attrVal)) el.setAttribute(attrName, "");
|
||
} else if (attrVal === true) {
|
||
el.setAttribute(attrName, "");
|
||
} else {
|
||
el.setAttribute(attrName, String(attrVal));
|
||
}
|
||
}
|
||
}
|
||
} else if (!isVoid) {
|
||
var child = asyncRenderToDom(arg, env, newNs);
|
||
if (isPromise(child)) {
|
||
var placeholder = document.createComment("async");
|
||
el.appendChild(placeholder);
|
||
(function(ph) {
|
||
pending.push(child.then(function(node) {
|
||
if (node) ph.parentNode.replaceChild(node, ph);
|
||
else ph.parentNode.removeChild(ph);
|
||
}));
|
||
})(placeholder);
|
||
} else if (child) {
|
||
if (child._spread) {
|
||
// Spread: merge attrs onto parent element
|
||
var sa = child.attrs || {};
|
||
for (var sk in sa) {
|
||
if (sk === "class") {
|
||
var ec = el.getAttribute("class") || "";
|
||
el.setAttribute("class", ec ? ec + " " + sa[sk] : sa[sk]);
|
||
} else if (sk === "style") {
|
||
var es = el.getAttribute("style") || "";
|
||
el.setAttribute("style", es ? es + ";" + sa[sk] : sa[sk]);
|
||
} else {
|
||
el.setAttribute(sk, String(sa[sk]));
|
||
}
|
||
}
|
||
} else {
|
||
el.appendChild(child);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
if (pending.length > 0) return Promise.all(pending).then(function() { return el; });
|
||
return el;
|
||
}
|
||
|
||
function asyncRenderComponent(comp, args, env, ns) {
|
||
var kwargs = {};
|
||
var children = [];
|
||
var pending = [];
|
||
for (var i = 0; i < args.length; i++) {
|
||
var arg = args[i];
|
||
if (arg && arg._kw && (i + 1) < args.length) {
|
||
var kName = arg.name;
|
||
var kVal = asyncEval(args[i + 1], env);
|
||
if (isPromise(kVal)) {
|
||
(function(k) { pending.push(kVal.then(function(v) { kwargs[k] = v; })); })(kName);
|
||
} else {
|
||
kwargs[kName] = kVal;
|
||
}
|
||
i++;
|
||
} else {
|
||
children.push(arg);
|
||
}
|
||
}
|
||
|
||
function doRender() {
|
||
var local = Object.create(componentClosure(comp));
|
||
for (var k in env) if (env.hasOwnProperty(k)) local[k] = env[k];
|
||
var params = componentParams(comp);
|
||
for (var j = 0; j < params.length; j++) {
|
||
local[params[j]] = params[j] in kwargs ? kwargs[params[j]] : NIL;
|
||
}
|
||
if (componentHasChildren(comp)) {
|
||
var childResult = asyncRenderChildren(children, env, ns);
|
||
if (isPromise(childResult)) {
|
||
return childResult.then(function(childFrag) {
|
||
local["children"] = childFrag;
|
||
return asyncRenderToDom(componentBody(comp), local, ns);
|
||
});
|
||
}
|
||
local["children"] = childResult;
|
||
}
|
||
return asyncRenderToDom(componentBody(comp), local, ns);
|
||
}
|
||
|
||
if (pending.length > 0) return Promise.all(pending).then(doRender);
|
||
return doRender();
|
||
}
|
||
|
||
function asyncRenderIf(expr, env, ns) {
|
||
var cond = asyncEval(expr[1], env);
|
||
if (isPromise(cond)) {
|
||
return cond.then(function(v) {
|
||
return isSxTruthy(v)
|
||
? asyncRenderToDom(expr[2], env, ns)
|
||
: (expr.length > 3 ? asyncRenderToDom(expr[3], env, ns) : null);
|
||
});
|
||
}
|
||
return isSxTruthy(cond)
|
||
? asyncRenderToDom(expr[2], env, ns)
|
||
: (expr.length > 3 ? asyncRenderToDom(expr[3], env, ns) : null);
|
||
}
|
||
|
||
function asyncRenderWhen(expr, env, ns) {
|
||
var cond = asyncEval(expr[1], env);
|
||
if (isPromise(cond)) {
|
||
return cond.then(function(v) {
|
||
return isSxTruthy(v) ? asyncRenderChildren(expr.slice(2), env, ns) : null;
|
||
});
|
||
}
|
||
return isSxTruthy(cond) ? asyncRenderChildren(expr.slice(2), env, ns) : null;
|
||
}
|
||
|
||
function asyncRenderCond(expr, env, ns) {
|
||
var clauses = expr.slice(1);
|
||
function step(idx) {
|
||
if (idx >= clauses.length) return null;
|
||
var clause = clauses[idx];
|
||
if (!Array.isArray(clause) || clause.length < 2) return step(idx + 1);
|
||
var test = clause[0];
|
||
if ((test && test._sym && (test.name === "else" || test.name === ":else")) ||
|
||
(test && test._kw && test.name === "else")) {
|
||
return asyncRenderToDom(clause[1], env, ns);
|
||
}
|
||
var v = asyncEval(test, env);
|
||
if (isPromise(v)) return v.then(function(r) { return isSxTruthy(r) ? asyncRenderToDom(clause[1], env, ns) : step(idx + 1); });
|
||
return isSxTruthy(v) ? asyncRenderToDom(clause[1], env, ns) : step(idx + 1);
|
||
}
|
||
return step(0);
|
||
}
|
||
|
||
function asyncRenderCase(expr, env, ns) {
|
||
var matchVal = asyncEval(expr[1], env);
|
||
function doCase(mv) {
|
||
var clauses = expr.slice(2);
|
||
for (var i = 0; i < clauses.length - 1; i += 2) {
|
||
var test = clauses[i];
|
||
if ((test && test._kw && test.name === "else") ||
|
||
(test && test._sym && (test.name === "else" || test.name === ":else"))) {
|
||
return asyncRenderToDom(clauses[i + 1], env, ns);
|
||
}
|
||
var tv = trampoline(evalExpr(test, env));
|
||
if (mv === tv || (typeof mv === "string" && typeof tv === "string" && mv === tv)) {
|
||
return asyncRenderToDom(clauses[i + 1], env, ns);
|
||
}
|
||
}
|
||
return null;
|
||
}
|
||
if (isPromise(matchVal)) return matchVal.then(doCase);
|
||
return doCase(matchVal);
|
||
}
|
||
|
||
function asyncRenderLet(expr, env, ns) {
|
||
var bindings = expr[1];
|
||
var local = Object.create(env);
|
||
for (var k in env) if (env.hasOwnProperty(k)) local[k] = env[k];
|
||
function bindStep(idx) {
|
||
if (!Array.isArray(bindings)) return asyncRenderChildren(expr.slice(2), local, ns);
|
||
// Nested pairs: ((a 1) (b 2))
|
||
if (bindings.length > 0 && Array.isArray(bindings[0])) {
|
||
if (idx >= bindings.length) return asyncRenderChildren(expr.slice(2), local, ns);
|
||
var b = bindings[idx];
|
||
var vname = b[0]._sym ? b[0].name : String(b[0]);
|
||
var val = asyncEval(b[1], local);
|
||
if (isPromise(val)) return val.then(function(v) { local[vname] = v; return bindStep(idx + 1); });
|
||
local[vname] = val;
|
||
return bindStep(idx + 1);
|
||
}
|
||
// Flat pairs: (a 1 b 2)
|
||
if (idx >= bindings.length) return asyncRenderChildren(expr.slice(2), local, ns);
|
||
var vn = bindings[idx]._sym ? bindings[idx].name : String(bindings[idx]);
|
||
var vv = asyncEval(bindings[idx + 1], local);
|
||
if (isPromise(vv)) return vv.then(function(v) { local[vn] = v; return bindStep(idx + 2); });
|
||
local[vn] = vv;
|
||
return bindStep(idx + 2);
|
||
}
|
||
return bindStep(0);
|
||
}
|
||
|
||
function asyncRenderMap(expr, env, ns) {
|
||
var fn = asyncEval(expr[1], env);
|
||
var coll = asyncEval(expr[2], env);
|
||
function doMap(f, c) {
|
||
if (!Array.isArray(c)) return null;
|
||
var frag = document.createDocumentFragment();
|
||
var pending = [];
|
||
for (var i = 0; i < c.length; i++) {
|
||
var item = c[i];
|
||
var result;
|
||
if (f && f._lambda) {
|
||
var lenv = Object.create(f.closure || env);
|
||
for (var k in env) if (env.hasOwnProperty(k)) lenv[k] = env[k];
|
||
lenv[f.params[0]] = item;
|
||
result = asyncRenderToDom(f.body, lenv, null);
|
||
} else if (typeof f === "function") {
|
||
var r = f(item);
|
||
result = isPromise(r) ? r.then(function(v) { return asyncRenderToDom(v, env, null); }) : asyncRenderToDom(r, env, null);
|
||
} else {
|
||
result = asyncRenderToDom(item, env, null);
|
||
}
|
||
if (isPromise(result)) {
|
||
var ph = document.createComment("async");
|
||
frag.appendChild(ph);
|
||
(function(p) { pending.push(result.then(function(n) { if (n) p.parentNode.replaceChild(n, p); else p.parentNode.removeChild(p); })); })(ph);
|
||
} else if (result && !result._spread) {
|
||
frag.appendChild(result);
|
||
}
|
||
}
|
||
if (pending.length) return Promise.all(pending).then(function() { return frag; });
|
||
return frag;
|
||
}
|
||
if (isPromise(fn) || isPromise(coll)) {
|
||
return Promise.all([isPromise(fn) ? fn : Promise.resolve(fn), isPromise(coll) ? coll : Promise.resolve(coll)])
|
||
.then(function(r) { return doMap(r[0], r[1]); });
|
||
}
|
||
return doMap(fn, coll);
|
||
}
|
||
|
||
function asyncRenderMapIndexed(expr, env, ns) {
|
||
var fn = asyncEval(expr[1], env);
|
||
var coll = asyncEval(expr[2], env);
|
||
function doMap(f, c) {
|
||
if (!Array.isArray(c)) return null;
|
||
var frag = document.createDocumentFragment();
|
||
var pending = [];
|
||
for (var i = 0; i < c.length; i++) {
|
||
var item = c[i];
|
||
var result;
|
||
if (f && f._lambda) {
|
||
var lenv = Object.create(f.closure || env);
|
||
for (var k in env) if (env.hasOwnProperty(k)) lenv[k] = env[k];
|
||
lenv[f.params[0]] = i;
|
||
lenv[f.params[1]] = item;
|
||
result = asyncRenderToDom(f.body, lenv, null);
|
||
} else if (typeof f === "function") {
|
||
var r = f(i, item);
|
||
result = isPromise(r) ? r.then(function(v) { return asyncRenderToDom(v, env, null); }) : asyncRenderToDom(r, env, null);
|
||
} else {
|
||
result = asyncRenderToDom(item, env, null);
|
||
}
|
||
if (isPromise(result)) {
|
||
var ph = document.createComment("async");
|
||
frag.appendChild(ph);
|
||
(function(p) { pending.push(result.then(function(n) { if (n) p.parentNode.replaceChild(n, p); else p.parentNode.removeChild(p); })); })(ph);
|
||
} else if (result && !result._spread) {
|
||
frag.appendChild(result);
|
||
}
|
||
}
|
||
if (pending.length) return Promise.all(pending).then(function() { return frag; });
|
||
return frag;
|
||
}
|
||
if (isPromise(fn) || isPromise(coll)) {
|
||
return Promise.all([isPromise(fn) ? fn : Promise.resolve(fn), isPromise(coll) ? coll : Promise.resolve(coll)])
|
||
.then(function(r) { return doMap(r[0], r[1]); });
|
||
}
|
||
return doMap(fn, coll);
|
||
}
|
||
|
||
function asyncEvalRaw(args, env) {
|
||
var parts = [];
|
||
var pending = [];
|
||
for (var i = 0; i < args.length; i++) {
|
||
var val = asyncEval(args[i], env);
|
||
if (isPromise(val)) {
|
||
(function(idx) {
|
||
pending.push(val.then(function(v) { parts[idx] = v; }));
|
||
})(parts.length);
|
||
parts.push(null);
|
||
} else {
|
||
parts.push(val);
|
||
}
|
||
}
|
||
function assemble() {
|
||
var html = "";
|
||
for (var j = 0; j < parts.length; j++) {
|
||
var p = parts[j];
|
||
if (p && p._rawHtml) html += p.html;
|
||
else if (typeof p === "string") html += p;
|
||
else if (p != null && !isNil(p)) html += String(p);
|
||
}
|
||
var el = document.createElement("span");
|
||
el.innerHTML = html;
|
||
var frag = document.createDocumentFragment();
|
||
while (el.firstChild) frag.appendChild(el.firstChild);
|
||
return frag;
|
||
}
|
||
if (pending.length) return Promise.all(pending).then(assemble);
|
||
return assemble();
|
||
}
|
||
|
||
// Async version of sxRenderWithEnv — returns Promise<DocumentFragment>
|
||
function asyncSxRenderWithEnv(source, extraEnv) {
|
||
var env = extraEnv ? merge(componentEnv, extraEnv) : componentEnv;
|
||
var exprs = parse(source);
|
||
if (!_hasDom) return Promise.resolve(null);
|
||
return asyncRenderChildren(exprs, env, null);
|
||
}
|
||
|
||
// IO proxy cache: key → { value, expires }
|
||
var _ioCache = {};
|
||
var IO_CACHE_TTL = 300000; // 5 minutes
|
||
|
||
// Register a server-proxied IO primitive: fetches from /sx/io/<name>
|
||
// Uses GET for short args, POST for long payloads (URL length safety).
|
||
// Results are cached client-side by (name + args) with a TTL.
|
||
function registerProxiedIo(name) {
|
||
registerIoPrimitive(name, function(args, kwargs) {
|
||
// Cache key: name + serialized args
|
||
var cacheKey = name;
|
||
for (var ci = 0; ci < args.length; ci++) cacheKey += " |