mirror of
https://github.com/overleaf/overleaf.git
synced 2025-01-10 04:32:26 +00:00
215 lines
5.9 KiB
JavaScript
215 lines
5.9 KiB
JavaScript
|
const { promisify, callbackify } = require('util')
|
||
|
const pLimit = require('p-limit')
|
||
|
|
||
|
module.exports = {
|
||
|
promisify,
|
||
|
promisifyAll,
|
||
|
promisifyClass,
|
||
|
promisifyMultiResult,
|
||
|
callbackify,
|
||
|
callbackifyAll,
|
||
|
callbackifyMultiResult,
|
||
|
expressify,
|
||
|
promiseMapWithLimit,
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Promisify all functions in a module.
|
||
|
*
|
||
|
* This is meant to be used only when all functions in the module are async
|
||
|
* callback-style functions.
|
||
|
*
|
||
|
* It's very much tailored to our current module structure. In particular, it
|
||
|
* binds `this` to the module when calling the function in order not to break
|
||
|
* modules that call sibling functions using `this`.
|
||
|
*
|
||
|
* This will not magically fix all modules. Special cases should be promisified
|
||
|
* manually.
|
||
|
*
|
||
|
* The second argument is a bag of options:
|
||
|
*
|
||
|
* - without: an array of function names that shouldn't be promisified
|
||
|
*
|
||
|
* - multiResult: an object whose keys are function names and values are lists
|
||
|
* of parameter names. This is meant for functions that invoke their callbacks
|
||
|
* with more than one result in separate parameters. The promisifed function
|
||
|
* will return these results as a single object, with each result keyed under
|
||
|
* the corresponding parameter name.
|
||
|
*/
|
||
|
function promisifyAll(module, opts = {}) {
|
||
|
const { without = [], multiResult = {} } = opts
|
||
|
const promises = {}
|
||
|
for (const propName of Object.getOwnPropertyNames(module)) {
|
||
|
if (without.includes(propName)) {
|
||
|
continue
|
||
|
}
|
||
|
const propValue = module[propName]
|
||
|
if (typeof propValue !== 'function') {
|
||
|
continue
|
||
|
}
|
||
|
if (multiResult[propName] != null) {
|
||
|
promises[propName] = promisifyMultiResult(
|
||
|
propValue,
|
||
|
multiResult[propName]
|
||
|
).bind(module)
|
||
|
} else {
|
||
|
promises[propName] = promisify(propValue).bind(module)
|
||
|
}
|
||
|
}
|
||
|
return promises
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Promisify all methods in a class.
|
||
|
*
|
||
|
* Options are the same as for promisifyAll
|
||
|
*/
|
||
|
function promisifyClass(cls, opts = {}) {
|
||
|
const promisified = class extends cls {}
|
||
|
const { without = [], multiResult = {} } = opts
|
||
|
for (const propName of Object.getOwnPropertyNames(cls.prototype)) {
|
||
|
if (propName === 'constructor' || without.includes(propName)) {
|
||
|
continue
|
||
|
}
|
||
|
const propValue = cls.prototype[propName]
|
||
|
if (typeof propValue !== 'function') {
|
||
|
continue
|
||
|
}
|
||
|
if (multiResult[propName] != null) {
|
||
|
promisified.prototype[propName] = promisifyMultiResult(
|
||
|
propValue,
|
||
|
multiResult[propName]
|
||
|
)
|
||
|
} else {
|
||
|
promisified.prototype[propName] = promisify(propValue)
|
||
|
}
|
||
|
}
|
||
|
return promisified
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Promisify a function that returns multiple results via additional callback
|
||
|
* parameters.
|
||
|
*
|
||
|
* The promisified function returns the results in a single object whose keys
|
||
|
* are the names given in the array `resultNames`.
|
||
|
*
|
||
|
* Example:
|
||
|
*
|
||
|
* function f(callback) {
|
||
|
* return callback(null, 1, 2, 3)
|
||
|
* }
|
||
|
*
|
||
|
* const g = promisifyMultiResult(f, ['a', 'b', 'c'])
|
||
|
*
|
||
|
* const result = await g() // returns {a: 1, b: 2, c: 3}
|
||
|
*/
|
||
|
function promisifyMultiResult(fn, resultNames) {
|
||
|
function promisified(...args) {
|
||
|
return new Promise((resolve, reject) => {
|
||
|
try {
|
||
|
fn.bind(this)(...args, (err, ...results) => {
|
||
|
if (err != null) {
|
||
|
return reject(err)
|
||
|
}
|
||
|
const promiseResult = {}
|
||
|
for (let i = 0; i < resultNames.length; i++) {
|
||
|
promiseResult[resultNames[i]] = results[i]
|
||
|
}
|
||
|
resolve(promiseResult)
|
||
|
})
|
||
|
} catch (err) {
|
||
|
reject(err)
|
||
|
}
|
||
|
})
|
||
|
}
|
||
|
return promisified
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Reverse of `promisifyAll`.
|
||
|
*
|
||
|
* Callbackify all async functions in a module and return them in an object. In
|
||
|
* contrast with `promisifyAll`, all other exports from the module are added to
|
||
|
* the result.
|
||
|
*
|
||
|
* This is meant to be used like this:
|
||
|
*
|
||
|
* const MyPromisifiedModule = {...}
|
||
|
* module.exports = {
|
||
|
* ...callbackifyAll(MyPromisifiedModule),
|
||
|
* promises: MyPromisifiedModule
|
||
|
* }
|
||
|
*
|
||
|
* @param {Object} module - The module to callbackify
|
||
|
* @param {Object} opts - Options
|
||
|
* @param {Object} opts.multiResult - Spec of methods to be callbackified with
|
||
|
* callbackifyMultiResult()
|
||
|
*/
|
||
|
function callbackifyAll(module, opts = {}) {
|
||
|
const { multiResult = {} } = opts
|
||
|
const callbacks = {}
|
||
|
for (const propName of Object.getOwnPropertyNames(module)) {
|
||
|
const propValue = module[propName]
|
||
|
if (typeof propValue === 'function') {
|
||
|
if (propValue.constructor.name === 'AsyncFunction') {
|
||
|
if (multiResult[propName] != null) {
|
||
|
callbacks[propName] = callbackifyMultiResult(
|
||
|
propValue,
|
||
|
multiResult[propName]
|
||
|
).bind(module)
|
||
|
} else {
|
||
|
callbacks[propName] = callbackify(propValue).bind(module)
|
||
|
}
|
||
|
} else {
|
||
|
callbacks[propName] = propValue.bind(module)
|
||
|
}
|
||
|
} else {
|
||
|
callbacks[propName] = propValue
|
||
|
}
|
||
|
}
|
||
|
return callbacks
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Reverse the effect of `promisifyMultiResult`.
|
||
|
*
|
||
|
* This is meant for providing a temporary backward compatible callback
|
||
|
* interface while we migrate to promises.
|
||
|
*/
|
||
|
function callbackifyMultiResult(fn, resultNames) {
|
||
|
function callbackified(...args) {
|
||
|
const [callback] = args.splice(-1)
|
||
|
fn(...args)
|
||
|
.then(result => {
|
||
|
const cbResults = resultNames.map(resultName => result[resultName])
|
||
|
callback(null, ...cbResults)
|
||
|
})
|
||
|
.catch(err => {
|
||
|
callback(err)
|
||
|
})
|
||
|
}
|
||
|
return callbackified
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Transform an async function into an Express middleware
|
||
|
*
|
||
|
* Any error will be passed to the error middlewares via `next()`
|
||
|
*/
|
||
|
function expressify(fn) {
|
||
|
return (req, res, next) => {
|
||
|
fn(req, res, next).catch(next)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Map values in `array` with the async function `fn`
|
||
|
*
|
||
|
* Limit the number of unresolved promises to `concurrency`.
|
||
|
*/
|
||
|
function promiseMapWithLimit(concurrency, array, fn) {
|
||
|
const limit = pLimit(concurrency)
|
||
|
return Promise.all(array.map(x => limit(() => fn(x))))
|
||
|
}
|