@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/LICENSE +19 -0
- package/context.d.ts +93 -0
- package/context.js +70 -0
- package/host/context.d.ts +51 -0
- package/host/context.js +104 -0
- package/host/emitter.d.ts +13 -0
- package/host/emitter.js +81 -0
- package/host/http.d.ts +30 -0
- package/host/http.js +205 -0
- package/host/logging.d.ts +4 -0
- package/host/logging.js +205 -0
- package/host/reflect.d.ts +24 -0
- package/host/reflect.js +57 -0
- package/host/registry.d.ts +26 -0
- package/host/registry.js +61 -0
- package/http.d.ts +39 -0
- package/http.js +18 -0
- package/package.json +38 -0
package/LICENSE
ADDED
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
Copyright © 2022 The Riddance Authors (see git commits)
|
|
2
|
+
|
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
4
|
+
of this software and associated documentation files (the “Software”), to deal
|
|
5
|
+
in the Software without restriction, including without limitation the rights
|
|
6
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
7
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
8
|
+
furnished to do so, subject to the following conditions:
|
|
9
|
+
|
|
10
|
+
The above copyright notice and this permission notice shall be included in all
|
|
11
|
+
copies or substantial portions of the Software.
|
|
12
|
+
|
|
13
|
+
THE SOFTWARE IS PROVIDED “AS IS”, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
14
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
15
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
16
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
17
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
18
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
19
|
+
SOFTWARE.
|
package/context.d.ts
ADDED
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
export interface Environment {
|
|
2
|
+
readonly [key: string]: string;
|
|
3
|
+
}
|
|
4
|
+
export interface Logger {
|
|
5
|
+
enrich(fields: object): Logger;
|
|
6
|
+
trace(message: string, error?: unknown, fields?: object): void;
|
|
7
|
+
debug(message: string, error?: unknown, fields?: object): void;
|
|
8
|
+
info(message: string, error?: unknown, fields?: object): void;
|
|
9
|
+
warn(message: string, error?: unknown, fields?: object): void;
|
|
10
|
+
error(message: string, error?: unknown, fields?: object): void;
|
|
11
|
+
fatal(message: string, error?: unknown, fields?: object): void;
|
|
12
|
+
}
|
|
13
|
+
export interface AbortSignal {
|
|
14
|
+
aborted: boolean;
|
|
15
|
+
addEventListener: (type: 'abort', listener: (this: AbortSignal, event: unknown) => unknown, options?: {
|
|
16
|
+
capture?: boolean;
|
|
17
|
+
once?: boolean;
|
|
18
|
+
passive?: boolean;
|
|
19
|
+
}) => void;
|
|
20
|
+
removeEventListener: (type: 'abort', listener: (this: AbortSignal, event: unknown) => unknown, options?: {
|
|
21
|
+
capture?: boolean;
|
|
22
|
+
}) => void;
|
|
23
|
+
}
|
|
24
|
+
export declare type MutableJson = null | boolean | number | string | MutableJson[] | {
|
|
25
|
+
[key: string]: MutableJson;
|
|
26
|
+
};
|
|
27
|
+
export declare type Json = null | boolean | number | string | readonly Json[] | {
|
|
28
|
+
readonly [key: string]: Json;
|
|
29
|
+
};
|
|
30
|
+
export declare function objectSpreadable(json?: Json): {
|
|
31
|
+
readonly [key: string]: Json;
|
|
32
|
+
};
|
|
33
|
+
export declare function arraySpreadable(json?: Json): readonly Json[];
|
|
34
|
+
export interface HandlerConfiguration {
|
|
35
|
+
/**
|
|
36
|
+
* An indication of CPU usage of the handler.
|
|
37
|
+
* @default 'low'
|
|
38
|
+
*/
|
|
39
|
+
readonly compute?: 'high' | 'low';
|
|
40
|
+
/**
|
|
41
|
+
* An indication of memory usage of the handler.
|
|
42
|
+
* @default 'low'
|
|
43
|
+
*/
|
|
44
|
+
readonly memory?: 'high' | 'low';
|
|
45
|
+
/**
|
|
46
|
+
* A boolean indicating whether to enrich the log with the body of events, requests or responses. Set to false if the body is large or contain very sensitive data.
|
|
47
|
+
* @default false
|
|
48
|
+
*/
|
|
49
|
+
readonly excludeBodyFromLogs?: boolean;
|
|
50
|
+
/**
|
|
51
|
+
* The level below which log entries will be discarded.
|
|
52
|
+
* @default 'trace'
|
|
53
|
+
*/
|
|
54
|
+
readonly minimumLogLevel?: 'trace' | 'debug' | 'info' | 'warning' | 'error' | 'fatal';
|
|
55
|
+
/**
|
|
56
|
+
* The number of seconds the function is expected to finish executing in.
|
|
57
|
+
*/
|
|
58
|
+
readonly timeout?: number;
|
|
59
|
+
}
|
|
60
|
+
export interface Context {
|
|
61
|
+
readonly env: Environment;
|
|
62
|
+
readonly log: Logger;
|
|
63
|
+
readonly signal: AbortSignal;
|
|
64
|
+
now(): Date;
|
|
65
|
+
readonly operationId?: string;
|
|
66
|
+
readonly client?: {
|
|
67
|
+
readonly id?: string;
|
|
68
|
+
readonly ip?: string;
|
|
69
|
+
readonly port?: number;
|
|
70
|
+
readonly userAgent?: string;
|
|
71
|
+
};
|
|
72
|
+
readonly meta?: {
|
|
73
|
+
readonly packageName: string;
|
|
74
|
+
readonly fileName: string;
|
|
75
|
+
readonly revision?: string;
|
|
76
|
+
};
|
|
77
|
+
emit(topic: string, type: string, subject: string, data?: Json, messageId?: string): void;
|
|
78
|
+
eventBarrier(): Promise<void>;
|
|
79
|
+
onSuccess(fn: () => Promise<void> | void): void;
|
|
80
|
+
}
|
|
81
|
+
export declare function httpRequestHeaders(context: Context): {
|
|
82
|
+
[key: string]: string;
|
|
83
|
+
};
|
|
84
|
+
export declare function throwOnNotOK<T extends {
|
|
85
|
+
ok?: boolean;
|
|
86
|
+
status?: number;
|
|
87
|
+
text?: () => Promise<string>;
|
|
88
|
+
}>(response: T, message: string, data?: {
|
|
89
|
+
[key: string]: unknown;
|
|
90
|
+
}): Promise<T>;
|
|
91
|
+
export declare function measure<T>(logger: {
|
|
92
|
+
trace: (message: string, _: undefined, f: object) => void;
|
|
93
|
+
}, name: string, fn: () => Promise<T> | T, fields?: object): Promise<T>;
|
package/context.js
ADDED
|
@@ -0,0 +1,70 @@
|
|
|
1
|
+
import { performance } from 'perf_hooks';
|
|
2
|
+
import { highPrecisionISODate } from './host/logging.js';
|
|
3
|
+
/*@__INLINE__*/
|
|
4
|
+
export function objectSpreadable(json) {
|
|
5
|
+
if (!json) {
|
|
6
|
+
return {};
|
|
7
|
+
}
|
|
8
|
+
return json;
|
|
9
|
+
}
|
|
10
|
+
/*@__INLINE__*/
|
|
11
|
+
export function arraySpreadable(json) {
|
|
12
|
+
if (!Array.isArray(json)) {
|
|
13
|
+
return [];
|
|
14
|
+
}
|
|
15
|
+
return json;
|
|
16
|
+
}
|
|
17
|
+
export function httpRequestHeaders(context) {
|
|
18
|
+
const headers = {
|
|
19
|
+
'user-agent': `${context.meta?.packageName ?? '?'}/${context.meta?.revision ?? '?'}`,
|
|
20
|
+
};
|
|
21
|
+
if (context.operationId) {
|
|
22
|
+
headers['x-request-id'] = context.operationId;
|
|
23
|
+
}
|
|
24
|
+
if (context.client) {
|
|
25
|
+
if (context.client.id) {
|
|
26
|
+
headers['x-client-id'] = context.client.id;
|
|
27
|
+
}
|
|
28
|
+
if (context.client.ip || context.client.port) {
|
|
29
|
+
headers['x-forwarded-for'] = `${context.client.ip ?? ''}:${context.client.port ?? ''}`;
|
|
30
|
+
}
|
|
31
|
+
if (context.client.userAgent) {
|
|
32
|
+
headers['x-forwarded-for-user-agent'] = context.client.userAgent;
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
return headers;
|
|
36
|
+
}
|
|
37
|
+
export async function throwOnNotOK(response, message, data) {
|
|
38
|
+
if (response.ok === false) {
|
|
39
|
+
throw Object.assign(new Error(message), {
|
|
40
|
+
response: {
|
|
41
|
+
status: response.status,
|
|
42
|
+
body: limitSize(await response.text?.()),
|
|
43
|
+
},
|
|
44
|
+
...data,
|
|
45
|
+
});
|
|
46
|
+
}
|
|
47
|
+
return response;
|
|
48
|
+
}
|
|
49
|
+
function limitSize(text) {
|
|
50
|
+
if ((text?.length ?? 0) > 2048) {
|
|
51
|
+
return text?.substring(0, 2048);
|
|
52
|
+
}
|
|
53
|
+
return text;
|
|
54
|
+
}
|
|
55
|
+
export async function measure(logger, name, fn, fields) {
|
|
56
|
+
const start = performance.now();
|
|
57
|
+
try {
|
|
58
|
+
return await fn();
|
|
59
|
+
}
|
|
60
|
+
finally {
|
|
61
|
+
const end = performance.now();
|
|
62
|
+
logger.trace(`Measurement of ${name} time`, undefined, {
|
|
63
|
+
start: highPrecisionISODate(start),
|
|
64
|
+
end: highPrecisionISODate(end),
|
|
65
|
+
duration: (Math.round(end * 10000) - Math.round(start * 10000)) / 10000,
|
|
66
|
+
...fields,
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"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","sourcesContent":["import { performance } from 'perf_hooks'\r\nimport { highPrecisionISODate } from './host/logging.js'\r\n\r\nexport interface Environment {\r\n    readonly [key: string]: string\r\n}\r\n\r\nexport interface Logger {\r\n    enrich(fields: object): Logger\r\n    trace(message: string, error?: unknown, fields?: object): void\r\n    debug(message: string, error?: unknown, fields?: object): void\r\n    info(message: string, error?: unknown, fields?: object): void\r\n    warn(message: string, error?: unknown, fields?: object): void\r\n    error(message: string, error?: unknown, fields?: object): void\r\n    fatal(message: string, error?: unknown, fields?: object): void\r\n}\r\n\r\nexport interface AbortSignal {\r\n    aborted: boolean\r\n\r\n    addEventListener: (\r\n        type: 'abort',\r\n        listener: (this: AbortSignal, event: unknown) => unknown,\r\n        options?: { capture?: boolean; once?: boolean; passive?: boolean },\r\n    ) => void\r\n\r\n    removeEventListener: (\r\n        type: 'abort',\r\n        listener: (this: AbortSignal, event: unknown) => unknown,\r\n        options?: { capture?: boolean },\r\n    ) => void\r\n}\r\n\r\nexport type MutableJson =\r\n    | null\r\n    | boolean\r\n    | number\r\n    | string\r\n    | MutableJson[]\r\n    | { [key: string]: MutableJson }\r\nexport type Json =\r\n    | null\r\n    | boolean\r\n    | number\r\n    | string\r\n    | readonly Json[]\r\n    | { readonly [key: string]: Json }\r\n\r\n/*@__INLINE__*/\r\nexport function objectSpreadable(json?: Json): { readonly [key: string]: Json } {\r\n    if (!json) {\r\n        return {}\r\n    }\r\n    return json as unknown as { readonly [key: string]: Json }\r\n}\r\n\r\n/*@__INLINE__*/\r\nexport function arraySpreadable(json?: Json): readonly Json[] {\r\n    if (!Array.isArray(json)) {\r\n        return []\r\n    }\r\n    return json as readonly Json[]\r\n}\r\n\r\nexport interface HandlerConfiguration {\r\n    /**\r\n     * An indication of CPU usage of the handler.\r\n     * @default 'low'\r\n     */\r\n    readonly compute?: 'high' | 'low'\r\n    /**\r\n     * An indication of memory usage of the handler.\r\n     * @default 'low'\r\n     */\r\n    readonly memory?: 'high' | 'low'\r\n    /**\r\n     * A boolean indicating whether to enrich the log with the body of events, requests or responses. Set to false if the body is large or contain very sensitive data.\r\n     * @default false\r\n     */\r\n    readonly excludeBodyFromLogs?: boolean\r\n    /**\r\n     * The level below which log entries will be discarded.\r\n     * @default 'trace'\r\n     */\r\n    readonly minimumLogLevel?: 'trace' | 'debug' | 'info' | 'warning' | 'error' | 'fatal'\r\n    /**\r\n     * The number of seconds the function is expected to finish executing in.\r\n     */\r\n    readonly timeout?: number\r\n}\r\n\r\nexport interface Context {\r\n    readonly env: Environment\r\n    readonly log: Logger\r\n    readonly signal: AbortSignal\r\n    now(): Date\r\n\r\n    readonly operationId?: string\r\n    readonly client?: {\r\n        readonly id?: string\r\n        readonly ip?: string\r\n        readonly port?: number\r\n        readonly userAgent?: string\r\n    }\r\n    readonly meta?: {\r\n        readonly packageName: string\r\n        readonly fileName: string\r\n        readonly revision?: string\r\n    }\r\n\r\n    emit(topic: string, type: string, subject: string, data?: Json, messageId?: string): void\r\n    eventBarrier(): Promise<void>\r\n\r\n    onSuccess(fn: () => Promise<void> | void): void\r\n}\r\n\r\nexport function httpRequestHeaders(context: Context) {\r\n    const headers: { [key: string]: string } = {\r\n        'user-agent': `${context.meta?.packageName ?? '?'}/${context.meta?.revision ?? '?'}`,\r\n    }\r\n    if (context.operationId) {\r\n        headers['x-request-id'] = context.operationId\r\n    }\r\n    if (context.client) {\r\n        if (context.client.id) {\r\n            headers['x-client-id'] = context.client.id\r\n        }\r\n        if (context.client.ip || context.client.port) {\r\n            headers['x-forwarded-for'] = `${context.client.ip ?? ''}:${context.client.port ?? ''}`\r\n        }\r\n        if (context.client.userAgent) {\r\n            headers['x-forwarded-for-user-agent'] = context.client.userAgent\r\n        }\r\n    }\r\n    return headers\r\n}\r\n\r\nexport async function throwOnNotOK<\r\n    T extends { ok?: boolean; status?: number; text?: () => Promise<string> },\r\n>(response: T, message: string, data?: { [key: string]: unknown }) {\r\n    if (response.ok === false) {\r\n        throw Object.assign(new Error(message), {\r\n            response: {\r\n                status: response.status,\r\n                body: limitSize(await response.text?.()),\r\n            },\r\n            ...data,\r\n        })\r\n    }\r\n    return response\r\n}\r\n\r\nfunction limitSize(text: string | undefined) {\r\n    if ((text?.length ?? 0) > 2048) {\r\n        return text?.substring(0, 2048)\r\n    }\r\n    return text\r\n}\r\n\r\nexport async function measure<T>(\r\n    logger: { trace: (message: string, _: undefined, f: object) => void },\r\n    name: string,\r\n    fn: () => Promise<T> | T,\r\n    fields?: object,\r\n) {\r\n    const start = performance.now()\r\n    try {\r\n        return await fn()\r\n    } finally {\r\n        const end = performance.now()\r\n        logger.trace(`Measurement of ${name} time`, undefined, {\r\n            start: highPrecisionISODate(start),\r\n            end: highPrecisionISODate(end),\r\n            duration: (Math.round(end * 10000) - Math.round(start * 10000)) / 10000,\r\n            ...fields,\r\n        })\r\n    }\r\n}\r\n"]}
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
/// <reference types="node" resolution-mode="require"/>
|
|
2
|
+
import { AbortSignal, Context, Environment, Logger } from '../context.js';
|
|
3
|
+
import { Metadata } from './registry.js';
|
|
4
|
+
export interface ClientInfo {
|
|
5
|
+
readonly operationId?: string;
|
|
6
|
+
readonly clientId?: string;
|
|
7
|
+
readonly clientIp?: string;
|
|
8
|
+
readonly clientPort?: number;
|
|
9
|
+
readonly userAgent?: string;
|
|
10
|
+
}
|
|
11
|
+
export interface EventMetadata {
|
|
12
|
+
topic: string;
|
|
13
|
+
type: string;
|
|
14
|
+
subject: string;
|
|
15
|
+
id?: string;
|
|
16
|
+
}
|
|
17
|
+
export interface BufferedEvent {
|
|
18
|
+
eventTime: Date;
|
|
19
|
+
meta: Omit<EventMetadata, 'topic'>;
|
|
20
|
+
ids: ClientInfo;
|
|
21
|
+
json?: string;
|
|
22
|
+
}
|
|
23
|
+
export interface EventTransport {
|
|
24
|
+
readonly publishRate: number;
|
|
25
|
+
sendEvents(topic: string, events: BufferedEvent[], signal: AbortSignal): Promise<void>;
|
|
26
|
+
}
|
|
27
|
+
export declare type LogLevel = 'trace' | 'debug' | 'info' | 'warning' | 'error' | 'fatal';
|
|
28
|
+
export interface LogEntry {
|
|
29
|
+
readonly level: LogLevel;
|
|
30
|
+
readonly timestamp: number;
|
|
31
|
+
readonly message: string;
|
|
32
|
+
readonly error: unknown;
|
|
33
|
+
readonly json: string;
|
|
34
|
+
}
|
|
35
|
+
export interface LogTransport {
|
|
36
|
+
readonly publishRate?: number;
|
|
37
|
+
sendEntries(entries: LogEntry[], signal: AbortSignal): Promise<void> | undefined;
|
|
38
|
+
}
|
|
39
|
+
export interface RootLogger extends Logger {
|
|
40
|
+
enrichReserved(fields: object): RootLogger;
|
|
41
|
+
flush(): Promise<void>;
|
|
42
|
+
}
|
|
43
|
+
export declare function createContext(clientInfo: ClientInfo, loggers: LogTransport[], eventTransport: EventTransport, timeouts: {
|
|
44
|
+
default: number;
|
|
45
|
+
cap?: number;
|
|
46
|
+
}, outerController: AbortController, meta?: Metadata, environment?: Environment | undefined, now?: (() => Date) | undefined): {
|
|
47
|
+
log: RootLogger;
|
|
48
|
+
context: Context;
|
|
49
|
+
success: () => Promise<unknown>;
|
|
50
|
+
flush: () => Promise<void>;
|
|
51
|
+
};
|
package/host/context.js
ADDED
|
@@ -0,0 +1,104 @@
|
|
|
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 _LogMulticaster_transports;
|
|
13
|
+
import { EventEmitter } from './emitter.js';
|
|
14
|
+
import { makeLogger } from './logging.js';
|
|
15
|
+
class LogMulticaster {
|
|
16
|
+
constructor(transports) {
|
|
17
|
+
_LogMulticaster_transports.set(this, void 0);
|
|
18
|
+
__classPrivateFieldSet(this, _LogMulticaster_transports, transports, "f");
|
|
19
|
+
this.publishRate = transports.map(t => t.publishRate).sort()[0] ?? Number.MAX_SAFE_INTEGER;
|
|
20
|
+
}
|
|
21
|
+
sendEntries(entries, signal) {
|
|
22
|
+
const promises = __classPrivateFieldGet(this, _LogMulticaster_transports, "f").map(t => t.sendEntries(entries, signal)).filter(p => !!p);
|
|
23
|
+
if (promises.length === 0) {
|
|
24
|
+
return;
|
|
25
|
+
}
|
|
26
|
+
return Promise.all(promises);
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
_LogMulticaster_transports = new WeakMap();
|
|
30
|
+
export function createContext(clientInfo, loggers, eventTransport, timeouts, outerController, meta, environment, now) {
|
|
31
|
+
const timeout = (timeouts.cap
|
|
32
|
+
? Math.min(meta?.config?.timeout ?? timeouts.default, timeouts.cap)
|
|
33
|
+
: meta?.config?.timeout ?? timeouts.default) * 1000;
|
|
34
|
+
const innerController = new AbortController();
|
|
35
|
+
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
|
36
|
+
const logTransport = loggers.length === 1 ? loggers[0] : new LogMulticaster(loggers);
|
|
37
|
+
const logger = makeLogger(logTransport, meta?.config?.minimumLogLevel, outerController.signal);
|
|
38
|
+
logger.enrichReserved({
|
|
39
|
+
operationId: clientInfo.operationId,
|
|
40
|
+
client: {
|
|
41
|
+
id: clientInfo.clientId,
|
|
42
|
+
ip: clientInfo.clientIp,
|
|
43
|
+
port: clientInfo.clientPort,
|
|
44
|
+
userAgent: clientInfo.userAgent,
|
|
45
|
+
},
|
|
46
|
+
});
|
|
47
|
+
globalLogger = logger;
|
|
48
|
+
const emitter = new EventEmitter(eventTransport, logger, clientInfo, timeout, outerController.signal);
|
|
49
|
+
const successHandlers = [];
|
|
50
|
+
const ctx = {
|
|
51
|
+
env: environment ?? process.env,
|
|
52
|
+
log: logger,
|
|
53
|
+
signal: innerController.signal,
|
|
54
|
+
now: now ?? (() => new Date()),
|
|
55
|
+
operationId: clientInfo.operationId,
|
|
56
|
+
client: {
|
|
57
|
+
id: clientInfo.clientId,
|
|
58
|
+
ip: clientInfo.clientIp,
|
|
59
|
+
port: clientInfo.clientPort,
|
|
60
|
+
userAgent: clientInfo.userAgent,
|
|
61
|
+
},
|
|
62
|
+
meta: meta
|
|
63
|
+
? {
|
|
64
|
+
packageName: meta.packageName,
|
|
65
|
+
fileName: meta.fileName,
|
|
66
|
+
revision: meta.revision,
|
|
67
|
+
}
|
|
68
|
+
: undefined,
|
|
69
|
+
emit: (topic, type, subject, data, messageId) => emitter.emit({ topic, type, subject, id: messageId }, data),
|
|
70
|
+
eventBarrier: () => emitter.flush(),
|
|
71
|
+
onSuccess: (fn) => successHandlers.push(fn),
|
|
72
|
+
};
|
|
73
|
+
const timeoutHandle = setTimeout(() => {
|
|
74
|
+
ctx.log.error('Timeout.', undefined, undefined);
|
|
75
|
+
innerController.abort();
|
|
76
|
+
// eslint-disable-next-line no-void
|
|
77
|
+
void logger.flush();
|
|
78
|
+
// eslint-disable-next-line no-void
|
|
79
|
+
void emitter.flush();
|
|
80
|
+
}, timeout);
|
|
81
|
+
const flushHandle = setTimeout(() => {
|
|
82
|
+
ctx.log.error('Aborting flush.', undefined, undefined);
|
|
83
|
+
outerController.abort();
|
|
84
|
+
}, timeout + 15000);
|
|
85
|
+
return {
|
|
86
|
+
log: logger,
|
|
87
|
+
context: ctx,
|
|
88
|
+
success: () => Promise.all(successHandlers.map(fn => fn())),
|
|
89
|
+
flush: async () => {
|
|
90
|
+
clearTimeout(timeoutHandle);
|
|
91
|
+
await emitter.flush();
|
|
92
|
+
await logger.flush();
|
|
93
|
+
clearTimeout(flushHandle);
|
|
94
|
+
},
|
|
95
|
+
};
|
|
96
|
+
}
|
|
97
|
+
let globalLogger;
|
|
98
|
+
process.on('uncaughtException', err => {
|
|
99
|
+
globalLogger?.fatal('Uncaught exception.', err, undefined);
|
|
100
|
+
});
|
|
101
|
+
process.on('unhandledRejection', reason => {
|
|
102
|
+
globalLogger?.fatal('Unhandled rejection.', reason, undefined);
|
|
103
|
+
});
|
|
104
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbortSignal, Context, Environment, Json, Logger } from '../context.js'\r\nimport { EventEmitter } from './emitter.js'\r\nimport { makeLogger } from './logging.js'\r\nimport { Metadata } from './registry.js'\r\n\r\nexport interface ClientInfo {\r\n    readonly operationId?: string\r\n    readonly clientId?: string\r\n    readonly clientIp?: string\r\n    readonly clientPort?: number\r\n    readonly userAgent?: string\r\n}\r\n\r\nexport interface EventMetadata {\r\n    topic: string\r\n    type: string\r\n    subject: string\r\n    id?: string\r\n}\r\n\r\nexport interface BufferedEvent {\r\n    eventTime: Date\r\n    meta: Omit<EventMetadata, 'topic'>\r\n    ids: ClientInfo\r\n    json?: string\r\n}\r\n\r\nexport interface EventTransport {\r\n    readonly publishRate: number\r\n    sendEvents(topic: string, events: BufferedEvent[], signal: AbortSignal): Promise<void>\r\n}\r\n\r\nexport type LogLevel = 'trace' | 'debug' | 'info' | 'warning' | 'error' | 'fatal'\r\n\r\nexport interface LogEntry {\r\n    readonly level: LogLevel\r\n    readonly timestamp: number\r\n    readonly message: string\r\n    readonly error: unknown\r\n    readonly json: string\r\n}\r\n\r\nexport interface LogTransport {\r\n    readonly publishRate?: number\r\n    sendEntries(entries: LogEntry[], signal: AbortSignal): Promise<void> | undefined\r\n}\r\n\r\nclass LogMulticaster implements LogTransport {\r\n    #transports: LogTransport[]\r\n    readonly publishRate: number\r\n\r\n    constructor(transports: LogTransport[]) {\r\n        this.#transports = transports\r\n        this.publishRate = transports.map(t => t.publishRate).sort()[0] ?? Number.MAX_SAFE_INTEGER\r\n    }\r\n\r\n    sendEntries(entries: LogEntry[], signal: AbortSignal) {\r\n        const promises = this.#transports.map(t => t.sendEntries(entries, signal)).filter(p => !!p)\r\n        if (promises.length === 0) {\r\n            return\r\n        }\r\n        return Promise.all(promises) as unknown as Promise<void>\r\n    }\r\n}\r\n\r\nexport interface RootLogger extends Logger {\r\n    enrichReserved(fields: object): RootLogger\r\n    flush(): Promise<void>\r\n}\r\n\r\nexport function createContext(\r\n    clientInfo: ClientInfo,\r\n    loggers: LogTransport[],\r\n    eventTransport: EventTransport,\r\n    timeouts: { default: number; cap?: number },\r\n    outerController: AbortController,\r\n    meta?: Metadata,\r\n    environment?: Environment | undefined,\r\n    now?: (() => Date) | undefined,\r\n): {\r\n    log: RootLogger\r\n    context: Context\r\n    success: () => Promise<unknown>\r\n    flush: () => Promise<void>\r\n} {\r\n    const timeout =\r\n        (timeouts.cap\r\n            ? Math.min(meta?.config?.timeout ?? timeouts.default, timeouts.cap)\r\n            : meta?.config?.timeout ?? timeouts.default) * 1000\r\n    const innerController = new AbortController()\r\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\r\n    const logTransport = loggers.length === 1 ? loggers[0]! : new LogMulticaster(loggers)\r\n    const logger = makeLogger(\r\n        logTransport,\r\n        meta?.config?.minimumLogLevel,\r\n        outerController.signal as AbortSignal,\r\n    )\r\n    logger.enrichReserved({\r\n        operationId: clientInfo.operationId,\r\n        client: {\r\n            id: clientInfo.clientId,\r\n            ip: clientInfo.clientIp,\r\n            port: clientInfo.clientPort,\r\n            userAgent: clientInfo.userAgent,\r\n        },\r\n    })\r\n    globalLogger = logger\r\n    const emitter = new EventEmitter(\r\n        eventTransport,\r\n        logger,\r\n        clientInfo,\r\n        timeout,\r\n        outerController.signal as AbortSignal,\r\n    )\r\n    const successHandlers: (() => Promise<void> | void)[] = []\r\n    const ctx = {\r\n        env: environment ?? (process.env as Environment),\r\n        log: logger,\r\n        signal: innerController.signal as AbortSignal,\r\n        now: now ?? (() => new Date()),\r\n        operationId: clientInfo.operationId,\r\n        client: {\r\n            id: clientInfo.clientId,\r\n            ip: clientInfo.clientIp,\r\n            port: clientInfo.clientPort,\r\n            userAgent: clientInfo.userAgent,\r\n        },\r\n        meta: meta\r\n            ? {\r\n                  packageName: meta.packageName,\r\n                  fileName: meta.fileName,\r\n                  revision: meta.revision,\r\n              }\r\n            : undefined,\r\n        emit: (topic: string, type: string, subject: string, data?: Json, messageId?: string) =>\r\n            emitter.emit({ topic, type, subject, id: messageId }, data),\r\n        eventBarrier: () => emitter.flush(),\r\n        onSuccess: (fn: () => Promise<void> | void) => successHandlers.push(fn),\r\n    }\r\n    const timeoutHandle = setTimeout(() => {\r\n        ctx.log.error('Timeout.', undefined, undefined)\r\n        innerController.abort()\r\n        // eslint-disable-next-line no-void\r\n        void logger.flush()\r\n        // eslint-disable-next-line no-void\r\n        void emitter.flush()\r\n    }, timeout)\r\n    const flushHandle = setTimeout(() => {\r\n        ctx.log.error('Aborting flush.', undefined, undefined)\r\n        outerController.abort()\r\n    }, timeout + 15000)\r\n    return {\r\n        log: logger,\r\n        context: ctx,\r\n        success: () => Promise.all(successHandlers.map(fn => fn())),\r\n        flush: async () => {\r\n            clearTimeout(timeoutHandle)\r\n            await emitter.flush()\r\n            await logger.flush()\r\n            clearTimeout(flushHandle)\r\n        },\r\n    }\r\n}\r\n\r\nlet globalLogger: Logger | undefined\r\n\r\nprocess.on('uncaughtException', err => {\r\n    globalLogger?.fatal('Uncaught exception.', err, undefined)\r\n})\r\nprocess.on('unhandledRejection', reason => {\r\n    globalLogger?.fatal('Unhandled rejection.', reason, undefined)\r\n})\r\n"]}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import { AbortSignal, Json, Logger } from '../context.js';
|
|
2
|
+
import { EventMetadata, EventTransport } from './context.js';
|
|
3
|
+
export declare class EventEmitter {
|
|
4
|
+
#private;
|
|
5
|
+
constructor(transport: EventTransport, logger: Logger, ids: {
|
|
6
|
+
operationId?: string;
|
|
7
|
+
clientId?: string;
|
|
8
|
+
clientIp?: string;
|
|
9
|
+
userAgent?: string;
|
|
10
|
+
}, timeout: number, signal: AbortSignal);
|
|
11
|
+
emit(meta: EventMetadata, data?: Json): void;
|
|
12
|
+
flush(): Promise<void>;
|
|
13
|
+
}
|
package/host/emitter.js
ADDED
|
@@ -0,0 +1,81 @@
|
|
|
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 _EventEmitter_instances, _EventEmitter_transport, _EventEmitter_logger, _EventEmitter_ids, _EventEmitter_emitted, _EventEmitter_size, _EventEmitter_flusher, _EventEmitter_deadline, _EventEmitter_buffered, _EventEmitter_signal, _EventEmitter_startFlush, _EventEmitter_flushEvents;
|
|
13
|
+
export class EventEmitter {
|
|
14
|
+
constructor(transport, logger, ids, timeout, signal) {
|
|
15
|
+
_EventEmitter_instances.add(this);
|
|
16
|
+
_EventEmitter_transport.set(this, void 0);
|
|
17
|
+
_EventEmitter_logger.set(this, void 0);
|
|
18
|
+
_EventEmitter_ids.set(this, void 0);
|
|
19
|
+
_EventEmitter_emitted.set(this, {});
|
|
20
|
+
_EventEmitter_size.set(this, 0);
|
|
21
|
+
_EventEmitter_flusher.set(this, void 0);
|
|
22
|
+
_EventEmitter_deadline.set(this, void 0);
|
|
23
|
+
_EventEmitter_buffered.set(this, void 0);
|
|
24
|
+
_EventEmitter_signal.set(this, void 0);
|
|
25
|
+
__classPrivateFieldSet(this, _EventEmitter_transport, transport, "f");
|
|
26
|
+
__classPrivateFieldSet(this, _EventEmitter_logger, logger, "f");
|
|
27
|
+
__classPrivateFieldSet(this, _EventEmitter_ids, ids, "f");
|
|
28
|
+
__classPrivateFieldSet(this, _EventEmitter_deadline, new Date().getTime() + timeout, "f");
|
|
29
|
+
__classPrivateFieldSet(this, _EventEmitter_buffered, 0, "f");
|
|
30
|
+
__classPrivateFieldSet(this, _EventEmitter_signal, signal, "f");
|
|
31
|
+
}
|
|
32
|
+
emit(meta, data) {
|
|
33
|
+
var _a;
|
|
34
|
+
const eventTime = new Date();
|
|
35
|
+
const timeLeft = __classPrivateFieldGet(this, _EventEmitter_deadline, "f") - new Date().getTime();
|
|
36
|
+
if (__classPrivateFieldGet(this, _EventEmitter_buffered, "f") / __classPrivateFieldGet(this, _EventEmitter_transport, "f").publishRate > timeLeft) {
|
|
37
|
+
throw new Error('Event overflow.');
|
|
38
|
+
}
|
|
39
|
+
const event = data === undefined
|
|
40
|
+
? { meta, ids: __classPrivateFieldGet(this, _EventEmitter_ids, "f"), eventTime }
|
|
41
|
+
: { meta, ids: __classPrivateFieldGet(this, _EventEmitter_ids, "f"), eventTime, json: JSON.stringify(data) };
|
|
42
|
+
const events = __classPrivateFieldGet(this, _EventEmitter_emitted, "f")[meta.topic];
|
|
43
|
+
if (!events) {
|
|
44
|
+
__classPrivateFieldGet(this, _EventEmitter_emitted, "f")[meta.topic] = [event];
|
|
45
|
+
}
|
|
46
|
+
else {
|
|
47
|
+
events.push(event);
|
|
48
|
+
if (events.length > 64 || __classPrivateFieldGet(this, _EventEmitter_size, "f") > 64000) {
|
|
49
|
+
// eslint-disable-next-line no-void
|
|
50
|
+
void this.flush();
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
__classPrivateFieldSet(this, _EventEmitter_buffered, (_a = __classPrivateFieldGet(this, _EventEmitter_buffered, "f"), ++_a), "f");
|
|
54
|
+
__classPrivateFieldSet(this, _EventEmitter_size, __classPrivateFieldGet(this, _EventEmitter_size, "f") + (event.json?.length ?? 0), "f");
|
|
55
|
+
}
|
|
56
|
+
async flush() {
|
|
57
|
+
__classPrivateFieldGet(this, _EventEmitter_instances, "m", _EventEmitter_startFlush).call(this, __classPrivateFieldGet(this, _EventEmitter_emitted, "f"));
|
|
58
|
+
__classPrivateFieldSet(this, _EventEmitter_emitted, {}, "f");
|
|
59
|
+
__classPrivateFieldSet(this, _EventEmitter_size, 0, "f");
|
|
60
|
+
return await __classPrivateFieldGet(this, _EventEmitter_flusher, "f");
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
_EventEmitter_transport = new WeakMap(), _EventEmitter_logger = new WeakMap(), _EventEmitter_ids = new WeakMap(), _EventEmitter_emitted = new WeakMap(), _EventEmitter_size = new WeakMap(), _EventEmitter_flusher = new WeakMap(), _EventEmitter_deadline = new WeakMap(), _EventEmitter_buffered = new WeakMap(), _EventEmitter_signal = new WeakMap(), _EventEmitter_instances = new WeakSet(), _EventEmitter_startFlush = function _EventEmitter_startFlush(emitted) {
|
|
64
|
+
if (__classPrivateFieldGet(this, _EventEmitter_flusher, "f")) {
|
|
65
|
+
__classPrivateFieldSet(this, _EventEmitter_flusher, __classPrivateFieldGet(this, _EventEmitter_flusher, "f").then(() => __classPrivateFieldGet(this, _EventEmitter_instances, "m", _EventEmitter_flushEvents).call(this, emitted)), "f");
|
|
66
|
+
}
|
|
67
|
+
else {
|
|
68
|
+
__classPrivateFieldSet(this, _EventEmitter_flusher, __classPrivateFieldGet(this, _EventEmitter_instances, "m", _EventEmitter_flushEvents).call(this, emitted), "f");
|
|
69
|
+
}
|
|
70
|
+
}, _EventEmitter_flushEvents = async function _EventEmitter_flushEvents(emitted) {
|
|
71
|
+
await Promise.all(Object.entries(emitted).map(async ([topic, events]) => {
|
|
72
|
+
try {
|
|
73
|
+
await __classPrivateFieldGet(this, _EventEmitter_transport, "f").sendEvents(topic, events, __classPrivateFieldGet(this, _EventEmitter_signal, "f"));
|
|
74
|
+
}
|
|
75
|
+
catch (e) {
|
|
76
|
+
__classPrivateFieldGet(this, _EventEmitter_logger, "f").fatal('Error sending events.', e, { events });
|
|
77
|
+
}
|
|
78
|
+
__classPrivateFieldSet(this, _EventEmitter_buffered, __classPrivateFieldGet(this, _EventEmitter_buffered, "f") - events.length, "f");
|
|
79
|
+
}));
|
|
80
|
+
};
|
|
81
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW1pdHRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImVtaXR0ZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7O0FBT0EsTUFBTSxPQUFPLFlBQVk7SUFXckIsWUFDSSxTQUF5QixFQUN6QixNQUFjLEVBQ2QsR0FBdUYsRUFDdkYsT0FBZSxFQUNmLE1BQW1COztRQWZ2QiwwQ0FBbUM7UUFDbkMsdUNBQXdCO1FBQ3hCLG9DQUF5QjtRQUN6QixnQ0FBdUIsRUFBRSxFQUFBO1FBQ3pCLDZCQUFRLENBQUMsRUFBQTtRQUNULHdDQUF3QjtRQUN4Qix5Q0FBMEI7UUFDMUIseUNBQWlCO1FBQ2pCLHVDQUE2QjtRQVN6Qix1QkFBQSxJQUFJLDJCQUFjLFNBQVMsTUFBQSxDQUFBO1FBQzNCLHVCQUFBLElBQUksd0JBQVcsTUFBTSxNQUFBLENBQUE7UUFDckIsdUJBQUEsSUFBSSxxQkFBUSxHQUFHLE1BQUEsQ0FBQTtRQUNmLHVCQUFBLElBQUksMEJBQWEsSUFBSSxJQUFJLEVBQUUsQ0FBQyxPQUFPLEVBQUUsR0FBRyxPQUFPLE1BQUEsQ0FBQTtRQUMvQyx1QkFBQSxJQUFJLDBCQUFhLENBQUMsTUFBQSxDQUFBO1FBQ2xCLHVCQUFBLElBQUksd0JBQVcsTUFBTSxNQUFBLENBQUE7SUFDekIsQ0FBQztJQUVELElBQUksQ0FBQyxJQUFtQixFQUFFLElBQVc7O1FBQ2pDLE1BQU0sU0FBUyxHQUFHLElBQUksSUFBSSxFQUFFLENBQUE7UUFDNUIsTUFBTSxRQUFRLEdBQUcsdUJBQUEsSUFBSSw4QkFBVSxHQUFHLElBQUksSUFBSSxFQUFFLENBQUMsT0FBTyxFQUFFLENBQUE7UUFDdEQsSUFBSSx1QkFBQSxJQUFJLDhCQUFVLEdBQUcsdUJBQUEsSUFBSSwrQkFBVyxDQUFDLFdBQVcsR0FBRyxRQUFRLEVBQUU7WUFDekQsTUFBTSxJQUFJLEtBQUssQ0FBQyxpQkFBaUIsQ0FBQyxDQUFBO1NBQ3JDO1FBQ0QsTUFBTSxLQUFLLEdBQ1AsSUFBSSxLQUFLLFNBQVM7WUFDZCxDQUFDLENBQUMsRUFBRSxJQUFJLEVBQUUsR0FBRyxFQUFFLHVCQUFBLElBQUkseUJBQUssRUFBRSxTQUFTLEVBQUU7WUFDckMsQ0FBQyxDQUFDLEVBQUUsSUFBSSxFQUFFLEdBQUcsRUFBRSx1QkFBQSxJQUFJLHlCQUFLLEVBQUUsU0FBUyxFQUFFLElBQUksRUFBRSxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUE7UUFDekUsTUFBTSxNQUFNLEdBQUcsdUJBQUEsSUFBSSw2QkFBUyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQTtRQUN4QyxJQUFJLENBQUMsTUFBTSxFQUFFO1lBQ1QsdUJBQUEsSUFBSSw2QkFBUyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLEtBQUssQ0FBQyxDQUFBO1NBQ3RDO2FBQU07WUFDSCxNQUFNLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFBO1lBQ2xCLElBQUksTUFBTSxDQUFDLE1BQU0sR0FBRyxFQUFFLElBQUksdUJBQUEsSUFBSSwwQkFBTSxHQUFHLEtBQUssRUFBRTtnQkFDMUMsbUNBQW1DO2dCQUNuQyxLQUFLLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQTthQUNwQjtTQUNKO1FBQ0QscURBQUEsQ0FBRSw4REFBYyxFQUFoQixJQUFnQixDQUFBLE1BQUEsQ0FBQTtRQUNoQiwwR0FBYyxLQUFLLENBQUMsSUFBSSxFQUFFLE1BQU0sSUFBSSxDQUFDLE9BQUEsQ0FBQTtJQUN6QyxDQUFDO0lBRUQsS0FBSyxDQUFDLEtBQUs7UUFDUCx1QkFBQSxJQUFJLHlEQUFZLE1BQWhCLElBQUksRUFBYSx1QkFBQSxJQUFJLDZCQUFTLENBQUMsQ0FBQTtRQUMvQix1QkFBQSxJQUFJLHlCQUFZLEVBQUUsTUFBQSxDQUFBO1FBQ2xCLHVCQUFBLElBQUksc0JBQVMsQ0FBQyxNQUFBLENBQUE7UUFDZCxPQUFPLE1BQU0sdUJBQUEsSUFBSSw2QkFBUyxDQUFBO0lBQzlCLENBQUM7Q0FzQko7Z2NBcEJlLE9BQW1CO0lBQzNCLElBQUksdUJBQUEsSUFBSSw2QkFBUyxFQUFFO1FBQ2YsdUJBQUEsSUFBSSx5QkFBWSx1QkFBQSxJQUFJLDZCQUFTLENBQUMsSUFBSSxDQUFDLEdBQUcsRUFBRSxDQUFDLHVCQUFBLElBQUksMERBQWEsTUFBakIsSUFBSSxFQUFjLE9BQU8sQ0FBQyxDQUFDLE1BQUEsQ0FBQTtLQUN2RTtTQUFNO1FBQ0gsdUJBQUEsSUFBSSx5QkFBWSx1QkFBQSxJQUFJLDBEQUFhLE1BQWpCLElBQUksRUFBYyxPQUFPLENBQUMsTUFBQSxDQUFBO0tBQzdDO0FBQ0wsQ0FBQyw4QkFFRCxLQUFLLG9DQUFjLE9BQW1CO0lBQ2xDLE1BQU0sT0FBTyxDQUFDLEdBQUcsQ0FDYixNQUFNLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDLEdBQUcsQ0FBQyxLQUFLLEVBQUUsQ0FBQyxLQUFLLEVBQUUsTUFBTSxDQUFDLEVBQUUsRUFBRTtRQUNsRCxJQUFJO1lBQ0EsTUFBTSx1QkFBQSxJQUFJLCtCQUFXLENBQUMsVUFBVSxDQUFDLEtBQUssRUFBRSxNQUFNLEVBQUUsdUJBQUEsSUFBSSw0QkFBUSxDQUFDLENBQUE7U0FDaEU7UUFBQyxPQUFPLENBQUMsRUFBRTtZQUNSLHVCQUFBLElBQUksNEJBQVEsQ0FBQyxLQUFLLENBQUMsdUJBQXVCLEVBQUUsQ0FBQyxFQUFFLEVBQUUsTUFBTSxFQUFFLENBQUMsQ0FBQTtTQUM3RDtRQUNELGlIQUFrQixNQUFNLENBQUMsTUFBTSxNQUFBLENBQUE7SUFDbkMsQ0FBQyxDQUFDLENBQ0wsQ0FBQTtBQUNMLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBYm9ydFNpZ25hbCwgSnNvbiwgTG9nZ2VyIH0gZnJvbSAnLi4vY29udGV4dC5qcydcclxuaW1wb3J0IHsgQnVmZmVyZWRFdmVudCwgQ2xpZW50SW5mbywgRXZlbnRNZXRhZGF0YSwgRXZlbnRUcmFuc3BvcnQgfSBmcm9tICcuL2NvbnRleHQuanMnXHJcblxyXG5pbnRlcmZhY2UgRW1pdEJ1ZmZlciB7XHJcbiAgICBbdG9waWM6IHN0cmluZ106IEJ1ZmZlcmVkRXZlbnRbXVxyXG59XHJcblxyXG5leHBvcnQgY2xhc3MgRXZlbnRFbWl0dGVyIHtcclxuICAgIHJlYWRvbmx5ICN0cmFuc3BvcnQ6IEV2ZW50VHJhbnNwb3J0XHJcbiAgICByZWFkb25seSAjbG9nZ2VyOiBMb2dnZXJcclxuICAgIHJlYWRvbmx5ICNpZHM6IENsaWVudEluZm9cclxuICAgICNlbWl0dGVkOiBFbWl0QnVmZmVyID0ge31cclxuICAgICNzaXplID0gMFxyXG4gICAgI2ZsdXNoZXI/OiBQcm9taXNlPHZvaWQ+XHJcbiAgICByZWFkb25seSAjZGVhZGxpbmU6IG51bWJlclxyXG4gICAgI2J1ZmZlcmVkOiBudW1iZXJcclxuICAgIHJlYWRvbmx5ICNzaWduYWw6IEFib3J0U2lnbmFsXHJcblxyXG4gICAgY29uc3RydWN0b3IoXHJcbiAgICAgICAgdHJhbnNwb3J0OiBFdmVudFRyYW5zcG9ydCxcclxuICAgICAgICBsb2dnZXI6IExvZ2dlcixcclxuICAgICAgICBpZHM6IHsgb3BlcmF0aW9uSWQ/OiBzdHJpbmc7IGNsaWVudElkPzogc3RyaW5nOyBjbGllbnRJcD86IHN0cmluZzsgdXNlckFnZW50Pzogc3RyaW5nIH0sXHJcbiAgICAgICAgdGltZW91dDogbnVtYmVyLFxyXG4gICAgICAgIHNpZ25hbDogQWJvcnRTaWduYWwsXHJcbiAgICApIHtcclxuICAgICAgICB0aGlzLiN0cmFuc3BvcnQgPSB0cmFuc3BvcnRcclxuICAgICAgICB0aGlzLiNsb2dnZXIgPSBsb2dnZXJcclxuICAgICAgICB0aGlzLiNpZHMgPSBpZHNcclxuICAgICAgICB0aGlzLiNkZWFkbGluZSA9IG5ldyBEYXRlKCkuZ2V0VGltZSgpICsgdGltZW91dFxyXG4gICAgICAgIHRoaXMuI2J1ZmZlcmVkID0gMFxyXG4gICAgICAgIHRoaXMuI3NpZ25hbCA9IHNpZ25hbFxyXG4gICAgfVxyXG5cclxuICAgIGVtaXQobWV0YTogRXZlbnRNZXRhZGF0YSwgZGF0YT86IEpzb24pOiB2b2lkIHtcclxuICAgICAgICBjb25zdCBldmVudFRpbWUgPSBuZXcgRGF0ZSgpXHJcbiAgICAgICAgY29uc3QgdGltZUxlZnQgPSB0aGlzLiNkZWFkbGluZSAtIG5ldyBEYXRlKCkuZ2V0VGltZSgpXHJcbiAgICAgICAgaWYgKHRoaXMuI2J1ZmZlcmVkIC8gdGhpcy4jdHJhbnNwb3J0LnB1Ymxpc2hSYXRlID4gdGltZUxlZnQpIHtcclxuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdFdmVudCBvdmVyZmxvdy4nKVxyXG4gICAgICAgIH1cclxuICAgICAgICBjb25zdCBldmVudCA9XHJcbiAgICAgICAgICAgIGRhdGEgPT09IHVuZGVmaW5lZFxyXG4gICAgICAgICAgICAgICAgPyB7IG1ldGEsIGlkczogdGhpcy4jaWRzLCBldmVudFRpbWUgfVxyXG4gICAgICAgICAgICAgICAgOiB7IG1ldGEsIGlkczogdGhpcy4jaWRzLCBldmVudFRpbWUsIGpzb246IEpTT04uc3RyaW5naWZ5KGRhdGEpIH1cclxuICAgICAgICBjb25zdCBldmVudHMgPSB0aGlzLiNlbWl0dGVkW21ldGEudG9waWNdXHJcbiAgICAgICAgaWYgKCFldmVudHMpIHtcclxuICAgICAgICAgICAgdGhpcy4jZW1pdHRlZFttZXRhLnRvcGljXSA9IFtldmVudF1cclxuICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICBldmVudHMucHVzaChldmVudClcclxuICAgICAgICAgICAgaWYgKGV2ZW50cy5sZW5ndGggPiA2NCB8fCB0aGlzLiNzaXplID4gNjQwMDApIHtcclxuICAgICAgICAgICAgICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBuby12b2lkXHJcbiAgICAgICAgICAgICAgICB2b2lkIHRoaXMuZmx1c2goKVxyXG4gICAgICAgICAgICB9XHJcbiAgICAgICAgfVxyXG4gICAgICAgICsrdGhpcy4jYnVmZmVyZWRcclxuICAgICAgICB0aGlzLiNzaXplICs9IGV2ZW50Lmpzb24/Lmxlbmd0aCA/PyAwXHJcbiAgICB9XHJcblxyXG4gICAgYXN5bmMgZmx1c2goKTogUHJvbWlzZTx2b2lkPiB7XHJcbiAgICAgICAgdGhpcy4jc3RhcnRGbHVzaCh0aGlzLiNlbWl0dGVkKVxyXG4gICAgICAgIHRoaXMuI2VtaXR0ZWQgPSB7fVxyXG4gICAgICAgIHRoaXMuI3NpemUgPSAwXHJcbiAgICAgICAgcmV0dXJuIGF3YWl0IHRoaXMuI2ZsdXNoZXJcclxuICAgIH1cclxuXHJcbiAgICAjc3RhcnRGbHVzaChlbWl0dGVkOiBFbWl0QnVmZmVyKSB7XHJcbiAgICAgICAgaWYgKHRoaXMuI2ZsdXNoZXIpIHtcclxuICAgICAgICAgICAgdGhpcy4jZmx1c2hlciA9IHRoaXMuI2ZsdXNoZXIudGhlbigoKSA9PiB0aGlzLiNmbHVzaEV2ZW50cyhlbWl0dGVkKSlcclxuICAgICAgICB9IGVsc2Uge1xyXG4gICAgICAgICAgICB0aGlzLiNmbHVzaGVyID0gdGhpcy4jZmx1c2hFdmVudHMoZW1pdHRlZClcclxuICAgICAgICB9XHJcbiAgICB9XHJcblxyXG4gICAgYXN5bmMgI2ZsdXNoRXZlbnRzKGVtaXR0ZWQ6IEVtaXRCdWZmZXIpIHtcclxuICAgICAgICBhd2FpdCBQcm9taXNlLmFsbChcclxuICAgICAgICAgICAgT2JqZWN0LmVudHJpZXMoZW1pdHRlZCkubWFwKGFzeW5jIChbdG9waWMsIGV2ZW50c10pID0+IHtcclxuICAgICAgICAgICAgICAgIHRyeSB7XHJcbiAgICAgICAgICAgICAgICAgICAgYXdhaXQgdGhpcy4jdHJhbnNwb3J0LnNlbmRFdmVudHModG9waWMsIGV2ZW50cywgdGhpcy4jc2lnbmFsKVxyXG4gICAgICAgICAgICAgICAgfSBjYXRjaCAoZSkge1xyXG4gICAgICAgICAgICAgICAgICAgIHRoaXMuI2xvZ2dlci5mYXRhbCgnRXJyb3Igc2VuZGluZyBldmVudHMuJywgZSwgeyBldmVudHMgfSlcclxuICAgICAgICAgICAgICAgIH1cclxuICAgICAgICAgICAgICAgIHRoaXMuI2J1ZmZlcmVkIC09IGV2ZW50cy5sZW5ndGhcclxuICAgICAgICAgICAgfSksXHJcbiAgICAgICAgKVxyXG4gICAgfVxyXG59XHJcbiJdfQ==
|
package/host/http.d.ts
ADDED
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
/// <reference types="node" resolution-mode="require"/>
|
|
2
|
+
import { Context } from '../context.js';
|
|
3
|
+
import type { Json } from '../http.js';
|
|
4
|
+
import { ClientInfo, RootLogger } from './context.js';
|
|
5
|
+
import type { HttpHandler } from './registry.js';
|
|
6
|
+
export interface Response {
|
|
7
|
+
headers: {
|
|
8
|
+
readonly [key: string]: string;
|
|
9
|
+
};
|
|
10
|
+
status: number;
|
|
11
|
+
body?: string | Buffer;
|
|
12
|
+
}
|
|
13
|
+
declare type RequestOptions = BodylessRequestOptions | StringRequestOptions | JsonRequestOptions;
|
|
14
|
+
interface BodylessRequestOptions {
|
|
15
|
+
uri: string;
|
|
16
|
+
headers?: {
|
|
17
|
+
readonly [key: string]: string;
|
|
18
|
+
};
|
|
19
|
+
}
|
|
20
|
+
interface StringRequestOptions extends BodylessRequestOptions {
|
|
21
|
+
body: string;
|
|
22
|
+
}
|
|
23
|
+
interface JsonRequestOptions extends BodylessRequestOptions {
|
|
24
|
+
json: Json;
|
|
25
|
+
}
|
|
26
|
+
export declare function executeRequest(log: RootLogger, context: Context, handler: HttpHandler, options: RequestOptions, success: () => Promise<unknown>): Promise<Response>;
|
|
27
|
+
export declare function clientFromHeaders(headers: {
|
|
28
|
+
readonly [key: string]: string;
|
|
29
|
+
} | undefined): ClientInfo;
|
|
30
|
+
export {};
|