haku: a very shitty tree-walk interpreter

This commit is contained in:
リキ萌え 2024-07-24 23:43:05 +02:00
parent d813675d47
commit b505c1bcfe
8 changed files with 403 additions and 26 deletions

View file

@ -10,3 +10,6 @@ end_of_line = lf
charset = utf-8 charset = utf-8
trim_trailing_whitespace = true trim_trailing_whitespace = true
insert_final_newline = true insert_final_newline = true
[*.js]
max_line_length = 100

View file

@ -2,7 +2,7 @@
scripts = ["treehouse/vendor/codejar.js", "treehouse/components/literate-programming.js"] scripts = ["treehouse/vendor/codejar.js", "treehouse/components/literate-programming.js"]
% id = "01J3K8A0D1774SFDPKDK5G9GPV" % id = "01J3K8A0D1774SFDPKDK5G9GPV"
- I've had this idea on my mind as of late, of a little lazily-evaluated pure functional programming language that would run in your browser. - I've had this idea on my mind as of late, of a little pure functional programming language that would run in your browser.
% id = "01J3K8A0D1WTM2KHERFZG2FWBJ" % id = "01J3K8A0D1WTM2KHERFZG2FWBJ"
+ the primary use case would be writing fun audiovisual sketches you can inspect and edit live, because after all everything is declarative. + the primary use case would be writing fun audiovisual sketches you can inspect and edit live, because after all everything is declarative.
@ -957,6 +957,238 @@ scripts = ["treehouse/vendor/codejar.js", "treehouse/components/literate-program
% id = "01J3K8A0D12VCHW6AJX0ZGPQBY" % id = "01J3K8A0D12VCHW6AJX0ZGPQBY"
- code style-wise it's also not my prettiest Nim code ever - it kind of abuses `template`s for referring to the current character with a single word, but that doesn't convey the fact that it's an effectful operation very well. - code style-wise it's also not my prettiest Nim code ever - it kind of abuses `template`s for referring to the current character with a single word, but that doesn't convey the fact that it's an effectful operation very well.
- ### interpretation
- with a parser now ready, it would be nice if we could execute some actual code!
- we'll again start off by setting a goal.
I want to be able to evaluate arbitrary arithmetic expressions, like this one:
```haku
(+ (* 2 1) 1 (/ 6 2) (- 10 3))
```
- the simplest way to get some code up and running would be to write a _tree-walk interpreter_.
{:program=haku}
```javascript
export const treewalk = {};
```
this kind of interpreter is actually really simple!
it just involves walking through your syntax tree, executing each node one by one.
- we'll again start off by defining a function that initializes our interpreter's state.
right now there isn't really anything to initialize, but recall that we don't have our tokens parsed into any meaningful data yet, so we'll have to have access the source string to do that.
{:program=haku}
```javascript
treewalk.init = (input) => {
return { input };
};
```
- the core of our interpretation will be a function that descends down the node tree and _evaluates_ each node, giving us a result.
{:program=haku}
```javascript
treewalk.eval = (state, node) => {
switch (node.kind) {
default:
throw new Error(`unhandled node kind: ${node.kind}`);
}
};
```
for now we'll leave it empty.
- in the meantime, let's prepare a couple convenient little wrappers to run our code:
{:program=haku}
```javascript
export function run(input, node) {
let state = treewalk.init(input);
return treewalk.eval(state, node);
}
export function printEvalResult(input) {
try {
let tokens = lex(input);
let ast = parse(tokens);
let result = run(input, ast);
console.log(result);
} catch (error) {
console.log(error.toString());
}
}
```
- now we can try running some code!
let's see what happens.
{:program=haku}
```javascript
printEvalResult("65");
```
{:program=haku}
```output
Error: unhandled node kind: integer
```
...of course.
- so let's patch those integers in!
this is where we'll need that source string of ours - we don't actually have a JavaScript `number` representation of the integers, so we'll need to parse them into place.
{:program=haku}
```javascript
treewalk.eval = (state, node) => {
switch (node.kind) {
case "integer":
let sourceString = state.input.substring(node.start, node.end);
return parseInt(sourceString);
default:
throw new Error(`unhandled node kind: ${node.kind}`);
}
};
```
- now when we run the program above...
{:program=haku}
```javascript
printEvalResult("65");
```
{:program=haku}
```output
65
```
we get sixty five!
- but that's of course a bit boring - it would be nice if we could like, y'know, _perform some arithmetic_.
- traditionally, in Lisp-like languages, a list expression always represents a function application, with the head of the list being the function to call, and the tail of the function being the arguments to apply to the function.
let's implement that logic then!
{:program=haku}
```javascript
export const builtins = {};
treewalk.eval = (state, node) => {
switch (node.kind) {
case "integer":
let sourceString = state.input.substring(node.start, node.end);
return parseInt(sourceString);
case "list": // <--
let functionToCall = node.children[0];
let builtin = builtins[state.input.substring(functionToCall.start, functionToCall.end)];
return builtin(state, node);
default:
throw new Error(`unhandled node kind: ${node.kind}`);
}
};
```
- we'm putting all of our built-in magic functions into a separate object `builtins`, so that they're easy to patch partially later.
you've seen my tricks already with hot-patching functions in objects, so this shouldn't be too surprising.
+ you'll note I'm kind of cheating here - because we have no mechanism to represent variables just yet, I'm using the node's text as the key to our `builtins` table.
- heck, I'm not even validating that this is an identifier - so you can technically do something like this, too:
```haku
((what the fuck) lol)
```
which will call the builtin named `(what the fuck)`.
- we could try this out now, except we don't actually have any builtins! so I'll add a few in, so that we can _finally_ perform our glorious arithmetic:
{:program=haku}
```javascript
function arithmeticBuiltin(op) {
return (state, node) => {
let result = treewalk.eval(state, node.children[1]);
for (let i = 2; i < node.children.length; ++i) {
result = op(result, treewalk.eval(state, node.children[i]));
}
return result;
};
}
builtins["+"] = arithmeticBuiltin((a, b) => a + b);
builtins["-"] = arithmeticBuiltin((a, b) => a - b);
builtins["*"] = arithmeticBuiltin((a, b) => a * b);
builtins["/"] = arithmeticBuiltin((a, b) => a / b);
```
- one thing of note is how `arithmeticBuiltin` accepts two or more arguments.
you're free to pass in more than that, which is common among Lisps.
- now let's try running our full arithmetic expression! drum roll please...
{:program=haku}
```javascript
printEvalResult("(+ (* 2 1) 1 (/ 6 2) (- 10 3))");
```
{:program=haku}
```output
13
```
- #### a brief intermission
- I will now pause here to say, I'm kind of tired of writing this `printEvalResult` ceremony over and over again.
so I took a bit of time to enhance the treehouse's capabilities, and it's now capable of running languages other than JavaScript!
- all we have to do is swap out the evaluation [kernel][]{title="like in Jupyter! Jupyter kernels are basically just support for different programming languages" style="cursor: help; text-decoration: 1px dotted underline;"}...
[kernel]: https://docs.jupyter.org/en/latest/projects/kernels.html
{:program=haku}
```javascript
import { getKernel } from "treehouse/components/literate-programming/eval.js";
let kernel = getKernel();
export const defaultKernelInit = kernel.init;
kernel.init = () => {
return defaultKernelInit();
};
export const defaultKernelEvalModule = kernel.evalModule;
kernel.evalModule = async (state, source, language, params) => {
if (language == "haku") {
printEvalResult(source);
return true;
} else {
return await defaultKernelEvalModule(state, source, language, params);
}
};
```
- and now we can write haku in code blocks!
{:program=haku}
```haku
(+ (* 2 1) 1 (/ 6 2) (- 10 3))
```
{:program=haku}
```output
13
```
% stage = "Draft" % stage = "Draft"
id = "01J3K8A0D1D0NTT3JYYFMRYVSC" id = "01J3K8A0D1D0NTT3JYYFMRYVSC"
- ### tests - ### tests
@ -979,4 +1211,28 @@ scripts = ["treehouse/vendor/codejar.js", "treehouse/components/literate-program
{:program=test-parser} {:program=test-parser}
```output ```output
( 0..1 '('
identifier 1..8 'example'
identifier 9..21 's-expression'
) 21..22 ')'
end of file 22..22 ''
(example s-expression)
```
- treewalk
{:program=test-treewalk}
```javascript
import { lex, parse, exprToString } from "haku/sexp.js";
import { run } from "haku/treewalk.js";
let input = "(+ (* 2 1) 1 (/ 6 2) (- 10 3))";
let tokens = lex(input);
let ast = parse(tokens);
console.log(run(input, ast));
```
{:program=test-treewalk}
```output
``` ```

View file

@ -337,6 +337,10 @@ impl<'a> Writer<'a> {
write_attr(program_name, out); write_attr(program_name, out);
out.push('"'); out.push('"');
out.push_str(r#" data-language=""#);
write_attr(language, out);
out.push('"');
if *language == "output" { if *language == "output" {
out.push_str(r#" data-mode="output""#); out.push_str(r#" data-mode="output""#);
} else { } else {

View file

@ -0,0 +1,42 @@
export const treewalk = {};
export const builtins = {};
treewalk.init = (input) => {
return { input };
};
treewalk.eval = (state, node) => {
switch (node.kind) {
case "integer":
let sourceString = state.input.substring(node.start, node.end);
return parseInt(sourceString);
case "list":
let functionToCall = node.children[0];
let builtin = builtins[state.input.substring(functionToCall.start, functionToCall.end)];
return builtin(state, node);
default:
throw new Error(`unhandled node kind: ${node.kind}`);
}
};
export function run(input, node) {
let state = treewalk.init(input);
return treewalk.eval(state, node);
}
function arithmeticBuiltin(op) {
return (state, node) => {
let result = treewalk.eval(state, node.children[1]);
for (let i = 2; i < node.children.length; ++i) {
result = op(result, treewalk.eval(state, node.children[i]));
}
return result;
};
}
builtins["+"] = arithmeticBuiltin((a, b) => a + b);
builtins["-"] = arithmeticBuiltin((a, b) => a - b);
builtins["*"] = arithmeticBuiltin((a, b) => a * b);
builtins["/"] = arithmeticBuiltin((a, b) => a / b);

View file

@ -26,7 +26,12 @@ function getLiterateProgramWorkerCommands(name, count) {
for (let i = 0; i < count; ++i) { for (let i = 0; i < count; ++i) {
let frame = literateProgram.frames[i]; let frame = literateProgram.frames[i];
if (frame.mode == "input") { if (frame.mode == "input") {
commands.push({ kind: "module", source: frame.textContent }); commands.push({
kind: "module",
source: frame.textContent,
language: frame.language,
kernelParameters: frame.kernelAttributes,
});
} else if (frame.mode == "output") { } else if (frame.mode == "output") {
commands.push({ kind: "output" }); commands.push({ kind: "output" });
} }
@ -35,27 +40,42 @@ function getLiterateProgramWorkerCommands(name, count) {
return commands; return commands;
} }
const javascriptJson = await (await fetch(`${TREEHOUSE_SITE}/static/syntax/javascript.json`)).text(); let compiledSyntaxes = new Map();
async function getCompiledSyntax(language) {
if (compiledSyntaxes.has(language)) {
return compiledSyntaxes.get(language);
} else {
let json = await (await fetch(TREEHOUSE_SYNTAX_URLS[language])).text();
let compiled = compileSyntax(JSON.parse(json));
compiledSyntaxes.set(language, compiled);
return compiled;
}
}
class InputMode { class InputMode {
static JAVASCRIPT = compileSyntax(JSON.parse(javascriptJson));
constructor(frame) { constructor(frame) {
this.frame = frame; this.frame = frame;
InputMode.highlight(frame); getCompiledSyntax(this.frame.language).then((syntax) => {
this.codeJar = CodeJar(frame, InputMode.highlight); this.syntax = syntax;
this.highlight();
});
this.codeJar = CodeJar(frame, (frame) => this.highlight(frame));
this.codeJar.onUpdate(() => { this.codeJar.onUpdate(() => {
for (let handler of frame.program.onChanged) { for (let handler of frame.program.onChanged) {
handler(frame.programName); handler(frame.programName);
} }
}) });
frame.addEventListener("click", event => event.preventDefault()); frame.addEventListener("click", (event) => event.preventDefault());
} }
static highlight(frame) { async highlight() {
highlight(frame, InputMode.JAVASCRIPT, (token, span) => { if (this.syntax == null) return;
highlight(this.frame, this.syntax, (token, span) => {
if (token.kind == "keyword1" && token.string == "export") { if (token.kind == "keyword1" && token.string == "export") {
// This is something a bit non-obvious about the treehouse's literate programs // This is something a bit non-obvious about the treehouse's literate programs
// so let's document it. // so let's document it.
@ -68,7 +88,7 @@ class InputMode {
function messageOutputArrayToString(output) { function messageOutputArrayToString(output) {
return output return output
.map(x => { .map((x) => {
if (typeof x === "object") return JSON.stringify(x); if (typeof x === "object") return JSON.stringify(x);
else return x + ""; else return x + "";
}) })
@ -97,7 +117,7 @@ class OutputMode {
this.iframe.contentWindow.treehouseSandboxInternals = { outputIndex: this.outputIndex }; this.iframe.contentWindow.treehouseSandboxInternals = { outputIndex: this.outputIndex };
this.iframe.contentWindow.addEventListener("message", event => { this.iframe.contentWindow.addEventListener("message", (event) => {
let message = event.data; let message = event.data;
if (message.kind == "ready") { if (message.kind == "ready") {
this.evaluate(); this.evaluate();
@ -121,14 +141,17 @@ class OutputMode {
this.frame.placeholderImage.classList.add("loading"); this.frame.placeholderImage.classList.add("loading");
} }
this.frame.program.onChanged.push(_ => this.evaluate()); this.frame.program.onChanged.push((_) => this.evaluate());
} }
evaluate() { evaluate() {
this.requestConsoleClear(); this.requestConsoleClear();
this.iframe.contentWindow.postMessage({ this.iframe.contentWindow.postMessage({
action: "eval", action: "eval",
input: getLiterateProgramWorkerCommands(this.frame.programName, this.frame.frameIndex + 1), input: getLiterateProgramWorkerCommands(
this.frame.programName,
this.frame.frameIndex + 1,
),
}); });
} }
@ -161,7 +184,7 @@ class OutputMode {
// One day this will be more fancy. Today is not that day. // One day this will be more fancy. Today is not that day.
line.textContent = output.message line.textContent = output.message
.map(x => { .map((x) => {
if (typeof x === "object") return JSON.stringify(x); if (typeof x === "object") return JSON.stringify(x);
else return x + ""; else return x + "";
}) })
@ -198,6 +221,7 @@ class OutputMode {
class LiterateProgram extends HTMLElement { class LiterateProgram extends HTMLElement {
connectedCallback() { connectedCallback() {
this.language = this.getAttribute("data-language");
this.programName = this.getAttribute("data-program"); this.programName = this.getAttribute("data-program");
this.frameIndex = this.program.frames.length; this.frameIndex = this.program.frames.length;
this.program.frames.push(this); this.program.frames.push(this);
@ -205,6 +229,13 @@ class LiterateProgram extends HTMLElement {
this.placeholderImage = this.getElementsByClassName("placeholder-image")[0]; this.placeholderImage = this.getElementsByClassName("placeholder-image")[0];
this.placeholderConsole = this.getElementsByClassName("placeholder-console")[0]; this.placeholderConsole = this.getElementsByClassName("placeholder-console")[0];
this.kernelAttributes = {};
for (let name of this.getAttributeNames()) {
if (name.startsWith("k-")) {
this.kernelAttributes[name] = this.getAttribute(name);
}
}
this.mode = this.getAttribute("data-mode"); this.mode = this.getAttribute("data-mode");
if (this.mode == "input") { if (this.mode == "input") {
this.modeImpl = new InputMode(this); this.modeImpl = new InputMode(this);

View file

@ -14,9 +14,32 @@ export const domConsole = {
}, },
outputIndex, outputIndex,
}); });
} },
}; };
let kernel = {
init() {
return {};
},
async evalModule(_state, source, language, _params) {
if (language == "javascript") {
let blobUrl = URL.createObjectURL(new Blob([source], { type: "text/javascript" }));
let module = await import(blobUrl);
for (let exportedKey in module) {
globalThis[exportedKey] = module[exportedKey];
}
return true;
} else {
return false;
}
},
};
export function getKernel() {
return kernel;
}
let evaluationComplete = null; let evaluationComplete = null;
export async function evaluate(commands, { error, newOutput }) { export async function evaluate(commands, { error, newOutput }) {
@ -27,17 +50,20 @@ export async function evaluate(commands, { error, newOutput }) {
let signalEvaluationComplete; let signalEvaluationComplete;
evaluationComplete = new Promise((resolve, _reject) => { evaluationComplete = new Promise((resolve, _reject) => {
signalEvaluationComplete = resolve; signalEvaluationComplete = resolve;
}) });
let kernelState = kernel.init();
outputIndex = 0; outputIndex = 0;
try { try {
for (let command of commands) { for (let command of commands) {
if (command.kind == "module") { if (command.kind == "module") {
let blobUrl = URL.createObjectURL(new Blob([command.source], { type: "text/javascript" })); await kernel.evalModule(
let module = await import(blobUrl); kernelState,
for (let exportedKey in module) { command.source,
globalThis[exportedKey] = module[exportedKey]; command.language,
} command.kernelParameters,
);
} else if (command.kind == "output") { } else if (command.kind == "output") {
if (newOutput != null) { if (newOutput != null) {
newOutput(outputIndex); newOutput(outputIndex);
@ -63,4 +89,3 @@ export async function evaluate(commands, { error, newOutput }) {
} }
signalEvaluationComplete(); signalEvaluationComplete();
} }

View file

@ -42,8 +42,16 @@ function tokenize(text, syntax) {
for (let i = 1; i < match.indices.length; ++i) { for (let i = 1; i < match.indices.length; ++i) {
let [start, end] = match.indices[i]; let [start, end] = match.indices[i];
if (match.indices[i] != null) { if (match.indices[i] != null) {
pushToken(tokens, pattern.is.default, text.substring(lastMatchEnd, start)); pushToken(
pushToken(tokens, pattern.is.captures[i], text.substring(start, end)); tokens,
pattern.is.default,
text.substring(lastMatchEnd, start),
);
pushToken(
tokens,
pattern.is.captures[i - 1],
text.substring(start, end),
);
} }
} }
} else { } else {

View file

@ -17,6 +17,14 @@ clever to do while browser vendors figure that out, we'll just have to do a cach
<script> <script>
const TREEHOUSE_SITE = `{{ config.site }}`; const TREEHOUSE_SITE = `{{ config.site }}`;
const TREEHOUSE_NEWS_COUNT = {{ len feeds.news.branches }}; const TREEHOUSE_NEWS_COUNT = {{ len feeds.news.branches }};
{{!-- Yeah, this should probably be solved in a better way somehow.
For now this is used to allow literate-programming.js to refer to syntax files with the ?cache attribute,
so that they don't need to be redownloaded every single time. --}}
const TREEHOUSE_SYNTAX_URLS = {
javascript: `{{{ asset 'syntax/javascript.json' }}}`,
haku: `{{{ asset 'syntax/haku.json' }}}`,
};
</script> </script>
<script type="module"> <script type="module">
import "treehouse/spells.js"; import "treehouse/spells.js";