rum-goggles/v1/vendor/github.com/robertkrimen/otto/cmpl_evaluate_expression.go

445 lines
12 KiB
Go
Raw Normal View History

package otto
import (
"fmt"
"math"
2024-04-04 14:46:14 +00:00
goruntime "runtime"
"github.com/robertkrimen/otto/token"
)
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeExpression(node nodeExpression) Value {
// Allow interpreter interruption
// If the Interrupt channel is nil, then
// we avoid runtime.Gosched() overhead (if any)
// FIXME: Test this
2024-04-04 14:46:14 +00:00
if rt.otto.Interrupt != nil {
goruntime.Gosched()
select {
2024-04-04 14:46:14 +00:00
case value := <-rt.otto.Interrupt:
value()
default:
}
}
switch node := node.(type) {
2024-04-04 14:46:14 +00:00
case *nodeArrayLiteral:
return rt.cmplEvaluateNodeArrayLiteral(node)
2024-04-04 14:46:14 +00:00
case *nodeAssignExpression:
return rt.cmplEvaluateNodeAssignExpression(node)
2024-04-04 14:46:14 +00:00
case *nodeBinaryExpression:
if node.comparison {
2024-04-04 14:46:14 +00:00
return rt.cmplEvaluateNodeBinaryExpressionComparison(node)
}
2024-04-04 14:46:14 +00:00
return rt.cmplEvaluateNodeBinaryExpression(node)
2024-04-04 14:46:14 +00:00
case *nodeBracketExpression:
return rt.cmplEvaluateNodeBracketExpression(node)
2024-04-04 14:46:14 +00:00
case *nodeCallExpression:
return rt.cmplEvaluateNodeCallExpression(node, nil)
2024-04-04 14:46:14 +00:00
case *nodeConditionalExpression:
return rt.cmplEvaluateNodeConditionalExpression(node)
2024-04-04 14:46:14 +00:00
case *nodeDotExpression:
return rt.cmplEvaluateNodeDotExpression(node)
2024-04-04 14:46:14 +00:00
case *nodeFunctionLiteral:
local := rt.scope.lexical
if node.name != "" {
2024-04-04 14:46:14 +00:00
local = rt.newDeclarationStash(local)
}
2024-04-04 14:46:14 +00:00
value := objectValue(rt.newNodeFunction(node, local))
if node.name != "" {
local.createBinding(node.name, false, value)
}
return value
2024-04-04 14:46:14 +00:00
case *nodeIdentifier:
name := node.name
// TODO Should be true or false (strictness) depending on context
// getIdentifierReference should not return nil, but we check anyway and panic
// so as not to propagate the nil into something else
2024-04-04 14:46:14 +00:00
reference := getIdentifierReference(rt, rt.scope.lexical, name, false, at(node.idx))
if reference == nil {
// Should never get here!
panic(hereBeDragons("referenceError == nil: " + name))
}
return toValue(reference)
2024-04-04 14:46:14 +00:00
case *nodeLiteral:
return node.value
2024-04-04 14:46:14 +00:00
case *nodeNewExpression:
return rt.cmplEvaluateNodeNewExpression(node)
2024-04-04 14:46:14 +00:00
case *nodeObjectLiteral:
return rt.cmplEvaluateNodeObjectLiteral(node)
2024-04-04 14:46:14 +00:00
case *nodeRegExpLiteral:
return objectValue(rt.newRegExpDirect(node.pattern, node.flags))
2024-04-04 14:46:14 +00:00
case *nodeSequenceExpression:
return rt.cmplEvaluateNodeSequenceExpression(node)
2024-04-04 14:46:14 +00:00
case *nodeThisExpression:
return objectValue(rt.scope.this)
2024-04-04 14:46:14 +00:00
case *nodeUnaryExpression:
return rt.cmplEvaluateNodeUnaryExpression(node)
2024-04-04 14:46:14 +00:00
case *nodeVariableExpression:
return rt.cmplEvaluateNodeVariableExpression(node)
default:
panic(fmt.Sprintf("unknown node type: %T", node))
}
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeArrayLiteral(node *nodeArrayLiteral) Value {
valueArray := []Value{}
for _, node := range node.value {
if node == nil {
valueArray = append(valueArray, emptyValue)
} else {
2024-04-04 14:46:14 +00:00
valueArray = append(valueArray, rt.cmplEvaluateNodeExpression(node).resolve())
}
}
2024-04-04 14:46:14 +00:00
result := rt.newArrayOf(valueArray)
2024-04-04 14:46:14 +00:00
return objectValue(result)
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeAssignExpression(node *nodeAssignExpression) Value {
left := rt.cmplEvaluateNodeExpression(node.left)
right := rt.cmplEvaluateNodeExpression(node.right)
rightValue := right.resolve()
result := rightValue
if node.operator != token.ASSIGN {
2024-04-04 14:46:14 +00:00
result = rt.calculateBinaryExpression(node.operator, left, rightValue)
}
2024-04-04 14:46:14 +00:00
rt.putValue(left.reference(), result)
return result
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeBinaryExpression(node *nodeBinaryExpression) Value {
left := rt.cmplEvaluateNodeExpression(node.left)
leftValue := left.resolve()
switch node.operator {
// Logical
case token.LOGICAL_AND:
if !leftValue.bool() {
return leftValue
}
2024-04-04 14:46:14 +00:00
right := rt.cmplEvaluateNodeExpression(node.right)
return right.resolve()
case token.LOGICAL_OR:
if leftValue.bool() {
return leftValue
}
2024-04-04 14:46:14 +00:00
right := rt.cmplEvaluateNodeExpression(node.right)
return right.resolve()
}
2024-04-04 14:46:14 +00:00
return rt.calculateBinaryExpression(node.operator, leftValue, rt.cmplEvaluateNodeExpression(node.right))
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeBinaryExpressionComparison(node *nodeBinaryExpression) Value {
left := rt.cmplEvaluateNodeExpression(node.left).resolve()
right := rt.cmplEvaluateNodeExpression(node.right).resolve()
2024-04-04 14:46:14 +00:00
return boolValue(rt.calculateComparison(node.operator, left, right))
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeBracketExpression(node *nodeBracketExpression) Value {
target := rt.cmplEvaluateNodeExpression(node.left)
targetValue := target.resolve()
2024-04-04 14:46:14 +00:00
member := rt.cmplEvaluateNodeExpression(node.member)
memberValue := member.resolve()
// TODO Pass in base value as-is, and defer toObject till later?
2024-04-04 14:46:14 +00:00
obj, err := rt.objectCoerce(targetValue)
if err != nil {
2024-04-04 14:46:14 +00:00
panic(rt.panicTypeError("Cannot access member %q of %s", memberValue.string(), err, at(node.idx)))
}
2024-04-04 14:46:14 +00:00
return toValue(newPropertyReference(rt, obj, memberValue.string(), false, at(node.idx)))
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeCallExpression(node *nodeCallExpression, withArgumentList []interface{}) Value {
this := Value{}
2024-04-04 14:46:14 +00:00
callee := rt.cmplEvaluateNodeExpression(node.callee)
argumentList := []Value{}
if withArgumentList != nil {
2024-04-04 14:46:14 +00:00
argumentList = rt.toValueArray(withArgumentList...)
} else {
for _, argumentNode := range node.argumentList {
2024-04-04 14:46:14 +00:00
argumentList = append(argumentList, rt.cmplEvaluateNodeExpression(argumentNode).resolve())
}
}
eval := false // Whether this call is a (candidate for) direct call to eval
name := ""
2024-04-04 14:46:14 +00:00
if rf := callee.reference(); rf != nil {
switch rf := rf.(type) {
2024-04-04 14:46:14 +00:00
case *propertyReference:
name = rf.name
2024-04-04 14:46:14 +00:00
this = objectValue(rf.base)
eval = rf.name == "eval" // Possible direct eval
2024-04-04 14:46:14 +00:00
case *stashReference:
// TODO ImplicitThisValue
name = rf.name
eval = rf.name == "eval" // Possible direct eval
default:
// FIXME?
2024-04-04 14:46:14 +00:00
panic(rt.panicTypeError("unexpected callee type %T to node call expression", rf))
}
}
2024-04-04 14:46:14 +00:00
atv := at(-1)
switch callee := node.callee.(type) {
2024-04-04 14:46:14 +00:00
case *nodeIdentifier:
atv = at(callee.idx)
case *nodeDotExpression:
atv = at(callee.idx)
case *nodeBracketExpression:
atv = at(callee.idx)
}
2024-04-04 14:46:14 +00:00
frm := frame{
callee: name,
2024-04-04 14:46:14 +00:00
file: rt.scope.frame.file,
}
2024-04-04 14:46:14 +00:00
vl := callee.resolve()
if !vl.IsFunction() {
if name == "" {
// FIXME Maybe typeof?
2024-04-04 14:46:14 +00:00
panic(rt.panicTypeError("%v is not a function", vl, atv))
}
2024-04-04 14:46:14 +00:00
panic(rt.panicTypeError("%q is not a function", name, atv))
}
2024-04-04 14:46:14 +00:00
rt.scope.frame.offset = int(atv)
2024-04-04 14:46:14 +00:00
return vl.object().call(this, argumentList, eval, frm)
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeConditionalExpression(node *nodeConditionalExpression) Value {
test := rt.cmplEvaluateNodeExpression(node.test)
testValue := test.resolve()
if testValue.bool() {
2024-04-04 14:46:14 +00:00
return rt.cmplEvaluateNodeExpression(node.consequent)
}
2024-04-04 14:46:14 +00:00
return rt.cmplEvaluateNodeExpression(node.alternate)
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeDotExpression(node *nodeDotExpression) Value {
target := rt.cmplEvaluateNodeExpression(node.left)
targetValue := target.resolve()
// TODO Pass in base value as-is, and defer toObject till later?
2024-04-04 14:46:14 +00:00
obj, err := rt.objectCoerce(targetValue)
if err != nil {
2024-04-04 14:46:14 +00:00
panic(rt.panicTypeError("Cannot access member %q of %s", node.identifier, err, at(node.idx)))
}
2024-04-04 14:46:14 +00:00
return toValue(newPropertyReference(rt, obj, node.identifier, false, at(node.idx)))
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeNewExpression(node *nodeNewExpression) Value {
callee := rt.cmplEvaluateNodeExpression(node.callee)
argumentList := []Value{}
for _, argumentNode := range node.argumentList {
2024-04-04 14:46:14 +00:00
argumentList = append(argumentList, rt.cmplEvaluateNodeExpression(argumentNode).resolve())
}
2024-04-04 14:46:14 +00:00
var name string
if rf := callee.reference(); rf != nil {
switch rf := rf.(type) {
2024-04-04 14:46:14 +00:00
case *propertyReference:
name = rf.name
2024-04-04 14:46:14 +00:00
case *stashReference:
name = rf.name
default:
2024-04-04 14:46:14 +00:00
panic(rt.panicTypeError("node new expression unexpected callee type %T", rf))
}
}
2024-04-04 14:46:14 +00:00
atv := at(-1)
switch callee := node.callee.(type) {
2024-04-04 14:46:14 +00:00
case *nodeIdentifier:
atv = at(callee.idx)
case *nodeDotExpression:
atv = at(callee.idx)
case *nodeBracketExpression:
atv = at(callee.idx)
}
2024-04-04 14:46:14 +00:00
vl := callee.resolve()
if !vl.IsFunction() {
if name == "" {
// FIXME Maybe typeof?
2024-04-04 14:46:14 +00:00
panic(rt.panicTypeError("%v is not a function", vl, atv))
}
2024-04-04 14:46:14 +00:00
panic(rt.panicTypeError("'%s' is not a function", name, atv))
}
2024-04-04 14:46:14 +00:00
rt.scope.frame.offset = int(atv)
2024-04-04 14:46:14 +00:00
return vl.object().construct(argumentList)
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeObjectLiteral(node *nodeObjectLiteral) Value {
result := rt.newObject()
for _, prop := range node.value {
switch prop.kind {
case "value":
2024-04-04 14:46:14 +00:00
result.defineProperty(prop.key, rt.cmplEvaluateNodeExpression(prop.value).resolve(), 0o111, false)
case "get":
2024-04-04 14:46:14 +00:00
getter := rt.newNodeFunction(prop.value.(*nodeFunctionLiteral), rt.scope.lexical)
descriptor := property{}
descriptor.mode = 0o211
descriptor.value = propertyGetSet{getter, nil}
result.defineOwnProperty(prop.key, descriptor, false)
case "set":
2024-04-04 14:46:14 +00:00
setter := rt.newNodeFunction(prop.value.(*nodeFunctionLiteral), rt.scope.lexical)
descriptor := property{}
descriptor.mode = 0o211
descriptor.value = propertyGetSet{nil, setter}
result.defineOwnProperty(prop.key, descriptor, false)
default:
2024-04-04 14:46:14 +00:00
panic(fmt.Sprintf("unknown node object literal property kind %T", prop.kind))
}
}
2024-04-04 14:46:14 +00:00
return objectValue(result)
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeSequenceExpression(node *nodeSequenceExpression) Value {
var result Value
for _, node := range node.sequence {
2024-04-04 14:46:14 +00:00
result = rt.cmplEvaluateNodeExpression(node)
result = result.resolve()
}
return result
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeUnaryExpression(node *nodeUnaryExpression) Value {
target := rt.cmplEvaluateNodeExpression(node.operand)
switch node.operator {
case token.TYPEOF, token.DELETE:
if target.kind == valueReference && target.reference().invalid() {
if node.operator == token.TYPEOF {
2024-04-04 14:46:14 +00:00
return stringValue("undefined")
}
return trueValue
}
}
switch node.operator {
case token.NOT:
targetValue := target.resolve()
if targetValue.bool() {
return falseValue
}
return trueValue
case token.BITWISE_NOT:
targetValue := target.resolve()
integerValue := toInt32(targetValue)
2024-04-04 14:46:14 +00:00
return int32Value(^integerValue)
case token.PLUS:
targetValue := target.resolve()
2024-04-04 14:46:14 +00:00
return float64Value(targetValue.float64())
case token.MINUS:
targetValue := target.resolve()
value := targetValue.float64()
// TODO Test this
sign := float64(-1)
if math.Signbit(value) {
sign = 1
}
2024-04-04 14:46:14 +00:00
return float64Value(math.Copysign(value, sign))
case token.INCREMENT:
targetValue := target.resolve()
if node.postfix {
// Postfix++
oldValue := targetValue.float64()
2024-04-04 14:46:14 +00:00
newValue := float64Value(+1 + oldValue)
rt.putValue(target.reference(), newValue)
return float64Value(oldValue)
}
2024-04-04 14:46:14 +00:00
// ++Prefix
newValue := float64Value(+1 + targetValue.float64())
rt.putValue(target.reference(), newValue)
return newValue
case token.DECREMENT:
targetValue := target.resolve()
if node.postfix {
// Postfix--
oldValue := targetValue.float64()
2024-04-04 14:46:14 +00:00
newValue := float64Value(-1 + oldValue)
rt.putValue(target.reference(), newValue)
return float64Value(oldValue)
}
2024-04-04 14:46:14 +00:00
// --Prefix
newValue := float64Value(-1 + targetValue.float64())
rt.putValue(target.reference(), newValue)
return newValue
case token.VOID:
target.resolve() // FIXME Side effect?
return Value{}
case token.DELETE:
reference := target.reference()
if reference == nil {
return trueValue
}
2024-04-04 14:46:14 +00:00
return boolValue(target.reference().delete())
case token.TYPEOF:
targetValue := target.resolve()
switch targetValue.kind {
case valueUndefined:
2024-04-04 14:46:14 +00:00
return stringValue("undefined")
case valueNull:
2024-04-04 14:46:14 +00:00
return stringValue("object")
case valueBoolean:
2024-04-04 14:46:14 +00:00
return stringValue("boolean")
case valueNumber:
2024-04-04 14:46:14 +00:00
return stringValue("number")
case valueString:
2024-04-04 14:46:14 +00:00
return stringValue("string")
case valueObject:
2024-04-04 14:46:14 +00:00
if targetValue.object().isCall() {
return stringValue("function")
}
2024-04-04 14:46:14 +00:00
return stringValue("object")
default:
// FIXME ?
}
}
panic(hereBeDragons())
}
2024-04-04 14:46:14 +00:00
func (rt *runtime) cmplEvaluateNodeVariableExpression(node *nodeVariableExpression) Value {
if node.initializer != nil {
// FIXME If reference is nil
2024-04-04 14:46:14 +00:00
left := getIdentifierReference(rt, rt.scope.lexical, node.name, false, at(node.idx))
right := rt.cmplEvaluateNodeExpression(node.initializer)
rightValue := right.resolve()
2024-04-04 14:46:14 +00:00
rt.putValue(left, rightValue)
}
2024-04-04 14:46:14 +00:00
return stringValue(node.name)
}