| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373 | 
							- "use strict";
 
- // These use the global symbol registry so that multiple copies of this
 
- // library can work together in case they are not deduped.
 
- const GENSYNC_START = Symbol.for("gensync:v1:start");
 
- const GENSYNC_SUSPEND = Symbol.for("gensync:v1:suspend");
 
- const GENSYNC_EXPECTED_START = "GENSYNC_EXPECTED_START";
 
- const GENSYNC_EXPECTED_SUSPEND = "GENSYNC_EXPECTED_SUSPEND";
 
- const GENSYNC_OPTIONS_ERROR = "GENSYNC_OPTIONS_ERROR";
 
- const GENSYNC_RACE_NONEMPTY = "GENSYNC_RACE_NONEMPTY";
 
- const GENSYNC_ERRBACK_NO_CALLBACK = "GENSYNC_ERRBACK_NO_CALLBACK";
 
- module.exports = Object.assign(
 
-   function gensync(optsOrFn) {
 
-     let genFn = optsOrFn;
 
-     if (typeof optsOrFn !== "function") {
 
-       genFn = newGenerator(optsOrFn);
 
-     } else {
 
-       genFn = wrapGenerator(optsOrFn);
 
-     }
 
-     return Object.assign(genFn, makeFunctionAPI(genFn));
 
-   },
 
-   {
 
-     all: buildOperation({
 
-       name: "all",
 
-       arity: 1,
 
-       sync: function(args) {
 
-         const items = Array.from(args[0]);
 
-         return items.map(item => evaluateSync(item));
 
-       },
 
-       async: function(args, resolve, reject) {
 
-         const items = Array.from(args[0]);
 
-         if (items.length === 0) {
 
-           Promise.resolve().then(() => resolve([]));
 
-           return;
 
-         }
 
-         let count = 0;
 
-         const results = items.map(() => undefined);
 
-         items.forEach((item, i) => {
 
-           evaluateAsync(
 
-             item,
 
-             val => {
 
-               results[i] = val;
 
-               count += 1;
 
-               if (count === results.length) resolve(results);
 
-             },
 
-             reject
 
-           );
 
-         });
 
-       },
 
-     }),
 
-     race: buildOperation({
 
-       name: "race",
 
-       arity: 1,
 
-       sync: function(args) {
 
-         const items = Array.from(args[0]);
 
-         if (items.length === 0) {
 
-           throw makeError("Must race at least 1 item", GENSYNC_RACE_NONEMPTY);
 
-         }
 
-         return evaluateSync(items[0]);
 
-       },
 
-       async: function(args, resolve, reject) {
 
-         const items = Array.from(args[0]);
 
-         if (items.length === 0) {
 
-           throw makeError("Must race at least 1 item", GENSYNC_RACE_NONEMPTY);
 
-         }
 
-         for (const item of items) {
 
-           evaluateAsync(item, resolve, reject);
 
-         }
 
-       },
 
-     }),
 
-   }
 
- );
 
- /**
 
-  * Given a generator function, return the standard API object that executes
 
-  * the generator and calls the callbacks.
 
-  */
 
- function makeFunctionAPI(genFn) {
 
-   const fns = {
 
-     sync: function(...args) {
 
-       return evaluateSync(genFn.apply(this, args));
 
-     },
 
-     async: function(...args) {
 
-       return new Promise((resolve, reject) => {
 
-         evaluateAsync(genFn.apply(this, args), resolve, reject);
 
-       });
 
-     },
 
-     errback: function(...args) {
 
-       const cb = args.pop();
 
-       if (typeof cb !== "function") {
 
-         throw makeError(
 
-           "Asynchronous function called without callback",
 
-           GENSYNC_ERRBACK_NO_CALLBACK
 
-         );
 
-       }
 
-       let gen;
 
-       try {
 
-         gen = genFn.apply(this, args);
 
-       } catch (err) {
 
-         cb(err);
 
-         return;
 
-       }
 
-       evaluateAsync(gen, val => cb(undefined, val), err => cb(err));
 
-     },
 
-   };
 
-   return fns;
 
- }
 
