-
-
Notifications
You must be signed in to change notification settings - Fork 1.3k
/
Copy pathnodeHTTPRequestHandler.ts
97 lines (88 loc) · 2.82 KB
/
nodeHTTPRequestHandler.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
/**
* If you're making an adapter for tRPC and looking at this file for reference, you should import types and functions from `@trpc/server` and `@trpc/server/http`
*
* @example
* ```ts
* import type { AnyTRPCRouter } from '@trpc/server'
* import type { HTTPBaseHandlerOptions } from '@trpc/server/http'
* ```
*/
// @trpc/server
import { getTRPCErrorFromUnknown, type AnyRouter } from '../../@trpc/server';
import type { ResolveHTTPRequestOptionsContextFn } from '../../@trpc/server/http';
import { resolveResponse } from '../../@trpc/server/http';
import { incomingMessageToRequest } from './incomingMessageToRequest';
import type {
NodeHTTPRequest,
NodeHTTPRequestHandlerOptions,
NodeHTTPResponse,
} from './types';
export async function nodeHTTPRequestHandler<
TRouter extends AnyRouter,
TRequest extends NodeHTTPRequest,
TResponse extends NodeHTTPResponse,
>(opts: NodeHTTPRequestHandlerOptions<TRouter, TRequest, TResponse>) {
const handleViaMiddleware = opts.middleware ?? ((_req, _res, next) => next());
return handleViaMiddleware(opts.req, opts.res, async (err) => {
const req = incomingMessageToRequest(opts.req, {
maxBodySize: opts.maxBodySize ?? null,
});
// Build tRPC dependencies
const createContext: ResolveHTTPRequestOptionsContextFn<TRouter> = async (
innerOpts,
) => {
return await opts.createContext?.({
...opts,
...innerOpts,
});
};
const response = await resolveResponse({
...opts,
req,
error: err ? getTRPCErrorFromUnknown(err) : null,
createContext,
onError(o) {
opts?.onError?.({
...o,
req: opts.req,
});
},
});
const { res } = opts;
if (res.statusCode === 200) {
// if the status code is set, we assume that it's been manually overridden
res.statusCode = response.status;
}
for (const [key, value] of response.headers) {
res.setHeader(key, value);
}
if (response.body) {
const reader = response.body.getReader();
const onAbort = () => {
// cancelling the reader will cause the whole stream to be cancelled
reader.cancel().catch(() => {
// console.error('reader.cancel() error', err);
});
};
req.signal.addEventListener('abort', onAbort);
while (true) {
const { done, value } = await reader.read();
if (done) {
break;
}
if (!res.writable) {
break;
}
if (!res.write(value)) {
await new Promise<void>((resolve) => {
res.once('drain', resolve);
});
}
// IMPORTANT - flush the response buffer, otherwise the client will not receive the data until `.end()`
res.flush?.();
}
req.signal.removeEventListener('abort', onAbort);
}
res.end();
});
}