'use strict'; var GetIntrinsic = require('get-intrinsic'); var $SyntaxError = GetIntrinsic('%SyntaxError%'); var $TypeError = GetIntrinsic('%TypeError%'); var $Promise = GetIntrinsic('%Promise%', true); var Call = require('./Call'); var CreateIterResultObject = require('./CreateIterResultObject'); var Get = require('./Get'); var GetMethod = require('./GetMethod'); var IteratorComplete = require('./IteratorComplete'); var IteratorNext = require('./IteratorNext'); var IteratorValue = require('./IteratorValue'); var ObjectCreate = require('./ObjectCreate'); var PromiseResolve = require('./PromiseResolve'); var Type = require('./Type'); var SLOT = require('internal-slot'); var callBound = require('call-bind/callBound'); var $then = callBound('Promise.prototype.then', true); var assertRecord = require('../helpers/assertRecord'); var AsyncFromSyncIteratorContinuation = function AsyncFromSyncIteratorContinuation(result) { if (Type(result) !== 'Object') { throw new $TypeError('Assertion failed: Type(O) is not Object'); } if (arguments.length > 1) { throw new $TypeError('although AsyncFromSyncIteratorContinuation should take a second argument, it is not used in this implementation'); } if (!$Promise) { throw new $SyntaxError('This environment does not support Promises.'); } return new Promise(function (resolve) { var done = IteratorComplete(result); // step 2 var value = IteratorValue(result); // step 4 var valueWrapper = PromiseResolve($Promise, value); // step 6 // eslint-disable-next-line no-shadow var onFulfilled = function (value) { // steps 8-9 return CreateIterResultObject(value, done); // step 8.a }; resolve($then(valueWrapper, onFulfilled)); // step 11 }); // step 12 }; var $AsyncFromSyncIteratorPrototype = GetIntrinsic('%AsyncFromSyncIteratorPrototype%', true) || { next: function next(value) { var O = this; // step 1 SLOT.assert(O, '[[SyncIteratorRecord]]'); // step 2 var argsLength = arguments.length; return new Promise(function (resolve) { // step 3 var syncIteratorRecord = SLOT.get(O, '[[SyncIteratorRecord]]'); // step 4 var result; if (argsLength > 0) { result = IteratorNext(syncIteratorRecord['[[Iterator]]'], value); // step 5.a } else { // step 6 result = IteratorNext(syncIteratorRecord['[[Iterator]]']);// step 6.a } resolve(AsyncFromSyncIteratorContinuation(result)); // step 8 }); }, 'return': function () { var O = this; // step 1 SLOT.assert(O, '[[SyncIteratorRecord]]'); // step 2 var valueIsPresent = arguments.length > 0; var value = valueIsPresent ? arguments[0] : void undefined; return new Promise(function (resolve, reject) { // step 3 var syncIterator = SLOT.get(O, '[[SyncIteratorRecord]]')['[[Iterator]]']; // step 4 var iteratorReturn = GetMethod(syncIterator, 'return'); // step 5 if (typeof iteratorReturn === 'undefined') { // step 7 var iterResult = CreateIterResultObject(value, true); // step 7.a Call(resolve, undefined, [iterResult]); // step 7.b return; } var result; if (valueIsPresent) { // step 8 result = Call(iteratorReturn, syncIterator, [value]); // step 8.a } else { // step 9 result = Call(iteratorReturn, syncIterator); // step 9.a } if (Type(result) !== 'Object') { // step 11 Call(reject, undefined, [new $TypeError('Iterator `return` method returned a non-object value.')]); // step 11.a return; } resolve(AsyncFromSyncIteratorContinuation(result)); // step 12 }); }, 'throw': function () { var O = this; // step 1 SLOT.assert(O, '[[SyncIteratorRecord]]'); // step 2 var valueIsPresent = arguments.length > 0; var value = valueIsPresent ? arguments[0] : void undefined; return new Promise(function (resolve, reject) { // step 3 var syncIterator = SLOT.get(O, '[[SyncIteratorRecord]]')['[[Iterator]]']; // step 4 var throwMethod = GetMethod(syncIterator, 'throw'); // step 5 if (typeof throwMethod === 'undefined') { // step 7 Call(reject, undefined, [value]); // step 7.a return; } var result; if (valueIsPresent) { // step 8 result = Call(throwMethod, syncIterator, [value]); // step 8.a } else { // step 9 result = Call(throwMethod, syncIterator); // step 9.a } if (Type(result) !== 'Object') { // step 11 Call(reject, undefined, [new $TypeError('Iterator `throw` method returned a non-object value.')]); // step 11.a return; } resolve(AsyncFromSyncIteratorContinuation(result/* , promiseCapability */)); // step 12 }); } }; // https://262.ecma-international.org/9.0/#sec-createasyncfromsynciterator module.exports = function CreateAsyncFromSyncIterator(syncIteratorRecord) { assertRecord(Type, 'Iterator Record', 'syncIteratorRecord', syncIteratorRecord); // var asyncIterator = ObjectCreate(%AsyncFromSyncIteratorPrototype%, « [[SyncIteratorRecord]] »); // step 1 var asyncIterator = ObjectCreate($AsyncFromSyncIteratorPrototype); SLOT.set(asyncIterator, '[[SyncIteratorRecord]]', syncIteratorRecord); // step 2 var nextMethod = Get(asyncIterator, 'next'); // step 3 return { // steps 3-4 '[[Iterator]]': asyncIterator, '[[NextMethod]]': nextMethod, '[[Done]]': false }; };