-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathindex.ts
92 lines (81 loc) · 2.28 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
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
export interface Context<T> {
value<K extends keyof T>(key: K): T[K];
}
/**
* The `BackgroundContext` object implements the default `Context` interface.
*/
class BackgroundContext implements Context<{}> {
value(): never {
return undefined as never;
}
}
/**
* The `ValueContext` object implements a chain-able `Context` interface.
*/
class ValueContext<P, T> implements Context<T> {
constructor(
private p: Context<P>,
private k: keyof T,
private v: T[typeof k]
) {}
value<K extends keyof (T & P)>(key: K): (T & P)[K] {
if (key === this.k) return this.v as (T & P)[K];
return this.p.value(key as keyof P) as (T & P)[K];
}
}
/**
* Initial context.
*/
export const background: Context<{}> = new BackgroundContext();
/**
* Create a `context` object that inherits from the parent values.
*/
export function withValue<T, K extends PropertyKey, V extends any>(
parent: Context<T>,
key: K,
value: V
): Context<T & Record<K, V>> {
return new ValueContext<T, Record<K, V>>(parent, key, value);
}
/**
* Abort function type.
*/
export type AbortFn = (reason: Error) => void;
/**
* Abort symbol for context.
*/
const abortKey = Symbol("abort");
/**
* Values used to manage `abort` in the context.
*/
export type AbortContextValue = Record<typeof abortKey, Promise<never>>;
/**
* Create a cancellable `context` object.
*/
export function withAbort<T>(
parent: Context<T & Partial<AbortContextValue>>
): [Context<T & AbortContextValue>, AbortFn] {
let abort: AbortFn;
let prev: Promise<never> | undefined;
const promise = new Promise<never>((_, reject) => (abort = reject));
(prev = parent.value(abortKey)) && prev.catch(abort!); // Propagate aborts.
return [withValue(parent, abortKey, promise), abort!];
}
/**
* Create a `context` which aborts after _ms_.
*/
export function withTimeout<T>(
parent: Context<T>,
ms: number
): [Context<T & AbortContextValue>, AbortFn] {
const [ctx, cancel] = withAbort(parent);
const timeout = setTimeout(cancel, ms, new Error("Context timed out"));
const abort = (reason: Error) => (clearTimeout(timeout), cancel(reason));
return [ctx, abort];
}
/**
* Use the abort signal.
*/
export function useAbort(ctx: Context<Partial<AbortContextValue>>) {
return ctx.value(abortKey) || new Promise<never>(() => 0);
}