prototype-3x5/notcl.js

158 lines
3.7 KiB
JavaScript
Raw Normal View History

2023-07-27 01:54:06 -04:00
/**
* @typedef {Command[]} Script
* @typedef {Word[]} Command
* @typedef {object} Word
* @property {string} text
*/
/**
* A Pattern is a function that matches against a string starting at a given index.
*
* If it matches successfully, it returns some captured value, and the index following the match.
*
* @template T
* @typedef {(source: string, index: number) => ([T, number] | null)} Pattern
*/
/**
* Creates a pattern that wraps another pattern, transforming the returned value on a match
* @template T, U
* @param {Pattern<T>} pattern
* @param {(value: T)=> U} map
* @return {Pattern<U>}
*/
function MapPattern(pattern, map) {
return function (source, index) {
const match = pattern(source, index);
return match ? [map(match[0]), match[1]] : null;
};
}
/**
* Creates a pattern matching a regex & returning any captures. The regex needs to be sticky (using the //y modifier)
* @param {RegExp} regex
* @return {Pattern<RegExpExecArray>}
*/
function RegexPattern(regex) {
return function (source, index) {
regex.lastIndex = index;
const matches = regex.exec(source);
return matches ? [matches, regex.lastIndex] : null;
};
}
/**
* @template T
* @param {...Pattern<T>} patterns
* @return {Pattern<T>}
*/
function Choose(...patterns) {
return function (source, index) {
for (const pattern of patterns) {
const match = pattern(source, index);
if (match) {
return match;
}
}
return null;
};
}
/**
* @template {unknown[]} T
* @param {{[K in keyof T]: Pattern<T[K]>}} patterns
* @return {Pattern<T>}
*/
function Sequence(...patterns) {
return function (source, index) {
const values = /** @type {T} */ (/** @type {unknown} */ ([]));
for (const pattern of patterns) {
const match = pattern(source, index);
if (match == null) {
return null;
}
values.push(match[0]);
index = match[1];
}
return [values, index];
};
}
const InterCommandWhitespace = RegexPattern(/[^\S\n;]*/y);
const CommentPattern = RegexPattern(/#.*\n/y);
const PreWordWhitespace = RegexPattern(/[^\S\n;]*/y);
const BasicWord = MapPattern(RegexPattern(/[^\s;]+/y), ([word]) => ({
text: word,
}));
const WordPattern = MapPattern(
Sequence(PreWordWhitespace, BasicWord),
([_, word]) => word
);
2023-07-27 01:54:06 -04:00
/**
* Parse out a Notcl script into an easier-to-interpret representation.
* No script is actually executed yet.
*
* @param {string} code
* @returns Script
*/
function parseNotcl(code) {
/* Preprocess */
// fold line endings
code = code.replace(/(?<!\\)((\\\\)*)\\\n/g, "$1");
2023-07-27 01:54:06 -04:00
/* Parse */
function nextWord(/* TODO: null/]/" terminator */) {
2023-07-27 01:54:06 -04:00
// TODO: handle all kinds of brace/substitution stuff
const [word, nextIndex] = WordPattern(code, 0) ?? [null, 0];
2023-07-27 01:54:06 -04:00
if (word) {
code = code.substring(nextIndex);
return word;
2023-07-27 01:54:06 -04:00
} else {
return null;
}
}
function nextCommand(/* TODO: null/]/" terminator */) {
2023-07-27 01:54:06 -04:00
const command = /** @type {Word[]} */ ([]);
while (true) {
// Strip whitespace
code = code.replace(/^\s*/, "");
// Strip comments
if (code[0] == "#") {
code = code.replace(/^.*\n/, "");
continue;
}
// Strip semicolons
if (code[0] == ";") {
code = code.substring(1);
continue;
}
break;
}
while (true) {
const word = nextWord();
if (word) {
command.push(word);
continue;
}
break;
}
return command;
}
/* Loop through commands, with safety check */
const script = /** @type {Command[]} */ ([]);
for (let i = 0; i < 1000 && code != ""; i++) {
script.push(nextCommand());
}
return script;
}