@riddance/host 0.1.2 → 0.2.0

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 CHANGED
@@ -1,4 +1,4 @@
1
- Copyright © 2022-2024 The Riddance Authors (see git commits)
1
+ Copyright © 2022-2025 The Riddance Authors (see git commits)
2
2
 
3
3
  Permission is hereby granted, free of charge, to any person obtaining a copy
4
4
  of this software and associated documentation files (the “Software”), to deal
package/context.d.ts CHANGED
@@ -63,8 +63,7 @@ export type Context = {
63
63
  readonly fileName: string;
64
64
  readonly revision?: string;
65
65
  };
66
- emit(topic: string, type: string, subject: string, data?: Json, messageId?: string): void;
67
- eventBarrier(): Promise<void>;
66
+ emit(topic: string, type: string, subject: string, data?: Json, messageId?: string): Promise<void>;
68
67
  onSuccess(fn: () => Promise<void> | void): void;
69
68
  };
70
69
  export declare function httpRequestHeaders(context: Context): {
package/context.js CHANGED
@@ -49,4 +49,4 @@ export async function measure(logger, name, fn, fields) {
49
49
  });
50
50
  }
51
51
  }
52
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAA;AAC7C,OAAO,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAA;AA+BxD,eAAe;AACf,MAAM,UAAU,gBAAgB,CAAC,IAAW;IACxC,IAAI,CAAC,IAAI,EAAE,CAAC;QACR,OAAO,EAAE,CAAA;IACb,CAAC;IACD,OAAO,IAAmD,CAAA;AAC9D,CAAC;AAED,eAAe;AACf,MAAM,UAAU,eAAe,CAAC,IAAW;IACvC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;QACvB,OAAO,EAAE,CAAA;IACb,CAAC;IACD,OAAO,IAAuB,CAAA;AAClC,CAAC;AAsDD,MAAM,UAAU,kBAAkB,CAAC,OAAgB;IAC/C,MAAM,OAAO,GAA8B;QACvC,YAAY,EAAE,GAAG,OAAO,CAAC,IAAI,EAAE,WAAW,IAAI,GAAG,IAAI,OAAO,CAAC,IAAI,EAAE,QAAQ,IAAI,GAAG,EAAE;KACvF,CAAA;IACD,IAAI,OAAO,CAAC,WAAW,EAAE,CAAC;QACtB,OAAO,CAAC,cAAc,CAAC,GAAG,OAAO,CAAC,WAAW,CAAA;IACjD,CAAC;IACD,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;QACjB,IAAI,OAAO,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC;YACpB,OAAO,CAAC,aAAa,CAAC,GAAG,OAAO,CAAC,MAAM,CAAC,EAAE,CAAA;QAC9C,CAAC;QACD,IAAI,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,IAAI,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;YAC/C,OAAO,CAAC,iBAAiB,CAAC,GAAG,GAAG,OAAO,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,IAAI,OAAO,CAAC,MAAM,CAAC,IAAI,IAAI,EAAE,EAAE,CAAA;QAC1F,CAAC;QACD,IAAI,OAAO,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC;YAC3B,OAAO,CAAC,4BAA4B,CAAC,GAAG,OAAO,CAAC,MAAM,CAAC,SAAS,CAAA;QACpE,CAAC;IACL,CAAC;IACD,OAAO,OAAO,CAAA;AAClB,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,OAAO,CACzB,MAAqE,EACrE,IAAY,EACZ,EAAwB,EACxB,MAAe;IAEf,MAAM,KAAK,GAAG,WAAW,CAAC,GAAG,EAAE,CAAA;IAC/B,IAAI,CAAC;QACD,OAAO,MAAM,EAAE,EAAE,CAAA;IACrB,CAAC;YAAS,CAAC;QACP,MAAM,GAAG,GAAG,WAAW,CAAC,GAAG,EAAE,CAAA;QAC7B,MAAM,CAAC,KAAK,CAAC,kBAAkB,IAAI,OAAO,EAAE,SAAS,EAAE;YACnD,KAAK,EAAE,oBAAoB,CAAC,KAAK,CAAC;YAClC,GAAG,EAAE,oBAAoB,CAAC,GAAG,CAAC;YAC9B,QAAQ,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,MAAM,CAAC,CAAC,GAAG,MAAM;YAC1E,GAAG,MAAM;SACZ,CAAC,CAAA;IACN,CAAC;AACL,CAAC","sourcesContent":["import { performance } from 'node:perf_hooks'\nimport { highPrecisionISODate } from './host/logging.js'\n\nexport type Environment = {\n    readonly [key: string]: string\n}\n\nexport type Logger = {\n    enrich(fields: object): Logger\n    trace(message: string, error?: unknown, fields?: object): void\n    debug(message: string, error?: unknown, fields?: object): void\n    info(message: string, error?: unknown, fields?: object): void\n    warn(message: string, error?: unknown, fields?: object): void\n    error(message: string, error?: unknown, fields?: object): void\n    fatal(message: string, error?: unknown, fields?: object): void\n}\n\nexport type MutableJson =\n    | null\n    | boolean\n    | number\n    | string\n    | MutableJson[]\n    | { [key: string]: MutableJson }\nexport type Json =\n    | null\n    | boolean\n    | number\n    | string\n    | readonly Json[]\n    | { readonly [key: string]: Json }\n\n/*@__INLINE__*/\nexport function objectSpreadable(json?: Json): { readonly [key: string]: Json } {\n    if (!json) {\n        return {}\n    }\n    return json as unknown as { readonly [key: string]: Json }\n}\n\n/*@__INLINE__*/\nexport function arraySpreadable(json?: Json): readonly Json[] {\n    if (!Array.isArray(json)) {\n        return []\n    }\n    return json as readonly Json[]\n}\n\nexport type HandlerConfiguration = {\n    /**\n     * An indication of CPU usage of the handler.\n     * @default 'low'\n     */\n    readonly compute?: 'high' | 'low'\n    /**\n     * An indication of memory usage of the handler.\n     * @default 'low'\n     */\n    readonly memory?: 'high' | 'low'\n    /**\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.\n     * @default false\n     */\n    readonly excludeBodyFromLogs?: boolean\n    /**\n     * The level below which log entries will be discarded.\n     * @default 'trace'\n     */\n    readonly minimumLogLevel?: 'trace' | 'debug' | 'info' | 'warning' | 'error' | 'fatal'\n    /**\n     * The number of seconds the function is expected to finish executing in.\n     */\n    readonly timeout?: number\n}\n\nexport type Context = {\n    readonly env: Environment\n    readonly log: Logger\n    readonly signal: AbortSignal\n    now(): Date\n\n    readonly operationId?: string\n    readonly client?: {\n        readonly id?: string\n        readonly ip?: string\n        readonly port?: number\n        readonly userAgent?: string\n    }\n    readonly meta?: {\n        readonly packageName: string\n        readonly fileName: string\n        readonly revision?: string\n    }\n\n    emit(topic: string, type: string, subject: string, data?: Json, messageId?: string): void\n    eventBarrier(): Promise<void>\n\n    onSuccess(fn: () => Promise<void> | void): void\n}\n\nexport function httpRequestHeaders(context: Context) {\n    const headers: { [key: string]: string } = {\n        'user-agent': `${context.meta?.packageName ?? '?'}/${context.meta?.revision ?? '?'}`,\n    }\n    if (context.operationId) {\n        headers['x-request-id'] = context.operationId\n    }\n    if (context.client) {\n        if (context.client.id) {\n            headers['x-client-id'] = context.client.id\n        }\n        if (!!context.client.ip || !!context.client.port) {\n            headers['x-forwarded-for'] = `${context.client.ip ?? ''}:${context.client.port ?? ''}`\n        }\n        if (context.client.userAgent) {\n            headers['x-forwarded-for-user-agent'] = context.client.userAgent\n        }\n    }\n    return headers\n}\n\nexport async function measure<T>(\n    logger: { trace: (message: string, _: undefined, f: object) => void },\n    name: string,\n    fn: () => Promise<T> | T,\n    fields?: object,\n) {\n    const start = performance.now()\n    try {\n        return await fn()\n    } finally {\n        const end = performance.now()\n        logger.trace(`Measurement of ${name} time`, undefined, {\n            start: highPrecisionISODate(start),\n            end: highPrecisionISODate(end),\n            duration: (Math.round(end * 10_000) - Math.round(start * 10_000)) / 10_000,\n            ...fields,\n        })\n    }\n}\n"]}
52
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAA;AAC7C,OAAO,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAA;AA+BxD,eAAe;AACf,MAAM,UAAU,gBAAgB,CAAC,IAAW;IACxC,IAAI,CAAC,IAAI,EAAE,CAAC;QACR,OAAO,EAAE,CAAA;IACb,CAAC;IACD,OAAO,IAAmD,CAAA;AAC9D,CAAC;AAED,eAAe;AACf,MAAM,UAAU,eAAe,CAAC,IAAW;IACvC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;QACvB,OAAO,EAAE,CAAA;IACb,CAAC;IACD,OAAO,IAAuB,CAAA;AAClC,CAAC;AA2DD,MAAM,UAAU,kBAAkB,CAAC,OAAgB;IAC/C,MAAM,OAAO,GAA8B;QACvC,YAAY,EAAE,GAAG,OAAO,CAAC,IAAI,EAAE,WAAW,IAAI,GAAG,IAAI,OAAO,CAAC,IAAI,EAAE,QAAQ,IAAI,GAAG,EAAE;KACvF,CAAA;IACD,IAAI,OAAO,CAAC,WAAW,EAAE,CAAC;QACtB,OAAO,CAAC,cAAc,CAAC,GAAG,OAAO,CAAC,WAAW,CAAA;IACjD,CAAC;IACD,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;QACjB,IAAI,OAAO,CAAC,MAAM,CAAC,EAAE,EAAE,CAAC;YACpB,OAAO,CAAC,aAAa,CAAC,GAAG,OAAO,CAAC,MAAM,CAAC,EAAE,CAAA;QAC9C,CAAC;QACD,IAAI,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,IAAI,CAAC,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC;YAC/C,OAAO,CAAC,iBAAiB,CAAC,GAAG,GAAG,OAAO,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,IAAI,OAAO,CAAC,MAAM,CAAC,IAAI,IAAI,EAAE,EAAE,CAAA;QAC1F,CAAC;QACD,IAAI,OAAO,CAAC,MAAM,CAAC,SAAS,EAAE,CAAC;YAC3B,OAAO,CAAC,4BAA4B,CAAC,GAAG,OAAO,CAAC,MAAM,CAAC,SAAS,CAAA;QACpE,CAAC;IACL,CAAC;IACD,OAAO,OAAO,CAAA;AAClB,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,OAAO,CACzB,MAAqE,EACrE,IAAY,EACZ,EAAwB,EACxB,MAAe;IAEf,MAAM,KAAK,GAAG,WAAW,CAAC,GAAG,EAAE,CAAA;IAC/B,IAAI,CAAC;QACD,OAAO,MAAM,EAAE,EAAE,CAAA;IACrB,CAAC;YAAS,CAAC;QACP,MAAM,GAAG,GAAG,WAAW,CAAC,GAAG,EAAE,CAAA;QAC7B,MAAM,CAAC,KAAK,CAAC,kBAAkB,IAAI,OAAO,EAAE,SAAS,EAAE;YACnD,KAAK,EAAE,oBAAoB,CAAC,KAAK,CAAC;YAClC,GAAG,EAAE,oBAAoB,CAAC,GAAG,CAAC;YAC9B,QAAQ,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,GAAG,MAAM,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,MAAM,CAAC,CAAC,GAAG,MAAM;YAC1E,GAAG,MAAM;SACZ,CAAC,CAAA;IACN,CAAC;AACL,CAAC","sourcesContent":["import { performance } from 'node:perf_hooks'\nimport { highPrecisionISODate } from './host/logging.js'\n\nexport type Environment = {\n    readonly [key: string]: string\n}\n\nexport type Logger = {\n    enrich(fields: object): Logger\n    trace(message: string, error?: unknown, fields?: object): void\n    debug(message: string, error?: unknown, fields?: object): void\n    info(message: string, error?: unknown, fields?: object): void\n    warn(message: string, error?: unknown, fields?: object): void\n    error(message: string, error?: unknown, fields?: object): void\n    fatal(message: string, error?: unknown, fields?: object): void\n}\n\nexport type MutableJson =\n    | null\n    | boolean\n    | number\n    | string\n    | MutableJson[]\n    | { [key: string]: MutableJson }\nexport type Json =\n    | null\n    | boolean\n    | number\n    | string\n    | readonly Json[]\n    | { readonly [key: string]: Json }\n\n/*@__INLINE__*/\nexport function objectSpreadable(json?: Json): { readonly [key: string]: Json } {\n    if (!json) {\n        return {}\n    }\n    return json as unknown as { readonly [key: string]: Json }\n}\n\n/*@__INLINE__*/\nexport function arraySpreadable(json?: Json): readonly Json[] {\n    if (!Array.isArray(json)) {\n        return []\n    }\n    return json as readonly Json[]\n}\n\nexport type HandlerConfiguration = {\n    /**\n     * An indication of CPU usage of the handler.\n     * @default 'low'\n     */\n    readonly compute?: 'high' | 'low'\n    /**\n     * An indication of memory usage of the handler.\n     * @default 'low'\n     */\n    readonly memory?: 'high' | 'low'\n    /**\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.\n     * @default false\n     */\n    readonly excludeBodyFromLogs?: boolean\n    /**\n     * The level below which log entries will be discarded.\n     * @default 'trace'\n     */\n    readonly minimumLogLevel?: 'trace' | 'debug' | 'info' | 'warning' | 'error' | 'fatal'\n    /**\n     * The number of seconds the function is expected to finish executing in.\n     */\n    readonly timeout?: number\n}\n\nexport type Context = {\n    readonly env: Environment\n    readonly log: Logger\n    readonly signal: AbortSignal\n    now(): Date\n\n    readonly operationId?: string\n    readonly client?: {\n        readonly id?: string\n        readonly ip?: string\n        readonly port?: number\n        readonly userAgent?: string\n    }\n    readonly meta?: {\n        readonly packageName: string\n        readonly fileName: string\n        readonly revision?: string\n    }\n\n    emit(\n        topic: string,\n        type: string,\n        subject: string,\n        data?: Json,\n        messageId?: string,\n    ): Promise<void>\n\n    onSuccess(fn: () => Promise<void> | void): void\n}\n\nexport function httpRequestHeaders(context: Context) {\n    const headers: { [key: string]: string } = {\n        'user-agent': `${context.meta?.packageName ?? '?'}/${context.meta?.revision ?? '?'}`,\n    }\n    if (context.operationId) {\n        headers['x-request-id'] = context.operationId\n    }\n    if (context.client) {\n        if (context.client.id) {\n            headers['x-client-id'] = context.client.id\n        }\n        if (!!context.client.ip || !!context.client.port) {\n            headers['x-forwarded-for'] = `${context.client.ip ?? ''}:${context.client.port ?? ''}`\n        }\n        if (context.client.userAgent) {\n            headers['x-forwarded-for-user-agent'] = context.client.userAgent\n        }\n    }\n    return headers\n}\n\nexport async function measure<T>(\n    logger: { trace: (message: string, _: undefined, f: object) => void },\n    name: string,\n    fn: () => Promise<T> | T,\n    fields?: object,\n) {\n    const start = performance.now()\n    try {\n        return await fn()\n    } finally {\n        const end = performance.now()\n        logger.trace(`Measurement of ${name} time`, undefined, {\n            start: highPrecisionISODate(start),\n            end: highPrecisionISODate(end),\n            duration: (Math.round(end * 10_000) - Math.round(start * 10_000)) / 10_000,\n            ...fields,\n        })\n    }\n}\n"]}
package/event.d.ts ADDED
@@ -0,0 +1,8 @@
1
+ import { Context, HandlerConfiguration, type Json } from './context.js';
2
+ export * from './context.js';
3
+ export type EventHandlerConfiguration = HandlerConfiguration & {};
4
+ export type Handler = (context: Context, subject: string, event: {
5
+ readonly [key: string]: Json;
6
+ } | undefined, timestamp: Date, messageId: string) => Promise<void> | void;
7
+ export declare function on(topic: string, event: string, fn: Handler): void;
8
+ export declare function on(topic: string, event: string, config: HandlerConfiguration, fn: Handler): void;
package/event.js ADDED
@@ -0,0 +1,6 @@
1
+ import { registerEventHandler } from './host/event-registry.js';
2
+ export * from './context.js';
3
+ export function on(topic, event, configOrHandler, fn) {
4
+ registerEventHandler(topic, event, configOrHandler, fn);
5
+ }
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXZlbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJldmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQTtBQUUvRCxjQUFjLGNBQWMsQ0FBQTtBQWtCNUIsTUFBTSxVQUFVLEVBQUUsQ0FDZCxLQUFhLEVBQ2IsS0FBYSxFQUNiLGVBQStDLEVBQy9DLEVBQVk7SUFFWixvQkFBb0IsQ0FBQyxLQUFLLEVBQUUsS0FBSyxFQUFFLGVBQWUsRUFBRSxFQUFFLENBQUMsQ0FBQTtBQUMzRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29udGV4dCwgSGFuZGxlckNvbmZpZ3VyYXRpb24sIHR5cGUgSnNvbiB9IGZyb20gJy4vY29udGV4dC5qcydcbmltcG9ydCB7IHJlZ2lzdGVyRXZlbnRIYW5kbGVyIH0gZnJvbSAnLi9ob3N0L2V2ZW50LXJlZ2lzdHJ5LmpzJ1xuXG5leHBvcnQgKiBmcm9tICcuL2NvbnRleHQuanMnXG5cbmV4cG9ydCB0eXBlIEV2ZW50SGFuZGxlckNvbmZpZ3VyYXRpb24gPSBIYW5kbGVyQ29uZmlndXJhdGlvbiAmIHt9XG5cbmV4cG9ydCB0eXBlIEhhbmRsZXIgPSAoXG4gICAgY29udGV4dDogQ29udGV4dCxcbiAgICBzdWJqZWN0OiBzdHJpbmcsXG4gICAgZXZlbnQ6XG4gICAgICAgIHwge1xuICAgICAgICAgICAgICByZWFkb25seSBba2V5OiBzdHJpbmddOiBKc29uXG4gICAgICAgICAgfVxuICAgICAgICB8IHVuZGVmaW5lZCxcbiAgICB0aW1lc3RhbXA6IERhdGUsXG4gICAgbWVzc2FnZUlkOiBzdHJpbmcsXG4pID0+IFByb21pc2U8dm9pZD4gfCB2b2lkXG5cbmV4cG9ydCBmdW5jdGlvbiBvbih0b3BpYzogc3RyaW5nLCBldmVudDogc3RyaW5nLCBmbjogSGFuZGxlcik6IHZvaWRcbmV4cG9ydCBmdW5jdGlvbiBvbih0b3BpYzogc3RyaW5nLCBldmVudDogc3RyaW5nLCBjb25maWc6IEhhbmRsZXJDb25maWd1cmF0aW9uLCBmbjogSGFuZGxlcik6IHZvaWRcbmV4cG9ydCBmdW5jdGlvbiBvbihcbiAgICB0b3BpYzogc3RyaW5nLFxuICAgIGV2ZW50OiBzdHJpbmcsXG4gICAgY29uZmlnT3JIYW5kbGVyOiBIYW5kbGVyQ29uZmlndXJhdGlvbiB8IEhhbmRsZXIsXG4gICAgZm4/OiBIYW5kbGVyLFxuKTogdm9pZCB7XG4gICAgcmVnaXN0ZXJFdmVudEhhbmRsZXIodG9waWMsIGV2ZW50LCBjb25maWdPckhhbmRsZXIsIGZuKVxufVxuIl19
package/host/context.d.ts CHANGED
@@ -1,5 +1,5 @@
1
- import { Context, Environment, Logger } from '../context.js';
2
- import { FullConfiguration, Metadata } from './registry.js';
1
+ import { Context, Environment, Json, Logger } from '../context.js';
2
+ import type { FullConfiguration, Metadata } from './meta.js';
3
3
  export type ClientInfo = {
4
4
  readonly operationId?: string;
5
5
  readonly clientId?: string;
@@ -7,21 +7,10 @@ export type ClientInfo = {
7
7
  readonly clientPort?: number;
8
8
  readonly userAgent?: string;
9
9
  };
10
- export type EventMetadata = {
11
- topic: string;
12
- type: string;
13
- subject: string;
14
- id?: string;
15
- };
16
- export type BufferedEvent = {
17
- eventTime: Date;
18
- meta: Omit<EventMetadata, 'topic'>;
19
- ids: ClientInfo;
20
- json?: string;
21
- };
22
10
  export type EventTransport = {
23
- readonly publishRate: number;
24
- sendEvents(topic: string, events: BufferedEvent[], signal: AbortSignal): Promise<void>;
11
+ sendEvent(topic: string, type: string, subject: string, data: {
12
+ readonly [key: string]: Json;
13
+ } | undefined, messageId: string | undefined, signal: AbortSignal): Promise<void>;
25
14
  };
26
15
  export type LogLevel = 'trace' | 'debug' | 'info' | 'warning' | 'error' | 'fatal';
27
16
  export type LogEntry = {
@@ -33,7 +22,7 @@ export type LogEntry = {
33
22
  };
34
23
  export type LogTransport = {
35
24
  readonly publishRate?: number;
36
- sendEntries(entries: LogEntry[], signal: AbortSignal): Promise<void> | undefined;
25
+ sendEntries(entries: LogEntry[], signal: AbortSignal): Promise<void> | undefined | void;
37
26
  };
38
27
  export type RootLogger = {
39
28
  enrichReserved(fields: object): RootLogger;
package/host/context.js CHANGED
@@ -1,4 +1,3 @@
1
- import { EventCollector } from './events.js';
2
1
  import { makeLogger } from './logging.js';
3
2
  class LogMulticaster {
4
3
  #transports;
@@ -32,7 +31,6 @@ export function createContext(clientInfo, loggers, eventTransport, timeouts, out
32
31
  },
33
32
  });
34
33
  globalLogger = logger;
35
- const emitter = new EventCollector(eventTransport, logger, clientInfo, timeout, outerController.signal);
36
34
  const successHandlers = [];
37
35
  const ctx = {
38
36
  env: environment ?? process.env,
@@ -52,10 +50,7 @@ export function createContext(clientInfo, loggers, eventTransport, timeouts, out
52
50
  revision: meta.revision,
53
51
  }
54
52
  : undefined,
55
- emit: (topic, type, subject, data, messageId) => {
56
- emitter.emit({ topic, type, subject, id: messageId }, data);
57
- },
58
- eventBarrier: () => emitter.flush(),
53
+ emit: (topic, type, subject, data, messageId) => eventTransport.sendEvent(topic, type, subject, data, messageId, outerController.signal),
59
54
  onSuccess: (fn) => successHandlers.push(fn),
60
55
  };
61
56
  const timeoutHandle = setTimeout(() => {
@@ -63,8 +58,6 @@ export function createContext(clientInfo, loggers, eventTransport, timeouts, out
63
58
  innerController.abort();
64
59
  // eslint-disable-next-line no-void
65
60
  void logger.flush();
66
- // eslint-disable-next-line no-void
67
- void emitter.flush();
68
61
  }, timeout);
69
62
  const flushHandle = setTimeout(() => {
70
63
  logger.error('Aborting flush.', undefined, undefined);
@@ -76,7 +69,6 @@ export function createContext(clientInfo, loggers, eventTransport, timeouts, out
76
69
  success: () => Promise.all(successHandlers.map(fn => fn())),
77
70
  flush: async () => {
78
71
  clearTimeout(timeoutHandle);
79
- await emitter.flush();
80
72
  await logger.flush();
81
73
  clearTimeout(flushHandle);
82
74
  },
@@ -89,4 +81,4 @@ process.on('uncaughtException', err => {
89
81
  process.on('unhandledRejection', reason => {
90
82
  globalLogger?.fatal('Unhandled rejection.', reason, undefined);
91
83
  });
92
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"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","sourcesContent":["import { Context, Environment, Json, Logger } from '../context.js'\nimport { EventCollector } from './events.js'\nimport { makeLogger } from './logging.js'\nimport { FullConfiguration, Metadata } from './registry.js'\n\nexport type ClientInfo = {\n    readonly operationId?: string\n    readonly clientId?: string\n    readonly clientIp?: string\n    readonly clientPort?: number\n    readonly userAgent?: string\n}\n\nexport type EventMetadata = {\n    topic: string\n    type: string\n    subject: string\n    id?: string\n}\n\nexport type BufferedEvent = {\n    eventTime: Date\n    meta: Omit<EventMetadata, 'topic'>\n    ids: ClientInfo\n    json?: string\n}\n\nexport type EventTransport = {\n    readonly publishRate: number\n    sendEvents(topic: string, events: BufferedEvent[], signal: AbortSignal): Promise<void>\n}\n\nexport type LogLevel = 'trace' | 'debug' | 'info' | 'warning' | 'error' | 'fatal'\n\nexport type LogEntry = {\n    readonly level: LogLevel\n    readonly timestamp: number\n    readonly message: string\n    readonly error: unknown\n    readonly json: string\n}\n\nexport type LogTransport = {\n    readonly publishRate?: number\n    sendEntries(entries: LogEntry[], signal: AbortSignal): Promise<void> | undefined\n}\n\nclass LogMulticaster implements LogTransport {\n    readonly #transports: LogTransport[]\n    readonly publishRate: number\n\n    constructor(transports: LogTransport[]) {\n        this.#transports = transports\n        this.publishRate = transports.map(t => t.publishRate).sort()[0] ?? Number.MAX_SAFE_INTEGER\n    }\n\n    sendEntries(entries: LogEntry[], signal: AbortSignal) {\n        const promises = this.#transports.map(t => t.sendEntries(entries, signal)).filter(p => !!p)\n        if (promises.length === 0) {\n            return\n        }\n        return Promise.all(promises) as unknown as Promise<void>\n    }\n}\n\nexport type RootLogger = {\n    enrichReserved(fields: object): RootLogger\n    flush(): Promise<void>\n} & Logger\n\nexport function createContext(\n    clientInfo: ClientInfo,\n    loggers: LogTransport[],\n    eventTransport: EventTransport,\n    timeouts: { default: number; cap?: number },\n    outerController: AbortController,\n    config?: FullConfiguration,\n    meta?: Metadata,\n    environment?: Environment,\n    now?: () => Date,\n): {\n    log: RootLogger\n    context: Omit<Context, 'log'>\n    success: () => Promise<unknown>\n    flush: () => Promise<void>\n} {\n    const timeout =\n        (timeouts.cap\n            ? Math.min(config?.timeout ?? timeouts.default, timeouts.cap)\n            : (config?.timeout ?? timeouts.default)) * 1000\n    const innerController = new AbortController()\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n    const logTransport = loggers.length === 1 ? loggers[0]! : new LogMulticaster(loggers)\n    const logger = makeLogger(\n        logTransport,\n        config?.minimumLogLevel,\n        outerController.signal,\n    ).enrichReserved({\n        operationId: clientInfo.operationId,\n        client: {\n            id: clientInfo.clientId,\n            ip: clientInfo.clientIp,\n            port: clientInfo.clientPort,\n            userAgent: clientInfo.userAgent,\n        },\n    })\n    globalLogger = logger\n    const emitter = new EventCollector(\n        eventTransport,\n        logger,\n        clientInfo,\n        timeout,\n        outerController.signal,\n    )\n    const successHandlers: (() => Promise<void> | void)[] = []\n    const ctx = {\n        env: environment ?? (process.env as Environment),\n        signal: innerController.signal,\n        now: now ?? (() => new Date()),\n        operationId: clientInfo.operationId,\n        client: {\n            id: clientInfo.clientId,\n            ip: clientInfo.clientIp,\n            port: clientInfo.clientPort,\n            userAgent: clientInfo.userAgent,\n        },\n        meta: meta\n            ? {\n                  packageName: meta.packageName,\n                  fileName: meta.fileName,\n                  revision: meta.revision,\n              }\n            : undefined,\n        emit: (topic: string, type: string, subject: string, data?: Json, messageId?: string) => {\n            emitter.emit({ topic, type, subject, id: messageId }, data)\n        },\n        eventBarrier: () => emitter.flush(),\n        onSuccess: (fn: () => Promise<void> | void) => successHandlers.push(fn),\n    }\n    const timeoutHandle = setTimeout(() => {\n        logger.error('Timeout.', undefined, undefined)\n        innerController.abort()\n        // eslint-disable-next-line no-void\n        void logger.flush()\n        // eslint-disable-next-line no-void\n        void emitter.flush()\n    }, timeout)\n    const flushHandle = setTimeout(() => {\n        logger.error('Aborting flush.', undefined, undefined)\n        outerController.abort()\n    }, timeout + 15_000)\n    return {\n        log: logger,\n        context: ctx,\n        success: () => Promise.all(successHandlers.map(fn => fn())),\n        flush: async () => {\n            clearTimeout(timeoutHandle)\n            await emitter.flush()\n            await logger.flush()\n            clearTimeout(flushHandle)\n        },\n    }\n}\n\nlet globalLogger: Logger | undefined\n\nprocess.on('uncaughtException', err => {\n    globalLogger?.fatal('Uncaught exception.', err, undefined)\n})\nprocess.on('unhandledRejection', reason => {\n    globalLogger?.fatal('Unhandled rejection.', reason, undefined)\n})\n"]}
84
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"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","sourcesContent":["import { Context, Environment, Json, Logger } from '../context.js'\nimport { makeLogger } from './logging.js'\nimport type { FullConfiguration, Metadata } from './meta.js'\n\nexport type ClientInfo = {\n    readonly operationId?: string\n    readonly clientId?: string\n    readonly clientIp?: string\n    readonly clientPort?: number\n    readonly userAgent?: string\n}\n\nexport type EventTransport = {\n    sendEvent(\n        topic: string,\n        type: string,\n        subject: string,\n        data:\n            | {\n                  readonly [key: string]: Json\n              }\n            | undefined,\n        messageId: string | undefined,\n        signal: AbortSignal,\n    ): Promise<void>\n}\n\nexport type LogLevel = 'trace' | 'debug' | 'info' | 'warning' | 'error' | 'fatal'\n\nexport type LogEntry = {\n    readonly level: LogLevel\n    readonly timestamp: number\n    readonly message: string\n    readonly error: unknown\n    readonly json: string\n}\n\nexport type LogTransport = {\n    readonly publishRate?: number\n    sendEntries(entries: LogEntry[], signal: AbortSignal): Promise<void> | undefined | void\n}\n\nclass LogMulticaster implements LogTransport {\n    readonly #transports: LogTransport[]\n    readonly publishRate: number\n\n    constructor(transports: LogTransport[]) {\n        this.#transports = transports\n        this.publishRate = transports.map(t => t.publishRate).sort()[0] ?? Number.MAX_SAFE_INTEGER\n    }\n\n    sendEntries(entries: LogEntry[], signal: AbortSignal) {\n        const promises = this.#transports.map(t => t.sendEntries(entries, signal)).filter(p => !!p)\n        if (promises.length === 0) {\n            return\n        }\n        return Promise.all(promises) as unknown as Promise<void>\n    }\n}\n\nexport type RootLogger = {\n    enrichReserved(fields: object): RootLogger\n    flush(): Promise<void>\n} & Logger\n\nexport function createContext(\n    clientInfo: ClientInfo,\n    loggers: LogTransport[],\n    eventTransport: EventTransport,\n    timeouts: { default: number; cap?: number },\n    outerController: AbortController,\n    config?: FullConfiguration,\n    meta?: Metadata,\n    environment?: Environment,\n    now?: () => Date,\n): {\n    log: RootLogger\n    context: Omit<Context, 'log'>\n    success: () => Promise<unknown>\n    flush: () => Promise<void>\n} {\n    const timeout =\n        (timeouts.cap\n            ? Math.min(config?.timeout ?? timeouts.default, timeouts.cap)\n            : (config?.timeout ?? timeouts.default)) * 1000\n    const innerController = new AbortController()\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n    const logTransport = loggers.length === 1 ? loggers[0]! : new LogMulticaster(loggers)\n    const logger = makeLogger(\n        logTransport,\n        config?.minimumLogLevel,\n        outerController.signal,\n    ).enrichReserved({\n        operationId: clientInfo.operationId,\n        client: {\n            id: clientInfo.clientId,\n            ip: clientInfo.clientIp,\n            port: clientInfo.clientPort,\n            userAgent: clientInfo.userAgent,\n        },\n    })\n    globalLogger = logger\n    const successHandlers: (() => Promise<void> | void)[] = []\n    const ctx = {\n        env: environment ?? (process.env as Environment),\n        signal: innerController.signal,\n        now: now ?? (() => new Date()),\n        operationId: clientInfo.operationId,\n        client: {\n            id: clientInfo.clientId,\n            ip: clientInfo.clientIp,\n            port: clientInfo.clientPort,\n            userAgent: clientInfo.userAgent,\n        },\n        meta: meta\n            ? {\n                  packageName: meta.packageName,\n                  fileName: meta.fileName,\n                  revision: meta.revision,\n              }\n            : undefined,\n        emit: (\n            topic: string,\n            type: string,\n            subject: string,\n            data?: {\n                readonly [key: string]: Json\n            },\n            messageId?: string,\n        ) =>\n            eventTransport.sendEvent(topic, type, subject, data, messageId, outerController.signal),\n        onSuccess: (fn: () => Promise<void> | void) => successHandlers.push(fn),\n    }\n    const timeoutHandle = setTimeout(() => {\n        logger.error('Timeout.', undefined, undefined)\n        innerController.abort()\n        // eslint-disable-next-line no-void\n        void logger.flush()\n    }, timeout)\n    const flushHandle = setTimeout(() => {\n        logger.error('Aborting flush.', undefined, undefined)\n        outerController.abort()\n    }, timeout + 15_000)\n    return {\n        log: logger,\n        context: ctx,\n        success: () => Promise.all(successHandlers.map(fn => fn())),\n        flush: async () => {\n            clearTimeout(timeoutHandle)\n            await logger.flush()\n            clearTimeout(flushHandle)\n        },\n    }\n}\n\nlet globalLogger: Logger | undefined\n\nprocess.on('uncaughtException', err => {\n    globalLogger?.fatal('Uncaught exception.', err, undefined)\n})\nprocess.on('unhandledRejection', reason => {\n    globalLogger?.fatal('Unhandled rejection.', reason, undefined)\n})\n"]}
@@ -0,0 +1,10 @@
1
+ import { type Handler as EventFunction, type HandlerConfiguration } from '../event.js';
2
+ import { type FullConfiguration, type Metadata } from './meta.js';
3
+ export type EventHandler = {
4
+ meta: Metadata | undefined;
5
+ config: FullConfiguration | undefined;
6
+ topic: string;
7
+ type: string;
8
+ entry: EventFunction;
9
+ };
10
+ export declare function registerEventHandler(topic: string, type: string, configOrHandler: HandlerConfiguration | EventFunction, fn?: EventFunction): void;
@@ -0,0 +1,28 @@
1
+ import { combineConfig, getMetadata } from './meta.js';
2
+ import { addHandler } from './registry.js';
3
+ let eventHostRegistry;
4
+ function setEventHost(host) {
5
+ eventHostRegistry = host;
6
+ }
7
+ function eventHost(meta, cfg, topic, type, entry) {
8
+ addHandler('event', {
9
+ meta,
10
+ config: combineConfig(meta?.config, cfg),
11
+ topic,
12
+ type,
13
+ entry,
14
+ });
15
+ }
16
+ setEventHost(eventHost);
17
+ export function registerEventHandler(topic, type, configOrHandler, fn) {
18
+ if (typeof configOrHandler === 'function') {
19
+ eventHostRegistry(getMetadata(), undefined, topic, type, configOrHandler);
20
+ }
21
+ else {
22
+ if (!fn) {
23
+ throw new Error('Please provide a handler function.');
24
+ }
25
+ eventHostRegistry(getMetadata(), configOrHandler, topic, type, fn);
26
+ }
27
+ }
28
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXZlbnQtcmVnaXN0cnkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJldmVudC1yZWdpc3RyeS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsYUFBYSxFQUFFLFdBQVcsRUFBeUMsTUFBTSxXQUFXLENBQUE7QUFDN0YsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQTtBQWtCMUMsSUFBSSxpQkFBNEIsQ0FBQTtBQUVoQyxTQUFTLFlBQVksQ0FBQyxJQUFlO0lBQ2pDLGlCQUFpQixHQUFHLElBQUksQ0FBQTtBQUM1QixDQUFDO0FBRUQsU0FBUyxTQUFTLENBQ2QsSUFBMEIsRUFDMUIsR0FBcUMsRUFDckMsS0FBYSxFQUNiLElBQVksRUFDWixLQUFvQjtJQUVwQixVQUFVLENBQUMsT0FBTyxFQUFFO1FBQ2hCLElBQUk7UUFDSixNQUFNLEVBQUUsYUFBYSxDQUFDLElBQUksRUFBRSxNQUFNLEVBQUUsR0FBRyxDQUFDO1FBQ3hDLEtBQUs7UUFDTCxJQUFJO1FBQ0osS0FBSztLQUNSLENBQUMsQ0FBQTtBQUNOLENBQUM7QUFFRCxZQUFZLENBQUMsU0FBUyxDQUFDLENBQUE7QUFFdkIsTUFBTSxVQUFVLG9CQUFvQixDQUNoQyxLQUFhLEVBQ2IsSUFBWSxFQUNaLGVBQXFELEVBQ3JELEVBQWtCO0lBRWxCLElBQUksT0FBTyxlQUFlLEtBQUssVUFBVSxFQUFFLENBQUM7UUFDeEMsaUJBQWlCLENBQUMsV0FBVyxFQUFFLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxJQUFJLEVBQUUsZUFBZSxDQUFDLENBQUE7SUFDN0UsQ0FBQztTQUFNLENBQUM7UUFDSixJQUFJLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDTixNQUFNLElBQUksS0FBSyxDQUFDLG9DQUFvQyxDQUFDLENBQUE7UUFDekQsQ0FBQztRQUNELGlCQUFpQixDQUFDLFdBQVcsRUFBRSxFQUFFLGVBQWUsRUFBRSxLQUFLLEVBQUUsSUFBSSxFQUFFLEVBQUUsQ0FBQyxDQUFBO0lBQ3RFLENBQUM7QUFDTCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgdHlwZSBIYW5kbGVyIGFzIEV2ZW50RnVuY3Rpb24sIHR5cGUgSGFuZGxlckNvbmZpZ3VyYXRpb24gfSBmcm9tICcuLi9ldmVudC5qcydcbmltcG9ydCB7IGNvbWJpbmVDb25maWcsIGdldE1ldGFkYXRhLCB0eXBlIEZ1bGxDb25maWd1cmF0aW9uLCB0eXBlIE1ldGFkYXRhIH0gZnJvbSAnLi9tZXRhLmpzJ1xuaW1wb3J0IHsgYWRkSGFuZGxlciB9IGZyb20gJy4vcmVnaXN0cnkuanMnXG5cbmV4cG9ydCB0eXBlIEV2ZW50SGFuZGxlciA9IHtcbiAgICBtZXRhOiBNZXRhZGF0YSB8IHVuZGVmaW5lZFxuICAgIGNvbmZpZzogRnVsbENvbmZpZ3VyYXRpb24gfCB1bmRlZmluZWRcbiAgICB0b3BpYzogc3RyaW5nXG4gICAgdHlwZTogc3RyaW5nXG4gICAgZW50cnk6IEV2ZW50RnVuY3Rpb25cbn1cblxudHlwZSBFdmVudEhvc3QgPSAoXG4gICAgbWV0YTogTWV0YWRhdGEgfCB1bmRlZmluZWQsXG4gICAgY29uZmlnOiBIYW5kbGVyQ29uZmlndXJhdGlvbiB8IHVuZGVmaW5lZCxcbiAgICB0b3BpYzogc3RyaW5nLFxuICAgIHR5cGU6IHN0cmluZyxcbiAgICBoYW5kbGVyOiBFdmVudEZ1bmN0aW9uLFxuKSA9PiB2b2lkXG5cbmxldCBldmVudEhvc3RSZWdpc3RyeTogRXZlbnRIb3N0XG5cbmZ1bmN0aW9uIHNldEV2ZW50SG9zdChob3N0OiBFdmVudEhvc3QpIHtcbiAgICBldmVudEhvc3RSZWdpc3RyeSA9IGhvc3Rcbn1cblxuZnVuY3Rpb24gZXZlbnRIb3N0KFxuICAgIG1ldGE6IE1ldGFkYXRhIHwgdW5kZWZpbmVkLFxuICAgIGNmZzogSGFuZGxlckNvbmZpZ3VyYXRpb24gfCB1bmRlZmluZWQsXG4gICAgdG9waWM6IHN0cmluZyxcbiAgICB0eXBlOiBzdHJpbmcsXG4gICAgZW50cnk6IEV2ZW50RnVuY3Rpb24sXG4pIHtcbiAgICBhZGRIYW5kbGVyKCdldmVudCcsIHtcbiAgICAgICAgbWV0YSxcbiAgICAgICAgY29uZmlnOiBjb21iaW5lQ29uZmlnKG1ldGE/LmNvbmZpZywgY2ZnKSxcbiAgICAgICAgdG9waWMsXG4gICAgICAgIHR5cGUsXG4gICAgICAgIGVudHJ5LFxuICAgIH0pXG59XG5cbnNldEV2ZW50SG9zdChldmVudEhvc3QpXG5cbmV4cG9ydCBmdW5jdGlvbiByZWdpc3RlckV2ZW50SGFuZGxlcihcbiAgICB0b3BpYzogc3RyaW5nLFxuICAgIHR5cGU6IHN0cmluZyxcbiAgICBjb25maWdPckhhbmRsZXI6IEhhbmRsZXJDb25maWd1cmF0aW9uIHwgRXZlbnRGdW5jdGlvbixcbiAgICBmbj86IEV2ZW50RnVuY3Rpb24sXG4pOiB2b2lkIHtcbiAgICBpZiAodHlwZW9mIGNvbmZpZ09ySGFuZGxlciA9PT0gJ2Z1bmN0aW9uJykge1xuICAgICAgICBldmVudEhvc3RSZWdpc3RyeShnZXRNZXRhZGF0YSgpLCB1bmRlZmluZWQsIHRvcGljLCB0eXBlLCBjb25maWdPckhhbmRsZXIpXG4gICAgfSBlbHNlIHtcbiAgICAgICAgaWYgKCFmbikge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdQbGVhc2UgcHJvdmlkZSBhIGhhbmRsZXIgZnVuY3Rpb24uJylcbiAgICAgICAgfVxuICAgICAgICBldmVudEhvc3RSZWdpc3RyeShnZXRNZXRhZGF0YSgpLCBjb25maWdPckhhbmRsZXIsIHRvcGljLCB0eXBlLCBmbilcbiAgICB9XG59XG4iXX0=
@@ -0,0 +1,11 @@
1
+ import { Context, type Json } from '../context.js';
2
+ import { RootLogger } from './context.js';
3
+ import type { EventHandler } from './event-registry.js';
4
+ export declare function handle(log: RootLogger, context: Omit<Context, 'log'>, handler: EventHandler, options: {
5
+ readonly subject: string;
6
+ readonly event: {
7
+ readonly [key: string]: Json;
8
+ } | undefined;
9
+ readonly timestamp: Date;
10
+ readonly messageId?: string;
11
+ }, success: () => Promise<unknown>): Promise<void>;
package/host/event.js ADDED
@@ -0,0 +1,15 @@
1
+ import { randomUUID } from 'node:crypto';
2
+ import { measure } from '../context.js';
3
+ export async function handle(log, context, handler, options, success) {
4
+ log = log.enrichReserved({ meta: context.meta, event: options });
5
+ log.trace('Event BEGIN');
6
+ try {
7
+ await measure(log, 'execution', () => handler.entry({ ...context, log }, options.subject, options.event, options.timestamp, options.messageId ?? randomUUID().replaceAll('-', '')));
8
+ log.debug('Event END');
9
+ await success();
10
+ }
11
+ catch (e) {
12
+ log.error('Event END', e);
13
+ }
14
+ }
15
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXZlbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJldmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sYUFBYSxDQUFBO0FBQ3hDLE9BQU8sRUFBVyxPQUFPLEVBQWEsTUFBTSxlQUFlLENBQUE7QUFJM0QsTUFBTSxDQUFDLEtBQUssVUFBVSxNQUFNLENBQ3hCLEdBQWUsRUFDZixPQUE2QixFQUM3QixPQUFxQixFQUNyQixPQVNDLEVBQ0QsT0FBK0I7SUFFL0IsR0FBRyxHQUFHLEdBQUcsQ0FBQyxjQUFjLENBQUMsRUFBRSxJQUFJLEVBQUUsT0FBTyxDQUFDLElBQUksRUFBRSxLQUFLLEVBQUUsT0FBTyxFQUFFLENBQUMsQ0FBQTtJQUNoRSxHQUFHLENBQUMsS0FBSyxDQUFDLGFBQWEsQ0FBQyxDQUFBO0lBQ3hCLElBQUksQ0FBQztRQUNELE1BQU0sT0FBTyxDQUFDLEdBQUcsRUFBRSxXQUFXLEVBQUUsR0FBRyxFQUFFLENBQ2pDLE9BQU8sQ0FBQyxLQUFLLENBQ1QsRUFBRSxHQUFHLE9BQU8sRUFBRSxHQUFHLEVBQUUsRUFDbkIsT0FBTyxDQUFDLE9BQU8sRUFDZixPQUFPLENBQUMsS0FBSyxFQUNiLE9BQU8sQ0FBQyxTQUFTLEVBQ2pCLE9BQU8sQ0FBQyxTQUFTLElBQUksVUFBVSxFQUFFLENBQUMsVUFBVSxDQUFDLEdBQUcsRUFBRSxFQUFFLENBQUMsQ0FDeEQsQ0FDSixDQUFBO1FBQ0QsR0FBRyxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUMsQ0FBQTtRQUN0QixNQUFNLE9BQU8sRUFBRSxDQUFBO0lBQ25CLENBQUM7SUFBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1FBQ1QsR0FBRyxDQUFDLEtBQUssQ0FBQyxXQUFXLEVBQUUsQ0FBQyxDQUFDLENBQUE7SUFDN0IsQ0FBQztBQUNMLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyByYW5kb21VVUlEIH0gZnJvbSAnbm9kZTpjcnlwdG8nXG5pbXBvcnQgeyBDb250ZXh0LCBtZWFzdXJlLCB0eXBlIEpzb24gfSBmcm9tICcuLi9jb250ZXh0LmpzJ1xuaW1wb3J0IHsgUm9vdExvZ2dlciB9IGZyb20gJy4vY29udGV4dC5qcydcbmltcG9ydCB0eXBlIHsgRXZlbnRIYW5kbGVyIH0gZnJvbSAnLi9ldmVudC1yZWdpc3RyeS5qcydcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGhhbmRsZShcbiAgICBsb2c6IFJvb3RMb2dnZXIsXG4gICAgY29udGV4dDogT21pdDxDb250ZXh0LCAnbG9nJz4sXG4gICAgaGFuZGxlcjogRXZlbnRIYW5kbGVyLFxuICAgIG9wdGlvbnM6IHtcbiAgICAgICAgcmVhZG9ubHkgc3ViamVjdDogc3RyaW5nXG4gICAgICAgIHJlYWRvbmx5IGV2ZW50OlxuICAgICAgICAgICAgfCB7XG4gICAgICAgICAgICAgICAgICByZWFkb25seSBba2V5OiBzdHJpbmddOiBKc29uXG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHwgdW5kZWZpbmVkXG4gICAgICAgIHJlYWRvbmx5IHRpbWVzdGFtcDogRGF0ZVxuICAgICAgICByZWFkb25seSBtZXNzYWdlSWQ/OiBzdHJpbmdcbiAgICB9LFxuICAgIHN1Y2Nlc3M6ICgpID0+IFByb21pc2U8dW5rbm93bj4sXG4pOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBsb2cgPSBsb2cuZW5yaWNoUmVzZXJ2ZWQoeyBtZXRhOiBjb250ZXh0Lm1ldGEsIGV2ZW50OiBvcHRpb25zIH0pXG4gICAgbG9nLnRyYWNlKCdFdmVudCBCRUdJTicpXG4gICAgdHJ5IHtcbiAgICAgICAgYXdhaXQgbWVhc3VyZShsb2csICdleGVjdXRpb24nLCAoKSA9PlxuICAgICAgICAgICAgaGFuZGxlci5lbnRyeShcbiAgICAgICAgICAgICAgICB7IC4uLmNvbnRleHQsIGxvZyB9LFxuICAgICAgICAgICAgICAgIG9wdGlvbnMuc3ViamVjdCxcbiAgICAgICAgICAgICAgICBvcHRpb25zLmV2ZW50LFxuICAgICAgICAgICAgICAgIG9wdGlvbnMudGltZXN0YW1wLFxuICAgICAgICAgICAgICAgIG9wdGlvbnMubWVzc2FnZUlkID8/IHJhbmRvbVVVSUQoKS5yZXBsYWNlQWxsKCctJywgJycpLFxuICAgICAgICAgICAgKSxcbiAgICAgICAgKVxuICAgICAgICBsb2cuZGVidWcoJ0V2ZW50IEVORCcpXG4gICAgICAgIGF3YWl0IHN1Y2Nlc3MoKVxuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgbG9nLmVycm9yKCdFdmVudCBFTkQnLCBlKVxuICAgIH1cbn1cbiJdfQ==
package/host/git.d.ts ADDED
@@ -0,0 +1 @@
1
+ export declare function getHash(path: string): Promise<string | undefined>;
package/host/git.js ADDED
@@ -0,0 +1,33 @@
1
+ import { exec } from 'node:child_process';
2
+ export async function getHash(path) {
3
+ try {
4
+ const [clean, hash = ''] = await Promise.all([isClean(path), getCommittedHash(path)]);
5
+ if (clean) {
6
+ return hash.slice(0, 7);
7
+ }
8
+ return hash.slice(0, 7) + '+';
9
+ }
10
+ catch {
11
+ return undefined;
12
+ }
13
+ }
14
+ async function isClean(path) {
15
+ const changes = await execAsync(path, 'git status --short');
16
+ return changes.length === 0;
17
+ }
18
+ async function getCommittedHash(path) {
19
+ const [long] = await execAsync(path, 'git rev-parse HEAD');
20
+ return long;
21
+ }
22
+ function execAsync(path, cmd) {
23
+ return new Promise((resolve, reject) => {
24
+ exec(cmd, { cwd: path }, (err, stdout) => {
25
+ if (err) {
26
+ reject(err);
27
+ return;
28
+ }
29
+ resolve(stdout.split('\n').slice(0, -1));
30
+ });
31
+ });
32
+ }
33
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2l0LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiZ2l0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxJQUFJLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQTtBQUV6QyxNQUFNLENBQUMsS0FBSyxVQUFVLE9BQU8sQ0FBQyxJQUFZO0lBQ3RDLElBQUksQ0FBQztRQUNELE1BQU0sQ0FBQyxLQUFLLEVBQUUsSUFBSSxHQUFHLEVBQUUsQ0FBQyxHQUFHLE1BQU0sT0FBTyxDQUFDLEdBQUcsQ0FBQyxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsRUFBRSxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUE7UUFDckYsSUFBSSxLQUFLLEVBQUUsQ0FBQztZQUNSLE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUE7UUFDM0IsQ0FBQztRQUNELE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLEdBQUcsR0FBRyxDQUFBO0lBQ2pDLENBQUM7SUFBQyxNQUFNLENBQUM7UUFDTCxPQUFPLFNBQVMsQ0FBQTtJQUNwQixDQUFDO0FBQ0wsQ0FBQztBQUVELEtBQUssVUFBVSxPQUFPLENBQUMsSUFBWTtJQUMvQixNQUFNLE9BQU8sR0FBRyxNQUFNLFNBQVMsQ0FBQyxJQUFJLEVBQUUsb0JBQW9CLENBQUMsQ0FBQTtJQUMzRCxPQUFPLE9BQU8sQ0FBQyxNQUFNLEtBQUssQ0FBQyxDQUFBO0FBQy9CLENBQUM7QUFFRCxLQUFLLFVBQVUsZ0JBQWdCLENBQUMsSUFBWTtJQUN4QyxNQUFNLENBQUMsSUFBSSxDQUFDLEdBQUcsTUFBTSxTQUFTLENBQUMsSUFBSSxFQUFFLG9CQUFvQixDQUFDLENBQUE7SUFDMUQsT0FBTyxJQUFJLENBQUE7QUFDZixDQUFDO0FBRUQsU0FBUyxTQUFTLENBQUMsSUFBWSxFQUFFLEdBQVc7SUFDeEMsT0FBTyxJQUFJLE9BQU8sQ0FBVyxDQUFDLE9BQU8sRUFBRSxNQUFNLEVBQUUsRUFBRTtRQUM3QyxJQUFJLENBQUMsR0FBRyxFQUFFLEVBQUUsR0FBRyxFQUFFLElBQUksRUFBRSxFQUFFLENBQUMsR0FBRyxFQUFFLE1BQU0sRUFBRSxFQUFFO1lBQ3JDLElBQUksR0FBRyxFQUFFLENBQUM7Z0JBQ04sTUFBTSxDQUFDLEdBQUcsQ0FBQyxDQUFBO2dCQUNYLE9BQU07WUFDVixDQUFDO1lBQ0QsT0FBTyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUE7UUFDNUMsQ0FBQyxDQUFDLENBQUE7SUFDTixDQUFDLENBQUMsQ0FBQTtBQUNOLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBleGVjIH0gZnJvbSAnbm9kZTpjaGlsZF9wcm9jZXNzJ1xuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gZ2V0SGFzaChwYXRoOiBzdHJpbmcpIHtcbiAgICB0cnkge1xuICAgICAgICBjb25zdCBbY2xlYW4sIGhhc2ggPSAnJ10gPSBhd2FpdCBQcm9taXNlLmFsbChbaXNDbGVhbihwYXRoKSwgZ2V0Q29tbWl0dGVkSGFzaChwYXRoKV0pXG4gICAgICAgIGlmIChjbGVhbikge1xuICAgICAgICAgICAgcmV0dXJuIGhhc2guc2xpY2UoMCwgNylcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gaGFzaC5zbGljZSgwLCA3KSArICcrJ1xuICAgIH0gY2F0Y2gge1xuICAgICAgICByZXR1cm4gdW5kZWZpbmVkXG4gICAgfVxufVxuXG5hc3luYyBmdW5jdGlvbiBpc0NsZWFuKHBhdGg6IHN0cmluZykge1xuICAgIGNvbnN0IGNoYW5nZXMgPSBhd2FpdCBleGVjQXN5bmMocGF0aCwgJ2dpdCBzdGF0dXMgLS1zaG9ydCcpXG4gICAgcmV0dXJuIGNoYW5nZXMubGVuZ3RoID09PSAwXG59XG5cbmFzeW5jIGZ1bmN0aW9uIGdldENvbW1pdHRlZEhhc2gocGF0aDogc3RyaW5nKSB7XG4gICAgY29uc3QgW2xvbmddID0gYXdhaXQgZXhlY0FzeW5jKHBhdGgsICdnaXQgcmV2LXBhcnNlIEhFQUQnKVxuICAgIHJldHVybiBsb25nXG59XG5cbmZ1bmN0aW9uIGV4ZWNBc3luYyhwYXRoOiBzdHJpbmcsIGNtZDogc3RyaW5nKSB7XG4gICAgcmV0dXJuIG5ldyBQcm9taXNlPHN0cmluZ1tdPigocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgICAgIGV4ZWMoY21kLCB7IGN3ZDogcGF0aCB9LCAoZXJyLCBzdGRvdXQpID0+IHtcbiAgICAgICAgICAgIGlmIChlcnIpIHtcbiAgICAgICAgICAgICAgICByZWplY3QoZXJyKVxuICAgICAgICAgICAgICAgIHJldHVyblxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgcmVzb2x2ZShzdGRvdXQuc3BsaXQoJ1xcbicpLnNsaWNlKDAsIC0xKSlcbiAgICAgICAgfSlcbiAgICB9KVxufVxuIl19
@@ -0,0 +1,12 @@
1
+ import { type HandlerConfiguration, type Handler as HttpFunction } from '../http.js';
2
+ import { type FullConfiguration, type Metadata } from './meta.js';
3
+ export type HttpHandler = {
4
+ meta: Metadata | undefined;
5
+ config: FullConfiguration | undefined;
6
+ method: Method;
7
+ pathPattern: string;
8
+ entry: HttpFunction;
9
+ };
10
+ export declare function pathToRegExp(path: string): RegExp;
11
+ export type Method = 'GET' | 'POST' | 'PUT' | 'PATCH' | 'DELETE';
12
+ export declare function registerHttpHandler(method: Method, path: string, configOrHandler: HandlerConfiguration | HttpFunction, fn?: HttpFunction): void;
@@ -0,0 +1,33 @@
1
+ import { combineConfig, getMetadata } from './meta.js';
2
+ import { addHandler } from './registry.js';
3
+ let httpHostRegistry;
4
+ function setHttpHost(host) {
5
+ httpHostRegistry = host;
6
+ }
7
+ export function pathToRegExp(path) {
8
+ return new RegExp(('^' +
9
+ path.replaceAll(/[/\\^$+?.()|[\]{}]/gu, '\\$&').replaceAll('*', '[^/\\?]+') +
10
+ '(\\?.*)?$').replace('[^/\\?]+[^/\\?]+(\\?.*)?$', ''), 'u');
11
+ }
12
+ function httpHost(meta, cfg, method, path, entry) {
13
+ addHandler('http', {
14
+ meta,
15
+ config: combineConfig(meta?.config, cfg),
16
+ method,
17
+ pathPattern: path,
18
+ entry,
19
+ });
20
+ }
21
+ setHttpHost(httpHost);
22
+ export function registerHttpHandler(method, path, configOrHandler, fn) {
23
+ if (typeof configOrHandler === 'function') {
24
+ httpHostRegistry(getMetadata(), undefined, method, path, configOrHandler);
25
+ }
26
+ else {
27
+ if (!fn) {
28
+ throw new Error('Please provide a handler function.');
29
+ }
30
+ httpHostRegistry(getMetadata(), configOrHandler, method, path, fn);
31
+ }
32
+ }
33
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaHR0cC1yZWdpc3RyeS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImh0dHAtcmVnaXN0cnkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLGFBQWEsRUFBRSxXQUFXLEVBQXlDLE1BQU0sV0FBVyxDQUFBO0FBQzdGLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUE7QUFrQjFDLElBQUksZ0JBQTBCLENBQUE7QUFFOUIsU0FBUyxXQUFXLENBQUMsSUFBYztJQUMvQixnQkFBZ0IsR0FBRyxJQUFJLENBQUE7QUFDM0IsQ0FBQztBQUVELE1BQU0sVUFBVSxZQUFZLENBQUMsSUFBWTtJQUNyQyxPQUFPLElBQUksTUFBTSxDQUNiLENBQ0ksR0FBRztRQUNILElBQUksQ0FBQyxVQUFVLENBQUMsc0JBQXNCLEVBQUUsTUFBTSxDQUFDLENBQUMsVUFBVSxDQUFDLEdBQUcsRUFBRSxVQUFVLENBQUM7UUFDM0UsV0FBVyxDQUNkLENBQUMsT0FBTyxDQUFDLDJCQUEyQixFQUFFLEVBQUUsQ0FBQyxFQUMxQyxHQUFHLENBQ04sQ0FBQTtBQUNMLENBQUM7QUFFRCxTQUFTLFFBQVEsQ0FDYixJQUEwQixFQUMxQixHQUFxQyxFQUNyQyxNQUFjLEVBQ2QsSUFBWSxFQUNaLEtBQW1CO0lBRW5CLFVBQVUsQ0FBQyxNQUFNLEVBQUU7UUFDZixJQUFJO1FBQ0osTUFBTSxFQUFFLGFBQWEsQ0FBQyxJQUFJLEVBQUUsTUFBTSxFQUFFLEdBQUcsQ0FBQztRQUN4QyxNQUFNO1FBQ04sV0FBVyxFQUFFLElBQUk7UUFDakIsS0FBSztLQUNSLENBQUMsQ0FBQTtBQUNOLENBQUM7QUFFRCxXQUFXLENBQUMsUUFBUSxDQUFDLENBQUE7QUFJckIsTUFBTSxVQUFVLG1CQUFtQixDQUMvQixNQUFjLEVBQ2QsSUFBWSxFQUNaLGVBQW9ELEVBQ3BELEVBQWlCO0lBRWpCLElBQUksT0FBTyxlQUFlLEtBQUssVUFBVSxFQUFFLENBQUM7UUFDeEMsZ0JBQWdCLENBQUMsV0FBVyxFQUFFLEVBQUUsU0FBUyxFQUFFLE1BQU0sRUFBRSxJQUFJLEVBQUUsZUFBZSxDQUFDLENBQUE7SUFDN0UsQ0FBQztTQUFNLENBQUM7UUFDSixJQUFJLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDTixNQUFNLElBQUksS0FBSyxDQUFDLG9DQUFvQyxDQUFDLENBQUE7UUFDekQsQ0FBQztRQUNELGdCQUFnQixDQUFDLFdBQVcsRUFBRSxFQUFFLGVBQWUsRUFBRSxNQUFNLEVBQUUsSUFBSSxFQUFFLEVBQUUsQ0FBQyxDQUFBO0lBQ3RFLENBQUM7QUFDTCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgdHlwZSBIYW5kbGVyQ29uZmlndXJhdGlvbiwgdHlwZSBIYW5kbGVyIGFzIEh0dHBGdW5jdGlvbiB9IGZyb20gJy4uL2h0dHAuanMnXG5pbXBvcnQgeyBjb21iaW5lQ29uZmlnLCBnZXRNZXRhZGF0YSwgdHlwZSBGdWxsQ29uZmlndXJhdGlvbiwgdHlwZSBNZXRhZGF0YSB9IGZyb20gJy4vbWV0YS5qcydcbmltcG9ydCB7IGFkZEhhbmRsZXIgfSBmcm9tICcuL3JlZ2lzdHJ5LmpzJ1xuXG5leHBvcnQgdHlwZSBIdHRwSGFuZGxlciA9IHtcbiAgICBtZXRhOiBNZXRhZGF0YSB8IHVuZGVmaW5lZFxuICAgIGNvbmZpZzogRnVsbENvbmZpZ3VyYXRpb24gfCB1bmRlZmluZWRcbiAgICBtZXRob2Q6IE1ldGhvZFxuICAgIHBhdGhQYXR0ZXJuOiBzdHJpbmdcbiAgICBlbnRyeTogSHR0cEZ1bmN0aW9uXG59XG5cbnR5cGUgSHR0cEhvc3QgPSAoXG4gICAgbWV0YTogTWV0YWRhdGEgfCB1bmRlZmluZWQsXG4gICAgY29uZmlnOiBIYW5kbGVyQ29uZmlndXJhdGlvbiB8IHVuZGVmaW5lZCxcbiAgICBtZXRob2Q6IE1ldGhvZCxcbiAgICBwYXRoOiBzdHJpbmcsXG4gICAgaGFuZGxlcjogSHR0cEZ1bmN0aW9uLFxuKSA9PiB2b2lkXG5cbmxldCBodHRwSG9zdFJlZ2lzdHJ5OiBIdHRwSG9zdFxuXG5mdW5jdGlvbiBzZXRIdHRwSG9zdChob3N0OiBIdHRwSG9zdCkge1xuICAgIGh0dHBIb3N0UmVnaXN0cnkgPSBob3N0XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBwYXRoVG9SZWdFeHAocGF0aDogc3RyaW5nKSB7XG4gICAgcmV0dXJuIG5ldyBSZWdFeHAoXG4gICAgICAgIChcbiAgICAgICAgICAgICdeJyArXG4gICAgICAgICAgICBwYXRoLnJlcGxhY2VBbGwoL1svXFxcXF4kKz8uKCl8W1xcXXt9XS9ndSwgJ1xcXFwkJicpLnJlcGxhY2VBbGwoJyonLCAnW14vXFxcXD9dKycpICtcbiAgICAgICAgICAgICcoXFxcXD8uKik/JCdcbiAgICAgICAgKS5yZXBsYWNlKCdbXi9cXFxcP10rW14vXFxcXD9dKyhcXFxcPy4qKT8kJywgJycpLFxuICAgICAgICAndScsXG4gICAgKVxufVxuXG5mdW5jdGlvbiBodHRwSG9zdChcbiAgICBtZXRhOiBNZXRhZGF0YSB8IHVuZGVmaW5lZCxcbiAgICBjZmc6IEhhbmRsZXJDb25maWd1cmF0aW9uIHwgdW5kZWZpbmVkLFxuICAgIG1ldGhvZDogTWV0aG9kLFxuICAgIHBhdGg6IHN0cmluZyxcbiAgICBlbnRyeTogSHR0cEZ1bmN0aW9uLFxuKSB7XG4gICAgYWRkSGFuZGxlcignaHR0cCcsIHtcbiAgICAgICAgbWV0YSxcbiAgICAgICAgY29uZmlnOiBjb21iaW5lQ29uZmlnKG1ldGE/LmNvbmZpZywgY2ZnKSxcbiAgICAgICAgbWV0aG9kLFxuICAgICAgICBwYXRoUGF0dGVybjogcGF0aCxcbiAgICAgICAgZW50cnksXG4gICAgfSlcbn1cblxuc2V0SHR0cEhvc3QoaHR0cEhvc3QpXG5cbmV4cG9ydCB0eXBlIE1ldGhvZCA9ICdHRVQnIHwgJ1BPU1QnIHwgJ1BVVCcgfCAnUEFUQ0gnIHwgJ0RFTEVURSdcblxuZXhwb3J0IGZ1bmN0aW9uIHJlZ2lzdGVySHR0cEhhbmRsZXIoXG4gICAgbWV0aG9kOiBNZXRob2QsXG4gICAgcGF0aDogc3RyaW5nLFxuICAgIGNvbmZpZ09ySGFuZGxlcjogSGFuZGxlckNvbmZpZ3VyYXRpb24gfCBIdHRwRnVuY3Rpb24sXG4gICAgZm4/OiBIdHRwRnVuY3Rpb24sXG4pOiB2b2lkIHtcbiAgICBpZiAodHlwZW9mIGNvbmZpZ09ySGFuZGxlciA9PT0gJ2Z1bmN0aW9uJykge1xuICAgICAgICBodHRwSG9zdFJlZ2lzdHJ5KGdldE1ldGFkYXRhKCksIHVuZGVmaW5lZCwgbWV0aG9kLCBwYXRoLCBjb25maWdPckhhbmRsZXIpXG4gICAgfSBlbHNlIHtcbiAgICAgICAgaWYgKCFmbikge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdQbGVhc2UgcHJvdmlkZSBhIGhhbmRsZXIgZnVuY3Rpb24uJylcbiAgICAgICAgfVxuICAgICAgICBodHRwSG9zdFJlZ2lzdHJ5KGdldE1ldGFkYXRhKCksIGNvbmZpZ09ySGFuZGxlciwgbWV0aG9kLCBwYXRoLCBmbilcbiAgICB9XG59XG4iXX0=
package/host/http.d.ts CHANGED
@@ -1,7 +1,7 @@
1
1
  import { Context } from '../context.js';
2
2
  import type { Json } from '../http.js';
3
3
  import { ClientInfo, RootLogger } from './context.js';
4
- import type { HttpHandler } from './registry.js';
4
+ import type { HttpHandler } from './http-registry.js';
5
5
  export type Response = {
6
6
  headers: {
7
7
  readonly [key: string]: string;
package/host/http.js CHANGED
@@ -1,3 +1,5 @@
1
+ import { hash } from 'node:crypto';
2
+ import { brotliCompress } from 'node:zlib';
1
3
  import { measure } from '../context.js';
2
4
  export async function executeRequest(log, context, handler, options, success) {
3
5
  const isShallow = context.env.SHALLOW_KEY && options.headers?.['x-shallow'] === context.env.SHALLOW_KEY;
@@ -83,7 +85,7 @@ export async function executeRequest(log, context, handler, options, success) {
83
85
  else {
84
86
  log.warn('Request END');
85
87
  }
86
- return response;
88
+ return await compressed(req.headers, eTagged(req.headers, response));
87
89
  }
88
90
  catch (e) {
89
91
  try {
@@ -181,9 +183,7 @@ function withContentType(headers, contentType) {
181
183
  'content-type': contentType,
182
184
  };
183
185
  }
184
- if (!headers['content-type']) {
185
- headers['content-type'] = contentType;
186
- }
186
+ headers['content-type'] ??= contentType;
187
187
  return headers;
188
188
  }
189
189
  function errorToResponse(e) {
@@ -217,14 +217,56 @@ export function clientFromHeaders(headers) {
217
217
  if (!headers) {
218
218
  return {};
219
219
  }
220
+ const address = headers['x-forwarded-for']?.split(':');
220
221
  return {
221
222
  operationId: headers['x-request-id'] ?? headers['request-id'],
222
223
  clientId: headers['x-client-id'] ??
223
224
  headers['x-installation-id'] ??
224
225
  headers['client-id'] ??
225
226
  headers['installation-id'],
226
- clientIp: headers['x-forwarded-for'],
227
+ clientIp: address?.[0],
228
+ clientPort: Number(address?.[1]) || undefined,
227
229
  userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],
228
230
  };
229
231
  }
230
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http.js","sourceRoot":"","sources":["http.ts"],"names":[],"mappings":"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","sourcesContent":["import { Context, measure } from '../context.js'\nimport type { Json, ParsedUrl, 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:\n            | (ParsedUrl & {\n                  /** @ignore */\n                  __proto__: unknown\n                  /** @ignore */\n                  toString: () => string\n              })\n            | undefined\n        let pathSteps: string[] | undefined\n        const req = {\n            rawUrl: options.uri,\n            get url() {\n                if (parsedUrl) {\n                    return parsedUrl\n                }\n                const url = new URL(this.rawUrl)\n                parsedUrl = {\n                    // eslint-disable-next-line unicorn/no-null\n                    __proto__: null,\n                    hash: url.hash,\n                    host: url.host,\n                    hostname: url.hostname,\n                    href: url.href,\n                    origin: url.origin,\n                    password: url.password,\n                    pathname: url.pathname,\n                    port: url.port,\n                    protocol: url.protocol,\n                    search: url.search,\n                    get searchParams() {\n                        return url.searchParams\n                    },\n                    toJSON: () => url.toJSON(),\n                    toString: () => url.toString(),\n                    username: url.username,\n                    pathStepAt: (index: number) => {\n                        const steps = (pathSteps ??= url.pathname.split('/'))\n                        const step = steps[index + 1]\n                        if (!step) {\n                            throw Object.assign(\n                                new RangeError(`Path does not have a step at index ${index}.`),\n                                {\n                                    rawUrl: this.rawUrl,\n                                    pathName: url.pathname,\n                                    steps: steps.slice(1),\n                                },\n                            )\n                        }\n                        return step\n                    },\n                }\n                return parsedUrl\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 ('json' in options) {\n        const { json, ...bodyless } = options\n        return bodyless\n    }\n    if ('body' in options) {\n        const { body, ...bodyless } = options\n        return bodyless\n    }\n    return options\n}\n\nfunction requestBody(options: RequestOptions): Json | string | undefined {\n    if ('json' in options) {\n        return options.json\n    }\n    if ('body' in options) {\n        return options.body\n    }\n    return 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"]}
232
+ function eTagged(requestHeaders, response) {
233
+ if (response.headers.etag || !response.body) {
234
+ return response;
235
+ }
236
+ const etag = hash('sha1', response.body, 'base64').slice(0, -1);
237
+ response.headers.etag = etag;
238
+ if (requestHeaders['if-none-match'] === etag) {
239
+ response.status = 304;
240
+ delete response.body;
241
+ }
242
+ return response;
243
+ }
244
+ async function compressed(requestHeaders, response) {
245
+ if (!response.body || response.body.length < 32_768 || response.headers['content-encoding']) {
246
+ return response;
247
+ }
248
+ const encodings = requestHeaders['accept-encoding']?.split(',').map(e => e.trim());
249
+ if (!encodings?.includes('br')) {
250
+ return response;
251
+ }
252
+ return {
253
+ status: response.status,
254
+ headers: {
255
+ 'content-encoding': 'br',
256
+ ...response.headers,
257
+ },
258
+ body: await compress(response.body),
259
+ };
260
+ }
261
+ function compress(body) {
262
+ return new Promise((resolve, reject) => {
263
+ brotliCompress(body, {}, (error, result) => {
264
+ if (error) {
265
+ reject(error);
266
+ return;
267
+ }
268
+ resolve(result);
269
+ });
270
+ });
271
+ }
272
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http.js","sourceRoot":"","sources":["http.ts"],"names":[],"mappings":"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","sourcesContent":["import { hash } from 'node:crypto'\nimport { brotliCompress } from 'node:zlib'\nimport { Context, measure } from '../context.js'\nimport type { Json, ParsedUrl, ResponseHeaders, Result } from '../http.js'\nimport { ClientInfo, RootLogger } from './context.js'\nimport type { HttpHandler } from './http-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:\n            | (ParsedUrl & {\n                  /** @ignore */\n                  __proto__: unknown\n                  /** @ignore */\n                  toString: () => string\n              })\n            | undefined\n        let pathSteps: string[] | undefined\n        const req = {\n            rawUrl: options.uri,\n            get url() {\n                if (parsedUrl) {\n                    return parsedUrl\n                }\n                const url = new URL(this.rawUrl)\n                parsedUrl = {\n                    // eslint-disable-next-line unicorn/no-null\n                    __proto__: null,\n                    hash: url.hash,\n                    host: url.host,\n                    hostname: url.hostname,\n                    href: url.href,\n                    origin: url.origin,\n                    password: url.password,\n                    pathname: url.pathname,\n                    port: url.port,\n                    protocol: url.protocol,\n                    search: url.search,\n                    get searchParams() {\n                        return url.searchParams\n                    },\n                    toJSON: () => url.toJSON(),\n                    toString: () => url.toString(),\n                    username: url.username,\n                    pathStepAt: (index: number) => {\n                        const steps = (pathSteps ??= url.pathname.split('/'))\n                        const step = steps[index + 1]\n                        if (!step) {\n                            throw Object.assign(\n                                new RangeError(`Path does not have a step at index ${index}.`),\n                                {\n                                    rawUrl: this.rawUrl,\n                                    pathName: url.pathname,\n                                    steps: steps.slice(1),\n                                },\n                            )\n                        }\n                        return step\n                    },\n                }\n                return parsedUrl\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 await compressed(req.headers, eTagged(req.headers, 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 ('json' in options) {\n        const { json, ...bodyless } = options\n        return bodyless\n    }\n    if ('body' in options) {\n        const { body, ...bodyless } = options\n        return bodyless\n    }\n    return options\n}\n\nfunction requestBody(options: RequestOptions): Json | string | undefined {\n    if ('json' in options) {\n        return options.json\n    }\n    if ('body' in options) {\n        return options.body\n    }\n    return undefined\n}\n\nfunction withContentType(headers: ResponseHeaders | undefined, contentType: string) {\n    if (!headers) {\n        return {\n            'content-type': contentType,\n        }\n    }\n    headers['content-type'] ??= contentType\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    const address = headers['x-forwarded-for']?.split(':')\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: address?.[0],\n        clientPort: Number(address?.[1]) || undefined,\n        userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],\n    }\n}\n\nfunction eTagged(\n    requestHeaders: { [key: string]: string },\n    response: Response & { headers: { [key: string]: string } },\n): Response {\n    if (response.headers.etag || !response.body) {\n        return response\n    }\n    const etag = hash('sha1', response.body, 'base64').slice(0, -1)\n    response.headers.etag = etag\n    if (requestHeaders['if-none-match'] === etag) {\n        response.status = 304\n        delete response.body\n    }\n    return response\n}\n\nasync function compressed(\n    requestHeaders: { [key: string]: string },\n    response: Response & { headers: { [key: string]: string } },\n): Promise<Response> {\n    if (!response.body || response.body.length < 32_768 || response.headers['content-encoding']) {\n        return response\n    }\n    const encodings = requestHeaders['accept-encoding']?.split(',').map(e => e.trim())\n    if (!encodings?.includes('br')) {\n        return response\n    }\n    return {\n        status: response.status,\n        headers: {\n            'content-encoding': 'br',\n            ...response.headers,\n        },\n        body: await compress(response.body),\n    }\n}\n\nfunction compress(body: string | Buffer) {\n    return new Promise<Buffer>((resolve, reject) => {\n        brotliCompress(body, {}, (error, result) => {\n            if (error) {\n                reject(error)\n                return\n            }\n            resolve(result)\n        })\n    })\n}\n"]}