mirror of
https://github.com/nodejs/node.git
synced 2024-11-27 22:16:50 +01:00
71785889c8
PR-URL: https://github.com/nodejs/node/pull/55044 Reviewed-By: James M Snell <jasnell@gmail.com> Reviewed-By: Chemi Atlow <chemi@atlow.co.il> Reviewed-By: Yagiz Nizipli <yagiz@nizipli.com> Reviewed-By: Trivikram Kamat <trivikr.dev@gmail.com> Reviewed-By: LiviaMedeiros <livia@cirno.name>
813 lines
22 KiB
JavaScript
813 lines
22 KiB
JavaScript
// Originally from narwhal.js (http://narwhaljs.org)
|
|
// Copyright (c) 2009 Thomas Robinson <280north.com>
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
// of this software and associated documentation files (the 'Software'), to
|
|
// deal in the Software without restriction, including without limitation the
|
|
// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
// sell copies of the Software, and to permit persons to whom the Software is
|
|
// furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
// ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
'use strict';
|
|
|
|
const {
|
|
ArrayPrototypeIndexOf,
|
|
ArrayPrototypeJoin,
|
|
ArrayPrototypePush,
|
|
ArrayPrototypeSlice,
|
|
Error,
|
|
NumberIsNaN,
|
|
ObjectAssign,
|
|
ObjectIs,
|
|
ObjectKeys,
|
|
ObjectPrototypeIsPrototypeOf,
|
|
ReflectApply,
|
|
RegExpPrototypeExec,
|
|
String,
|
|
StringPrototypeIndexOf,
|
|
StringPrototypeSlice,
|
|
StringPrototypeSplit,
|
|
} = primordials;
|
|
|
|
const {
|
|
codes: {
|
|
ERR_AMBIGUOUS_ARGUMENT,
|
|
ERR_INVALID_ARG_TYPE,
|
|
ERR_INVALID_ARG_VALUE,
|
|
ERR_INVALID_RETURN_VALUE,
|
|
ERR_MISSING_ARGS,
|
|
},
|
|
} = require('internal/errors');
|
|
const AssertionError = require('internal/assert/assertion_error');
|
|
const { inspect } = require('internal/util/inspect');
|
|
const { isPromise, isRegExp } = require('internal/util/types');
|
|
const { isError, deprecate } = require('internal/util');
|
|
const { innerOk } = require('internal/assert/utils');
|
|
|
|
const CallTracker = require('internal/assert/calltracker');
|
|
const {
|
|
validateFunction,
|
|
} = require('internal/validators');
|
|
|
|
let isDeepEqual;
|
|
let isDeepStrictEqual;
|
|
|
|
function lazyLoadComparison() {
|
|
const comparison = require('internal/util/comparisons');
|
|
isDeepEqual = comparison.isDeepEqual;
|
|
isDeepStrictEqual = comparison.isDeepStrictEqual;
|
|
}
|
|
|
|
let warned = false;
|
|
|
|
// The assert module provides functions that throw
|
|
// AssertionError's when particular conditions are not met. The
|
|
// assert module must conform to the following interface.
|
|
|
|
const assert = module.exports = ok;
|
|
|
|
const NO_EXCEPTION_SENTINEL = {};
|
|
|
|
// All of the following functions must throw an AssertionError
|
|
// when a corresponding condition is not met, with a message that
|
|
// may be undefined if not provided. All assertion methods provide
|
|
// both the actual and expected values to the assertion error for
|
|
// display purposes.
|
|
|
|
function innerFail(obj) {
|
|
if (obj.message instanceof Error) throw obj.message;
|
|
|
|
throw new AssertionError(obj);
|
|
}
|
|
|
|
/**
|
|
* @param {any} actual
|
|
* @param {any} expected
|
|
* @param {string | Error} [message]
|
|
* @param {string} [operator]
|
|
* @param {Function} [stackStartFn]
|
|
*/
|
|
function fail(actual, expected, message, operator, stackStartFn) {
|
|
const argsLen = arguments.length;
|
|
|
|
let internalMessage = false;
|
|
if (actual == null && argsLen <= 1) {
|
|
internalMessage = true;
|
|
message = 'Failed';
|
|
} else if (argsLen === 1) {
|
|
message = actual;
|
|
actual = undefined;
|
|
} else {
|
|
if (warned === false) {
|
|
warned = true;
|
|
process.emitWarning(
|
|
'assert.fail() with more than one argument is deprecated. ' +
|
|
'Please use assert.strictEqual() instead or only pass a message.',
|
|
'DeprecationWarning',
|
|
'DEP0094',
|
|
);
|
|
}
|
|
if (argsLen === 2)
|
|
operator = '!=';
|
|
}
|
|
|
|
if (message instanceof Error) throw message;
|
|
|
|
const errArgs = {
|
|
actual,
|
|
expected,
|
|
operator: operator === undefined ? 'fail' : operator,
|
|
stackStartFn: stackStartFn || fail,
|
|
message,
|
|
};
|
|
const err = new AssertionError(errArgs);
|
|
if (internalMessage) {
|
|
err.generatedMessage = true;
|
|
}
|
|
throw err;
|
|
}
|
|
|
|
assert.fail = fail;
|
|
|
|
// The AssertionError is defined in internal/error.
|
|
assert.AssertionError = AssertionError;
|
|
|
|
/**
|
|
* Pure assertion tests whether a value is truthy, as determined
|
|
* by !!value.
|
|
* @param {...any} args
|
|
* @returns {void}
|
|
*/
|
|
function ok(...args) {
|
|
innerOk(ok, args.length, ...args);
|
|
}
|
|
assert.ok = ok;
|
|
|
|
/**
|
|
* The equality assertion tests shallow, coercive equality with ==.
|
|
* @param {any} actual
|
|
* @param {any} expected
|
|
* @param {string | Error} [message]
|
|
* @returns {void}
|
|
*/
|
|
/* eslint-disable no-restricted-properties */
|
|
assert.equal = function equal(actual, expected, message) {
|
|
if (arguments.length < 2) {
|
|
throw new ERR_MISSING_ARGS('actual', 'expected');
|
|
}
|
|
// eslint-disable-next-line eqeqeq
|
|
if (actual != expected && (!NumberIsNaN(actual) || !NumberIsNaN(expected))) {
|
|
innerFail({
|
|
actual,
|
|
expected,
|
|
message,
|
|
operator: '==',
|
|
stackStartFn: equal,
|
|
});
|
|
}
|
|
};
|
|
|
|
/**
|
|
* The non-equality assertion tests for whether two objects are not
|
|
* equal with !=.
|
|
* @param {any} actual
|
|
* @param {any} expected
|
|
* @param {string | Error} [message]
|
|
* @returns {void}
|
|
*/
|
|
assert.notEqual = function notEqual(actual, expected, message) {
|
|
if (arguments.length < 2) {
|
|
throw new ERR_MISSING_ARGS('actual', 'expected');
|
|
}
|
|
// eslint-disable-next-line eqeqeq
|
|
if (actual == expected || (NumberIsNaN(actual) && NumberIsNaN(expected))) {
|
|
innerFail({
|
|
actual,
|
|
expected,
|
|
message,
|
|
operator: '!=',
|
|
stackStartFn: notEqual,
|
|
});
|
|
}
|
|
};
|
|
|
|
/**
|
|
* The deep equivalence assertion tests a deep equality relation.
|
|
* @param {any} actual
|
|
* @param {any} expected
|
|
* @param {string | Error} [message]
|
|
* @returns {void}
|
|
*/
|
|
assert.deepEqual = function deepEqual(actual, expected, message) {
|
|
if (arguments.length < 2) {
|
|
throw new ERR_MISSING_ARGS('actual', 'expected');
|
|
}
|
|
if (isDeepEqual === undefined) lazyLoadComparison();
|
|
if (!isDeepEqual(actual, expected)) {
|
|
innerFail({
|
|
actual,
|
|
expected,
|
|
message,
|
|
operator: 'deepEqual',
|
|
stackStartFn: deepEqual,
|
|
});
|
|
}
|
|
};
|
|
|
|
/**
|
|
* The deep non-equivalence assertion tests for any deep inequality.
|
|
* @param {any} actual
|
|
* @param {any} expected
|
|
* @param {string | Error} [message]
|
|
* @returns {void}
|
|
*/
|
|
assert.notDeepEqual = function notDeepEqual(actual, expected, message) {
|
|
if (arguments.length < 2) {
|
|
throw new ERR_MISSING_ARGS('actual', 'expected');
|
|
}
|
|
if (isDeepEqual === undefined) lazyLoadComparison();
|
|
if (isDeepEqual(actual, expected)) {
|
|
innerFail({
|
|
actual,
|
|
expected,
|
|
message,
|
|
operator: 'notDeepEqual',
|
|
stackStartFn: notDeepEqual,
|
|
});
|
|
}
|
|
};
|
|
/* eslint-enable */
|
|
|
|
/**
|
|
* The deep strict equivalence assertion tests a deep strict equality
|
|
* relation.
|
|
* @param {any} actual
|
|
* @param {any} expected
|
|
* @param {string | Error} [message]
|
|
* @returns {void}
|
|
*/
|
|
assert.deepStrictEqual = function deepStrictEqual(actual, expected, message) {
|
|
if (arguments.length < 2) {
|
|
throw new ERR_MISSING_ARGS('actual', 'expected');
|
|
}
|
|
if (isDeepEqual === undefined) lazyLoadComparison();
|
|
if (!isDeepStrictEqual(actual, expected)) {
|
|
innerFail({
|
|
actual,
|
|
expected,
|
|
message,
|
|
operator: 'deepStrictEqual',
|
|
stackStartFn: deepStrictEqual,
|
|
});
|
|
}
|
|
};
|
|
|
|
/**
|
|
* The deep strict non-equivalence assertion tests for any deep strict
|
|
* inequality.
|
|
* @param {any} actual
|
|
* @param {any} expected
|
|
* @param {string | Error} [message]
|
|
* @returns {void}
|
|
*/
|
|
assert.notDeepStrictEqual = notDeepStrictEqual;
|
|
function notDeepStrictEqual(actual, expected, message) {
|
|
if (arguments.length < 2) {
|
|
throw new ERR_MISSING_ARGS('actual', 'expected');
|
|
}
|
|
if (isDeepEqual === undefined) lazyLoadComparison();
|
|
if (isDeepStrictEqual(actual, expected)) {
|
|
innerFail({
|
|
actual,
|
|
expected,
|
|
message,
|
|
operator: 'notDeepStrictEqual',
|
|
stackStartFn: notDeepStrictEqual,
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* The strict equivalence assertion tests a strict equality relation.
|
|
* @param {any} actual
|
|
* @param {any} expected
|
|
* @param {string | Error} [message]
|
|
* @returns {void}
|
|
*/
|
|
assert.strictEqual = function strictEqual(actual, expected, message) {
|
|
if (arguments.length < 2) {
|
|
throw new ERR_MISSING_ARGS('actual', 'expected');
|
|
}
|
|
if (!ObjectIs(actual, expected)) {
|
|
innerFail({
|
|
actual,
|
|
expected,
|
|
message,
|
|
operator: 'strictEqual',
|
|
stackStartFn: strictEqual,
|
|
});
|
|
}
|
|
};
|
|
|
|
/**
|
|
* The strict non-equivalence assertion tests for any strict inequality.
|
|
* @param {any} actual
|
|
* @param {any} expected
|
|
* @param {string | Error} [message]
|
|
* @returns {void}
|
|
*/
|
|
assert.notStrictEqual = function notStrictEqual(actual, expected, message) {
|
|
if (arguments.length < 2) {
|
|
throw new ERR_MISSING_ARGS('actual', 'expected');
|
|
}
|
|
if (ObjectIs(actual, expected)) {
|
|
innerFail({
|
|
actual,
|
|
expected,
|
|
message,
|
|
operator: 'notStrictEqual',
|
|
stackStartFn: notStrictEqual,
|
|
});
|
|
}
|
|
};
|
|
|
|
class Comparison {
|
|
constructor(obj, keys, actual) {
|
|
for (const key of keys) {
|
|
if (key in obj) {
|
|
if (actual !== undefined &&
|
|
typeof actual[key] === 'string' &&
|
|
isRegExp(obj[key]) &&
|
|
RegExpPrototypeExec(obj[key], actual[key]) !== null) {
|
|
this[key] = actual[key];
|
|
} else {
|
|
this[key] = obj[key];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function compareExceptionKey(actual, expected, key, message, keys, fn) {
|
|
if (!(key in actual) || !isDeepStrictEqual(actual[key], expected[key])) {
|
|
if (!message) {
|
|
// Create placeholder objects to create a nice output.
|
|
const a = new Comparison(actual, keys);
|
|
const b = new Comparison(expected, keys, actual);
|
|
|
|
const err = new AssertionError({
|
|
actual: a,
|
|
expected: b,
|
|
operator: 'deepStrictEqual',
|
|
stackStartFn: fn,
|
|
});
|
|
err.actual = actual;
|
|
err.expected = expected;
|
|
err.operator = fn.name;
|
|
throw err;
|
|
}
|
|
innerFail({
|
|
actual,
|
|
expected,
|
|
message,
|
|
operator: fn.name,
|
|
stackStartFn: fn,
|
|
});
|
|
}
|
|
}
|
|
|
|
function expectedException(actual, expected, message, fn) {
|
|
let generatedMessage = false;
|
|
let throwError = false;
|
|
|
|
if (typeof expected !== 'function') {
|
|
// Handle regular expressions.
|
|
if (isRegExp(expected)) {
|
|
const str = String(actual);
|
|
if (RegExpPrototypeExec(expected, str) !== null)
|
|
return;
|
|
|
|
if (!message) {
|
|
generatedMessage = true;
|
|
message = 'The input did not match the regular expression ' +
|
|
`${inspect(expected)}. Input:\n\n${inspect(str)}\n`;
|
|
}
|
|
throwError = true;
|
|
// Handle primitives properly.
|
|
} else if (typeof actual !== 'object' || actual === null) {
|
|
const err = new AssertionError({
|
|
actual,
|
|
expected,
|
|
message,
|
|
operator: 'deepStrictEqual',
|
|
stackStartFn: fn,
|
|
});
|
|
err.operator = fn.name;
|
|
throw err;
|
|
} else {
|
|
// Handle validation objects.
|
|
const keys = ObjectKeys(expected);
|
|
// Special handle errors to make sure the name and the message are
|
|
// compared as well.
|
|
if (expected instanceof Error) {
|
|
ArrayPrototypePush(keys, 'name', 'message');
|
|
} else if (keys.length === 0) {
|
|
throw new ERR_INVALID_ARG_VALUE('error',
|
|
expected, 'may not be an empty object');
|
|
}
|
|
if (isDeepEqual === undefined) lazyLoadComparison();
|
|
for (const key of keys) {
|
|
if (typeof actual[key] === 'string' &&
|
|
isRegExp(expected[key]) &&
|
|
RegExpPrototypeExec(expected[key], actual[key]) !== null) {
|
|
continue;
|
|
}
|
|
compareExceptionKey(actual, expected, key, message, keys, fn);
|
|
}
|
|
return;
|
|
}
|
|
// Guard instanceof against arrow functions as they don't have a prototype.
|
|
// Check for matching Error classes.
|
|
} else if (expected.prototype !== undefined && actual instanceof expected) {
|
|
return;
|
|
} else if (ObjectPrototypeIsPrototypeOf(Error, expected)) {
|
|
if (!message) {
|
|
generatedMessage = true;
|
|
message = 'The error is expected to be an instance of ' +
|
|
`"${expected.name}". Received `;
|
|
if (isError(actual)) {
|
|
const name = (actual.constructor?.name) ||
|
|
actual.name;
|
|
if (expected.name === name) {
|
|
message += 'an error with identical name but a different prototype.';
|
|
} else {
|
|
message += `"${name}"`;
|
|
}
|
|
if (actual.message) {
|
|
message += `\n\nError message:\n\n${actual.message}`;
|
|
}
|
|
} else {
|
|
message += `"${inspect(actual, { depth: -1 })}"`;
|
|
}
|
|
}
|
|
throwError = true;
|
|
} else {
|
|
// Check validation functions return value.
|
|
const res = ReflectApply(expected, {}, [actual]);
|
|
if (res !== true) {
|
|
if (!message) {
|
|
generatedMessage = true;
|
|
const name = expected.name ? `"${expected.name}" ` : '';
|
|
message = `The ${name}validation function is expected to return` +
|
|
` "true". Received ${inspect(res)}`;
|
|
|
|
if (isError(actual)) {
|
|
message += `\n\nCaught error:\n\n${actual}`;
|
|
}
|
|
}
|
|
throwError = true;
|
|
}
|
|
}
|
|
|
|
if (throwError) {
|
|
const err = new AssertionError({
|
|
actual,
|
|
expected,
|
|
message,
|
|
operator: fn.name,
|
|
stackStartFn: fn,
|
|
});
|
|
err.generatedMessage = generatedMessage;
|
|
throw err;
|
|
}
|
|
}
|
|
|
|
function getActual(fn) {
|
|
validateFunction(fn, 'fn');
|
|
try {
|
|
fn();
|
|
} catch (e) {
|
|
return e;
|
|
}
|
|
return NO_EXCEPTION_SENTINEL;
|
|
}
|
|
|
|
function checkIsPromise(obj) {
|
|
// Accept native ES6 promises and promises that are implemented in a similar
|
|
// way. Do not accept thenables that use a function as `obj` and that have no
|
|
// `catch` handler.
|
|
return isPromise(obj) ||
|
|
(obj !== null && typeof obj === 'object' &&
|
|
typeof obj.then === 'function' &&
|
|
typeof obj.catch === 'function');
|
|
}
|
|
|
|
async function waitForActual(promiseFn) {
|
|
let resultPromise;
|
|
if (typeof promiseFn === 'function') {
|
|
// Return a rejected promise if `promiseFn` throws synchronously.
|
|
resultPromise = promiseFn();
|
|
// Fail in case no promise is returned.
|
|
if (!checkIsPromise(resultPromise)) {
|
|
throw new ERR_INVALID_RETURN_VALUE('instance of Promise',
|
|
'promiseFn', resultPromise);
|
|
}
|
|
} else if (checkIsPromise(promiseFn)) {
|
|
resultPromise = promiseFn;
|
|
} else {
|
|
throw new ERR_INVALID_ARG_TYPE(
|
|
'promiseFn', ['Function', 'Promise'], promiseFn);
|
|
}
|
|
|
|
try {
|
|
await resultPromise;
|
|
} catch (e) {
|
|
return e;
|
|
}
|
|
return NO_EXCEPTION_SENTINEL;
|
|
}
|
|
|
|
function expectsError(stackStartFn, actual, error, message) {
|
|
if (typeof error === 'string') {
|
|
if (arguments.length === 4) {
|
|
throw new ERR_INVALID_ARG_TYPE('error',
|
|
['Object', 'Error', 'Function', 'RegExp'],
|
|
error);
|
|
}
|
|
if (typeof actual === 'object' && actual !== null) {
|
|
if (actual.message === error) {
|
|
throw new ERR_AMBIGUOUS_ARGUMENT(
|
|
'error/message',
|
|
`The error message "${actual.message}" is identical to the message.`,
|
|
);
|
|
}
|
|
} else if (actual === error) {
|
|
throw new ERR_AMBIGUOUS_ARGUMENT(
|
|
'error/message',
|
|
`The error "${actual}" is identical to the message.`,
|
|
);
|
|
}
|
|
message = error;
|
|
error = undefined;
|
|
} else if (error != null &&
|
|
typeof error !== 'object' &&
|
|
typeof error !== 'function') {
|
|
throw new ERR_INVALID_ARG_TYPE('error',
|
|
['Object', 'Error', 'Function', 'RegExp'],
|
|
error);
|
|
}
|
|
|
|
if (actual === NO_EXCEPTION_SENTINEL) {
|
|
let details = '';
|
|
if (error?.name) {
|
|
details += ` (${error.name})`;
|
|
}
|
|
details += message ? `: ${message}` : '.';
|
|
const fnType = stackStartFn === assert.rejects ? 'rejection' : 'exception';
|
|
innerFail({
|
|
actual: undefined,
|
|
expected: error,
|
|
operator: stackStartFn.name,
|
|
message: `Missing expected ${fnType}${details}`,
|
|
stackStartFn,
|
|
});
|
|
}
|
|
|
|
if (!error)
|
|
return;
|
|
|
|
expectedException(actual, error, message, stackStartFn);
|
|
}
|
|
|
|
function hasMatchingError(actual, expected) {
|
|
if (typeof expected !== 'function') {
|
|
if (isRegExp(expected)) {
|
|
const str = String(actual);
|
|
return RegExpPrototypeExec(expected, str) !== null;
|
|
}
|
|
throw new ERR_INVALID_ARG_TYPE(
|
|
'expected', ['Function', 'RegExp'], expected,
|
|
);
|
|
}
|
|
// Guard instanceof against arrow functions as they don't have a prototype.
|
|
if (expected.prototype !== undefined && actual instanceof expected) {
|
|
return true;
|
|
}
|
|
if (ObjectPrototypeIsPrototypeOf(Error, expected)) {
|
|
return false;
|
|
}
|
|
return ReflectApply(expected, {}, [actual]) === true;
|
|
}
|
|
|
|
function expectsNoError(stackStartFn, actual, error, message) {
|
|
if (actual === NO_EXCEPTION_SENTINEL)
|
|
return;
|
|
|
|
if (typeof error === 'string') {
|
|
message = error;
|
|
error = undefined;
|
|
}
|
|
|
|
if (!error || hasMatchingError(actual, error)) {
|
|
const details = message ? `: ${message}` : '.';
|
|
const fnType = stackStartFn === assert.doesNotReject ?
|
|
'rejection' : 'exception';
|
|
innerFail({
|
|
actual,
|
|
expected: error,
|
|
operator: stackStartFn.name,
|
|
message: `Got unwanted ${fnType}${details}\n` +
|
|
`Actual message: "${actual?.message}"`,
|
|
stackStartFn,
|
|
});
|
|
}
|
|
throw actual;
|
|
}
|
|
|
|
/**
|
|
* Expects the function `promiseFn` to throw an error.
|
|
* @param {() => any} promiseFn
|
|
* @param {...any} [args]
|
|
* @returns {void}
|
|
*/
|
|
assert.throws = function throws(promiseFn, ...args) {
|
|
expectsError(throws, getActual(promiseFn), ...args);
|
|
};
|
|
|
|
/**
|
|
* Expects `promiseFn` function or its value to reject.
|
|
* @param {() => Promise<any>} promiseFn
|
|
* @param {...any} [args]
|
|
* @returns {Promise<void>}
|
|
*/
|
|
assert.rejects = async function rejects(promiseFn, ...args) {
|
|
expectsError(rejects, await waitForActual(promiseFn), ...args);
|
|
};
|
|
|
|
/**
|
|
* Asserts that the function `fn` does not throw an error.
|
|
* @param {() => any} fn
|
|
* @param {...any} [args]
|
|
* @returns {void}
|
|
*/
|
|
assert.doesNotThrow = function doesNotThrow(fn, ...args) {
|
|
expectsNoError(doesNotThrow, getActual(fn), ...args);
|
|
};
|
|
|
|
/**
|
|
* Expects `fn` or its value to not reject.
|
|
* @param {() => Promise<any>} fn
|
|
* @param {...any} [args]
|
|
* @returns {Promise<void>}
|
|
*/
|
|
assert.doesNotReject = async function doesNotReject(fn, ...args) {
|
|
expectsNoError(doesNotReject, await waitForActual(fn), ...args);
|
|
};
|
|
|
|
/**
|
|
* Throws `value` if the value is not `null` or `undefined`.
|
|
* @param {any} err
|
|
* @returns {void}
|
|
*/
|
|
assert.ifError = function ifError(err) {
|
|
if (err !== null && err !== undefined) {
|
|
let message = 'ifError got unwanted exception: ';
|
|
if (typeof err === 'object' && typeof err.message === 'string') {
|
|
if (err.message.length === 0 && err.constructor) {
|
|
message += err.constructor.name;
|
|
} else {
|
|
message += err.message;
|
|
}
|
|
} else {
|
|
message += inspect(err);
|
|
}
|
|
|
|
const newErr = new AssertionError({
|
|
actual: err,
|
|
expected: null,
|
|
operator: 'ifError',
|
|
message,
|
|
stackStartFn: ifError,
|
|
});
|
|
|
|
// Make sure we actually have a stack trace!
|
|
const origStack = err.stack;
|
|
|
|
if (typeof origStack === 'string') {
|
|
// This will remove any duplicated frames from the error frames taken
|
|
// from within `ifError` and add the original error frames to the newly
|
|
// created ones.
|
|
const origStackStart = StringPrototypeIndexOf(origStack, '\n at');
|
|
if (origStackStart !== -1) {
|
|
const originalFrames = StringPrototypeSplit(
|
|
StringPrototypeSlice(origStack, origStackStart + 1),
|
|
'\n',
|
|
);
|
|
// Filter all frames existing in err.stack.
|
|
let newFrames = StringPrototypeSplit(newErr.stack, '\n');
|
|
for (const errFrame of originalFrames) {
|
|
// Find the first occurrence of the frame.
|
|
const pos = ArrayPrototypeIndexOf(newFrames, errFrame);
|
|
if (pos !== -1) {
|
|
// Only keep new frames.
|
|
newFrames = ArrayPrototypeSlice(newFrames, 0, pos);
|
|
break;
|
|
}
|
|
}
|
|
const stackStart = ArrayPrototypeJoin(newFrames, '\n');
|
|
const stackEnd = ArrayPrototypeJoin(originalFrames, '\n');
|
|
newErr.stack = `${stackStart}\n${stackEnd}`;
|
|
}
|
|
}
|
|
|
|
throw newErr;
|
|
}
|
|
};
|
|
|
|
function internalMatch(string, regexp, message, fn) {
|
|
if (!isRegExp(regexp)) {
|
|
throw new ERR_INVALID_ARG_TYPE(
|
|
'regexp', 'RegExp', regexp,
|
|
);
|
|
}
|
|
const match = fn === assert.match;
|
|
if (typeof string !== 'string' ||
|
|
RegExpPrototypeExec(regexp, string) !== null !== match) {
|
|
if (message instanceof Error) {
|
|
throw message;
|
|
}
|
|
|
|
const generatedMessage = !message;
|
|
|
|
// 'The input was expected to not match the regular expression ' +
|
|
message ||= (typeof string !== 'string' ?
|
|
'The "string" argument must be of type string. Received type ' +
|
|
`${typeof string} (${inspect(string)})` :
|
|
(match ?
|
|
'The input did not match the regular expression ' :
|
|
'The input was expected to not match the regular expression ') +
|
|
`${inspect(regexp)}. Input:\n\n${inspect(string)}\n`);
|
|
const err = new AssertionError({
|
|
actual: string,
|
|
expected: regexp,
|
|
message,
|
|
operator: fn.name,
|
|
stackStartFn: fn,
|
|
});
|
|
err.generatedMessage = generatedMessage;
|
|
throw err;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Expects the `string` input to match the regular expression.
|
|
* @param {string} string
|
|
* @param {RegExp} regexp
|
|
* @param {string | Error} [message]
|
|
* @returns {void}
|
|
*/
|
|
assert.match = function match(string, regexp, message) {
|
|
internalMatch(string, regexp, message, match);
|
|
};
|
|
|
|
/**
|
|
* Expects the `string` input not to match the regular expression.
|
|
* @param {string} string
|
|
* @param {RegExp} regexp
|
|
* @param {string | Error} [message]
|
|
* @returns {void}
|
|
*/
|
|
assert.doesNotMatch = function doesNotMatch(string, regexp, message) {
|
|
internalMatch(string, regexp, message, doesNotMatch);
|
|
};
|
|
|
|
assert.CallTracker = deprecate(CallTracker, 'assert.CallTracker is deprecated.', 'DEP0173');
|
|
|
|
/**
|
|
* Expose a strict only variant of assert.
|
|
* @param {...any} args
|
|
* @returns {void}
|
|
*/
|
|
function strict(...args) {
|
|
innerOk(strict, args.length, ...args);
|
|
}
|
|
|
|
assert.strict = ObjectAssign(strict, assert, {
|
|
equal: assert.strictEqual,
|
|
deepEqual: assert.deepStrictEqual,
|
|
notEqual: assert.notStrictEqual,
|
|
notDeepEqual: assert.notDeepStrictEqual,
|
|
});
|
|
|
|
assert.strict.strict = assert.strict;
|