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

160 lines
5.5 KiB
TypeScript
Raw Normal View History

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