-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex.ts
130 lines (106 loc) · 3.2 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
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
const customInspectSymbol = Symbol.for("nodejs.util.inspect.custom");
const isContext = Symbol("isContext");
export type Context = {
readonly signal?: AbortSignal;
readonly deadline?: number;
// hidden symbol to identify Context objects
[isContext]: true;
};
export type Abortable = {
readonly ctx: Context;
readonly abort: () => void;
};
function rootContext(): Context {
return Object.freeze(
Object.create(null, {
[customInspectSymbol]: { value: inspectContext },
[isContext]: { value: true },
})
);
}
export const background = rootContext();
export const TODO = rootContext();
export const requestIdKey = Symbol("requestId");
export function getRequestId(ctx: Context): string | undefined {
if (requestIdKey in ctx) {
// typescript doesn't like symbols, but it's safe
return String((ctx as Record<typeof requestIdKey, unknown>)[requestIdKey]);
}
}
export function withValues(
parent: Context,
values: {
readonly signal?: AbortSignal;
readonly deadline?: number;
} & Record<symbol | string, unknown>
): Context {
if (!parent[isContext]) throw new TypeError("parent is not a Context");
const ctx = Object.create(parent);
for (const key of Object.keys(values)) {
Object.defineProperty(ctx, key, { value: values[key], enumerable: true });
}
for (const key of Object.getOwnPropertySymbols(values)) {
Object.defineProperty(ctx, key, {
// typescript doesn't like this, but it's safe
// eslint-disable-next-line @typescript-eslint/no-explicit-any
value: values[key as any],
enumerable: true,
});
}
return Object.freeze(ctx);
}
export function withAbort(parent: Context): Abortable {
const ac = new AbortController();
function abort() {
parent.signal?.removeEventListener("abort", abort);
ac.abort();
}
parent.signal?.addEventListener("abort", abort, { once: true });
const ctx = withValues(parent, { signal: ac.signal });
return { ctx, abort };
}
export function withDeadline(
parent: Context,
deadline: Date | number
): Abortable {
const dl = typeof deadline === "number" ? deadline : deadline.getTime();
if (parent.deadline && dl > parent.deadline) {
return withAbort(parent);
}
const ac = new AbortController();
const t = setTimeout(() => {
abort();
}, dl - Date.now());
function abort(): void {
parent.signal?.removeEventListener("abort", abort);
clearTimeout(t);
ac.abort();
}
parent.signal?.addEventListener("abort", abort);
const ctx = withValues(parent, { signal: ac.signal, deadline: dl });
return { ctx, abort };
}
export function withTimeout(parent: Context, duration: number): Abortable {
return withDeadline(parent, Date.now() + duration);
}
function recursiveSpread(obj: object): object {
const p = Object.getPrototypeOf(obj);
if (p === null) {
return { ...obj };
}
return { ...recursiveSpread(p), ...obj };
}
function inspectContext(
this: Context,
depth: number,
options: unknown,
inspect: (obj: unknown, opts: unknown) => string
): string {
switch (this) {
case background:
return "Context< background >";
case TODO:
return "Context< TODO >";
}
return `Context< ${inspect(recursiveSpread(this), options)} >`;
}