diff --git a/doc/api/assert.md b/doc/api/assert.md index 70f4ac6c6db5bd..06b27fa261d3a3 100644 --- a/doc/api/assert.md +++ b/doc/api/assert.md @@ -2548,6 +2548,232 @@ assert.throws(throwingFirst, /Second$/); Due to the confusing error-prone notation, avoid a string as the second argument. +## `assert.partialDeepEqual(actual, expected[, message])` + + + +* `actual` {any} +* `expected` {any} +* `message` {string|Error} + +[`assert.partialDeepEqual()`][] evaluates the equivalence between the `actual` and `expected` parameters by +performing a deep comparison. This function ensures that all properties defined +in the `expected` parameter match those in the `actual` parameter in +both value and type, allowing type coercion. The main difference with [`assert.deepEqual()`][] is that +[`assert.partialDeepEqual()`][] does not require all properties in the `actual` parameter to be present in the +`expected` parameter. + +```mjs +import assert from 'node:assert'; + +assert.partialDeepEqual({ a: 1, b: '2' }, { a: 1, b: 2 }); +// OK + +assert.partialDeepEqual({ a: 1, b: '2', c: 3 }, { a: 1, b: 2 }); +// OK + +assert.partialDeepEqual({ a: { b: { c: '1' } } }, { a: { b: { c: 1 } } }); +// OK + +assert.partialDeepEqual(new Set(['value1', 'value2']), new Set(['value1', 'value2'])); +// OK + +assert.partialDeepEqual(new Map([['key1', 'value1']]), new Map([['key1', 'value1']])); +// OK + +assert.partialDeepEqual(new Uint8Array([1, 2, 3]), new Uint8Array([1, 2, 3])); +// OK + +assert.partialDeepEqual(/abc/, /abc/); +// OK + +assert.partialDeepEqual(new Date(0), new Date(0)); +// OK + +assert.partialDeepEqual({ a: 1 }, { a: 1, b: 2 }); +// AssertionError + +assert.partialDeepEqual({ a: 1, b: true }, { a: 1, b: 'true' }); +// AssertionError + +assert.partialDeepEqual({ a: { b: 2 } }, { a: { b: 2, c: 3 } }); +// AssertionError +``` + +```cjs +const assert = require('node:assert'); + +assert.partialDeepEqual({ a: 1, b: '2' }, { a: 1, b: 2 }); +// OK + +assert.partialDeepEqual({ a: 1, b: '2', c: 3 }, { a: 1, b: 2 }); +// OK + +assert.partialDeepEqual({ a: { b: { c: '1' } } }, { a: { b: { c: 1 } } }); +// OK + +assert.partialDeepEqual(new Set(['value1', 'value2']), new Set(['value1', 'value2'])); +// OK + +assert.partialDeepEqual(new Map([['key1', 'value1']]), new Map([['key1', 'value1']])); +// OK + +assert.partialDeepEqual(new Uint8Array([1, 2, 3]), new Uint8Array([1, 2, 3])); +// OK + +assert.partialDeepEqual(/abc/, /abc/); +// OK + +assert.partialDeepEqual(new Date(0), new Date(0)); +// OK + +assert.partialDeepEqual({ a: 1 }, { a: 1, b: 2 }); +// AssertionError: Expected key b + +assert.partialDeepEqual({ a: 1, b: true }, { a: 1, b: 'true' }); +// AssertionError + +assert.partialDeepEqual({ a: { b: 2, d: 4 } }, { a: { b: 2, c: 3 } }); +// AssertionError: Expected key c +``` + +If the values or keys are not equal in the `expected` parameter, an [`AssertionError`][] is thrown with a `message` +property set equal to the value of the `message` parameter. If the `message` +parameter is undefined, a default error message is assigned. If the `message` +parameter is an instance of an [`Error`][] then it will be thrown instead of the +`AssertionError`. + +## `assert.partialDeepStrictEqual(actual, expected[, message])` + + + +* `actual` {any} +* `expected` {any} +* `message` {string|Error} + +[`assert.partialDeepStrictEqual()`][] Assesses the equivalence between the `actual` and `expected` parameters through a +deep comparison, ensuring that all properties in the `expected` parameter are +present in the `actual` parameter with equivalent values, not allowing type coercion. +The main difference with [`assert.deepStrictEqual()`][] is that [`assert.partialDeepStrictEqual()`][] does not require +all properties in the `actual` parameter to be present in the `expected` parameter. + +```mjs +import assert from 'node:assert'; + +assert.partialDeepStrictEqual({ a: 1, b: 2 }, { a: 1, b: 2 }); +// OK + +assert.partialDeepStrictEqual({ a: { b: { c: 1 } } }, { a: { b: { c: 1 } } }); +// OK + +assert.partialDeepStrictEqual({ a: 1, b: 2, c: 3 }, { a: 1, b: 2 }); +// OK + +assert.partialDeepStrictEqual(new Set(['value1', 'value2']), new Set(['value1', 'value2'])); +// OK + +assert.partialDeepStrictEqual(new Map([['key1', 'value1']]), new Map([['key1', 'value1']])); +// OK + +assert.partialDeepStrictEqual(new Uint8Array([1, 2, 3]), new Uint8Array([1, 2, 3])); +// OK + +assert.partialDeepStrictEqual(/abc/, /abc/); +// OK + +assert.partialDeepStrictEqual(new Date(0), new Date(0)); +// OK + +assert.partialDeepStrictEqual({ a: 1 }, { a: 1, b: 2 }); +// AssertionError + +assert.partialDeepStrictEqual({ a: 1, b: '2' }, { a: 1, b: 2 }); +// AssertionError + +assert.partialDeepStrictEqual({ a: { b: 2 } }, { a: { b: '2' } }); +// AssertionError +``` + +```cjs +const assert = require('node:assert'); + +assert.partialDeepStrictEqual({ a: 1, b: 2 }, { a: 1, b: 2 }); +// OK + +assert.partialDeepStrictEqual({ a: { b: { c: 1 } } }, { a: { b: { c: 1 } } }); +// OK + +assert.partialDeepStrictEqual({ a: 1, b: 2, c: 3 }, { a: 1, b: 2 }); +// OK + +assert.partialDeepStrictEqual({ a: 1 }, { a: 1, b: 2 }); +// AssertionError + +assert.partialDeepStrictEqual({ a: 1, b: '2' }, { a: 1, b: 2 }); +// AssertionError + +assert.partialDeepStrictEqual({ a: { b: 2 } }, { a: { b: '2' } }); +// AssertionError +``` + +## `assert.includes(actual, expected[, message])` + + + +* `actual` {Array | string} +* `expected` {any} +* `message` {string|Error} + +[`assert.includes()`][] compares the `actual` and `expected` parameters to determine if the `expected` +parameter is included in the `actual` parameter; the comparison is done with type coercion. +The `actual` and the `expected` parameters can be +either an array or a string. If the `actual` parameter is an array, +the `expected` parameter must be an array and vice versa for strings. + +```mjs +import assert from 'node:assert'; + +assert.includes([1, 2, 3], 2); +// OK + +assert.includes('Hello World!', 'World'); +// OK + +assert.includes([1, 2, 3], '2'); +// AssertionError + +assert.includes('Hello World!', 'Node.js'); +// AssertionError +``` + +```cjs +const assert = require('node:assert'); + +assert.includes([1, 2, 3], 2); +// OK + +assert.includes('Hello World!', 'World'); +// OK + +assert.includes([1, 2, 3], '2'); +// AssertionError + +assert.includes('Hello World!', 'Node.js'); +// AssertionError +``` + +If the assertion fails, an [`AssertionError`][] is thrown with a `message` +property set equal to the value of the `message` parameter. If the `message` +parameter is undefined, a default error message is assigned. If the `message` +parameter is an instance of an [`Error`][] then it will be thrown instead of the +`AssertionError`. + [Object wrappers]: https://developer.mozilla.org/en-US/docs/Glossary/Primitive#Primitive_wrapper_objects_in_JavaScript [Object.prototype.toString()]: https://tc39.github.io/ecma262/#sec-object.prototype.tostring [`!=` operator]: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Operators/Inequality @@ -2571,11 +2797,14 @@ argument. [`assert.deepStrictEqual()`]: #assertdeepstrictequalactual-expected-message [`assert.doesNotThrow()`]: #assertdoesnotthrowfn-error-message [`assert.equal()`]: #assertequalactual-expected-message +[`assert.includes()`]: #assertincludesactual-expected-message [`assert.notDeepEqual()`]: #assertnotdeepequalactual-expected-message [`assert.notDeepStrictEqual()`]: #assertnotdeepstrictequalactual-expected-message [`assert.notEqual()`]: #assertnotequalactual-expected-message [`assert.notStrictEqual()`]: #assertnotstrictequalactual-expected-message [`assert.ok()`]: #assertokvalue-message +[`assert.partialDeepEqual()`]: #assertpartialdeepequalactual-expected-message +[`assert.partialDeepStrictEqual()`]: #assertpartialdeepstrictequalactual-expected-message [`assert.strictEqual()`]: #assertstrictequalactual-expected-message [`assert.throws()`]: #assertthrowsfn-error-message [`getColorDepth()`]: tty.md#writestreamgetcolordepthenv diff --git a/lib/assert.js b/lib/assert.js index 9e54983c8682a3..1a15b51f20357e 100644 --- a/lib/assert.js +++ b/lib/assert.js @@ -21,22 +21,36 @@ 'use strict'; const { + ArrayIsArray, + ArrayPrototypeEvery, + ArrayPrototypeIncludes, ArrayPrototypeIndexOf, ArrayPrototypeJoin, ArrayPrototypePush, ArrayPrototypeSlice, + ArrayPrototypeSome, Error, + FunctionPrototypeCall, + MapPrototypeGet, + MapPrototypeHas, NumberIsNaN, ObjectAssign, ObjectIs, ObjectKeys, ObjectPrototypeIsPrototypeOf, ReflectApply, + ReflectHas, + ReflectOwnKeys, RegExpPrototypeExec, + SafeMap, + SafeSet, + SetPrototypeHas, String, + StringPrototypeIncludes, StringPrototypeIndexOf, StringPrototypeSlice, StringPrototypeSplit, + SymbolIterator, } = primordials; const { @@ -50,7 +64,7 @@ const { } = require('internal/errors'); const AssertionError = require('internal/assert/assertion_error'); const { inspect } = require('internal/util/inspect'); -const { isPromise, isRegExp } = require('internal/util/types'); +const { isPromise, isRegExp, isMap, isSet, isDate } = require('internal/util/types'); const { isError, deprecate } = require('internal/util'); const { innerOk } = require('internal/assert/utils'); @@ -341,6 +355,194 @@ assert.notStrictEqual = function notStrictEqual(actual, expected, message) { } }; +function isSpecial(obj) { + return obj == null || typeof obj !== 'object' || isError(obj) || isRegExp(obj) || isDate(obj); +} + +/** + * Compares two objects or values recursively to check if they are equal. + * @param {any} actual - The actual value to compare. + * @param {any} expected - The expected value to compare. + * @param {boolean} [loose=false] - Whether to use loose comparison (==) or strict comparison (===). Defaults to false. + * @param {Set} [comparedObjects=new Set()] - Set to track compared objects for handling circular references. + * @returns {boolean} - Returns `true` if the actual value matches the expected value, otherwise `false`. + * @example + * // Loose comparison (default) + * compareBranch({a: 1, b: 2, c: 3}, {a: 1, b: '2'}); // true + * + * // Strict comparison + * compareBranch({a: 1, b: 2, c: 3}, {a: 1, b: 2}, true); // true + */ +function compareBranch( + actual, + expected, + loose = false, + comparedObjects = new SafeSet(), +) { + // Check for Map object equality + if (isMap(actual) && isMap(expected)) { + if (actual.size !== expected.size) { + return false; + } + const safeIterator = FunctionPrototypeCall(SafeMap.prototype[SymbolIterator], actual); + + for (const { 0: key, 1: val } of safeIterator) { + if (!MapPrototypeHas(expected, key)) { + return false; + } + if (!compareBranch(val, MapPrototypeGet(expected, key), loose, comparedObjects)) { + return false; + } + } + return true; + } + + // Check for Set object equality + // TODO(aduh95): switch to `SetPrototypeIsSubsetOf` when it's available + if (isSet(actual) && isSet(expected)) { + if (actual.size > expected.size) { + return false; + } + const safeIterator = FunctionPrototypeCall(SafeSet.prototype[SymbolIterator], actual); + + for (const item of safeIterator) { + if (!SetPrototypeHas(expected, item)) { + return false; + } + } + return true; + } + + // Check if expected array is a subset of actual array + if (ArrayIsArray(actual) && ArrayIsArray(expected)) { + if (expected.length > actual.length) { + return false; + } + if (typeof actual === 'string') { + return StringPrototypeIncludes(actual, expected); + } + + if (loose) { + return ArrayPrototypeEvery(expected, (item) => ArrayPrototypeSome(actual, (actualItem) => { + // eslint-disable-next-line eqeqeq + return actualItem == item; + })); + } + + return ArrayPrototypeEvery(expected, (item) => ArrayPrototypeIncludes(actual, item)); + } + + // Comparison done when at least one of the values is not an object + if (isSpecial(actual) || isSpecial(expected)) { + if (isDeepEqual === undefined) { + lazyLoadComparison(); + } + return loose ? isDeepEqual(actual, expected) : isDeepStrictEqual(actual, expected); + } + + // Use Reflect.ownKeys() instead of Object.keys() to include symbol properties + const keysExpected = ReflectOwnKeys(expected); + + // Handle circular references + if (SetPrototypeHas(comparedObjects, actual)) { + return true; + } + comparedObjects.add(actual); + + // Check if all expected keys and values match + for (let i = 0; i < keysExpected.length; i++) { + const key = keysExpected[i]; + assert( + ReflectHas(actual, key), + new AssertionError({ message: `Expected key ${String(key)} not found in actual object` }), + ); + if (!compareBranch(actual[key], expected[key], loose, comparedObjects)) { + return false; + } + } + + return true; +} + +/** + * The strict equivalence assertion test between two objects + * @param {any} actual + * @param {any} expected + * @param {string | Error} [message] + * @returns {void} + */ +assert.partialDeepStrictEqual = function partialDeepStrictEqual( + actual, + expected, + message, +) { + if (arguments.length < 2) { + throw new ERR_MISSING_ARGS('actual', 'expected'); + } + + if (!compareBranch(actual, expected)) { + innerFail({ + actual, + expected, + message, + operator: 'partialDeepStrictEqual', + stackStartFn: partialDeepStrictEqual, + }); + } +}; + +/** + * The equivalence assertion test between two objects + * @param {any} actual + * @param {any} expected + * @param {string | Error} [message] + * @returns {void} + */ +assert.partialDeepEqual = function partialDeepEqual(actual, expected, message) { + if (arguments.length < 2) { + throw new ERR_MISSING_ARGS('actual', 'expected'); + } + + if (!compareBranch(actual, expected, true)) { + innerFail({ + actual, + expected, + message, + operator: 'partialDeepEqual', + stackStartFn: partialDeepEqual, + }); + } +}; + +/** + * The inclusion assertion test between two arrays or strings + * @param {Array | string} actual + * @param {any} expected + * @param {string | Error} [message] + * @returns {void} + */ +assert.includes = function includes(actual, expected, message) { + if (arguments.length < 2) { + throw new ERR_MISSING_ARGS('actual', 'expected'); + } + + const isString = typeof actual === 'string'; + + if (!isString && !ArrayIsArray(actual)) { + throw new ERR_INVALID_ARG_TYPE('actual', ['string', 'Array'], actual); + } + + if (!(isString ? StringPrototypeIncludes : ArrayPrototypeIncludes)(actual, expected)) { + innerFail({ + actual, + expected, + message, + operator: 'includes', + stackStartFn: includes, + }); + } +}; + class Comparison { constructor(obj, keys, actual) { for (const key of keys) { @@ -807,6 +1009,7 @@ assert.strict = ObjectAssign(strict, assert, { deepEqual: assert.deepStrictEqual, notEqual: assert.notStrictEqual, notDeepEqual: assert.notDeepStrictEqual, + partialDeepEqual: assert.partialDeepStrictEqual, }); assert.strict.strict = assert.strict; diff --git a/lib/internal/test_runner/test.js b/lib/internal/test_runner/test.js index 4331079f74e95a..4c2bb662e9a6e0 100644 --- a/lib/internal/test_runner/test.js +++ b/lib/internal/test_runner/test.js @@ -109,11 +109,14 @@ function lazyAssertObject(harness) { 'equal', 'fail', 'ifError', + 'includes', 'match', 'notDeepEqual', 'notDeepStrictEqual', 'notEqual', 'notStrictEqual', + 'partialDeepEqual', + 'partialDeepStrictEqual', 'rejects', 'strictEqual', 'throws', diff --git a/test/parallel/test-assert-objects.js b/test/parallel/test-assert-objects.js new file mode 100644 index 00000000000000..765751818dab34 --- /dev/null +++ b/test/parallel/test-assert-objects.js @@ -0,0 +1,617 @@ +'use strict'; + +require('../common'); +const vm = require('node:vm'); +const assert = require('node:assert'); +const { describe, it } = require('node:test'); +const { KeyObject } = require('node:crypto'); + +const { subtle } = globalThis.crypto; + +function createCircularObject() { + const obj = {}; + obj.self = obj; + return obj; +} + +function createDeepNestedObject() { + return { level1: { level2: { level3: 'deepValue' } } }; +} + +async function generateCryptoKey() { + const cryptoKey = await subtle.generateKey( + { + name: 'HMAC', + hash: 'SHA-256', + length: 256, + }, + true, + ['sign', 'verify'] + ); + + const keyObject = KeyObject.from(cryptoKey); + + return { cryptoKey, keyObject }; +} + +describe('Object Comparison Tests', () => { + describe('partialDeepStrictEqual', () => { + describe('throws an error', () => { + [ + { + description: 'throws when only one argument is provided', + actual: { a: 1 }, + expected: undefined, + }, + { + description: 'throws when comparing two different objects', + actual: { a: 1, b: 'string' }, + expected: { a: 2, b: 'string' }, + }, + { + description: + 'throws when comparing two objects with different nested objects', + actual: createDeepNestedObject(), + expected: { level1: { level2: { level3: 'differentValue' } } }, + }, + { + description: + 'throws when comparing two objects with different RegExp properties', + actual: { pattern: /abc/ }, + expected: { pattern: /def/ }, + }, + { + description: + 'throws when comparing two arrays with different elements', + actual: [1, 'two', true], + expected: [1, 'two', false], + }, + { + description: + 'throws when comparing two Date objects with different times', + actual: new Date(0), + expected: new Date(1), + }, + { + description: + 'throws when comparing two objects with different large number of properties', + actual: Object.fromEntries( + Array.from({ length: 100 }, (_, i) => [`key${i}`, i]) + ), + expected: Object.fromEntries( + Array.from({ length: 100 }, (_, i) => [`key${i}`, i + 1]) + ), + }, + { + description: + 'throws when comparing two objects with different Symbols', + actual: { [Symbol('test')]: 'symbol' }, + expected: { [Symbol('test')]: 'symbol' }, + }, + { + description: + 'throws when comparing two objects with different array properties', + actual: { a: [1, 2, 3] }, + expected: { a: [1, 2, 4] }, + }, + { + description: + 'throws when comparing two objects with different function properties', + actual: { fn: () => {} }, + expected: { fn: () => {} }, + }, + { + description: + 'throws when comparing two objects with different Error instances', + actual: { error: new Error('Test error 1') }, + expected: { error: new Error('Test error 2') }, + }, + { + description: + 'throws when comparing two objects with different TypedArray instances and content', + actual: { typedArray: new Uint8Array([1, 2, 3]) }, + expected: { typedArray: new Uint8Array([4, 5, 6]) }, + }, + { + description: + 'throws when comparing two Map objects with different entries', + actual: new Map([ + ['key1', 'value1'], + ['key2', 'value2'], + ]), + expected: new Map([ + ['key1', 'value1'], + ['key3', 'value3'], + ]), + }, + { + description: + 'throws when comparing two Map objects with different keys', + actual: new Map([ + ['key1', 'value1'], + ['key2', 'value2'], + ]), + expected: new Map([ + ['key1', 'value1'], + ['key3', 'value2'], + ]), + }, + { + description: + 'throws when comparing two Map objects with different length', + actual: new Map([ + ['key1', 'value1'], + ['key2', 'value2'], + ]), + expected: new Map([['key1', 'value1']]), + }, + { + description: + 'throws when comparing two Set objects from different realms with different values', + actual: new vm.runInNewContext('new Set(["value1", "value2"])'), + expected: new Set(['value1', 'value3']), + }, + { + description: + 'throws when comparing two Set objects with different values', + actual: new Set(['value1', 'value2']), + expected: new Set(['value1', 'value3']), + }, + { + description: + 'throws when comparing two objects with different CryptoKey instances objects', + actual: async () => { + return generateCryptoKey(); + }, + expected: async () => { + return generateCryptoKey(); + }, + }, + { + description: 'throws when comparing one subset object with another', + actual: { a: 1, b: 2, c: 3 }, + expected: { b: '2' }, + }, + { + description: 'throws when comparing one subset array with another', + actual: [1, 2, 3], + expected: ['2'], + }, + ].forEach(({ description, actual, expected }) => { + it(description, () => { + assert.throws(() => assert.partialDeepStrictEqual(actual, expected), Error); + }); + }); + }); + }); + + describe('does not throw an error', () => { + const sym = Symbol('test'); + const func = () => {}; + + [ + { + description: 'compares two identical simple objects', + actual: { a: 1, b: 'string' }, + expected: { a: 1, b: 'string' }, + }, + { + description: 'compares two objects with different property order', + actual: { a: 1, b: 'string' }, + expected: { b: 'string', a: 1 }, + }, + { + description: 'compares two deeply nested objects with partial equality', + actual: { a: { nested: { property: true, some: 'other' } } }, + expected: { a: { nested: { property: true } } }, + }, + { + description: + 'compares plain objects from different realms', + actual: vm.runInNewContext(`({ + a: 1, + b: 2n, + c: "3", + d: /4/, + e: new Set([5]), + f: [6], + g: new Uint8Array() + })`), + expected: { b: 2n, e: new Set([5]), f: [6], g: new Uint8Array() }, + }, + { + description: 'compares two integers', + actual: 1, + expected: 1, + }, + { + description: 'compares two strings', + actual: '1', + expected: '1', + }, + { + description: 'compares two objects with nested objects', + actual: createDeepNestedObject(), + expected: createDeepNestedObject(), + }, + { + description: 'compares two objects with circular references', + actual: createCircularObject(), + expected: createCircularObject(), + }, + { + description: 'compares two arrays with identical elements', + actual: [1, 'two', true], + expected: [1, 'two', true], + }, + { + description: 'compares two Date objects with the same time', + actual: new Date(0), + expected: new Date(0), + }, + { + description: 'compares two objects with large number of properties', + actual: Object.fromEntries( + Array.from({ length: 100 }, (_, i) => [`key${i}`, i]) + ), + expected: Object.fromEntries( + Array.from({ length: 100 }, (_, i) => [`key${i}`, i]) + ), + }, + { + description: 'compares two objects with Symbol properties', + actual: { [sym]: 'symbol' }, + expected: { [sym]: 'symbol' }, + }, + { + description: 'compares two objects with RegExp properties', + actual: { pattern: /abc/ }, + expected: { pattern: /abc/ }, + }, + { + description: 'compares two objects with identical function properties', + actual: { fn: func }, + expected: { fn: func }, + }, + { + description: 'compares two objects with mixed types of properties', + actual: { num: 1, str: 'test', bool: true, sym }, + expected: { num: 1, str: 'test', bool: true, sym }, + }, + { + description: 'compares two objects with Buffers', + actual: { buf: Buffer.from('Node.js') }, + expected: { buf: Buffer.from('Node.js') }, + }, + { + description: 'compares two objects with identical Error properties', + actual: { error: new Error('Test error') }, + expected: { error: new Error('Test error') }, + }, + { + description: 'compares two objects with the same TypedArray instance', + actual: { typedArray: new Uint8Array([1, 2, 3]) }, + expected: { typedArray: new Uint8Array([1, 2, 3]) }, + }, + { + description: 'compares two Map objects with identical entries', + actual: new Map([ + ['key1', 'value1'], + ['key2', 'value2'], + ]), + expected: new Map([ + ['key1', 'value1'], + ['key2', 'value2'], + ]), + }, + { + description: 'compares two Set objects with identical values', + actual: new Set(['value1', 'value2']), + expected: new Set(['value1', 'value2']), + }, + { + description: 'compares two Set objects', + actual: new Set(['value1', 'value2']), + expected: new Set(['value1', 'value2', 'value3']), + }, + { + description: + 'compares two Map objects from different realms with identical entries', + actual: new vm.runInNewContext( + 'new Map([["key1", "value1"], ["key2", "value2"]])' + ), + expected: new Map([ + ['key1', 'value1'], + ['key2', 'value2'], + ]), + }, + { + description: + 'compares two objects with identical getter/setter properties', + actual: (() => { + let value = 'test'; + return Object.defineProperty({}, 'prop', { + get: () => value, + set: (newValue) => { + value = newValue; + }, + enumerable: true, + configurable: true, + }); + })(), + expected: (() => { + let value = 'test'; + return Object.defineProperty({}, 'prop', { + get: () => value, + set: (newValue) => { + value = newValue; + }, + enumerable: true, + configurable: true, + }); + })(), + }, + { + description: 'compares two objects with no prototype', + actual: { __proto__: null, prop: 'value' }, + expected: { __proto__: null, prop: 'value' }, + }, + { + description: + 'compares two objects with identical non-enumerable properties', + actual: (() => { + const obj = {}; + Object.defineProperty(obj, 'hidden', { + value: 'secret', + enumerable: false, + }); + return obj; + })(), + expected: (() => { + const obj = {}; + Object.defineProperty(obj, 'hidden', { + value: 'secret', + enumerable: false, + }); + return obj; + })(), + }, + { + description: 'compares two identical primitives, string', + actual: 'foo', + expected: 'foo', + }, + { + description: 'compares two identical primitives, number', + actual: 1, + expected: 1, + }, + { + description: 'compares two identical primitives, boolean', + actual: false, + expected: false, + }, + { + description: 'compares two identical primitives, null', + actual: null, + expected: null, + }, + { + description: 'compares two identical primitives, undefined', + actual: undefined, + expected: undefined, + }, + { + description: 'compares two identical primitives, Symbol', + actual: sym, + expected: sym, + }, + { + description: + 'compares one subset object with another', + actual: { a: 1, b: 2, c: 3 }, + expected: { b: 2 }, + }, + { + description: + 'compares one subset array with another', + actual: [1, 2, 3], + expected: [2], + }, + ].forEach(({ description, actual, expected }) => { + it(description, () => { + assert.partialDeepStrictEqual(actual, expected); + }); + }); + }); + + describe('partialDeepEqual', () => { + describe('throws an error', () => { + [ + { + description: + 'throws because the expected value is longer than the actual value', + actual: [1, 2, 3], + expected: [1, 2, 3, 4], + }, + { + description: + 'partialDeepEqual throws when comparing two objects with null and empty object', + actual: { a: null }, + expected: { a: {} }, + }, + { + description: 'throws because only one argument is provided', + actual: { a: 1 }, + expected: undefined, + }, + { + description: 'throws because the first argument is null', + actual: null, + expected: { a: 1 }, + }, + { + description: 'throws because the second argument is null', + actual: { a: 1 }, + expected: null, + }, + ].forEach(({ description, actual, expected }) => { + it(description, () => { + assert.throws(() => assert.partialDeepEqual(actual, expected), Error); + }); + }); + }); + + describe('does not throw an error', () => { + [ + { + description: 'compares two objects with Buffers', + actual: { buf: Buffer.from('Node.js') }, + expected: { buf: Buffer.from('Node.js') }, + }, + { + description: 'compares two identical primitives, string', + actual: 'foo', + expected: 'foo', + }, + + { + description: 'compares two identical primitives, number', + actual: 1, + expected: 1, + }, + + { + description: 'compares two identical primitives, bigint', + actual: 1n, + expected: 1n, + }, + + { + description: 'compares two non identical simple objects', + actual: { a: 1, b: 'foo', c: '1' }, + expected: { a: 1, c: 1 }, + }, + { + description: 'compares two similar objects with type coercion', + actual: { a: 1, b: '2' }, + expected: { a: 1, b: 2 }, + }, + { + description: + 'compares two objects with nested objects and type coercion', + actual: { level1: { level2: { level3: '42' } } }, + expected: { level1: { level2: { level3: 42 } } }, + }, + { + description: 'compares two objects with circular references', + actual: createCircularObject(), + expected: createCircularObject(), + }, + { + description: 'compares two arrays with type coercion', + actual: [1, '2', true], + expected: [1, 2, 1], + }, + { + description: 'compares two objects with numeric string and number', + actual: { a: '100' }, + expected: { a: 100 }, + }, + { + description: + 'compares two objects with boolean and numeric representations', + actual: { a: 1, b: 0 }, + expected: { a: true, b: false }, + }, + { + description: + 'compares two objects with undefined and missing properties', + actual: { a: undefined }, + expected: {}, + }, + { + description: 'compares one subset object with another', + actual: { a: 1, b: 2, c: 3 }, + expected: { b: '2' }, + }, + { + description: 'compares one subset array with another', + actual: [true], + expected: [1], + }, + ].forEach(({ description, actual, expected }) => { + it(description, () => { + assert.partialDeepEqual(actual, expected); + }); + }); + }); + }); + + describe('includes', () => { + describe('throws an error', () => { + [ + { + description: 'throws because expected is not a valid type', + actual: { a: 1 }, + expected: undefined, + errorCode: 'ERR_INVALID_ARG_TYPE' + }, + { + description: 'throws because actual is nor an Array or a string', + actual: { a: 1 }, + expected: [1], + errorCode: 'ERR_INVALID_ARG_TYPE' + }, + { + description: + 'throws because an string is being compared to an array', + actual: 'abc', + expected: ['a', 'b', 'c'], + }, + { + description: + 'throws because an array is being compared to a string', + actual: ['a', 'b', 'c'], + expected: 'abc', + }, + { + description: + 'throws because the expected value is longer than the actual value', + actual: 'abc', + expected: 'abcd', + }, + ].forEach(({ description, actual, expected, errorCode }) => { + it(description, () => { + assert.throws(() => assert.includes(actual, expected), errorCode ? + { code: errorCode } : + assert.AssertionError); + }); + }); + }); + + describe('does not throw an error', () => { + [ + { + description: 'compares one subset array with another', + actual: [1, 2, 3], + expected: 2, + }, + { + description: 'compares one subset string with another', + actual: 'abc', + expected: 'b', + }, + { + description: 'compares a null-prototype array with an element', + actual: Object.setPrototypeOf([1, 2, 3], null), + expected: 2, + }, + ].forEach(({ description, actual, expected }) => { + it(description, () => { + assert.includes(actual, expected); + }); + }); + }); + }); +});