@xnoxs/flux-lang 4.0.0 → 4.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/src/self/lexer.js CHANGED
@@ -1 +1,713 @@
1
- function map(arr, fn) { return arr.map(fn); } function some(arr, fn) { return arr.some(fn); } function join(arr, sep) { return arr.join(sep != null ? sep : ','); } function trim(s) { return String(s).trim(); } function trimEnd(s) { return String(s).trimEnd(); } "use strict"; const T = { NUMBER: "NUMBER", STRING: "STRING", BOOL: "BOOL", NULL: "NULL", IDENT: "IDENT", VAR: "VAR", VAL: "VAL", FN: "FN", RETURN: "RETURN", IF: "IF", ELSE: "ELSE", FOR: "FOR", IN: "IN", WHILE: "WHILE", BREAK: "BREAK", CONTINUE: "CONTINUE", DO: "DO", CLASS: "CLASS", EXTENDS: "EXTENDS", SELF: "SELF", NEW: "NEW", INTERFACE: "INTERFACE", IMPLEMENTS: "IMPLEMENTS", PRIVATE: "PRIVATE", PUBLIC: "PUBLIC", PROTECTED: "PROTECTED", READONLY: "READONLY", STATIC: "STATIC", ABSTRACT: "ABSTRACT", OVERRIDE: "OVERRIDE", MATCH: "MATCH", WHEN: "WHEN", IMPORT: "IMPORT", EXPORT: "EXPORT", FROM: "FROM", AS: "AS", DEFAULT: "DEFAULT", AND: "AND", OR: "OR", NOT: "NOT", ASYNC: "ASYNC", AWAIT: "AWAIT", TRY: "TRY", CATCH: "CATCH", FINALLY: "FINALLY", THROW: "THROW", TYPEOF: "TYPEOF", INSTANCEOF: "INSTANCEOF", TYPE: "TYPE", ENUM: "ENUM", SATISFIES: "SATISFIES", IS: "IS", CONST: "CONST", PLUS: "PLUS", MINUS: "MINUS", STAR: "STAR", SLASH: "SLASH", PERCENT: "PERCENT", REGEX: "REGEX", STARSTAR: "STARSTAR", EQ: "EQ", EQEQ: "EQEQ", NEQ: "NEQ", EQEQEQ: "EQEQEQ", NEQEQ: "NEQEQ", LT: "LT", LTE: "LTE", GT: "GT", GTE: "GTE", PLUSEQ: "PLUSEQ", MINUSEQ: "MINUSEQ", STAREQ: "STAREQ", SLASHEQ: "SLASHEQ", PERCENTEQ: "PERCENTEQ", PLUSPLUS: "PLUSPLUS", MINUSMINUS: "MINUSMINUS", AMPERSAND: "AMPERSAND", ANDAND: "ANDAND", PIPEB: "PIPEB", OROR: "OROR", CARET: "CARET", TILDE: "TILDE", LSHIFT: "LSHIFT", RSHIFT: "RSHIFT", ARROW: "ARROW", FATARROW: "FATARROW", PIPE: "PIPE", DOTDOT: "DOTDOT", DOTDOTDOT: "DOTDOTDOT", WILDCARD: "WILDCARD", NULLISH: "NULLISH", QUESTIONDOT: "QUESTIONDOT", BANG: "BANG", AT: "AT", LPAREN: "LPAREN", RPAREN: "RPAREN", LBRACKET: "LBRACKET", RBRACKET: "RBRACKET", LBRACE: "LBRACE", RBRACE: "RBRACE", COMMA: "COMMA", DOT: "DOT", COLON: "COLON", QUESTION: "QUESTION", NEWLINE: "NEWLINE", INDENT: "INDENT", DEDENT: "DEDENT", EOF: "EOF" }; module.exports.T = T; const TokenType = T; module.exports.TokenType = TokenType; const _a = { var: "VAR", val: "VAL", fn: "FN", return: "RETURN", declare: "DECLARE", if: "IF", else: "ELSE", for: "FOR", in: "IN", while: "WHILE", break: "BREAK", continue: "CONTINUE", do: "DO", class: "CLASS", extends: "EXTENDS", self: "SELF", new: "NEW", interface: "INTERFACE", implements: "IMPLEMENTS", private: "PRIVATE", public: "PUBLIC", protected: "PROTECTED", readonly: "READONLY", static: "STATIC", abstract: "ABSTRACT", override: "OVERRIDE", match: "MATCH", when: "WHEN", import: "IMPORT", export: "EXPORT", from: "FROM", as: "AS", default: "DEFAULT", and: "AND", or: "OR", not: "NOT", async: "ASYNC", await: "AWAIT", try: "TRY", catch: "CATCH", finally: "FINALLY", throw: "THROW", typeof: "TYPEOF", instanceof: "INSTANCEOF", type: "TYPE", enum: "ENUM", satisfies: "SATISFIES", is: "IS", const: "CONST", true: "__TRUE__", false: "__FALSE__", null: "__NULL__" }; class Lexer { constructor(src, pos, line, col, tokens, indentStack, nestDepth) { this.src = src; this.pos = pos; this.line = line; this.col = col; this.tokens = tokens; this.indentStack = indentStack; this.nestDepth = nestDepth; } err(_b) { throw new Error(`[Lexer ${this.line}:${this.col}] ${_b}`); } ch(_c = 0) { return (this.src[(this.pos + _c)] ?? ""); } adv() { const c = this.src[this.pos]; this.pos = (this.pos + 1); if ((c == "\n")) { this.line = (this.line + 1); this.col = 1; } else { this.col = (this.col + 1); } return c; } tok(_d, _e, _f, _g) { const v = ((_e != undefined) ? _e : _d); const ln = ((_f != undefined) ? _f : this.line); const co = ((_g != undefined) ? _g : this.col); this.tokens.push({ type: _d, value: v, line: ln, col: co }); } applyIndent(_h) { if ((this.nestDepth > 0)) { return; } const top = this.indentStack[(this.indentStack.length - 1)]; if ((_h > top)) { this.indentStack.push(_h); this.tok(T.INDENT, _h, this.line, 1); } else if ((_h < top)) { while (((this.indentStack.length > 1) && (this.indentStack[(this.indentStack.length - 1)] > _h))) { this.indentStack.pop(); this.tok(T.DEDENT, null, this.line, 1); } if ((this.indentStack[(this.indentStack.length - 1)] != _h)) { this.err(`Inconsistent indentation (${_h} spaces)`); } } } scanBacktick(_f, _g) { this.adv(); let s = ""; while (((this.pos < this.src.length) && (this.ch() != "`"))) { if ((this.ch() == "\\")) { this.adv(); const e = this.adv(); if ((e == "n")) { s += "\n"; } else if ((e == "t")) { s += "\t"; } else if ((e == "`")) { s += "`"; } else if ((e == "\\")) { s += "\\"; } else { s += ("\\" + e); } } else { s += this.adv(); } } if ((this.ch() != "`")) { this.err("Unterminated backtick string"); } this.adv(); const rawLines = s.split("\n"); const trimmed = rawLines.map((_i) => _i.trimEnd()); while ((trimmed.length && !trimmed[0].trim())) { trimmed.shift(); } while ((trimmed.length && !trimmed[(trimmed.length - 1)].trim())) { trimmed.pop(); } let minIndent = 999999; for (const _i of trimmed) { if (_i.trim()) { const m = _i.match(/^(\s*)/); const ind = m[1].length; if ((ind < minIndent)) { minIndent = ind; } } } if ((minIndent == 999999)) { minIndent = 0; } const dedented = trimmed.map((_i) => _i.slice(minIndent)); this.tok(T.STRING, dedented.join("\n"), _f, _g); } scanStr(_f, _g) { this.adv(); const parts = []; let text = ""; while (((this.pos < this.src.length) && (this.ch() != "\""))) { if ((this.ch() == "\\")) { this.adv(); const e = this.adv(); if ((e == "n")) { text += "\n"; } else if ((e == "t")) { text += "\t"; } else if ((e == "\"")) { text += "\""; } else if ((e == "'")) { text += "'"; } else if ((e == "\\")) { text += "\\"; } else if ((e == "{")) { text += "{"; } else if ((e == "}")) { text += "}"; } else { text += ("\\" + e); } } else if ((this.ch() == "{")) { parts.push({ type: "text", value: text }); text = ""; this.adv(); let depth = 1; let expr = ""; while ((this.pos < this.src.length)) { const cc = this.ch(); if ((cc == "{")) { depth = (depth + 1); } if ((cc == "}")) { depth = (depth - 1); if ((depth == 0)) { break; } } if (((cc == "\\") && ((this.src[(this.pos + 1)] == "\"") || (this.src[(this.pos + 1)] == "'")))) { this.adv(); expr += this.adv(); } else { expr += this.adv(); } } if ((this.ch() != "}")) { this.err("Unclosed string interpolation"); } this.adv(); parts.push({ type: "expr", value: expr.trim() }); } else { text += this.adv(); } } if ((this.ch() != "\"")) { this.err("Unterminated string"); } this.adv(); parts.push({ type: "text", value: text }); if (parts.some((_j) => (_j.type == "expr"))) { this.tok(T.STRING, { template: true, parts }, _f, _g); } else { this.tok(T.STRING, parts.map((_j) => _j.value).join(""), _f, _g); } } scanRegexBody(_f, _g) { let pattern = ""; let inClass = false; while ((this.pos < this.src.length)) { const ch = this.ch(); if ((ch == "\n")) { this.err("Unterminated regex literal"); } if ((ch == "\\")) { pattern += this.adv(); if ((this.pos < this.src.length)) { pattern += this.adv(); } continue; } if ((ch == "[")) { inClass = true; pattern += this.adv(); continue; } if ((ch == "]")) { inClass = false; pattern += this.adv(); continue; } if (((ch == "/") && !inClass)) { break; } pattern += this.adv(); } if ((this.ch() != "/")) { this.err("Unterminated regex literal"); } this.adv(); let flags = ""; while (/[gimsuy]/.test(this.ch())) { flags += this.adv(); } this.tok(T.REGEX, { pattern, flags }, _f, _g); } scanBlockComment() { this.adv(); this.adv(); while ((this.pos < this.src.length)) { if (((this.ch() == "*") && (this.ch(1) == "/"))) { this.adv(); this.adv(); return; } this.adv(); } this.err("Unterminated block comment"); } tokenize() { let bol = true; while ((this.pos < this.src.length)) { if (bol) { bol = false; let _h = 0; while (((this.ch() == " ") || (this.ch() == "\t"))) { if ((this.ch() == "\t")) { _h = (_h + 4); } else { _h = (_h + 1); } this.adv(); } if (((this.ch() == "\n") || !this.ch())) { if ((this.ch() == "\n")) { this.adv(); bol = true; } continue; } if (((this.ch() == "/") && (this.ch(1) == "/"))) { while (((this.pos < this.src.length) && (this.ch() != "\n"))) { this.adv(); } if ((this.ch() == "\n")) { this.adv(); bol = true; } continue; } if (((this.ch() == "/") && (this.ch(1) == "*"))) { this.scanBlockComment(); continue; } const isContinuation = ((((this.ch() == "|") && (this.ch(1) == ">")) || (this.ch() == ".")) || ((this.ch() == "?") && (this.ch(1) == "."))); if (isContinuation) { const lastTok = this.tokens[(this.tokens.length - 1)]; if ((lastTok && (lastTok.type == T.NEWLINE))) { this.tokens.pop(); } } else { this.applyIndent(_h); } } const _f = this.line; const _g = this.col; const cur = this.ch(); if ((cur == "\n")) { this.adv(); bol = true; if ((this.nestDepth == 0)) { const lastTok = this.tokens[(this.tokens.length - 1)]; if ((((lastTok && (lastTok.type != T.NEWLINE)) && (lastTok.type != T.INDENT)) && (lastTok.type != T.DEDENT))) { this.tok(T.NEWLINE, null, _f, _g); } } continue; } if (((cur == " ") || (cur == "\t"))) { this.adv(); continue; } if (((cur == "/") && (this.ch(1) == "/"))) { while (((this.pos < this.src.length) && (this.ch() != "\n"))) { this.adv(); } continue; } if (((cur == "/") && (this.ch(1) == "*"))) { this.scanBlockComment(); continue; } if (((cur >= "0") && (cur <= "9"))) { if (((cur == "0") && ((this.ch(1) == "x") || (this.ch(1) == "X")))) { this.adv(); this.adv(); let h = ""; while (/[0-9a-fA-F_]/.test(this.ch())) { const hc = this.adv(); if ((hc != "_")) { h += hc; } } this.tok(T.NUMBER, parseInt(((h.length > 0) ? h : "0"), 16), _f, _g); continue; } if (((cur == "0") && ((this.ch(1) == "b") || (this.ch(1) == "B")))) { this.adv(); this.adv(); let b = ""; while (/[01_]/.test(this.ch())) { const bc = this.adv(); if ((bc != "_")) { b += bc; } } this.tok(T.NUMBER, parseInt(((b.length > 0) ? b : "0"), 2), _f, _g); continue; } let numStr = ""; while ((this.pos < this.src.length)) { if (((this.ch() == ".") && (this.ch(1) == "."))) { break; } if ((((this.ch() >= "0") && (this.ch() <= "9")) || (this.ch() == "."))) { numStr += this.adv(); } else if ((this.ch() == "_")) { this.adv(); } else if ((((this.ch() == "e") || (this.ch() == "E")) && (numStr.length > 0))) { numStr += this.adv(); if (((this.ch() == "+") || (this.ch() == "-"))) { numStr += this.adv(); } while (((this.ch() >= "0") && (this.ch() <= "9"))) { numStr += this.adv(); } break; } else { break; } } this.tok(T.NUMBER, parseFloat(numStr), _f, _g); continue; } if ((cur == "\"")) { this.scanStr(_f, _g); continue; } if ((cur == "`")) { this.scanBacktick(_f, _g); continue; } if ((cur == "'")) { this.adv(); let sq = ""; while (((this.pos < this.src.length) && (this.ch() != "'"))) { if ((this.ch() == "\\")) { this.adv(); const e = this.adv(); if ((e == "n")) { sq += "\n"; } else if ((e == "t")) { sq += "\t"; } else if ((e == "r")) { sq += "\r"; } else if ((e == "'")) { sq += "'"; } else if ((e == "\\")) { sq += "\\"; } else { sq += ("\\" + e); } } else { sq += this.adv(); } } if ((this.ch() != "'")) { this.err("Unterminated string"); } this.adv(); this.tok(T.STRING, sq, _f, _g); continue; } if (((((cur >= "a") && (cur <= "z")) || ((cur >= "A") && (cur <= "Z"))) || (cur == "_"))) { let word = ""; while (/[a-zA-Z0-9_]/.test(this.ch())) { word += this.adv(); } if (((word == "_") && !/[a-zA-Z0-9_]/.test(this.ch()))) { this.tok(T.WILDCARD, "_", _f, _g); continue; } const kw = _a[word]; if ((kw == "__TRUE__")) { this.tok(T.BOOL, true, _f, _g); } else if ((kw == "__FALSE__")) { this.tok(T.BOOL, false, _f, _g); } else if ((kw == "__NULL__")) { this.tok(T.NULL, null, _f, _g); } else if ((kw != undefined)) { this.tok(kw, word, _f, _g); } else { this.tok(T.IDENT, word, _f, _g); } continue; } this.adv(); if ((cur == "+")) { if ((this.ch() == "+")) { this.adv(); this.tok(T.PLUSPLUS, "++", _f, _g); } else if ((this.ch() == "=")) { this.adv(); this.tok(T.PLUSEQ, "+=", _f, _g); } else { this.tok(T.PLUS, "+", _f, _g); } } else if ((cur == "-")) { if ((this.ch() == "-")) { this.adv(); this.tok(T.MINUSMINUS, "--", _f, _g); } else if ((this.ch() == ">")) { this.adv(); this.tok(T.ARROW, "->", _f, _g); } else if ((this.ch() == "=")) { this.adv(); this.tok(T.MINUSEQ, "-=", _f, _g); } else { this.tok(T.MINUS, "-", _f, _g); } } else if ((cur == "*")) { if ((this.ch() == "*")) { this.adv(); this.tok(T.STARSTAR, "**", _f, _g); } else if ((this.ch() == "=")) { this.adv(); this.tok(T.STAREQ, "*=", _f, _g); } else { this.tok(T.STAR, "*", _f, _g); } } else if ((cur == "/")) { if ((this.ch() == "=")) { this.adv(); this.tok(T.SLASHEQ, "/=", _f, _g); } else { const lastTok = this.tokens[(this.tokens.length - 1)]; const afterVal = ((lastTok != null) && (((((((((((lastTok.type == T.IDENT) || (lastTok.type == T.NUMBER)) || (lastTok.type == T.STRING)) || (lastTok.type == T.BOOL)) || (lastTok.type == T.NULL)) || (lastTok.type == T.REGEX)) || (lastTok.type == T.RPAREN)) || (lastTok.type == T.RBRACKET)) || (lastTok.type == T.PLUSPLUS)) || (lastTok.type == T.MINUSMINUS)) || (lastTok.type == T.BANG))); if (afterVal) { this.tok(T.SLASH, "/", _f, _g); } else { this.scanRegexBody(_f, _g); } } } else if ((cur == "%")) { if ((this.ch() == "=")) { this.adv(); this.tok(T.PERCENTEQ, "%=", _f, _g); } else { this.tok(T.PERCENT, "%", _f, _g); } } else if ((cur == "=")) { if (((this.ch() == "=") && (this.src[(this.pos + 1)] == "="))) { this.adv(); this.adv(); this.tok(T.EQEQEQ, "===", _f, _g); } else if ((this.ch() == "=")) { this.adv(); this.tok(T.EQEQ, "==", _f, _g); } else if ((this.ch() == ">")) { this.adv(); this.tok(T.FATARROW, "=>", _f, _g); } else { this.tok(T.EQ, "=", _f, _g); } } else if ((cur == "!")) { if (((this.ch() == "=") && (this.src[(this.pos + 1)] == "="))) { this.adv(); this.adv(); this.tok(T.NEQEQ, "!==", _f, _g); } else if ((this.ch() == "=")) { this.adv(); this.tok(T.NEQ, "!=", _f, _g); } else { this.tok(T.BANG, "!", _f, _g); } } else if ((cur == "<")) { if ((this.ch() == "<")) { this.adv(); this.tok(T.LSHIFT, "<<", _f, _g); } else if ((this.ch() == "=")) { this.adv(); this.tok(T.LTE, "<=", _f, _g); } else { this.tok(T.LT, "<", _f, _g); } } else if ((cur == ">")) { if ((this.ch() == ">")) { this.adv(); this.tok(T.RSHIFT, ">>", _f, _g); } else if ((this.ch() == "=")) { this.adv(); this.tok(T.GTE, ">=", _f, _g); } else { this.tok(T.GT, ">", _f, _g); } } else if ((cur == ".")) { if (((this.ch() == ".") && (this.src[(this.pos + 1)] == "."))) { this.adv(); this.adv(); this.tok(T.DOTDOTDOT, "...", _f, _g); } else if ((this.ch() == ".")) { this.adv(); this.tok(T.DOTDOT, "..", _f, _g); } else { this.tok(T.DOT, ".", _f, _g); } } else if ((cur == "?")) { if ((this.ch() == ".")) { this.adv(); this.tok(T.QUESTIONDOT, "?.", _f, _g); } else if ((this.ch() == "?")) { this.adv(); this.tok(T.NULLISH, "??", _f, _g); } else { this.tok(T.QUESTION, "?", _f, _g); } } else if ((cur == "|")) { if ((this.ch() == ">")) { this.adv(); this.tok(T.PIPE, "|>", _f, _g); } else if ((this.ch() == "|")) { this.adv(); this.tok(T.OROR, "||", _f, _g); } else { this.tok(T.PIPEB, "|", _f, _g); } } else if ((cur == "&")) { if ((this.ch() == "&")) { this.adv(); this.tok(T.ANDAND, "&&", _f, _g); } else { this.tok(T.AMPERSAND, "&", _f, _g); } } else if ((cur == "^")) { this.tok(T.CARET, "^", _f, _g); } else if ((cur == "~")) { this.tok(T.TILDE, "~", _f, _g); } else if ((cur == "@")) { this.tok(T.AT, "@", _f, _g); } else if ((cur == "(")) { this.nestDepth = (this.nestDepth + 1); this.tok(T.LPAREN, "(", _f, _g); } else if ((cur == ")")) { this.nestDepth = (this.nestDepth - 1); this.tok(T.RPAREN, ")", _f, _g); } else if ((cur == "[")) { this.nestDepth = (this.nestDepth + 1); this.tok(T.LBRACKET, "[", _f, _g); } else if ((cur == "]")) { this.nestDepth = (this.nestDepth - 1); this.tok(T.RBRACKET, "]", _f, _g); } else if ((cur == "{")) { this.nestDepth = (this.nestDepth + 1); this.tok(T.LBRACE, "{", _f, _g); } else if ((cur == "}")) { this.nestDepth = (this.nestDepth - 1); this.tok(T.RBRACE, "}", _f, _g); } else if ((cur == ",")) { this.tok(T.COMMA, ",", _f, _g); } else if ((cur == ":")) { this.tok(T.COLON, ":", _f, _g); } else if ((cur != ";")) { this.err(`Unknown character: '${cur}'`); } } while ((this.indentStack.length > 1)) { this.indentStack.pop(); this.tok(T.DEDENT, null, this.line, 1); } const lastTok = this.tokens[(this.tokens.length - 1)]; if (((lastTok && (lastTok.type != T.NEWLINE)) && (lastTok.type != T.DEDENT))) { this.tok(T.NEWLINE, null, this.line, this.col); } this.tok(T.EOF, null, this.line, this.col); return this.tokens; } } module.exports.Lexer = Lexer; function lexerize(_k) { const processed = _k.replace(/\r\n/g, "\n").replace(/\r/g, "\n"); return new Lexer(processed, 0, 1, 1, [], [0], 0); } module.exports.lexerize = lexerize;
1
+ // ── Flux stdlib ──
2
+
3
+ function map(arr, fn) { return arr.map(fn); }
4
+
5
+ function some(arr, fn) { return arr.some(fn); }
6
+
7
+ function join(arr, sep) { return arr.join(sep != null ? sep : ','); }
8
+
9
+ function trim(s) { return String(s).trim(); }
10
+
11
+ function trimEnd(s) { return String(s).trimEnd(); }
12
+ // ── end stdlib ──
13
+
14
+ // Generated by Flux Transpiler v3.2.0
15
+ "use strict";
16
+
17
+ const T = { NUMBER: "NUMBER", STRING: "STRING", BOOL: "BOOL", NULL: "NULL", IDENT: "IDENT", VAR: "VAR", VAL: "VAL", FN: "FN", RETURN: "RETURN", IF: "IF", ELSE: "ELSE", FOR: "FOR", IN: "IN", WHILE: "WHILE", BREAK: "BREAK", CONTINUE: "CONTINUE", DO: "DO", CLASS: "CLASS", EXTENDS: "EXTENDS", SELF: "SELF", NEW: "NEW", INTERFACE: "INTERFACE", IMPLEMENTS: "IMPLEMENTS", PRIVATE: "PRIVATE", PUBLIC: "PUBLIC", PROTECTED: "PROTECTED", READONLY: "READONLY", STATIC: "STATIC", ABSTRACT: "ABSTRACT", OVERRIDE: "OVERRIDE", MATCH: "MATCH", WHEN: "WHEN", IMPORT: "IMPORT", EXPORT: "EXPORT", FROM: "FROM", AS: "AS", DEFAULT: "DEFAULT", AND: "AND", OR: "OR", NOT: "NOT", ASYNC: "ASYNC", AWAIT: "AWAIT", TRY: "TRY", CATCH: "CATCH", FINALLY: "FINALLY", THROW: "THROW", TYPEOF: "TYPEOF", INSTANCEOF: "INSTANCEOF", TYPE: "TYPE", ENUM: "ENUM", SATISFIES: "SATISFIES", IS: "IS", CONST: "CONST", PLUS: "PLUS", MINUS: "MINUS", STAR: "STAR", SLASH: "SLASH", PERCENT: "PERCENT", REGEX: "REGEX", STARSTAR: "STARSTAR", EQ: "EQ", EQEQ: "EQEQ", NEQ: "NEQ", EQEQEQ: "EQEQEQ", NEQEQ: "NEQEQ", LT: "LT", LTE: "LTE", GT: "GT", GTE: "GTE", PLUSEQ: "PLUSEQ", MINUSEQ: "MINUSEQ", STAREQ: "STAREQ", SLASHEQ: "SLASHEQ", PERCENTEQ: "PERCENTEQ", PLUSPLUS: "PLUSPLUS", MINUSMINUS: "MINUSMINUS", AMPERSAND: "AMPERSAND", ANDAND: "ANDAND", PIPEB: "PIPEB", OROR: "OROR", CARET: "CARET", TILDE: "TILDE", LSHIFT: "LSHIFT", RSHIFT: "RSHIFT", ARROW: "ARROW", FATARROW: "FATARROW", PIPE: "PIPE", DOTDOT: "DOTDOT", DOTDOTDOT: "DOTDOTDOT", WILDCARD: "WILDCARD", NULLISH: "NULLISH", QUESTIONDOT: "QUESTIONDOT", BANG: "BANG", AT: "AT", LPAREN: "LPAREN", RPAREN: "RPAREN", LBRACKET: "LBRACKET", RBRACKET: "RBRACKET", LBRACE: "LBRACE", RBRACE: "RBRACE", COMMA: "COMMA", DOT: "DOT", COLON: "COLON", QUESTION: "QUESTION", NEWLINE: "NEWLINE", INDENT: "INDENT", DEDENT: "DEDENT", EOF: "EOF" };
18
+ module.exports.T = T;
19
+ const TokenType = T;
20
+ module.exports.TokenType = TokenType;
21
+ const KEYWORDS = { var: "VAR", val: "VAL", fn: "FN", return: "RETURN", declare: "DECLARE", if: "IF", else: "ELSE", for: "FOR", in: "IN", while: "WHILE", break: "BREAK", continue: "CONTINUE", do: "DO", class: "CLASS", extends: "EXTENDS", self: "SELF", new: "NEW", interface: "INTERFACE", implements: "IMPLEMENTS", private: "PRIVATE", public: "PUBLIC", protected: "PROTECTED", readonly: "READONLY", static: "STATIC", abstract: "ABSTRACT", override: "OVERRIDE", match: "MATCH", when: "WHEN", import: "IMPORT", export: "EXPORT", from: "FROM", as: "AS", default: "DEFAULT", and: "AND", or: "OR", not: "NOT", async: "ASYNC", await: "AWAIT", try: "TRY", catch: "CATCH", finally: "FINALLY", throw: "THROW", typeof: "TYPEOF", instanceof: "INSTANCEOF", type: "TYPE", enum: "ENUM", satisfies: "SATISFIES", is: "IS", const: "CONST", true: "__TRUE__", false: "__FALSE__", null: "__NULL__" };
22
+ class Lexer {
23
+ constructor(src, pos, line, col, tokens, indentStack, nestDepth) {
24
+ this.src = src;
25
+ this.pos = pos;
26
+ this.line = line;
27
+ this.col = col;
28
+ this.tokens = tokens;
29
+ this.indentStack = indentStack;
30
+ this.nestDepth = nestDepth;
31
+ }
32
+
33
+ err(msg) {
34
+ throw new Error(`[Lexer ${this.line}:${this.col}] ${msg}`);
35
+ }
36
+
37
+ ch(n = 0) {
38
+ return (this.src[(this.pos + n)] ?? "");
39
+ }
40
+
41
+ adv() {
42
+ const c = this.src[this.pos];
43
+ this.pos = (this.pos + 1);
44
+ if ((c == "\n")) {
45
+ this.line = (this.line + 1);
46
+ this.col = 1;
47
+ }
48
+ else {
49
+ this.col = (this.col + 1);
50
+ }
51
+ return c;
52
+ }
53
+
54
+ tok(typ, value, l, c) {
55
+ const v = ((value != undefined) ? value : typ);
56
+ const ln = ((l != undefined) ? l : this.line);
57
+ const co = ((c != undefined) ? c : this.col);
58
+ this.tokens.push({ type: typ, value: v, line: ln, col: co });
59
+ }
60
+
61
+ applyIndent(indent) {
62
+ if ((this.nestDepth > 0)) {
63
+ return;
64
+ }
65
+ const top = this.indentStack[(this.indentStack.length - 1)];
66
+ if ((indent > top)) {
67
+ this.indentStack.push(indent);
68
+ this.tok(T.INDENT, indent, this.line, 1);
69
+ }
70
+ else if ((indent < top)) {
71
+ while (((this.indentStack.length > 1) && (this.indentStack[(this.indentStack.length - 1)] > indent))) {
72
+ this.indentStack.pop();
73
+ this.tok(T.DEDENT, null, this.line, 1);
74
+ }
75
+ if ((this.indentStack[(this.indentStack.length - 1)] != indent)) {
76
+ this.err(`Inconsistent indentation (${indent} spaces)`);
77
+ }
78
+ }
79
+ }
80
+
81
+ scanBacktick(l, c) {
82
+ this.adv();
83
+ let s = "";
84
+ while (((this.pos < this.src.length) && (this.ch() != "`"))) {
85
+ if ((this.ch() == "\\")) {
86
+ this.adv();
87
+ const e = this.adv();
88
+ if ((e == "n")) {
89
+ s += "\n";
90
+ }
91
+ else if ((e == "t")) {
92
+ s += "\t";
93
+ }
94
+ else if ((e == "`")) {
95
+ s += "`";
96
+ }
97
+ else if ((e == "\\")) {
98
+ s += "\\";
99
+ }
100
+ else {
101
+ s += ("\\" + e);
102
+ }
103
+ }
104
+ else {
105
+ s += this.adv();
106
+ }
107
+ }
108
+ if ((this.ch() != "`")) {
109
+ this.err("Unterminated backtick string");
110
+ }
111
+ this.adv();
112
+ const rawLines = s.split("\n");
113
+ const trimmed = rawLines.map((ln) => ln.trimEnd());
114
+ while ((trimmed.length && !trimmed[0].trim())) {
115
+ trimmed.shift();
116
+ }
117
+ while ((trimmed.length && !trimmed[(trimmed.length - 1)].trim())) {
118
+ trimmed.pop();
119
+ }
120
+ let minIndent = 999999;
121
+ for (const ln of trimmed) {
122
+ if (ln.trim()) {
123
+ const m = ln.match(/^(\s*)/);
124
+ const ind = m[1].length;
125
+ if ((ind < minIndent)) {
126
+ minIndent = ind;
127
+ }
128
+ }
129
+ }
130
+ if ((minIndent == 999999)) {
131
+ minIndent = 0;
132
+ }
133
+ const dedented = trimmed.map((ln) => ln.slice(minIndent));
134
+ this.tok(T.STRING, dedented.join("\n"), l, c);
135
+ }
136
+
137
+ scanStr(l, c) {
138
+ this.adv();
139
+ const parts = [];
140
+ let text = "";
141
+ while (((this.pos < this.src.length) && (this.ch() != "\""))) {
142
+ if ((this.ch() == "\\")) {
143
+ this.adv();
144
+ const e = this.adv();
145
+ if ((e == "n")) {
146
+ text += "\n";
147
+ }
148
+ else if ((e == "t")) {
149
+ text += "\t";
150
+ }
151
+ else if ((e == "\"")) {
152
+ text += "\"";
153
+ }
154
+ else if ((e == "'")) {
155
+ text += "'";
156
+ }
157
+ else if ((e == "\\")) {
158
+ text += "\\";
159
+ }
160
+ else if ((e == "{")) {
161
+ text += "{";
162
+ }
163
+ else if ((e == "}")) {
164
+ text += "}";
165
+ }
166
+ else {
167
+ text += ("\\" + e);
168
+ }
169
+ }
170
+ else if ((this.ch() == "{")) {
171
+ parts.push({ type: "text", value: text });
172
+ text = "";
173
+ this.adv();
174
+ let depth = 1;
175
+ let expr = "";
176
+ while ((this.pos < this.src.length)) {
177
+ const cc = this.ch();
178
+ if ((cc == "{")) {
179
+ depth = (depth + 1);
180
+ }
181
+ if ((cc == "}")) {
182
+ depth = (depth - 1);
183
+ if ((depth == 0)) {
184
+ break;
185
+ }
186
+ }
187
+ if (((cc == "\\") && ((this.src[(this.pos + 1)] == "\"") || (this.src[(this.pos + 1)] == "'")))) {
188
+ this.adv();
189
+ expr += this.adv();
190
+ }
191
+ else {
192
+ expr += this.adv();
193
+ }
194
+ }
195
+ if ((this.ch() != "}")) {
196
+ this.err("Unclosed string interpolation");
197
+ }
198
+ this.adv();
199
+ parts.push({ type: "expr", value: expr.trim() });
200
+ }
201
+ else {
202
+ text += this.adv();
203
+ }
204
+ }
205
+ if ((this.ch() != "\"")) {
206
+ this.err("Unterminated string");
207
+ }
208
+ this.adv();
209
+ parts.push({ type: "text", value: text });
210
+ if (parts.some((p) => (p.type == "expr"))) {
211
+ this.tok(T.STRING, { template: true, parts }, l, c);
212
+ }
213
+ else {
214
+ this.tok(T.STRING, parts.map((p) => p.value).join(""), l, c);
215
+ }
216
+ }
217
+
218
+ scanRegexBody(l, c) {
219
+ let pattern = "";
220
+ let inClass = false;
221
+ while ((this.pos < this.src.length)) {
222
+ const ch = this.ch();
223
+ if ((ch == "\n")) {
224
+ this.err("Unterminated regex literal");
225
+ }
226
+ if ((ch == "\\")) {
227
+ pattern += this.adv();
228
+ if ((this.pos < this.src.length)) {
229
+ pattern += this.adv();
230
+ }
231
+ continue;
232
+ }
233
+ if ((ch == "[")) {
234
+ inClass = true;
235
+ pattern += this.adv();
236
+ continue;
237
+ }
238
+ if ((ch == "]")) {
239
+ inClass = false;
240
+ pattern += this.adv();
241
+ continue;
242
+ }
243
+ if (((ch == "/") && !inClass)) {
244
+ break;
245
+ }
246
+ pattern += this.adv();
247
+ }
248
+ if ((this.ch() != "/")) {
249
+ this.err("Unterminated regex literal");
250
+ }
251
+ this.adv();
252
+ let flags = "";
253
+ while (/[gimsuy]/.test(this.ch())) {
254
+ flags += this.adv();
255
+ }
256
+ this.tok(T.REGEX, { pattern, flags }, l, c);
257
+ }
258
+
259
+ scanBlockComment() {
260
+ this.adv();
261
+ this.adv();
262
+ while ((this.pos < this.src.length)) {
263
+ if (((this.ch() == "*") && (this.ch(1) == "/"))) {
264
+ this.adv();
265
+ this.adv();
266
+ return;
267
+ }
268
+ this.adv();
269
+ }
270
+ this.err("Unterminated block comment");
271
+ }
272
+
273
+ tokenize() {
274
+ let bol = true;
275
+ while ((this.pos < this.src.length)) {
276
+ if (bol) {
277
+ bol = false;
278
+ let indent = 0;
279
+ while (((this.ch() == " ") || (this.ch() == "\t"))) {
280
+ if ((this.ch() == "\t")) {
281
+ indent = (indent + 4);
282
+ }
283
+ else {
284
+ indent = (indent + 1);
285
+ }
286
+ this.adv();
287
+ }
288
+ if (((this.ch() == "\n") || !this.ch())) {
289
+ if ((this.ch() == "\n")) {
290
+ this.adv();
291
+ bol = true;
292
+ }
293
+ continue;
294
+ }
295
+ if (((this.ch() == "/") && (this.ch(1) == "/"))) {
296
+ while (((this.pos < this.src.length) && (this.ch() != "\n"))) {
297
+ this.adv();
298
+ }
299
+ if ((this.ch() == "\n")) {
300
+ this.adv();
301
+ bol = true;
302
+ }
303
+ continue;
304
+ }
305
+ if (((this.ch() == "/") && (this.ch(1) == "*"))) {
306
+ this.scanBlockComment();
307
+ continue;
308
+ }
309
+ const isContinuation = ((((this.ch() == "|") && (this.ch(1) == ">")) || (this.ch() == ".")) || ((this.ch() == "?") && (this.ch(1) == ".")));
310
+ if (isContinuation) {
311
+ const lastTok = this.tokens[(this.tokens.length - 1)];
312
+ if ((lastTok && (lastTok.type == T.NEWLINE))) {
313
+ this.tokens.pop();
314
+ }
315
+ }
316
+ else {
317
+ this.applyIndent(indent);
318
+ }
319
+ }
320
+ const l = this.line;
321
+ const c = this.col;
322
+ const cur = this.ch();
323
+ if ((cur == "\n")) {
324
+ this.adv();
325
+ bol = true;
326
+ if ((this.nestDepth == 0)) {
327
+ const lastTok = this.tokens[(this.tokens.length - 1)];
328
+ if ((((lastTok && (lastTok.type != T.NEWLINE)) && (lastTok.type != T.INDENT)) && (lastTok.type != T.DEDENT))) {
329
+ this.tok(T.NEWLINE, null, l, c);
330
+ }
331
+ }
332
+ continue;
333
+ }
334
+ if (((cur == " ") || (cur == "\t"))) {
335
+ this.adv();
336
+ continue;
337
+ }
338
+ if (((cur == "/") && (this.ch(1) == "/"))) {
339
+ while (((this.pos < this.src.length) && (this.ch() != "\n"))) {
340
+ this.adv();
341
+ }
342
+ continue;
343
+ }
344
+ if (((cur == "/") && (this.ch(1) == "*"))) {
345
+ this.scanBlockComment();
346
+ continue;
347
+ }
348
+ if (((cur >= "0") && (cur <= "9"))) {
349
+ if (((cur == "0") && ((this.ch(1) == "x") || (this.ch(1) == "X")))) {
350
+ this.adv();
351
+ this.adv();
352
+ let h = "";
353
+ while (/[0-9a-fA-F_]/.test(this.ch())) {
354
+ const hc = this.adv();
355
+ if ((hc != "_")) {
356
+ h += hc;
357
+ }
358
+ }
359
+ this.tok(T.NUMBER, parseInt(((h.length > 0) ? h : "0"), 16), l, c);
360
+ continue;
361
+ }
362
+ if (((cur == "0") && ((this.ch(1) == "b") || (this.ch(1) == "B")))) {
363
+ this.adv();
364
+ this.adv();
365
+ let b = "";
366
+ while (/[01_]/.test(this.ch())) {
367
+ const bc = this.adv();
368
+ if ((bc != "_")) {
369
+ b += bc;
370
+ }
371
+ }
372
+ this.tok(T.NUMBER, parseInt(((b.length > 0) ? b : "0"), 2), l, c);
373
+ continue;
374
+ }
375
+ let numStr = "";
376
+ while ((this.pos < this.src.length)) {
377
+ if (((this.ch() == ".") && (this.ch(1) == "."))) {
378
+ break;
379
+ }
380
+ if ((((this.ch() >= "0") && (this.ch() <= "9")) || (this.ch() == "."))) {
381
+ numStr += this.adv();
382
+ }
383
+ else if ((this.ch() == "_")) {
384
+ this.adv();
385
+ }
386
+ else if ((((this.ch() == "e") || (this.ch() == "E")) && (numStr.length > 0))) {
387
+ numStr += this.adv();
388
+ if (((this.ch() == "+") || (this.ch() == "-"))) {
389
+ numStr += this.adv();
390
+ }
391
+ while (((this.ch() >= "0") && (this.ch() <= "9"))) {
392
+ numStr += this.adv();
393
+ }
394
+ break;
395
+ }
396
+ else {
397
+ break;
398
+ }
399
+ }
400
+ this.tok(T.NUMBER, parseFloat(numStr), l, c);
401
+ continue;
402
+ }
403
+ if ((cur == "\"")) {
404
+ this.scanStr(l, c);
405
+ continue;
406
+ }
407
+ if ((cur == "`")) {
408
+ this.scanBacktick(l, c);
409
+ continue;
410
+ }
411
+ if ((cur == "'")) {
412
+ this.adv();
413
+ let sq = "";
414
+ while (((this.pos < this.src.length) && (this.ch() != "'"))) {
415
+ if ((this.ch() == "\\")) {
416
+ this.adv();
417
+ const e = this.adv();
418
+ if ((e == "n")) {
419
+ sq += "\n";
420
+ }
421
+ else if ((e == "t")) {
422
+ sq += "\t";
423
+ }
424
+ else if ((e == "r")) {
425
+ sq += "\r";
426
+ }
427
+ else if ((e == "'")) {
428
+ sq += "'";
429
+ }
430
+ else if ((e == "\\")) {
431
+ sq += "\\";
432
+ }
433
+ else {
434
+ sq += ("\\" + e);
435
+ }
436
+ }
437
+ else {
438
+ sq += this.adv();
439
+ }
440
+ }
441
+ if ((this.ch() != "'")) {
442
+ this.err("Unterminated string");
443
+ }
444
+ this.adv();
445
+ this.tok(T.STRING, sq, l, c);
446
+ continue;
447
+ }
448
+ if (((((cur >= "a") && (cur <= "z")) || ((cur >= "A") && (cur <= "Z"))) || (cur == "_"))) {
449
+ let word = "";
450
+ while (/[a-zA-Z0-9_]/.test(this.ch())) {
451
+ word += this.adv();
452
+ }
453
+ if (((word == "_") && !/[a-zA-Z0-9_]/.test(this.ch()))) {
454
+ this.tok(T.WILDCARD, "_", l, c);
455
+ continue;
456
+ }
457
+ const kw = (Object.prototype.hasOwnProperty.call(KEYWORDS, word) ? KEYWORDS[word] : undefined);
458
+ if ((kw == "__TRUE__")) {
459
+ this.tok(T.BOOL, true, l, c);
460
+ }
461
+ else if ((kw == "__FALSE__")) {
462
+ this.tok(T.BOOL, false, l, c);
463
+ }
464
+ else if ((kw == "__NULL__")) {
465
+ this.tok(T.NULL, null, l, c);
466
+ }
467
+ else if ((kw != undefined)) {
468
+ this.tok(kw, word, l, c);
469
+ }
470
+ else {
471
+ this.tok(T.IDENT, word, l, c);
472
+ }
473
+ continue;
474
+ }
475
+ this.adv();
476
+ if ((cur == "+")) {
477
+ if ((this.ch() == "+")) {
478
+ this.adv();
479
+ this.tok(T.PLUSPLUS, "++", l, c);
480
+ }
481
+ else if ((this.ch() == "=")) {
482
+ this.adv();
483
+ this.tok(T.PLUSEQ, "+=", l, c);
484
+ }
485
+ else {
486
+ this.tok(T.PLUS, "+", l, c);
487
+ }
488
+ }
489
+ else if ((cur == "-")) {
490
+ if ((this.ch() == "-")) {
491
+ this.adv();
492
+ this.tok(T.MINUSMINUS, "--", l, c);
493
+ }
494
+ else if ((this.ch() == ">")) {
495
+ this.adv();
496
+ this.tok(T.ARROW, "->", l, c);
497
+ }
498
+ else if ((this.ch() == "=")) {
499
+ this.adv();
500
+ this.tok(T.MINUSEQ, "-=", l, c);
501
+ }
502
+ else {
503
+ this.tok(T.MINUS, "-", l, c);
504
+ }
505
+ }
506
+ else if ((cur == "*")) {
507
+ if ((this.ch() == "*")) {
508
+ this.adv();
509
+ this.tok(T.STARSTAR, "**", l, c);
510
+ }
511
+ else if ((this.ch() == "=")) {
512
+ this.adv();
513
+ this.tok(T.STAREQ, "*=", l, c);
514
+ }
515
+ else {
516
+ this.tok(T.STAR, "*", l, c);
517
+ }
518
+ }
519
+ else if ((cur == "/")) {
520
+ if ((this.ch() == "=")) {
521
+ this.adv();
522
+ this.tok(T.SLASHEQ, "/=", l, c);
523
+ }
524
+ else {
525
+ const lastTok = this.tokens[(this.tokens.length - 1)];
526
+ const afterVal = ((lastTok != null) && (((((((((((lastTok.type == T.IDENT) || (lastTok.type == T.NUMBER)) || (lastTok.type == T.STRING)) || (lastTok.type == T.BOOL)) || (lastTok.type == T.NULL)) || (lastTok.type == T.REGEX)) || (lastTok.type == T.RPAREN)) || (lastTok.type == T.RBRACKET)) || (lastTok.type == T.PLUSPLUS)) || (lastTok.type == T.MINUSMINUS)) || (lastTok.type == T.BANG)));
527
+ if (afterVal) {
528
+ this.tok(T.SLASH, "/", l, c);
529
+ }
530
+ else {
531
+ this.scanRegexBody(l, c);
532
+ }
533
+ }
534
+ }
535
+ else if ((cur == "%")) {
536
+ if ((this.ch() == "=")) {
537
+ this.adv();
538
+ this.tok(T.PERCENTEQ, "%=", l, c);
539
+ }
540
+ else {
541
+ this.tok(T.PERCENT, "%", l, c);
542
+ }
543
+ }
544
+ else if ((cur == "=")) {
545
+ if (((this.ch() == "=") && (this.src[(this.pos + 1)] == "="))) {
546
+ this.adv();
547
+ this.adv();
548
+ this.tok(T.EQEQEQ, "===", l, c);
549
+ }
550
+ else if ((this.ch() == "=")) {
551
+ this.adv();
552
+ this.tok(T.EQEQ, "==", l, c);
553
+ }
554
+ else if ((this.ch() == ">")) {
555
+ this.adv();
556
+ this.tok(T.FATARROW, "=>", l, c);
557
+ }
558
+ else {
559
+ this.tok(T.EQ, "=", l, c);
560
+ }
561
+ }
562
+ else if ((cur == "!")) {
563
+ if (((this.ch() == "=") && (this.src[(this.pos + 1)] == "="))) {
564
+ this.adv();
565
+ this.adv();
566
+ this.tok(T.NEQEQ, "!==", l, c);
567
+ }
568
+ else if ((this.ch() == "=")) {
569
+ this.adv();
570
+ this.tok(T.NEQ, "!=", l, c);
571
+ }
572
+ else {
573
+ this.tok(T.BANG, "!", l, c);
574
+ }
575
+ }
576
+ else if ((cur == "<")) {
577
+ if ((this.ch() == "<")) {
578
+ this.adv();
579
+ this.tok(T.LSHIFT, "<<", l, c);
580
+ }
581
+ else if ((this.ch() == "=")) {
582
+ this.adv();
583
+ this.tok(T.LTE, "<=", l, c);
584
+ }
585
+ else {
586
+ this.tok(T.LT, "<", l, c);
587
+ }
588
+ }
589
+ else if ((cur == ">")) {
590
+ if ((this.ch() == ">")) {
591
+ this.adv();
592
+ this.tok(T.RSHIFT, ">>", l, c);
593
+ }
594
+ else if ((this.ch() == "=")) {
595
+ this.adv();
596
+ this.tok(T.GTE, ">=", l, c);
597
+ }
598
+ else {
599
+ this.tok(T.GT, ">", l, c);
600
+ }
601
+ }
602
+ else if ((cur == ".")) {
603
+ if (((this.ch() == ".") && (this.src[(this.pos + 1)] == "."))) {
604
+ this.adv();
605
+ this.adv();
606
+ this.tok(T.DOTDOTDOT, "...", l, c);
607
+ }
608
+ else if ((this.ch() == ".")) {
609
+ this.adv();
610
+ this.tok(T.DOTDOT, "..", l, c);
611
+ }
612
+ else {
613
+ this.tok(T.DOT, ".", l, c);
614
+ }
615
+ }
616
+ else if ((cur == "?")) {
617
+ if ((this.ch() == ".")) {
618
+ this.adv();
619
+ this.tok(T.QUESTIONDOT, "?.", l, c);
620
+ }
621
+ else if ((this.ch() == "?")) {
622
+ this.adv();
623
+ this.tok(T.NULLISH, "??", l, c);
624
+ }
625
+ else {
626
+ this.tok(T.QUESTION, "?", l, c);
627
+ }
628
+ }
629
+ else if ((cur == "|")) {
630
+ if ((this.ch() == ">")) {
631
+ this.adv();
632
+ this.tok(T.PIPE, "|>", l, c);
633
+ }
634
+ else if ((this.ch() == "|")) {
635
+ this.adv();
636
+ this.tok(T.OROR, "||", l, c);
637
+ }
638
+ else {
639
+ this.tok(T.PIPEB, "|", l, c);
640
+ }
641
+ }
642
+ else if ((cur == "&")) {
643
+ if ((this.ch() == "&")) {
644
+ this.adv();
645
+ this.tok(T.ANDAND, "&&", l, c);
646
+ }
647
+ else {
648
+ this.tok(T.AMPERSAND, "&", l, c);
649
+ }
650
+ }
651
+ else if ((cur == "^")) {
652
+ this.tok(T.CARET, "^", l, c);
653
+ }
654
+ else if ((cur == "~")) {
655
+ this.tok(T.TILDE, "~", l, c);
656
+ }
657
+ else if ((cur == "@")) {
658
+ this.tok(T.AT, "@", l, c);
659
+ }
660
+ else if ((cur == "(")) {
661
+ this.nestDepth = (this.nestDepth + 1);
662
+ this.tok(T.LPAREN, "(", l, c);
663
+ }
664
+ else if ((cur == ")")) {
665
+ this.nestDepth = (this.nestDepth - 1);
666
+ this.tok(T.RPAREN, ")", l, c);
667
+ }
668
+ else if ((cur == "[")) {
669
+ this.nestDepth = (this.nestDepth + 1);
670
+ this.tok(T.LBRACKET, "[", l, c);
671
+ }
672
+ else if ((cur == "]")) {
673
+ this.nestDepth = (this.nestDepth - 1);
674
+ this.tok(T.RBRACKET, "]", l, c);
675
+ }
676
+ else if ((cur == "{")) {
677
+ this.nestDepth = (this.nestDepth + 1);
678
+ this.tok(T.LBRACE, "{", l, c);
679
+ }
680
+ else if ((cur == "}")) {
681
+ this.nestDepth = (this.nestDepth - 1);
682
+ this.tok(T.RBRACE, "}", l, c);
683
+ }
684
+ else if ((cur == ",")) {
685
+ this.tok(T.COMMA, ",", l, c);
686
+ }
687
+ else if ((cur == ":")) {
688
+ this.tok(T.COLON, ":", l, c);
689
+ }
690
+ else if ((cur != ";")) {
691
+ this.err(`Unknown character: '${cur}'`);
692
+ }
693
+ }
694
+ while ((this.indentStack.length > 1)) {
695
+ this.indentStack.pop();
696
+ this.tok(T.DEDENT, null, this.line, 1);
697
+ }
698
+ const lastTok = this.tokens[(this.tokens.length - 1)];
699
+ if (((lastTok && (lastTok.type != T.NEWLINE)) && (lastTok.type != T.DEDENT))) {
700
+ this.tok(T.NEWLINE, null, this.line, this.col);
701
+ }
702
+ this.tok(T.EOF, null, this.line, this.col);
703
+ return this.tokens;
704
+ }
705
+
706
+ }
707
+
708
+ module.exports.Lexer = Lexer;
709
+ function lexerize(source) {
710
+ const processed = source.replace(/\r\n/g, "\n").replace(/\r/g, "\n");
711
+ return new Lexer(processed, 0, 1, 1, [], [0], 0);
712
+ }
713
+ module.exports.lexerize = lexerize;