-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.ts
73 lines (66 loc) · 1.77 KB
/
index.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
/**
* An async deferrable statement.
*/
export type Deferrable = () => void | Promise<void>;
/**
* Wraps `fn` to allow for deferred functions. Use `yield () => ...` to
* execute code after the function has been resolved or rejected.
*/
export function defer<Args extends unknown[], Return>(
fn: (
...args: Args
) => AsyncGenerator<Deferrable, Return> | Generator<Deferrable, Return>,
): (...args: Args) => Promise<Return> {
return async (...args) => {
const stack: Array<Deferrable> = [];
try {
const generator = fn(...args);
while (true) {
const result = await generator.next();
if (result.done) return result.value;
stack.push(result.value);
}
} finally {
let result = Promise.resolve();
while (stack.length) {
const fn = stack.pop()!;
result = result.finally(() => fn());
}
await result;
}
};
}
/**
* A deferrable statement.
*/
export type DeferrableSync = () => void;
/**
* Wraps `fn` to allow for deferred functions. Use `yield () => ...` to
* execute code after the function has returned or thrown.
*/
export function deferSync<Args extends unknown[], Return>(
fn: (...args: Args) => Generator<DeferrableSync, Return>,
): (...args: Args) => Return {
return (...args) => {
const stack: Array<DeferrableSync> = [];
try {
const generator = fn(...args);
while (true) {
const result = generator.next();
if (result.done) return result.value;
stack.push(result.value);
}
} finally {
let error: unknown = stack;
while (stack.length) {
const fn = stack.pop()!;
try {
fn();
} catch (_error) {
error = _error;
}
}
if (error !== stack) throw error;
}
};
}