Skip to content

Commit

Permalink
[Breaking] 2018+: GetIterator: Add hint parameter
Browse files Browse the repository at this point in the history
BREAKING CHANGE: This is backwards incompatible, as the `hint` parameter
is added between the `obj` and `method` parameters.
  • Loading branch information
ExE-Boss authored and ljharb committed Nov 4, 2019
1 parent ae3c4ed commit 224dd6d
Show file tree
Hide file tree
Showing 6 changed files with 124 additions and 22 deletions.
2 changes: 0 additions & 2 deletions .gitattributes
Original file line number Diff line number Diff line change
Expand Up @@ -268,7 +268,6 @@
2018/DeletePropertyOrThrow.js spackled
2018/FromPropertyDescriptor.js spackled
2018/Get.js spackled
2018/GetIterator.js spackled
2018/GetMethod.js spackled
2018/GetOwnPropertyKeys.js spackled
2018/GetPrototypeFromConstructor.js spackled
Expand All @@ -291,7 +290,6 @@
2018/IsPromise.js spackled
2018/IsPropertyKey.js spackled
2018/IsRegExp.js spackled
2018/IterableToList.js spackled
2018/IteratorClose.js spackled
2018/IteratorComplete.js spackled
2018/IteratorNext.js spackled
Expand Down
45 changes: 36 additions & 9 deletions 2018/GetIterator.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,10 @@
var GetIntrinsic = require('../GetIntrinsic');

var $TypeError = GetIntrinsic('%TypeError%');
var $asyncIterator = GetIntrinsic('%Symbol.asyncIterator%', true);

var inspect = require('object-inspect');
var hasSymbols = require('has-symbols')();

var getIteratorMethod = require('../helpers/getIteratorMethod');
var AdvanceStringIndex = require('./AdvanceStringIndex');
Expand All @@ -11,25 +15,48 @@ var GetMethod = require('./GetMethod');
var IsArray = require('./IsArray');
var Type = require('./Type');

// https://ecma-international.org/ecma-262/6.0/#sec-getiterator

