| 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;}
 |