2016-09-15 06:12:25 +03:00

719 lines
18 KiB
Go

package otto
import (
"testing"
)
func TestArray(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
var abc = [ undefined, "Nothing happens." ];
abc.length;
`, 2)
test(`
abc = ""+[0, 1, 2, 3];
def = [].toString();
ghi = [null, 4, "null"].toString();
[ abc, def, ghi ];
`, "0,1,2,3,,,4,null")
test(`new Array(0).length`, 0)
test(`new Array(11).length`, 11)
test(`new Array(11, 1).length`, 2)
test(`
abc = [0, 1, 2, 3];
abc.xyzzy = "Nothing happens.";
delete abc[1];
var xyzzy = delete abc.xyzzy;
[ abc, xyzzy, abc.xyzzy ];
`, "0,,2,3,true,")
test(`
var abc = [0, 1, 2, 3, 4];
abc.length = 2;
abc;
`, "0,1")
test(`raise:
[].length = 3.14159;
`, "RangeError")
test(`raise:
new Array(3.14159);
`, "RangeError")
test(`
Object.defineProperty(Array.prototype, "0", {
value: 100,
writable: false,
configurable: true
});
abc = [101];
abc.hasOwnProperty("0") && abc[0] === 101;
`, true)
test(`
abc = [,,undefined];
[ abc.hasOwnProperty(0), abc.hasOwnProperty(1), abc.hasOwnProperty(2) ];
`, "false,false,true")
test(`
abc = Object.getOwnPropertyDescriptor(Array, "prototype");
[ [ typeof Array.prototype ],
[ abc.writable, abc.enumerable, abc.configurable ] ];
`, "object,false,false,false")
})
}
func TestArray_toString(t *testing.T) {
tt(t, func() {
{
test(`
Array.prototype.toString = function() {
return "Nothing happens.";
}
abc = Array.prototype.toString();
def = [].toString();
ghi = [null, 4, "null"].toString();
[ abc, def, ghi ].join(",");
`, "Nothing happens.,Nothing happens.,Nothing happens.")
}
{
test(`
Array.prototype.join = undefined
abc = Array.prototype.toString()
def = [].toString()
ghi = [null, 4, "null"].toString()
abc + "," + def + "," + ghi;
`, "[object Array],[object Array],[object Array]")
}
})
}
func TestArray_toLocaleString(t *testing.T) {
tt(t, func() {
test, _ := test()
defer mockUTC()()
test(`
[ 3.14159, "abc", undefined, new Date(0) ].toLocaleString();
`, "3.14159,abc,,1970-01-01 00:00:00")
test(`raise:
[ { toLocaleString: undefined } ].toLocaleString();
`, "TypeError")
})
}
func TestArray_concat(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
abc = [0, 1, 2];
def = [-1, -2, -3];
ghi = abc.concat(def);
jkl = abc.concat(def, 3, 4, 5);
mno = def.concat(-4, -5, abc);
[ ghi, jkl, mno ].join(";");
`, "0,1,2,-1,-2,-3;0,1,2,-1,-2,-3,3,4,5;-1,-2,-3,-4,-5,0,1,2")
test(`
var abc = [,1];
var def = abc.concat([], [,]);
def.getClass = Object.prototype.toString;
[ def.getClass(), typeof def[0], def[1], typeof def[2], def.length ];
`, "[object Array],undefined,1,undefined,3")
test(`
Object.defineProperty(Array.prototype, "0", {
value: 100,
writable: false,
configurable: true
});
var abc = Array.prototype.concat.call(101);
var hasProperty = abc.hasOwnProperty("0");
var instanceOfVerify = typeof abc[0] === "object";
var verifyValue = false;
verifyValue = abc[0] == 101;
var verifyEnumerable = false;
for (var property in abc) {
if (property === "0" && abc.hasOwnProperty("0")) {
verifyEnumerable = true;
}
}
var verifyWritable = false;
abc[0] = 12;
verifyWritable = abc[0] === 12;
var verifyConfigurable = false;
delete abc[0];
verifyConfigurable = abc.hasOwnProperty("0");
[ hasProperty, instanceOfVerify, verifyValue, !verifyConfigurable, verifyEnumerable, verifyWritable ];
`, "true,true,true,true,true,true")
})
}
func TestArray_splice(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
abc = [0, 1, 2];
def = abc.splice(1, 2, 3, 4, 5);
ghi = [].concat(abc);
jkl = ghi.splice(17, 21, 7, 8, 9);
[ abc, def, ghi, jkl ].join(";");
`, "0,3,4,5;1,2;0,3,4,5,7,8,9;")
})
}
func TestArray_shift(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
abc = [0, 1, 2];
def = abc.shift();
ghi = [].concat(abc);
jkl = abc.shift();
mno = [].concat(abc);
pqr = abc.shift();
stu = [].concat(abc);
vwx = abc.shift();
[ abc, def, ghi, jkl, mno, pqr, stu, vwx ].join(";");
`, ";0;1,2;1;2;2;;")
})
}
func TestArray_push(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
abc = [0];
def = abc.push(1);
ghi = [].concat(abc);
jkl = abc.push(2,3,4);
[ abc, def, ghi, jkl ].join(";");
`, "0,1,2,3,4;2;0,1;5")
})
}
func TestArray_pop(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
abc = [0,1];
def = abc.pop();
ghi = [].concat(abc);
jkl = abc.pop();
mno = [].concat(abc);
pqr = abc.pop();
[ abc, def, ghi, jkl, mno, pqr ].join(";");
`, ";1;0;0;;")
})
}
func TestArray_slice(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
abc = [0,1,2,3];
def = abc.slice();
ghi = abc.slice(1);
jkl = abc.slice(3,-1);
mno = abc.slice(2,-1);
pqr = abc.slice(-1, -10);
[ abc, def, ghi, jkl, mno, pqr ].join(";");
`, "0,1,2,3;0,1,2,3;1,2,3;;2;")
// Array.protoype.slice is generic
test(`
abc = { 0: 0, 1: 1, 2: 2, 3: 3 };
abc.length = 4;
def = Array.prototype.slice.call(abc);
ghi = Array.prototype.slice.call(abc,1);
jkl = Array.prototype.slice.call(abc,3,-1);
mno = Array.prototype.slice.call(abc,2,-1);
pqr = Array.prototype.slice.call(abc,-1,-10);
[ abc, def, ghi, jkl, pqr ].join(";");
`, "[object Object];0,1,2,3;1,2,3;;")
})
}
func TestArray_sliceArguments(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
(function(){
return Array.prototype.slice.call(arguments, 1)
})({}, 1, 2, 3);
`, "1,2,3")
})
}
func TestArray_unshift(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
abc = [];
def = abc.unshift(0);
ghi = [].concat(abc);
jkl = abc.unshift(1,2,3,4);
[ abc, def, ghi, jkl ].join(";");
`, "1,2,3,4,0;1;0;5")
})
}
func TestArray_reverse(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
abc = [0,1,2,3].reverse();
def = [0,1,2].reverse();
[ abc, def ];
`, "3,2,1,0,2,1,0")
})
}
func TestArray_sort(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
abc = [0,1,2,3].sort();
def = [3,2,1,0].sort();
ghi = [].sort();
jkl = [0].sort();
mno = [1,0].sort();
pqr = [1,5,-10, 100, 8, 72, 401, 0.05].sort();
stu = [1,5,-10, 100, 8, 72, 401, 0.05].sort(function(x, y){
return x == y ? 0 : x < y ? -1 : 1
});
vwx = [1,2,3,1,2,3].sort();
yza = [1,2,3,1,0,1,-1,0].sort();
[ abc, def, ghi, jkl, mno, pqr, stu, vwx, yza ].join(";");
`, "0,1,2,3;0,1,2,3;;0;0,1;-10,0.05,1,100,401,5,72,8;-10,0.05,1,5,8,72,100,401;1,1,2,2,3,3;-1,0,0,1,1,1,2,3")
test(`Array.prototype.sort.length`, 1)
})
}
func TestArray_isArray(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
[ Array.isArray.length, Array.isArray(), Array.isArray([]), Array.isArray({}) ];
`, "1,false,true,false")
test(`Array.isArray(Math)`, false)
})
}
func TestArray_indexOf(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`['a', 'b', 'c', 'b'].indexOf('b')`, 1)
test(`['a', 'b', 'c', 'b'].indexOf('b', 2)`, 3)
test(`['a', 'b', 'c', 'b'].indexOf('b', -2)`, 3)
test(`
Object.prototype.indexOf = Array.prototype.indexOf;
var abc = {0: 'a', 1: 'b', 2: 'c', length: 3};
abc.indexOf('c');
`, 2)
test(`[true].indexOf(true, "-Infinity")`, 0)
test(`
var target = {};
Math[3] = target;
Math.length = 5;
Array.prototype.indexOf.call(Math, target) === 3;
`, true)
test(`
var _NaN = NaN;
var abc = new Array("NaN", undefined, 0, false, null, {toString:function(){return NaN}}, "false", _NaN, NaN);
abc.indexOf(NaN);
`, -1)
test(`
var abc = {toString:function (){return 0}};
var def = 1;
var ghi = -(4/3);
var jkl = new Array(false, undefined, null, "0", abc, -1.3333333333333, "string", -0, true, +0, def, 1, 0, false, ghi, -(4/3));
[ jkl.indexOf(-(4/3)), jkl.indexOf(0), jkl.indexOf(-0), jkl.indexOf(1) ];
`, "14,7,7,10")
})
}
func TestArray_lastIndexOf(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`['a', 'b', 'c', 'b'].lastIndexOf('b')`, 3)
test(`['a', 'b', 'c', 'b'].lastIndexOf('b', 2)`, 1)
test(`['a', 'b', 'c', 'b'].lastIndexOf('b', -2)`, 1)
test(`
Object.prototype.lastIndexOf = Array.prototype.lastIndexOf;
var abc = {0: 'a', 1: 'b', 2: 'c', 3: 'b', length: 4};
abc.lastIndexOf('b');
`, 3)
test(`
var target = {};
Math[3] = target;
Math.length = 5;
[ Array.prototype.lastIndexOf.call(Math, target) === 3 ];
`, "true")
test(`
var _NaN = NaN;
var abc = new Array("NaN", undefined, 0, false, null, {toString:function(){return NaN}}, "false", _NaN, NaN);
abc.lastIndexOf(NaN);
`, -1)
test(`
var abc = {toString:function (){return 0}};
var def = 1;
var ghi = -(4/3);
var jkl = new Array(false, undefined, null, "0", abc, -1.3333333333333, "string", -0, true, +0, def, 1, 0, false, ghi, -(4/3));
[ jkl.lastIndexOf(-(4/3)), jkl.indexOf(0), jkl.indexOf(-0), jkl.indexOf(1) ];
`, "15,7,7,10")
})
}
func TestArray_every(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`raise: [].every()`, "TypeError")
test(`raise: [].every("abc")`, "TypeError")
test(`[].every(function() { return false })`, true)
test(`[1,2,3].every(function() { return false })`, false)
test(`[1,2,3].every(function() { return true })`, true)
test(`[1,2,3].every(function(_, index) { if (index === 1) return true })`, false)
test(`
var abc = function(value, index, object) {
return ('[object Math]' !== Object.prototype.toString.call(object));
};
Math.length = 1;
Math[0] = 1;
!Array.prototype.every.call(Math, abc);
`, true)
test(`
var def = false;
var abc = function(value, index, object) {
def = true;
return this === Math;
};
[11].every(abc, Math) && def;
`, true)
test(`
var def = false;
var abc = function(value, index, object) {
def = true;
return Math;
};
[11].every(abc) && def;
`, true)
})
}
func TestArray_some(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`raise: [].some("abc")`, "TypeError")
test(`[].some(function() { return true })`, false)
test(`[1,2,3].some(function() { return false })`, false)
test(`[1,2,3].some(function() { return true })`, true)
test(`[1,2,3].some(function(_, index) { if (index === 1) return true })`, true)
test(`
var abc = function(value, index, object) {
return ('[object Math]' !== Object.prototype.toString.call(object));
};
Math.length = 1;
Math[0] = 1;
!Array.prototype.some.call(Math, abc);
`, true)
test(`
var abc = function(value, index, object) {
return this === Math;
};
[11].some(abc, Math);
`, true)
test(`
var abc = function(value, index, object) {
return Math;
};
[11].some(abc);
`, true)
})
}
func TestArray_forEach(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`raise: [].forEach("abc")`, "TypeError")
test(`
var abc = 0;
[].forEach(function(value) {
abc += value;
});
abc;
`, 0)
test(`
abc = 0;
var def = [];
[1,2,3].forEach(function(value, index) {
abc += value;
def.push(index);
});
[ abc, def ];
`, "6,0,1,2")
test(`
var def = false;
var abc = function(value, index, object) {
def = ('[object Math]' === Object.prototype.toString.call(object));
};
Math.length = 1;
Math[0] = 1;
Array.prototype.forEach.call(Math, abc);
def;
`, true)
test(`
var def = false;
var abc = function(value, index, object) {
def = this === Math;
};
[11].forEach(abc, Math);
def;
`, true)
})
}
func TestArray_indexing(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
var abc = new Array(0, 1);
var def = abc.length;
abc[4294967296] = 10; // 2^32 => 0
abc[4294967297] = 11; // 2^32+1 => 1
[ def, abc.length, abc[0], abc[1], abc[4294967296] ];
`, "2,2,0,1,10")
test(`
abc = new Array(0, 1);
def = abc.length;
abc[4294967295] = 10;
var ghi = abc.length;
abc[4294967299] = 12;
var jkl = abc.length;
abc[4294967294] = 11;
[ def, ghi, jkl, abc.length, abc[4294967295], abc[4294967299] ];
`, "2,2,2,4294967295,10,12")
})
}
func TestArray_map(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`raise: [].map("abc")`, "TypeError")
test(`[].map(function() { return 1 }).length`, 0)
test(`[1,2,3].map(function(value) { return value * value })`, "1,4,9")
test(`[1,2,3].map(function(value) { return 1 })`, "1,1,1")
test(`
var abc = function(value, index, object) {
return ('[object Math]' === Object.prototype.toString.call(object));
};
Math.length = 1;
Math[0] = 1;
Array.prototype.map.call(Math, abc)[0];
`, true)
test(`
var abc = function(value, index, object) {
return this === Math;
};
[11].map(abc, Math)[0];
`, true)
})
}
func TestArray_filter(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`raise: [].filter("abc")`, "TypeError")
test(`[].filter(function() { return 1 }).length`, 0)
test(`[1,2,3].filter(function() { return false }).length`, 0)
test(`[1,2,3].filter(function() { return true })`, "1,2,3")
})
}
func TestArray_reduce(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`raise: [].reduce("abc")`, "TypeError")
test(`raise: [].reduce(function() {})`, "TypeError")
test(`[].reduce(function() {}, 0)`, 0)
test(`[].reduce(function() {}, undefined)`, "undefined")
test(`['a','b','c'].reduce(function(result, value) { return result+', '+value })`, "a, b, c")
test(`[1,2,3].reduce(function(result, value) { return result + value }, 4)`, 10)
test(`[1,2,3].reduce(function(result, value) { return result + value })`, 6)
})
}
func TestArray_reduceRight(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`raise: [].reduceRight("abc")`, "TypeError")
test(`raise: [].reduceRight(function() {})`, "TypeError")
test(`[].reduceRight(function() {}, 0)`, 0)
test(`[].reduceRight(function() {}, undefined)`, "undefined")
test(`['a','b','c'].reduceRight(function(result, value) { return result+', '+value })`, "c, b, a")
test(`[1,2,3].reduceRight(function(result, value) { return result + value }, 4)`, 10)
test(`[1,2,3].reduceRight(function(result, value) { return result + value })`, 6)
})
}
func TestArray_defineOwnProperty(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
var abc = [];
Object.defineProperty(abc, "length", {
writable: false
});
abc.length;
`, 0)
test(`raise:
var abc = [];
var exception;
Object.defineProperty(abc, "length", {
writable: false
});
Object.defineProperty(abc, "length", {
writable: true
});
`, "TypeError")
})
}
func TestArray_new(t *testing.T) {
tt(t, func() {
test, _ := test()
test(`
var abc = new Array(null);
var def = new Array(undefined);
[ abc.length, abc[0] === null, def.length, def[0] === undefined ]
`, "1,true,1,true")
test(`
var abc = new Array(new Number(0));
var def = new Array(new Number(4294967295));
[ abc.length, typeof abc[0], abc[0] == 0, def.length, typeof def[0], def[0] == 4294967295 ]
`, "1,object,true,1,object,true")
})
}