-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmiddleware.ts
125 lines (107 loc) · 4.59 KB
/
middleware.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
import { FunctionHandler, InvocationContext } from '@azure/functions';
import { errorHandler } from './error';
import { stringify } from './util/stringify';
export type ExceptionalResult = { $failed: true; $error: Error };
export type MiddlewareResult<T> = ExceptionalResult | { $failed: false; $result: Awaited<T | undefined> };
export type BeforeExecutionFunction<T = FunctionHandler> = T extends (...a: infer U) => infer R
? (...a: [...U, MiddlewareResult<R>]) => unknown
: never;
export type PostExecutionFunction<T = FunctionHandler> = BeforeExecutionFunction<T>;
export const isErrorResult = <T>(result: MiddlewareResult<T> | ExceptionalResult): result is ExceptionalResult =>
(result as ExceptionalResult)?.$failed;
const middlewareCore =
<T extends FunctionHandler>(
beforeExecution: (BeforeExecutionFunction<T> | false)[],
handler: T,
postExecution: (PostExecutionFunction<T> | false)[],
) =>
async (request: Parameters<T>, context: InvocationContext): Promise<ReturnType<T> | undefined | Error> => {
let handlerResult: MiddlewareResult<ReturnType<T>> = { $failed: false, $result: undefined };
if (beforeExecution) {
const middlewareFunctions = beforeExecution.filter(
(predicate): predicate is BeforeExecutionFunction<T> => predicate !== false,
);
for (const middlewareFunction of middlewareFunctions) {
try {
await middlewareFunction(request, context, handlerResult);
} catch (err) {
if (err instanceof Error) {
handlerResult = { $failed: true, $error: err };
continue;
}
handlerResult = {
$failed: true,
$error: new Error(`Caught ${stringify(err)} which is not of type Error`),
};
}
}
}
if (!handlerResult.$failed) {
try {
handlerResult = { $failed: false, $result: await handler(request, context) };
} catch (err) {
if (err instanceof Error) {
handlerResult = { $failed: true, $error: err };
}
}
}
if (postExecution) {
const middlewareFunctions = postExecution.filter(
(predicate): predicate is PostExecutionFunction<T> => predicate !== false,
);
for (const middlewareFunction of middlewareFunctions) {
await middlewareFunction(request, context, handlerResult);
}
}
if (isErrorResult(handlerResult)) {
context.error(`An caught error occurred in the execution of the handler: ${handlerResult.$error}`);
return handlerResult.$error;
}
return handlerResult.$result;
};
export type Options = {
errorResponseHandler?: (
error: unknown,
context: InvocationContext,
) => {
[key: string]: unknown;
};
disableErrorHandling?: boolean;
};
async function middlewareWrapper<T extends FunctionHandler>(
beforeExecution: (BeforeExecutionFunction<T> | false)[],
handler: T,
postExecution: (PostExecutionFunction<T> | false)[],
request: Parameters<T>,
context: InvocationContext,
opts?: Options,
) {
const result = await middlewareCore(beforeExecution, handler, postExecution)(request, context);
if (result instanceof Error) {
if (opts?.disableErrorHandling) {
throw result;
}
return errorHandler(result, context, opts);
}
return result;
}
export const middleware =
<T extends FunctionHandler>(
beforeExecution: (BeforeExecutionFunction<T> | false)[],
handler: T,
postExecution: (PostExecutionFunction<T> | false)[],
opts?: Options,
) =>
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore - Sorry, but the type `FunctionResult<T = unknown> = T | Promise<T>` is very unhandy
async (request: Parameters<T>[0], context: InvocationContext): ReturnType<T> => {
if (opts?.disableErrorHandling) {
return await middlewareWrapper(beforeExecution, handler, postExecution, request, context, opts);
}
try {
return await middlewareWrapper(beforeExecution, handler, postExecution, request, context, opts);
} catch (error) {
context.error(`An caught error occurred in the execution of the middleware: ${stringify(error)}`);
return errorHandler(error, context, opts);
}
};