407 lines
13 KiB
TypeScript
407 lines
13 KiB
TypeScript
import { lezerToParseTree } from "$common/markdown_parser/parse_tree.ts";
|
|
import {
|
|
type AST as CrudeAST,
|
|
parseTreeToAST,
|
|
} from "@silverbulletmd/silverbullet/lib/tree";
|
|
import { parser } from "./parse-lua.js";
|
|
import { styleTags } from "@lezer/highlight";
|
|
import type {
|
|
LuaAttName,
|
|
LuaBlock,
|
|
LuaExpression,
|
|
LuaFunctionBody,
|
|
LuaFunctionCallExpression,
|
|
LuaFunctionName,
|
|
LuaLValue,
|
|
LuaPrefixExpression,
|
|
LuaStatement,
|
|
LuaTableField,
|
|
} from "./ast.ts";
|
|
|
|
const luaStyleTags = styleTags({
|
|
// Identifier: t.variableName,
|
|
// TagIdentifier: t.variableName,
|
|
// GlobalIdentifier: t.variableName,
|
|
// String: t.string,
|
|
// Number: t.number,
|
|
// PageRef: ct.WikiLinkTag,
|
|
// BinExpression: t.operator,
|
|
// TernaryExpression: t.operator,
|
|
// Regex: t.regexp,
|
|
// "where limit select render Order OrderKW and or null as InKW NotKW BooleanKW each all":
|
|
// t.keyword,
|
|
});
|
|
|
|
export const highlightingQueryParser = parser.configure({
|
|
props: [
|
|
luaStyleTags,
|
|
],
|
|
});
|
|
|
|
function parseChunk(n: CrudeAST): LuaBlock {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
if (t[0] !== "Chunk") {
|
|
throw new Error(`Expected Chunk, got ${t[0]}`);
|
|
}
|
|
return parseBlock(t[1]);
|
|
}
|
|
|
|
function parseBlock(n: CrudeAST): LuaBlock {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
if (t[0] !== "Block") {
|
|
throw new Error(`Expected Block, got ${t[0]}`);
|
|
}
|
|
const statements = t.slice(1).map(parseStatement);
|
|
return { type: "Block", statements };
|
|
}
|
|
|
|
function parseStatement(n: CrudeAST): LuaStatement {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
switch (t[0]) {
|
|
case "Block":
|
|
return parseChunk(t[1]);
|
|
case "Semicolon":
|
|
return { type: "Semicolon" };
|
|
case "Label":
|
|
return { type: "Label", name: t[2][1] as string };
|
|
case "Break":
|
|
return { type: "Break" };
|
|
case "Goto":
|
|
return { type: "Goto", name: t[2][1] as string };
|
|
case "Scope":
|
|
return parseBlock(t[2]);
|
|
case ";":
|
|
return { type: "Semicolon" };
|
|
case "WhileStatement":
|
|
return {
|
|
type: "While",
|
|
condition: parseExpression(t[2]),
|
|
block: parseBlock(t[4]),
|
|
};
|
|
case "RepeatStatement":
|
|
return {
|
|
type: "Repeat",
|
|
block: parseBlock(t[2]),
|
|
condition: parseExpression(t[4]),
|
|
};
|
|
case "IfStatement": {
|
|
const conditions: { condition: LuaExpression; block: LuaBlock }[] =
|
|
[];
|
|
let elseBlock: LuaBlock | undefined = undefined;
|
|
for (let i = 1; i < t.length; i += 4) {
|
|
console.log("Looking at", t[i]);
|
|
if (t[i][0] === "if" || t[i][0] === "elseif") {
|
|
conditions.push({
|
|
condition: parseExpression(t[i + 1]),
|
|
block: parseBlock(t[i + 3]),
|
|
});
|
|
} else if (t[i][0] === "else") {
|
|
elseBlock = parseBlock(t[i + 1]);
|
|
} else if (t[i][0] === "end") {
|
|
break;
|
|
} else {
|
|
throw new Error(`Unknown if clause type: ${t[i][0]}`);
|
|
}
|
|
}
|
|
return {
|
|
type: "If",
|
|
conditions,
|
|
elseBlock,
|
|
};
|
|
}
|
|
case "ForStatement":
|
|
if (t[2][0] === "ForNumeric") {
|
|
const forNumeric = t[2] as [string, ...CrudeAST[]];
|
|
return {
|
|
type: "For",
|
|
name: forNumeric[1][1] as string,
|
|
start: parseExpression(forNumeric[3]),
|
|
end: parseExpression(forNumeric[5]),
|
|
step: forNumeric[6]
|
|
? parseExpression(forNumeric[7])
|
|
: undefined,
|
|
block: parseBlock(t[4]),
|
|
};
|
|
} else {
|
|
const forGeneric = t[2] as [string, ...CrudeAST[]];
|
|
return {
|
|
type: "ForIn",
|
|
names: parseNameList(forGeneric[1]),
|
|
expressions: parseExpList(forGeneric[3]),
|
|
block: parseBlock(t[4]),
|
|
};
|
|
}
|
|
case "Function":
|
|
return {
|
|
type: "Function",
|
|
name: parseFunctionName(t[2]),
|
|
body: parseFunctionBody(t[3]),
|
|
};
|
|
case "LocalFunction":
|
|
return {
|
|
type: "LocalFunction",
|
|
name: t[3][1] as string,
|
|
body: parseFunctionBody(t[4]),
|
|
};
|
|
case "FunctionCall":
|
|
return {
|
|
type: "FunctionCallStatement",
|
|
call: parseExpression([
|
|
"FunctionCall",
|
|
...t.slice(1),
|
|
]) as LuaFunctionCallExpression,
|
|
};
|
|
case "Assign":
|
|
return {
|
|
type: "Assignment",
|
|
variables: (t[1].slice(1) as CrudeAST[]).filter((t) =>
|
|
t[0] != ","
|
|
).map(parseLValue),
|
|
expressions: parseExpList(t[3]),
|
|
};
|
|
case "Local":
|
|
return {
|
|
type: "Local",
|
|
names: parseAttNames(t[2]),
|
|
expressions: t[4] ? parseExpList(t[4]) : [],
|
|
};
|
|
default:
|
|
console.error(t);
|
|
throw new Error(`Unknown statement type: ${t[0]}`);
|
|
}
|
|
}
|
|
|
|
function parseAttNames(n: CrudeAST): LuaAttName[] {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
if (t[0] !== "AttNameList") {
|
|
throw new Error(`Expected AttNameList, got ${t[0]}`);
|
|
}
|
|
return t.slice(1).filter((t) => t[0] !== ",").map(parseAttName);
|
|
}
|
|
|
|
function parseAttName(n: CrudeAST): LuaAttName {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
if (t[0] !== "AttName") {
|
|
throw new Error(`Expected AttName, got ${t[0]}`);
|
|
}
|
|
return {
|
|
type: "AttName",
|
|
name: t[1][1] as string,
|
|
attribute: t[2][2] ? t[2][2][1] as string : undefined,
|
|
};
|
|
}
|
|
|
|
function parseLValue(n: CrudeAST): LuaLValue {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
switch (t[0]) {
|
|
case "Name":
|
|
return { type: "Variable", name: t[1] as string };
|
|
case "Property":
|
|
return {
|
|
type: "PropertyAccess",
|
|
object: parsePrefixExpression(t[1]),
|
|
property: t[3][1] as string,
|
|
};
|
|
case "MemberExpression":
|
|
return {
|
|
type: "TableAccess",
|
|
object: parsePrefixExpression(t[1]),
|
|
key: parseExpression(t[3]),
|
|
};
|
|
default:
|
|
console.error(t);
|
|
throw new Error(`Unknown lvalue type: ${t[0]}`);
|
|
}
|
|
}
|
|
|
|
function parseFunctionName(n: CrudeAST): LuaFunctionName {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
if (t[0] !== "FuncName") {
|
|
throw new Error(`Expected FunctionName, got ${t[0]}`);
|
|
}
|
|
const propNames: string[] = [];
|
|
let colonName: string | undefined = undefined;
|
|
for (let i = 1; i < t.length; i += 2) {
|
|
propNames.push(t[i][1] as string);
|
|
if (t[i + 1] && t[i + 1][0] === ":") {
|
|
colonName = t[i + 2][1] as string;
|
|
break;
|
|
}
|
|
}
|
|
return { type: "FunctionName", propNames, colonName };
|
|
}
|
|
|
|
function parseNameList(n: CrudeAST): string[] {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
if (t[0] !== "NameList") {
|
|
throw new Error(`Expected NameList, got ${t[0]}`);
|
|
}
|
|
return t.slice(1).filter((t) => t[0] === "Name").map((t) => t[1] as string);
|
|
}
|
|
|
|
function parseExpList(n: CrudeAST): LuaExpression[] {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
if (t[0] !== "ExpList") {
|
|
throw new Error(`Expected ExpList, got ${t[0]}`);
|
|
}
|
|
return t.slice(1).filter((t) => t[0] !== ",").map(parseExpression);
|
|
}
|
|
|
|
function parseExpression(n: CrudeAST): LuaExpression {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
switch (t[0]) {
|
|
case "LiteralString": {
|
|
let cleanString = t[1] as string;
|
|
// Remove quotes etc
|
|
cleanString = cleanString.slice(1, -1);
|
|
return { type: "String", value: cleanString };
|
|
}
|
|
case "Number":
|
|
return { type: "Number", value: parseFloat(t[1] as string) };
|
|
case "BinaryExpression":
|
|
return {
|
|
type: "Binary",
|
|
operator: t[2][1] as string,
|
|
left: parseExpression(t[1]),
|
|
right: parseExpression(t[3]),
|
|
};
|
|
case "UnaryExpression":
|
|
return {
|
|
type: "Unary",
|
|
operator: t[1][1] as string,
|
|
argument: parseExpression(t[2]),
|
|
};
|
|
case "Property":
|
|
return {
|
|
type: "PropertyAccess",
|
|
object: parsePrefixExpression(t[1]),
|
|
property: t[3][1] as string,
|
|
};
|
|
|
|
case "Parens":
|
|
return parseExpression(t[2]);
|
|
case "FunctionCall": {
|
|
if (t[2][0] === ":") {
|
|
return {
|
|
type: "FunctionCall",
|
|
prefix: parsePrefixExpression(t[1]),
|
|
name: t[3][1] as string,
|
|
args: parseFunctionArgs(t.slice(4)),
|
|
};
|
|
}
|
|
return {
|
|
type: "FunctionCall",
|
|
prefix: parsePrefixExpression(t[1]),
|
|
args: parseFunctionArgs(t.slice(2)),
|
|
};
|
|
}
|
|
case "FunctionDef": {
|
|
const body = parseFunctionBody(t[2]);
|
|
return {
|
|
type: "FunctionDefinition",
|
|
body,
|
|
};
|
|
}
|
|
case "Name":
|
|
return { type: "Variable", name: t[1] as string };
|
|
case "Ellipsis":
|
|
return { type: "Variable", name: "..." };
|
|
case "true":
|
|
return { type: "Boolean", value: true };
|
|
case "false":
|
|
return { type: "Boolean", value: false };
|
|
case "TableConstructor":
|
|
return {
|
|
type: "TableConstructor",
|
|
fields: t.slice(2, -1).filter((t) =>
|
|
!(typeof t === "string" ||
|
|
["{", "}"].includes(t[1] as string))
|
|
).map(parseTableField),
|
|
};
|
|
case "nil":
|
|
return { type: "Nil" };
|
|
default:
|
|
console.error(t);
|
|
throw new Error(`Unknown expression type: ${t[0]}`);
|
|
}
|
|
}
|
|
|
|
function parseFunctionArgs(n: CrudeAST[]): LuaExpression[] {
|
|
console.log("Parsing function args", n);
|
|
return n.filter((t) => ![",", "(", ")"].includes(t[0])).map(
|
|
parseExpression,
|
|
);
|
|
}
|
|
|
|
function parseFunctionBody(n: CrudeAST): LuaFunctionBody {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
if (t[0] !== "FuncBody") {
|
|
throw new Error(`Expected FunctionBody, got ${t[0]}`);
|
|
}
|
|
return {
|
|
type: "FunctionBody",
|
|
parameters: (t[2] as CrudeAST[]).slice(1).filter((t) =>
|
|
["Name", "Ellipsis"].includes(t[0])
|
|
)
|
|
.map((t) => t[1] as string),
|
|
block: parseBlock(t[4]),
|
|
};
|
|
}
|
|
|
|
function parsePrefixExpression(n: CrudeAST): LuaPrefixExpression {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
switch (t[0]) {
|
|
case "Name":
|
|
return { type: "Variable", name: t[1] as string };
|
|
case "Property":
|
|
return {
|
|
type: "PropertyAccess",
|
|
object: parsePrefixExpression(t[1]),
|
|
property: t[3][1] as string,
|
|
};
|
|
case "Parens":
|
|
return { type: "Parenthesized", expression: parseExpression(t[2]) };
|
|
default:
|
|
console.error(t);
|
|
throw new Error(`Unknown prefix expression type: ${t[0]}`);
|
|
}
|
|
}
|
|
|
|
function parseTableField(n: CrudeAST): LuaTableField {
|
|
const t = n as [string, ...CrudeAST[]];
|
|
switch (t[0]) {
|
|
case "FieldExp":
|
|
return {
|
|
type: "ExpressionField",
|
|
value: parseExpression(t[1]),
|
|
};
|
|
case "FieldProp":
|
|
return {
|
|
type: "PropField",
|
|
key: t[1][1] as string,
|
|
value: parseExpression(t[3]),
|
|
};
|
|
case "FieldDynamic":
|
|
return {
|
|
type: "DynamicField",
|
|
key: parseExpression(t[2]),
|
|
value: parseExpression(t[5]),
|
|
};
|
|
default:
|
|
console.error(t);
|
|
throw new Error(`Unknown table field type: ${t[0]}`);
|
|
}
|
|
}
|
|
|
|
export function parse(t: string): LuaBlock {
|
|
const crudeAst = parseToCrudeAST(t);
|
|
console.log("Crude AST", JSON.stringify(crudeAst, null, 2));
|
|
const result = parseChunk(crudeAst);
|
|
console.log("Parsed AST", JSON.stringify(result, null, 2));
|
|
return result;
|
|
}
|
|
|
|
export function parseToCrudeAST(t: string): CrudeAST {
|
|
return parseTreeToAST(lezerToParseTree(t, parser.parse(t).topNode), true);
|
|
}
|