Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
MR414N-ID
GitHub Repository: MR414N-ID/botku2
Path: blob/master/node_modules/assert/build/internal/util/comparisons.js
1129 views
1
// Currently in sync with Node.js lib/internal/util/comparisons.js
2
// https://github.com/nodejs/node/commit/112cc7c27551254aa2b17098fb774867f05ed0d9
3
'use strict';
4
5
function _slicedToArray(arr, i) { return _arrayWithHoles(arr) || _iterableToArrayLimit(arr, i) || _nonIterableRest(); }
6
7
function _nonIterableRest() { throw new TypeError("Invalid attempt to destructure non-iterable instance"); }
8
9
function _iterableToArrayLimit(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"] != null) _i["return"](); } finally { if (_d) throw _e; } } return _arr; }
10
11
function _arrayWithHoles(arr) { if (Array.isArray(arr)) return arr; }
12
13
function _typeof(obj) { if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; } return _typeof(obj); }
14
15
var regexFlagsSupported = /a/g.flags !== undefined;
16
17
var arrayFromSet = function arrayFromSet(set) {
18
var array = [];
19
set.forEach(function (value) {
20
return array.push(value);
21
});
22
return array;
23
};
24
25
var arrayFromMap = function arrayFromMap(map) {
26
var array = [];
27
map.forEach(function (value, key) {
28
return array.push([key, value]);
29
});
30
return array;
31
};
32
33
var objectIs = Object.is ? Object.is : require('object-is');
34
var objectGetOwnPropertySymbols = Object.getOwnPropertySymbols ? Object.getOwnPropertySymbols : function () {
35
return [];
36
};
37
var numberIsNaN = Number.isNaN ? Number.isNaN : require('is-nan');
38
39
function uncurryThis(f) {
40
return f.call.bind(f);
41
}
42
43
var hasOwnProperty = uncurryThis(Object.prototype.hasOwnProperty);
44
var propertyIsEnumerable = uncurryThis(Object.prototype.propertyIsEnumerable);
45
var objectToString = uncurryThis(Object.prototype.toString);
46
47
var _require$types = require('util/').types,
48
isAnyArrayBuffer = _require$types.isAnyArrayBuffer,
49
isArrayBufferView = _require$types.isArrayBufferView,
50
isDate = _require$types.isDate,
51
isMap = _require$types.isMap,
52
isRegExp = _require$types.isRegExp,
53
isSet = _require$types.isSet,
54
isNativeError = _require$types.isNativeError,
55
isBoxedPrimitive = _require$types.isBoxedPrimitive,
56
isNumberObject = _require$types.isNumberObject,
57
isStringObject = _require$types.isStringObject,
58
isBooleanObject = _require$types.isBooleanObject,
59
isBigIntObject = _require$types.isBigIntObject,
60
isSymbolObject = _require$types.isSymbolObject,
61
isFloat32Array = _require$types.isFloat32Array,
62
isFloat64Array = _require$types.isFloat64Array;
63
64
function isNonIndex(key) {
65
if (key.length === 0 || key.length > 10) return true;
66
67
for (var i = 0; i < key.length; i++) {
68
var code = key.charCodeAt(i);
69
if (code < 48 || code > 57) return true;
70
} // The maximum size for an array is 2 ** 32 -1.
71
72
73
return key.length === 10 && key >= Math.pow(2, 32);
74
}
75
76
function getOwnNonIndexProperties(value) {
77
return Object.keys(value).filter(isNonIndex).concat(objectGetOwnPropertySymbols(value).filter(Object.prototype.propertyIsEnumerable.bind(value)));
78
} // Taken from https://github.com/feross/buffer/blob/680e9e5e488f22aac27599a57dc844a6315928dd/index.js
79
// original notice:
80
81
/*!
82
* The buffer module from node.js, for the browser.
83
*
84
* @author Feross Aboukhadijeh <[email protected]> <http://feross.org>
85
* @license MIT
86
*/
87
88
89
function compare(a, b) {
90
if (a === b) {
91
return 0;
92
}
93
94
var x = a.length;
95
var y = b.length;
96
97
for (var i = 0, len = Math.min(x, y); i < len; ++i) {
98
if (a[i] !== b[i]) {
99
x = a[i];
100
y = b[i];
101
break;
102
}
103
}
104
105
if (x < y) {
106
return -1;
107
}
108
109
if (y < x) {
110
return 1;
111
}
112
113
return 0;
114
}
115
116
var ONLY_ENUMERABLE = undefined;
117
var kStrict = true;
118
var kLoose = false;
119
var kNoIterator = 0;
120
var kIsArray = 1;
121
var kIsSet = 2;
122
var kIsMap = 3; // Check if they have the same source and flags
123
124
function areSimilarRegExps(a, b) {
125
return regexFlagsSupported ? a.source === b.source && a.flags === b.flags : RegExp.prototype.toString.call(a) === RegExp.prototype.toString.call(b);
126
}
127
128
function areSimilarFloatArrays(a, b) {
129
if (a.byteLength !== b.byteLength) {
130
return false;
131
}
132
133
for (var offset = 0; offset < a.byteLength; offset++) {
134
if (a[offset] !== b[offset]) {
135
return false;
136
}
137
}
138
139
return true;
140
}
141
142
function areSimilarTypedArrays(a, b) {
143
if (a.byteLength !== b.byteLength) {
144
return false;
145
}
146
147
return compare(new Uint8Array(a.buffer, a.byteOffset, a.byteLength), new Uint8Array(b.buffer, b.byteOffset, b.byteLength)) === 0;
148
}
149
150
function areEqualArrayBuffers(buf1, buf2) {
151
return buf1.byteLength === buf2.byteLength && compare(new Uint8Array(buf1), new Uint8Array(buf2)) === 0;
152
}
153
154
function isEqualBoxedPrimitive(val1, val2) {
155
if (isNumberObject(val1)) {
156
return isNumberObject(val2) && objectIs(Number.prototype.valueOf.call(val1), Number.prototype.valueOf.call(val2));
157
}
158
159
if (isStringObject(val1)) {
160
return isStringObject(val2) && String.prototype.valueOf.call(val1) === String.prototype.valueOf.call(val2);
161
}
162
163
if (isBooleanObject(val1)) {
164
return isBooleanObject(val2) && Boolean.prototype.valueOf.call(val1) === Boolean.prototype.valueOf.call(val2);
165
}
166
167
if (isBigIntObject(val1)) {
168
return isBigIntObject(val2) && BigInt.prototype.valueOf.call(val1) === BigInt.prototype.valueOf.call(val2);
169
}
170
171
return isSymbolObject(val2) && Symbol.prototype.valueOf.call(val1) === Symbol.prototype.valueOf.call(val2);
172
} // Notes: Type tags are historical [[Class]] properties that can be set by
173
// FunctionTemplate::SetClassName() in C++ or Symbol.toStringTag in JS
174
// and retrieved using Object.prototype.toString.call(obj) in JS
175
// See https://tc39.github.io/ecma262/#sec-object.prototype.tostring
176
// for a list of tags pre-defined in the spec.
177
// There are some unspecified tags in the wild too (e.g. typed array tags).
178
// Since tags can be altered, they only serve fast failures
179
//
180
// Typed arrays and buffers are checked by comparing the content in their
181
// underlying ArrayBuffer. This optimization requires that it's
182
// reasonable to interpret their underlying memory in the same way,
183
// which is checked by comparing their type tags.
184
// (e.g. a Uint8Array and a Uint16Array with the same memory content
185
// could still be different because they will be interpreted differently).
186
//
187
// For strict comparison, objects should have
188
// a) The same built-in type tags
189
// b) The same prototypes.
190
191
192
function innerDeepEqual(val1, val2, strict, memos) {
193
// All identical values are equivalent, as determined by ===.
194
if (val1 === val2) {
195
if (val1 !== 0) return true;
196
return strict ? objectIs(val1, val2) : true;
197
} // Check more closely if val1 and val2 are equal.
198
199
200
if (strict) {
201
if (_typeof(val1) !== 'object') {
202
return typeof val1 === 'number' && numberIsNaN(val1) && numberIsNaN(val2);
203
}
204
205
if (_typeof(val2) !== 'object' || val1 === null || val2 === null) {
206
return false;
207
}
208
209
if (Object.getPrototypeOf(val1) !== Object.getPrototypeOf(val2)) {
210
return false;
211
}
212
} else {
213
if (val1 === null || _typeof(val1) !== 'object') {
214
if (val2 === null || _typeof(val2) !== 'object') {
215
// eslint-disable-next-line eqeqeq
216
return val1 == val2;
217
}
218
219
return false;
220
}
221
222
if (val2 === null || _typeof(val2) !== 'object') {
223
return false;
224
}
225
}
226
227
var val1Tag = objectToString(val1);
228
var val2Tag = objectToString(val2);
229
230
if (val1Tag !== val2Tag) {
231
return false;
232
}
233
234
if (Array.isArray(val1)) {
235
// Check for sparse arrays and general fast path
236
if (val1.length !== val2.length) {
237
return false;
238
}
239
240
var keys1 = getOwnNonIndexProperties(val1, ONLY_ENUMERABLE);
241
var keys2 = getOwnNonIndexProperties(val2, ONLY_ENUMERABLE);
242
243
if (keys1.length !== keys2.length) {
244
return false;
245
}
246
247
return keyCheck(val1, val2, strict, memos, kIsArray, keys1);
248
} // [browserify] This triggers on certain types in IE (Map/Set) so we don't
249
// wan't to early return out of the rest of the checks. However we can check
250
// if the second value is one of these values and the first isn't.
251
252
253
if (val1Tag === '[object Object]') {
254
// return keyCheck(val1, val2, strict, memos, kNoIterator);
255
if (!isMap(val1) && isMap(val2) || !isSet(val1) && isSet(val2)) {
256
return false;
257
}
258
}
259
260
if (isDate(val1)) {
261
if (!isDate(val2) || Date.prototype.getTime.call(val1) !== Date.prototype.getTime.call(val2)) {
262
return false;
263
}
264
} else if (isRegExp(val1)) {
265
if (!isRegExp(val2) || !areSimilarRegExps(val1, val2)) {
266
return false;
267
}
268
} else if (isNativeError(val1) || val1 instanceof Error) {
269
// Do not compare the stack as it might differ even though the error itself
270
// is otherwise identical.
271
if (val1.message !== val2.message || val1.name !== val2.name) {
272
return false;
273
}
274
} else if (isArrayBufferView(val1)) {
275
if (!strict && (isFloat32Array(val1) || isFloat64Array(val1))) {
276
if (!areSimilarFloatArrays(val1, val2)) {
277
return false;
278
}
279
} else if (!areSimilarTypedArrays(val1, val2)) {
280
return false;
281
} // Buffer.compare returns true, so val1.length === val2.length. If they both
282
// only contain numeric keys, we don't need to exam further than checking
283
// the symbols.
284
285
286
var _keys = getOwnNonIndexProperties(val1, ONLY_ENUMERABLE);
287
288
var _keys2 = getOwnNonIndexProperties(val2, ONLY_ENUMERABLE);
289
290
if (_keys.length !== _keys2.length) {
291
return false;
292
}
293
294
return keyCheck(val1, val2, strict, memos, kNoIterator, _keys);
295
} else if (isSet(val1)) {
296
if (!isSet(val2) || val1.size !== val2.size) {
297
return false;
298
}
299
300
return keyCheck(val1, val2, strict, memos, kIsSet);
301
} else if (isMap(val1)) {
302
if (!isMap(val2) || val1.size !== val2.size) {
303
return false;
304
}
305
306
return keyCheck(val1, val2, strict, memos, kIsMap);
307
} else if (isAnyArrayBuffer(val1)) {
308
if (!areEqualArrayBuffers(val1, val2)) {
309
return false;
310
}
311
} else if (isBoxedPrimitive(val1) && !isEqualBoxedPrimitive(val1, val2)) {
312
return false;
313
}
314
315
return keyCheck(val1, val2, strict, memos, kNoIterator);
316
}
317
318
function getEnumerables(val, keys) {
319
return keys.filter(function (k) {
320
return propertyIsEnumerable(val, k);
321
});
322
}
323
324
function keyCheck(val1, val2, strict, memos, iterationType, aKeys) {
325
// For all remaining Object pairs, including Array, objects and Maps,
326
// equivalence is determined by having:
327
// a) The same number of owned enumerable properties
328
// b) The same set of keys/indexes (although not necessarily the same order)
329
// c) Equivalent values for every corresponding key/index
330
// d) For Sets and Maps, equal contents
331
// Note: this accounts for both named and indexed properties on Arrays.
332
if (arguments.length === 5) {
333
aKeys = Object.keys(val1);
334
var bKeys = Object.keys(val2); // The pair must have the same number of owned properties.
335
336
if (aKeys.length !== bKeys.length) {
337
return false;
338
}
339
} // Cheap key test
340
341
342
var i = 0;
343
344
for (; i < aKeys.length; i++) {
345
if (!hasOwnProperty(val2, aKeys[i])) {
346
return false;
347
}
348
}
349
350
if (strict && arguments.length === 5) {
351
var symbolKeysA = objectGetOwnPropertySymbols(val1);
352
353
if (symbolKeysA.length !== 0) {
354
var count = 0;
355
356
for (i = 0; i < symbolKeysA.length; i++) {
357
var key = symbolKeysA[i];
358
359
if (propertyIsEnumerable(val1, key)) {
360
if (!propertyIsEnumerable(val2, key)) {
361
return false;
362
}
363
364
aKeys.push(key);
365
count++;
366
} else if (propertyIsEnumerable(val2, key)) {
367
return false;
368
}
369
}
370
371
var symbolKeysB = objectGetOwnPropertySymbols(val2);
372
373
if (symbolKeysA.length !== symbolKeysB.length && getEnumerables(val2, symbolKeysB).length !== count) {
374
return false;
375
}
376
} else {
377
var _symbolKeysB = objectGetOwnPropertySymbols(val2);
378
379
if (_symbolKeysB.length !== 0 && getEnumerables(val2, _symbolKeysB).length !== 0) {
380
return false;
381
}
382
}
383
}
384
385
if (aKeys.length === 0 && (iterationType === kNoIterator || iterationType === kIsArray && val1.length === 0 || val1.size === 0)) {
386
return true;
387
} // Use memos to handle cycles.
388
389
390
if (memos === undefined) {
391
memos = {
392
val1: new Map(),
393
val2: new Map(),
394
position: 0
395
};
396
} else {
397
// We prevent up to two map.has(x) calls by directly retrieving the value
398
// and checking for undefined. The map can only contain numbers, so it is
399
// safe to check for undefined only.
400
var val2MemoA = memos.val1.get(val1);
401
402
if (val2MemoA !== undefined) {
403
var val2MemoB = memos.val2.get(val2);
404
405
if (val2MemoB !== undefined) {
406
return val2MemoA === val2MemoB;
407
}
408
}
409
410
memos.position++;
411
}
412
413
memos.val1.set(val1, memos.position);
414
memos.val2.set(val2, memos.position);
415
var areEq = objEquiv(val1, val2, strict, aKeys, memos, iterationType);
416
memos.val1.delete(val1);
417
memos.val2.delete(val2);
418
return areEq;
419
}
420
421
function setHasEqualElement(set, val1, strict, memo) {
422
// Go looking.
423
var setValues = arrayFromSet(set);
424
425
for (var i = 0; i < setValues.length; i++) {
426
var val2 = setValues[i];
427
428
if (innerDeepEqual(val1, val2, strict, memo)) {
429
// Remove the matching element to make sure we do not check that again.
430
set.delete(val2);
431
return true;
432
}
433
}
434
435
return false;
436
} // See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Equality_comparisons_and_sameness#Loose_equality_using
437
// Sadly it is not possible to detect corresponding values properly in case the
438
// type is a string, number, bigint or boolean. The reason is that those values
439
// can match lots of different string values (e.g., 1n == '+00001').
440
441
442
function findLooseMatchingPrimitives(prim) {
443
switch (_typeof(prim)) {
444
case 'undefined':
445
return null;
446
447
case 'object':
448
// Only pass in null as object!
449
return undefined;
450
451
case 'symbol':
452
return false;
453
454
case 'string':
455
prim = +prim;
456
// Loose equal entries exist only if the string is possible to convert to
457
// a regular number and not NaN.
458
// Fall through
459
460
case 'number':
461
if (numberIsNaN(prim)) {
462
return false;
463
}
464
465
}
466
467
return true;
468
}
469
470
function setMightHaveLoosePrim(a, b, prim) {
471
var altValue = findLooseMatchingPrimitives(prim);
472
if (altValue != null) return altValue;
473
return b.has(altValue) && !a.has(altValue);
474
}
475
476
function mapMightHaveLoosePrim(a, b, prim, item, memo) {
477
var altValue = findLooseMatchingPrimitives(prim);
478
479
if (altValue != null) {
480
return altValue;
481
}
482
483
var curB = b.get(altValue);
484
485
if (curB === undefined && !b.has(altValue) || !innerDeepEqual(item, curB, false, memo)) {
486
return false;
487
}
488
489
return !a.has(altValue) && innerDeepEqual(item, curB, false, memo);
490
}
491
492
function setEquiv(a, b, strict, memo) {
493
// This is a lazily initiated Set of entries which have to be compared
494
// pairwise.
495
var set = null;
496
var aValues = arrayFromSet(a);
497
498
for (var i = 0; i < aValues.length; i++) {
499
var val = aValues[i]; // Note: Checking for the objects first improves the performance for object
500
// heavy sets but it is a minor slow down for primitives. As they are fast
501
// to check this improves the worst case scenario instead.
502
503
if (_typeof(val) === 'object' && val !== null) {
504
if (set === null) {
505
set = new Set();
506
} // If the specified value doesn't exist in the second set its an not null
507
// object (or non strict only: a not matching primitive) we'll need to go
508
// hunting for something thats deep-(strict-)equal to it. To make this
509
// O(n log n) complexity we have to copy these values in a new set first.
510
511
512
set.add(val);
513
} else if (!b.has(val)) {
514
if (strict) return false; // Fast path to detect missing string, symbol, undefined and null values.
515
516
if (!setMightHaveLoosePrim(a, b, val)) {
517
return false;
518
}
519
520
if (set === null) {
521
set = new Set();
522
}
523
524
set.add(val);
525
}
526
}
527
528
if (set !== null) {
529
var bValues = arrayFromSet(b);
530
531
for (var _i = 0; _i < bValues.length; _i++) {
532
var _val = bValues[_i]; // We have to check if a primitive value is already
533
// matching and only if it's not, go hunting for it.
534
535
if (_typeof(_val) === 'object' && _val !== null) {
536
if (!setHasEqualElement(set, _val, strict, memo)) return false;
537
} else if (!strict && !a.has(_val) && !setHasEqualElement(set, _val, strict, memo)) {
538
return false;
539
}
540
}
541
542
return set.size === 0;
543
}
544
545
return true;
546
}
547
548
function mapHasEqualEntry(set, map, key1, item1, strict, memo) {
549
// To be able to handle cases like:
550
// Map([[{}, 'a'], [{}, 'b']]) vs Map([[{}, 'b'], [{}, 'a']])
551
// ... we need to consider *all* matching keys, not just the first we find.
552
var setValues = arrayFromSet(set);
553
554
for (var i = 0; i < setValues.length; i++) {
555
var key2 = setValues[i];
556
557
if (innerDeepEqual(key1, key2, strict, memo) && innerDeepEqual(item1, map.get(key2), strict, memo)) {
558
set.delete(key2);
559
return true;
560
}
561
}
562
563
return false;
564
}
565
566
function mapEquiv(a, b, strict, memo) {
567
var set = null;
568
var aEntries = arrayFromMap(a);
569
570
for (var i = 0; i < aEntries.length; i++) {
571
var _aEntries$i = _slicedToArray(aEntries[i], 2),
572
key = _aEntries$i[0],
573
item1 = _aEntries$i[1];
574
575
if (_typeof(key) === 'object' && key !== null) {
576
if (set === null) {
577
set = new Set();
578
}
579
580
set.add(key);
581
} else {
582
// By directly retrieving the value we prevent another b.has(key) check in
583
// almost all possible cases.
584
var item2 = b.get(key);
585
586
if (item2 === undefined && !b.has(key) || !innerDeepEqual(item1, item2, strict, memo)) {
587
if (strict) return false; // Fast path to detect missing string, symbol, undefined and null
588
// keys.
589
590
if (!mapMightHaveLoosePrim(a, b, key, item1, memo)) return false;
591
592
if (set === null) {
593
set = new Set();
594
}
595
596
set.add(key);
597
}
598
}
599
}
600
601
if (set !== null) {
602
var bEntries = arrayFromMap(b);
603
604
for (var _i2 = 0; _i2 < bEntries.length; _i2++) {
605
var _bEntries$_i = _slicedToArray(bEntries[_i2], 2),
606
key = _bEntries$_i[0],
607
item = _bEntries$_i[1];
608
609
if (_typeof(key) === 'object' && key !== null) {
610
if (!mapHasEqualEntry(set, a, key, item, strict, memo)) return false;
611
} else if (!strict && (!a.has(key) || !innerDeepEqual(a.get(key), item, false, memo)) && !mapHasEqualEntry(set, a, key, item, false, memo)) {
612
return false;
613
}
614
}
615
616
return set.size === 0;
617
}
618
619
return true;
620
}
621
622
function objEquiv(a, b, strict, keys, memos, iterationType) {
623
// Sets and maps don't have their entries accessible via normal object
624
// properties.
625
var i = 0;
626
627
if (iterationType === kIsSet) {
628
if (!setEquiv(a, b, strict, memos)) {
629
return false;
630
}
631
} else if (iterationType === kIsMap) {
632
if (!mapEquiv(a, b, strict, memos)) {
633
return false;
634
}
635
} else if (iterationType === kIsArray) {
636
for (; i < a.length; i++) {
637
if (hasOwnProperty(a, i)) {
638
if (!hasOwnProperty(b, i) || !innerDeepEqual(a[i], b[i], strict, memos)) {
639
return false;
640
}
641
} else if (hasOwnProperty(b, i)) {
642
return false;
643
} else {
644
// Array is sparse.
645
var keysA = Object.keys(a);
646
647
for (; i < keysA.length; i++) {
648
var key = keysA[i];
649
650
if (!hasOwnProperty(b, key) || !innerDeepEqual(a[key], b[key], strict, memos)) {
651
return false;
652
}
653
}
654
655
if (keysA.length !== Object.keys(b).length) {
656
return false;
657
}
658
659
return true;
660
}
661
}
662
} // The pair must have equivalent values for every corresponding key.
663
// Possibly expensive deep test:
664
665
666
for (i = 0; i < keys.length; i++) {
667
var _key = keys[i];
668
669
if (!innerDeepEqual(a[_key], b[_key], strict, memos)) {
670
return false;
671
}
672
}
673
674
return true;
675
}
676
677
function isDeepEqual(val1, val2) {
678
return innerDeepEqual(val1, val2, kLoose);
679
}
680
681
function isDeepStrictEqual(val1, val2) {
682
return innerDeepEqual(val1, val2, kStrict);
683
}
684
685
module.exports = {
686
isDeepEqual: isDeepEqual,
687
isDeepStrictEqual: isDeepStrictEqual
688
};
689