2015-02-16 13:28:33 +00:00
|
|
|
package otto
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"reflect"
|
2015-03-20 12:22:01 +00:00
|
|
|
"sync"
|
2015-02-16 13:28:33 +00:00
|
|
|
|
|
|
|
"github.com/robertkrimen/otto/ast"
|
|
|
|
"github.com/robertkrimen/otto/parser"
|
|
|
|
)
|
|
|
|
|
|
|
|
type _global struct {
|
|
|
|
Object *_object // Object( ... ), new Object( ... ) - 1 (length)
|
|
|
|
Function *_object // Function( ... ), new Function( ... ) - 1
|
|
|
|
Array *_object // Array( ... ), new Array( ... ) - 1
|
|
|
|
String *_object // String( ... ), new String( ... ) - 1
|
|
|
|
Boolean *_object // Boolean( ... ), new Boolean( ... ) - 1
|
|
|
|
Number *_object // Number( ... ), new Number( ... ) - 1
|
|
|
|
Math *_object
|
|
|
|
Date *_object // Date( ... ), new Date( ... ) - 7
|
|
|
|
RegExp *_object // RegExp( ... ), new RegExp( ... ) - 2
|
|
|
|
Error *_object // Error( ... ), new Error( ... ) - 1
|
|
|
|
EvalError *_object
|
|
|
|
TypeError *_object
|
|
|
|
RangeError *_object
|
|
|
|
ReferenceError *_object
|
|
|
|
SyntaxError *_object
|
|
|
|
URIError *_object
|
|
|
|
JSON *_object
|
|
|
|
|
|
|
|
ObjectPrototype *_object // Object.prototype
|
|
|
|
FunctionPrototype *_object // Function.prototype
|
|
|
|
ArrayPrototype *_object // Array.prototype
|
|
|
|
StringPrototype *_object // String.prototype
|
|
|
|
BooleanPrototype *_object // Boolean.prototype
|
|
|
|
NumberPrototype *_object // Number.prototype
|
|
|
|
DatePrototype *_object // Date.prototype
|
|
|
|
RegExpPrototype *_object // RegExp.prototype
|
|
|
|
ErrorPrototype *_object // Error.prototype
|
|
|
|
EvalErrorPrototype *_object
|
|
|
|
TypeErrorPrototype *_object
|
|
|
|
RangeErrorPrototype *_object
|
|
|
|
ReferenceErrorPrototype *_object
|
|
|
|
SyntaxErrorPrototype *_object
|
|
|
|
URIErrorPrototype *_object
|
|
|
|
}
|
|
|
|
|
|
|
|
type _runtime struct {
|
2015-03-20 12:22:01 +00:00
|
|
|
global _global
|
|
|
|
globalObject *_object
|
|
|
|
globalStash *_objectStash
|
|
|
|
scope *_scope
|
|
|
|
otto *Otto
|
|
|
|
eval *_object // The builtin eval, for determine indirect versus direct invocation
|
2015-02-16 13:28:33 +00:00
|
|
|
|
|
|
|
labels []string // FIXME
|
2015-03-20 12:22:01 +00:00
|
|
|
lck sync.Mutex
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
|
2015-03-20 12:22:01 +00:00
|
|
|
func (self *_runtime) enterScope(scope *_scope) {
|
|
|
|
scope.outer = self.scope
|
|
|
|
self.scope = scope
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
|
2015-03-20 12:22:01 +00:00
|
|
|
func (self *_runtime) leaveScope() {
|
|
|
|
self.scope = self.scope.outer
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
|
2015-03-20 12:22:01 +00:00
|
|
|
// FIXME This is used in two places (cloning)
|
|
|
|
func (self *_runtime) enterGlobalScope() {
|
|
|
|
self.enterScope(newScope(self.globalStash, self.globalStash, self.globalObject))
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
|
2015-03-20 12:22:01 +00:00
|
|
|
func (self *_runtime) enterFunctionScope(outer _stash, this Value) *_fnStash {
|
|
|
|
if outer == nil {
|
|
|
|
outer = self.globalStash
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
2015-03-20 12:22:01 +00:00
|
|
|
stash := self.newFunctionStash(outer)
|
2015-02-16 13:28:33 +00:00
|
|
|
var thisObject *_object
|
2015-03-20 12:22:01 +00:00
|
|
|
switch this.kind {
|
2015-02-16 13:28:33 +00:00
|
|
|
case valueUndefined, valueNull:
|
2015-03-20 12:22:01 +00:00
|
|
|
thisObject = self.globalObject
|
2015-02-16 13:28:33 +00:00
|
|
|
default:
|
|
|
|
thisObject = self.toObject(this)
|
|
|
|
}
|
2015-03-20 12:22:01 +00:00
|
|
|
self.enterScope(newScope(stash, stash, thisObject))
|
|
|
|
return stash
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
|
2015-03-20 12:22:01 +00:00
|
|
|
func (self *_runtime) putValue(reference _reference, value Value) {
|
|
|
|
name := reference.putValue(value)
|
|
|
|
if name != "" {
|
|
|
|
// Why? -- If reference.base == nil
|
|
|
|
// strict = false
|
|
|
|
self.globalObject.defineProperty(name, value, 0111, false)
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *_runtime) tryCatchEvaluate(inner func() Value) (tryValue Value, exception bool) {
|
|
|
|
// resultValue = The value of the block (e.g. the last statement)
|
|
|
|
// throw = Something was thrown
|
|
|
|
// throwValue = The value of what was thrown
|
|
|
|
// other = Something that changes flow (return, break, continue) that is not a throw
|
|
|
|
// Otherwise, some sort of unknown panic happened, we'll just propagate it
|
|
|
|
defer func() {
|
|
|
|
if caught := recover(); caught != nil {
|
|
|
|
if exception, ok := caught.(*_exception); ok {
|
|
|
|
caught = exception.eject()
|
|
|
|
}
|
|
|
|
switch caught := caught.(type) {
|
|
|
|
case _error:
|
|
|
|
exception = true
|
2015-03-20 12:22:01 +00:00
|
|
|
tryValue = toValue_object(self.newError(caught.name, caught.messageValue()))
|
2015-02-16 13:28:33 +00:00
|
|
|
case Value:
|
|
|
|
exception = true
|
|
|
|
tryValue = caught
|
|
|
|
default:
|
|
|
|
panic(caught)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
tryValue = inner()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// toObject
|
|
|
|
|
|
|
|
func (self *_runtime) toObject(value Value) *_object {
|
2015-03-20 12:22:01 +00:00
|
|
|
switch value.kind {
|
2015-02-16 13:28:33 +00:00
|
|
|
case valueEmpty, valueUndefined, valueNull:
|
2015-03-20 12:22:01 +00:00
|
|
|
panic(self.panicTypeError())
|
2015-02-16 13:28:33 +00:00
|
|
|
case valueBoolean:
|
|
|
|
return self.newBoolean(value)
|
|
|
|
case valueString:
|
|
|
|
return self.newString(value)
|
|
|
|
case valueNumber:
|
|
|
|
return self.newNumber(value)
|
|
|
|
case valueObject:
|
|
|
|
return value._object()
|
|
|
|
}
|
2015-03-20 12:22:01 +00:00
|
|
|
panic(self.panicTypeError())
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (self *_runtime) objectCoerce(value Value) (*_object, error) {
|
2015-03-20 12:22:01 +00:00
|
|
|
switch value.kind {
|
2015-02-16 13:28:33 +00:00
|
|
|
case valueUndefined:
|
|
|
|
return nil, errors.New("undefined")
|
|
|
|
case valueNull:
|
|
|
|
return nil, errors.New("null")
|
|
|
|
case valueBoolean:
|
|
|
|
return self.newBoolean(value), nil
|
|
|
|
case valueString:
|
|
|
|
return self.newString(value), nil
|
|
|
|
case valueNumber:
|
|
|
|
return self.newNumber(value), nil
|
|
|
|
case valueObject:
|
|
|
|
return value._object(), nil
|
|
|
|
}
|
2015-03-20 12:22:01 +00:00
|
|
|
panic(self.panicTypeError())
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
|
2015-03-20 12:22:01 +00:00
|
|
|
func checkObjectCoercible(rt *_runtime, value Value) {
|
2015-02-16 13:28:33 +00:00
|
|
|
isObject, mustCoerce := testObjectCoercible(value)
|
|
|
|
if !isObject && !mustCoerce {
|
2015-03-20 12:22:01 +00:00
|
|
|
panic(rt.panicTypeError())
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// testObjectCoercible
|
|
|
|
|
|
|
|
func testObjectCoercible(value Value) (isObject bool, mustCoerce bool) {
|
2015-03-20 12:22:01 +00:00
|
|
|
switch value.kind {
|
2015-02-16 13:28:33 +00:00
|
|
|
case valueReference, valueEmpty, valueNull, valueUndefined:
|
|
|
|
return false, false
|
|
|
|
case valueNumber, valueString, valueBoolean:
|
|
|
|
isObject = false
|
|
|
|
mustCoerce = true
|
|
|
|
case valueObject:
|
|
|
|
isObject = true
|
|
|
|
mustCoerce = false
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2015-03-20 12:22:01 +00:00
|
|
|
func (self *_runtime) safeToValue(value interface{}) (Value, error) {
|
|
|
|
result := Value{}
|
2015-02-16 13:28:33 +00:00
|
|
|
err := catchPanic(func() {
|
|
|
|
result = self.toValue(value)
|
|
|
|
})
|
|
|
|
return result, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *_runtime) toValue(value interface{}) Value {
|
|
|
|
switch value := value.(type) {
|
|
|
|
case Value:
|
|
|
|
return value
|
|
|
|
case func(FunctionCall) Value:
|
|
|
|
return toValue_object(self.newNativeFunction("", value))
|
|
|
|
case _nativeFunction:
|
|
|
|
return toValue_object(self.newNativeFunction("", value))
|
|
|
|
case Object, *Object, _object, *_object:
|
|
|
|
// Nothing happens.
|
2015-03-20 12:22:01 +00:00
|
|
|
// FIXME We should really figure out what can come here.
|
|
|
|
// This catch-all is ugly.
|
2015-02-16 13:28:33 +00:00
|
|
|
default:
|
|
|
|
{
|
|
|
|
value := reflect.ValueOf(value)
|
|
|
|
switch value.Kind() {
|
|
|
|
case reflect.Ptr:
|
|
|
|
switch reflect.Indirect(value).Kind() {
|
|
|
|
case reflect.Struct:
|
|
|
|
return toValue_object(self.newGoStructObject(value))
|
|
|
|
case reflect.Array:
|
|
|
|
return toValue_object(self.newGoArray(value))
|
|
|
|
}
|
|
|
|
case reflect.Func:
|
2015-03-20 12:22:01 +00:00
|
|
|
// TODO Maybe cache this?
|
2015-02-16 13:28:33 +00:00
|
|
|
return toValue_object(self.newNativeFunction("", func(call FunctionCall) Value {
|
2015-03-20 12:22:01 +00:00
|
|
|
in := make([]reflect.Value, len(call.ArgumentList))
|
|
|
|
for i, value := range call.ArgumentList {
|
|
|
|
in[i] = reflect.ValueOf(value.export())
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
|
2015-03-20 12:22:01 +00:00
|
|
|
out := value.Call(in)
|
|
|
|
if len(out) == 1 {
|
|
|
|
return self.toValue(out[0].Interface())
|
|
|
|
} else if len(out) == 0 {
|
|
|
|
return Value{}
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
2015-03-20 12:22:01 +00:00
|
|
|
|
|
|
|
panic(call.runtime.panicTypeError())
|
2015-02-16 13:28:33 +00:00
|
|
|
}))
|
|
|
|
case reflect.Struct:
|
|
|
|
return toValue_object(self.newGoStructObject(value))
|
|
|
|
case reflect.Map:
|
|
|
|
return toValue_object(self.newGoMapObject(value))
|
|
|
|
case reflect.Slice:
|
|
|
|
return toValue_object(self.newGoSlice(value))
|
|
|
|
case reflect.Array:
|
|
|
|
return toValue_object(self.newGoArray(value))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return toValue(value)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (runtime *_runtime) newGoSlice(value reflect.Value) *_object {
|
|
|
|
self := runtime.newGoSliceObject(value)
|
2015-03-20 12:22:01 +00:00
|
|
|
self.prototype = runtime.global.ArrayPrototype
|
2015-02-16 13:28:33 +00:00
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
|
|
|
func (runtime *_runtime) newGoArray(value reflect.Value) *_object {
|
|
|
|
self := runtime.newGoArrayObject(value)
|
2015-03-20 12:22:01 +00:00
|
|
|
self.prototype = runtime.global.ArrayPrototype
|
2015-02-16 13:28:33 +00:00
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
|
|
|
func (runtime *_runtime) parse(filename string, src interface{}) (*ast.Program, error) {
|
|
|
|
return parser.ParseFile(nil, filename, src, 0)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (runtime *_runtime) cmpl_parse(filename string, src interface{}) (*_nodeProgram, error) {
|
|
|
|
program, err := parser.ParseFile(nil, filename, src, 0)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return cmpl_parse(program), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *_runtime) parseSource(src interface{}) (*_nodeProgram, *ast.Program, error) {
|
|
|
|
switch src := src.(type) {
|
|
|
|
case *ast.Program:
|
|
|
|
return nil, src, nil
|
|
|
|
case *Script:
|
|
|
|
return src.program, nil, nil
|
|
|
|
}
|
|
|
|
program, err := self.parse("", src)
|
|
|
|
return nil, program, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *_runtime) cmpl_run(src interface{}) (Value, error) {
|
2015-03-20 12:22:01 +00:00
|
|
|
result := Value{}
|
2015-02-16 13:28:33 +00:00
|
|
|
cmpl_program, program, err := self.parseSource(src)
|
|
|
|
if err != nil {
|
|
|
|
return result, err
|
|
|
|
}
|
|
|
|
if cmpl_program == nil {
|
|
|
|
cmpl_program = cmpl_parse(program)
|
|
|
|
}
|
|
|
|
err = catchPanic(func() {
|
2015-03-20 12:22:01 +00:00
|
|
|
result = self.cmpl_evaluate_nodeProgram(cmpl_program, false)
|
2015-02-16 13:28:33 +00:00
|
|
|
})
|
2015-03-20 12:22:01 +00:00
|
|
|
switch result.kind {
|
2015-02-16 13:28:33 +00:00
|
|
|
case valueEmpty:
|
2015-03-20 12:22:01 +00:00
|
|
|
result = Value{}
|
2015-02-16 13:28:33 +00:00
|
|
|
case valueReference:
|
2015-03-20 12:22:01 +00:00
|
|
|
result = result.resolve()
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
return result, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *_runtime) parseThrow(err error) {
|
|
|
|
if err == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
switch err := err.(type) {
|
|
|
|
case parser.ErrorList:
|
|
|
|
{
|
|
|
|
err := err[0]
|
|
|
|
if err.Message == "Invalid left-hand side in assignment" {
|
2015-03-20 12:22:01 +00:00
|
|
|
panic(self.panicReferenceError(err.Message))
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
2015-03-20 12:22:01 +00:00
|
|
|
panic(self.panicSyntaxError(err.Message))
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
}
|
2015-03-20 12:22:01 +00:00
|
|
|
panic(self.panicSyntaxError(err.Error()))
|
2015-02-16 13:28:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (self *_runtime) parseOrThrow(source string) *ast.Program {
|
|
|
|
program, err := self.parse("", source)
|
|
|
|
self.parseThrow(err) // Will panic/throw appropriately
|
|
|
|
return program
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *_runtime) cmpl_parseOrThrow(source string) *_nodeProgram {
|
|
|
|
program, err := self.cmpl_parse("", source)
|
|
|
|
self.parseThrow(err) // Will panic/throw appropriately
|
|
|
|
return program
|
|
|
|
}
|