UNPKG

es-iterator-helpers

Version:

An ESnext spec-compliant iterator helpers shim/polyfill/replacement that works as far down as ES3.

464 lines (383 loc) 13 kB
'use strict'; var defineProperties = require('define-properties'); var test = require('tape'); var callBind = require('call-bind'); var functionsHaveNames = require('functions-have-names')(); var hasStrictMode = require('has-strict-mode')(); var forEach = require('for-each'); var debug = require('object-inspect'); var v = require('es-value-fixtures'); var hasSymbols = require('has-symbols/shams')(); var hasPropertyDescriptors = require('has-property-descriptors')(); var generators = require('make-generator-function')(); var iterate = require('iterate-iterator'); var index = require('../Iterator.prototype.map'); var impl = require('../Iterator.prototype.map/implementation'); var fnName = 'map'; var isEnumerable = Object.prototype.propertyIsEnumerable; var testIterator = require('./helpers/testIterator'); module.exports = { tests: function (map, name, t) { t['throws']( function () { return new map(); }, // eslint-disable-line new-cap TypeError, '`' + name + '` itself is not a constructor' ); forEach(v.primitives.concat(v.objects), function (nonIterator) { t['throws']( function () { iterate(map(nonIterator, function () {})); }, TypeError, debug(nonIterator) + ' is not an Object with a callable `next` method' ); var badNext = { next: nonIterator }; t['throws']( function () { iterate(map(badNext, function () {})); }, TypeError, debug(badNext) + ' is not an Object with a callable `next` method' ); }); forEach(v.nonFunctions, function (nonFunction) { t['throws']( function () { map({ next: function () {} }, nonFunction); }, TypeError, debug(nonFunction) + ' is not a function' ); }); var sentinel = {}; var done = false; var fakeIterator = { next: function () { try { return { done: !!done, value: sentinel }; } finally { done = done === false ? null : true; } } }; var result = {}; testIterator( map(fakeIterator, function (x, i) { result.value = x; result.counter = i; result.receiver = this; result.args = arguments.length; return fakeIterator; }), [fakeIterator, fakeIterator], t, 'fake iterator, mapped, runs as expected' ); t.deepEqual( result, { value: sentinel, counter: 1, receiver: undefined, args: 2 }, 'callback is called with the correct arguments' ); t.test('actual iteration', { skip: !hasSymbols }, function (st) { var arr = [1, 2, 3]; var iterator = callBind(arr[Symbol.iterator], arr); st['throws']( function () { return new map(iterator()); }, // eslint-disable-line new-cap TypeError, '`' + name + '` iterator is not a constructor' ); st['throws']( function () { return new map(iterator(), function () {}); }, // eslint-disable-line new-cap TypeError, '`' + name + '` iterator is not a constructor' ); testIterator(iterator(), [1, 2, 3], st, 'original'); testIterator(map(iterator(), function (x) { return x; }), [1, 2, 3], st, 'identity mapper'); testIterator(map(iterator(), function (x) { return 2 * x; }), [2, 4, 6], st, 'doubler mapper'); st.test('generators', { skip: generators.length === 0 }, function (s2t) { forEach(generators, function (gen) { s2t.doesNotThrow( function () { map(gen(), function () {}); }, 'generator function ' + debug(gen) + ' does not need to be from-wrapped first' ); }); s2t.end(); }); st.test('262: mapper-this', function (s2t) { var thisValue; var iter = map( [1][Symbol.iterator](), function () { thisValue = this; return 42; } ); iter.next(); s2t.equal(thisValue, undefined, 'mapper this is undefined in sloppy mode'); s2t.end(); }); st.test('262: result is iterator', function (s2t) { var mappedIter = map([1, 2][Symbol.iterator](), function (x) { return x; }); s2t.equal(typeof mappedIter.next, 'function', 'has next method'); s2t.equal(typeof mappedIter[Symbol.iterator], 'function', 'has Symbol.iterator method'); s2t.equal(mappedIter[Symbol.iterator](), mappedIter, 'Symbol.iterator returns itself'); s2t.end(); }); st.test('262: get next method only once', { skip: !hasPropertyDescriptors }, function (s2t) { var nextGets = 0; var testIter = { next: function () { return { done: true, value: undefined }; } }; Object.defineProperty(testIter, 'next', { get: function () { nextGets += 1; return function () { return { done: true, value: undefined }; }; } }); var iter = map(testIter, function (x) { return x; }); s2t.equal(nextGets, 1, 'next retrieved once on creation'); iter.next(); s2t.equal(nextGets, 1, 'next not retrieved again'); s2t.end(); }); st.test('262: get next method throws', { skip: !hasPropertyDescriptors }, function (s2t) { var testIter = {}; Object.defineProperty(testIter, 'next', { get: function () { throw new EvalError('next getter threw'); } }); s2t['throws']( function () { map(testIter, function () {}); }, EvalError, 'throws when getting next throws' ); s2t.end(); }); st.test('262: next method returns non-object throws', function (s2t) { var badIterator = { next: function () { return null; } }; var iter = map(badIterator, function (x) { return x; }); s2t['throws'](function () { iter.next(); }, TypeError, 'throws when next returns null'); var badIterator2 = { next: function () { return 42; } }; var iter2 = map(badIterator2, function (x) { return x; }); s2t['throws'](function () { iter2.next(); }, TypeError, 'throws when next returns number'); s2t.end(); }); st.test('262: next method returns throwing done', { skip: !hasPropertyDescriptors }, function (s2t) { var throwingIterator = { next: function () { var iterResult = { value: 1 }; Object.defineProperty(iterResult, 'done', { get: function () { throw new EvalError('done getter threw'); } }); return iterResult; } }; var iter = map(throwingIterator, function (x) { return x; }); s2t['throws'](function () { iter.next(); }, EvalError, 'throws when done getter throws'); s2t.end(); }); st.test('262: next method returns throwing value', { skip: !hasPropertyDescriptors }, function (s2t) { var throwingIterator = { next: function () { var iterResult = { done: false }; Object.defineProperty(iterResult, 'value', { get: function () { throw new EvalError('value getter threw'); } }); return iterResult; } }; var iter = map(throwingIterator, function (x) { return x; }); s2t['throws'](function () { iter.next(); }, EvalError, 'throws when value getter throws'); s2t.end(); }); st.test('262: next method throws', function (s2t) { var throwingIterator = { next: function () { throw new EvalError('next threw'); } }; var iter = map(throwingIterator, function (x) { return x; }); s2t['throws'](function () { iter.next(); }, EvalError, 'throws error from next'); s2t.end(); }); st.test('262: mapper throws', function (s2t) { var returnCalls = 0; var testIter = { next: function () { return { done: false, value: 1 }; }, 'return': function () { returnCalls += 1; return { done: true, value: undefined }; } }; var iter = map(testIter, function () { throw new SyntaxError('mapper threw'); }); s2t['throws'](function () { iter.next(); }, SyntaxError, 'throws error from mapper'); s2t.equal(returnCalls, 1, 'return called on underlying iterator'); s2t.end(); }); st.test('262: mapper throws then closing iterator also throws', function (s2t) { var testIter = { next: function () { return { done: false, value: 1 }; }, 'return': function () { throw new EvalError('return threw'); } }; var iter = map(testIter, function () { throw new SyntaxError('mapper threw'); }); s2t['throws'](function () { iter.next(); }, SyntaxError, 'mapper exception wins over return exception'); s2t.end(); }); st.test('262: get return method throws', { skip: !hasPropertyDescriptors }, function (s2t) { var badIterator = { next: function () { return { done: false, value: 1 }; } }; Object.defineProperty(badIterator, 'return', { get: function () { throw new SyntaxError('return getter threw'); } }); var iter = map(badIterator, function (x) { return x; }); iter.next(); s2t['throws'](function () { iter['return'](); }, SyntaxError, 'throws error from return getter'); s2t.end(); }); st.test('262: return is forwarded to underlying iterator', function (s2t) { var returnCalls = 0; var testIter = { next: function () { return { done: false, value: 1 }; }, 'return': function () { returnCalls += 1; return { done: true, value: undefined }; } }; var iter = map(testIter, function (x) { return x; }); iter.next(); s2t.equal(returnCalls, 0, 'return not called before calling return()'); iter['return'](); s2t.equal(returnCalls, 1, 'return called once'); s2t.end(); }); st.test('262: return is not forwarded after exhaustion', function (s2t) { var returnCalls = 0; var testIter = { next: function () { return { done: true, value: undefined }; }, 'return': function () { returnCalls += 1; throw new EvalError('return should not be called'); } }; var iter = map(testIter, function (x) { return x; }); iter.next(); iter['return'](); s2t.equal(returnCalls, 0, 'return not called after exhaustion'); s2t.end(); }); st.test('262: iterator already exhausted', function (s2t) { var testIter = { next: function () { return { done: true, value: undefined }; } }; var mapperCalls = 0; var iter = map(testIter, function (x) { mapperCalls += 1; return x; }); var nextResult = iter.next(); s2t.equal(nextResult.done, true, 'done is true'); s2t.equal(nextResult.value, undefined, 'value is undefined'); s2t.equal(mapperCalls, 0, 'mapper not called for exhausted iterator'); s2t.end(); }); st.test('262: throws TypeError when generator is running', function (s2t) { var reentrantIterator; var testIter = { next: function () { reentrantIterator.next(); return { done: false, value: 1 }; } }; reentrantIterator = map(testIter, function (x) { return x; }); s2t['throws'](function () { reentrantIterator.next(); }, TypeError, 'throws on reentrant next()'); s2t.end(); }); st.test('262: underlying iterator closed when mapper returns abrupt', function (s2t) { var returnCalls = 0; var testIter = { next: function () { return { done: false, value: 1 }; }, 'return': function () { returnCalls += 1; return { done: true, value: undefined }; } }; var iter = map(testIter, function () { throw new EvalError('mapper threw'); }); s2t['throws'](function () { iter.next(); }, EvalError); s2t.equal(returnCalls, 1, 'underlying iterator closed on mapper throw'); s2t.end(); }); st.end(); }); }, index: function () { test('Iterator.prototype.' + fnName + ': index', function (t) { module.exports.tests(index, 'Iterator.prototype.' + fnName, t); t.end(); }); }, implementation: function () { test('Iterator.prototype.' + fnName + ': implementation', function (t) { module.exports.tests(callBind(impl), 'Iterator.prototype.' + fnName, t); t.end(); }); }, shimmed: function () { test('Iterator.prototype.' + fnName + ': shimmed', function (t) { t.equal(typeof Iterator.prototype[fnName], 'function', 'exists and is a function'); t.test('Function name', { skip: !functionsHaveNames }, function (st) { st.equal(Iterator.prototype[fnName].name, fnName, 'Iterator#' + fnName + ' has name "' + fnName + '"'); st.end(); }); t.test('enumerability', { skip: !defineProperties.supportsDescriptors }, function (et) { et.equal(false, isEnumerable.call(Iterator.prototype, fnName), 'Iterator#' + fnName + ' is not enumerable'); et.end(); }); t.test('bad string/this value', { skip: !hasStrictMode }, function (st) { st['throws'](function () { return Iterator.prototype[fnName].call(undefined, 'a'); }, TypeError, 'undefined is not an object'); st['throws'](function () { return Iterator.prototype[fnName].call(null, 'a'); }, TypeError, 'null is not an object'); st.end(); }); module.exports.tests(callBind(Iterator.prototype[fnName]), 'Iterator.prototype.' + fnName, t); t.end(); }); } };