module.exports = function GetIterator(obj, method) {
var actualMethod = method;
// https://ecma-international.org/ecma-262/9.0/#sec-getiterator
module.exports = function GetIterator(obj, hint, method) {
var actualHint = hint;
if (arguments.length < 2) {
actualMethod = getIteratorMethod(
{
actualHint = 'sync';
}
if (actualHint !== 'sync' && actualHint !== 'async') {
throw new $TypeError("Assertion failed: `hint` must be one of 'sync' or 'async', got " + inspect(hint));
}
var actualMethod = method;
if (arguments.length < 3) {
if (actualHint === 'async') {
if (hasSymbols && $asyncIterator) {
actualMethod = GetMethod(obj, $asyncIterator);
}
if (actualMethod === undefined) {
throw new $TypeError("async from sync iterators aren't currently supported");
}
} else {
actualMethod = getIteratorMethod({
AdvanceStringIndex: AdvanceStringIndex,
GetMethod: GetMethod,
IsArray: IsArray,
Type: Type
},
obj
);
}, obj);
}
}

var iterator = Call(actualMethod, obj);
if (Type(iterator) !== 'Object') {
throw new $TypeError('iterator must return an object');
}

return iterator;

// TODO: This should return an IteratorRecord
/*
var nextMethod = GetV(iterator, 'next');
return {
'[[Iterator]]': iterator,
'[[NextMethod]]': nextMethod,
'[[Done]]': false
};
*/
};
2 changes: 1 addition & 1 deletion 2018/IterableToList.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ var IteratorValue = require('./IteratorValue');
// https://www.ecma-international.org/ecma-262/8.0/#sec-iterabletolist

module.exports = function IterableToList(items, method) {
var iterator = GetIterator(items, method);
var iterator = GetIterator(items, 'sync', method);
var values = [];
var next = true;
while (next) {
Expand Down
45 changes: 36 additions & 9 deletions 2019/GetIterator.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,10 @@
var GetIntrinsic = require('../GetIntrinsic');

var $TypeError = GetIntrinsic('%TypeError%');
var $asyncIterator = GetIntrinsic('%Symbol.asyncIterator%', true);

var inspect = require('object-inspect');
var hasSymbols = require('has-symbols')();

var getIteratorMethod = require('../helpers/getIteratorMethod');
var AdvanceStringIndex = require('./AdvanceStringIndex');
Expand All @@ -11,25 +15,48 @@ var GetMethod = require('./GetMethod');
var IsArray = require('./IsArray');
var Type = require('./Type');

// https://ecma-international.org/ecma-262/6.0/#sec-getiterator

module.exports = function GetIterator(obj, method) {
var actualMethod = method;
// https://ecma-international.org/ecma-262/9.0/#sec-getiterator
module.exports = function GetIterator(obj, hint, method) {
var actualHint = hint;
if (arguments.length < 2) {
actualMethod = getIteratorMethod(
{
actualHint = 'sync';
}
if (actualHint !== 'sync' && actualHint !== 'async') {
throw new $TypeError("Assertion failed: `hint` must be one of 'sync' or 'async', got " + inspect(hint));
}
var actualMethod = method;
if (arguments.length < 3) {
if (actualHint === 'async') {
if (hasSymbols && $asyncIterator) {
actualMethod = GetMethod(obj, $asyncIterator);
}
if (actualMethod === undefined) {
throw new $TypeError("async from sync iterators aren't currently supported");
}
} else {
actualMethod = getIteratorMethod({
AdvanceStringIndex: AdvanceStringIndex,
GetMethod: GetMethod,
IsArray: IsArray,
Type: Type
},
obj
);
}, obj);
}
}

var iterator = Call(actualMethod, obj);
if (Type(iterator) !== 'Object') {
throw new $TypeError('iterator must return an object');
}

return iterator;

// TODO: This should return an IteratorRecord
/*
var nextMethod = GetV(iterator, 'next');
return {
'[[Iterator]]': iterator,
'[[NextMethod]]': nextMethod,
'[[Done]]': false
};
*/
};
2 changes: 1 addition & 1 deletion 2019/IterableToList.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ var IteratorValue = require('./IteratorValue');
// https://www.ecma-international.org/ecma-262/8.0/#sec-iterabletolist

module.exports = function IterableToList(items, method) {
var iterator = GetIterator(items, method);
var iterator = GetIterator(items, 'sync', method);
var values = [];
var next = true;
while (next) {
Expand Down
50 changes: 50 additions & 0 deletions test/tests.js
Original file line number Diff line number Diff line change
Expand Up @@ -3732,6 +3732,56 @@ var es2018 = function ES2018(ES, ops, expectedMissing, skips) {
t.end();
});

test('GetIterator', function (t) {
try {
ES.GetIterator({}, null);
} catch (e) {
t.ok(e.message.indexOf("Assertion failed: `hint` must be one of 'sync' or 'async'" >= 0));
}

var arr = [1, 2];
testIterator(t, ES.GetIterator(arr), arr);

testIterator(t, ES.GetIterator('abc'), 'abc'.split(''));

t.test('Symbol.iterator', { skip: !v.hasSymbols }, function (st) {
var m = new Map();
m.set(1, 'a');
m.set(2, 'b');

testIterator(st, ES.GetIterator(m), [[1, 'a'], [2, 'b']]);

st.end();
});

t.test('Symbol.asyncIterator', { skip: !v.hasSymbols || !Symbol.asyncIterator }, function (st) {
try {
ES.GetIterator(arr, 'async');
} catch (e) {
st.ok(e.message.indexOf("async from sync iterators aren't currently supported") >= 0);
}

/** @type {AsyncIterator<any>} */
var it = {
next: function () {
return Promise.resolve({
done: true
});
}
};
var obj = {};
obj[Symbol.asyncIterator] = function () {
return it;
};

st.equal(ES.GetIterator(obj, 'async'), it);

st.end();
});

t.end();
});

test('PromiseResolve', function (t) {
t.test('Promises unsupported', { skip: typeof Promise === 'function' }, function (st) {
st['throws'](
Expand Down

0 comments on commit 224dd6d

Please sign in to comment.