|
"use strict"; |
|
|
|
|
|
|
|
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); |
|
} |
|
}, |
|
}), |
|
} |
|
); |
|
|
|
|
|
|
|
|
|
|
|
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 }); |
|
} |
|
|
|
|
|
|
|
|
|
|
|
function newGenerator({ name, arity, sync, async, errback }) { |
|
assertTypeof("string", "name", name, true ); |
|
assertTypeof("number", "arity", arity, true ); |
|
assertTypeof("function", "sync", sync); |
|
assertTypeof("function", "async", async, true ); |
|
assertTypeof("function", "errback", errback, true ); |
|
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) { |
|
|
|
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(); |
|
} |
|
|
|
|
|
|
|
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); |
|
|
|
|
|
|
|
|
|
let sync = true; |
|
let didSyncResume = false; |
|
const out = gen.next(() => { |
|
if (sync) { |
|
didSyncResume = true; |
|
} else { |
|
step(); |
|
} |
|
}); |
|
sync = false; |
|
|
|
assertSuspend(out, gen); |
|
|
|
if (!didSyncResume) { |
|
|
|
|
|
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) { |
|
|
|
|
|
|
|
|
|
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") { |
|
|
|
|
|
|
|
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; |
|
} |
|
|