-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathcreateDeferredExecutor.ts
57 lines (46 loc) · 1.42 KB
/
createDeferredExecutor.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
export type PromiseState = 'pending' | 'fulfilled' | 'rejected'
export type Executor<Value> = ConstructorParameters<typeof Promise<Value>>[0]
export type ResolveFunction<Value> = Parameters<Executor<Value>>[0]
export type RejectFunction<Reason> = Parameters<Executor<Reason>>[1]
export type DeferredPromiseExecutor<Input = never, Output = Input> = {
(resolve?: ResolveFunction<Input>, reject?: RejectFunction<any>): void
resolve: ResolveFunction<Input>
reject: RejectFunction<any>
result?: Output
state: PromiseState
rejectionReason?: unknown
}
export function createDeferredExecutor<
Input = never,
Output = Input
>(): DeferredPromiseExecutor<Input, Output> {
const executor = <DeferredPromiseExecutor<Input, Output>>((
resolve,
reject
) => {
executor.state = 'pending'
executor.resolve = (data) => {
if (executor.state !== 'pending') {
return
}
executor.result = data as Output
const onFulfilled = <Value>(value: Value) => {
executor.state = 'fulfilled'
return value
}
return resolve(
data instanceof Promise ? data : Promise.resolve(data).then(onFulfilled)
)
}
executor.reject = (reason) => {
if (executor.state !== 'pending') {
return
}
queueMicrotask(() => {
executor.state = 'rejected'
})
return reject((executor.rejectionReason = reason))
}
})
return executor
}