@riddance/host 0.0.12 → 0.0.14

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/host/http.js CHANGED
@@ -1,211 +1,212 @@
1
- import { parse } from 'node:url';
2
- import { measure } from '../context.js';
3
- export async function executeRequest(log, context, handler, options, success) {
4
- const isShallow = context.env.SHALLOW_KEY && options.headers?.['x-shallow'] === context.env.SHALLOW_KEY;
5
- const includeBodyInLogs = !handler.config?.excludeBodyFromLogs;
6
- const logRequest = includeBodyInLogs
7
- ? { method: handler.method, ...options }
8
- : withoutRequestBody({ method: handler.method, ...options });
9
- log = log.enrichReserved({ meta: context.meta, request: logRequest });
10
- if (isShallow) {
11
- log.trace('Shallow request');
12
- return {
13
- headers: {},
14
- status: 204,
15
- };
16
- }
17
- log.trace('Request BEGIN');
18
- try {
19
- let parsedUrl;
20
- let pathSteps;
21
- const req = {
22
- rawUrl: options.uri,
23
- get url() {
24
- return (parsedUrl ??= {
25
- ...parse(this.rawUrl, true),
26
- pathStepAt: (index) => {
27
- const steps = (pathSteps ??= parsedUrl.pathname?.split('/') ?? []);
28
- const step = steps[index + 1];
29
- if (!step) {
30
- throw new RangeError(`Path does not have a step at index ${index}.`);
31
- }
32
- return step;
33
- },
34
- });
35
- },
36
- body: requestBody(options),
37
- headers: options.headers ?? {},
38
- };
39
- const result = await measure(log, 'execution', () => handler.entry({ ...context, log }, req));
40
- const response = resultToResponse(result, includeBodyInLogs);
41
- if (context.signal.aborted) {
42
- response.headers = {
43
- 'x-timeout': '1',
44
- ...response.headers,
45
- };
46
- }
47
- log = log.enrichReserved({
48
- response: {
49
- status: response.status,
50
- headers: response.headers,
51
- body: response.logBody,
52
- },
53
- });
54
- if (response.status < 300) {
55
- log.debug('Request END');
56
- await success();
57
- }
58
- else {
59
- log.warn('Request END');
60
- }
61
- return response;
62
- }
63
- catch (e) {
64
- try {
65
- const response = errorToResponse(e);
66
- log = log.enrichReserved({ response });
67
- log.error('Request END', e);
68
- return response;
69
- }
70
- catch (convertError) {
71
- log.error('Could not convert exception to error response.', convertError);
72
- return {
73
- headers: {},
74
- status: 500,
75
- };
76
- }
77
- }
78
- }
79
- function resultToResponse(result, withLogBody) {
80
- if (!result) {
81
- return {
82
- headers: {},
83
- status: 204,
84
- };
85
- }
86
- else if (typeof result === 'string') {
87
- const logBody = withLogBody ? result : undefined;
88
- return {
89
- headers: {
90
- 'content-type': 'text/plain',
91
- },
92
- status: 200,
93
- body: result,
94
- logBody,
95
- };
96
- }
97
- else {
98
- if (result.body === undefined) {
99
- return {
100
- headers: result.headers ?? {},
101
- status: result.status ?? 200,
102
- };
103
- }
104
- else if (typeof result.body === 'string') {
105
- const logBody = withLogBody ? result.body : undefined;
106
- return {
107
- headers: withContentType(result.headers, 'text/plain'),
108
- status: result.status ?? 200,
109
- body: result.body,
110
- logBody,
111
- };
112
- }
113
- else if (Buffer.isBuffer(result.body)) {
114
- const logBody = withLogBody ? result.body.toString('base64') : undefined;
115
- return {
116
- headers: withContentType(result.headers, 'application/octet-stream'),
117
- status: result.status ?? 200,
118
- body: result.body,
119
- logBody,
120
- };
121
- }
122
- else {
123
- const logBody = withLogBody ? result.body : undefined;
124
- return {
125
- headers: withContentType(result.headers, 'application/json'),
126
- status: result.status ?? 200,
127
- body: JSON.stringify(result.body),
128
- logBody,
129
- };
130
- }
131
- }
132
- }
133
- function withoutRequestBody(options) {
134
- if (hasJsonBody(options)) {
135
- const { json, ...bodyless } = options;
136
- return bodyless;
137
- }
138
- if (hasStringBody(options)) {
139
- const { body, ...bodyless } = options;
140
- return bodyless;
141
- }
142
- return options;
143
- }
144
- function requestBody(options) {
145
- if (hasJsonBody(options)) {
146
- return options.json;
147
- }
148
- if (hasStringBody(options)) {
149
- return options.body;
150
- }
151
- return undefined;
152
- }
153
- function hasJsonBody(options) {
154
- return options.json !== undefined;
155
- }
156
- function hasStringBody(options) {
157
- return options.body !== undefined;
158
- }
159
- function withContentType(headers, contentType) {
160
- if (!headers) {
161
- return {
162
- 'content-type': contentType,
163
- };
164
- }
165
- if (!headers['content-type']) {
166
- headers['content-type'] = contentType;
167
- }
168
- return headers;
169
- }
170
- function errorToResponse(e) {
171
- const { body, statusCode: status } = e;
172
- if (typeof body === 'string') {
173
- return {
174
- headers: {
175
- 'content-type': 'text/plain',
176
- },
177
- status: status ?? 500,
178
- body,
179
- };
180
- }
181
- else if (typeof body === 'object') {
182
- return {
183
- headers: {
184
- 'content-type': 'application/json',
185
- },
186
- status: status ?? 500,
187
- body: JSON.stringify(body),
188
- };
189
- }
190
- else {
191
- return {
192
- headers: {},
193
- status: status ?? 500,
194
- };
195
- }
196
- }
197
- export function clientFromHeaders(headers) {
198
- if (!headers) {
199
- return {};
200
- }
201
- return {
202
- operationId: headers['x-request-id'] ?? headers['request-id'],
203
- clientId: headers['x-client-id'] ??
204
- headers['x-installation-id'] ??
205
- headers['client-id'] ??
206
- headers['installation-id'],
207
- clientIp: headers['x-forwarded-for'],
208
- userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],
209
- };
210
- }
211
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http.js","sourceRoot":"","sources":["http.ts"],"names":[],"mappings":"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","sourcesContent":["import { parse, UrlWithParsedQuery } from 'node:url'\nimport { Context, measure } from '../context.js'\nimport type { Json, ResponseHeaders, Result } from '../http.js'\nimport { ClientInfo, RootLogger } from './context.js'\nimport type { HttpHandler } from './registry.js'\n\nexport type Response = {\n    headers: { readonly [key: string]: string }\n    status: number\n    body?: string | Buffer\n}\n\ntype RequestOptions = BodylessRequestOptions | StringRequestOptions | JsonRequestOptions\n\ntype BodylessRequestOptions = {\n    uri: string\n    headers?: { readonly [key: string]: string }\n}\n\ntype StringRequestOptions = BodylessRequestOptions & {\n    body: string\n}\n\ntype JsonRequestOptions = BodylessRequestOptions & {\n    json: Json\n}\n\nexport async function executeRequest(\n    log: RootLogger,\n    context: Omit<Context, 'log'>,\n    handler: HttpHandler,\n    options: RequestOptions,\n    success: () => Promise<unknown>,\n): Promise<Response> {\n    const isShallow =\n        context.env.SHALLOW_KEY && options.headers?.['x-shallow'] === context.env.SHALLOW_KEY\n    const includeBodyInLogs = !handler.config?.excludeBodyFromLogs\n    const logRequest = includeBodyInLogs\n        ? { method: handler.method, ...options }\n        : withoutRequestBody({ method: handler.method, ...options })\n    log = log.enrichReserved({ meta: context.meta, request: logRequest })\n    if (isShallow) {\n        log.trace('Shallow request')\n        return {\n            headers: {},\n            status: 204,\n        }\n    }\n    log.trace('Request BEGIN')\n    try {\n        let parsedUrl: UrlWithParsedQuery & { pathStepAt: (index: number) => string }\n        let pathSteps: string[]\n        const req = {\n            rawUrl: options.uri,\n            get url() {\n                return (parsedUrl ??= {\n                    ...parse(this.rawUrl, true),\n                    pathStepAt: (index: number) => {\n                        const steps = (pathSteps ??= parsedUrl.pathname?.split('/') ?? [])\n                        const step = steps[index + 1]\n                        if (!step) {\n                            throw new RangeError(`Path does not have a step at index ${index}.`)\n                        }\n                        return step\n                    },\n                })\n            },\n            body: requestBody(options),\n            headers: options.headers ?? {},\n        }\n\n        const result = await measure(log, 'execution', () =>\n            handler.entry({ ...context, log }, req),\n        )\n\n        const response = resultToResponse(result, includeBodyInLogs)\n\n        if (context.signal.aborted) {\n            response.headers = {\n                'x-timeout': '1',\n                ...response.headers,\n            }\n        }\n\n        log = log.enrichReserved({\n            response: {\n                status: response.status,\n                headers: response.headers,\n                body: response.logBody,\n            },\n        })\n        if (response.status < 300) {\n            log.debug('Request END')\n            await success()\n        } else {\n            log.warn('Request END')\n        }\n        return response\n    } catch (e) {\n        try {\n            const response = errorToResponse(e)\n            log = log.enrichReserved({ response })\n            log.error('Request END', e)\n            return response\n        } catch (convertError) {\n            log.error('Could not convert exception to error response.', convertError)\n            return {\n                headers: {},\n                status: 500,\n            }\n        }\n    }\n}\n\nfunction resultToResponse(result: Result, withLogBody: boolean): Response & { logBody?: unknown } {\n    if (!result) {\n        return {\n            headers: {},\n            status: 204,\n        }\n    } else if (typeof result === 'string') {\n        const logBody = withLogBody ? result : undefined\n        return {\n            headers: {\n                'content-type': 'text/plain',\n            },\n            status: 200,\n            body: result,\n            logBody,\n        }\n    } else {\n        if (result.body === undefined) {\n            return {\n                headers: result.headers ?? {},\n                status: result.status ?? 200,\n            }\n        } else if (typeof result.body === 'string') {\n            const logBody = withLogBody ? result.body : undefined\n            return {\n                headers: withContentType(result.headers, 'text/plain'),\n                status: result.status ?? 200,\n                body: result.body,\n                logBody,\n            }\n        } else if (Buffer.isBuffer(result.body)) {\n            const logBody = withLogBody ? result.body.toString('base64') : undefined\n            return {\n                headers: withContentType(result.headers, 'application/octet-stream'),\n                status: result.status ?? 200,\n                body: result.body,\n                logBody,\n            }\n        } else {\n            const logBody = withLogBody ? result.body : undefined\n            return {\n                headers: withContentType(result.headers, 'application/json'),\n                status: result.status ?? 200,\n                body: JSON.stringify(result.body),\n                logBody,\n            }\n        }\n    }\n}\n\nfunction withoutRequestBody(options: RequestOptions & { method: string }) {\n    if (hasJsonBody(options)) {\n        const { json, ...bodyless } = options\n        return bodyless\n    }\n    if (hasStringBody(options)) {\n        const { body, ...bodyless } = options\n        return bodyless\n    }\n    return options\n}\n\nfunction requestBody(options: RequestOptions): Json | string | undefined {\n    if (hasJsonBody(options)) {\n        return options.json\n    }\n    if (hasStringBody(options)) {\n        return options.body\n    }\n    return undefined\n}\n\nfunction hasJsonBody(options: RequestOptions): options is JsonRequestOptions {\n    return (options as { json?: unknown }).json !== undefined\n}\n\nfunction hasStringBody(options: RequestOptions): options is StringRequestOptions {\n    return (options as { body?: unknown }).body !== undefined\n}\n\nfunction withContentType(headers: ResponseHeaders | undefined, contentType: string) {\n    if (!headers) {\n        return {\n            'content-type': contentType,\n        }\n    }\n    if (!headers['content-type']) {\n        headers['content-type'] = contentType\n    }\n    return headers\n}\n\nfunction errorToResponse(e: unknown): Response {\n    const { body, statusCode: status } = e as { body?: unknown; statusCode?: number }\n    if (typeof body === 'string') {\n        return {\n            headers: {\n                'content-type': 'text/plain',\n            },\n            status: status ?? 500,\n            body,\n        }\n    } else if (typeof body === 'object') {\n        return {\n            headers: {\n                'content-type': 'application/json',\n            },\n            status: status ?? 500,\n            body: JSON.stringify(body),\n        }\n    } else {\n        return {\n            headers: {},\n            status: status ?? 500,\n        }\n    }\n}\n\nexport function clientFromHeaders(\n    headers: { readonly [key: string]: string } | undefined,\n): ClientInfo {\n    if (!headers) {\n        return {}\n    }\n    return {\n        operationId: headers['x-request-id'] ?? headers['request-id'],\n        clientId:\n            headers['x-client-id'] ??\n            headers['x-installation-id'] ??\n            headers['client-id'] ??\n            headers['installation-id'],\n        clientIp: headers['x-forwarded-for'],\n        userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],\n    }\n}\n"]}
1
+ import { parse } from 'node:url';
2
+ import { measure } from '../context.js';
3
+ export async function executeRequest(log, context, handler, options, success) {
4
+ const isShallow = context.env.SHALLOW_KEY && options.headers?.['x-shallow'] === context.env.SHALLOW_KEY;
5
+ const includeBodyInLogs = !handler.config?.excludeBodyFromLogs;
6
+ const logRequest = includeBodyInLogs
7
+ ? { method: handler.method, ...options }
8
+ : withoutRequestBody({ method: handler.method, ...options });
9
+ log = log.enrichReserved({ meta: context.meta, request: logRequest });
10
+ if (isShallow) {
11
+ log.trace('Shallow request');
12
+ return {
13
+ headers: {},
14
+ status: 204,
15
+ };
16
+ }
17
+ log.trace('Request BEGIN');
18
+ try {
19
+ let parsedUrl;
20
+ let pathSteps;
21
+ const req = {
22
+ rawUrl: options.uri,
23
+ get url() {
24
+ return (parsedUrl ??= {
25
+ ...parse(this.rawUrl, true),
26
+ pathStepAt: (index) => {
27
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
28
+ const steps = (pathSteps ??= parsedUrl.pathname?.split('/') ?? []);
29
+ const step = steps[index + 1];
30
+ if (!step) {
31
+ throw new RangeError(`Path does not have a step at index ${index}.`);
32
+ }
33
+ return step;
34
+ },
35
+ });
36
+ },
37
+ body: requestBody(options),
38
+ headers: options.headers ?? {},
39
+ };
40
+ const result = await measure(log, 'execution', () => handler.entry({ ...context, log }, req));
41
+ const response = resultToResponse(result, includeBodyInLogs);
42
+ if (context.signal.aborted) {
43
+ response.headers = {
44
+ 'x-timeout': '1',
45
+ ...response.headers,
46
+ };
47
+ }
48
+ log = log.enrichReserved({
49
+ response: {
50
+ status: response.status,
51
+ headers: response.headers,
52
+ body: response.logBody,
53
+ },
54
+ });
55
+ if (response.status < 300) {
56
+ log.debug('Request END');
57
+ await success();
58
+ }
59
+ else {
60
+ log.warn('Request END');
61
+ }
62
+ return response;
63
+ }
64
+ catch (e) {
65
+ try {
66
+ const response = errorToResponse(e);
67
+ log = log.enrichReserved({ response });
68
+ log.error('Request END', e);
69
+ return response;
70
+ }
71
+ catch (convertError) {
72
+ log.error('Could not convert exception to error response.', convertError);
73
+ return {
74
+ headers: {},
75
+ status: 500,
76
+ };
77
+ }
78
+ }
79
+ }
80
+ function resultToResponse(result, withLogBody) {
81
+ if (!result) {
82
+ return {
83
+ headers: {},
84
+ status: 204,
85
+ };
86
+ }
87
+ else if (typeof result === 'string') {
88
+ const logBody = withLogBody ? result : undefined;
89
+ return {
90
+ headers: {
91
+ 'content-type': 'text/plain',
92
+ },
93
+ status: 200,
94
+ body: result,
95
+ logBody,
96
+ };
97
+ }
98
+ else {
99
+ if (result.body === undefined) {
100
+ return {
101
+ headers: result.headers ?? {},
102
+ status: result.status ?? 200,
103
+ };
104
+ }
105
+ else if (typeof result.body === 'string') {
106
+ const logBody = withLogBody ? result.body : undefined;
107
+ return {
108
+ headers: withContentType(result.headers, 'text/plain'),
109
+ status: result.status ?? 200,
110
+ body: result.body,
111
+ logBody,
112
+ };
113
+ }
114
+ else if (Buffer.isBuffer(result.body)) {
115
+ const logBody = withLogBody ? result.body.toString('base64') : undefined;
116
+ return {
117
+ headers: withContentType(result.headers, 'application/octet-stream'),
118
+ status: result.status ?? 200,
119
+ body: result.body,
120
+ logBody,
121
+ };
122
+ }
123
+ else {
124
+ const logBody = withLogBody ? result.body : undefined;
125
+ return {
126
+ headers: withContentType(result.headers, 'application/json'),
127
+ status: result.status ?? 200,
128
+ body: JSON.stringify(result.body),
129
+ logBody,
130
+ };
131
+ }
132
+ }
133
+ }
134
+ function withoutRequestBody(options) {
135
+ if (hasJsonBody(options)) {
136
+ const { json, ...bodyless } = options;
137
+ return bodyless;
138
+ }
139
+ if (hasStringBody(options)) {
140
+ const { body, ...bodyless } = options;
141
+ return bodyless;
142
+ }
143
+ return options;
144
+ }
145
+ function requestBody(options) {
146
+ if (hasJsonBody(options)) {
147
+ return options.json;
148
+ }
149
+ if (hasStringBody(options)) {
150
+ return options.body;
151
+ }
152
+ return undefined;
153
+ }
154
+ function hasJsonBody(options) {
155
+ return options.json !== undefined;
156
+ }
157
+ function hasStringBody(options) {
158
+ return options.body !== undefined;
159
+ }
160
+ function withContentType(headers, contentType) {
161
+ if (!headers) {
162
+ return {
163
+ 'content-type': contentType,
164
+ };
165
+ }
166
+ if (!headers['content-type']) {
167
+ headers['content-type'] = contentType;
168
+ }
169
+ return headers;
170
+ }
171
+ function errorToResponse(e) {
172
+ const { body, statusCode: status } = e;
173
+ if (typeof body === 'string') {
174
+ return {
175
+ headers: {
176
+ 'content-type': 'text/plain',
177
+ },
178
+ status: status ?? 500,
179
+ body,
180
+ };
181
+ }
182
+ else if (typeof body === 'object') {
183
+ return {
184
+ headers: {
185
+ 'content-type': 'application/json',
186
+ },
187
+ status: status ?? 500,
188
+ body: JSON.stringify(body),
189
+ };
190
+ }
191
+ else {
192
+ return {
193
+ headers: {},
194
+ status: status ?? 500,
195
+ };
196
+ }
197
+ }
198
+ export function clientFromHeaders(headers) {
199
+ if (!headers) {
200
+ return {};
201
+ }
202
+ return {
203
+ operationId: headers['x-request-id'] ?? headers['request-id'],
204
+ clientId: headers['x-client-id'] ??
205
+ headers['x-installation-id'] ??
206
+ headers['client-id'] ??
207
+ headers['installation-id'],
208
+ clientIp: headers['x-forwarded-for'],
209
+ userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],
210
+ };
211
+ }
212
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http.js","sourceRoot":"","sources":["http.ts"],"names":[],"mappings":"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","sourcesContent":["import { parse, UrlWithParsedQuery } from 'node:url'\nimport { Context, measure } from '../context.js'\nimport type { Json, ResponseHeaders, Result } from '../http.js'\nimport { ClientInfo, RootLogger } from './context.js'\nimport type { HttpHandler } from './registry.js'\n\nexport type Response = {\n    headers: { readonly [key: string]: string }\n    status: number\n    body?: string | Buffer\n}\n\ntype RequestOptions = BodylessRequestOptions | StringRequestOptions | JsonRequestOptions\n\ntype BodylessRequestOptions = {\n    uri: string\n    headers?: { readonly [key: string]: string }\n}\n\ntype StringRequestOptions = BodylessRequestOptions & {\n    body: string\n}\n\ntype JsonRequestOptions = BodylessRequestOptions & {\n    json: Json\n}\n\nexport async function executeRequest(\n    log: RootLogger,\n    context: Omit<Context, 'log'>,\n    handler: HttpHandler,\n    options: RequestOptions,\n    success: () => Promise<unknown>,\n): Promise<Response> {\n    const isShallow =\n        context.env.SHALLOW_KEY && options.headers?.['x-shallow'] === context.env.SHALLOW_KEY\n    const includeBodyInLogs = !handler.config?.excludeBodyFromLogs\n    const logRequest = includeBodyInLogs\n        ? { method: handler.method, ...options }\n        : withoutRequestBody({ method: handler.method, ...options })\n    log = log.enrichReserved({ meta: context.meta, request: logRequest })\n    if (isShallow) {\n        log.trace('Shallow request')\n        return {\n            headers: {},\n            status: 204,\n        }\n    }\n    log.trace('Request BEGIN')\n    try {\n        let parsedUrl: (UrlWithParsedQuery & { pathStepAt: (index: number) => string }) | undefined\n        let pathSteps: string[] | undefined\n        const req = {\n            rawUrl: options.uri,\n            get url() {\n                return (parsedUrl ??= {\n                    ...parse(this.rawUrl, true),\n                    pathStepAt: (index: number) => {\n                        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n                        const steps = (pathSteps ??= parsedUrl!.pathname?.split('/') ?? [])\n                        const step = steps[index + 1]\n                        if (!step) {\n                            throw new RangeError(`Path does not have a step at index ${index}.`)\n                        }\n                        return step\n                    },\n                })\n            },\n            body: requestBody(options),\n            headers: options.headers ?? {},\n        }\n\n        const result = await measure(log, 'execution', () =>\n            handler.entry({ ...context, log }, req),\n        )\n\n        const response = resultToResponse(result, includeBodyInLogs)\n\n        if (context.signal.aborted) {\n            response.headers = {\n                'x-timeout': '1',\n                ...response.headers,\n            }\n        }\n\n        log = log.enrichReserved({\n            response: {\n                status: response.status,\n                headers: response.headers,\n                body: response.logBody,\n            },\n        })\n        if (response.status < 300) {\n            log.debug('Request END')\n            await success()\n        } else {\n            log.warn('Request END')\n        }\n        return response\n    } catch (e) {\n        try {\n            const response = errorToResponse(e)\n            log = log.enrichReserved({ response })\n            log.error('Request END', e)\n            return response\n        } catch (convertError) {\n            log.error('Could not convert exception to error response.', convertError)\n            return {\n                headers: {},\n                status: 500,\n            }\n        }\n    }\n}\n\nfunction resultToResponse(result: Result, withLogBody: boolean): Response & { logBody?: unknown } {\n    if (!result) {\n        return {\n            headers: {},\n            status: 204,\n        }\n    } else if (typeof result === 'string') {\n        const logBody = withLogBody ? result : undefined\n        return {\n            headers: {\n                'content-type': 'text/plain',\n            },\n            status: 200,\n            body: result,\n            logBody,\n        }\n    } else {\n        if (result.body === undefined) {\n            return {\n                headers: result.headers ?? {},\n                status: result.status ?? 200,\n            }\n        } else if (typeof result.body === 'string') {\n            const logBody = withLogBody ? result.body : undefined\n            return {\n                headers: withContentType(result.headers, 'text/plain'),\n                status: result.status ?? 200,\n                body: result.body,\n                logBody,\n            }\n        } else if (Buffer.isBuffer(result.body)) {\n            const logBody = withLogBody ? result.body.toString('base64') : undefined\n            return {\n                headers: withContentType(result.headers, 'application/octet-stream'),\n                status: result.status ?? 200,\n                body: result.body,\n                logBody,\n            }\n        } else {\n            const logBody = withLogBody ? result.body : undefined\n            return {\n                headers: withContentType(result.headers, 'application/json'),\n                status: result.status ?? 200,\n                body: JSON.stringify(result.body),\n                logBody,\n            }\n        }\n    }\n}\n\nfunction withoutRequestBody(options: RequestOptions & { method: string }) {\n    if (hasJsonBody(options)) {\n        const { json, ...bodyless } = options\n        return bodyless\n    }\n    if (hasStringBody(options)) {\n        const { body, ...bodyless } = options\n        return bodyless\n    }\n    return options\n}\n\nfunction requestBody(options: RequestOptions): Json | string | undefined {\n    if (hasJsonBody(options)) {\n        return options.json\n    }\n    if (hasStringBody(options)) {\n        return options.body\n    }\n    return undefined\n}\n\nfunction hasJsonBody(options: RequestOptions): options is JsonRequestOptions {\n    return (options as { json?: unknown }).json !== undefined\n}\n\nfunction hasStringBody(options: RequestOptions): options is StringRequestOptions {\n    return (options as { body?: unknown }).body !== undefined\n}\n\nfunction withContentType(headers: ResponseHeaders | undefined, contentType: string) {\n    if (!headers) {\n        return {\n            'content-type': contentType,\n        }\n    }\n    if (!headers['content-type']) {\n        headers['content-type'] = contentType\n    }\n    return headers\n}\n\nfunction errorToResponse(e: unknown): Response {\n    const { body, statusCode: status } = e as { body?: unknown; statusCode?: number }\n    if (typeof body === 'string') {\n        return {\n            headers: {\n                'content-type': 'text/plain',\n            },\n            status: status ?? 500,\n            body,\n        }\n    } else if (typeof body === 'object') {\n        return {\n            headers: {\n                'content-type': 'application/json',\n            },\n            status: status ?? 500,\n            body: JSON.stringify(body),\n        }\n    } else {\n        return {\n            headers: {},\n            status: status ?? 500,\n        }\n    }\n}\n\nexport function clientFromHeaders(\n    headers: { readonly [key: string]: string } | undefined,\n): ClientInfo {\n    if (!headers) {\n        return {}\n    }\n    return {\n        operationId: headers['x-request-id'] ?? headers['request-id'],\n        clientId:\n            headers['x-client-id'] ??\n            headers['x-installation-id'] ??\n            headers['client-id'] ??\n            headers['installation-id'],\n        clientIp: headers['x-forwarded-for'],\n        userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],\n    }\n}\n"]}
package/host/logging.d.ts CHANGED
@@ -1,4 +1,4 @@
1
- import { AbortSignal } from '../context.js';
2
- import { LogLevel, LogTransport, RootLogger } from './context.js';
3
- export declare function makeLogger(transport: LogTransport, minimumLogLevel: LogLevel | undefined, signal: AbortSignal): RootLogger;
4
- export declare function highPrecisionISODate(performanceNow: number): string;
1
+ import { AbortSignal } from '../context.js';
2
+ import { LogLevel, LogTransport, RootLogger } from './context.js';
3
+ export declare function makeLogger(transport: LogTransport, minimumLogLevel: LogLevel | undefined, signal: AbortSignal): RootLogger;
4
+ export declare function highPrecisionISODate(performanceNow: number): string;