From 7c30734a75e37ab1fb2ba5704c11f931b4925c3c Mon Sep 17 00:00:00 2001 From: Giovanni Date: Thu, 12 Dec 2024 11:54:48 +0100 Subject: [PATCH] assert: make partialDeepStrictEqual throw when comparing [0] with [-0] Fixes: https://github.com/nodejs/node/issues/56230 --- lib/assert.js | 66 ++++++++++++++++++++-------- test/parallel/test-assert-objects.js | 50 +++++++++++++++++++++ 2 files changed, 98 insertions(+), 18 deletions(-) diff --git a/lib/assert.js b/lib/assert.js index a2991a096ac081..1cbbf0fb07ee95 100644 --- a/lib/assert.js +++ b/lib/assert.js @@ -497,6 +497,19 @@ function partiallyCompareSets(actual, expected, comparedObjects) { return true; } +// Helper function to determine if an item is 0, -0, '0', or '-0' +function isZeroOrMinusZero(item) { + return ObjectIs(item, -0) || item === '-0' || ObjectIs(item, 0) || item === '0'; +} + +// Helper function to get a unique key for 0, -0, '0', and '-0' to avoid collisions +function getZeroKey(item) { + if (ObjectIs(item, -0)) return '-0(number)'; + if (item === '-0') return '-0(string)'; + if (ObjectIs(item, 0)) return '0(number)'; + if (item === '0') return '0(string)'; +} + function partiallyCompareArrays(actual, expected, comparedObjects) { if (expected.length > actual.length) { return false; @@ -507,38 +520,55 @@ function partiallyCompareArrays(actual, expected, comparedObjects) { // 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)) { - expectedCounts.set(key, count + 1); - found = true; - break; + // Check if the item is a zero or a -0, as these need to be handled separately + if (isZeroOrMinusZero(expectedItem)) { + const zeroKey = getZeroKey(expectedItem); + expectedCounts.set(zeroKey, (expectedCounts.get(zeroKey) || 0) + 1); + } else { + let found = false; + for (const { 0: key, 1: count } of expectedCounts) { + if (isDeepStrictEqual(key, expectedItem)) { + expectedCounts.set(key, count + 1); + found = true; + break; + } + } + if (!found) { + expectedCounts.set(expectedItem, 1); } - } - if (!found) { - expectedCounts.set(expectedItem, 1); } } const safeActual = new SafeArrayIterator(actual); - // Create a map to count occurrences of relevant elements in the actual array for (const actualItem of safeActual) { - for (const { 0: key, 1: count } of expectedCounts) { - if (isDeepStrictEqual(key, actualItem)) { + // Check if the item is a zero or a -0, as these need to be handled separately + if (isZeroOrMinusZero(actualItem)) { + const zeroKey = getZeroKey(actualItem); + + if (expectedCounts.has(zeroKey)) { + const count = expectedCounts.get(zeroKey); if (count === 1) { - expectedCounts.delete(key); + expectedCounts.delete(zeroKey); } else { - expectedCounts.set(key, count - 1); + expectedCounts.set(zeroKey, count - 1); + } + } + } else { + for (const { 0: expectedItem, 1: count } of expectedCounts) { + if (isDeepStrictEqual(expectedItem, actualItem)) { + if (count === 1) { + expectedCounts.delete(expectedItem); + } else { + expectedCounts.set(expectedItem, count - 1); + } + break; } - break; } } } - const { size } = expectedCounts; - expectedCounts.clear(); - return size === 0; + return expectedCounts.size === 0; } /** diff --git a/test/parallel/test-assert-objects.js b/test/parallel/test-assert-objects.js index 3f02ff3c274daa..af0dd5f553b615 100644 --- a/test/parallel/test-assert-objects.js +++ b/test/parallel/test-assert-objects.js @@ -97,6 +97,41 @@ describe('Object Comparison Tests', () => { actual: [1, 'two', true], expected: [1, 'two', false], }, + { + description: 'throws when comparing [0] with [-0]', + actual: [0], + expected: [-0], + }, + { + description: 'throws when comparing [0, 0, 0] with [0, -0]', + actual: [0, 0, 0], + expected: [0, -0], + }, + { + description: 'throws when comparing [-0] with [0]', + actual: [0], + expected: [-0], + }, + { + description: 'throws when comparing ["-0"] with [-0]', + actual: ['-0'], + expected: [-0], + }, + { + description: 'throws when comparing [-0] with ["-0"]', + actual: [-0], + expected: ['-0'], + }, + { + description: 'throws when comparing ["0"] with [0]', + actual: ['0'], + expected: [0], + }, + { + description: 'throws when comparing [0] with ["0"]', + actual: [0], + expected: ['0'], + }, { description: 'throws when comparing two Date objects with different times', @@ -385,6 +420,21 @@ describe('Object Comparison Tests', () => { actual: [1, 'two', true], expected: [1, 'two', true], }, + { + description: 'compares [0] with [0]', + actual: [0], + expected: [0], + }, + { + description: 'compares [-0] with [-0]', + actual: [-0], + expected: [-0], + }, + { + description: 'compares [0, -0, 0] with [0, 0]', + actual: [0, -0, 0], + expected: [0, 0], + }, { description: 'compares two Date objects with the same time', actual: new Date(0),