'use strict' const { format } = require('node:util') /** * An object that provides methods for creating process warning, emitting them, * and inspecting/managing the emission state of warning. * * @typedef {object} ProcessWarningManager */ /** * @typedef {object} CreateOptions * @property {boolean} [unlimited=false] Indicates if the warning should be * emitted every time (`true`) or only the first time (`false`). */ /** * An error instance representing a process warning. This is what will be * emitted to listeners when the warning is emitted. * * @typedef {Error} ProcessWarning * @property {string} code * @property {string} name * @property {string} message */ /** * A function used to create new {@link ProcessWarning} instances. * * @callback ProcessWarningBuilder * @param {*} [param1] First possible string interpolation value. * @param {*} [param2] Second possible string interpolation value. * @param {*} [param3] Third possible string interpolation value. * @returns ProcessWarning */ /** * Factory that builds a new {@link ProcessWarningManager} and returns it. * * @returns ProcessWarningManager */ function processWarning () { const codes = {} const emitted = new Map() const opts = Object.create(null) /** * Builds a new {@link ProcessWarning} and adds it to the * {@link ProcessWarningManager} such that it can be emitted through the * {@link ProcessWarningManager#emit} method. * * @memberof! ProcessWarningManager * @instance * * @param {string} name Warning name, e.g. `'MyCustomWarning'`. * @param {string} code A unique code for the warning, e.g. `'WARN_001'`. * @param {string} message The body of the warning. * @param {CreateOptions} [options] * @returns {ProcessWarningBuilder} */ function create (name, code, message, { unlimited = false } = {}) { if (!name) throw new Error('Warning name must not be empty') if (!code) throw new Error('Warning code must not be empty') if (!message) throw new Error('Warning message must not be empty') if (typeof unlimited !== 'boolean') throw new Error('Warning opts.unlimited must be a boolean') code = code.toUpperCase() if (codes[code] !== undefined) { throw new Error(`The code '${code}' already exist`) } function buildWarnOpts (a, b, c) { // more performant than spread (...) operator let formatted if (a && b && c) { formatted = format(message, a, b, c) } else if (a && b) { formatted = format(message, a, b) } else if (a) { formatted = format(message, a) } else { formatted = message } return { code, name, message: formatted } } Object.assign(opts, { unlimited }) emitted.set(code, unlimited) codes[code] = buildWarnOpts return codes[code] } /** * A wrapper for {@link ProcessWarningManager#create} that builds a new * deprecation warning. This method is equivalent to passing * `name = 'DeprecationWarning'` to the create method. * * Deprecation warnings have extended support for the Node.js CLI options: * `--throw-deprecation`, `--no-deprecation`, and `--trace-deprecation`. * * @memberof! ProcessWarningManager * @instance * * @param {string} code * @param {string} message * @param {CreateOptions} opts * @returns {ProcessWarningBuilder} */ function createDeprecation (code, message, opts = {}) { return create('DeprecationWarning', code, message, opts) } /** * Emits a registered warning associated with the given `code`. If the * warning message has interpolation strings present, up to the first three * of them can be supplied values with the optional interpolation value * parameters. * * If a warning is set to `unlimited: false`, and has already been emitted * once, invoking this method is a no-operation. * * @memberof! ProcessWarningManager * @instance * * @param {string} code The code for the error to emit, e.g. `'WARN_001'`. * This is the same code that was provided to {@link ProcessWarningManager#create}. * @param {*} [a] Possible message interpolation value. * @param {*} [b] Possible message interpolation value. * @param {*} [c] Possible message interpolation value. */ function emit (code, a, b, c) { if (emitted.get(code) === true && opts.unlimited === false) return if (codes[code] === undefined) throw new Error(`The code '${code}' does not exist`) emitted.set(code, true) const warning = codes[code](a, b, c) process.emitWarning(warning.message, warning.name, warning.code) } return { create, createDeprecation, emit, emitted } } module.exports = processWarning module.exports.default = processWarning module.exports.processWarning = processWarning