2017-07-13 00:10:52 +02:00
|
|
|
const promiseToCallback = require('promise-to-callback')
|
2019-03-29 05:37:40 +01:00
|
|
|
const callbackNoop = function (err) { if (err) throw err }
|
2016-11-28 21:43:44 +01:00
|
|
|
|
2018-04-16 19:08:04 +02:00
|
|
|
/**
|
|
|
|
* A generator that returns a function which, when passed a promise, can treat that promise as a node style callback.
|
2018-04-17 01:53:29 +02:00
|
|
|
* The prime advantage being that callbacks are better for error handling.
|
2018-04-16 19:08:04 +02:00
|
|
|
*
|
2018-04-17 00:15:27 +02:00
|
|
|
* @param {Function} fn The function to handle as a callback
|
|
|
|
* @param {Object} context The context in which the fn is to be called, most often a this reference
|
2018-04-16 19:08:04 +02:00
|
|
|
*
|
|
|
|
*/
|
2017-08-04 03:20:01 +02:00
|
|
|
module.exports = function nodeify (fn, context) {
|
2017-10-21 21:06:39 +02:00
|
|
|
return function () {
|
2019-03-29 05:37:40 +01:00
|
|
|
// parse arguments
|
2017-07-13 00:06:49 +02:00
|
|
|
const args = [].slice.call(arguments)
|
2017-10-06 22:02:34 +02:00
|
|
|
const lastArg = args[args.length - 1]
|
2017-10-06 21:36:08 +02:00
|
|
|
const lastArgIsCallback = typeof lastArg === 'function'
|
|
|
|
let callback
|
|
|
|
if (lastArgIsCallback) {
|
|
|
|
callback = lastArg
|
|
|
|
args.pop()
|
|
|
|
} else {
|
2019-03-29 05:37:40 +01:00
|
|
|
callback = callbackNoop
|
2017-10-06 21:36:08 +02:00
|
|
|
}
|
2019-03-29 05:37:40 +01:00
|
|
|
// call the provided function and ensure result is a promise
|
|
|
|
let result
|
|
|
|
try {
|
|
|
|
result = Promise.resolve(fn.apply(context, args))
|
|
|
|
} catch (err) {
|
|
|
|
result = Promise.reject(err)
|
|
|
|
}
|
|
|
|
// wire up promise resolution to callback
|
|
|
|
promiseToCallback(result)(callback)
|
2016-11-28 21:43:44 +01:00
|
|
|
}
|
|
|
|
}
|