package otto import ( "encoding" "encoding/json" "errors" "fmt" "math" "path" "reflect" goruntime "runtime" "strconv" "strings" "sync" "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 { global global globalObject *object globalStash *objectStash scope *scope otto *Otto eval *object // The builtin eval, for determine indirect versus direct invocation debugger func(*Otto) random func() float64 stackLimit int traceLimit int labels []string // FIXME lck sync.Mutex } func (rt *runtime) enterScope(scop *scope) { scop.outer = rt.scope if rt.scope != nil { if rt.stackLimit != 0 && rt.scope.depth+1 >= rt.stackLimit { panic(rt.panicRangeError("Maximum call stack size exceeded")) } scop.depth = rt.scope.depth + 1 } rt.scope = scop } func (rt *runtime) leaveScope() { rt.scope = rt.scope.outer } // FIXME This is used in two places (cloning). func (rt *runtime) enterGlobalScope() { rt.enterScope(newScope(rt.globalStash, rt.globalStash, rt.globalObject)) } func (rt *runtime) enterFunctionScope(outer stasher, this Value) *fnStash { if outer == nil { outer = rt.globalStash } stash := rt.newFunctionStash(outer) var thisObject *object switch this.kind { case valueUndefined, valueNull: thisObject = rt.globalObject default: thisObject = rt.toObject(this) } rt.enterScope(newScope(stash, stash, thisObject)) return stash } func (rt *runtime) putValue(reference referencer, value Value) { name := reference.putValue(value) if name != "" { // Why? -- If reference.base == nil // strict = false rt.globalObject.defineProperty(name, value, 0o111, false) } } func (rt *runtime) tryCatchEvaluate(inner func() Value) (tryValue Value, isException bool) { //nolint: nonamedreturns // 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 excep, ok := caught.(*exception); ok { caught = excep.eject() } switch caught := caught.(type) { case ottoError: isException = true tryValue = objectValue(rt.newErrorObjectError(caught)) case Value: isException = true tryValue = caught default: isException = true tryValue = toValue(caught) } } }() return inner(), false } func (rt *runtime) toObject(value Value) *object { switch value.kind { case valueEmpty, valueUndefined, valueNull: panic(rt.panicTypeError("toObject unsupported kind %s", value.kind)) case valueBoolean: return rt.newBoolean(value) case valueString: return rt.newString(value) case valueNumber: return rt.newNumber(value) case valueObject: return value.object() default: panic(rt.panicTypeError("toObject unknown kind %s", value.kind)) } } func (rt *runtime) objectCoerce(value Value) (*object, error) { switch value.kind { case valueUndefined: return nil, errors.New("undefined") case valueNull: return nil, errors.New("null") case valueBoolean: return rt.newBoolean(value), nil case valueString: return rt.newString(value), nil case valueNumber: return rt.newNumber(value), nil case valueObject: return value.object(), nil default: panic(rt.panicTypeError("objectCoerce unknown kind %s", value.kind)) } } func checkObjectCoercible(rt *runtime, value Value) { isObject, mustCoerce := testObjectCoercible(value) if !isObject && !mustCoerce { panic(rt.panicTypeError("checkObjectCoercible not object or mustCoerce")) } } // testObjectCoercible. func testObjectCoercible(value Value) (isObject, mustCoerce bool) { //nolint: nonamedreturns switch value.kind { case valueReference, valueEmpty, valueNull, valueUndefined: return false, false case valueNumber, valueString, valueBoolean: return false, true case valueObject: return true, false default: panic(fmt.Sprintf("testObjectCoercible unknown kind %s", value.kind)) } } func (rt *runtime) safeToValue(value interface{}) (Value, error) { result := Value{} err := catchPanic(func() { result = rt.toValue(value) }) return result, err } // convertNumeric converts numeric parameter val from js to that of type t if it is safe to do so, otherwise it panics. // This allows literals (int64), bitwise values (int32) and the general form (float64) of javascript numerics to be passed as parameters to go functions easily. func (rt *runtime) convertNumeric(v Value, t reflect.Type) reflect.Value { val := reflect.ValueOf(v.export()) if val.Kind() == t.Kind() { return val } if val.Kind() == reflect.Interface { val = reflect.ValueOf(val.Interface()) } switch val.Kind() { case reflect.Float32, reflect.Float64: f64 := val.Float() switch t.Kind() { case reflect.Float64: return reflect.ValueOf(f64) case reflect.Float32: if reflect.Zero(t).OverflowFloat(f64) { panic(rt.panicRangeError("converting float64 to float32 would overflow")) } return val.Convert(t) case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64, reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64: i64 := int64(f64) if float64(i64) != f64 { panic(rt.panicRangeError(fmt.Sprintf("converting %v to %v would cause loss of precision", val.Type(), t))) } // The float represents an integer val = reflect.ValueOf(i64) default: panic(rt.panicTypeError(fmt.Sprintf("cannot convert %v to %v", val.Type(), t))) } } switch val.Kind() { case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: i64 := val.Int() switch t.Kind() { case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: if reflect.Zero(t).OverflowInt(i64) { panic(rt.panicRangeError(fmt.Sprintf("converting %v to %v would overflow", val.Type(), t))) } return val.Convert(t) case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64: if i64 < 0 { panic(rt.panicRangeError(fmt.Sprintf("converting %v to %v would underflow", val.Type(), t))) } if reflect.Zero(t).OverflowUint(uint64(i64)) { panic(rt.panicRangeError(fmt.Sprintf("converting %v to %v would overflow", val.Type(), t))) } return val.Convert(t) case reflect.Float32, reflect.Float64: return val.Convert(t) } case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64: u64 := val.Uint() switch t.Kind() { case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: if u64 > math.MaxInt64 || reflect.Zero(t).OverflowInt(int64(u64)) { panic(rt.panicRangeError(fmt.Sprintf("converting %v to %v would overflow", val.Type(), t))) } return val.Convert(t) case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64: if reflect.Zero(t).OverflowUint(u64) { panic(rt.panicRangeError(fmt.Sprintf("converting %v to %v would overflow", val.Type(), t))) } return val.Convert(t) case reflect.Float32, reflect.Float64: return val.Convert(t) } } panic(rt.panicTypeError(fmt.Sprintf("unsupported type %v -> %v for numeric conversion", val.Type(), t))) } func fieldIndexByName(t reflect.Type, name string) []int { for t.Kind() == reflect.Ptr { t = t.Elem() } for i := 0; i < t.NumField(); i++ { f := t.Field(i) if !validGoStructName(f.Name) { continue } if f.Anonymous { for t.Kind() == reflect.Ptr { t = t.Elem() } if f.Type.Kind() == reflect.Struct { if a := fieldIndexByName(f.Type, name); a != nil { return append([]int{i}, a...) } } } if a := strings.SplitN(f.Tag.Get("json"), ",", 2); a[0] != "" { if a[0] == "-" { continue } if a[0] == name { return []int{i} } } if f.Name == name { return []int{i} } } return nil } var ( typeOfValue = reflect.TypeOf(Value{}) typeOfJSONRawMessage = reflect.TypeOf(json.RawMessage{}) ) // convertCallParameter converts request val to type t if possible. // If the conversion fails due to overflow or type miss-match then it panics. // If no conversion is known then the original value is returned. func (rt *runtime) convertCallParameter(v Value, t reflect.Type) (reflect.Value, error) { if t == typeOfValue { return reflect.ValueOf(v), nil } if t == typeOfJSONRawMessage { if d, err := json.Marshal(v.export()); err == nil { return reflect.ValueOf(d), nil } } if v.kind == valueObject { if gso, ok := v.object().value.(*goStructObject); ok { if gso.value.Type().AssignableTo(t) { // please see TestDynamicFunctionReturningInterface for why this exists if t.Kind() == reflect.Interface && gso.value.Type().ConvertibleTo(t) { return gso.value.Convert(t), nil } return gso.value, nil } } if gao, ok := v.object().value.(*goArrayObject); ok { if gao.value.Type().AssignableTo(t) { // please see TestDynamicFunctionReturningInterface for why this exists if t.Kind() == reflect.Interface && gao.value.Type().ConvertibleTo(t) { return gao.value.Convert(t), nil } return gao.value, nil } } } tk := t.Kind() if tk == reflect.Interface { e := v.export() if e == nil { return reflect.Zero(t), nil } iv := reflect.ValueOf(e) if iv.Type().AssignableTo(t) { return iv, nil } } if tk == reflect.Ptr { switch v.kind { case valueEmpty, valueNull, valueUndefined: return reflect.Zero(t), nil default: var vv reflect.Value vv, err := rt.convertCallParameter(v, t.Elem()) if err != nil { return reflect.Zero(t), fmt.Errorf("can't convert to %s: %w", t, err) } if vv.CanAddr() { return vv.Addr(), nil } pv := reflect.New(vv.Type()) pv.Elem().Set(vv) return pv, nil } } switch tk { case reflect.Bool: return reflect.ValueOf(v.bool()), nil case reflect.String: switch v.kind { case valueString: return reflect.ValueOf(v.value), nil case valueNumber: return reflect.ValueOf(fmt.Sprintf("%v", v.value)), nil } case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64, reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Float32, reflect.Float64: if v.kind == valueNumber { return rt.convertNumeric(v, t), nil } case reflect.Slice: if o := v.object(); o != nil { if lv := o.get(propertyLength); lv.IsNumber() { l := lv.number().int64 s := reflect.MakeSlice(t, int(l), int(l)) tt := t.Elem() switch o.class { case classArrayName: for i := int64(0); i < l; i++ { p, ok := o.property[strconv.FormatInt(i, 10)] if !ok { continue } e, ok := p.value.(Value) if !ok { continue } ev, err := rt.convertCallParameter(e, tt) if err != nil { return reflect.Zero(t), fmt.Errorf("couldn't convert element %d of %s: %w", i, t, err) } s.Index(int(i)).Set(ev) } case classGoArrayName, classGoSliceName: var gslice bool switch o.value.(type) { case *goSliceObject: gslice = true case *goArrayObject: gslice = false } for i := int64(0); i < l; i++ { var p *property if gslice { p = goSliceGetOwnProperty(o, strconv.FormatInt(i, 10)) } else { p = goArrayGetOwnProperty(o, strconv.FormatInt(i, 10)) } if p == nil { continue } e, ok := p.value.(Value) if !ok { continue } ev, err := rt.convertCallParameter(e, tt) if err != nil { return reflect.Zero(t), fmt.Errorf("couldn't convert element %d of %s: %w", i, t, err) } s.Index(int(i)).Set(ev) } } return s, nil } } case reflect.Map: if o := v.object(); o != nil && t.Key().Kind() == reflect.String { m := reflect.MakeMap(t) var err error o.enumerate(false, func(k string) bool { v, verr := rt.convertCallParameter(o.get(k), t.Elem()) if verr != nil { err = fmt.Errorf("couldn't convert property %q of %s: %w", k, t, verr) return false } m.SetMapIndex(reflect.ValueOf(k), v) return true }) if err != nil { return reflect.Zero(t), err } return m, nil } case reflect.Func: if t.NumOut() > 1 { return reflect.Zero(t), fmt.Errorf("converting JavaScript values to Go functions with more than one return value is currently not supported") } if o := v.object(); o != nil && o.class == classFunctionName { return reflect.MakeFunc(t, func(args []reflect.Value) []reflect.Value { l := make([]interface{}, len(args)) for i, a := range args { if a.CanInterface() { l[i] = a.Interface() } } rv, err := v.Call(nullValue, l...) if err != nil { panic(err) } if t.NumOut() == 0 { return nil } r, err := rt.convertCallParameter(rv, t.Out(0)) if err != nil { panic(rt.panicTypeError("convertCallParameter Func: %s", err)) } return []reflect.Value{r} }), nil } case reflect.Struct: if o := v.object(); o != nil && o.class == classObjectName { s := reflect.New(t) for _, k := range o.propertyOrder { idx := fieldIndexByName(t, k) if idx == nil { return reflect.Zero(t), fmt.Errorf("can't convert property %q of %s: field does not exist", k, t) } ss := s for _, i := range idx { if ss.Kind() == reflect.Ptr { if ss.IsNil() { if !ss.CanSet() { return reflect.Zero(t), fmt.Errorf("can't convert property %q of %s: %s is unexported", k, t, ss.Type().Elem()) } ss.Set(reflect.New(ss.Type().Elem())) } ss = ss.Elem() } ss = ss.Field(i) } v, err := rt.convertCallParameter(o.get(k), ss.Type()) if err != nil { return reflect.Zero(t), fmt.Errorf("couldn't convert property %q of %s: %w", k, t, err) } ss.Set(v) } return s.Elem(), nil } } if tk == reflect.String { if o := v.object(); o != nil && o.hasProperty("toString") { if fn := o.get("toString"); fn.IsFunction() { sv, err := fn.Call(v) if err != nil { return reflect.Zero(t), fmt.Errorf("couldn't call toString: %w", err) } r, err := rt.convertCallParameter(sv, t) if err != nil { return reflect.Zero(t), fmt.Errorf("couldn't convert toString result: %w", err) } return r, nil } } return reflect.ValueOf(v.String()), nil } if v.kind == valueString { var s encoding.TextUnmarshaler if reflect.PtrTo(t).Implements(reflect.TypeOf(&s).Elem()) { r := reflect.New(t) if err := r.Interface().(encoding.TextUnmarshaler).UnmarshalText([]byte(v.string())); err != nil { return reflect.Zero(t), fmt.Errorf("can't convert to %s as TextUnmarshaller: %w", t.String(), err) } return r.Elem(), nil } } s := "OTTO DOES NOT UNDERSTAND THIS TYPE" switch v.kind { case valueBoolean: s = "boolean" case valueNull: s = "null" case valueNumber: s = "number" case valueString: s = "string" case valueUndefined: s = "undefined" case valueObject: s = v.Class() } return reflect.Zero(t), fmt.Errorf("can't convert from %q to %q", s, t) } func (rt *runtime) toValue(value interface{}) Value { rv, ok := value.(reflect.Value) if ok { value = rv.Interface() } switch value := value.(type) { case Value: return value case func(FunctionCall) Value: var name, file string var line int pc := reflect.ValueOf(value).Pointer() fn := goruntime.FuncForPC(pc) if fn != nil { name = fn.Name() file, line = fn.FileLine(pc) file = path.Base(file) } return objectValue(rt.newNativeFunction(name, file, line, value)) case nativeFunction: var name, file string var line int pc := reflect.ValueOf(value).Pointer() fn := goruntime.FuncForPC(pc) if fn != nil { name = fn.Name() file, line = fn.FileLine(pc) file = path.Base(file) } return objectValue(rt.newNativeFunction(name, file, line, value)) case Object, *Object, object, *object: // Nothing happens. // FIXME We should really figure out what can come here. // This catch-all is ugly. default: val := reflect.ValueOf(value) if ok && val.Kind() == rv.Kind() { // Use passed in rv which may be writable. val = rv } switch val.Kind() { case reflect.Ptr: switch reflect.Indirect(val).Kind() { case reflect.Struct: return objectValue(rt.newGoStructObject(val)) case reflect.Array: return objectValue(rt.newGoArray(val)) } case reflect.Struct: return objectValue(rt.newGoStructObject(val)) case reflect.Map: return objectValue(rt.newGoMapObject(val)) case reflect.Slice: return objectValue(rt.newGoSlice(val)) case reflect.Array: return objectValue(rt.newGoArray(val)) case reflect.Func: var name, file string var line int if v := reflect.ValueOf(val); v.Kind() == reflect.Ptr { pc := v.Pointer() fn := goruntime.FuncForPC(pc) if fn != nil { name = fn.Name() file, line = fn.FileLine(pc) file = path.Base(file) } } typ := val.Type() return objectValue(rt.newNativeFunction(name, file, line, func(c FunctionCall) Value { nargs := typ.NumIn() if len(c.ArgumentList) != nargs { if typ.IsVariadic() { if len(c.ArgumentList) < nargs-1 { panic(rt.panicRangeError(fmt.Sprintf("expected at least %d arguments; got %d", nargs-1, len(c.ArgumentList)))) } } else { panic(rt.panicRangeError(fmt.Sprintf("expected %d argument(s); got %d", nargs, len(c.ArgumentList)))) } } in := make([]reflect.Value, len(c.ArgumentList)) callSlice := false for i, a := range c.ArgumentList { var t reflect.Type n := i if n >= nargs-1 && typ.IsVariadic() { if n > nargs-1 { n = nargs - 1 } t = typ.In(n).Elem() } else { t = typ.In(n) } // if this is a variadic Go function, and the caller has supplied // exactly the number of JavaScript arguments required, and this // is the last JavaScript argument, try treating the it as the // actual set of variadic Go arguments. if that succeeds, break // out of the loop. if typ.IsVariadic() && len(c.ArgumentList) == nargs && i == nargs-1 { if v, err := rt.convertCallParameter(a, typ.In(n)); err == nil { in[i] = v callSlice = true break } } v, err := rt.convertCallParameter(a, t) if err != nil { panic(rt.panicTypeError(err.Error())) } in[i] = v } var out []reflect.Value if callSlice { out = val.CallSlice(in) } else { out = val.Call(in) } switch len(out) { case 0: return Value{} case 1: return rt.toValue(out[0].Interface()) default: s := make([]interface{}, len(out)) for i, v := range out { s[i] = rt.toValue(v.Interface()) } return rt.toValue(s) } })) } } return toValue(value) } func (rt *runtime) newGoSlice(value reflect.Value) *object { obj := rt.newGoSliceObject(value) obj.prototype = rt.global.ArrayPrototype return obj } func (rt *runtime) newGoArray(value reflect.Value) *object { obj := rt.newGoArrayObject(value) obj.prototype = rt.global.ArrayPrototype return obj } func (rt *runtime) parse(filename string, src, sm interface{}) (*ast.Program, error) { return parser.ParseFileWithSourceMap(nil, filename, src, sm, 0) } func (rt *runtime) cmplParse(filename string, src, sm interface{}) (*nodeProgram, error) { program, err := parser.ParseFileWithSourceMap(nil, filename, src, sm, 0) if err != nil { return nil, err } return cmplParse(program), nil } func (rt *runtime) parseSource(src, sm 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 := rt.parse("", src, sm) return nil, program, err } func (rt *runtime) cmplRunOrEval(src, sm interface{}, eval bool) (Value, error) { result := Value{} node, program, err := rt.parseSource(src, sm) if err != nil { return result, err } if node == nil { node = cmplParse(program) } err = catchPanic(func() { result = rt.cmplEvaluateNodeProgram(node, eval) }) switch result.kind { case valueEmpty: result = Value{} case valueReference: result = result.resolve() } return result, err } func (rt *runtime) cmplRun(src, sm interface{}) (Value, error) { return rt.cmplRunOrEval(src, sm, false) } func (rt *runtime) cmplEval(src, sm interface{}) (Value, error) { return rt.cmplRunOrEval(src, sm, true) } func (rt *runtime) parseThrow(err error) { if err == nil { return } var errl parser.ErrorList if errors.Is(err, &errl) { err := errl[0] if err.Message == "invalid left-hand side in assignment" { panic(rt.panicReferenceError(err.Message)) } panic(rt.panicSyntaxError(err.Message)) } panic(rt.panicSyntaxError(err.Error())) } func (rt *runtime) cmplParseOrThrow(src, sm interface{}) *nodeProgram { program, err := rt.cmplParse("", src, sm) rt.parseThrow(err) // Will panic/throw appropriately return program }