From c912e20a6b2f65c3c4bfb11f88aff8c6649138c3 Mon Sep 17 00:00:00 2001 From: Giovanni Date: Sun, 1 Dec 2024 15:11:44 +0100 Subject: [PATCH] assert: make partialDeepStrictEqual work with ArrayBuffers Fixes: /~https://github.com/nodejs/node/issues/56097 --- lib/assert.js | 235 ++++++++++++------ test/parallel/test-assert-objects.js | 69 ++++- .../test-assert-typedarray-deepequal.js | 4 + 3 files changed, 231 insertions(+), 77 deletions(-) diff --git a/lib/assert.js b/lib/assert.js index 3e212a1c3aebbe..b96c474be2dc3f 100644 --- a/lib/assert.js +++ b/lib/assert.js @@ -21,12 +21,17 @@ 'use strict'; const { + ArrayBufferIsView, + ArrayBufferPrototypeGetByteLength, ArrayFrom, ArrayIsArray, ArrayPrototypeIndexOf, ArrayPrototypeJoin, ArrayPrototypePush, ArrayPrototypeSlice, + DataViewPrototypeGetBuffer, + DataViewPrototypeGetByteLength, + DataViewPrototypeGetByteOffset, Error, FunctionPrototypeCall, MapPrototypeDelete, @@ -38,6 +43,7 @@ const { ObjectIs, ObjectKeys, ObjectPrototypeIsPrototypeOf, + ObjectPrototypeToString, ReflectApply, ReflectHas, ReflectOwnKeys, @@ -50,6 +56,8 @@ const { StringPrototypeSlice, StringPrototypeSplit, SymbolIterator, + TypedArrayPrototypeGetLength, + Uint8Array, } = primordials; const { @@ -65,6 +73,8 @@ const AssertionError = require('internal/assert/assertion_error'); const { inspect } = require('internal/util/inspect'); const { Buffer } = require('buffer'); const { + isArrayBuffer, + isDataView, isKeyObject, isPromise, isRegExp, @@ -73,6 +83,7 @@ const { isDate, isWeakSet, isWeakMap, + isSharedArrayBuffer, } = require('internal/util/types'); const { isError, deprecate, emitExperimentalWarning } = require('internal/util'); const { innerOk } = require('internal/assert/utils'); @@ -369,9 +380,149 @@ function isSpecial(obj) { } const typesToCallDeepStrictEqualWith = [ - isKeyObject, isWeakSet, isWeakMap, Buffer.isBuffer, + isKeyObject, isWeakSet, isWeakMap, Buffer.isBuffer, isSharedArrayBuffer, ]; +function compareMaps(actual, expected, comparedObjects) { + if (actual.size !== expected.size) { + return false; + } + const safeIterator = FunctionPrototypeCall(SafeMap.prototype[SymbolIterator], actual); + + comparedObjects ??= new SafeWeakSet(); + + for (const { 0: key, 1: val } of safeIterator) { + if (!MapPrototypeHas(expected, key)) { + return false; + } + if (!compareBranch(val, MapPrototypeGet(expected, key), comparedObjects)) { + return false; + } + } + return true; +} + +function compareArrayBuffers(actual, expected) { + let actualView, expectedView, expectedViewLength; + + if (!ArrayBufferIsView(actual)) { + if (ArrayBufferIsView(expected)) { + return false; + } + expectedViewLength = ArrayBufferPrototypeGetByteLength(expected); + + if (expectedViewLength > ArrayBufferPrototypeGetByteLength(actual)) { + return false; + } + actualView = new Uint8Array(actual); + expectedView = new Uint8Array(expected); + + } else if (isDataView(actual)) { + if (!isDataView(expected)) { + return false; + } + const actualByteLength = DataViewPrototypeGetByteLength(actual); + expectedViewLength = DataViewPrototypeGetByteLength(expected); + if (expectedViewLength > actualByteLength) { + return false; + } + + actualView = new Uint8Array( + DataViewPrototypeGetBuffer(actual), + DataViewPrototypeGetByteOffset(actual), + actualByteLength, + ); + expectedView = new Uint8Array( + DataViewPrototypeGetBuffer(expected), + DataViewPrototypeGetByteOffset(expected), + expectedViewLength, + ); + } else { + if (ObjectPrototypeToString(actual) !== ObjectPrototypeToString(expected)) { + return false; + } + actualView = actual; + expectedView = expected; + expectedViewLength = TypedArrayPrototypeGetLength(expected); + + if (expectedViewLength > TypedArrayPrototypeGetLength(actual)) { + return false; + } + } + + for (let i = 0; i < expectedViewLength; i++) { + if (actualView[i] !== expectedView[i]) { + return false; + } + } + + return true; +} + +function compareSets(actual, expected, comparedObjects) { + if (expected.size > actual.size) { + return false; // `expected` can't be a subset if it has more elements + } + + if (isDeepEqual === undefined) lazyLoadComparison(); + + const actualArray = ArrayFrom(FunctionPrototypeCall(SafeSet.prototype[SymbolIterator], actual)); + const expectedIterator = FunctionPrototypeCall(SafeSet.prototype[SymbolIterator], expected); + const usedIndices = new SafeSet(); + + expectedIteration: for (const expectedItem of expectedIterator) { + for (let actualIdx = 0; actualIdx < actualArray.length; actualIdx++) { + if (!usedIndices.has(actualIdx) && isDeepStrictEqual(actualArray[actualIdx], expectedItem)) { + usedIndices.add(actualIdx); + continue expectedIteration; + } + } + return false; + } + + return true; +} + +function compareArrays(actual, expected, comparedObjects) { + if (expected.length > actual.length) { + return false; + } + + if (isDeepEqual === undefined) lazyLoadComparison(); + + // Create a map to count occurrences of each element in the expected array + const expectedCounts = new SafeMap(); + for (const expectedItem of expected) { + let found = false; + for (const { 0: key, 1: count } of expectedCounts) { + if (isDeepStrictEqual(key, expectedItem)) { + MapPrototypeSet(expectedCounts, key, count + 1); + found = true; + break; + } + } + if (!found) { + MapPrototypeSet(expectedCounts, expectedItem, 1); + } + } + + // Create a map to count occurrences of relevant elements in the actual array + for (const actualItem of actual) { + for (const { 0: key, 1: count } of expectedCounts) { + if (isDeepStrictEqual(key, actualItem)) { + if (count === 1) { + MapPrototypeDelete(expectedCounts, key); + } else { + MapPrototypeSet(expectedCounts, key, count - 1); + } + break; + } + } + } + + return !expectedCounts.size; +} + /** * Compares two objects or values recursively to check if they are equal. * @param {any} actual - The actual value to compare. @@ -388,22 +539,14 @@ function compareBranch( ) { // Check for Map object equality if (isMap(actual) && isMap(expected)) { - if (actual.size !== expected.size) { - return false; - } - const safeIterator = FunctionPrototypeCall(SafeMap.prototype[SymbolIterator], actual); - - comparedObjects ??= new SafeWeakSet(); + return compareMaps(actual, expected, comparedObjects); + } - for (const { 0: key, 1: val } of safeIterator) { - if (!MapPrototypeHas(expected, key)) { - return false; - } - if (!compareBranch(val, MapPrototypeGet(expected, key), comparedObjects)) { - return false; - } - } - return true; + if ( + (ArrayBufferIsView(actual) && ArrayBufferIsView(expected)) || + (isArrayBuffer(actual) && isArrayBuffer(expected)) + ) { + return compareArrayBuffers(actual, expected); } for (const type of typesToCallDeepStrictEqualWith) { @@ -415,68 +558,12 @@ function compareBranch( // Check for Set object equality if (isSet(actual) && isSet(expected)) { - if (expected.size > actual.size) { - return false; // `expected` can't be a subset if it has more elements - } - - if (isDeepEqual === undefined) lazyLoadComparison(); - - const actualArray = ArrayFrom(FunctionPrototypeCall(SafeSet.prototype[SymbolIterator], actual)); - const expectedIterator = FunctionPrototypeCall(SafeSet.prototype[SymbolIterator], expected); - const usedIndices = new SafeSet(); - - expectedIteration: for (const expectedItem of expectedIterator) { - for (let actualIdx = 0; actualIdx < actualArray.length; actualIdx++) { - if (!usedIndices.has(actualIdx) && isDeepStrictEqual(actualArray[actualIdx], expectedItem)) { - usedIndices.add(actualIdx); - continue expectedIteration; - } - } - return false; - } - - return true; + return compareSets(actual, expected, comparedObjects); } // Check if expected array is a subset of actual array if (ArrayIsArray(actual) && ArrayIsArray(expected)) { - if (expected.length > actual.length) { - return false; - } - - if (isDeepEqual === undefined) lazyLoadComparison(); - - // Create a map to count occurrences of each element in the expected array - const expectedCounts = new SafeMap(); - for (const expectedItem of expected) { - let found = false; - for (const { 0: key, 1: count } of expectedCounts) { - if (isDeepStrictEqual(key, expectedItem)) { - MapPrototypeSet(expectedCounts, key, count + 1); - found = true; - break; - } - } - if (!found) { - MapPrototypeSet(expectedCounts, expectedItem, 1); - } - } - - // Create a map to count occurrences of relevant elements in the actual array - for (const actualItem of actual) { - for (const { 0: key, 1: count } of expectedCounts) { - if (isDeepStrictEqual(key, actualItem)) { - if (count === 1) { - MapPrototypeDelete(expectedCounts, key); - } else { - MapPrototypeSet(expectedCounts, key, count - 1); - } - break; - } - } - } - - return !expectedCounts.size; + return compareArrays(actual, expected, comparedObjects); } // Comparison done when at least one of the values is not an object diff --git a/test/parallel/test-assert-objects.js b/test/parallel/test-assert-objects.js index d1c8bb854babb0..af86d23030f65a 100644 --- a/test/parallel/test-assert-objects.js +++ b/test/parallel/test-assert-objects.js @@ -39,10 +39,15 @@ describe('Object Comparison Tests', () => { describe('throws an error', () => { const tests = [ { - description: 'throws when only one argument is provided', + description: 'throws when only actual is provided', actual: { a: 1 }, expected: undefined, }, + { + description: 'throws when only expected is provided', + actual: undefined, + expected: { a: 1 }, + }, { description: 'throws when expected has more properties than actual', actual: [1, 'two'], @@ -207,6 +212,31 @@ describe('Object Comparison Tests', () => { actual: [1, 2, 3], expected: ['2'], }, + { + description: 'throws when comparing a ArrayBuffer with a SharedArrayBuffer', + actual: new ArrayBuffer(3), + expected: new SharedArrayBuffer(3), + }, + { + description: 'throws when comparing an Int16Array with a Uint16Array', + actual: new Int16Array(3), + expected: new Uint16Array(3), + }, + { + description: 'throws when comparing two dataviews with different buffers', + actual: { dataView: new DataView(new ArrayBuffer(3)) }, + expected: { dataView: new DataView(new ArrayBuffer(4)) }, + }, + { + description: 'throws when comparing a DataView with a TypedArray', + actual: { dataView: new DataView(new ArrayBuffer(3)) }, + expected: { dataView: new Uint8Array(3) }, + }, + { + description: 'throws when comparing a TypedArray with a DataView', + actual: { dataView: new Uint8Array(3) }, + expected: { dataView: new DataView(new ArrayBuffer(3)) }, + }, ]; if (common.hasCrypto) { @@ -343,10 +373,30 @@ describe('Object Comparison Tests', () => { expected: { error: new Error('Test error') }, }, { - description: 'compares two objects with TypedArray instances with the same content', - actual: { typedArray: new Uint8Array([1, 2, 3]) }, + description: 'compares two Uint8Array objects', + actual: { typedArray: new Uint8Array([1, 2, 3, 4, 5]) }, expected: { typedArray: new Uint8Array([1, 2, 3]) }, }, + { + description: 'compares two Int16Array objects', + actual: { typedArray: new Int16Array([1, 2, 3, 4, 5]) }, + expected: { typedArray: new Int16Array([1, 2, 3]) }, + }, + { + description: 'compares two DataView objects with the same buffer and different views', + actual: { dataView: new DataView(new ArrayBuffer(8), 0, 4) }, + expected: { dataView: new DataView(new ArrayBuffer(8), 4, 4) }, + }, + { + description: 'compares two DataView objects with different buffers', + actual: { dataView: new DataView(new ArrayBuffer(8)) }, + expected: { dataView: new DataView(new ArrayBuffer(8)) }, + }, + { + description: 'compares two DataView objects with the same buffer and same views', + actual: { dataView: new DataView(new ArrayBuffer(8), 0, 8) }, + expected: { dataView: new DataView(new ArrayBuffer(8), 0, 8) }, + }, { description: 'compares two Map objects with identical entries', actual: new Map([ @@ -358,6 +408,19 @@ describe('Object Comparison Tests', () => { ['key2', 'value2'], ]), }, + { + description: 'compares two Map where one is a subset of the other', + actual: new Map([ + ['key1', { nested: { property: true } }], + ['key2', new Set([1, 2, 3])], + ['key3', new Uint8Array([1, 2, 3])], + ]), + expected: new Map([ + ['key1', { nested: { property: true } }], + ['key2', new Set([1, 2, 3])], + ['key3', new Uint8Array([1, 2, 3])], + ]) + }, { describe: 'compares two array of objects', actual: [{ a: 5 }], diff --git a/test/parallel/test-assert-typedarray-deepequal.js b/test/parallel/test-assert-typedarray-deepequal.js index 1c1c4c030a267e..243b51de1cd4a3 100644 --- a/test/parallel/test-assert-typedarray-deepequal.js +++ b/test/parallel/test-assert-typedarray-deepequal.js @@ -99,6 +99,10 @@ suite('notEqualArrayPairs', () => { makeBlock(assert.deepStrictEqual, arrayPair[0], arrayPair[1]), assert.AssertionError ); + assert.throws( + makeBlock(assert.partialDeepStrictEqual, arrayPair[0], arrayPair[1]), + assert.AssertionError + ); }); } });