CreateAsyncFromSyncIterator.js 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. 'use strict';
  2. var GetIntrinsic = require('get-intrinsic');
  3. var $SyntaxError = require('es-errors/syntax');
  4. var $TypeError = require('es-errors/type');
  5. var $Promise = GetIntrinsic('%Promise%', true);
  6. var Call = require('./Call');
  7. var CreateIterResultObject = require('./CreateIterResultObject');
  8. var Get = require('./Get');
  9. var GetMethod = require('./GetMethod');
  10. var IteratorComplete = require('./IteratorComplete');
  11. var IteratorNext = require('./IteratorNext');
  12. var IteratorValue = require('./IteratorValue');
  13. var ObjectCreate = require('./ObjectCreate');
  14. var PromiseResolve = require('./PromiseResolve');
  15. var Type = require('./Type');
  16. var isIteratorRecord = require('../helpers/records/iterator-record');
  17. var SLOT = require('internal-slot');
  18. var callBound = require('call-bind/callBound');
  19. var $then = callBound('Promise.prototype.then', true);
  20. var AsyncFromSyncIteratorContinuation = function AsyncFromSyncIteratorContinuation(result) {
  21. if (Type(result) !== 'Object') {
  22. throw new $TypeError('Assertion failed: Type(O) is not Object');
  23. }
  24. if (arguments.length > 1) {
  25. throw new $TypeError('although AsyncFromSyncIteratorContinuation should take a second argument, it is not used in this implementation');
  26. }
  27. if (!$Promise) {
  28. throw new $SyntaxError('This environment does not support Promises.');
  29. }
  30. return new Promise(function (resolve) {
  31. var done = IteratorComplete(result); // step 2
  32. var value = IteratorValue(result); // step 4
  33. var valueWrapper = PromiseResolve($Promise, value); // step 6
  34. // eslint-disable-next-line no-shadow
  35. var onFulfilled = function (value) { // steps 8-9
  36. return CreateIterResultObject(value, done); // step 8.a
  37. };
  38. resolve($then(valueWrapper, onFulfilled)); // step 11
  39. }); // step 12
  40. };
  41. var $AsyncFromSyncIteratorPrototype = GetIntrinsic('%AsyncFromSyncIteratorPrototype%', true) || {
  42. next: function next(value) {
  43. var O = this; // step 1
  44. SLOT.assert(O, '[[SyncIteratorRecord]]'); // step 2
  45. var argsLength = arguments.length;
  46. return new Promise(function (resolve) { // step 3
  47. var syncIteratorRecord = SLOT.get(O, '[[SyncIteratorRecord]]'); // step 4
  48. var result;
  49. if (argsLength > 0) {
  50. result = IteratorNext(syncIteratorRecord['[[Iterator]]'], value); // step 5.a
  51. } else { // step 6
  52. result = IteratorNext(syncIteratorRecord['[[Iterator]]']);// step 6.a
  53. }
  54. resolve(AsyncFromSyncIteratorContinuation(result)); // step 8
  55. });
  56. },
  57. 'return': function () {
  58. var O = this; // step 1
  59. SLOT.assert(O, '[[SyncIteratorRecord]]'); // step 2
  60. var valueIsPresent = arguments.length > 0;
  61. var value = valueIsPresent ? arguments[0] : void undefined;
  62. return new Promise(function (resolve, reject) { // step 3
  63. var syncIterator = SLOT.get(O, '[[SyncIteratorRecord]]')['[[Iterator]]']; // step 4
  64. var iteratorReturn = GetMethod(syncIterator, 'return'); // step 5
  65. if (typeof iteratorReturn === 'undefined') { // step 7
  66. var iterResult = CreateIterResultObject(value, true); // step 7.a
  67. Call(resolve, undefined, [iterResult]); // step 7.b
  68. return;
  69. }
  70. var result;
  71. if (valueIsPresent) { // step 8
  72. result = Call(iteratorReturn, syncIterator, [value]); // step 8.a
  73. } else { // step 9
  74. result = Call(iteratorReturn, syncIterator); // step 9.a
  75. }
  76. if (Type(result) !== 'Object') { // step 11
  77. Call(reject, undefined, [new $TypeError('Iterator `return` method returned a non-object value.')]); // step 11.a
  78. return;
  79. }
  80. resolve(AsyncFromSyncIteratorContinuation(result)); // step 12
  81. });
  82. },
  83. 'throw': function () {
  84. var O = this; // step 1
  85. SLOT.assert(O, '[[SyncIteratorRecord]]'); // step 2
  86. var valueIsPresent = arguments.length > 0;
  87. var value = valueIsPresent ? arguments[0] : void undefined;
  88. return new Promise(function (resolve, reject) { // step 3
  89. var syncIterator = SLOT.get(O, '[[SyncIteratorRecord]]')['[[Iterator]]']; // step 4
  90. var throwMethod = GetMethod(syncIterator, 'throw'); // step 5
  91. if (typeof throwMethod === 'undefined') { // step 7
  92. Call(reject, undefined, [value]); // step 7.a
  93. return;
  94. }
  95. var result;
  96. if (valueIsPresent) { // step 8
  97. result = Call(throwMethod, syncIterator, [value]); // step 8.a
  98. } else { // step 9
  99. result = Call(throwMethod, syncIterator); // step 9.a
  100. }
  101. if (Type(result) !== 'Object') { // step 11
  102. Call(reject, undefined, [new $TypeError('Iterator `throw` method returned a non-object value.')]); // step 11.a
  103. return;
  104. }
  105. resolve(AsyncFromSyncIteratorContinuation(result/* , promiseCapability */)); // step 12
  106. });
  107. }
  108. };
  109. // https://262.ecma-international.org/9.0/#sec-createasyncfromsynciterator
  110. module.exports = function CreateAsyncFromSyncIterator(syncIteratorRecord) {
  111. if (!isIteratorRecord(syncIteratorRecord)) {
  112. throw new $TypeError('Assertion failed: `syncIteratorRecord` is not an Iterator Record');
  113. }
  114. // var asyncIterator = ObjectCreate(%AsyncFromSyncIteratorPrototype%, « [[SyncIteratorRecord]] »); // step 1
  115. var asyncIterator = ObjectCreate($AsyncFromSyncIteratorPrototype);
  116. SLOT.set(asyncIterator, '[[SyncIteratorRecord]]', syncIteratorRecord); // step 2
  117. var nextMethod = Get(asyncIterator, 'next'); // step 3
  118. return { // steps 3-4
  119. '[[Iterator]]': asyncIterator,
  120. '[[NextMethod]]': nextMethod,
  121. '[[Done]]': false
  122. };
  123. };