123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330 |
- using Jint.Native;
- namespace Jint.Tests.Runtime;
- public class ArrayTests
- {
- private readonly Engine _engine;
- public ArrayTests()
- {
- _engine = new Engine()
- .SetValue("log", new Action<object>(Console.WriteLine))
- .SetValue("assert", new Action<bool>(Assert.True))
- .SetValue("equal", new Action<object, object>(Assert.Equal));
- }
- [Fact]
- public void ArrayPrototypeToStringWithArray()
- {
- var result = _engine.Evaluate("Array.prototype.toString.call([1,2,3]);").AsString();
- Assert.Equal("1,2,3", result);
- }
- [Fact]
- public void ArrayPrototypeToStringWithNumber()
- {
- var result = _engine.Evaluate("Array.prototype.toString.call(1);").AsString();
- Assert.Equal("[object Number]", result);
- }
- [Fact]
- public void ArrayPrototypeToStringWithObject()
- {
- var result = _engine.Evaluate("Array.prototype.toString.call({});").AsString();
- Assert.Equal("[object Object]", result);
- }
- [Fact]
- public void EmptyStringKey()
- {
- var result = _engine.Evaluate("var x=[];x[\"\"]=8;x[\"\"];").AsNumber();
- Assert.Equal(8, result);
- }
- [Fact]
- public void LargeArraySize()
- {
- const string code = @"
- let arr = [];
- for (let i = 0; i < 10000; i++) arr.push(i);
- for (let i=0;i<10000;i++) arr.splice(0, 1);
- ";
- var engine = new Engine();
- engine.Execute(code);
- }
- [Fact]
- public void ArrayLengthFromInitialState()
- {
- var engine = new Engine();
- var array = new JsArray(engine);
- var length = (int) array.Length;
- Assert.Equal(0, length);
- }
- [Fact]
- public void ArraySortIsStable()
- {
- const string code = @"
- var items = [
- { name: 'Edward', value: 0 },
- { name: 'Sharpe', value: 0 },
- { name: 'And', value: 0 },
- { name: 'The', value: 1 },
- { name: 'Magnetic', value: 0 },
- { name: 'Zeros', value: 0 }
- ];
- // sort by value
- function compare(a, b) {
- return a.value - b.value;
- }
- var a = items.sort();
- assert(a[0].name == 'Edward');
- assert(a[1].name == 'Sharpe');
- assert(a[2].name == 'And');
- assert(a[3].name == 'The');
- assert(a[4].name == 'Magnetic');
- assert(a[5].name == 'Zeros');
- var a = items.sort(compare);
- assert(a[0].name == 'Edward');
- assert(a[1].name == 'Sharpe');
- assert(a[2].name == 'And');
- assert(a[3].name == 'Magnetic');
- assert(a[4].name == 'Zeros');
- assert(a[5].name == 'The');
- ";
- _engine.Execute(code);
- }
- #if !NETCOREAPP
- // this test case only triggers on older full framework where the is no checks for infinite comparisons
- [Fact]
- public void ArraySortShouldObeyExecutionConstraints()
- {
- const string script = @"
- let cases = [5,5];
- let latestCase = cases.sort((c1, c2) => c1 > c2 ? -1: 1);";
- var engine = new Engine(options => options
- .TimeoutInterval(TimeSpan.FromSeconds(1))
- );
- Assert.Throws<TimeoutException>(() => engine.Evaluate(script));
- }
- #endif
- [Fact]
- public void ExtendingArrayAndInstanceOf()
- {
- const string script = @"
- class MyArr extends Array {
- constructor(...args) {
- super(...args);
- }
- }";
- _engine.Execute(script);
- _engine.Evaluate("const a = new MyArr(1,2);");
- Assert.True(_engine.Evaluate("a instanceof MyArr").AsBoolean());
- }
- [Fact]
- public void IteratorShouldBeConvertibleToArray()
- {
- Assert.Equal("hello;again", _engine.Evaluate("Array.from(['hello', 'again'].values()).join(';')"));
- Assert.Equal("hello;another", _engine.Evaluate("Array.from(new Map([['hello', 'world'], ['another', 'value']]).keys()).join(';')"));
- }
- [Fact]
- public void ArrayFromShouldNotFlattenInputArray()
- {
- Assert.Equal("a;b", _engine.Evaluate("[...['a', 'b']].join(';')"));
- Assert.Equal("0,a;1,b", _engine.Evaluate("[...['a', 'b'].entries()].join(';')"));
- Assert.Equal("0,c;1,d", _engine.Evaluate("Array.from(['c', 'd'].entries()).join(';')"));
- Assert.Equal("0,e;1,f", _engine.Evaluate("Array.from([[0, 'e'],[1, 'f']]).join(';')"));
- }
- [Fact]
- public void ArrayEntriesShouldReturnKeyValuePairs()
- {
- Assert.Equal("0,hello,1,world", _engine.Evaluate("Array.from(['hello', 'world'].entries()).join()"));
- Assert.Equal("0,hello;1,world", _engine.Evaluate("Array.from(['hello', 'world'].entries()).join(';')"));
- Assert.Equal("0,;1,1;2,5", _engine.Evaluate("Array.from([,1,5,].entries()).join(';')"));
- }
- [Fact]
- public void IteratorsShouldHaveIteratorSymbol()
- {
- _engine.Execute("assert(!!['hello'].values()[Symbol.iterator])");
- _engine.Execute("assert(!!new Map([['hello', 'world']]).keys()[Symbol.iterator])");
- }
- [Fact]
- public void ArraySortDoesNotCrashInDebugMode()
- {
- var engine = new Engine(o =>
- {
- o.DebugMode(true);
- });
- engine.SetValue("equal", new Action<object, object>(Assert.Equal));
- const string code = @"
- var items = [5,2,4,1];
- items.sort((a,b) => a - b);
- equal('1,2,4,5', items.join());
- ";
- engine.Execute(code);
- }
- [Fact]
- public void ArrayConstructorFromHoles()
- {
- _engine.Evaluate("var a = Array(...[,,]);");
- Assert.True(_engine.Evaluate("\"0\" in a").AsBoolean());
- Assert.True(_engine.Evaluate("\"1\" in a").AsBoolean());
- Assert.Equal("undefinedundefined", _engine.Evaluate("'' + a[0] + a[1]"));
- }
- [Fact]
- public void ArrayIsSubclassable()
- {
- _engine.Evaluate("class C extends Array {}");
- _engine.Evaluate("var c = new C();");
- Assert.True(_engine.Evaluate("c.map(Boolean) instanceof C").AsBoolean());
- }
- [Fact]
- public void HasProperIteratorPrototypeChain()
- {
- const string Script = @"
- // Iterator instance
- var iterator = [][Symbol.iterator]();
- // %ArrayIteratorPrototype%
- var proto1 = Object.getPrototypeOf(iterator);
- // %IteratorPrototype%
- var proto2 = Object.getPrototypeOf(proto1);";
- var engine = new Engine();
- engine.Execute(Script);
- Assert.True(engine.Evaluate("proto2.hasOwnProperty(Symbol.iterator)").AsBoolean());
- Assert.True(engine.Evaluate("!proto1.hasOwnProperty(Symbol.iterator)").AsBoolean());
- Assert.True(engine.Evaluate("!iterator.hasOwnProperty(Symbol.iterator)").AsBoolean());
- Assert.True(engine.Evaluate("iterator[Symbol.iterator]() === iterator").AsBoolean());
- }
- [Fact]
- public void ArrayFrom()
- {
- const string Script = @"
- // Array.from -> Get -> [[Get]]
- var get = [];
- var p = new Proxy({length: 2, 0: '', 1: ''}, { get: function(o, k) { get.push(k); return o[k]; }});
- Array.from(p);";
- var engine = new Engine();
- engine.Execute(Script);
- Assert.True(engine.Evaluate("get[0] === Symbol.iterator").AsBoolean());
- Assert.Equal("length,0,1", engine.Evaluate("get.slice(1) + ''").AsString());
- }
- [Fact]
- public void ArrayFromStringUsingMapping()
- {
- var engine = new Engine();
- var array = engine.Evaluate("Array.from('fff', (s) => Number.parseInt(s, 16))").AsArray();
- Assert.Equal((uint) 3, array.Length);
- Assert.Equal((uint) 15, array[0]);
- Assert.Equal((uint) 15, array[1]);
- Assert.Equal((uint) 15, array[2]);
- }
- [Fact]
- public void Iteration()
- {
- const string Script = @"
- // Array.prototype methods -> Get -> [[Get]]
- var methods = ['copyWithin', 'every', 'fill', 'filter', 'find', 'findIndex', 'forEach',
- 'indexOf', 'join', 'lastIndexOf', 'map', 'reduce', 'reduceRight', 'some'];
- var get;
- var p = new Proxy({length: 2, 0: '', 1: ''}, { get: function(o, k) { get.push(k); return o[k]; }});
- for(var i = 0; i < methods.length; i+=1) {
- get = [];
- Array.prototype[methods[i]].call(p, Function());
- var actual = get + '';
- var expected = (
- methods[i] === 'fill' ? ""length"" :
- methods[i] === 'every' ? ""length,0"" :
- methods[i] === 'lastIndexOf' || methods[i] === 'reduceRight' ? ""length,1,0"" :
- ""length,0,1"");
- if (actual !== expected) {
- throw methods[i] + ': ' + actual + ' !== ' + expected;
- }
- }
- return true;";
- var engine = new Engine();
- Assert.True(engine.Evaluate(Script).AsBoolean());
- }
- [Fact]
- public void Concat()
- {
- const string Script = @"
- // Array.prototype.concat -> Get -> [[Get]]
- var get = [];
- var arr = [1];
- arr.constructor = void undefined;
- var p = new Proxy(arr, { get: function(o, k) { get.push(k); return o[k]; }});
- Array.prototype.concat.call(p,p);";
- var engine = new Engine();
- engine.Execute(Script);
- Assert.Equal("constructor", engine.Evaluate("get[0]"));
- Assert.True(engine.Evaluate("get[1] === Symbol.isConcatSpreadable").AsBoolean());
- Assert.Equal("length", engine.Evaluate("get[2]"));
- Assert.Equal("0", engine.Evaluate("get[3]"));
- Assert.True(engine.Evaluate("get[4] === get[1] && get[5] === get[2] && get[6] === get[3]").AsBoolean());
- Assert.Equal(7, engine.Evaluate("get.length"));
- }
- [Fact]
- public void Shift()
- {
- const string Script = @"
- // Array.prototype.shift -> Get -> [[Get]]
- var get = [];
- var p = new Proxy([0,1,2,3], { get: function(o, k) { get.push(k); return o[k]; }});
- Array.prototype.shift.call(p);
- return get + '' === ""length,0,1,2,3"";";
- var engine = new Engine();
- Assert.True(engine.Evaluate(Script).AsBoolean());
- }
- [Fact]
- public void ShouldBeAbleToInitFromArray()
- {
- var engine = new Engine();
- var propertyDescriptors = new JsArray(engine, new JsValue[] { 1 }).GetOwnProperties().ToArray();
- Assert.Equal(2, propertyDescriptors.Length);
- Assert.Equal("0", propertyDescriptors[0].Key);
- Assert.Equal(1, propertyDescriptors[0].Value.Value);
- Assert.Equal("length", propertyDescriptors[1].Key);
- Assert.Equal(1, propertyDescriptors[1].Value.Value);
- }
- }
|