356 lines
11 KiB
Go
Raw Normal View History

package otto
import (
"fmt"
"math"
"strings"
"testing"
)
func TestGlobal(t *testing.T) {
tt(t, func() {
test, vm := test()
runtime := vm.vm.runtime
{
call := func(object interface{}, src string, argumentList ...interface{}) Value {
var tgt *Object
switch object := object.(type) {
case Value:
tgt = object.Object()
case *Object:
tgt = object
case *_object:
tgt = toValue_object(object).Object()
default:
panic("Here be dragons.")
}
value, err := tgt.Call(src, argumentList...)
is(err, nil)
return value
}
// FIXME enterGlobalScope
if false {
value := runtime.scope.lexical.getBinding("Object", false)._object().call(UndefinedValue(), []Value{toValue(runtime.newObject())}, false, nativeFrame)
is(value.IsObject(), true)
is(value, "[object Object]")
is(value._object().prototype == runtime.global.ObjectPrototype, true)
is(value._object().prototype == runtime.global.Object.get("prototype")._object(), true)
is(value._object().get("toString"), "function toString() { [native code] }")
is(call(value.Object(), "hasOwnProperty", "hasOwnProperty"), false)
is(call(value._object().get("toString")._object().prototype, "toString"), "function () { [native code] }") // TODO Is this right?
is(value._object().get("toString")._object().get("toString"), "function toString() { [native code] }")
is(value._object().get("toString")._object().get("toString")._object(), "function toString() { [native code] }")
is(call(value._object(), "propertyIsEnumerable", "isPrototypeOf"), false)
value._object().put("xyzzy", toValue_string("Nothing happens."), false)
is(call(value, "propertyIsEnumerable", "isPrototypeOf"), false)
is(call(value, "propertyIsEnumerable", "xyzzy"), true)
is(value._object().get("xyzzy"), "Nothing happens.")
is(call(runtime.scope.lexical.getBinding("Object", false), "isPrototypeOf", value), false)
is(call(runtime.scope.lexical.getBinding("Object", false)._object().get("prototype"), "isPrototypeOf", value), true)
is(call(runtime.scope.lexical.getBinding("Function", false), "isPrototypeOf", value), false)
is(runtime.newObject().prototype == runtime.global.Object.get("prototype")._object(), true)
abc := runtime.newBoolean(toValue_bool(true))
is(toValue_object(abc), "true") // TODO Call primitive?
//def := runtime.localGet("Boolean")._object().Construct(UndefinedValue(), []Value{})
//is(def, "false") // TODO Call primitive?
}
}
test(`new Number().constructor == Number`, true)
test(`this.hasOwnProperty`, "function hasOwnProperty() { [native code] }")
test(`eval.length === 1`, true)
test(`eval.prototype === undefined`, true)
test(`raise: new eval()`, "TypeError: function eval() { [native code] } is not a constructor")
test(`
[
[ delete undefined, undefined ],
[ delete NaN, NaN ],
[ delete Infinity, Infinity ],
];
`, "false,,false,NaN,false,Infinity")
test(`
Object.getOwnPropertyNames(Function('return this')()).sort();
`, "Array,Boolean,Date,Error,EvalError,Function,Infinity,JSON,Math,NaN,Number,Object,RangeError,ReferenceError,RegExp,String,SyntaxError,TypeError,URIError,console,decodeURI,decodeURIComponent,encodeURI,encodeURIComponent,escape,eval,isFinite,isNaN,parseFloat,parseInt,undefined,unescape")
// __defineGetter__,__defineSetter__,__lookupGetter__,__lookupSetter__,constructor,hasOwnProperty,isPrototypeOf,propertyIsEnumerable,toLocaleString,toString,valueOf
test(`
Object.getOwnPropertyNames(Object.prototype).sort();
`, "constructor,hasOwnProperty,isPrototypeOf,propertyIsEnumerable,toLocaleString,toString,valueOf")
// arguments,caller,length,name,prototype
test(`
Object.getOwnPropertyNames(EvalError).sort();
`, "length,prototype")
test(`
var abc = [];
var def = [EvalError, RangeError, ReferenceError, SyntaxError, TypeError, URIError];
for (constructor in def) {
abc.push(def[constructor] === def[constructor].prototype.constructor);
}
def = [Array, Boolean, Date, Function, Number, Object, RegExp, String, SyntaxError];
for (constructor in def) {
abc.push(def[constructor] === def[constructor].prototype.constructor);
}
abc;
`, "true,true,true,true,true,true,true,true,true,true,true,true,true,true,true")
test(`
[ Array.prototype.constructor === Array, Array.constructor === Function ];
`, "true,true")
test(`
[ Number.prototype.constructor === Number, Number.constructor === Function ];
`, "true,true")
test(`
[ Function.prototype.constructor === Function, Function.constructor === Function ];
`, "true,true")
})
}
func TestGlobalLength(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
[ Object.length, Function.length, RegExp.length, Math.length ];
`, "1,1,2,")
})
}
func TestGlobalError(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
[ TypeError.length, TypeError(), TypeError("Nothing happens.") ];
`, "1,TypeError,TypeError: Nothing happens.")
test(`
[ URIError.length, URIError(), URIError("Nothing happens.") ];
`, "1,URIError,URIError: Nothing happens.")
})
}
func TestGlobalReadOnly(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`Number.POSITIVE_INFINITY`, math.Inf(1))
test(`
Number.POSITIVE_INFINITY = 1;
`, 1)
test(`Number.POSITIVE_INFINITY`, math.Inf(1))
test(`
Number.POSITIVE_INFINITY = 1;
Number.POSITIVE_INFINITY;
`, math.Inf(1))
})
}
func Test_isNaN(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`isNaN(0)`, false)
test(`isNaN("Xyzzy")`, true)
test(`isNaN()`, true)
test(`isNaN(NaN)`, true)
test(`isNaN(Infinity)`, false)
test(`isNaN.length === 1`, true)
test(`isNaN.prototype === undefined`, true)
})
}
func Test_isFinite(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`isFinite(0)`, true)
test(`isFinite("Xyzzy")`, false)
test(`isFinite()`, false)
test(`isFinite(NaN)`, false)
test(`isFinite(Infinity)`, false)
test(`isFinite(new Number(451));`, true)
test(`isFinite.length === 1`, true)
test(`isFinite.prototype === undefined`, true)
})
}
func Test_parseInt(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`parseInt("0")`, 0)
test(`parseInt("11")`, 11)
test(`parseInt(" 11")`, 11)
test(`parseInt("11 ")`, 11)
test(`parseInt(" 11 ")`, 11)
test(`parseInt(" 11\n")`, 11)
test(`parseInt(" 11\n", 16)`, 17)
test(`parseInt("Xyzzy")`, _NaN)
test(`parseInt(" 0x11\n", 16)`, 17)
test(`parseInt("0x0aXyzzy", 16)`, 10)
test(`parseInt("0x1", 0)`, 1)
test(`parseInt("0x10000000000000000000", 16)`, float64(75557863725914323419136))
test(`parseInt.length === 2`, true)
test(`parseInt.prototype === undefined`, true)
})
}
func Test_parseFloat(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`parseFloat("0")`, 0)
test(`parseFloat("11")`, 11)
test(`parseFloat(" 11")`, 11)
test(`parseFloat("11 ")`, 11)
test(`parseFloat(" 11 ")`, 11)
test(`parseFloat(" 11\n")`, 11)
test(`parseFloat(" 11\n", 16)`, 11)
test(`parseFloat("11.1")`, 11.1)
test(`parseFloat("Xyzzy")`, _NaN)
test(`parseFloat(" 0x11\n", 16)`, 0)
test(`parseFloat("0x0a")`, 0)
test(`parseFloat("0x0aXyzzy")`, 0)
test(`parseFloat("Infinity")`, _Infinity)
test(`parseFloat("infinity")`, _NaN)
test(`parseFloat("0x")`, 0)
test(`parseFloat("11x")`, 11)
test(`parseFloat("Infinity1")`, _Infinity)
test(`parseFloat.length === 1`, true)
test(`parseFloat.prototype === undefined`, true)
})
}
func Test_encodeURI(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`encodeURI("http://example.com/ Nothing happens.")`, "http://example.com/%20Nothing%20happens.")
test(`encodeURI("http://example.com/ _^#")`, "http://example.com/%20_%5E#")
test(`encodeURI(String.fromCharCode("0xE000"))`, "%EE%80%80")
test(`encodeURI(String.fromCharCode("0xFFFD"))`, "%EF%BF%BD")
test(`raise: encodeURI(String.fromCharCode("0xDC00"))`, "URIError: URI malformed")
test(`encodeURI.length === 1`, true)
test(`encodeURI.prototype === undefined`, true)
})
}
func Test_encodeURIComponent(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`encodeURIComponent("http://example.com/ Nothing happens.")`, "http%3A%2F%2Fexample.com%2F%20Nothing%20happens.")
test(`encodeURIComponent("http://example.com/ _^#")`, "http%3A%2F%2Fexample.com%2F%20_%5E%23")
})
}
func Test_decodeURI(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`decodeURI(encodeURI("http://example.com/ Nothing happens."))`, "http://example.com/ Nothing happens.")
test(`decodeURI(encodeURI("http://example.com/ _^#"))`, "http://example.com/ _^#")
test(`raise: decodeURI("http://example.com/ _^#%")`, "URIError: URI malformed")
test(`raise: decodeURI("%DF%7F")`, "URIError: URI malformed")
for _, check := range strings.Fields("+ %3B %2F %3F %3A %40 %26 %3D %2B %24 %2C %23") {
test(fmt.Sprintf(`decodeURI("%s")`, check), check)
}
test(`decodeURI.length === 1`, true)
test(`decodeURI.prototype === undefined`, true)
})
}
func Test_decodeURIComponent(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`decodeURIComponent(encodeURI("http://example.com/ Nothing happens."))`, "http://example.com/ Nothing happens.")
test(`decodeURIComponent(encodeURI("http://example.com/ _^#"))`, "http://example.com/ _^#")
test(`decodeURIComponent.length === 1`, true)
test(`decodeURIComponent.prototype === undefined`, true)
test(`
var global = Function('return this')();
var abc = Object.getOwnPropertyDescriptor(global, "decodeURIComponent");
[ abc.value === global.decodeURIComponent, abc.writable, abc.enumerable, abc.configurable ];
`, "true,true,false,true")
})
}
func TestGlobal_skipEnumeration(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
var found = [];
for (var test in this) {
if (false ||
test === 'NaN' ||
test === 'undefined' ||
test === 'Infinity' ||
false) {
found.push(test)
}
}
found.length;
`, 0)
test(`
var found = [];
for (var test in this) {
if (false ||
test === 'Object' ||
test === 'Function' ||
test === 'String' ||
test === 'Number' ||
test === 'Array' ||
test === 'Boolean' ||
test === 'Date' ||
test === 'RegExp' ||
test === 'Error' ||
test === 'EvalError' ||
test === 'RangeError' ||
test === 'ReferenceError' ||
test === 'SyntaxError' ||
test === 'TypeError' ||
test === 'URIError' ||
false) {
found.push(test)
}
}
found.length;
`, 0)
})
}