overleaf/libraries/o-error/index.js
2020-04-17 09:14:35 +01:00

93 lines
2.3 KiB
JavaScript

'use strict'
/**
* Make custom error types that pass `instanceof` checks, have stack traces,
* support custom messages and properties, and support wrapping errors (causes).
*
* @module
*/
/**
* A base class for custom errors that handles:
*
* 1. Wrapping an optional 'cause'.
* 2. Storing an 'info' object with additional data.
* 3. Setting the name to the subclass name.
*
* @extends Error
*/
class OError extends Error {
/**
* @param {string} message as for built-in Error
* @param {?object} info extra data to attach to the error
*/
constructor({ message, info }) {
super(message)
this.name = this.constructor.name
if (info) {
this.info = info
}
}
/**
* Wrap the given error, which caused this error.
*
* @param {Error} cause
* @return {this}
*/
withCause(cause) {
this.cause = cause
if (this.message && cause.message) {
this.message += ': ' + cause.message
}
return this
}
}
/**
* Return the `info` property from `error` and recursively merge the `info`
* properties from the error's causes, if any.
*
* If a property is repeated, the first one in the cause chain wins.
*
* @param {?Error} error assumed not to have circular causes
* @return {Object}
*/
function getFullInfo(error) {
if (!error) return {}
const info = getFullInfo(error.cause)
if (typeof error.info === 'object') Object.assign(info, error.info)
return info
}
/**
* Return the `stack` property from `error` and recursively append the `stack`
* properties from the error's causes, if any.
*
* @param {?Error} error assumed not to have circular causes
* @return {string}
*/
function getFullStack(error) {
if (!error) return ''
const causeStack = getFullStack(error.cause)
if (causeStack) return error.stack + '\ncaused by: ' + causeStack
return error.stack
}
/**
* Is `error` or one of its causes an instance of `klass`?
*
* @param {?Error} error assumed not to have circular causes
* @param {function} klass
* @return {Boolean}
*/
function hasCauseInstanceOf(error, klass) {
if (!error) return false
return error instanceof klass || hasCauseInstanceOf(error.cause, klass)
}
OError.getFullInfo = getFullInfo
OError.getFullStack = getFullStack
OError.hasCauseInstanceOf = hasCauseInstanceOf
module.exports = OError