mirror of https://github.com/tc39/test262.git
Object.fromEntries: add basic tests
This commit is contained in:
parent
8ae2a51e29
commit
7f694d4718
|
@ -7,6 +7,10 @@
|
|||
#
|
||||
# https://github.com/tc39/process-document
|
||||
|
||||
# Object.fromEntries
|
||||
# https://github.com/tc39/proposal-object-from-entries
|
||||
Object.fromEntries
|
||||
|
||||
# BigInt
|
||||
# https://github.com/tc39/proposal-bigint
|
||||
BigInt
|
||||
|
|
|
@ -0,0 +1,74 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Evaluation order is iterator.next(), get '0', get '1', toPropertyKey, repeat.
|
||||
esid: sec-object.fromentries
|
||||
includes: [compareArray.js]
|
||||
features: [Symbol.iterator, Object.fromEntries]
|
||||
---*/
|
||||
|
||||
var effects = [];
|
||||
|
||||
function makeEntry(label) {
|
||||
return {
|
||||
get '0'() {
|
||||
effects.push('access property "0" of ' + label + ' entry');
|
||||
return {
|
||||
toString: function() {
|
||||
effects.push('toString of ' + label + ' key');
|
||||
return label + ' key';
|
||||
},
|
||||
};
|
||||
},
|
||||
get '1'() {
|
||||
effects.push('access property "1" of ' + label + ' entry');
|
||||
return label + ' value';
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
var iterable = {
|
||||
[Symbol.iterator]: function() {
|
||||
effects.push('get Symbol.iterator');
|
||||
var count = 0;
|
||||
return {
|
||||
next: function() {
|
||||
effects.push('next ' + count);
|
||||
if (count === 0) {
|
||||
++count;
|
||||
return {
|
||||
done: false,
|
||||
value: makeEntry('first', 'first key', 'first value'),
|
||||
};
|
||||
} else if (count === 1) {
|
||||
++count;
|
||||
return {
|
||||
done: false,
|
||||
value: makeEntry('second', 'second key', 'second value'),
|
||||
};
|
||||
} else {
|
||||
return {
|
||||
done: true,
|
||||
};
|
||||
}
|
||||
},
|
||||
};
|
||||
},
|
||||
};
|
||||
|
||||
var result = Object.fromEntries(iterable);
|
||||
assert.compareArray(effects, [
|
||||
'get Symbol.iterator',
|
||||
'next 0',
|
||||
'access property "0" of first entry',
|
||||
'access property "1" of first entry',
|
||||
'toString of first key',
|
||||
'next 1',
|
||||
'access property "0" of second entry',
|
||||
'access property "1" of second entry',
|
||||
'toString of second key',
|
||||
'next 2',
|
||||
], 'Object.fromEntries evaluation order');
|
||||
assert.sameValue(result['first key'], 'first value');
|
||||
assert.sameValue(result['second key'], 'second value');
|
|
@ -0,0 +1,39 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Closes iterators when they return entries which are null.
|
||||
esid: sec-object.fromentries
|
||||
features: [Symbol.iterator, Object.fromEntries]
|
||||
---*/
|
||||
|
||||
var returned = false;
|
||||
var iterable = {
|
||||
[Symbol.iterator]: function() {
|
||||
var advanced = false;
|
||||
return {
|
||||
next: function() {
|
||||
if (advanced) {
|
||||
throw new Test262Error('should only advance once');
|
||||
}
|
||||
advanced = true;
|
||||
return {
|
||||
done: false,
|
||||
value: 'null',
|
||||
};
|
||||
},
|
||||
return: function() {
|
||||
if (returned) {
|
||||
throw new Test262Error('should only return once');
|
||||
}
|
||||
returned = true;
|
||||
},
|
||||
};
|
||||
},
|
||||
};
|
||||
|
||||
assert.throws(TypeError, function() {
|
||||
Object.fromEntries(iterable);
|
||||
});
|
||||
|
||||
assert(returned, 'iterator should be closed when entry is null');
|
|
@ -0,0 +1,39 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Closes iterators when they return entries which are strings.
|
||||
esid: sec-object.fromentries
|
||||
features: [Symbol.iterator, Object.fromEntries]
|
||||
---*/
|
||||
|
||||
var returned = false;
|
||||
var iterable = {
|
||||
[Symbol.iterator]: function() {
|
||||
var advanced = false;
|
||||
return {
|
||||
next: function() {
|
||||
if (advanced) {
|
||||
throw new Test262Error('should only advance once');
|
||||
}
|
||||
advanced = true;
|
||||
return {
|
||||
done: false,
|
||||
value: 'ab',
|
||||
};
|
||||
},
|
||||
return: function() {
|
||||
if (returned) {
|
||||
throw new Test262Error('should only return once');
|
||||
}
|
||||
returned = true;
|
||||
},
|
||||
};
|
||||
},
|
||||
};
|
||||
|
||||
assert.throws(TypeError, function() {
|
||||
Object.fromEntries(iterable);
|
||||
});
|
||||
|
||||
assert(returned, 'iterator should be closed when entry is a string');
|
|
@ -0,0 +1,45 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Closes iterators when accessing an entry's properties throws.
|
||||
esid: sec-object.fromentries
|
||||
features: [Symbol.iterator, Object.fromEntries]
|
||||
---*/
|
||||
|
||||
function DummyError() {}
|
||||
|
||||
var returned = false;
|
||||
var iterable = {
|
||||
[Symbol.iterator]: function() {
|
||||
var advanced = false;
|
||||
return {
|
||||
next: function() {
|
||||
if (advanced) {
|
||||
throw new Test262Error('should only advance once');
|
||||
}
|
||||
advanced = true;
|
||||
return {
|
||||
done: false,
|
||||
value: {
|
||||
get '0'() {
|
||||
throw new DummyError();
|
||||
},
|
||||
},
|
||||
};
|
||||
},
|
||||
return: function() {
|
||||
if (returned) {
|
||||
throw new Test262Error('should only return once');
|
||||
}
|
||||
returned = true;
|
||||
},
|
||||
};
|
||||
},
|
||||
};
|
||||
|
||||
assert.throws(DummyError, function() {
|
||||
Object.fromEntries(iterable);
|
||||
});
|
||||
|
||||
assert(returned, 'iterator should be closed when entry property access throws');
|
|
@ -0,0 +1,47 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Closes iterators when toString on a key throws.
|
||||
esid: sec-object.fromentries
|
||||
features: [Symbol.iterator, Object.fromEntries]
|
||||
---*/
|
||||
|
||||
function DummyError() {}
|
||||
|
||||
var returned = false;
|
||||
var iterable = {
|
||||
[Symbol.iterator]: function() {
|
||||
var advanced = false;
|
||||
return {
|
||||
next: function() {
|
||||
if (advanced) {
|
||||
throw new Test262Error('should only advance once');
|
||||
}
|
||||
advanced = true;
|
||||
return {
|
||||
done: false,
|
||||
value: {
|
||||
0: {
|
||||
toString: function() {
|
||||
throw new DummyError();
|
||||
},
|
||||
},
|
||||
},
|
||||
};
|
||||
},
|
||||
return: function() {
|
||||
if (returned) {
|
||||
throw new Test262Error('should only return once');
|
||||
}
|
||||
returned = true;
|
||||
},
|
||||
};
|
||||
},
|
||||
};
|
||||
|
||||
assert.throws(DummyError, function() {
|
||||
Object.fromEntries(iterable);
|
||||
});
|
||||
|
||||
assert(returned, 'iterator should be closed when key toString throws');
|
|
@ -0,0 +1,25 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Does not close iterators with a `next` method which returns a non-object.
|
||||
esid: sec-object.fromentries
|
||||
features: [Symbol.iterator, Object.fromEntries]
|
||||
---*/
|
||||
|
||||
var iterable = {
|
||||
[Symbol.iterator]: function() {
|
||||
return {
|
||||
next: function() {
|
||||
return null;
|
||||
},
|
||||
return: function() {
|
||||
throw new Test262Error('should not call return');
|
||||
},
|
||||
};
|
||||
},
|
||||
};
|
||||
|
||||
assert.throws(TypeError, function() {
|
||||
Object.fromEntries(iterable);
|
||||
});
|
|
@ -0,0 +1,32 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Does not close iterators with a `done` accessor which throws.
|
||||
esid: sec-object.fromentries
|
||||
features: [Symbol.iterator, Object.fromEntries]
|
||||
---*/
|
||||
|
||||
function DummyError() {}
|
||||
|
||||
var returned = false;
|
||||
var iterable = {
|
||||
[Symbol.iterator]: function() {
|
||||
return {
|
||||
next: function() {
|
||||
return {
|
||||
get done() {
|
||||
throw new DummyError();
|
||||
},
|
||||
};
|
||||
},
|
||||
return: function() {
|
||||
throw new Test262Error('should not call return');
|
||||
},
|
||||
};
|
||||
},
|
||||
};
|
||||
|
||||
assert.throws(DummyError, function() {
|
||||
Object.fromEntries(iterable);
|
||||
});
|
|
@ -0,0 +1,27 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Does not close iterators with a `next` method which throws.
|
||||
esid: sec-object.fromentries
|
||||
features: [Symbol.iterator, Object.fromEntries]
|
||||
---*/
|
||||
|
||||
function DummyError() {}
|
||||
|
||||
var iterable = {
|
||||
[Symbol.iterator]: function() {
|
||||
return {
|
||||
next: function() {
|
||||
throw new DummyError();
|
||||
},
|
||||
return: function() {
|
||||
throw new Test262Error('should not call return');
|
||||
},
|
||||
};
|
||||
},
|
||||
};
|
||||
|
||||
assert.throws(DummyError, function() {
|
||||
Object.fromEntries(iterable);
|
||||
});
|
|
@ -0,0 +1,23 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Does not close iterators with an uncallable `next` property.
|
||||
esid: sec-object.fromentries
|
||||
features: [Symbol.iterator, Object.fromEntries]
|
||||
---*/
|
||||
|
||||
var iterable = {
|
||||
[Symbol.iterator]: function() {
|
||||
return {
|
||||
next: null,
|
||||
return: function() {
|
||||
throw new Test262Error('should not call return');
|
||||
},
|
||||
};
|
||||
},
|
||||
};
|
||||
|
||||
assert.throws(TypeError, function() {
|
||||
Object.fromEntries(iterable);
|
||||
});
|
|
@ -0,0 +1,22 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Key enumeration order of result objects matches the order of entries in the iterable.
|
||||
esid: sec-object.fromentries
|
||||
includes: [compareArray.js]
|
||||
features: [Object.fromEntries]
|
||||
---*/
|
||||
|
||||
var entries = [
|
||||
['z', 1],
|
||||
['y', 2],
|
||||
['x', 3],
|
||||
['y', 4],
|
||||
];
|
||||
|
||||
var result = Object.fromEntries(entries);
|
||||
assert.sameValue(result['z'], 1);
|
||||
assert.sameValue(result['y'], 4);
|
||||
assert.sameValue(result['x'], 3);
|
||||
assert.compareArray(Object.keys(result), ['z', 'y', 'x']);
|
|
@ -0,0 +1,15 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Object.fromEntries.length is 1.
|
||||
esid: sec-object.fromentries
|
||||
includes: [propertyHelper.js]
|
||||
features: [Object.fromEntries]
|
||||
---*/
|
||||
|
||||
assert.sameValue(Object.fromEntries.length, 1);
|
||||
|
||||
verifyNotEnumerable(Object.fromEntries, "length");
|
||||
verifyNotWritable(Object.fromEntries, "length");
|
||||
verifyConfigurable(Object.fromEntries, "length");
|
|
@ -0,0 +1,15 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Object.fromEntries.name is "fromEntries".
|
||||
esid: sec-object.fromentries
|
||||
includes: [propertyHelper.js]
|
||||
features: [Object.fromEntries]
|
||||
---*/
|
||||
|
||||
assert.sameValue(Object.fromEntries.name, "fromEntries");
|
||||
|
||||
verifyNotEnumerable(Object.fromEntries, "name");
|
||||
verifyNotWritable(Object.fromEntries, "name");
|
||||
verifyConfigurable(Object.fromEntries, "name");
|
|
@ -0,0 +1,12 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Throws when called without an argument.
|
||||
esid: sec-object.fromentries
|
||||
features: [Object.fromEntries]
|
||||
---*/
|
||||
|
||||
assert.throws(TypeError, function() {
|
||||
Object.fromEntries();
|
||||
});
|
|
@ -0,0 +1,14 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Creates data properties which are enumerable, writable, and configurable.
|
||||
esid: sec-object.fromentries
|
||||
includes: [propertyHelper.js]
|
||||
features: [Object.fromEntries]
|
||||
---*/
|
||||
|
||||
var result = Object.fromEntries([['key', 'value']]);
|
||||
verifyEnumerable(result, 'key');
|
||||
verifyWritable(result, 'key');
|
||||
verifyConfigurable(result, 'key');
|
|
@ -0,0 +1,11 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Succeeds when an entry object is a boxed string.
|
||||
esid: sec-object.fromentries
|
||||
features: [Object.fromEntries]
|
||||
---*/
|
||||
|
||||
var result = Object.fromEntries([Object('ab')]);
|
||||
assert.sameValue(result['a'], 'b');
|
|
@ -0,0 +1,12 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Throws when an entry object is a primitive string.
|
||||
esid: sec-object.fromentries
|
||||
features: [Object.fromEntries]
|
||||
---*/
|
||||
|
||||
assert.throws(TypeError, function() {
|
||||
Object.fromEntries(['ab']);
|
||||
});
|
|
@ -0,0 +1,12 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Allows symbol keys.
|
||||
esid: sec-object.fromentries
|
||||
features: [Symbol, Object.fromEntries]
|
||||
---*/
|
||||
|
||||
var key = Symbol();
|
||||
var result = Object.fromEntries([[key, 'value']]);
|
||||
assert.sameValue(result[key], 'value');
|
|
@ -0,0 +1,17 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Coerces keys to strings using ToPropertyKey.
|
||||
esid: sec-object.fromentries
|
||||
features: [Symbol.toPrimitive, Object.fromEntries]
|
||||
---*/
|
||||
|
||||
var key = {
|
||||
[Symbol.toPrimitive]: function(hint) {
|
||||
assert.sameValue(hint, 'string');
|
||||
return 'key';
|
||||
},
|
||||
};
|
||||
var result = Object.fromEntries([[key, 'value']]);
|
||||
assert.sameValue(result.key, 'value');
|
|
@ -0,0 +1,20 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Uses [[DefineOwnProperty]] rather than [[Set]].
|
||||
esid: sec-object.fromentries
|
||||
features: [Object.fromEntries]
|
||||
---*/
|
||||
|
||||
Object.defineProperty(Object.prototype, 'property', {
|
||||
get: function() {
|
||||
throw new Test262Error('should not trigger getter on Object.prototype');
|
||||
},
|
||||
set: function() {
|
||||
throw new Test262Error('should not trigger setter on Object.prototype');
|
||||
},
|
||||
});
|
||||
|
||||
var result = Object.fromEntries([['property', 'value']]);
|
||||
assert.sameValue(result['property'], 'value', '');
|
|
@ -0,0 +1,48 @@
|
|||
// Copyright (C) 2018 Kevin Gibbons. All rights reserved.
|
||||
// This code is governed by the BSD license found in the LICENSE file.
|
||||
|
||||
/*---
|
||||
description: Reads properties rather than iterating.
|
||||
esid: sec-object.fromentries
|
||||
features: [Symbol.iterator, Object.fromEntries]
|
||||
---*/
|
||||
|
||||
var iterable = {
|
||||
[Symbol.iterator]: function() {
|
||||
var count = 0;
|
||||
return {
|
||||
next: function() {
|
||||
if (count === 0) {
|
||||
++count;
|
||||
return {
|
||||
done: false,
|
||||
value: {
|
||||
'0': 'first key',
|
||||
'1': 'first value',
|
||||
get [Symbol.iterator]() {
|
||||
throw new Test262Error('Object.fromEntries should not access Symbol.iterator on entry objects');
|
||||
},
|
||||
},
|
||||
};
|
||||
} else if (count === 1) {
|
||||
++count;
|
||||
Array.prototype[Symbol.iterator] = function() {
|
||||
throw new Test262Error('Object.fromEntries should not access Symbol.iterator on entry arrays');
|
||||
};
|
||||
return {
|
||||
done: false,
|
||||
value: ['second key', 'second value'],
|
||||
};
|
||||
} else {
|
||||
return {
|
||||
done: true,
|
||||
};
|
||||
}
|
||||
},
|
||||
};
|
||||
},
|
||||
};
|
||||
|
||||
var result = Object.fromEntries(iterable);
|
||||
assert.sameValue(result['first key'], 'first value');
|
||||
assert.sameValue(result['second key'], 'second value');
|
Loading…
Reference in New Issue