-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathindex.ts
151 lines (144 loc) · 4.41 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
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
// eslint-disable-next-line import/no-unresolved
import type {NextFunction, Request, RequestHandler, Response} from 'express'
import type * as core from 'express-serve-static-core'
const isPromise = (maybePromise: any): maybePromise is typeof Promise => !!maybePromise
&& (typeof maybePromise === 'object' || typeof maybePromise === 'function')
&& typeof maybePromise.then === 'function'
export const asyncMiddleware = <
P = core.ParamsDictionary,
ResBody = any,
ReqBody = any,
ReqQuery = core.Query,
Locals extends Record<string, any> = Record<string, any>
>(
middleware: (
req: Request<P, ResBody, ReqBody, ReqQuery, Locals>,
res: Response<ResBody, Locals>,
next: NextFunction
) => Promise<any> | any
) => (
req: Request<P, ResBody, ReqBody, ReqQuery, Locals>,
res: Response<ResBody, Locals>,
next: NextFunction
) => (() => {
let called = false
const cb = <T>(...args: ReadonlyArray<T>) => {
if (called) return
called = true
return next(...args)
}
let maybePromise
try {
maybePromise = middleware(req, res, cb)
} catch (err) {
return cb(err)
}
if (isPromise(maybePromise)) return (async () => {
try {
await maybePromise
} catch (err) {
return cb(err)
}
})()
})()
/**
* wrap async function to connect-like middleware
* @param middleware can return Promise or throw error
* @returns {Function} connect-like middleware
* next function is always called at most once
*/
export default asyncMiddleware
type IRequestHandler<
P,
ResBody,
ReqBody,
ReqQuery,
Locals extends Record<string, any>
> = RequestHandler<P, ResBody, ReqBody, ReqQuery, Locals>
// eslint-disable-next-line no-use-before-define
| IRequestHandlerArray<P, ResBody, ReqBody, ReqQuery, Locals>
type IRequestHandlerArray<P, ResBody, ReqBody, ReqQuery, Locals extends Record<string, any>> = ReadonlyArray<
IRequestHandler<P, ResBody, ReqBody, ReqQuery, Locals>
>
/**
* combine list of middlewares into 1 middlewares
* the combined chain does not break if any middleware returns a rejected promise
* to catch these errors, wrap the middlewares with asyncMiddleware
* @param first
* @param middlewares
* @returns {Function}
*/
export const combineMiddlewares = <
P = core.ParamsDictionary,
ResBody = any,
ReqBody = any,
ReqQuery = core.Query,
Locals extends Record<string, any> = Record<string, any>
>(
first?: IRequestHandler<P, ResBody, ReqBody, ReqQuery, Locals>,
...middlewares: ReadonlyArray<IRequestHandler<P, ResBody, ReqBody, ReqQuery, Locals>>
) => {
while (Array.isArray(first)) [first, ...middlewares] = [...first, ...middlewares]
return (
req: Request<P, ResBody, ReqBody, ReqQuery, Locals>,
res: Response<ResBody, Locals>,
next: NextFunction
) => first
? (first as RequestHandler<P, ResBody, ReqBody, ReqQuery, Locals>)(req, res, (err?: any) => err
? next(err)
: combineMiddlewares(...middlewares)(req, res, next))
: next()
}
let expressMajorVersion = 4
export const mockExpressMajorVersion = (v: number) => expressMajorVersion = v
/**
* mimic the next middleware. For express <= 4.x, synchronous error is caught, and returned rejected promise is ignored.
* While with express >= 5.x, both are caught.
* @param middleware a single middleware
* @return result/error promise
*/
export const middlewareToPromise = <
P = core.ParamsDictionary,
ResBody = any,
ReqBody = any,
ReqQuery = core.Query,
Locals extends Record<string, any> = Record<string, any>
>(
middleware: RequestHandler<P, ResBody, ReqBody, ReqQuery, Locals>
) => (
req: Request<P, ResBody, ReqBody, ReqQuery, Locals>, res: Response<ResBody, Locals>
): Promise<void> => new Promise(
async (resolve, reject) => {
let maybePromise
try {
maybePromise = middleware(req, res, (err?: any) => {
if (err) reject(err)
else resolve()
})
} catch (err) {
reject(err)
return
}
if (isPromise(maybePromise)) {
try {
await maybePromise
} catch (err) {
// ignore rejected promise in express <= 4.x
if (expressMajorVersion >= 5) reject(err)
}
}
}
)
/**
* extended version of middlewareToPromise which allows one or more middleware / array of middlewares
* @param args
*/
export const combineToAsync = <
P = core.ParamsDictionary,
ResBody = any,
ReqBody = any,
ReqQuery = core.Query,
Locals extends Record<string, any> = Record<string, any>
>(
...args: IRequestHandlerArray<P, ResBody, ReqBody, ReqQuery, Locals>
) => middlewareToPromise(combineMiddlewares(...args))