@riddance/host 0.0.1

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 ADDED
@@ -0,0 +1,205 @@
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
+ context.log.trace('Shallow request');
12
+ return {
13
+ headers: {},
14
+ status: 204,
15
+ };
16
+ }
17
+ context.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];
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(context.log, 'execution', () => handler.entry(context, 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({ request: logRequest });
48
+ if (response.status < 300) {
49
+ log.debug('Request END');
50
+ await success();
51
+ }
52
+ else {
53
+ log.warn('Request END');
54
+ }
55
+ return response;
56
+ }
57
+ catch (e) {
58
+ try {
59
+ const response = errorToResponse(e);
60
+ log = log.enrichReserved({ response });
61
+ log.error('Request END', e);
62
+ return response;
63
+ }
64
+ catch (convertError) {
65
+ log.error('Could not convert exception to error response.', convertError);
66
+ return {
67
+ headers: {},
68
+ status: 500,
69
+ };
70
+ }
71
+ }
72
+ }
73
+ function resultToResponse(result, withLogBody) {
74
+ if (!result) {
75
+ return {
76
+ headers: {},
77
+ status: 204,
78
+ };
79
+ }
80
+ else if (typeof result === 'string') {
81
+ const logBody = withLogBody ? result : undefined;
82
+ return {
83
+ headers: {
84
+ 'content-type': 'text/plain',
85
+ },
86
+ status: 200,
87
+ body: result,
88
+ logBody,
89
+ };
90
+ }
91
+ else {
92
+ if (result.body === undefined) {
93
+ return {
94
+ headers: result.headers ?? {},
95
+ status: result.status ?? 200,
96
+ };
97
+ }
98
+ else if (typeof result.body === 'string') {
99
+ const logBody = withLogBody ? result.body : undefined;
100
+ return {
101
+ headers: withContentType(result.headers, 'text/plain'),
102
+ status: result.status ?? 200,
103
+ body: result.body,
104
+ logBody,
105
+ };
106
+ }
107
+ else if (Buffer.isBuffer(result.body)) {
108
+ const logBody = withLogBody ? result.body.toString('base64') : undefined;
109
+ return {
110
+ headers: withContentType(result.headers, 'application/octet-stream'),
111
+ status: result.status ?? 200,
112
+ body: result.body,
113
+ logBody,
114
+ };
115
+ }
116
+ else {
117
+ const logBody = withLogBody ? result.body : undefined;
118
+ return {
119
+ headers: withContentType(result.headers, 'application/json'),
120
+ status: result.status ?? 200,
121
+ body: JSON.stringify(result.body),
122
+ logBody,
123
+ };
124
+ }
125
+ }
126
+ }
127
+ function withoutRequestBody(options) {
128
+ if (hasJsonBody(options)) {
129
+ const { json, ...bodyless } = options;
130
+ return bodyless;
131
+ }
132
+ if (hasStringBody(options)) {
133
+ const { body, ...bodyless } = options;
134
+ return bodyless;
135
+ }
136
+ return options;
137
+ }
138
+ function requestBody(options) {
139
+ if (hasJsonBody(options)) {
140
+ return options.json;
141
+ }
142
+ if (hasStringBody(options)) {
143
+ return options.body;
144
+ }
145
+ return undefined;
146
+ }
147
+ function hasJsonBody(options) {
148
+ return options.json !== undefined;
149
+ }
150
+ function hasStringBody(options) {
151
+ return options.body !== undefined;
152
+ }
153
+ function withContentType(headers, contentType) {
154
+ if (!headers) {
155
+ return {
156
+ 'content-type': contentType,
157
+ };
158
+ }
159
+ if (!headers['content-type']) {
160
+ headers['content-type'] = contentType;
161
+ }
162
+ return headers;
163
+ }
164
+ function errorToResponse(e) {
165
+ const { body, statusCode: status } = e;
166
+ if (typeof body === 'string') {
167
+ return {
168
+ headers: {
169
+ 'content-type': 'text/plain',
170
+ },
171
+ status: status ?? 500,
172
+ body,
173
+ };
174
+ }
175
+ else if (typeof body === 'object') {
176
+ return {
177
+ headers: {
178
+ 'content-type': 'application/json',
179
+ },
180
+ status: status ?? 500,
181
+ body: JSON.stringify(body),
182
+ };
183
+ }
184
+ else {
185
+ return {
186
+ headers: {},
187
+ status: status ?? 500,
188
+ };
189
+ }
190
+ }
191
+ export function clientFromHeaders(headers) {
192
+ if (!headers) {
193
+ return {};
194
+ }
195
+ return {
196
+ operationId: headers['x-request-id'] ?? headers['request-id'],
197
+ clientId: headers['x-client-id'] ??
198
+ headers['x-installation-id'] ??
199
+ headers['client-id'] ??
200
+ headers['installation-id'],
201
+ clientIp: headers['x-forwarded-for'],
202
+ userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],
203
+ };
204
+ }
205
+ //# 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'\r\nimport { Context, measure } from '../context.js'\r\nimport type { Json, ResponseHeaders, Result } from '../http.js'\r\nimport { ClientInfo, RootLogger } from './context.js'\r\nimport type { HttpHandler } from './registry.js'\r\n\r\nexport interface Response {\r\n    headers: { readonly [key: string]: string }\r\n    status: number\r\n    body?: string | Buffer\r\n}\r\n\r\ntype RequestOptions = BodylessRequestOptions | StringRequestOptions | JsonRequestOptions\r\n\r\ninterface BodylessRequestOptions {\r\n    uri: string\r\n    headers?: { readonly [key: string]: string }\r\n}\r\n\r\ninterface StringRequestOptions extends BodylessRequestOptions {\r\n    body: string\r\n}\r\n\r\ninterface JsonRequestOptions extends BodylessRequestOptions {\r\n    json: Json\r\n}\r\n\r\nexport async function executeRequest(\r\n    log: RootLogger,\r\n    context: Context,\r\n    handler: HttpHandler,\r\n    options: RequestOptions,\r\n    success: () => Promise<unknown>,\r\n): Promise<Response> {\r\n    const isShallow =\r\n        context.env.SHALLOW_KEY && options.headers?.['x-shallow'] === context.env.SHALLOW_KEY\r\n    const includeBodyInLogs = !handler.config?.excludeBodyFromLogs\r\n    const logRequest = includeBodyInLogs\r\n        ? { method: handler.method, ...options }\r\n        : withoutRequestBody({ method: handler.method, ...options })\r\n    log = log.enrichReserved({ meta: context.meta, request: logRequest })\r\n    if (isShallow) {\r\n        context.log.trace('Shallow request')\r\n        return {\r\n            headers: {},\r\n            status: 204,\r\n        }\r\n    }\r\n    context.log.trace('Request BEGIN')\r\n    try {\r\n        let parsedUrl: UrlWithParsedQuery & { pathStepAt: (index: number) => string }\r\n        let pathSteps: string[]\r\n        const req = {\r\n            rawUrl: options.uri,\r\n            get url() {\r\n                return (parsedUrl ??= {\r\n                    ...parse(this.rawUrl, true),\r\n                    pathStepAt: (index: number) => {\r\n                        const steps = (pathSteps ??= parsedUrl.pathname?.split('/') ?? [])\r\n                        const step = steps[index]\r\n                        if (!step) {\r\n                            throw new RangeError(`Path does not have a step at index ${index}.`)\r\n                        }\r\n                        return step\r\n                    },\r\n                })\r\n            },\r\n            body: requestBody(options),\r\n            headers: options.headers ?? {},\r\n        }\r\n\r\n        const result = await measure(context.log, 'execution', () => handler.entry(context, req))\r\n\r\n        const response = resultToResponse(result, includeBodyInLogs)\r\n\r\n        if (context.signal.aborted) {\r\n            response.headers = {\r\n                'x-timeout': '1',\r\n                ...response.headers,\r\n            }\r\n        }\r\n\r\n        log = log.enrichReserved({ request: logRequest })\r\n        if (response.status < 300) {\r\n            log.debug('Request END')\r\n            await success()\r\n        } else {\r\n            log.warn('Request END')\r\n        }\r\n        return response\r\n    } catch (e) {\r\n        try {\r\n            const response = errorToResponse(e)\r\n            log = log.enrichReserved({ response })\r\n            log.error('Request END', e)\r\n            return response\r\n        } catch (convertError) {\r\n            log.error('Could not convert exception to error response.', convertError)\r\n            return {\r\n                headers: {},\r\n                status: 500,\r\n            }\r\n        }\r\n    }\r\n}\r\n\r\nfunction resultToResponse(result: Result, withLogBody: boolean): Response & { logBody?: unknown } {\r\n    if (!result) {\r\n        return {\r\n            headers: {},\r\n            status: 204,\r\n        }\r\n    } else if (typeof result === 'string') {\r\n        const logBody = withLogBody ? result : undefined\r\n        return {\r\n            headers: {\r\n                'content-type': 'text/plain',\r\n            },\r\n            status: 200,\r\n            body: result,\r\n            logBody,\r\n        }\r\n    } else {\r\n        if (result.body === undefined) {\r\n            return {\r\n                headers: result.headers ?? {},\r\n                status: result.status ?? 200,\r\n            }\r\n        } else if (typeof result.body === 'string') {\r\n            const logBody = withLogBody ? result.body : undefined\r\n            return {\r\n                headers: withContentType(result.headers, 'text/plain'),\r\n                status: result.status ?? 200,\r\n                body: result.body,\r\n                logBody,\r\n            }\r\n        } else if (Buffer.isBuffer(result.body)) {\r\n            const logBody = withLogBody ? result.body.toString('base64') : undefined\r\n            return {\r\n                headers: withContentType(result.headers, 'application/octet-stream'),\r\n                status: result.status ?? 200,\r\n                body: result.body,\r\n                logBody,\r\n            }\r\n        } else {\r\n            const logBody = withLogBody ? result.body : undefined\r\n            return {\r\n                headers: withContentType(result.headers, 'application/json'),\r\n                status: result.status ?? 200,\r\n                body: JSON.stringify(result.body),\r\n                logBody,\r\n            }\r\n        }\r\n    }\r\n}\r\n\r\nfunction withoutRequestBody(options: RequestOptions & { method: string }) {\r\n    if (hasJsonBody(options)) {\r\n        const { json, ...bodyless } = options\r\n        return bodyless\r\n    }\r\n    if (hasStringBody(options)) {\r\n        const { body, ...bodyless } = options\r\n        return bodyless\r\n    }\r\n    return options\r\n}\r\n\r\nfunction requestBody(options: RequestOptions): Json | string | undefined {\r\n    if (hasJsonBody(options)) {\r\n        return options.json\r\n    }\r\n    if (hasStringBody(options)) {\r\n        return options.body\r\n    }\r\n    return undefined\r\n}\r\n\r\nfunction hasJsonBody(options: RequestOptions): options is JsonRequestOptions {\r\n    return (options as { json?: unknown }).json !== undefined\r\n}\r\n\r\nfunction hasStringBody(options: RequestOptions): options is StringRequestOptions {\r\n    return (options as { body?: unknown }).body !== undefined\r\n}\r\n\r\nfunction withContentType(headers: ResponseHeaders | undefined, contentType: string) {\r\n    if (!headers) {\r\n        return {\r\n            'content-type': contentType,\r\n        }\r\n    }\r\n    if (!headers['content-type']) {\r\n        headers['content-type'] = contentType\r\n    }\r\n    return headers\r\n}\r\n\r\nfunction errorToResponse(e: unknown): Response {\r\n    const { body, statusCode: status } = e as { body?: unknown; statusCode?: number }\r\n    if (typeof body === 'string') {\r\n        return {\r\n            headers: {\r\n                'content-type': 'text/plain',\r\n            },\r\n            status: status ?? 500,\r\n            body,\r\n        }\r\n    } else if (typeof body === 'object') {\r\n        return {\r\n            headers: {\r\n                'content-type': 'application/json',\r\n            },\r\n            status: status ?? 500,\r\n            body: JSON.stringify(body),\r\n        }\r\n    } else {\r\n        return {\r\n            headers: {},\r\n            status: status ?? 500,\r\n        }\r\n    }\r\n}\r\n\r\nexport function clientFromHeaders(\r\n    headers: { readonly [key: string]: string } | undefined,\r\n): ClientInfo {\r\n    if (!headers) {\r\n        return {}\r\n    }\r\n    return {\r\n        operationId: headers['x-request-id'] ?? headers['request-id'],\r\n        clientId:\r\n            headers['x-client-id'] ??\r\n            headers['x-installation-id'] ??\r\n            headers['client-id'] ??\r\n            headers['installation-id'],\r\n        clientIp: headers['x-forwarded-for'],\r\n        userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],\r\n    }\r\n}\r\n"]}
@@ -0,0 +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;
@@ -0,0 +1,205 @@
1
+ var __classPrivateFieldSet = (this && this.__classPrivateFieldSet) || function (receiver, state, value, kind, f) {
2
+ if (kind === "m") throw new TypeError("Private method is not writable");
3
+ if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a setter");
4
+ if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot write private member to an object whose class did not declare it");
5
+ return (kind === "a" ? f.call(receiver, value) : f ? f.value = value : state.set(receiver, value)), value;
6
+ };
7
+ var __classPrivateFieldGet = (this && this.__classPrivateFieldGet) || function (receiver, state, kind, f) {
8
+ if (kind === "a" && !f) throw new TypeError("Private accessor was defined without a getter");
9
+ if (typeof state === "function" ? receiver !== state || !f : !state.has(receiver)) throw new TypeError("Cannot read private member from an object whose class did not declare it");
10
+ return kind === "m" ? f : kind === "a" ? f.call(receiver) : f ? f.value : state.get(receiver);
11
+ };
12
+ var _LogBuffer_instances, _LogBuffer_transport, _LogBuffer_entries, _LogBuffer_size, _LogBuffer_flusher, _LogBuffer_signal, _LogBuffer_asyncTransport, _LogBuffer_timeout, _LogBuffer_startFlush, _EnrichingLogger_buffer, _EnrichingLogger_reservedEnrichment, _EnrichingLogger_customEnrichment, _EnrichingLogger_level;
13
+ import { performance } from 'perf_hooks';
14
+ export function makeLogger(transport, minimumLogLevel, signal) {
15
+ return new EnrichingLogger(new LogBuffer(transport, signal), minimumLogLevel
16
+ ? ['fatal', 'error', 'warning', 'info', 'debug', 'trace'].indexOf(minimumLogLevel)
17
+ : 5, signal);
18
+ }
19
+ const performanceTimeOrigin100ns = Math.round(performance.timeOrigin * 10000);
20
+ export function highPrecisionISODate(performanceNow) {
21
+ const now100ns = performanceTimeOrigin100ns + Math.round(performanceNow * 10000);
22
+ return (new Date(now100ns / 10000).toISOString().substring(0, 20) +
23
+ (now100ns % 10000000).toString().padStart(7, '0') +
24
+ 'Z');
25
+ }
26
+ class LogBuffer {
27
+ constructor(transport, signal) {
28
+ _LogBuffer_instances.add(this);
29
+ _LogBuffer_transport.set(this, void 0);
30
+ _LogBuffer_entries.set(this, []);
31
+ _LogBuffer_size.set(this, 0);
32
+ _LogBuffer_flusher.set(this, void 0);
33
+ _LogBuffer_signal.set(this, void 0);
34
+ _LogBuffer_asyncTransport.set(this, void 0);
35
+ _LogBuffer_timeout.set(this, void 0);
36
+ __classPrivateFieldSet(this, _LogBuffer_transport, transport, "f");
37
+ __classPrivateFieldSet(this, _LogBuffer_signal, signal, "f");
38
+ }
39
+ collect(level, numericLogLevel, message, error, fields, reservedEnrichment, customEnrichment) {
40
+ const offset = performance.now();
41
+ const json = JSON.stringify({
42
+ timestamp: highPrecisionISODate(offset),
43
+ level,
44
+ message,
45
+ error: errorAsJson(error),
46
+ ...reservedEnrichment,
47
+ ...((fields || customEnrichment) && {
48
+ fields: customEnrichment
49
+ ? fields
50
+ ? { ...customEnrichment, ...fields }
51
+ : customEnrichment
52
+ : fields,
53
+ }),
54
+ });
55
+ __classPrivateFieldGet(this, _LogBuffer_entries, "f").push({
56
+ timestamp: offset,
57
+ level,
58
+ message,
59
+ error,
60
+ json,
61
+ });
62
+ __classPrivateFieldSet(this, _LogBuffer_size, __classPrivateFieldGet(this, _LogBuffer_size, "f") + json.length, "f");
63
+ if (__classPrivateFieldGet(this, _LogBuffer_asyncTransport, "f") === false) {
64
+ // eslint-disable-next-line no-void
65
+ void __classPrivateFieldGet(this, _LogBuffer_transport, "f").sendEntries(__classPrivateFieldGet(this, _LogBuffer_entries, "f"), __classPrivateFieldGet(this, _LogBuffer_signal, "f"));
66
+ __classPrivateFieldSet(this, _LogBuffer_entries, [], "f");
67
+ __classPrivateFieldSet(this, _LogBuffer_size, 0, "f");
68
+ }
69
+ else if (__classPrivateFieldGet(this, _LogBuffer_asyncTransport, "f") === undefined) {
70
+ __classPrivateFieldSet(this, _LogBuffer_asyncTransport, true, "f");
71
+ setImmediate(() => {
72
+ if (__classPrivateFieldGet(this, _LogBuffer_flusher, "f")) {
73
+ return;
74
+ }
75
+ const sendResult = __classPrivateFieldGet(this, _LogBuffer_transport, "f").sendEntries(__classPrivateFieldGet(this, _LogBuffer_entries, "f"), __classPrivateFieldGet(this, _LogBuffer_signal, "f"));
76
+ __classPrivateFieldSet(this, _LogBuffer_entries, [], "f");
77
+ __classPrivateFieldSet(this, _LogBuffer_size, 0, "f");
78
+ if (sendResult) {
79
+ __classPrivateFieldSet(this, _LogBuffer_flusher, sendResult, "f");
80
+ }
81
+ else {
82
+ __classPrivateFieldSet(this, _LogBuffer_asyncTransport, false, "f");
83
+ }
84
+ });
85
+ }
86
+ else {
87
+ if (numericLogLevel < 2 || __classPrivateFieldGet(this, _LogBuffer_entries, "f").length > 8 || __classPrivateFieldGet(this, _LogBuffer_size, "f") > 64000) {
88
+ // eslint-disable-next-line no-void
89
+ void this.flush();
90
+ }
91
+ else {
92
+ __classPrivateFieldSet(this, _LogBuffer_timeout, setTimeout(() => {
93
+ // eslint-disable-next-line no-void
94
+ void this.flush();
95
+ __classPrivateFieldSet(this, _LogBuffer_timeout, undefined, "f");
96
+ }, 2000), "f");
97
+ }
98
+ }
99
+ }
100
+ async flush() {
101
+ if (__classPrivateFieldGet(this, _LogBuffer_entries, "f").length === 0) {
102
+ return;
103
+ }
104
+ __classPrivateFieldGet(this, _LogBuffer_instances, "m", _LogBuffer_startFlush).call(this, __classPrivateFieldGet(this, _LogBuffer_entries, "f"));
105
+ __classPrivateFieldSet(this, _LogBuffer_entries, [], "f");
106
+ __classPrivateFieldSet(this, _LogBuffer_size, 0, "f");
107
+ if (__classPrivateFieldGet(this, _LogBuffer_timeout, "f")) {
108
+ clearTimeout(__classPrivateFieldGet(this, _LogBuffer_timeout, "f"));
109
+ __classPrivateFieldSet(this, _LogBuffer_timeout, undefined, "f");
110
+ }
111
+ return await __classPrivateFieldGet(this, _LogBuffer_flusher, "f");
112
+ }
113
+ }
114
+ _LogBuffer_transport = new WeakMap(), _LogBuffer_entries = new WeakMap(), _LogBuffer_size = new WeakMap(), _LogBuffer_flusher = new WeakMap(), _LogBuffer_signal = new WeakMap(), _LogBuffer_asyncTransport = new WeakMap(), _LogBuffer_timeout = new WeakMap(), _LogBuffer_instances = new WeakSet(), _LogBuffer_startFlush = function _LogBuffer_startFlush(entries) {
115
+ if (__classPrivateFieldGet(this, _LogBuffer_flusher, "f")) {
116
+ __classPrivateFieldSet(this, _LogBuffer_flusher, __classPrivateFieldGet(this, _LogBuffer_flusher, "f").then(() => __classPrivateFieldGet(this, _LogBuffer_transport, "f").sendEntries(entries, __classPrivateFieldGet(this, _LogBuffer_signal, "f"))), "f");
117
+ }
118
+ else {
119
+ __classPrivateFieldSet(this, _LogBuffer_flusher, __classPrivateFieldGet(this, _LogBuffer_transport, "f").sendEntries(entries, __classPrivateFieldGet(this, _LogBuffer_signal, "f")), "f");
120
+ }
121
+ };
122
+ class EnrichingLogger {
123
+ constructor(buffer, level, reservedEnrichment, customEnrichment) {
124
+ _EnrichingLogger_buffer.set(this, void 0);
125
+ _EnrichingLogger_reservedEnrichment.set(this, void 0);
126
+ _EnrichingLogger_customEnrichment.set(this, void 0);
127
+ _EnrichingLogger_level.set(this, void 0);
128
+ __classPrivateFieldSet(this, _EnrichingLogger_buffer, buffer, "f");
129
+ __classPrivateFieldSet(this, _EnrichingLogger_level, level, "f");
130
+ __classPrivateFieldSet(this, _EnrichingLogger_reservedEnrichment, reservedEnrichment, "f");
131
+ __classPrivateFieldSet(this, _EnrichingLogger_customEnrichment, customEnrichment, "f");
132
+ }
133
+ enrich(fields) {
134
+ return new EnrichingLogger(__classPrivateFieldGet(this, _EnrichingLogger_buffer, "f"), __classPrivateFieldGet(this, _EnrichingLogger_level, "f"), __classPrivateFieldGet(this, _EnrichingLogger_reservedEnrichment, "f"), {
135
+ ...(__classPrivateFieldGet(this, _EnrichingLogger_customEnrichment, "f") ?? {}),
136
+ ...fields,
137
+ });
138
+ }
139
+ flush() {
140
+ return __classPrivateFieldGet(this, _EnrichingLogger_buffer, "f").flush();
141
+ }
142
+ enrichReserved(fields) {
143
+ return new EnrichingLogger(__classPrivateFieldGet(this, _EnrichingLogger_buffer, "f"), __classPrivateFieldGet(this, _EnrichingLogger_level, "f"), {
144
+ ...(__classPrivateFieldGet(this, _EnrichingLogger_reservedEnrichment, "f") ?? {}),
145
+ ...fields,
146
+ }, __classPrivateFieldGet(this, _EnrichingLogger_customEnrichment, "f"));
147
+ }
148
+ trace(message, error, fields) {
149
+ if (__classPrivateFieldGet(this, _EnrichingLogger_level, "f") < 5) {
150
+ return;
151
+ }
152
+ __classPrivateFieldGet(this, _EnrichingLogger_buffer, "f").collect('trace', 5, message, error, fields, __classPrivateFieldGet(this, _EnrichingLogger_reservedEnrichment, "f"), __classPrivateFieldGet(this, _EnrichingLogger_customEnrichment, "f"));
153
+ }
154
+ debug(message, error, fields) {
155
+ if (__classPrivateFieldGet(this, _EnrichingLogger_level, "f") < 4) {
156
+ return;
157
+ }
158
+ __classPrivateFieldGet(this, _EnrichingLogger_buffer, "f").collect('debug', 4, message, error, fields, __classPrivateFieldGet(this, _EnrichingLogger_reservedEnrichment, "f"), __classPrivateFieldGet(this, _EnrichingLogger_customEnrichment, "f"));
159
+ }
160
+ info(message, error, fields) {
161
+ if (__classPrivateFieldGet(this, _EnrichingLogger_level, "f") < 3) {
162
+ return;
163
+ }
164
+ __classPrivateFieldGet(this, _EnrichingLogger_buffer, "f").collect('debug', 3, message, error, fields, __classPrivateFieldGet(this, _EnrichingLogger_reservedEnrichment, "f"), __classPrivateFieldGet(this, _EnrichingLogger_customEnrichment, "f"));
165
+ }
166
+ warn(message, error, fields) {
167
+ if (__classPrivateFieldGet(this, _EnrichingLogger_level, "f") < 2) {
168
+ return;
169
+ }
170
+ __classPrivateFieldGet(this, _EnrichingLogger_buffer, "f").collect('debug', 2, message, error, fields, __classPrivateFieldGet(this, _EnrichingLogger_reservedEnrichment, "f"), __classPrivateFieldGet(this, _EnrichingLogger_customEnrichment, "f"));
171
+ }
172
+ error(message, error, fields) {
173
+ if (__classPrivateFieldGet(this, _EnrichingLogger_level, "f") < 1) {
174
+ return;
175
+ }
176
+ __classPrivateFieldGet(this, _EnrichingLogger_buffer, "f").collect('error', 1, message, error, fields, __classPrivateFieldGet(this, _EnrichingLogger_reservedEnrichment, "f"), __classPrivateFieldGet(this, _EnrichingLogger_customEnrichment, "f"));
177
+ }
178
+ fatal(message, error, fields) {
179
+ __classPrivateFieldGet(this, _EnrichingLogger_buffer, "f").collect('fatal', 0, message, error, fields, __classPrivateFieldGet(this, _EnrichingLogger_reservedEnrichment, "f"), __classPrivateFieldGet(this, _EnrichingLogger_customEnrichment, "f"));
180
+ }
181
+ }
182
+ _EnrichingLogger_buffer = new WeakMap(), _EnrichingLogger_reservedEnrichment = new WeakMap(), _EnrichingLogger_customEnrichment = new WeakMap(), _EnrichingLogger_level = new WeakMap();
183
+ function errorAsJson(error) {
184
+ if (error === undefined || error === null) {
185
+ return undefined;
186
+ }
187
+ if (error instanceof Error) {
188
+ return {
189
+ message: error.message,
190
+ name: error.name,
191
+ stack: error.stack,
192
+ ...error,
193
+ };
194
+ }
195
+ if (error instanceof Object) {
196
+ return {
197
+ ...error,
198
+ };
199
+ }
200
+ return {
201
+ message: error?.toString(),
202
+ name: typeof error,
203
+ };
204
+ }
205
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logging.js","sourceRoot":"","sources":["logging.ts"],"names":[],"mappings":"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","sourcesContent":["import { performance } from 'perf_hooks'\r\nimport { AbortSignal, Json, Logger } from '../context.js'\r\nimport { LogEntry, LogLevel, LogTransport, RootLogger } from './context.js'\r\n\r\nexport function makeLogger(\r\n    transport: LogTransport,\r\n    minimumLogLevel: LogLevel | undefined,\r\n    signal: AbortSignal,\r\n): RootLogger {\r\n    return new EnrichingLogger(\r\n        new LogBuffer(transport, signal),\r\n        minimumLogLevel\r\n            ? ['fatal', 'error', 'warning', 'info', 'debug', 'trace'].indexOf(minimumLogLevel)\r\n            : 5,\r\n        signal,\r\n    )\r\n}\r\n\r\nconst performanceTimeOrigin100ns = Math.round(performance.timeOrigin * 10000)\r\n\r\nexport function highPrecisionISODate(performanceNow: number) {\r\n    const now100ns = performanceTimeOrigin100ns + Math.round(performanceNow * 10000)\r\n    return (\r\n        new Date(now100ns / 10000).toISOString().substring(0, 20) +\r\n        (now100ns % 10000000).toString().padStart(7, '0') +\r\n        'Z'\r\n    )\r\n}\r\n\r\nclass LogBuffer {\r\n    readonly #transport: LogTransport\r\n    #entries: LogEntry[] = []\r\n    #size = 0\r\n    #flusher?: Promise<void> | undefined\r\n    readonly #signal: AbortSignal\r\n    #asyncTransport: boolean | undefined\r\n    #timeout: NodeJS.Timeout | undefined\r\n\r\n    constructor(transport: LogTransport, signal: AbortSignal) {\r\n        this.#transport = transport\r\n        this.#signal = signal\r\n    }\r\n\r\n    collect(\r\n        level: LogLevel,\r\n        numericLogLevel: number,\r\n        message: string,\r\n        error: unknown,\r\n        fields: object | undefined,\r\n        reservedEnrichment: object | undefined,\r\n        customEnrichment: object | undefined,\r\n    ) {\r\n        const offset = performance.now()\r\n        const json = JSON.stringify({\r\n            timestamp: highPrecisionISODate(offset),\r\n            level,\r\n            message,\r\n            error: errorAsJson(error),\r\n            ...reservedEnrichment,\r\n            ...((fields || customEnrichment) && {\r\n                fields: customEnrichment\r\n                    ? fields\r\n                        ? { ...customEnrichment, ...fields }\r\n                        : customEnrichment\r\n                    : fields,\r\n            }),\r\n        })\r\n        this.#entries.push({\r\n            timestamp: offset,\r\n            level,\r\n            message,\r\n            error,\r\n            json,\r\n        })\r\n        this.#size += json.length\r\n\r\n        if (this.#asyncTransport === false) {\r\n            // eslint-disable-next-line no-void\r\n            void this.#transport.sendEntries(this.#entries, this.#signal)\r\n            this.#entries = []\r\n            this.#size = 0\r\n        } else if (this.#asyncTransport === undefined) {\r\n            this.#asyncTransport = true\r\n            setImmediate(() => {\r\n                if (this.#flusher) {\r\n                    return\r\n                }\r\n                const sendResult = this.#transport.sendEntries(this.#entries, this.#signal)\r\n                this.#entries = []\r\n                this.#size = 0\r\n                if (sendResult) {\r\n                    this.#flusher = sendResult\r\n                } else {\r\n                    this.#asyncTransport = false\r\n                }\r\n            })\r\n        } else {\r\n            if (numericLogLevel < 2 || this.#entries.length > 8 || this.#size > 64000) {\r\n                // eslint-disable-next-line no-void\r\n                void this.flush()\r\n            } else {\r\n                this.#timeout = setTimeout(() => {\r\n                    // eslint-disable-next-line no-void\r\n                    void this.flush()\r\n                    this.#timeout = undefined\r\n                }, 2000)\r\n            }\r\n        }\r\n    }\r\n\r\n    async flush(): Promise<void> {\r\n        if (this.#entries.length === 0) {\r\n            return\r\n        }\r\n        this.#startFlush(this.#entries)\r\n        this.#entries = []\r\n        this.#size = 0\r\n        if (this.#timeout) {\r\n            clearTimeout(this.#timeout)\r\n            this.#timeout = undefined\r\n        }\r\n        return await this.#flusher\r\n    }\r\n\r\n    #startFlush(entries: LogEntry[]) {\r\n        if (this.#flusher) {\r\n            this.#flusher = this.#flusher.then(() =>\r\n                this.#transport.sendEntries(entries, this.#signal),\r\n            )\r\n        } else {\r\n            this.#flusher = this.#transport.sendEntries(entries, this.#signal)\r\n        }\r\n    }\r\n}\r\n\r\nclass EnrichingLogger implements Logger {\r\n    readonly #buffer: LogBuffer\r\n    readonly #reservedEnrichment?: object\r\n    readonly #customEnrichment?: object\r\n    readonly #level: number\r\n\r\n    constructor(\r\n        buffer: LogBuffer,\r\n        level: number,\r\n        reservedEnrichment?: object,\r\n        customEnrichment?: object,\r\n    ) {\r\n        this.#buffer = buffer\r\n        this.#level = level\r\n        this.#reservedEnrichment = reservedEnrichment\r\n        this.#customEnrichment = customEnrichment\r\n    }\r\n\r\n    enrich(fields: object): Logger {\r\n        return new EnrichingLogger(this.#buffer, this.#level, this.#reservedEnrichment, {\r\n            ...(this.#customEnrichment ?? {}),\r\n            ...fields,\r\n        })\r\n    }\r\n\r\n    flush() {\r\n        return this.#buffer.flush()\r\n    }\r\n\r\n    enrichReserved(fields: object): EnrichingLogger {\r\n        return new EnrichingLogger(\r\n            this.#buffer,\r\n            this.#level,\r\n            {\r\n                ...(this.#reservedEnrichment ?? {}),\r\n                ...fields,\r\n            },\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n\r\n    trace(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 5) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'trace',\r\n            5,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    debug(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 4) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'debug',\r\n            4,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    info(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 3) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'debug',\r\n            3,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    warn(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 2) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'debug',\r\n            2,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    error(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 1) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'error',\r\n            1,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    fatal(message: string, error: unknown, fields: object | undefined): void {\r\n        this.#buffer.collect(\r\n            'fatal',\r\n            0,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n}\r\n\r\nfunction errorAsJson(error: unknown): Json | undefined {\r\n    if (error === undefined || error === null) {\r\n        return undefined\r\n    }\r\n    if (error instanceof Error) {\r\n        return {\r\n            message: error.message,\r\n            name: error.name,\r\n            stack: error.stack,\r\n            ...(error as unknown as { [key: string]: unknown }),\r\n        } as Json\r\n    }\r\n    if (error instanceof Object) {\r\n        return {\r\n            ...error,\r\n        } as Json\r\n    }\r\n    return {\r\n        message: (error as object | undefined)?.toString(),\r\n        name: typeof error,\r\n    } as Json\r\n}\r\n"]}
@@ -0,0 +1,24 @@
1
+ /// <reference types="node" resolution-mode="require"/>
2
+ import { HttpHandlerConfiguration } from '../http.js';
3
+ declare type CPU = 'arm' | 'arm64' | 'ia32' | 'mips' | 'mipsel' | 'ppc' | 'ppc64' | 's390' | 's390x' | 'x32' | 'x64';
4
+ declare type CpuConfig = CPU | `!${CPU}`;
5
+ declare type OSConfig = NodeJS.Platform | `!${NodeJS.Platform}`;
6
+ export interface PackageJsonConfiguration {
7
+ nodeVersion?: string;
8
+ cpus?: CpuConfig[];
9
+ os?: OSConfig[];
10
+ }
11
+ export interface Reflection {
12
+ name: string;
13
+ http: {
14
+ name: string;
15
+ method: string;
16
+ pathPattern: string;
17
+ pathRegExp: RegExp;
18
+ config: HttpHandlerConfiguration & PackageJsonConfiguration;
19
+ }[];
20
+ }
21
+ export declare function resolveCpu(config: PackageJsonConfiguration, supported: CPU[]): CPU;
22
+ export declare function resolveOS(config: PackageJsonConfiguration, supported: NodeJS.Platform[]): NodeJS.Platform;
23
+ export declare function reflect(path: string): Promise<Reflection>;
24
+ export {};
@@ -0,0 +1,57 @@
1
+ import { readdir, readFile } from 'node:fs/promises';
2
+ import { basename, extname, join } from 'node:path';
3
+ import { pathToFileURL } from 'node:url';
4
+ export function resolveCpu(config, supported) {
5
+ const resolved = resolve(config.cpus, supported);
6
+ if (!resolved) {
7
+ // resolve<T>(config, supported) actually asserts config is (T | `!${T}`)[], but that's not supported yet.
8
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
9
+ throw new Error('Unsupported CPUs: ' + config.cpus.join(', '));
10
+ }
11
+ return resolved;
12
+ }
13
+ export function resolveOS(config, supported) {
14
+ const resolved = resolve(config.os, supported);
15
+ if (!resolved) {
16
+ // resolve<T>(config, supported) actually asserts config is (T | `!${T}`)[], but that's not supported yet.
17
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
18
+ throw new Error('Unsupported operating systems: ' + config.os.join(', '));
19
+ }
20
+ return resolved;
21
+ }
22
+ function resolve(config, supported) {
23
+ if (!config) {
24
+ return supported[0];
25
+ }
26
+ return supported.find(s => config.includes(s) && !config.includes(`!${s}`));
27
+ }
28
+ export async function reflect(path) {
29
+ const packageJson = await readConfig();
30
+ const files = (await readdir(path)).filter(file => extname(file) === '.ts' && !file.endsWith('.d.ts'));
31
+ const { getHandlers, setMeta } = (await import(pathToFileURL(join(process.cwd(), 'node_modules/@riddance/host/host/registry.js')).toString()));
32
+ for (const file of files) {
33
+ const base = basename(file, '.ts');
34
+ setMeta(packageJson.name, base, undefined, packageJson.config);
35
+ await import(pathToFileURL(join(process.cwd(), path, base + '.js')).toString());
36
+ }
37
+ return {
38
+ name: packageJson.name,
39
+ http: getHandlers('http').map(h => ({
40
+ config: {
41
+ ...h.config,
42
+ cpus: packageJson.cpu,
43
+ os: packageJson.os,
44
+ nodeVersion: packageJson.engines?.node,
45
+ },
46
+ name: h.meta?.fileName ?? '',
47
+ method: h.method,
48
+ pathPattern: h.pathPattern,
49
+ pathRegExp: h.pathRegExp,
50
+ })),
51
+ };
52
+ }
53
+ async function readConfig() {
54
+ const packageJson = JSON.parse(await readFile('package.json', 'utf-8'));
55
+ return packageJson;
56
+ }
57
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"reflect.js","sourceRoot":"","sources":["reflect.ts"],"names":[],"mappings":"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","sourcesContent":["import { readdir, readFile } from 'node:fs/promises'\r\nimport { basename, extname, join } from 'node:path'\r\nimport { pathToFileURL } from 'node:url'\r\nimport { HttpHandlerConfiguration } from '../http.js'\r\nimport { PackageConfiguration } from './registry.js'\r\n\r\ntype CPU =\r\n    | 'arm'\r\n    | 'arm64'\r\n    | 'ia32'\r\n    | 'mips'\r\n    | 'mipsel'\r\n    | 'ppc'\r\n    | 'ppc64'\r\n    | 's390'\r\n    | 's390x'\r\n    | 'x32'\r\n    | 'x64'\r\ntype CpuConfig = CPU | `!${CPU}`\r\ntype OSConfig = NodeJS.Platform | `!${NodeJS.Platform}`\r\n\r\nexport interface PackageJsonConfiguration {\r\n    nodeVersion?: string\r\n    cpus?: CpuConfig[]\r\n    os?: OSConfig[]\r\n}\r\n\r\nexport interface Reflection {\r\n    name: string\r\n    http: {\r\n        name: string\r\n        method: string\r\n        pathPattern: string\r\n        pathRegExp: RegExp\r\n        config: HttpHandlerConfiguration & PackageJsonConfiguration\r\n    }[]\r\n}\r\n\r\nexport function resolveCpu(config: PackageJsonConfiguration, supported: CPU[]): CPU {\r\n    const resolved = resolve(config.cpus, supported)\r\n    if (!resolved) {\r\n        // resolve<T>(config, supported) actually asserts config is (T | `!${T}`)[], but that's not supported yet.\r\n        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\r\n        throw new Error('Unsupported CPUs: ' + config.cpus!.join(', '))\r\n    }\r\n    return resolved\r\n}\r\n\r\nexport function resolveOS(\r\n    config: PackageJsonConfiguration,\r\n    supported: NodeJS.Platform[],\r\n): NodeJS.Platform {\r\n    const resolved = resolve(config.os, supported)\r\n    if (!resolved) {\r\n        // resolve<T>(config, supported) actually asserts config is (T | `!${T}`)[], but that's not supported yet.\r\n        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\r\n        throw new Error('Unsupported operating systems: ' + config.os!.join(', '))\r\n    }\r\n    return resolved\r\n}\r\n\r\nfunction resolve<T extends string>(\r\n    config: (T | `!${T}`)[] | undefined,\r\n    supported: T[],\r\n): T | undefined {\r\n    if (!config) {\r\n        return supported[0]\r\n    }\r\n    return supported.find(s => config.includes(s) && !config.includes(`!${s}`))\r\n}\r\n\r\nexport async function reflect(path: string): Promise<Reflection> {\r\n    const packageJson = await readConfig()\r\n    const files = (await readdir(path)).filter(\r\n        file => extname(file) === '.ts' && !file.endsWith('.d.ts'),\r\n    )\r\n    const { getHandlers, setMeta } = (await import(\r\n        pathToFileURL(\r\n            join(process.cwd(), 'node_modules/@riddance/host/host/registry.js'),\r\n        ).toString()\r\n    )) as {\r\n        getHandlers: (type: string) => {\r\n            name: string\r\n            meta?: { fileName: string }\r\n            config: HttpHandlerConfiguration\r\n            method: string\r\n            pathPattern: string\r\n            pathRegExp: RegExp\r\n        }[]\r\n        setMeta: (\r\n            packageName: string,\r\n            fileName: string,\r\n            rev: string | undefined,\r\n            cfg: PackageConfiguration | undefined,\r\n        ) => void\r\n    }\r\n\r\n    for (const file of files) {\r\n        const base = basename(file, '.ts')\r\n        setMeta(packageJson.name, base, undefined, packageJson.config)\r\n        await import(pathToFileURL(join(process.cwd(), path, base + '.js')).toString())\r\n    }\r\n\r\n    return {\r\n        name: packageJson.name,\r\n        http: getHandlers('http').map(h => ({\r\n            config: {\r\n                ...h.config,\r\n                cpus: packageJson.cpu,\r\n                os: packageJson.os,\r\n                nodeVersion: packageJson.engines?.node,\r\n            },\r\n            name: h.meta?.fileName ?? '',\r\n            method: h.method,\r\n            pathPattern: h.pathPattern,\r\n            pathRegExp: h.pathRegExp,\r\n        })),\r\n    }\r\n}\r\n\r\nasync function readConfig() {\r\n    const packageJson = JSON.parse(await readFile('package.json', 'utf-8')) as {\r\n        name: string\r\n        engines?: { [engine: string]: string }\r\n        cpu?: CpuConfig[]\r\n        os?: OSConfig[]\r\n        config?: object\r\n    }\r\n    return packageJson\r\n}\r\n"]}
@@ -0,0 +1,26 @@
1
+ import { HandlerConfiguration } from '../context.js';
2
+ import { Handler } from '../http.js';
3
+ export interface HttpHandler {
4
+ meta: Metadata | undefined;
5
+ config: (PackageConfiguration & HandlerConfiguration) | undefined;
6
+ method: Method;
7
+ pathPattern: string;
8
+ pathRegExp: RegExp;
9
+ entry: Handler;
10
+ }
11
+ interface HandlerTypes {
12
+ http: HttpHandler;
13
+ }
14
+ export declare function getHandlers(type: keyof HandlerTypes): HttpHandler[];
15
+ export declare function setMeta(packageName: string, fileName: string, revision: string | undefined, config: PackageConfiguration | undefined): void;
16
+ export interface PackageConfiguration extends HandlerConfiguration {
17
+ }
18
+ export interface Metadata {
19
+ packageName: string;
20
+ fileName: string;
21
+ revision: string | undefined;
22
+ config?: PackageConfiguration;
23
+ }
24
+ export declare type Method = 'GET' | 'POST' | 'PUT' | 'PATCH' | 'DELETE';
25
+ export declare function registerHttpHandler(method: Method, path: string, configOrHandler: HandlerConfiguration | Handler, fn?: Handler): void;
26
+ export {};