mirror of https://github.com/lukechilds/node.git
Browse Source
Add the internal/errors.js core mechanism. PR-URL: https://github.com/nodejs/node/pull/11220 Reviewed-By: Anna Henningsen <anna@addaleax.net> Reviewed-By: Joyee Cheung <joyeec9h3@gmail.com>v6
5 changed files with 361 additions and 0 deletions
@ -0,0 +1,141 @@ |
|||
# Using the internal/errors.js Module |
|||
|
|||
## What is internal/errors.js |
|||
|
|||
The `require('internal/errors')` module is an internal-only module that can be |
|||
used to produce `Error`, `TypeError` and `RangeError` instances that use a |
|||
static, permanent error code and an optionally parameterized message. |
|||
|
|||
The intent of the module is to allow errors provided by Node.js to be assigned a |
|||
permanent identifier. Without a permanent identifier, userland code may need to |
|||
inspect error messages to distinguish one error from another. An unfortunate |
|||
result of that practice is that changes to error messages result in broken code |
|||
in the ecosystem. For that reason, Node.js has considered error message changes |
|||
to be breaking changes. By providing a permanent identifier for a specific |
|||
error, we reduce the need for userland code to inspect error messages. |
|||
|
|||
*Note*: Switching an existing error to use the `internal/errors` module must be |
|||
considered a `semver-major` change. However, once using `internal/errors`, |
|||
changes to `internal/errors` error messages will be handled as `semver-minor` |
|||
or `semver-patch`. |
|||
|
|||
## Using internal/errors.js |
|||
|
|||
The `internal/errors` module exposes three custom `Error` classes that |
|||
are intended to replace existing `Error` objects within the Node.js source. |
|||
|
|||
For instance, an existing `Error` such as: |
|||
|
|||
```js |
|||
var err = new TypeError('Expected string received ' + type); |
|||
``` |
|||
|
|||
Can be replaced by first adding a new error key into the `internal/errors.js` |
|||
file: |
|||
|
|||
```js |
|||
E('FOO', 'Expected string received %s'); |
|||
``` |
|||
|
|||
Then replacing the existing `new TypeError` in the code: |
|||
|
|||
```js |
|||
const errors = require('internal/errors'); |
|||
// ... |
|||
var err = new errors.TypeError('FOO', type); |
|||
``` |
|||
|
|||
## Adding new errors |
|||
|
|||
New static error codes are added by modifying the `internal/errors.js` file |
|||
and appending the new error codes to the end using the utility `E()` method. |
|||
|
|||
```js |
|||
E('EXAMPLE_KEY1', 'This is the error value'); |
|||
E('EXAMPLE_KEY2', (a, b) => return `${a} ${b}`); |
|||
``` |
|||
|
|||
The first argument passed to `E()` is the static identifier. The second |
|||
argument is either a String with optional `util.format()` style replacement |
|||
tags (e.g. `%s`, `%d`), or a function returning a String. The optional |
|||
additional arguments passed to the `errors.message()` function (which is |
|||
used by the `errors.Error`, `errors.TypeError` and `errors.RangeError` classes), |
|||
will be used to format the error message. |
|||
|
|||
## Documenting new errors |
|||
|
|||
Whenever a new static error code is added and used, corresponding documentation |
|||
for the error code should be added to the `doc/api/errors.md` file. This will |
|||
give users a place to go to easily look up the meaning of individual error |
|||
codes. |
|||
|
|||
|
|||
## API |
|||
|
|||
### Class: errors.Error(key[, args...]) |
|||
|
|||
* `key` {String} The static error identifier |
|||
* `args...` {Any} Zero or more optional arguments |
|||
|
|||
```js |
|||
const errors = require('internal/errors'); |
|||
|
|||
var arg1 = 'foo'; |
|||
var arg2 = 'bar'; |
|||
const myError = new errors.Error('KEY', arg1, arg2); |
|||
throw myError; |
|||
``` |
|||
|
|||
The specific error message for the `myError` instance will depend on the |
|||
associated value of `KEY` (see "Adding new errors"). |
|||
|
|||
The `myError` object will have a `code` property equal to the `key` and a |
|||
`name` property equal to `Error[${key}]`. |
|||
|
|||
### Class: errors.TypeError(key[, args...]) |
|||
|
|||
* `key` {String} The static error identifier |
|||
* `args...` {Any} Zero or more optional arguments |
|||
|
|||
```js |
|||
const errors = require('internal/errors'); |
|||
|
|||
var arg1 = 'foo'; |
|||
var arg2 = 'bar'; |
|||
const myError = new errors.TypeError('KEY', arg1, arg2); |
|||
throw myError; |
|||
``` |
|||
|
|||
The specific error message for the `myError` instance will depend on the |
|||
associated value of `KEY` (see "Adding new errors"). |
|||
|
|||
The `myError` object will have a `code` property equal to the `key` and a |
|||
`name` property equal to `TypeError[${key}]`. |
|||
|
|||
### Class: errors.RangeError(key[, args...]) |
|||
|
|||
* `key` {String} The static error identifier |
|||
* `args...` {Any} Zero or more optional arguments |
|||
|
|||
```js |
|||
const errors = require('internal/errors'); |
|||
|
|||
var arg1 = 'foo'; |
|||
var arg2 = 'bar'; |
|||
const myError = new errors.RangeError('KEY', arg1, arg2); |
|||
throw myError; |
|||
``` |
|||
|
|||
The specific error message for the `myError` instance will depend on the |
|||
associated value of `KEY` (see "Adding new errors"). |
|||
|
|||
The `myError` object will have a `code` property equal to the `key` and a |
|||
`name` property equal to `RangeError[${key}]`. |
|||
|
|||
### Method: errors.message(key, args) |
|||
|
|||
* `key` {String} The static error identifier |
|||
* `args` {Array} Zero or more optional arguments passed as an Array |
|||
* Returns: {String} |
|||
|
|||
Returns the formatted error message string for the given `key`. |
@ -0,0 +1,88 @@ |
|||
'use strict'; |
|||
|
|||
// The whole point behind this internal module is to allow Node.js to no
|
|||
// longer be forced to treat every error message change as a semver-major
|
|||
// change. The NodeError classes here all expose a `code` property whose
|
|||
// value statically and permanently identifies the error. While the error
|
|||
// message may change, the code should not.
|
|||
|
|||
const kCode = Symbol('code'); |
|||
const messages = new Map(); |
|||
|
|||
var assert, util; |
|||
function lazyAssert() { |
|||
if (!assert) |
|||
assert = require('assert'); |
|||
return assert; |
|||
} |
|||
|
|||
function lazyUtil() { |
|||
if (!util) |
|||
util = require('util'); |
|||
return util; |
|||
} |
|||
|
|||
function makeNodeError(Base) { |
|||
return class NodeError extends Base { |
|||
constructor(key, ...args) { |
|||
super(message(key, args)); |
|||
this[kCode] = key; |
|||
Error.captureStackTrace(this, NodeError); |
|||
} |
|||
|
|||
get name() { |
|||
return `${super.name}[${this[kCode]}]`; |
|||
} |
|||
|
|||
get code() { |
|||
return this[kCode]; |
|||
} |
|||
}; |
|||
} |
|||
|
|||
function message(key, args) { |
|||
const assert = lazyAssert(); |
|||
assert.strictEqual(typeof key, 'string'); |
|||
const util = lazyUtil(); |
|||
const msg = messages.get(key); |
|||
assert(msg, `An invalid error message key was used: ${key}.`); |
|||
let fmt = util.format; |
|||
if (typeof msg === 'function') { |
|||
fmt = msg; |
|||
} else { |
|||
if (args === undefined || args.length === 0) |
|||
return msg; |
|||
args.unshift(msg); |
|||
} |
|||
return String(fmt.apply(null, args)); |
|||
} |
|||
|
|||
// Utility function for registering the error codes. Only used here. Exported
|
|||
// *only* to allow for testing.
|
|||
function E(sym, val) { |
|||
messages.set(sym, typeof val === 'function' ? val : String(val)); |
|||
} |
|||
|
|||
module.exports = exports = { |
|||
message, |
|||
Error: makeNodeError(Error), |
|||
TypeError: makeNodeError(TypeError), |
|||
RangeError: makeNodeError(RangeError), |
|||
E // This is exported only to facilitate testing.
|
|||
}; |
|||
|
|||
// To declare an error message, use the E(sym, val) function above. The sym
|
|||
// must be an upper case string. The val can be either a function or a string.
|
|||
// The return value of the function must be a string.
|
|||
// Examples:
|
|||
// E('EXAMPLE_KEY1', 'This is the error value');
|
|||
// E('EXAMPLE_KEY2', (a, b) => return `${a} ${b}`);
|
|||
//
|
|||
// Once an error code has been assigned, the code itself MUST NOT change and
|
|||
// any given error code must never be reused to identify a different error.
|
|||
//
|
|||
// Any error code added here should also be added to the documentation
|
|||
//
|
|||
// Note: Please try to keep these in alphabetical order
|
|||
E('ERR_ASSERTION', (msg) => msg); |
|||
// Add new errors from here...
|
@ -0,0 +1,116 @@ |
|||
// Flags: --expose-internals
|
|||
'use strict'; |
|||
|
|||
const common = require('../common'); |
|||
const errors = require('internal/errors'); |
|||
const assert = require('assert'); |
|||
|
|||
errors.E('TEST_ERROR_1', 'Error for testing purposes: %s'); |
|||
errors.E('TEST_ERROR_2', (a, b) => `${a} ${b}`); |
|||
|
|||
const err1 = new errors.Error('TEST_ERROR_1', 'test'); |
|||
const err2 = new errors.TypeError('TEST_ERROR_1', 'test'); |
|||
const err3 = new errors.RangeError('TEST_ERROR_1', 'test'); |
|||
const err4 = new errors.Error('TEST_ERROR_2', 'abc', 'xyz'); |
|||
|
|||
assert(err1 instanceof Error); |
|||
assert.strictEqual(err1.name, 'Error[TEST_ERROR_1]'); |
|||
assert.strictEqual(err1.message, 'Error for testing purposes: test'); |
|||
assert.strictEqual(err1.code, 'TEST_ERROR_1'); |
|||
|
|||
assert(err2 instanceof TypeError); |
|||
assert.strictEqual(err2.name, 'TypeError[TEST_ERROR_1]'); |
|||
assert.strictEqual(err2.message, 'Error for testing purposes: test'); |
|||
assert.strictEqual(err2.code, 'TEST_ERROR_1'); |
|||
|
|||
assert(err3 instanceof RangeError); |
|||
assert.strictEqual(err3.name, 'RangeError[TEST_ERROR_1]'); |
|||
assert.strictEqual(err3.message, 'Error for testing purposes: test'); |
|||
assert.strictEqual(err3.code, 'TEST_ERROR_1'); |
|||
|
|||
assert(err4 instanceof Error); |
|||
assert.strictEqual(err4.name, 'Error[TEST_ERROR_2]'); |
|||
assert.strictEqual(err4.message, 'abc xyz'); |
|||
assert.strictEqual(err4.code, 'TEST_ERROR_2'); |
|||
|
|||
assert.throws( |
|||
() => new errors.Error('TEST_FOO_KEY'), |
|||
/^AssertionError: An invalid error message key was used: TEST_FOO_KEY.$/); |
|||
// Calling it twice yields same result (using the key does not create it)
|
|||
assert.throws( |
|||
() => new errors.Error('TEST_FOO_KEY'), |
|||
/^AssertionError: An invalid error message key was used: TEST_FOO_KEY.$/); |
|||
assert.throws( |
|||
() => new errors.Error(1), |
|||
/^AssertionError: 'number' === 'string'$/); |
|||
assert.throws( |
|||
() => new errors.Error({}), |
|||
/^AssertionError: 'object' === 'string'$/); |
|||
assert.throws( |
|||
() => new errors.Error([]), |
|||
/^AssertionError: 'object' === 'string'$/); |
|||
assert.throws( |
|||
() => new errors.Error(true), |
|||
/^AssertionError: 'boolean' === 'string'$/); |
|||
assert.throws( |
|||
() => new errors.TypeError(1), |
|||
/^AssertionError: 'number' === 'string'$/); |
|||
assert.throws( |
|||
() => new errors.TypeError({}), |
|||
/^AssertionError: 'object' === 'string'$/); |
|||
assert.throws( |
|||
() => new errors.TypeError([]), |
|||
/^AssertionError: 'object' === 'string'$/); |
|||
assert.throws( |
|||
() => new errors.TypeError(true), |
|||
/^AssertionError: 'boolean' === 'string'$/); |
|||
assert.throws( |
|||
() => new errors.RangeError(1), |
|||
/^AssertionError: 'number' === 'string'$/); |
|||
assert.throws( |
|||
() => new errors.RangeError({}), |
|||
/^AssertionError: 'object' === 'string'$/); |
|||
assert.throws( |
|||
() => new errors.RangeError([]), |
|||
/^AssertionError: 'object' === 'string'$/); |
|||
assert.throws( |
|||
() => new errors.RangeError(true), |
|||
/^AssertionError: 'boolean' === 'string'$/); |
|||
|
|||
|
|||
// Tests for common.expectsError
|
|||
assert.doesNotThrow(() => { |
|||
assert.throws(() => { |
|||
throw new errors.TypeError('TEST_ERROR_1', 'a'); |
|||
}, common.expectsError('TEST_ERROR_1')); |
|||
}); |
|||
|
|||
assert.doesNotThrow(() => { |
|||
assert.throws(() => { |
|||
throw new errors.TypeError('TEST_ERROR_1', 'a'); |
|||
}, common.expectsError('TEST_ERROR_1', TypeError, /^Error for testing/)); |
|||
}); |
|||
|
|||
assert.doesNotThrow(() => { |
|||
assert.throws(() => { |
|||
throw new errors.TypeError('TEST_ERROR_1', 'a'); |
|||
}, common.expectsError('TEST_ERROR_1', TypeError)); |
|||
}); |
|||
|
|||
assert.doesNotThrow(() => { |
|||
assert.throws(() => { |
|||
throw new errors.TypeError('TEST_ERROR_1', 'a'); |
|||
}, common.expectsError('TEST_ERROR_1', Error)); |
|||
}); |
|||
|
|||
assert.throws(() => { |
|||
assert.throws(() => { |
|||
throw new errors.TypeError('TEST_ERROR_1', 'a'); |
|||
}, common.expectsError('TEST_ERROR_1', RangeError)); |
|||
}, /^AssertionError: error is not the expected type/); |
|||
|
|||
assert.throws(() => { |
|||
assert.throws(() => { |
|||
throw new errors.TypeError('TEST_ERROR_1', 'a'); |
|||
}, common.expectsError('TEST_ERROR_1', TypeError, /^Error for testing 2/)); |
|||
}, /^AssertionError: error.message does not match/); |
Loading…
Reference in new issue