2022-11-20 09:41:21 +11:00
|
|
|
import { Program, BinaryPart, BinaryExpression } from "./abstractSyntaxTree";
|
2022-11-14 14:04:23 +11:00
|
|
|
|
2022-11-20 09:41:21 +11:00
|
|
|
interface ProgramMemory {
|
|
|
|
root: { [key: string]: any };
|
|
|
|
return?: any;
|
|
|
|
}
|
|
|
|
|
|
|
|
export const executor = (
|
|
|
|
node: Program,
|
|
|
|
programMemory: ProgramMemory = { root: {} }
|
|
|
|
): any => {
|
|
|
|
const _programMemory: ProgramMemory = {
|
2022-11-14 14:04:23 +11:00
|
|
|
root: {
|
2022-11-20 09:41:21 +11:00
|
|
|
...programMemory.root,
|
2022-11-14 14:04:23 +11:00
|
|
|
},
|
2022-11-20 09:41:21 +11:00
|
|
|
return: programMemory.return,
|
2022-11-14 14:04:23 +11:00
|
|
|
};
|
2022-11-20 09:41:21 +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") {
|
2022-11-20 09:41:21 +11:00
|
|
|
_programMemory.root[variableName] = declaration.init.value;
|
2022-11-14 14:04:23 +11:00
|
|
|
} else if (declaration.init.type === "BinaryExpression") {
|
2022-11-20 09:41:21 +11:00
|
|
|
_programMemory.root[variableName] = getBinaryExpressionResult(declaration.init, _programMemory);
|
2022-11-20 08:18:45 +11:00
|
|
|
} else if (declaration.init.type === "FunctionExpression") {
|
|
|
|
const fnInit = declaration.init;
|
2022-11-20 09:41:21 +11:00
|
|
|
|
|
|
|
_programMemory.root[declaration.id.name] = (...args: any[]) => {
|
|
|
|
const fnMemory: ProgramMemory = {
|
2022-11-20 08:18:45 +11:00
|
|
|
root: {
|
2022-11-20 09:41:21 +11:00
|
|
|
..._programMemory.root,
|
2022-11-20 08:18:45 +11:00
|
|
|
},
|
|
|
|
};
|
2022-11-20 09:41:21 +11:00
|
|
|
if (args.length > fnInit.params.length) {
|
|
|
|
throw new Error(
|
|
|
|
`Too many arguments passed to function ${declaration.id.name}`
|
|
|
|
);
|
2022-11-20 08:18:45 +11:00
|
|
|
} else if (args.length < fnInit.params.length) {
|
2022-11-20 09:41:21 +11:00
|
|
|
throw new Error(
|
|
|
|
`Too few arguments passed to function ${declaration.id.name}`
|
|
|
|
);
|
2022-11-20 08:18:45 +11:00
|
|
|
}
|
|
|
|
fnInit.params.forEach((param, index) => {
|
|
|
|
fnMemory.root[param.name] = args[index];
|
|
|
|
});
|
2022-11-20 09:41:21 +11:00
|
|
|
return executor(fnInit.body, fnMemory).return;
|
|
|
|
};
|
|
|
|
} else if (declaration.init.type === "CallExpression") {
|
2022-11-20 08:18:45 +11:00
|
|
|
const fnName = declaration.init.callee.name;
|
|
|
|
const fnArgs = declaration.init.arguments.map((arg) => {
|
2022-11-20 09:41:21 +11:00
|
|
|
if (arg.type === "Literal") {
|
2022-11-20 08:18:45 +11:00
|
|
|
return arg.value;
|
2022-11-20 09:41:21 +11:00
|
|
|
} else if (arg.type === "Identifier") {
|
|
|
|
return _programMemory.root[arg.name];
|
2022-11-20 08:18:45 +11:00
|
|
|
}
|
2022-11-20 09:41:21 +11:00
|
|
|
});
|
|
|
|
_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") {
|
2022-11-20 09:41:21 +11:00
|
|
|
return _programMemory.root[arg.name];
|
2022-11-14 14:04:23 +11:00
|
|
|
}
|
|
|
|
});
|
2022-11-20 09:41:21 +11:00
|
|
|
_programMemory.root[functionName](...args);
|
|
|
|
}
|
|
|
|
} else if (statement.type === "ReturnStatement") {
|
|
|
|
if(statement.argument.type === "BinaryExpression") {
|
|
|
|
const returnValue = getBinaryExpressionResult(statement.argument, _programMemory);
|
|
|
|
_programMemory.return = returnValue;
|
2022-11-14 14:04:23 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2022-11-20 09:41:21 +11:00
|
|
|
return _programMemory;
|
2022-11-14 14:04:23 +11:00
|
|
|
};
|
2022-11-20 09:41:21 +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;
|
|
|
|
}
|