Spaces:
Running
Running
/** | |
* @fileoverview A set of methods that mimic a bit of the Jasmine testing library, but simpler and | |
* more succinct for manipulating a comfy integration test. | |
*/ | |
import { wait } from "rgthree/common/shared_utils.js"; | |
type TestContext = { | |
label?: string; | |
beforeEach?: Function[]; | |
}; | |
let contexts: TestContext[] = []; | |
export function describe(label: string, fn: Function) { | |
return async () => { | |
await describeRun(label, fn); | |
}; | |
} | |
export async function describeRun(label: string, fn: Function) { | |
await wait(); | |
contexts.push({ label }); | |
console.group(`[Start] ${contexts[contexts.length - 1]!.label}`); | |
await fn(); | |
contexts.pop(); | |
console.groupEnd(); | |
} | |
export async function should(declaration: string, fn: Function) { | |
if (!contexts[contexts.length - 1]) { | |
throw Error("Called should outside of a describe."); | |
} | |
console.group(`...should ${declaration}`); | |
try { | |
for (const context of contexts) { | |
for (const beforeEachFn of context?.beforeEach || []) { | |
await beforeEachFn(); | |
} | |
} | |
await fn(); | |
} catch (e: any) { | |
fail(e); | |
} | |
console.groupEnd(); | |
} | |
export async function beforeEach(fn: Function) { | |
if (!contexts[contexts.length - 1]) { | |
throw Error("Called beforeEach outside of a describe."); | |
} | |
const last = contexts[contexts.length - 1]!; | |
last.beforeEach = last?.beforeEach || []; | |
last.beforeEach.push(fn); | |
} | |
export function fail(e: Error) { | |
log(`X Failure: ${e}`, "color:#600; background:#fdd; padding: 2px 6px;"); | |
} | |
function log(msg: string, styles: string) { | |
if (styles) { | |
console.log(`%c ${msg}`, styles); | |
} else { | |
console.log(msg); | |
} | |
} | |
class Expectation { | |
private propertyLabel: string | null = ""; | |
private expectedLabel: string | null = ""; | |
private expectedFn!: (v: any) => boolean; | |
private value: any; | |
constructor(value: any) { | |
this.value = value; | |
} | |
toBe(labelOrExpected: any, maybeExpected?: any) { | |
const expected = maybeExpected !== undefined ? maybeExpected : labelOrExpected; | |
this.propertyLabel = maybeExpected !== undefined ? labelOrExpected : null; | |
this.expectedLabel = JSON.stringify(expected); | |
this.expectedFn = (v) => v == expected; | |
return this.toBeEval(); | |
} | |
toBeUndefined(propertyLabel: string) { | |
this.expectedFn = (v) => v === undefined; | |
this.propertyLabel = propertyLabel || ""; | |
this.expectedLabel = "undefined"; | |
return this.toBeEval(true); | |
} | |
toBeNullOrUndefined(propertyLabel: string) { | |
this.expectedFn = (v) => v == null; | |
this.propertyLabel = propertyLabel || ""; | |
this.expectedLabel = "null or undefined"; | |
return this.toBeEval(true); | |
} | |
toBeTruthy(propertyLabel: string) { | |
this.expectedFn = (v) => !v; | |
this.propertyLabel = propertyLabel || ""; | |
this.expectedLabel = "truthy"; | |
return this.toBeEval(false); | |
} | |
toBeANumber(propertyLabel: string) { | |
this.expectedFn = (v) => typeof v === "number"; | |
this.propertyLabel = propertyLabel || ""; | |
this.expectedLabel = "a number"; | |
return this.toBeEval(); | |
} | |
toBeEval(strict = false) { | |
let evaluation = this.expectedFn(this.value); | |
let msg = `Expected ${this.propertyLabel ? this.propertyLabel + " to be " : ""}${ | |
this.expectedLabel | |
}`; | |
msg += evaluation ? "." : `, but was ${JSON.stringify(this.value)}`; | |
this.log(evaluation, msg); | |
return evaluation; | |
} | |
log(value: boolean, msg: string) { | |
if (value) { | |
log(`🗸 ${msg}`, "color:#060; background:#cec; padding: 2px 6px;"); | |
} else { | |
log(`X ${msg}`, "color:#600; background:#fdd; padding: 2px 6px;"); | |
} | |
} | |
} | |
export function expect(value: any, msg?: string) { | |
const expectation = new Expectation(value); | |
if (msg) { | |
expectation.log(value, msg); | |
} | |
return expectation; | |
} | |