@riddance/host 0.1.2 → 0.2.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 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<boolean>;
package/host/event.js ADDED
@@ -0,0 +1,17 @@
1
+ import { randomUUID } from 'node:crypto';
2
+ import { measure } from '../context.js';
3
+ export async function handle(log, context, handler, options, success) {
4
+ const enrichedLog = log.enrichReserved({ meta: context.meta, event: options });
5
+ enrichedLog.trace('Event BEGIN');
6
+ try {
7
+ await measure(log.enrichReserved({ meta: context.meta }), 'execution', () => handler.entry({ ...context, log: enrichedLog }, options.subject, options.event, options.timestamp, options.messageId ?? randomUUID().replaceAll('-', '')));
8
+ enrichedLog.debug('Event END');
9
+ await success();
10
+ return true;
11
+ }
12
+ catch (e) {
13
+ enrichedLog.error('Event END', e);
14
+ return false;
15
+ }
16
+ }
17
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXZlbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJldmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sYUFBYSxDQUFBO0FBQ3hDLE9BQU8sRUFBVyxPQUFPLEVBQWEsTUFBTSxlQUFlLENBQUE7QUFJM0QsTUFBTSxDQUFDLEtBQUssVUFBVSxNQUFNLENBQ3hCLEdBQWUsRUFDZixPQUE2QixFQUM3QixPQUFxQixFQUNyQixPQVNDLEVBQ0QsT0FBK0I7SUFFL0IsTUFBTSxXQUFXLEdBQUcsR0FBRyxDQUFDLGNBQWMsQ0FBQyxFQUFFLElBQUksRUFBRSxPQUFPLENBQUMsSUFBSSxFQUFFLEtBQUssRUFBRSxPQUFPLEVBQUUsQ0FBQyxDQUFBO0lBQzlFLFdBQVcsQ0FBQyxLQUFLLENBQUMsYUFBYSxDQUFDLENBQUE7SUFDaEMsSUFBSSxDQUFDO1FBQ0QsTUFBTSxPQUFPLENBQUMsR0FBRyxDQUFDLGNBQWMsQ0FBQyxFQUFFLElBQUksRUFBRSxPQUFPLENBQUMsSUFBSSxFQUFFLENBQUMsRUFBRSxXQUFXLEVBQUUsR0FBRyxFQUFFLENBQ3hFLE9BQU8sQ0FBQyxLQUFLLENBQ1QsRUFBRSxHQUFHLE9BQU8sRUFBRSxHQUFHLEVBQUUsV0FBVyxFQUFFLEVBQ2hDLE9BQU8sQ0FBQyxPQUFPLEVBQ2YsT0FBTyxDQUFDLEtBQUssRUFDYixPQUFPLENBQUMsU0FBUyxFQUNqQixPQUFPLENBQUMsU0FBUyxJQUFJLFVBQVUsRUFBRSxDQUFDLFVBQVUsQ0FBQyxHQUFHLEVBQUUsRUFBRSxDQUFDLENBQ3hELENBQ0osQ0FBQTtRQUNELFdBQVcsQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLENBQUE7UUFDOUIsTUFBTSxPQUFPLEVBQUUsQ0FBQTtRQUNmLE9BQU8sSUFBSSxDQUFBO0lBQ2YsQ0FBQztJQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7UUFDVCxXQUFXLENBQUMsS0FBSyxDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUMsQ0FBQTtRQUNqQyxPQUFPLEtBQUssQ0FBQTtJQUNoQixDQUFDO0FBQ0wsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHJhbmRvbVVVSUQgfSBmcm9tICdub2RlOmNyeXB0bydcbmltcG9ydCB7IENvbnRleHQsIG1lYXN1cmUsIHR5cGUgSnNvbiB9IGZyb20gJy4uL2NvbnRleHQuanMnXG5pbXBvcnQgeyBSb290TG9nZ2VyIH0gZnJvbSAnLi9jb250ZXh0LmpzJ1xuaW1wb3J0IHR5cGUgeyBFdmVudEhhbmRsZXIgfSBmcm9tICcuL2V2ZW50LXJlZ2lzdHJ5LmpzJ1xuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gaGFuZGxlKFxuICAgIGxvZzogUm9vdExvZ2dlcixcbiAgICBjb250ZXh0OiBPbWl0PENvbnRleHQsICdsb2cnPixcbiAgICBoYW5kbGVyOiBFdmVudEhhbmRsZXIsXG4gICAgb3B0aW9uczoge1xuICAgICAgICByZWFkb25seSBzdWJqZWN0OiBzdHJpbmdcbiAgICAgICAgcmVhZG9ubHkgZXZlbnQ6XG4gICAgICAgICAgICB8IHtcbiAgICAgICAgICAgICAgICAgIHJlYWRvbmx5IFtrZXk6IHN0cmluZ106IEpzb25cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfCB1bmRlZmluZWRcbiAgICAgICAgcmVhZG9ubHkgdGltZXN0YW1wOiBEYXRlXG4gICAgICAgIHJlYWRvbmx5IG1lc3NhZ2VJZD86IHN0cmluZ1xuICAgIH0sXG4gICAgc3VjY2VzczogKCkgPT4gUHJvbWlzZTx1bmtub3duPixcbik6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGNvbnN0IGVucmljaGVkTG9nID0gbG9nLmVucmljaFJlc2VydmVkKHsgbWV0YTogY29udGV4dC5tZXRhLCBldmVudDogb3B0aW9ucyB9KVxuICAgIGVucmljaGVkTG9nLnRyYWNlKCdFdmVudCBCRUdJTicpXG4gICAgdHJ5IHtcbiAgICAgICAgYXdhaXQgbWVhc3VyZShsb2cuZW5yaWNoUmVzZXJ2ZWQoeyBtZXRhOiBjb250ZXh0Lm1ldGEgfSksICdleGVjdXRpb24nLCAoKSA9PlxuICAgICAgICAgICAgaGFuZGxlci5lbnRyeShcbiAgICAgICAgICAgICAgICB7IC4uLmNvbnRleHQsIGxvZzogZW5yaWNoZWRMb2cgfSxcbiAgICAgICAgICAgICAgICBvcHRpb25zLnN1YmplY3QsXG4gICAgICAgICAgICAgICAgb3B0aW9ucy5ldmVudCxcbiAgICAgICAgICAgICAgICBvcHRpb25zLnRpbWVzdGFtcCxcbiAgICAgICAgICAgICAgICBvcHRpb25zLm1lc3NhZ2VJZCA/PyByYW5kb21VVUlEKCkucmVwbGFjZUFsbCgnLScsICcnKSxcbiAgICAgICAgICAgICksXG4gICAgICAgIClcbiAgICAgICAgZW5yaWNoZWRMb2cuZGVidWcoJ0V2ZW50IEVORCcpXG4gICAgICAgIGF3YWl0IHN1Y2Nlc3MoKVxuICAgICAgICByZXR1cm4gdHJ1ZVxuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgZW5yaWNoZWRMb2cuZXJyb3IoJ0V2ZW50IEVORCcsIGUpXG4gICAgICAgIHJldHVybiBmYWxzZVxuICAgIH1cbn1cbiJdfQ==
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;