2017-07-14 17:37:24 +02:00
|
|
|
// Copyright (C) 2017 Ecma International. All rights reserved.
|
|
|
|
// This code is governed by the BSD license found in the LICENSE file.
|
|
|
|
/*---
|
|
|
|
description: |
|
|
|
|
Collection of assertion functions used throughout test262
|
|
|
|
---*/
|
|
|
|
|
2019-09-18 18:39:12 +02:00
|
|
|
|
2014-12-01 21:37:39 +01:00
|
|
|
function assert(mustBeTrue, message) {
|
2017-04-13 16:37:32 +02:00
|
|
|
if (mustBeTrue === true) {
|
|
|
|
return;
|
|
|
|
}
|
2014-12-01 21:37:39 +01:00
|
|
|
|
2017-04-13 16:37:32 +02:00
|
|
|
if (message === undefined) {
|
2019-09-18 18:25:38 +02:00
|
|
|
message = 'Expected true but got ' + assert._toString(mustBeTrue);
|
2017-04-13 16:37:32 +02:00
|
|
|
}
|
|
|
|
$ERROR(message);
|
2014-12-01 21:37:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
assert._isSameValue = function (a, b) {
|
2017-04-13 16:37:32 +02:00
|
|
|
if (a === b) {
|
|
|
|
// Handle +/-0 vs. -/+0
|
|
|
|
return a !== 0 || 1 / a === 1 / b;
|
|
|
|
}
|
2014-12-01 21:37:39 +01:00
|
|
|
|
2017-04-13 16:37:32 +02:00
|
|
|
// Handle NaN vs. NaN
|
|
|
|
return a !== a && b !== b;
|
2014-12-01 21:37:39 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
assert.sameValue = function (actual, expected, message) {
|
2018-10-23 23:25:58 +02:00
|
|
|
try {
|
|
|
|
if (assert._isSameValue(actual, expected)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} catch (error) {
|
|
|
|
$ERROR(message + ' (_isSameValue operation threw) ' + error);
|
2017-04-13 16:37:32 +02:00
|
|
|
return;
|
|
|
|
}
|
2014-12-01 21:37:39 +01:00
|
|
|
|
2017-04-13 16:37:32 +02:00
|
|
|
if (message === undefined) {
|
|
|
|
message = '';
|
|
|
|
} else {
|
|
|
|
message += ' ';
|
|
|
|
}
|
2015-04-21 19:15:19 +02:00
|
|
|
|
2019-09-18 18:25:38 +02:00
|
|
|
message += 'Expected SameValue(«' + assert._toString(actual) + '», «' + assert._toString(expected) + '») to be true';
|
2015-04-21 19:15:19 +02:00
|
|
|
|
2017-04-13 16:37:32 +02:00
|
|
|
$ERROR(message);
|
2014-12-01 21:37:39 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
assert.notSameValue = function (actual, unexpected, message) {
|
2017-04-13 16:37:32 +02:00
|
|
|
if (!assert._isSameValue(actual, unexpected)) {
|
|
|
|
return;
|
|
|
|
}
|
2014-12-01 21:37:39 +01:00
|
|
|
|
2017-04-13 16:37:32 +02:00
|
|
|
if (message === undefined) {
|
|
|
|
message = '';
|
|
|
|
} else {
|
|
|
|
message += ' ';
|
|
|
|
}
|
2015-04-21 19:15:19 +02:00
|
|
|
|
2019-09-18 18:25:38 +02:00
|
|
|
message += 'Expected SameValue(«' + assert._toString(actual) + '», «' + assert._toString(unexpected) + '») to be false';
|
2015-04-21 19:15:19 +02:00
|
|
|
|
2017-04-13 16:37:32 +02:00
|
|
|
$ERROR(message);
|
2014-12-01 21:37:39 +01:00
|
|
|
};
|
2014-12-04 00:29:52 +01:00
|
|
|
|
2015-07-08 17:19:59 +02:00
|
|
|
assert.throws = function (expectedErrorConstructor, func, message) {
|
2017-04-13 16:37:32 +02:00
|
|
|
if (typeof func !== "function") {
|
|
|
|
$ERROR('assert.throws requires two arguments: the error constructor ' +
|
|
|
|
'and a function to run');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (message === undefined) {
|
|
|
|
message = '';
|
|
|
|
} else {
|
|
|
|
message += ' ';
|
|
|
|
}
|
2014-12-04 00:29:52 +01:00
|
|
|
|
2017-04-13 16:37:32 +02:00
|
|
|
try {
|
|
|
|
func();
|
|
|
|
} catch (thrown) {
|
|
|
|
if (typeof thrown !== 'object' || thrown === null) {
|
|
|
|
message += 'Thrown value was not an object!';
|
|
|
|
$ERROR(message);
|
|
|
|
} else if (thrown.constructor !== expectedErrorConstructor) {
|
|
|
|
message += 'Expected a ' + expectedErrorConstructor.name + ' but got a ' + thrown.constructor.name;
|
|
|
|
$ERROR(message);
|
2014-12-04 00:29:52 +01:00
|
|
|
}
|
2017-04-13 16:37:32 +02:00
|
|
|
return;
|
|
|
|
}
|
2014-12-04 00:29:52 +01:00
|
|
|
|
2017-04-13 16:37:32 +02:00
|
|
|
message += 'Expected a ' + expectedErrorConstructor.name + ' to be thrown but no exception was thrown at all';
|
|
|
|
$ERROR(message);
|
2014-12-04 00:29:52 +01:00
|
|
|
};
|
2019-09-18 18:25:38 +02:00
|
|
|
|
2019-09-18 19:15:53 +02:00
|
|
|
assert._formatValue = (value, seen) => {
|
2019-09-18 18:39:12 +02:00
|
|
|
switch (typeof value) {
|
|
|
|
case 'string':
|
2019-09-18 19:15:53 +02:00
|
|
|
return typeof JSON !== "undefined" ? JSON.stringify(value) : `"${value}"`;
|
2019-09-18 18:39:12 +02:00
|
|
|
case 'number':
|
|
|
|
case 'boolean':
|
|
|
|
case 'symbol':
|
|
|
|
case 'bigint':
|
|
|
|
return value.toString();
|
|
|
|
case 'undefined':
|
|
|
|
return 'undefined';
|
|
|
|
case 'function':
|
2019-09-18 19:15:53 +02:00
|
|
|
return `[Function${value.name ? `: ${value.name}` : ''}]`;
|
2019-09-18 18:39:12 +02:00
|
|
|
case 'object':
|
|
|
|
if (value === null) return 'null';
|
2019-09-18 19:15:53 +02:00
|
|
|
if (value instanceof Date) return `Date "${value.toISOString()}"`;
|
2019-09-18 18:39:12 +02:00
|
|
|
if (value instanceof RegExp) return value.toString();
|
|
|
|
if (!seen) {
|
|
|
|
seen = {
|
|
|
|
counter: 0,
|
|
|
|
map: new Map()
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-09-18 19:15:53 +02:00
|
|
|
let usage = seen.map.get(value);
|
2019-09-18 18:39:12 +02:00
|
|
|
if (usage) {
|
|
|
|
usage.used = true;
|
2019-09-18 19:15:53 +02:00
|
|
|
return `[Ref: #${usage.id}]`;
|
2019-09-18 18:39:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
usage = { id: ++seen.counter, used: false };
|
|
|
|
seen.map.set(value, usage);
|
|
|
|
|
|
|
|
if (typeof Set !== "undefined" && value instanceof Set) {
|
2019-09-18 19:15:53 +02:00
|
|
|
return `Set {${Array.from(value).map(value => assert._formatValue(value, seen)).join(', ')}}${usage.used ? ` as #${usage.id}` : ''}`;
|
2019-09-18 18:39:12 +02:00
|
|
|
}
|
|
|
|
if (typeof Map !== "undefined" && value instanceof Map) {
|
2019-09-18 19:15:53 +02:00
|
|
|
return `Map {${Array.from(value).map(pair => `${assert._formatValue(pair[0], seen)} => ${assert._formatValue(pair[1], seen)}}`).join(', ')}}${usage.used ? ` as #${usage.id}` : ''}`;
|
2019-09-18 18:39:12 +02:00
|
|
|
}
|
|
|
|
if (Array.isArray ? Array.isArray(value) : value instanceof Array) {
|
2019-09-18 19:15:53 +02:00
|
|
|
return `[${value.map(value => assert._formatValue(value, seen)).join(', ')}]${usage.used ? ` as #${usage.id}` : ''}`;
|
2019-09-18 18:39:12 +02:00
|
|
|
}
|
2019-09-18 19:15:53 +02:00
|
|
|
let tag = Symbol.toStringTag in value ? value[Symbol.toStringTag] : 'Object';
|
2019-09-18 18:39:12 +02:00
|
|
|
if (tag === 'Object' && Object.getPrototypeOf(value) === null) {
|
|
|
|
tag = '[Object: null prototype]';
|
|
|
|
}
|
2019-09-18 19:15:53 +02:00
|
|
|
return `${tag ? `${tag} ` : ''}{ ${Object.keys(value).map(key => `${key.toString()}: ${assert._formatValue(value[key], seen)}`).join(', ')} }${usage.used ? ` as #${usage.id}` : ''}`;
|
2019-09-18 18:39:12 +02:00
|
|
|
default:
|
|
|
|
return typeof value;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-09-18 18:25:38 +02:00
|
|
|
assert._toString = function (value) {
|
|
|
|
try {
|
|
|
|
return String(value);
|
|
|
|
} catch (err) {
|
|
|
|
if (err.name === 'TypeError') {
|
|
|
|
return Object.prototype.toString.call(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
};
|