- function assertTypeof(type, name, value, allowUndefined) {
 
-   if (
 
-     typeof value === type ||
 
-     (allowUndefined && typeof value === "undefined")
 
-   ) {
 
-     return;
 
-   }
 
-   let msg;
 
-   if (allowUndefined) {
 
-     msg = `Expected opts.${name} to be either a ${type}, or undefined.`;
 
-   } else {
 
-     msg = `Expected opts.${name} to be a ${type}.`;
 
-   }
 
-   throw makeError(msg, GENSYNC_OPTIONS_ERROR);
 
- }
 
- function makeError(msg, code) {
 
-   return Object.assign(new Error(msg), { code });
 
- }
 
- /**
 
-  * Given an options object, return a new generator that dispatches the
 
-  * correct handler based on sync or async execution.
 
-  */
 
- function newGenerator({ name, arity, sync, async, errback }) {
 
-   assertTypeof("string", "name", name, true /* allowUndefined */);
 
-   assertTypeof("number", "arity", arity, true /* allowUndefined */);
 
-   assertTypeof("function", "sync", sync);
 
-   assertTypeof("function", "async", async, true /* allowUndefined */);
 
-   assertTypeof("function", "errback", errback, true /* allowUndefined */);
 
-   if (async && errback) {
 
-     throw makeError(
 
-       "Expected one of either opts.async or opts.errback, but got _both_.",
 
-       GENSYNC_OPTIONS_ERROR
 
-     );
 
-   }
 
-   if (typeof name !== "string") {
 
-     let fnName;
 
-     if (errback && errback.name && errback.name !== "errback") {
 
-       fnName = errback.name;
 
-     }
 
-     if (async && async.name && async.name !== "async") {
 
-       fnName = async.name.replace(/Async$/, "");
 
-     }
 
-     if (sync && sync.name && sync.name !== "sync") {
 
-       fnName = sync.name.replace(/Sync$/, "");
 
-     }
 
-     if (typeof fnName === "string") {
 
-       name = fnName;
 
-     }
 
-   }
 
-   if (typeof arity !== "number") {
 
-     arity = sync.length;
 
-   }
 
-   return buildOperation({
 
-     name,
 
-     arity,
 
-     sync: function(args) {
 
-       return sync.apply(this, args);
 
-     },
 
-     async: function(args, resolve, reject) {
 
-       if (async) {
 
-         async.apply(this, args).then(resolve, reject);
 
-       } else if (errback) {
 
-         errback.call(this, ...args, (err, value) => {
 
-           if (err == null) resolve(value);
 
-           else reject(err);
 
-         });
 
-       } else {
 
-         resolve(sync.apply(this, args));
 
-       }
 
-     },
 
-   });
 
- }
 
- function wrapGenerator(genFn) {
 
-   return setFunctionMetadata(genFn.name, genFn.length, function(...args) {
 
-     return genFn.apply(this, args);
 
-   });
 
- }
 
- function buildOperation({ name, arity, sync, async }) {
 
-   return setFunctionMetadata(name, arity, function*(...args) {
 
-     const resume = yield GENSYNC_START;
 
-     if (!resume) {
 
-       // Break the tail call to avoid a bug in V8 v6.X with --harmony enabled.
 
-       const res = sync.call(this, args);
 
-       return res;
 
-     }
 
-     let result;
 
-     try {
 
-       async.call(
 
-         this,
 
-         args,
 
-         value => {
 
-           if (result) return;
 
-           result = { value };
 
-           resume();
 
-         },
 
-         err => {
 
-           if (result) return;
 
-           result = { err };
 
-           resume();
 
-         }
 
-       );
 
-     } catch (err) {
 
-       result = { err };
 
-       resume();
 
-     }
 
-     // Suspend until the callbacks run. Will resume synchronously if the
 
-     // callback was already called.
 
-     yield GENSYNC_SUSPEND;
 
-     if (result.hasOwnProperty("err")) {
 
-       throw result.err;
 
-     }
 
-     return result.value;
 
-   });
 
- }
 
