2020-01-02 15:13:47 -05:00
|
|
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
2019-08-24 10:38:18 -07:00
|
|
|
import { red, green, white, gray, bold } from "../fmt/colors.ts";
|
2019-08-14 16:22:31 +02:00
|
|
|
import diff, { DiffType, DiffResult } from "./diff.ts";
|
|
|
|
import { format } from "./format.ts";
|
|
|
|
|
|
|
|
const CAN_NOT_DISPLAY = "[Cannot display]";
|
2019-03-05 20:58:28 +01:00
|
|
|
|
|
|
|
interface Constructor {
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
|
|
new (...args: any[]): any;
|
|
|
|
}
|
|
|
|
|
2019-03-08 22:04:43 +01:00
|
|
|
export class AssertionError extends Error {
|
|
|
|
constructor(message: string) {
|
|
|
|
super(message);
|
|
|
|
this.name = "AssertionError";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-14 16:22:31 +02:00
|
|
|
function createStr(v: unknown): string {
|
|
|
|
try {
|
|
|
|
return format(v);
|
|
|
|
} catch (e) {
|
|
|
|
return red(CAN_NOT_DISPLAY);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function createColor(diffType: DiffType): (s: string) => string {
|
|
|
|
switch (diffType) {
|
|
|
|
case DiffType.added:
|
|
|
|
return (s: string): string => green(bold(s));
|
|
|
|
case DiffType.removed:
|
|
|
|
return (s: string): string => red(bold(s));
|
|
|
|
default:
|
|
|
|
return white;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function createSign(diffType: DiffType): string {
|
|
|
|
switch (diffType) {
|
|
|
|
case DiffType.added:
|
|
|
|
return "+ ";
|
|
|
|
case DiffType.removed:
|
|
|
|
return "- ";
|
|
|
|
default:
|
|
|
|
return " ";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function buildMessage(diffResult: ReadonlyArray<DiffResult<string>>): string[] {
|
|
|
|
const messages: string[] = [];
|
|
|
|
messages.push("");
|
|
|
|
messages.push("");
|
|
|
|
messages.push(
|
|
|
|
` ${gray(bold("[Diff]"))} ${red(bold("Left"))} / ${green(bold("Right"))}`
|
|
|
|
);
|
|
|
|
messages.push("");
|
|
|
|
messages.push("");
|
2019-11-14 05:42:34 +11:00
|
|
|
diffResult.forEach((result: DiffResult<string>): void => {
|
|
|
|
const c = createColor(result.type);
|
|
|
|
messages.push(c(`${createSign(result.type)}${result.value}`));
|
|
|
|
});
|
2019-08-14 16:22:31 +02:00
|
|
|
messages.push("");
|
|
|
|
|
|
|
|
return messages;
|
|
|
|
}
|
|
|
|
|
2019-11-04 15:21:43 +00:00
|
|
|
function isKeyedCollection(x: unknown): x is Set<unknown> {
|
|
|
|
return [Symbol.iterator, "size"].every(k => k in (x as Set<unknown>));
|
|
|
|
}
|
|
|
|
|
2019-03-06 22:39:50 +01:00
|
|
|
export function equal(c: unknown, d: unknown): boolean {
|
|
|
|
const seen = new Map();
|
2019-04-24 13:41:23 +02:00
|
|
|
return (function compare(a: unknown, b: unknown): boolean {
|
2019-03-26 13:15:16 +01:00
|
|
|
// Have to render RegExp & Date for string comparison
|
|
|
|
// unless it's mistreated as object
|
|
|
|
if (
|
|
|
|
a &&
|
|
|
|
b &&
|
|
|
|
((a instanceof RegExp && b instanceof RegExp) ||
|
|
|
|
(a instanceof Date && b instanceof Date))
|
|
|
|
) {
|
|
|
|
return String(a) === String(b);
|
|
|
|
}
|
2019-03-06 22:39:50 +01:00
|
|
|
if (Object.is(a, b)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (a && typeof a === "object" && b && typeof b === "object") {
|
|
|
|
if (seen.get(a) === b) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (Object.keys(a || {}).length !== Object.keys(b || {}).length) {
|
|
|
|
return false;
|
|
|
|
}
|
2019-11-04 15:21:43 +00:00
|
|
|
if (isKeyedCollection(a) && isKeyedCollection(b)) {
|
|
|
|
if (a.size !== b.size) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
let unmatchedEntries = a.size;
|
|
|
|
|
|
|
|
for (const [aKey, aValue] of a.entries()) {
|
|
|
|
for (const [bKey, bValue] of b.entries()) {
|
|
|
|
/* Given that Map keys can be references, we need
|
|
|
|
* to ensure that they are also deeply equal */
|
|
|
|
if (
|
|
|
|
(aKey === aValue && bKey === bValue && compare(aKey, bKey)) ||
|
|
|
|
(compare(aKey, bKey) && compare(aValue, bValue))
|
|
|
|
) {
|
|
|
|
unmatchedEntries--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return unmatchedEntries === 0;
|
|
|
|
}
|
2019-03-06 22:39:50 +01:00
|
|
|
const merged = { ...a, ...b };
|
|
|
|
for (const key in merged) {
|
|
|
|
type Key = keyof typeof merged;
|
|
|
|
if (!compare(a && a[key as Key], b && b[key as Key])) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
seen.set(a, b);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
})(c, d);
|
|
|
|
}
|
|
|
|
|
2019-03-05 20:58:28 +01:00
|
|
|
/** Make an assertion, if not `true`, then throw. */
|
2019-11-14 05:42:34 +11:00
|
|
|
export function assert(expr: unknown, msg = ""): asserts expr {
|
2019-03-05 20:58:28 +01:00
|
|
|
if (!expr) {
|
2019-03-08 22:04:43 +01:00
|
|
|
throw new AssertionError(msg);
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Make an assertion that `actual` and `expected` are equal, deeply. If not
|
|
|
|
* deeply equal, then throw.
|
|
|
|
*/
|
2019-03-06 19:42:24 -05:00
|
|
|
export function assertEquals(
|
2019-03-06 22:39:50 +01:00
|
|
|
actual: unknown,
|
|
|
|
expected: unknown,
|
|
|
|
msg?: string
|
|
|
|
): void {
|
2019-08-14 16:22:31 +02:00
|
|
|
if (equal(actual, expected)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let message = "";
|
|
|
|
const actualString = createStr(actual);
|
|
|
|
const expectedString = createStr(expected);
|
|
|
|
try {
|
|
|
|
const diffResult = diff(
|
|
|
|
actualString.split("\n"),
|
|
|
|
expectedString.split("\n")
|
|
|
|
);
|
|
|
|
message = buildMessage(diffResult).join("\n");
|
|
|
|
} catch (e) {
|
|
|
|
message = `\n${red(CAN_NOT_DISPLAY)} + \n\n`;
|
|
|
|
}
|
|
|
|
if (msg) {
|
|
|
|
message = msg;
|
|
|
|
}
|
|
|
|
throw new AssertionError(message);
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
|
|
|
|
2019-03-07 15:08:19 +01:00
|
|
|
/**
|
|
|
|
* Make an assertion that `actual` and `expected` are not equal, deeply.
|
|
|
|
* If not then throw.
|
|
|
|
*/
|
|
|
|
export function assertNotEquals(
|
|
|
|
actual: unknown,
|
|
|
|
expected: unknown,
|
|
|
|
msg?: string
|
|
|
|
): void {
|
|
|
|
if (!equal(actual, expected)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let actualString: string;
|
|
|
|
let expectedString: string;
|
|
|
|
try {
|
|
|
|
actualString = String(actual);
|
|
|
|
} catch (e) {
|
|
|
|
actualString = "[Cannot display]";
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
expectedString = String(expected);
|
|
|
|
} catch (e) {
|
|
|
|
expectedString = "[Cannot display]";
|
|
|
|
}
|
|
|
|
if (!msg) {
|
|
|
|
msg = `actual: ${actualString} expected: ${expectedString}`;
|
|
|
|
}
|
2019-03-08 22:04:43 +01:00
|
|
|
throw new AssertionError(msg);
|
2019-03-07 15:08:19 +01:00
|
|
|
}
|
|
|
|
|
2019-03-05 20:58:28 +01:00
|
|
|
/**
|
|
|
|
* Make an assertion that `actual` and `expected` are strictly equal. If
|
|
|
|
* not then throw.
|
|
|
|
*/
|
|
|
|
export function assertStrictEq(
|
|
|
|
actual: unknown,
|
|
|
|
expected: unknown,
|
|
|
|
msg?: string
|
|
|
|
): void {
|
|
|
|
if (actual !== expected) {
|
|
|
|
let actualString: string;
|
|
|
|
let expectedString: string;
|
|
|
|
try {
|
|
|
|
actualString = String(actual);
|
|
|
|
} catch (e) {
|
|
|
|
actualString = "[Cannot display]";
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
expectedString = String(expected);
|
|
|
|
} catch (e) {
|
|
|
|
expectedString = "[Cannot display]";
|
|
|
|
}
|
|
|
|
if (!msg) {
|
|
|
|
msg = `actual: ${actualString} expected: ${expectedString}`;
|
|
|
|
}
|
2019-03-08 22:04:43 +01:00
|
|
|
throw new AssertionError(msg);
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Make an assertion that actual contains expected. If not
|
|
|
|
* then thrown.
|
|
|
|
*/
|
|
|
|
export function assertStrContains(
|
|
|
|
actual: string,
|
|
|
|
expected: string,
|
|
|
|
msg?: string
|
|
|
|
): void {
|
|
|
|
if (!actual.includes(expected)) {
|
|
|
|
if (!msg) {
|
|
|
|
msg = `actual: "${actual}" expected to contains: "${expected}"`;
|
|
|
|
}
|
2019-03-08 22:04:43 +01:00
|
|
|
throw new AssertionError(msg);
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-07 15:08:19 +01:00
|
|
|
/**
|
|
|
|
* Make an assertion that `actual` contains the `expected` values
|
|
|
|
* If not then thrown.
|
|
|
|
*/
|
|
|
|
export function assertArrayContains(
|
|
|
|
actual: unknown[],
|
|
|
|
expected: unknown[],
|
|
|
|
msg?: string
|
2019-03-12 14:51:51 +09:00
|
|
|
): void {
|
2019-10-06 01:02:34 +09:00
|
|
|
const missing: unknown[] = [];
|
2019-03-07 15:08:19 +01:00
|
|
|
for (let i = 0; i < expected.length; i++) {
|
|
|
|
let found = false;
|
|
|
|
for (let j = 0; j < actual.length; j++) {
|
|
|
|
if (equal(expected[i], actual[j])) {
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!found) {
|
|
|
|
missing.push(expected[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (missing.length === 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!msg) {
|
|
|
|
msg = `actual: "${actual}" expected to contains: "${expected}"`;
|
|
|
|
msg += "\n";
|
|
|
|
msg += `missing: ${missing}`;
|
|
|
|
}
|
2019-03-08 22:04:43 +01:00
|
|
|
throw new AssertionError(msg);
|
2019-03-07 15:08:19 +01:00
|
|
|
}
|
|
|
|
|
2019-03-05 20:58:28 +01:00
|
|
|
/**
|
|
|
|
* Make an assertion that `actual` match RegExp `expected`. If not
|
|
|
|
* then thrown
|
|
|
|
*/
|
|
|
|
export function assertMatch(
|
|
|
|
actual: string,
|
|
|
|
expected: RegExp,
|
|
|
|
msg?: string
|
|
|
|
): void {
|
|
|
|
if (!expected.test(actual)) {
|
|
|
|
if (!msg) {
|
|
|
|
msg = `actual: "${actual}" expected to match: "${expected}"`;
|
|
|
|
}
|
2019-03-08 22:04:43 +01:00
|
|
|
throw new AssertionError(msg);
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Forcefully throws a failed assertion
|
|
|
|
*/
|
|
|
|
export function fail(msg?: string): void {
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-use-before-define
|
|
|
|
assert(false, `Failed assertion${msg ? `: ${msg}` : "."}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Executes a function, expecting it to throw. If it does not, then it
|
|
|
|
* throws. An error class and a string that should be included in the
|
|
|
|
* error message can also be asserted.
|
|
|
|
*/
|
|
|
|
export function assertThrows(
|
|
|
|
fn: () => void,
|
|
|
|
ErrorClass?: Constructor,
|
|
|
|
msgIncludes = "",
|
|
|
|
msg?: string
|
2019-11-15 03:22:33 +00:00
|
|
|
): Error {
|
2019-03-05 20:58:28 +01:00
|
|
|
let doesThrow = false;
|
2019-11-15 03:22:33 +00:00
|
|
|
let error = null;
|
2019-03-05 20:58:28 +01:00
|
|
|
try {
|
|
|
|
fn();
|
|
|
|
} catch (e) {
|
|
|
|
if (ErrorClass && !(Object.getPrototypeOf(e) === ErrorClass.prototype)) {
|
|
|
|
msg = `Expected error to be instance of "${ErrorClass.name}"${
|
|
|
|
msg ? `: ${msg}` : "."
|
|
|
|
}`;
|
2019-03-08 22:04:43 +01:00
|
|
|
throw new AssertionError(msg);
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
2019-03-08 22:04:43 +01:00
|
|
|
if (msgIncludes && !e.message.includes(msgIncludes)) {
|
|
|
|
msg = `Expected error message to include "${msgIncludes}", but got "${
|
|
|
|
e.message
|
|
|
|
}"${msg ? `: ${msg}` : "."}`;
|
|
|
|
throw new AssertionError(msg);
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
|
|
|
doesThrow = true;
|
2019-11-15 03:22:33 +00:00
|
|
|
error = e;
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
|
|
|
if (!doesThrow) {
|
|
|
|
msg = `Expected function to throw${msg ? `: ${msg}` : "."}`;
|
2019-03-08 22:04:43 +01:00
|
|
|
throw new AssertionError(msg);
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
2019-11-15 03:22:33 +00:00
|
|
|
return error;
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
export async function assertThrowsAsync(
|
|
|
|
fn: () => Promise<void>,
|
|
|
|
ErrorClass?: Constructor,
|
|
|
|
msgIncludes = "",
|
|
|
|
msg?: string
|
2019-11-15 03:22:33 +00:00
|
|
|
): Promise<Error> {
|
2019-03-05 20:58:28 +01:00
|
|
|
let doesThrow = false;
|
2019-11-15 03:22:33 +00:00
|
|
|
let error = null;
|
2019-03-05 20:58:28 +01:00
|
|
|
try {
|
|
|
|
await fn();
|
|
|
|
} catch (e) {
|
|
|
|
if (ErrorClass && !(Object.getPrototypeOf(e) === ErrorClass.prototype)) {
|
|
|
|
msg = `Expected error to be instance of "${ErrorClass.name}"${
|
|
|
|
msg ? `: ${msg}` : "."
|
|
|
|
}`;
|
2019-03-08 22:04:43 +01:00
|
|
|
throw new AssertionError(msg);
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
2019-03-08 22:04:43 +01:00
|
|
|
if (msgIncludes && !e.message.includes(msgIncludes)) {
|
|
|
|
msg = `Expected error message to include "${msgIncludes}", but got "${
|
|
|
|
e.message
|
|
|
|
}"${msg ? `: ${msg}` : "."}`;
|
|
|
|
throw new AssertionError(msg);
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
|
|
|
doesThrow = true;
|
2019-11-15 03:22:33 +00:00
|
|
|
error = e;
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
|
|
|
if (!doesThrow) {
|
|
|
|
msg = `Expected function to throw${msg ? `: ${msg}` : "."}`;
|
2019-03-08 22:04:43 +01:00
|
|
|
throw new AssertionError(msg);
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
2019-11-15 03:22:33 +00:00
|
|
|
return error;
|
2019-03-05 20:58:28 +01:00
|
|
|
}
|
2019-03-08 02:32:46 -05:00
|
|
|
|
|
|
|
/** Use this to stub out methods that will throw when invoked. */
|
|
|
|
export function unimplemented(msg?: string): never {
|
2019-03-08 22:04:43 +01:00
|
|
|
throw new AssertionError(msg || "unimplemented");
|
2019-03-08 02:32:46 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Use this to assert unreachable code. */
|
|
|
|
export function unreachable(): never {
|
2019-03-08 22:04:43 +01:00
|
|
|
throw new AssertionError("unreachable");
|
2019-03-08 02:32:46 -05:00
|
|
|
}
|
2020-02-24 01:59:36 +09:00
|
|
|
|
|
|
|
export function assertNotEOF<T extends {}>(val: T | Deno.EOF): T {
|
|
|
|
assertNotEquals(val, Deno.EOF);
|
|
|
|
return val as T;
|
|
|
|
}
|