Files
modeling-app/src/lang/executor.ts

158 lines
5.4 KiB
TypeScript
Raw Normal View History

import { Program, BinaryPart, BinaryExpression } from "./abstractSyntaxTree";
import {Path, sketchFns } from "./sketch";
2022-11-14 14:04:23 +11:00
export interface ProgramMemory {
root: { [key: string]: any };
return?: any;
_sketch: Path[];
}
export const executor = (
node: Program,
programMemory: ProgramMemory = { root: {}, _sketch: [] },
2022-11-21 09:16:24 +11:00
options: { bodyType: "root" | "sketch" | "block" } = { bodyType: "root" }
): any => {
const _programMemory: ProgramMemory = {
2022-11-14 14:04:23 +11:00
root: {
...programMemory.root,
2022-11-14 14:04:23 +11:00
},
_sketch: [],
return: programMemory.return,
2022-11-14 14:04:23 +11:00
};
const { body } = node;
2022-11-14 14:04:23 +11:00
body.forEach((statement) => {
if (statement.type === "VariableDeclaration") {
statement.declarations.forEach((declaration) => {
const variableName = declaration.id.name;
if (declaration.init.type === "Literal") {
_programMemory.root[variableName] = declaration.init.value;
2022-11-14 14:04:23 +11:00
} else if (declaration.init.type === "BinaryExpression") {
_programMemory.root[variableName] = getBinaryExpressionResult(
declaration.init,
_programMemory
);
} else if (declaration.init.type === "SketchExpression") {
const sketchInit = declaration.init;
const fnMemory: ProgramMemory = {
root: {
..._programMemory.root,
},
_sketch: [],
};
const { _sketch } = executor(sketchInit.body, fnMemory, {
bodyType: "sketch",
});
_programMemory.root[variableName] = _sketch;
} else if (declaration.init.type === "FunctionExpression") {
const fnInit = declaration.init;
_programMemory.root[declaration.id.name] = (...args: any[]) => {
const fnMemory: ProgramMemory = {
root: {
..._programMemory.root,
},
_sketch: [],
};
if (args.length > fnInit.params.length) {
throw new Error(
`Too many arguments passed to function ${declaration.id.name}`
);
} else if (args.length < fnInit.params.length) {
throw new Error(
`Too few arguments passed to function ${declaration.id.name}`
);
}
fnInit.params.forEach((param, index) => {
fnMemory.root[param.name] = args[index];
});
2022-11-21 09:16:24 +11:00
return executor(fnInit.body, fnMemory, {bodyType: 'block'}).return;
};
} else if (declaration.init.type === "CallExpression") {
const fnName = declaration.init.callee.name;
const fnArgs = declaration.init.arguments.map((arg) => {
if (arg.type === "Literal") {
return arg.value;
} else if (arg.type === "Identifier") {
return _programMemory.root[arg.name];
}
});
if ("lineTo" === fnName || "close" === fnName) {
if (options.bodyType !== "sketch") {
throw new Error(
`Cannot call ${fnName} outside of a sketch declaration`
);
}
const result = sketchFns[fnName](
_programMemory,
variableName,
[declaration.start, declaration.end],
...fnArgs
);
_programMemory._sketch = result.programMemory._sketch;
_programMemory.root[variableName] = result.currentPath;
} else {
_programMemory.root[variableName] = _programMemory.root[fnName](
...fnArgs
);
}
2022-11-14 14:04:23 +11:00
}
});
} else if (statement.type === "ExpressionStatement") {
const expression = statement.expression;
if (expression.type === "CallExpression") {
const functionName = expression.callee.name;
const args = expression.arguments.map((arg) => {
if (arg.type === "Literal") {
return arg.value;
} else if (arg.type === "Identifier") {
return _programMemory.root[arg.name];
2022-11-14 14:04:23 +11:00
}
});
if ("lineTo" === functionName || "close" === functionName) {
if (options.bodyType !== "sketch") {
throw new Error(
`Cannot call ${functionName} outside of a sketch declaration`
);
}
const result = sketchFns[functionName](_programMemory, "", [statement.start, statement.end], ...args);
_programMemory._sketch = [...result.programMemory._sketch];
2022-11-21 09:16:24 +11:00
} else if("show" === functionName) {
if (options.bodyType !== "root") {
throw new Error(
`Cannot call ${functionName} outside of a root`
);
}
_programMemory.return = expression.arguments;
} else {
_programMemory.root[functionName](...args);
}
}
} else if (statement.type === "ReturnStatement") {
if (statement.argument.type === "BinaryExpression") {
_programMemory.return = getBinaryExpressionResult(
statement.argument,
_programMemory
);
2022-11-14 14:04:23 +11:00
}
}
});
return _programMemory;
2022-11-14 14:04:23 +11:00
};
function getBinaryExpressionResult(
expression: BinaryExpression,
programMemory: ProgramMemory
) {
const getVal = (part: BinaryPart) => {
if (part.type === "Literal") {
return part.value;
} else if (part.type === "Identifier") {
return programMemory.root[part.name];
}
};
const left = getVal(expression.left);
const right = getVal(expression.right);
return left + right;
}