mirror of
https://github.com/nodejs/node.git
synced 2025-05-05 10:27:05 +00:00

Due to a new API it's possible to skip the indices. That allows to use auto completion with big (typed) arrays. PR-URL: https://github.com/nodejs/node/pull/22408 Fixes: https://github.com/nodejs/node/issues/21446 Reviewed-By: Benjamin Gruenbaum <benjamingr@gmail.com> Reviewed-By: Anna Henningsen <anna@addaleax.net>
630 lines
18 KiB
JavaScript
630 lines
18 KiB
JavaScript
'use strict';
|
|
|
|
const { compare } = process.binding('buffer');
|
|
const { isArrayBufferView } = require('internal/util/types');
|
|
const { internalBinding } = require('internal/bootstrap/loaders');
|
|
const {
|
|
isAnyArrayBuffer,
|
|
isDate,
|
|
isMap,
|
|
isRegExp,
|
|
isSet
|
|
} = internalBinding('types');
|
|
const {
|
|
getOwnNonIndexProperties,
|
|
propertyFilter: {
|
|
ONLY_ENUMERABLE
|
|
}
|
|
} = process.binding('util');
|
|
|
|
const ReflectApply = Reflect.apply;
|
|
|
|
function uncurryThis(func) {
|
|
return (thisArg, ...args) => ReflectApply(func, thisArg, args);
|
|
}
|
|
|
|
const kStrict = true;
|
|
const kLoose = false;
|
|
|
|
const kNoIterator = 0;
|
|
const kIsArray = 1;
|
|
const kIsSet = 2;
|
|
const kIsMap = 3;
|
|
|
|
const objectToString = uncurryThis(Object.prototype.toString);
|
|
const hasOwnProperty = uncurryThis(Object.prototype.hasOwnProperty);
|
|
const propertyIsEnumerable = uncurryThis(Object.prototype.propertyIsEnumerable);
|
|
|
|
const objectKeys = Object.keys;
|
|
const getPrototypeOf = Object.getPrototypeOf;
|
|
const getOwnPropertySymbols = Object.getOwnPropertySymbols;
|
|
const objectIs = Object.is;
|
|
const numberIsNaN = Number.isNaN;
|
|
|
|
// Check if they have the same source and flags
|
|
function areSimilarRegExps(a, b) {
|
|
return a.source === b.source && a.flags === b.flags;
|
|
}
|
|
|
|
function areSimilarFloatArrays(a, b) {
|
|
if (a.byteLength !== b.byteLength) {
|
|
return false;
|
|
}
|
|
for (var offset = 0; offset < a.byteLength; offset++) {
|
|
if (a[offset] !== b[offset]) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
function areSimilarTypedArrays(a, b) {
|
|
if (a.byteLength !== b.byteLength) {
|
|
return false;
|
|
}
|
|
return compare(new Uint8Array(a.buffer, a.byteOffset, a.byteLength),
|
|
new Uint8Array(b.buffer, b.byteOffset, b.byteLength)) === 0;
|
|
}
|
|
|
|
function areEqualArrayBuffers(buf1, buf2) {
|
|
return buf1.byteLength === buf2.byteLength &&
|
|
compare(new Uint8Array(buf1), new Uint8Array(buf2)) === 0;
|
|
}
|
|
|
|
function isFloatTypedArrayTag(tag) {
|
|
return tag === '[object Float32Array]' || tag === '[object Float64Array]';
|
|
}
|
|
|
|
function isArguments(tag) {
|
|
return tag === '[object Arguments]';
|
|
}
|
|
|
|
function isObjectOrArrayTag(tag) {
|
|
return tag === '[object Array]' || tag === '[object Object]';
|
|
}
|
|
|
|
// Notes: Type tags are historical [[Class]] properties that can be set by
|
|
// FunctionTemplate::SetClassName() in C++ or Symbol.toStringTag in JS
|
|
// and retrieved using Object.prototype.toString.call(obj) in JS
|
|
// See https://tc39.github.io/ecma262/#sec-object.prototype.tostring
|
|
// for a list of tags pre-defined in the spec.
|
|
// There are some unspecified tags in the wild too (e.g. typed array tags).
|
|
// Since tags can be altered, they only serve fast failures
|
|
//
|
|
// Typed arrays and buffers are checked by comparing the content in their
|
|
// underlying ArrayBuffer. This optimization requires that it's
|
|
// reasonable to interpret their underlying memory in the same way,
|
|
// which is checked by comparing their type tags.
|
|
// (e.g. a Uint8Array and a Uint16Array with the same memory content
|
|
// could still be different because they will be interpreted differently).
|
|
//
|
|
// For strict comparison, objects should have
|
|
// a) The same built-in type tags
|
|
// b) The same prototypes.
|
|
function strictDeepEqual(val1, val2, memos) {
|
|
if (typeof val1 !== 'object') {
|
|
return typeof val1 === 'number' && numberIsNaN(val1) &&
|
|
numberIsNaN(val2);
|
|
}
|
|
if (typeof val2 !== 'object' || val1 === null || val2 === null) {
|
|
return false;
|
|
}
|
|
const val1Tag = objectToString(val1);
|
|
const val2Tag = objectToString(val2);
|
|
|
|
if (val1Tag !== val2Tag) {
|
|
return false;
|
|
}
|
|
if (getPrototypeOf(val1) !== getPrototypeOf(val2)) {
|
|
return false;
|
|
}
|
|
if (val1Tag === '[object Array]') {
|
|
// Check for sparse arrays and general fast path
|
|
if (val1.length !== val2.length) {
|
|
return false;
|
|
}
|
|
const keys1 = getOwnNonIndexProperties(val1, ONLY_ENUMERABLE);
|
|
const keys2 = getOwnNonIndexProperties(val2, ONLY_ENUMERABLE);
|
|
if (keys1.length !== keys2.length) {
|
|
return false;
|
|
}
|
|
return keyCheck(val1, val2, kStrict, memos, kIsArray, keys1);
|
|
}
|
|
if (val1Tag === '[object Object]') {
|
|
return keyCheck(val1, val2, kStrict, memos, kNoIterator);
|
|
}
|
|
if (isDate(val1)) {
|
|
// TODO: Make these safe.
|
|
if (val1.getTime() !== val2.getTime()) {
|
|
return false;
|
|
}
|
|
} else if (isRegExp(val1)) {
|
|
if (!areSimilarRegExps(val1, val2)) {
|
|
return false;
|
|
}
|
|
} else if (val1Tag === '[object Error]') {
|
|
// Do not compare the stack as it might differ even though the error itself
|
|
// is otherwise identical. The non-enumerable name should be identical as
|
|
// the prototype is also identical. Otherwise this is caught later on.
|
|
if (val1.message !== val2.message) {
|
|
return false;
|
|
}
|
|
} else if (isArrayBufferView(val1)) {
|
|
if (!areSimilarTypedArrays(val1, val2)) {
|
|
return false;
|
|
}
|
|
// Buffer.compare returns true, so val1.length === val2.length. If they both
|
|
// only contain numeric keys, we don't need to exam further than checking
|
|
// the symbols.
|
|
const keys1 = getOwnNonIndexProperties(val1, ONLY_ENUMERABLE);
|
|
const keys2 = getOwnNonIndexProperties(val2, ONLY_ENUMERABLE);
|
|
if (keys1.length !== keys2.length) {
|
|
return false;
|
|
}
|
|
return keyCheck(val1, val2, kStrict, memos, kNoIterator, keys1);
|
|
} else if (isSet(val1)) {
|
|
if (!isSet(val2) || val1.size !== val2.size) {
|
|
return false;
|
|
}
|
|
return keyCheck(val1, val2, kStrict, memos, kIsSet);
|
|
} else if (isMap(val1)) {
|
|
if (!isMap(val2) || val1.size !== val2.size) {
|
|
return false;
|
|
}
|
|
return keyCheck(val1, val2, kStrict, memos, kIsMap);
|
|
} else if (isAnyArrayBuffer(val1)) {
|
|
if (!areEqualArrayBuffers(val1, val2)) {
|
|
return false;
|
|
}
|
|
// TODO: Make the valueOf checks safe.
|
|
} else if (typeof val1.valueOf === 'function') {
|
|
const val1Value = val1.valueOf();
|
|
if (val1Value !== val1 &&
|
|
(typeof val2.valueOf !== 'function' ||
|
|
!innerDeepEqual(val1Value, val2.valueOf(), kStrict))) {
|
|
return false;
|
|
}
|
|
}
|
|
return keyCheck(val1, val2, kStrict, memos, kNoIterator);
|
|
}
|
|
|
|
function looseDeepEqual(val1, val2, memos) {
|
|
if (val1 === null || typeof val1 !== 'object') {
|
|
if (val2 === null || typeof val2 !== 'object') {
|
|
// eslint-disable-next-line eqeqeq
|
|
return val1 == val2;
|
|
}
|
|
return false;
|
|
}
|
|
if (val2 === null || typeof val2 !== 'object') {
|
|
return false;
|
|
}
|
|
const val1Tag = objectToString(val1);
|
|
const val2Tag = objectToString(val2);
|
|
if (val1Tag === val2Tag) {
|
|
if (isObjectOrArrayTag(val1Tag)) {
|
|
return keyCheck(val1, val2, kLoose, memos, kNoIterator);
|
|
}
|
|
if (isArrayBufferView(val1)) {
|
|
if (isFloatTypedArrayTag(val1Tag)) {
|
|
return areSimilarFloatArrays(val1, val2);
|
|
}
|
|
return areSimilarTypedArrays(val1, val2);
|
|
}
|
|
if (isDate(val1) && isDate(val2)) {
|
|
return val1.getTime() === val2.getTime();
|
|
}
|
|
if (isRegExp(val1) && isRegExp(val2)) {
|
|
return areSimilarRegExps(val1, val2);
|
|
}
|
|
if (val1 instanceof Error && val2 instanceof Error) {
|
|
if (val1.message !== val2.message || val1.name !== val2.name)
|
|
return false;
|
|
}
|
|
// Ensure reflexivity of deepEqual with `arguments` objects.
|
|
// See https://github.com/nodejs/node-v0.x-archive/pull/7178
|
|
} else if (isArguments(val1Tag) || isArguments(val2Tag)) {
|
|
return false;
|
|
}
|
|
if (isSet(val1)) {
|
|
if (!isSet(val2) || val1.size !== val2.size) {
|
|
return false;
|
|
}
|
|
return keyCheck(val1, val2, kLoose, memos, kIsSet);
|
|
} else if (isMap(val1)) {
|
|
if (!isMap(val2) || val1.size !== val2.size) {
|
|
return false;
|
|
}
|
|
return keyCheck(val1, val2, kLoose, memos, kIsMap);
|
|
} else if (isSet(val2) || isMap(val2)) {
|
|
return false;
|
|
}
|
|
if (isAnyArrayBuffer(val1) && isAnyArrayBuffer(val2)) {
|
|
if (!areEqualArrayBuffers(val1, val2)) {
|
|
return false;
|
|
}
|
|
}
|
|
return keyCheck(val1, val2, kLoose, memos, kNoIterator);
|
|
}
|
|
|
|
function getEnumerables(val, keys) {
|
|
return keys.filter((k) => propertyIsEnumerable(val, k));
|
|
}
|
|
|
|
function keyCheck(val1, val2, strict, memos, iterationType, aKeys) {
|
|
// For all remaining Object pairs, including Array, objects and Maps,
|
|
// equivalence is determined by having:
|
|
// a) The same number of owned enumerable properties
|
|
// b) The same set of keys/indexes (although not necessarily the same order)
|
|
// c) Equivalent values for every corresponding key/index
|
|
// d) For Sets and Maps, equal contents
|
|
// Note: this accounts for both named and indexed properties on Arrays.
|
|
if (arguments.length === 5) {
|
|
aKeys = objectKeys(val1);
|
|
const bKeys = objectKeys(val2);
|
|
|
|
// The pair must have the same number of owned properties.
|
|
if (aKeys.length !== bKeys.length) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Cheap key test
|
|
let i = 0;
|
|
for (; i < aKeys.length; i++) {
|
|
if (!hasOwnProperty(val2, aKeys[i])) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (strict && arguments.length === 5) {
|
|
const symbolKeysA = getOwnPropertySymbols(val1);
|
|
if (symbolKeysA.length !== 0) {
|
|
let count = 0;
|
|
for (i = 0; i < symbolKeysA.length; i++) {
|
|
const key = symbolKeysA[i];
|
|
if (propertyIsEnumerable(val1, key)) {
|
|
if (!propertyIsEnumerable(val2, key)) {
|
|
return false;
|
|
}
|
|
aKeys.push(key);
|
|
count++;
|
|
} else if (propertyIsEnumerable(val2, key)) {
|
|
return false;
|
|
}
|
|
}
|
|
const symbolKeysB = getOwnPropertySymbols(val2);
|
|
if (symbolKeysA.length !== symbolKeysB.length &&
|
|
getEnumerables(val2, symbolKeysB).length !== count) {
|
|
return false;
|
|
}
|
|
} else {
|
|
const symbolKeysB = getOwnPropertySymbols(val2);
|
|
if (symbolKeysB.length !== 0 &&
|
|
getEnumerables(val2, symbolKeysB).length !== 0) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (aKeys.length === 0 &&
|
|
(iterationType === kNoIterator ||
|
|
iterationType === kIsArray && val1.length === 0 ||
|
|
val1.size === 0)) {
|
|
return true;
|
|
}
|
|
|
|
// Use memos to handle cycles.
|
|
if (memos === undefined) {
|
|
memos = {
|
|
val1: new Map(),
|
|
val2: new Map(),
|
|
position: 0
|
|
};
|
|
} else {
|
|
// We prevent up to two map.has(x) calls by directly retrieving the value
|
|
// and checking for undefined. The map can only contain numbers, so it is
|
|
// safe to check for undefined only.
|
|
const val2MemoA = memos.val1.get(val1);
|
|
if (val2MemoA !== undefined) {
|
|
const val2MemoB = memos.val2.get(val2);
|
|
if (val2MemoB !== undefined) {
|
|
return val2MemoA === val2MemoB;
|
|
}
|
|
}
|
|
memos.position++;
|
|
}
|
|
|
|
memos.val1.set(val1, memos.position);
|
|
memos.val2.set(val2, memos.position);
|
|
|
|
const areEq = objEquiv(val1, val2, strict, aKeys, memos, iterationType);
|
|
|
|
memos.val1.delete(val1);
|
|
memos.val2.delete(val2);
|
|
|
|
return areEq;
|
|
}
|
|
|
|
function innerDeepEqual(val1, val2, strict, memos) {
|
|
// All identical values are equivalent, as determined by ===.
|
|
if (val1 === val2) {
|
|
if (val1 !== 0)
|
|
return true;
|
|
return strict ? objectIs(val1, val2) : true;
|
|
}
|
|
|
|
// Check more closely if val1 and val2 are equal.
|
|
if (strict === true)
|
|
return strictDeepEqual(val1, val2, memos);
|
|
|
|
return looseDeepEqual(val1, val2, memos);
|
|
}
|
|
|
|
function setHasEqualElement(set, val1, strict, memo) {
|
|
// Go looking.
|
|
for (const val2 of set) {
|
|
if (innerDeepEqual(val1, val2, strict, memo)) {
|
|
// Remove the matching element to make sure we do not check that again.
|
|
set.delete(val2);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
// Note: we currently run this multiple times for each loose key!
|
|
// This is done to prevent slowing down the average case.
|
|
function setHasLoosePrim(a, b, val) {
|
|
const altValues = findLooseMatchingPrimitives(val);
|
|
if (altValues === undefined)
|
|
return false;
|
|
|
|
let matches = 1;
|
|
for (var i = 0; i < altValues.length; i++) {
|
|
if (b.has(altValues[i])) {
|
|
matches--;
|
|
}
|
|
if (a.has(altValues[i])) {
|
|
matches++;
|
|
}
|
|
}
|
|
return matches === 0;
|
|
}
|
|
|
|
function setEquiv(a, b, strict, memo) {
|
|
// This is a lazily initiated Set of entries which have to be compared
|
|
// pairwise.
|
|
let set = null;
|
|
for (const val of a) {
|
|
// Note: Checking for the objects first improves the performance for object
|
|
// heavy sets but it is a minor slow down for primitives. As they are fast
|
|
// to check this improves the worst case scenario instead.
|
|
if (typeof val === 'object' && val !== null) {
|
|
if (set === null) {
|
|
set = new Set();
|
|
}
|
|
// If the specified value doesn't exist in the second set its an not null
|
|
// object (or non strict only: a not matching primitive) we'll need to go
|
|
// hunting for something thats deep-(strict-)equal to it. To make this
|
|
// O(n log n) complexity we have to copy these values in a new set first.
|
|
set.add(val);
|
|
} else if (!b.has(val) && (strict || !setHasLoosePrim(a, b, val))) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (set !== null) {
|
|
for (const val of b) {
|
|
// We have to check if a primitive value is already
|
|
// matching and only if it's not, go hunting for it.
|
|
if (typeof val === 'object' && val !== null) {
|
|
if (!setHasEqualElement(set, val, strict, memo))
|
|
return false;
|
|
} else if (!a.has(val) && (strict || !setHasLoosePrim(b, a, val))) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Equality_comparisons_and_sameness#Loose_equality_using
|
|
function findLooseMatchingPrimitives(prim) {
|
|
switch (typeof prim) {
|
|
case 'number':
|
|
if (prim === 0) {
|
|
return ['', '0', false];
|
|
}
|
|
if (prim === 1) {
|
|
return ['1', true];
|
|
}
|
|
return ['' + prim];
|
|
case 'string':
|
|
if (prim === '' || prim === '0') {
|
|
return [0, false];
|
|
}
|
|
if (prim === '1') {
|
|
return [1, true];
|
|
}
|
|
const number = +prim;
|
|
if ('' + number === prim) {
|
|
return [number];
|
|
}
|
|
return;
|
|
case 'undefined':
|
|
return [null];
|
|
case 'object': // Only pass in null as object!
|
|
return [undefined];
|
|
case 'boolean':
|
|
if (prim === false) {
|
|
return ['', '0', 0];
|
|
}
|
|
return ['1', 1];
|
|
}
|
|
}
|
|
|
|
// This is a ugly but relatively fast way to determine if a loose equal entry
|
|
// currently has a correspondent matching entry. Otherwise checking for such
|
|
// values would be way more expensive (O(n^2)).
|
|
// Note: we currently run this multiple times for each loose key!
|
|
// This is done to prevent slowing down the average case.
|
|
function mapHasLoosePrim(a, b, key1, memo, item1, item2) {
|
|
const altKeys = findLooseMatchingPrimitives(key1);
|
|
if (altKeys === undefined)
|
|
return false;
|
|
|
|
const setA = new Set();
|
|
const setB = new Set();
|
|
|
|
let keyCount = 1;
|
|
|
|
setA.add(item1);
|
|
if (b.has(key1)) {
|
|
keyCount--;
|
|
setB.add(item2);
|
|
}
|
|
|
|
for (var i = 0; i < altKeys.length; i++) {
|
|
const key2 = altKeys[i];
|
|
if (a.has(key2)) {
|
|
keyCount++;
|
|
setA.add(a.get(key2));
|
|
}
|
|
if (b.has(key2)) {
|
|
keyCount--;
|
|
setB.add(b.get(key2));
|
|
}
|
|
}
|
|
if (keyCount !== 0 || setA.size !== setB.size)
|
|
return false;
|
|
|
|
for (const val of setA) {
|
|
if (typeof val === 'object' && val !== null) {
|
|
if (!setHasEqualElement(setB, val, false, memo))
|
|
return false;
|
|
} else if (!setB.has(val) && !setHasLoosePrim(setA, setB, val)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
function mapHasEqualEntry(set, map, key1, item1, strict, memo) {
|
|
// To be able to handle cases like:
|
|
// Map([[{}, 'a'], [{}, 'b']]) vs Map([[{}, 'b'], [{}, 'a']])
|
|
// ... we need to consider *all* matching keys, not just the first we find.
|
|
for (const key2 of set) {
|
|
if (innerDeepEqual(key1, key2, strict, memo) &&
|
|
innerDeepEqual(item1, map.get(key2), strict, memo)) {
|
|
set.delete(key2);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function mapEquiv(a, b, strict, memo) {
|
|
let set = null;
|
|
|
|
for (const [key, item1] of a) {
|
|
if (typeof key === 'object' && key !== null) {
|
|
if (set === null) {
|
|
set = new Set();
|
|
}
|
|
set.add(key);
|
|
} else {
|
|
// By directly retrieving the value we prevent another b.has(key) check in
|
|
// almost all possible cases.
|
|
const item2 = b.get(key);
|
|
if ((item2 === undefined && !b.has(key) ||
|
|
!innerDeepEqual(item1, item2, strict, memo)) &&
|
|
(strict || !mapHasLoosePrim(a, b, key, memo, item1, item2))) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (set !== null) {
|
|
for (const [key, item] of b) {
|
|
if (typeof key === 'object' && key !== null) {
|
|
if (!mapHasEqualEntry(set, a, key, item, strict, memo))
|
|
return false;
|
|
} else if (!a.has(key) &&
|
|
(strict || !mapHasLoosePrim(b, a, key, memo, item))) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
function objEquiv(a, b, strict, keys, memos, iterationType) {
|
|
// Sets and maps don't have their entries accessible via normal object
|
|
// properties.
|
|
let i = 0;
|
|
|
|
if (iterationType === kIsSet) {
|
|
if (!setEquiv(a, b, strict, memos)) {
|
|
return false;
|
|
}
|
|
} else if (iterationType === kIsMap) {
|
|
if (!mapEquiv(a, b, strict, memos)) {
|
|
return false;
|
|
}
|
|
} else if (iterationType === kIsArray) {
|
|
for (; i < a.length; i++) {
|
|
if (hasOwnProperty(a, i)) {
|
|
if (!hasOwnProperty(b, i) ||
|
|
!innerDeepEqual(a[i], b[i], strict, memos)) {
|
|
return false;
|
|
}
|
|
} else if (hasOwnProperty(b, i)) {
|
|
return false;
|
|
} else {
|
|
// Array is sparse.
|
|
const keysA = objectKeys(a);
|
|
i++;
|
|
for (; i < keysA.length; i++) {
|
|
const key = keysA[i];
|
|
if (!hasOwnProperty(b, key) ||
|
|
!innerDeepEqual(a[key], b[i], strict, memos)) {
|
|
return false;
|
|
}
|
|
}
|
|
if (keysA.length !== objectKeys(b).length) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
// The pair must have equivalent values for every corresponding key.
|
|
// Possibly expensive deep test:
|
|
for (i = 0; i < keys.length; i++) {
|
|
const key = keys[i];
|
|
if (!innerDeepEqual(a[key], b[key], strict, memos)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
function isDeepEqual(val1, val2) {
|
|
return innerDeepEqual(val1, val2, kLoose);
|
|
}
|
|
|
|
function isDeepStrictEqual(val1, val2) {
|
|
return innerDeepEqual(val1, val2, kStrict);
|
|
}
|
|
|
|
module.exports = {
|
|
isDeepEqual,
|
|
isDeepStrictEqual
|
|
};
|