- function evaluateSync(gen) {
 
-   let value;
 
-   while (!({ value } = gen.next()).done) {
 
-     assertStart(value, gen);
 
-   }
 
-   return value;
 
- }
 
- function evaluateAsync(gen, resolve, reject) {
 
-   (function step() {
 
-     try {
 
-       let value;
 
-       while (!({ value } = gen.next()).done) {
 
-         assertStart(value, gen);
 
-         // If this throws, it is considered to have broken the contract
 
-         // established for async handlers. If these handlers are called
 
-         // synchronously, it is also considered bad behavior.
 
-         let sync = true;
 
-         let didSyncResume = false;
 
-         const out = gen.next(() => {
 
-           if (sync) {
 
-             didSyncResume = true;
 
-           } else {
 
-             step();
 
-           }
 
-         });
 
-         sync = false;
 
-         assertSuspend(out, gen);
 
-         if (!didSyncResume) {
 
-           // Callback wasn't called synchronously, so break out of the loop
 
-           // and let it call 'step' later.
 
-           return;
 
-         }
 
-       }
 
-       return resolve(value);
 
-     } catch (err) {
 
-       return reject(err);
 
-     }
 
-   })();
 
- }
 
- function assertStart(value, gen) {
 
-   if (value === GENSYNC_START) return;
 
-   throwError(
 
-     gen,
 
-     makeError(
 
-       `Got unexpected yielded value in gensync generator: ${JSON.stringify(
 
-         value
 
-       )}. Did you perhaps mean to use 'yield*' instead of 'yield'?`,
 
-       GENSYNC_EXPECTED_START
 
-     )
 
-   );
 
- }
 
- function assertSuspend({ value, done }, gen) {
 
-   if (!done && value === GENSYNC_SUSPEND) return;
 
-   throwError(
 
-     gen,
 
-     makeError(
 
-       done
 
-         ? "Unexpected generator completion. If you get this, it is probably a gensync bug."
 
-         : `Expected GENSYNC_SUSPEND, got ${JSON.stringify(
 
-             value
 
-           )}. If you get this, it is probably a gensync bug.`,
 
-       GENSYNC_EXPECTED_SUSPEND
 
-     )
 
-   );
 
- }
 
- function throwError(gen, err) {
 
-   // Call `.throw` so that users can step in a debugger to easily see which
 
-   // 'yield' passed an unexpected value. If the `.throw` call didn't throw
 
-   // back to the generator, we explicitly do it to stop the error
 
-   // from being swallowed by user code try/catches.
 
-   if (gen.throw) gen.throw(err);
 
-   throw err;
 
- }
 
- function isIterable(value) {
 
-   return (
 
-     !!value &&
 
-     (typeof value === "object" || typeof value === "function") &&
 
-     !value[Symbol.iterator]
 
-   );
 
- }
 
- function setFunctionMetadata(name, arity, fn) {
 
-   if (typeof name === "string") {
 
-     // This should always work on the supported Node versions, but for the
 
-     // sake of users that are compiling to older versions, we check for
 
-     // configurability so we don't throw.
 
-     const nameDesc = Object.getOwnPropertyDescriptor(fn, "name");
 
-     if (!nameDesc || nameDesc.configurable) {
 
-       Object.defineProperty(
 
-         fn,
 
-         "name",
 
-         Object.assign(nameDesc || {}, {
 
-           configurable: true,
 
-           value: name,
 
-         })
 
-       );
 
-     }
 
-   }
 
-   if (typeof arity === "number") {
 
-     const lengthDesc = Object.getOwnPropertyDescriptor(fn, "length");
 
-     if (!lengthDesc || lengthDesc.configurable) {
 
-       Object.defineProperty(
 
-         fn,
 
-         "length",
 
-         Object.assign(lengthDesc || {}, {
 
-           configurable: true,
 
-           value: arity,
 
-         })
 
-       );
 
-     }
 
-   }
 
-   return fn;
 
- }
 
 
  |