mirror of
https://github.com/denoland/deno.git
synced 2025-01-23 15:39:49 -05:00
b40086fd7d
This commit changes "include_js_files!" macro from "deno_core" in a way that "dir" option doesn't cause specifiers to be rewritten to include it. Example: ``` include_js_files! { dir "js", "hello.js", } ``` The above definition required embedders to use: `import ... from "internal:<ext_name>/js/hello.js"`. But with this change, the "js" directory in which the files are stored is an implementation detail, which for embedders results in: `import ... from "internal:<ext_name>/hello.js"`. The directory the files are stored in, is an implementation detail and in some cases might result in a significant size difference for the snapshot. As an example, in "deno_node" extension, we store the source code in "polyfills" directory; which resulted in each specifier to look like "internal:deno_node/polyfills/<module_name>", but with this change it's "internal:deno_node/<module_name>". Given that "deno_node" has over 100 files, many of them having several import specifiers to the same extension, this change removes 10 characters from each import specifier.
681 lines
16 KiB
TypeScript
681 lines
16 KiB
TypeScript
// Copyright 2018-2023 the Deno authors. All rights reserved. MIT license.
|
|
// Copyright Joyent and Node contributors. All rights reserved. MIT license.
|
|
|
|
// deno-lint-ignore-file
|
|
import {
|
|
isAnyArrayBuffer,
|
|
isArrayBufferView,
|
|
isBigIntObject,
|
|
isBooleanObject,
|
|
isBoxedPrimitive,
|
|
isDate,
|
|
isFloat32Array,
|
|
isFloat64Array,
|
|
isMap,
|
|
isNativeError,
|
|
isNumberObject,
|
|
isRegExp,
|
|
isSet,
|
|
isStringObject,
|
|
isSymbolObject,
|
|
isTypedArray,
|
|
} from "internal:deno_node/internal/util/types.ts";
|
|
|
|
import { Buffer } from "internal:deno_node/buffer.ts";
|
|
import {
|
|
getOwnNonIndexProperties,
|
|
ONLY_ENUMERABLE,
|
|
SKIP_SYMBOLS,
|
|
} from "internal:deno_node/internal_binding/util.ts";
|
|
|
|
enum valueType {
|
|
noIterator,
|
|
isArray,
|
|
isSet,
|
|
isMap,
|
|
}
|
|
|
|
interface Memo {
|
|
val1: Map<unknown, unknown>;
|
|
val2: Map<unknown, unknown>;
|
|
position: number;
|
|
}
|
|
let memo: Memo;
|
|
|
|
export function isDeepStrictEqual(val1: unknown, val2: unknown): boolean {
|
|
return innerDeepEqual(val1, val2, true);
|
|
}
|
|
export function isDeepEqual(val1: unknown, val2: unknown): boolean {
|
|
return innerDeepEqual(val1, val2, false);
|
|
}
|
|
|
|
function innerDeepEqual(
|
|
val1: unknown,
|
|
val2: unknown,
|
|
strict: boolean,
|
|
memos = memo,
|
|
): boolean {
|
|
// Basic case covered by Strict Equality Comparison
|
|
if (val1 === val2) {
|
|
if (val1 !== 0) return true;
|
|
return strict ? Object.is(val1, val2) : true;
|
|
}
|
|
if (strict) {
|
|
// Cases where the values are not objects
|
|
// If both values are Not a Number NaN
|
|
if (typeof val1 !== "object") {
|
|
return (
|
|
typeof val1 === "number" && Number.isNaN(val1) && Number.isNaN(val2)
|
|
);
|
|
}
|
|
// If either value is null
|
|
if (typeof val2 !== "object" || val1 === null || val2 === null) {
|
|
return false;
|
|
}
|
|
// If the prototype are not the same
|
|
if (Object.getPrototypeOf(val1) !== Object.getPrototypeOf(val2)) {
|
|
return false;
|
|
}
|
|
} else {
|
|
// Non strict case where values are either null or NaN
|
|
if (val1 === null || typeof val1 !== "object") {
|
|
if (val2 === null || typeof val2 !== "object") {
|
|
return val1 == val2 || (Number.isNaN(val1) && Number.isNaN(val2));
|
|
}
|
|
return false;
|
|
}
|
|
if (val2 === null || typeof val2 !== "object") {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
const val1Tag = Object.prototype.toString.call(val1);
|
|
const val2Tag = Object.prototype.toString.call(val2);
|
|
|
|
// prototype must be Strictly Equal
|
|
if (
|
|
val1Tag !== val2Tag
|
|
) {
|
|
return false;
|
|
}
|
|
|
|
// handling when values are array
|
|
if (Array.isArray(val1)) {
|
|
// quick rejection cases
|
|
if (!Array.isArray(val2) || val1.length !== val2.length) {
|
|
return false;
|
|
}
|
|
const filter = strict ? ONLY_ENUMERABLE : ONLY_ENUMERABLE | SKIP_SYMBOLS;
|
|
const keys1 = getOwnNonIndexProperties(val1, filter);
|
|
const keys2 = getOwnNonIndexProperties(val2, filter);
|
|
if (keys1.length !== keys2.length) {
|
|
return false;
|
|
}
|
|
return keyCheck(val1, val2, strict, memos, valueType.isArray, keys1);
|
|
} else if (val1Tag === "[object Object]") {
|
|
return keyCheck(
|
|
val1 as object,
|
|
val2 as object,
|
|
strict,
|
|
memos,
|
|
valueType.noIterator,
|
|
);
|
|
} else if (val1 instanceof Date) {
|
|
if (!(val2 instanceof Date) || val1.getTime() !== val2.getTime()) {
|
|
return false;
|
|
}
|
|
} else if (val1 instanceof RegExp) {
|
|
if (!(val2 instanceof RegExp) || !areSimilarRegExps(val1, val2)) {
|
|
return false;
|
|
}
|
|
} else if (isNativeError(val1) || val1 instanceof Error) {
|
|
// stack may or may not be same, hence it shouldn't be compared
|
|
if (
|
|
// How to handle the type errors here
|
|
(!isNativeError(val2) && !(val2 instanceof Error)) ||
|
|
(val1 as Error).message !== (val2 as Error).message ||
|
|
(val1 as Error).name !== (val2 as Error).name
|
|
) {
|
|
return false;
|
|
}
|
|
} else if (isArrayBufferView(val1)) {
|
|
const TypedArrayPrototypeGetSymbolToStringTag = (
|
|
val:
|
|
| BigInt64Array
|
|
| BigUint64Array
|
|
| Float32Array
|
|
| Float64Array
|
|
| Int8Array
|
|
| Int16Array
|
|
| Int32Array
|
|
| Uint8Array
|
|
| Uint8ClampedArray
|
|
| Uint16Array
|
|
| Uint32Array,
|
|
) =>
|
|
Object.getOwnPropertySymbols(val)
|
|
.map((item) => item.toString())
|
|
.toString();
|
|
if (
|
|
isTypedArray(val1) &&
|
|
isTypedArray(val2) &&
|
|
(TypedArrayPrototypeGetSymbolToStringTag(val1) !==
|
|
TypedArrayPrototypeGetSymbolToStringTag(val2))
|
|
) {
|
|
return false;
|
|
}
|
|
|
|
if (!strict && (isFloat32Array(val1) || isFloat64Array(val1))) {
|
|
if (!areSimilarFloatArrays(val1, val2)) {
|
|
return false;
|
|
}
|
|
} else if (!areSimilarTypedArrays(val1, val2)) {
|
|
return false;
|
|
}
|
|
const filter = strict ? ONLY_ENUMERABLE : ONLY_ENUMERABLE | SKIP_SYMBOLS;
|
|
const keysVal1 = getOwnNonIndexProperties(val1 as object, filter);
|
|
const keysVal2 = getOwnNonIndexProperties(val2 as object, filter);
|
|
if (keysVal1.length !== keysVal2.length) {
|
|
return false;
|
|
}
|
|
return keyCheck(
|
|
val1 as object,
|
|
val2 as object,
|
|
strict,
|
|
memos,
|
|
valueType.noIterator,
|
|
keysVal1,
|
|
);
|
|
} else if (isSet(val1)) {
|
|
if (
|
|
!isSet(val2) ||
|
|
(val1 as Set<unknown>).size !== (val2 as Set<unknown>).size
|
|
) {
|
|
return false;
|
|
}
|
|
return keyCheck(
|
|
val1 as object,
|
|
val2 as object,
|
|
strict,
|
|
memos,
|
|
valueType.isSet,
|
|
);
|
|
} else if (isMap(val1)) {
|
|
if (
|
|
!isMap(val2) || val1.size !== val2.size
|
|
) {
|
|
return false;
|
|
}
|
|
return keyCheck(
|
|
val1 as object,
|
|
val2 as object,
|
|
strict,
|
|
memos,
|
|
valueType.isMap,
|
|
);
|
|
} else if (isAnyArrayBuffer(val1)) {
|
|
if (!isAnyArrayBuffer(val2) || !areEqualArrayBuffers(val1, val2)) {
|
|
return false;
|
|
}
|
|
} else if (isBoxedPrimitive(val1)) {
|
|
if (!isEqualBoxedPrimitive(val1, val2)) {
|
|
return false;
|
|
}
|
|
} else if (
|
|
Array.isArray(val2) ||
|
|
isArrayBufferView(val2) ||
|
|
isSet(val2) ||
|
|
isMap(val2) ||
|
|
isDate(val2) ||
|
|
isRegExp(val2) ||
|
|
isAnyArrayBuffer(val2) ||
|
|
isBoxedPrimitive(val2) ||
|
|
isNativeError(val2) ||
|
|
val2 instanceof Error
|
|
) {
|
|
return false;
|
|
}
|
|
return keyCheck(
|
|
val1 as object,
|
|
val2 as object,
|
|
strict,
|
|
memos,
|
|
valueType.noIterator,
|
|
);
|
|
}
|
|
|
|
function keyCheck(
|
|
val1: object,
|
|
val2: object,
|
|
strict: boolean,
|
|
memos: Memo,
|
|
iterationType: valueType,
|
|
aKeys: (string | symbol)[] = [],
|
|
) {
|
|
if (arguments.length === 5) {
|
|
aKeys = Object.keys(val1);
|
|
const bKeys = Object.keys(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 (!val2.propertyIsEnumerable(aKeys[i])) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (strict && arguments.length === 5) {
|
|
const symbolKeysA = Object.getOwnPropertySymbols(val1);
|
|
if (symbolKeysA.length !== 0) {
|
|
let count = 0;
|
|
for (i = 0; i < symbolKeysA.length; i++) {
|
|
const key = symbolKeysA[i];
|
|
if (val1.propertyIsEnumerable(key)) {
|
|
if (!val2.propertyIsEnumerable(key)) {
|
|
return false;
|
|
}
|
|
// added toString here
|
|
aKeys.push(key.toString());
|
|
count++;
|
|
} else if (val2.propertyIsEnumerable(key)) {
|
|
return false;
|
|
}
|
|
}
|
|
const symbolKeysB = Object.getOwnPropertySymbols(val2);
|
|
if (
|
|
symbolKeysA.length !== symbolKeysB.length &&
|
|
getEnumerables(val2, symbolKeysB).length !== count
|
|
) {
|
|
return false;
|
|
}
|
|
} else {
|
|
const symbolKeysB = Object.getOwnPropertySymbols(val2);
|
|
if (
|
|
symbolKeysB.length !== 0 &&
|
|
getEnumerables(val2, symbolKeysB).length !== 0
|
|
) {
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
if (
|
|
aKeys.length === 0 &&
|
|
(iterationType === valueType.noIterator ||
|
|
(iterationType === valueType.isArray && (val1 as []).length === 0) ||
|
|
(val1 as Set<unknown>).size === 0)
|
|
) {
|
|
return true;
|
|
}
|
|
|
|
if (memos === undefined) {
|
|
memos = {
|
|
val1: new Map(),
|
|
val2: new Map(),
|
|
position: 0,
|
|
};
|
|
} else {
|
|
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 areSimilarRegExps(a: RegExp, b: RegExp) {
|
|
return a.source === b.source && a.flags === b.flags &&
|
|
a.lastIndex === b.lastIndex;
|
|
}
|
|
|
|
// TODO(standvpmnt): add type for arguments
|
|
function areSimilarFloatArrays(arr1: any, arr2: any): boolean {
|
|
if (arr1.byteLength !== arr2.byteLength) {
|
|
return false;
|
|
}
|
|
for (let i = 0; i < arr1.byteLength; i++) {
|
|
if (arr1[i] !== arr2[i]) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// TODO(standvpmnt): add type for arguments
|
|
function areSimilarTypedArrays(arr1: any, arr2: any): boolean {
|
|
if (arr1.byteLength !== arr2.byteLength) {
|
|
return false;
|
|
}
|
|
return (
|
|
Buffer.compare(
|
|
new Uint8Array(arr1.buffer, arr1.byteOffset, arr1.byteLength),
|
|
new Uint8Array(arr2.buffer, arr2.byteOffset, arr2.byteLength),
|
|
) === 0
|
|
);
|
|
}
|
|
// TODO(standvpmnt): add type for arguments
|
|
function areEqualArrayBuffers(buf1: any, buf2: any): boolean {
|
|
return (
|
|
buf1.byteLength === buf2.byteLength &&
|
|
Buffer.compare(new Uint8Array(buf1), new Uint8Array(buf2)) === 0
|
|
);
|
|
}
|
|
|
|
// TODO(standvpmnt): this check of getOwnPropertySymbols and getOwnPropertyNames
|
|
// length is sufficient to handle the current test case, however this will fail
|
|
// to catch a scenario wherein the getOwnPropertySymbols and getOwnPropertyNames
|
|
// length is the same(will be very contrived but a possible shortcoming
|
|
function isEqualBoxedPrimitive(a: any, b: any): boolean {
|
|
if (
|
|
Object.getOwnPropertyNames(a).length !==
|
|
Object.getOwnPropertyNames(b).length
|
|
) {
|
|
return false;
|
|
}
|
|
if (
|
|
Object.getOwnPropertySymbols(a).length !==
|
|
Object.getOwnPropertySymbols(b).length
|
|
) {
|
|
return false;
|
|
}
|
|
if (isNumberObject(a)) {
|
|
return (
|
|
isNumberObject(b) &&
|
|
Object.is(
|
|
Number.prototype.valueOf.call(a),
|
|
Number.prototype.valueOf.call(b),
|
|
)
|
|
);
|
|
}
|
|
if (isStringObject(a)) {
|
|
return (
|
|
isStringObject(b) &&
|
|
(String.prototype.valueOf.call(a) === String.prototype.valueOf.call(b))
|
|
);
|
|
}
|
|
if (isBooleanObject(a)) {
|
|
return (
|
|
isBooleanObject(b) &&
|
|
(Boolean.prototype.valueOf.call(a) === Boolean.prototype.valueOf.call(b))
|
|
);
|
|
}
|
|
if (isBigIntObject(a)) {
|
|
return (
|
|
isBigIntObject(b) &&
|
|
(BigInt.prototype.valueOf.call(a) === BigInt.prototype.valueOf.call(b))
|
|
);
|
|
}
|
|
if (isSymbolObject(a)) {
|
|
return (
|
|
isSymbolObject(b) &&
|
|
(Symbol.prototype.valueOf.call(a) ===
|
|
Symbol.prototype.valueOf.call(b))
|
|
);
|
|
}
|
|
// assert.fail(`Unknown boxed type ${val1}`);
|
|
// return false;
|
|
throw Error(`Unknown boxed type`);
|
|
}
|
|
|
|
function getEnumerables(val: any, keys: any) {
|
|
return keys.filter((key: string) => val.propertyIsEnumerable(key));
|
|
}
|
|
|
|
function objEquiv(
|
|
obj1: any,
|
|
obj2: any,
|
|
strict: boolean,
|
|
keys: any,
|
|
memos: Memo,
|
|
iterationType: valueType,
|
|
): boolean {
|
|
let i = 0;
|
|
|
|
if (iterationType === valueType.isSet) {
|
|
if (!setEquiv(obj1, obj2, strict, memos)) {
|
|
return false;
|
|
}
|
|
} else if (iterationType === valueType.isMap) {
|
|
if (!mapEquiv(obj1, obj2, strict, memos)) {
|
|
return false;
|
|
}
|
|
} else if (iterationType === valueType.isArray) {
|
|
for (; i < obj1.length; i++) {
|
|
if (obj1.hasOwnProperty(i)) {
|
|
if (
|
|
!obj2.hasOwnProperty(i) ||
|
|
!innerDeepEqual(obj1[i], obj2[i], strict, memos)
|
|
) {
|
|
return false;
|
|
}
|
|
} else if (obj2.hasOwnProperty(i)) {
|
|
return false;
|
|
} else {
|
|
const keys1 = Object.keys(obj1);
|
|
for (; i < keys1.length; i++) {
|
|
const key = keys1[i];
|
|
if (
|
|
!obj2.hasOwnProperty(key) ||
|
|
!innerDeepEqual(obj1[key], obj2[key], strict, memos)
|
|
) {
|
|
return false;
|
|
}
|
|
}
|
|
if (keys1.length !== Object.keys(obj2).length) {
|
|
return false;
|
|
}
|
|
if (keys1.length !== Object.keys(obj2).length) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Expensive test
|
|
for (i = 0; i < keys.length; i++) {
|
|
const key = keys[i];
|
|
if (!innerDeepEqual(obj1[key], obj2[key], strict, memos)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
function findLooseMatchingPrimitives(
|
|
primitive: unknown,
|
|
): boolean | null | undefined {
|
|
switch (typeof primitive) {
|
|
case "undefined":
|
|
return null;
|
|
case "object":
|
|
return undefined;
|
|
case "symbol":
|
|
return false;
|
|
case "string":
|
|
primitive = +primitive;
|
|
case "number":
|
|
if (Number.isNaN(primitive)) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
function setMightHaveLoosePrim(
|
|
set1: Set<unknown>,
|
|
set2: Set<unknown>,
|
|
primitive: any,
|
|
) {
|
|
const altValue = findLooseMatchingPrimitives(primitive);
|
|
if (altValue != null) return altValue;
|
|
|
|
return set2.has(altValue) && !set1.has(altValue);
|
|
}
|
|
|
|
function setHasEqualElement(
|
|
set: any,
|
|
val1: any,
|
|
strict: boolean,
|
|
memos: Memo,
|
|
): boolean {
|
|
for (const val2 of set) {
|
|
if (innerDeepEqual(val1, val2, strict, memos)) {
|
|
set.delete(val2);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
function setEquiv(set1: any, set2: any, strict: boolean, memos: Memo): boolean {
|
|
let set = null;
|
|
for (const item of set1) {
|
|
if (typeof item === "object" && item !== null) {
|
|
if (set === null) {
|
|
// What is SafeSet from primordials?
|
|
// set = new SafeSet();
|
|
set = new Set();
|
|
}
|
|
set.add(item);
|
|
} else if (!set2.has(item)) {
|
|
if (strict) return false;
|
|
|
|
if (!setMightHaveLoosePrim(set1, set2, item)) {
|
|
return false;
|
|
}
|
|
|
|
if (set === null) {
|
|
set = new Set();
|
|
}
|
|
set.add(item);
|
|
}
|
|
}
|
|
|
|
if (set !== null) {
|
|
for (const item of set2) {
|
|
if (typeof item === "object" && item !== null) {
|
|
if (!setHasEqualElement(set, item, strict, memos)) return false;
|
|
} else if (
|
|
!strict &&
|
|
!set1.has(item) &&
|
|
!setHasEqualElement(set, item, strict, memos)
|
|
) {
|
|
return false;
|
|
}
|
|
}
|
|
return set.size === 0;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// TODO(standvpmnt): add types for argument
|
|
function mapMightHaveLoosePrimitive(
|
|
map1: Map<unknown, unknown>,
|
|
map2: Map<unknown, unknown>,
|
|
primitive: any,
|
|
item: any,
|
|
memos: Memo,
|
|
): boolean {
|
|
const altValue = findLooseMatchingPrimitives(primitive);
|
|
if (altValue != null) {
|
|
return altValue;
|
|
}
|
|
const curB = map2.get(altValue);
|
|
if (
|
|
(curB === undefined && !map2.has(altValue)) ||
|
|
!innerDeepEqual(item, curB, false, memo)
|
|
) {
|
|
return false;
|
|
}
|
|
return !map1.has(altValue) && innerDeepEqual(item, curB, false, memos);
|
|
}
|
|
|
|
function mapEquiv(map1: any, map2: any, strict: boolean, memos: Memo): boolean {
|
|
let set = null;
|
|
|
|
for (const { 0: key, 1: item1 } of map1) {
|
|
if (typeof key === "object" && key !== null) {
|
|
if (set === null) {
|
|
set = new Set();
|
|
}
|
|
set.add(key);
|
|
} else {
|
|
const item2 = map2.get(key);
|
|
if (
|
|
(
|
|
(item2 === undefined && !map2.has(key)) ||
|
|
!innerDeepEqual(item1, item2, strict, memos)
|
|
)
|
|
) {
|
|
if (strict) return false;
|
|
if (!mapMightHaveLoosePrimitive(map1, map2, key, item1, memos)) {
|
|
return false;
|
|
}
|
|
if (set === null) {
|
|
set = new Set();
|
|
}
|
|
set.add(key);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (set !== null) {
|
|
for (const { 0: key, 1: item } of map2) {
|
|
if (typeof key === "object" && key !== null) {
|
|
if (!mapHasEqualEntry(set, map1, key, item, strict, memos)) {
|
|
return false;
|
|
}
|
|
} else if (
|
|
!strict && (!map1.has(key) ||
|
|
!innerDeepEqual(map1.get(key), item, false, memos)) &&
|
|
!mapHasEqualEntry(set, map1, key, item, false, memos)
|
|
) {
|
|
return false;
|
|
}
|
|
}
|
|
return set.size === 0;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
function mapHasEqualEntry(
|
|
set: any,
|
|
map: any,
|
|
key1: any,
|
|
item1: any,
|
|
strict: boolean,
|
|
memos: Memo,
|
|
): boolean {
|
|
for (const key2 of set) {
|
|
if (
|
|
innerDeepEqual(key1, key2, strict, memos) &&
|
|
innerDeepEqual(item1, map.get(key2), strict, memos)
|
|
) {
|
|
set.delete(key2);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|