mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-22 00:07:52 +01:00
381 lines
9.7 KiB
JavaScript
381 lines
9.7 KiB
JavaScript
'use strict'
|
|
|
|
const { format, inspect, AggregateError: CustomAggregateError } = require('./util')
|
|
/*
|
|
This file is a reduced and adapted version of the main lib/internal/errors.js file defined at
|
|
|
|
https://github.com/nodejs/node/blob/master/lib/internal/errors.js
|
|
|
|
Don't try to replace with the original file and keep it up to date (starting from E(...) definitions)
|
|
with the upstream file.
|
|
*/
|
|
|
|
const AggregateError = globalThis.AggregateError || CustomAggregateError
|
|
const kIsNodeError = Symbol('kIsNodeError')
|
|
const kTypes = [
|
|
'string',
|
|
'function',
|
|
'number',
|
|
'object', // Accept 'Function' and 'Object' as alternative to the lower cased version.
|
|
'Function',
|
|
'Object',
|
|
'boolean',
|
|
'bigint',
|
|
'symbol'
|
|
]
|
|
const classRegExp = /^([A-Z][a-z0-9]*)+$/
|
|
const nodeInternalPrefix = '__node_internal_'
|
|
const codes = {}
|
|
|
|
function assert(value, message) {
|
|
if (!value) {
|
|
throw new codes.ERR_INTERNAL_ASSERTION(message)
|
|
}
|
|
} // Only use this for integers! Decimal numbers do not work with this function.
|
|
|
|
function addNumericalSeparator(val) {
|
|
let res = ''
|
|
let i = val.length
|
|
const start = val[0] === '-' ? 1 : 0
|
|
|
|
for (; i >= start + 4; i -= 3) {
|
|
res = `_${val.slice(i - 3, i)}${res}`
|
|
}
|
|
|
|
return `${val.slice(0, i)}${res}`
|
|
}
|
|
|
|
function getMessage(key, msg, args) {
|
|
if (typeof msg === 'function') {
|
|
assert(
|
|
msg.length <= args.length, // Default options do not count.
|
|
`Code: ${key}; The provided arguments length (${args.length}) does not match the required ones (${msg.length}).`
|
|
)
|
|
return msg(...args)
|
|
}
|
|
|
|
const expectedLength = (msg.match(/%[dfijoOs]/g) || []).length
|
|
assert(
|
|
expectedLength === args.length,
|
|
`Code: ${key}; The provided arguments length (${args.length}) does not match the required ones (${expectedLength}).`
|
|
)
|
|
|
|
if (args.length === 0) {
|
|
return msg
|
|
}
|
|
|
|
return format(msg, ...args)
|
|
}
|
|
|
|
function E(code, message, Base) {
|
|
if (!Base) {
|
|
Base = Error
|
|
}
|
|
|
|
class NodeError extends Base {
|
|
constructor(...args) {
|
|
super(getMessage(code, message, args))
|
|
}
|
|
|
|
toString() {
|
|
return `${this.name} [${code}]: ${this.message}`
|
|
}
|
|
}
|
|
|
|
NodeError.prototype.name = Base.name
|
|
NodeError.prototype.code = code
|
|
NodeError.prototype[kIsNodeError] = true
|
|
|
|
NodeError.prototype.toString = function () {
|
|
return `${this.name} [${code}]: ${this.message}`
|
|
}
|
|
|
|
codes[code] = NodeError
|
|
}
|
|
|
|
function hideStackFrames(fn) {
|
|
// We rename the functions that will be hidden to cut off the stacktrace
|
|
// at the outermost one
|
|
const hidden = nodeInternalPrefix + fn.name
|
|
Object.defineProperty(fn, 'name', {
|
|
value: hidden
|
|
})
|
|
return fn
|
|
}
|
|
|
|
function aggregateTwoErrors(innerError, outerError) {
|
|
if (innerError && outerError && innerError !== outerError) {
|
|
if (Array.isArray(outerError.errors)) {
|
|
// If `outerError` is already an `AggregateError`.
|
|
outerError.errors.push(innerError)
|
|
return outerError
|
|
}
|
|
|
|
const err = new AggregateError([outerError, innerError], outerError.message)
|
|
err.code = outerError.code
|
|
return err
|
|
}
|
|
|
|
return innerError || outerError
|
|
}
|
|
|
|
class AbortError extends Error {
|
|
constructor(message = 'The operation was aborted', options = undefined) {
|
|
if (options !== undefined && typeof options !== 'object') {
|
|
throw new codes.ERR_INVALID_ARG_TYPE('options', 'Object', options)
|
|
}
|
|
|
|
super(message, options)
|
|
this.code = 'ABORT_ERR'
|
|
this.name = 'AbortError'
|
|
}
|
|
}
|
|
|
|
E('ERR_ASSERTION', '%s', Error)
|
|
E(
|
|
'ERR_INVALID_ARG_TYPE',
|
|
(name, expected, actual) => {
|
|
assert(typeof name === 'string', "'name' must be a string")
|
|
|
|
if (!Array.isArray(expected)) {
|
|
expected = [expected]
|
|
}
|
|
|
|
let msg = 'The '
|
|
|
|
if (name.endsWith(' argument')) {
|
|
// For cases like 'first argument'
|
|
msg += `${name} `
|
|
} else {
|
|
msg += `"${name}" ${name.includes('.') ? 'property' : 'argument'} `
|
|
}
|
|
|
|
msg += 'must be '
|
|
const types = []
|
|
const instances = []
|
|
const other = []
|
|
|
|
for (const value of expected) {
|
|
assert(typeof value === 'string', 'All expected entries have to be of type string')
|
|
|
|
if (kTypes.includes(value)) {
|
|
types.push(value.toLowerCase())
|
|
} else if (classRegExp.test(value)) {
|
|
instances.push(value)
|
|
} else {
|
|
assert(value !== 'object', 'The value "object" should be written as "Object"')
|
|
other.push(value)
|
|
}
|
|
} // Special handle `object` in case other instances are allowed to outline
|
|
// the differences between each other.
|
|
|
|
if (instances.length > 0) {
|
|
const pos = types.indexOf('object')
|
|
|
|
if (pos !== -1) {
|
|
types.splice(types, pos, 1)
|
|
instances.push('Object')
|
|
}
|
|
}
|
|
|
|
if (types.length > 0) {
|
|
switch (types.length) {
|
|
case 1:
|
|
msg += `of type ${types[0]}`
|
|
break
|
|
|
|
case 2:
|
|
msg += `one of type ${types[0]} or ${types[1]}`
|
|
break
|
|
|
|
default: {
|
|
const last = types.pop()
|
|
msg += `one of type ${types.join(', ')}, or ${last}`
|
|
}
|
|
}
|
|
|
|
if (instances.length > 0 || other.length > 0) {
|
|
msg += ' or '
|
|
}
|
|
}
|
|
|
|
if (instances.length > 0) {
|
|
switch (instances.length) {
|
|
case 1:
|
|
msg += `an instance of ${instances[0]}`
|
|
break
|
|
|
|
case 2:
|
|
msg += `an instance of ${instances[0]} or ${instances[1]}`
|
|
break
|
|
|
|
default: {
|
|
const last = instances.pop()
|
|
msg += `an instance of ${instances.join(', ')}, or ${last}`
|
|
}
|
|
}
|
|
|
|
if (other.length > 0) {
|
|
msg += ' or '
|
|
}
|
|
}
|
|
|
|
switch (other.length) {
|
|
case 0:
|
|
break
|
|
|
|
case 1:
|
|
if (other[0].toLowerCase() !== other[0]) {
|
|
msg += 'an '
|
|
}
|
|
|
|
msg += `${other[0]}`
|
|
break
|
|
|
|
case 2:
|
|
msg += `one of ${other[0]} or ${other[1]}`
|
|
break
|
|
|
|
default: {
|
|
const last = other.pop()
|
|
msg += `one of ${other.join(', ')}, or ${last}`
|
|
}
|
|
}
|
|
|
|
if (actual == null) {
|
|
msg += `. Received ${actual}`
|
|
} else if (typeof actual === 'function' && actual.name) {
|
|
msg += `. Received function ${actual.name}`
|
|
} else if (typeof actual === 'object') {
|
|
var _actual$constructor
|
|
|
|
if (
|
|
(_actual$constructor = actual.constructor) !== null &&
|
|
_actual$constructor !== undefined &&
|
|
_actual$constructor.name
|
|
) {
|
|
msg += `. Received an instance of ${actual.constructor.name}`
|
|
} else {
|
|
const inspected = inspect(actual, {
|
|
depth: -1
|
|
})
|
|
msg += `. Received ${inspected}`
|
|
}
|
|
} else {
|
|
let inspected = inspect(actual, {
|
|
colors: false
|
|
})
|
|
|
|
if (inspected.length > 25) {
|
|
inspected = `${inspected.slice(0, 25)}...`
|
|
}
|
|
|
|
msg += `. Received type ${typeof actual} (${inspected})`
|
|
}
|
|
|
|
return msg
|
|
},
|
|
TypeError
|
|
)
|
|
E(
|
|
'ERR_INVALID_ARG_VALUE',
|
|
(name, value, reason = 'is invalid') => {
|
|
let inspected = inspect(value)
|
|
|
|
if (inspected.length > 128) {
|
|
inspected = inspected.slice(0, 128) + '...'
|
|
}
|
|
|
|
const type = name.includes('.') ? 'property' : 'argument'
|
|
return `The ${type} '${name}' ${reason}. Received ${inspected}`
|
|
},
|
|
TypeError
|
|
)
|
|
E(
|
|
'ERR_INVALID_RETURN_VALUE',
|
|
(input, name, value) => {
|
|
var _value$constructor
|
|
|
|
const type =
|
|
value !== null &&
|
|
value !== undefined &&
|
|
(_value$constructor = value.constructor) !== null &&
|
|
_value$constructor !== undefined &&
|
|
_value$constructor.name
|
|
? `instance of ${value.constructor.name}`
|
|
: `type ${typeof value}`
|
|
return `Expected ${input} to be returned from the "${name}"` + ` function but got ${type}.`
|
|
},
|
|
TypeError
|
|
)
|
|
E(
|
|
'ERR_MISSING_ARGS',
|
|
(...args) => {
|
|
assert(args.length > 0, 'At least one arg needs to be specified')
|
|
let msg
|
|
const len = args.length
|
|
args = (Array.isArray(args) ? args : [args]).map((a) => `"${a}"`).join(' or ')
|
|
|
|
switch (len) {
|
|
case 1:
|
|
msg += `The ${args[0]} argument`
|
|
break
|
|
|
|
case 2:
|
|
msg += `The ${args[0]} and ${args[1]} arguments`
|
|
break
|
|
|
|
default:
|
|
{
|
|
const last = args.pop()
|
|
msg += `The ${args.join(', ')}, and ${last} arguments`
|
|
}
|
|
break
|
|
}
|
|
|
|
return `${msg} must be specified`
|
|
},
|
|
TypeError
|
|
)
|
|
E(
|
|
'ERR_OUT_OF_RANGE',
|
|
(str, range, input) => {
|
|
assert(range, 'Missing "range" argument')
|
|
let received
|
|
|
|
if (Number.isInteger(input) && Math.abs(input) > 2 ** 32) {
|
|
received = addNumericalSeparator(String(input))
|
|
} else if (typeof input === 'bigint') {
|
|
received = String(input)
|
|
|
|
if (input > 2n ** 32n || input < -(2n ** 32n)) {
|
|
received = addNumericalSeparator(received)
|
|
}
|
|
|
|
received += 'n'
|
|
} else {
|
|
received = inspect(input)
|
|
}
|
|
|
|
return `The value of "${str}" is out of range. It must be ${range}. Received ${received}`
|
|
},
|
|
RangeError
|
|
)
|
|
E('ERR_MULTIPLE_CALLBACK', 'Callback called multiple times', Error)
|
|
E('ERR_METHOD_NOT_IMPLEMENTED', 'The %s method is not implemented', Error)
|
|
E('ERR_STREAM_ALREADY_FINISHED', 'Cannot call %s after a stream was finished', Error)
|
|
E('ERR_STREAM_CANNOT_PIPE', 'Cannot pipe, not readable', Error)
|
|
E('ERR_STREAM_DESTROYED', 'Cannot call %s after a stream was destroyed', Error)
|
|
E('ERR_STREAM_NULL_VALUES', 'May not write null values to stream', TypeError)
|
|
E('ERR_STREAM_PREMATURE_CLOSE', 'Premature close', Error)
|
|
E('ERR_STREAM_PUSH_AFTER_EOF', 'stream.push() after EOF', Error)
|
|
E('ERR_STREAM_UNSHIFT_AFTER_END_EVENT', 'stream.unshift() after end event', Error)
|
|
E('ERR_STREAM_WRITE_AFTER_END', 'write after end', Error)
|
|
E('ERR_UNKNOWN_ENCODING', 'Unknown encoding: %s', TypeError)
|
|
module.exports = {
|
|
AbortError,
|
|
aggregateTwoErrors: hideStackFrames(aggregateTwoErrors),
|
|
hideStackFrames,
|
|
codes
|
|
}
|