forked from tjjfvi/subshape
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtest-util.ts
100 lines (92 loc) · 3.26 KB
/
test-util.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
/// <reference lib="deno.unstable"/>
import { assertEquals, assertThrows } from "https://deno.land/std@0.161.0/testing/asserts.ts"
import { assertSnapshot } from "https://deno.land/std@0.161.0/testing/snapshot.ts"
import { AnyShape, assert, DecodeBuffer, Shape, ShapeAssertError } from "./common/mod.ts"
const [lipsum, words, cargoLock] = ["lipsum.txt", "words.txt", "Cargo.lock"].map((fileName) => () =>
Deno.readTextFile(fileName)
)
export const files = { lipsum: lipsum!, words: words!, cargoLock: cargoLock! }
export function testShape<T>(
shape: Shape<T>,
values: NoInfer<T>[] | Record<string, NoInfer<T> | (() => NoInfer<T>)>,
async?: boolean,
): void
export function testShape<T>(
shape: Shape<T>,
values: T[] | Record<string, T | (() => T)>,
async = false,
) {
for (const key in values) {
let value = values[key as never] as T | (() => T)
const label = values instanceof Array
? Deno.inspect(value, {
depth: Infinity,
trailingComma: true,
compact: true,
iterableLimit: Infinity,
})
: key
Deno.test(`${Deno.inspect(shape)} ${label}`, async (t) => {
if (typeof value === "function") {
value = (value as () => T)()
}
assert(shape, value)
const encoded = async ? await shape.encodeAsync(value) : shape.encode(value)
await assertSnapshot(t, encoded, { serializer: serializeU8A })
const decodeBuffer = new DecodeBuffer(encoded)
const decoded = shape.subDecode(decodeBuffer)
assertEquals(decoded, value)
assertEquals(decodeBuffer.index, encoded.length)
assert(shape, decoded)
})
}
}
export function testInvalid(shape: AnyShape, values: unknown[] | Record<string, unknown | (() => unknown)>) {
for (const key in values) {
let value: unknown = values[key as never]
const label = values instanceof Array
? Deno.inspect(value, {
depth: Infinity,
trailingComma: true,
compact: true,
iterableLimit: Infinity,
})
: key
Deno.test(`${Deno.inspect(shape)} invalid ${label}`, async (t) => {
if (typeof value === "function") {
value = (value as () => unknown)()
}
await assertThrowsSnapshot(t, () => assert(shape as Shape<any>, value), ShapeAssertError)
})
}
}
function serializeU8A(array: Uint8Array) {
return [...array].map((x) => x.toString(16).padStart(2, "0")).join("\n")
}
type NoInfer<T> = T extends infer U ? U : never
export function benchShape<T>(name: string, shape: Shape<T>, value: NoInfer<T>): void
export function benchShape<T>(name: string, shape: Shape<T>, value: T) {
const encoded = shape.encode(value)
Deno.bench(`- ${name} (${encoded.length}B) [encode] `, () => {
shape.encode(value)
})
Deno.bench(` ${name} (${encoded.length}B) [decode]`, () => {
shape.decode(encoded)
})
}
export async function assertThrowsSnapshot(
t: Deno.TestContext,
fn: () => unknown,
ctor: abstract new(...args: any) => Error = Error,
) {
try {
fn()
} catch (error) {
if (!(error instanceof ctor)) {
throw new Error("Expected " + ctor.name)
}
return await assertSnapshot(t, `${error.name}: ${error.message}`, { serializer: (x) => x })
}
throw new Error("Expected function to throw")
}
export { assertEquals, assertSnapshot, assertThrows }