@riddance/host 0.2.4 → 0.2.6

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/context.d.ts CHANGED
@@ -2,33 +2,34 @@ export type Environment = {
2
2
  readonly [key: string]: string;
3
3
  };
4
4
  export type Logger = {
5
- enrich(fields: object): void;
6
- trace(message: string, error?: unknown, fields?: object): void;
7
- debug(message: string, error?: unknown, fields?: object): void;
8
- info(message: string, error?: unknown, fields?: object): void;
9
- warn(message: string, error?: unknown, fields?: object): void;
10
- error(message: string, error?: unknown, fields?: object): void;
11
- fatal(message: string, error?: unknown, fields?: object): void;
5
+ enrich(fields: JsonSafeObject): void;
6
+ trace(message: string, error?: unknown, fields?: JsonSafeObject): void;
7
+ debug(message: string, error?: unknown, fields?: JsonSafeObject): void;
8
+ info(message: string, error?: unknown, fields?: JsonSafeObject): void;
9
+ warn(message: string, error?: unknown, fields?: JsonSafeObject): void;
10
+ error(message: string, error?: unknown, fields?: JsonSafeObject): void;
11
+ fatal(message: string, error?: unknown, fields?: JsonSafeObject): void;
12
12
  };
13
13
  export type MutableJson = null | boolean | number | string | MutableJson[] | {
14
14
  [key: string]: MutableJson;
15
15
  };
16
- export type Json = null | boolean | number | string | readonly Json[] | {
16
+ export type Json = null | boolean | number | string | readonly Json[] | JsonObject;
17
+ export type JsonObject = {
17
18
  readonly [key: string]: Json;
18
19
  };
19
- export declare function objectSpreadable(json?: Json): {
20
- readonly [key: string]: Json;
20
+ export type JsonSafe = undefined | null | boolean | number | string | {
21
+ toJSON: () => string;
22
+ } | readonly JsonSafe[] | JsonSafeObject;
23
+ export type JsonSafeObject = {
24
+ readonly [key: string]: JsonSafe;
21
25
  };
22
- export declare function arraySpreadable(json?: Json): readonly Json[];
23
26
  export type HandlerConfiguration = {
24
27
  /**
25
- * An indication of CPU usage of the handler.
26
- * @default 'low'
28
+ * An indication of CPU usage of the handler. If undefined, a generic conservative value will be used.
27
29
  */
28
30
  readonly compute?: 'high' | 'low';
29
31
  /**
30
- * An indication of memory usage of the handler.
31
- * @default 'low'
32
+ * An indication of memory usage of the handler. If undefined, a generic conservative value will be used.
32
33
  */
33
34
  readonly memory?: 'high' | 'low';
34
35
  /**
@@ -66,9 +67,9 @@ export type Context = {
66
67
  emit(topic: string, type: string, subject: string, data?: Json, messageId?: string): Promise<void>;
67
68
  onSuccess(fn: () => Promise<void> | void): void;
68
69
  };
69
- export declare function httpRequestHeaders(context: Context): {
70
+ export declare function httpRequestHeaders({ meta, operationId, client, }: Pick<Context, 'meta' | 'operationId' | 'client'>): {
70
71
  [key: string]: string;
71
72
  };
72
73
  export declare function measure<T>(logger: {
73
- trace: (message: string, _: undefined, f: object) => void;
74
- }, name: string, fn: () => Promise<T> | T, fields?: object): Promise<T>;
74
+ trace: (message: string, _: undefined, f: JsonSafeObject) => void;
75
+ }, name: string, fn: () => Promise<T> | T, fields?: JsonSafeObject): Promise<T>;
package/context.js CHANGED
@@ -1,39 +1,45 @@
1
+ import { isIPv4 } from 'node:net';
1
2
  import { performance } from 'node:perf_hooks';
2
3
  import { highPrecisionISODate } from './host/logging.js';
3
- /*@__INLINE__*/
4
- export function objectSpreadable(json) {
5
- if (!json) {
6
- return {};
7
- }
8
- return json;
9
- }
10
- /*@__INLINE__*/
11
- export function arraySpreadable(json) {
12
- if (!Array.isArray(json)) {
13
- return [];
14
- }
15
- return json;
16
- }
17
- export function httpRequestHeaders(context) {
4
+ export function httpRequestHeaders({ meta, operationId, client, }) {
18
5
  const headers = {
19
- 'user-agent': `${context.meta?.packageName ?? '?'}/${context.meta?.revision ?? '?'}`,
6
+ 'user-agent': `${meta?.packageName ?? '?'}/${meta?.revision ?? '?'}`,
20
7
  };
21
- if (context.operationId) {
22
- headers['x-request-id'] = context.operationId;
8
+ if (operationId) {
9
+ headers['x-request-id'] = operationId;
23
10
  }
24
- if (context.client) {
25
- if (context.client.id) {
26
- headers['x-client-id'] = context.client.id;
11
+ if (client) {
12
+ if (client.id) {
13
+ headers['x-client-id'] = client.id;
27
14
  }
28
- if (!!context.client.ip || !!context.client.port) {
29
- headers['x-forwarded-for'] = `${context.client.ip ?? ''}:${context.client.port ?? ''}`;
15
+ const { ip, port } = client;
16
+ if (!!ip || !!port) {
17
+ const xff = forwardedFor(ip, port);
18
+ if (xff) {
19
+ headers['x-forwarded-for'] = xff;
20
+ }
30
21
  }
31
- if (context.client.userAgent) {
32
- headers['x-forwarded-for-user-agent'] = context.client.userAgent;
22
+ if (client.userAgent) {
23
+ headers['x-forwarded-for-user-agent'] = client.userAgent;
33
24
  }
34
25
  }
35
26
  return headers;
36
27
  }
28
+ function forwardedFor(ip, port) {
29
+ if (!port) {
30
+ if (ip) {
31
+ return ip;
32
+ }
33
+ return undefined;
34
+ }
35
+ if (!ip) {
36
+ return `:${port}`;
37
+ }
38
+ if (isIPv4(ip)) {
39
+ return `${ip}:${port}`;
40
+ }
41
+ return `[${ip}]:${port}`;
42
+ }
37
43
  export async function measure(logger, name, fn, fields) {
38
44
  const start = performance.now();
39
45
  try {
@@ -49,4 +55,4 @@ export async function measure(logger, name, fn, fields) {
49
55
  });
50
56
  }
51
57
  }
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): void\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"]}
58
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,UAAU,CAAA;AACjC,OAAO,EAAE,WAAW,EAAE,MAAM,iBAAiB,CAAA;AAC7C,OAAO,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAA;AA6FxD,MAAM,UAAU,kBAAkB,CAAC,EAC/B,IAAI,EACJ,WAAW,EACX,MAAM,GACyC;IAC/C,MAAM,OAAO,GAA8B;QACvC,YAAY,EAAE,GAAG,IAAI,EAAE,WAAW,IAAI,GAAG,IAAI,IAAI,EAAE,QAAQ,IAAI,GAAG,EAAE;KACvE,CAAA;IACD,IAAI,WAAW,EAAE,CAAC;QACd,OAAO,CAAC,cAAc,CAAC,GAAG,WAAW,CAAA;IACzC,CAAC;IACD,IAAI,MAAM,EAAE,CAAC;QACT,IAAI,MAAM,CAAC,EAAE,EAAE,CAAC;YACZ,OAAO,CAAC,aAAa,CAAC,GAAG,MAAM,CAAC,EAAE,CAAA;QACtC,CAAC;QACD,MAAM,EAAE,EAAE,EAAE,IAAI,EAAE,GAAG,MAAM,CAAA;QAC3B,IAAI,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC;YACjB,MAAM,GAAG,GAAG,YAAY,CAAC,EAAE,EAAE,IAAI,CAAC,CAAA;YAClC,IAAI,GAAG,EAAE,CAAC;gBACN,OAAO,CAAC,iBAAiB,CAAC,GAAG,GAAG,CAAA;YACpC,CAAC;QACL,CAAC;QACD,IAAI,MAAM,CAAC,SAAS,EAAE,CAAC;YACnB,OAAO,CAAC,4BAA4B,CAAC,GAAG,MAAM,CAAC,SAAS,CAAA;QAC5D,CAAC;IACL,CAAC;IACD,OAAO,OAAO,CAAA;AAClB,CAAC;AAED,SAAS,YAAY,CAAC,EAAsB,EAAE,IAAwB;IAClE,IAAI,CAAC,IAAI,EAAE,CAAC;QACR,IAAI,EAAE,EAAE,CAAC;YACL,OAAO,EAAE,CAAA;QACb,CAAC;QACD,OAAO,SAAS,CAAA;IACpB,CAAC;IACD,IAAI,CAAC,EAAE,EAAE,CAAC;QACN,OAAO,IAAI,IAAI,EAAE,CAAA;IACrB,CAAC;IACD,IAAI,MAAM,CAAC,EAAE,CAAC,EAAE,CAAC;QACb,OAAO,GAAG,EAAE,IAAI,IAAI,EAAE,CAAA;IAC1B,CAAC;IACD,OAAO,IAAI,EAAE,KAAK,IAAI,EAAE,CAAA;AAC5B,CAAC;AAED,MAAM,CAAC,KAAK,UAAU,OAAO,CACzB,MAA6E,EAC7E,IAAY,EACZ,EAAwB,EACxB,MAAuB;IAEvB,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 { isIPv4 } from 'node:net'\nimport { 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: JsonSafeObject): void\n    trace(message: string, error?: unknown, fields?: JsonSafeObject): void\n    debug(message: string, error?: unknown, fields?: JsonSafeObject): void\n    info(message: string, error?: unknown, fields?: JsonSafeObject): void\n    warn(message: string, error?: unknown, fields?: JsonSafeObject): void\n    error(message: string, error?: unknown, fields?: JsonSafeObject): void\n    fatal(message: string, error?: unknown, fields?: JsonSafeObject): void\n}\n\nexport type MutableJson =\n    | null\n    | boolean\n    | number\n    | string\n    | MutableJson[]\n    | { [key: string]: MutableJson }\nexport type Json = null | boolean | number | string | readonly Json[] | JsonObject\nexport type JsonObject = { readonly [key: string]: Json }\n\nexport type JsonSafe =\n    | undefined\n    | null\n    | boolean\n    | number\n    | string\n    | { toJSON: () => string }\n    | readonly JsonSafe[]\n    | JsonSafeObject\n\nexport type JsonSafeObject = { readonly [key: string]: JsonSafe }\n\nexport type HandlerConfiguration = {\n    /**\n     * An indication of CPU usage of the handler. If undefined, a generic conservative value will be used.\n     */\n    readonly compute?: 'high' | 'low'\n    /**\n     * An indication of memory usage of the handler. If undefined, a generic conservative value will be used.\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({\n    meta,\n    operationId,\n    client,\n}: Pick<Context, 'meta' | 'operationId' | 'client'>) {\n    const headers: { [key: string]: string } = {\n        'user-agent': `${meta?.packageName ?? '?'}/${meta?.revision ?? '?'}`,\n    }\n    if (operationId) {\n        headers['x-request-id'] = operationId\n    }\n    if (client) {\n        if (client.id) {\n            headers['x-client-id'] = client.id\n        }\n        const { ip, port } = client\n        if (!!ip || !!port) {\n            const xff = forwardedFor(ip, port)\n            if (xff) {\n                headers['x-forwarded-for'] = xff\n            }\n        }\n        if (client.userAgent) {\n            headers['x-forwarded-for-user-agent'] = client.userAgent\n        }\n    }\n    return headers\n}\n\nfunction forwardedFor(ip: string | undefined, port: number | undefined) {\n    if (!port) {\n        if (ip) {\n            return ip\n        }\n        return undefined\n    }\n    if (!ip) {\n        return `:${port}`\n    }\n    if (isIPv4(ip)) {\n        return `${ip}:${port}`\n    }\n    return `[${ip}]:${port}`\n}\n\nexport async function measure<T>(\n    logger: { trace: (message: string, _: undefined, f: JsonSafeObject) => void },\n    name: string,\n    fn: () => Promise<T> | T,\n    fields?: JsonSafeObject,\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 CHANGED
@@ -1,8 +1,6 @@
1
- import { Context, HandlerConfiguration, type Json } from './context.js';
1
+ import { Context, HandlerConfiguration, type JsonObject } from './context.js';
2
2
  export * from './context.js';
3
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;
4
+ export type Handler = (context: Context, subject: string, event: JsonObject | undefined, timestamp: Date, messageId: string) => Promise<void> | void;
7
5
  export declare function on(topic: string, event: string, fn: Handler): void;
8
6
  export declare function on(topic: string, event: string, config: HandlerConfiguration, fn: Handler): void;
package/event.js CHANGED
@@ -3,4 +3,4 @@ export * from './context.js';
3
3
  export function on(topic, event, configOrHandler, fn) {
4
4
  registerEventHandler(topic, event, configOrHandler, fn);
5
5
  }
6
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXZlbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJldmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQTtBQUUvRCxjQUFjLGNBQWMsQ0FBQTtBQWtCNUIsTUFBTSxVQUFVLEVBQUUsQ0FDZCxLQUFhLEVBQ2IsS0FBYSxFQUNiLGVBQStDLEVBQy9DLEVBQVk7SUFFWixvQkFBb0IsQ0FBQyxLQUFLLEVBQUUsS0FBSyxFQUFFLGVBQWUsRUFBRSxFQUFFLENBQUMsQ0FBQTtBQUMzRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29udGV4dCwgSGFuZGxlckNvbmZpZ3VyYXRpb24sIHR5cGUgSnNvbiB9IGZyb20gJy4vY29udGV4dC5qcydcbmltcG9ydCB7IHJlZ2lzdGVyRXZlbnRIYW5kbGVyIH0gZnJvbSAnLi9ob3N0L2V2ZW50LXJlZ2lzdHJ5LmpzJ1xuXG5leHBvcnQgKiBmcm9tICcuL2NvbnRleHQuanMnXG5cbmV4cG9ydCB0eXBlIEV2ZW50SGFuZGxlckNvbmZpZ3VyYXRpb24gPSBIYW5kbGVyQ29uZmlndXJhdGlvbiAmIHt9XG5cbmV4cG9ydCB0eXBlIEhhbmRsZXIgPSAoXG4gICAgY29udGV4dDogQ29udGV4dCxcbiAgICBzdWJqZWN0OiBzdHJpbmcsXG4gICAgZXZlbnQ6XG4gICAgICAgIHwge1xuICAgICAgICAgICAgICByZWFkb25seSBba2V5OiBzdHJpbmddOiBKc29uXG4gICAgICAgICAgfVxuICAgICAgICB8IHVuZGVmaW5lZCxcbiAgICB0aW1lc3RhbXA6IERhdGUsXG4gICAgbWVzc2FnZUlkOiBzdHJpbmcsXG4pID0+IFByb21pc2U8dm9pZD4gfCB2b2lkXG5cbmV4cG9ydCBmdW5jdGlvbiBvbih0b3BpYzogc3RyaW5nLCBldmVudDogc3RyaW5nLCBmbjogSGFuZGxlcik6IHZvaWRcbmV4cG9ydCBmdW5jdGlvbiBvbih0b3BpYzogc3RyaW5nLCBldmVudDogc3RyaW5nLCBjb25maWc6IEhhbmRsZXJDb25maWd1cmF0aW9uLCBmbjogSGFuZGxlcik6IHZvaWRcbmV4cG9ydCBmdW5jdGlvbiBvbihcbiAgICB0b3BpYzogc3RyaW5nLFxuICAgIGV2ZW50OiBzdHJpbmcsXG4gICAgY29uZmlnT3JIYW5kbGVyOiBIYW5kbGVyQ29uZmlndXJhdGlvbiB8IEhhbmRsZXIsXG4gICAgZm4/OiBIYW5kbGVyLFxuKTogdm9pZCB7XG4gICAgcmVnaXN0ZXJFdmVudEhhbmRsZXIodG9waWMsIGV2ZW50LCBjb25maWdPckhhbmRsZXIsIGZuKVxufVxuIl19
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXZlbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJldmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQTtBQUUvRCxjQUFjLGNBQWMsQ0FBQTtBQWM1QixNQUFNLFVBQVUsRUFBRSxDQUNkLEtBQWEsRUFDYixLQUFhLEVBQ2IsZUFBK0MsRUFDL0MsRUFBWTtJQUVaLG9CQUFvQixDQUFDLEtBQUssRUFBRSxLQUFLLEVBQUUsZUFBZSxFQUFFLEVBQUUsQ0FBQyxDQUFBO0FBQzNELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb250ZXh0LCBIYW5kbGVyQ29uZmlndXJhdGlvbiwgdHlwZSBKc29uT2JqZWN0IH0gZnJvbSAnLi9jb250ZXh0LmpzJ1xuaW1wb3J0IHsgcmVnaXN0ZXJFdmVudEhhbmRsZXIgfSBmcm9tICcuL2hvc3QvZXZlbnQtcmVnaXN0cnkuanMnXG5cbmV4cG9ydCAqIGZyb20gJy4vY29udGV4dC5qcydcblxuZXhwb3J0IHR5cGUgRXZlbnRIYW5kbGVyQ29uZmlndXJhdGlvbiA9IEhhbmRsZXJDb25maWd1cmF0aW9uICYge31cblxuZXhwb3J0IHR5cGUgSGFuZGxlciA9IChcbiAgICBjb250ZXh0OiBDb250ZXh0LFxuICAgIHN1YmplY3Q6IHN0cmluZyxcbiAgICBldmVudDogSnNvbk9iamVjdCB8IHVuZGVmaW5lZCxcbiAgICB0aW1lc3RhbXA6IERhdGUsXG4gICAgbWVzc2FnZUlkOiBzdHJpbmcsXG4pID0+IFByb21pc2U8dm9pZD4gfCB2b2lkXG5cbmV4cG9ydCBmdW5jdGlvbiBvbih0b3BpYzogc3RyaW5nLCBldmVudDogc3RyaW5nLCBmbjogSGFuZGxlcik6IHZvaWRcbmV4cG9ydCBmdW5jdGlvbiBvbih0b3BpYzogc3RyaW5nLCBldmVudDogc3RyaW5nLCBjb25maWc6IEhhbmRsZXJDb25maWd1cmF0aW9uLCBmbjogSGFuZGxlcik6IHZvaWRcbmV4cG9ydCBmdW5jdGlvbiBvbihcbiAgICB0b3BpYzogc3RyaW5nLFxuICAgIGV2ZW50OiBzdHJpbmcsXG4gICAgY29uZmlnT3JIYW5kbGVyOiBIYW5kbGVyQ29uZmlndXJhdGlvbiB8IEhhbmRsZXIsXG4gICAgZm4/OiBIYW5kbGVyLFxuKTogdm9pZCB7XG4gICAgcmVnaXN0ZXJFdmVudEhhbmRsZXIodG9waWMsIGV2ZW50LCBjb25maWdPckhhbmRsZXIsIGZuKVxufVxuIl19
package/host/context.d.ts CHANGED
@@ -1,4 +1,4 @@
1
- import { Context, Environment, Json, Logger } from '../context.js';
1
+ import { Context, Environment, Logger, type JsonSafeObject } from '../context.js';
2
2
  import type { FullConfiguration, Metadata } from './meta.js';
3
3
  export type ClientInfo = {
4
4
  readonly operationId?: string;
@@ -8,9 +8,7 @@ export type ClientInfo = {
8
8
  readonly userAgent?: string;
9
9
  };
10
10
  export type EventTransport = {
11
- sendEvent(topic: string, type: string, subject: string, data: {
12
- readonly [key: string]: Json;
13
- } | undefined, messageId: string | undefined, signal: AbortSignal): Promise<void>;
11
+ sendEvent(topic: string, type: string, subject: string, data: JsonSafeObject | undefined, messageId: string | undefined, signal: AbortSignal): Promise<void>;
14
12
  };
15
13
  export type LogLevel = 'trace' | 'debug' | 'info' | 'warning' | 'error' | 'fatal';
16
14
  export type LogEntry = {
@@ -31,7 +29,7 @@ export type RootLogger = {
31
29
  export declare function createContext(clientInfo: ClientInfo, loggers: LogTransport[], eventTransport: EventTransport, timeouts: {
32
30
  default: number;
33
31
  cap?: number;
34
- }, outerController: AbortController, config?: FullConfiguration, meta?: Metadata, environment?: Environment, now?: () => Date): {
32
+ }, outerController: AbortController, config?: FullConfiguration, meta?: Metadata, environment?: Partial<Environment>, now?: () => Date): {
35
33
  log: RootLogger;
36
34
  context: Omit<Context, 'log'>;
37
35
  success: () => Promise<unknown>;
package/host/context.js CHANGED
@@ -33,7 +33,7 @@ export function createContext(clientInfo, loggers, eventTransport, timeouts, out
33
33
  globalLogger = logger;
34
34
  const successHandlers = [];
35
35
  const ctx = {
36
- env: environment ?? process.env,
36
+ env: (environment ?? process.env),
37
37
  signal: innerController.signal,
38
38
  now: now ?? (() => new Date()),
39
39
  operationId: clientInfo.operationId,
@@ -81,4 +81,4 @@ process.on('uncaughtException', err => {
81
81
  process.on('unhandledRejection', reason => {
82
82
  globalLogger?.fatal('Unhandled rejection.', reason, undefined);
83
83
  });
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()).filter(r => !!r)),\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"]}
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, type JsonSafeObject } 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: JsonSafeObject | 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?: Partial<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()).filter(r => !!r)),\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"]}
package/host/event.d.ts CHANGED
@@ -1,11 +1,9 @@
1
- import { Context, type Json } from '../context.js';
1
+ import { Context, type JsonObject } from '../context.js';
2
2
  import { RootLogger } from './context.js';
3
3
  import type { EventHandler } from './event-registry.js';
4
4
  export declare function handle(log: RootLogger, context: Omit<Context, 'log'>, handler: EventHandler, options: {
5
5
  readonly subject: string;
6
- readonly event: {
7
- readonly [key: string]: Json;
8
- } | undefined;
6
+ readonly event: JsonObject | undefined;
9
7
  readonly timestamp: Date;
10
8
  readonly messageId?: string;
11
9
  }, success: () => Promise<unknown>): Promise<boolean>;
package/host/event.js CHANGED
@@ -14,4 +14,4 @@ export async function handle(log, context, handler, options, success) {
14
14
  return false;
15
15
  }
16
16
  }
17
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXZlbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJldmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sYUFBYSxDQUFBO0FBQ3hDLE9BQU8sRUFBVyxPQUFPLEVBQWEsTUFBTSxlQUFlLENBQUE7QUFJM0QsTUFBTSxDQUFDLEtBQUssVUFBVSxNQUFNLENBQ3hCLEdBQWUsRUFDZixPQUE2QixFQUM3QixPQUFxQixFQUNyQixPQVNDLEVBQ0QsT0FBK0I7SUFFL0IsTUFBTSxXQUFXLEdBQUcsR0FBRyxDQUFDLGNBQWMsQ0FBQyxFQUFFLElBQUksRUFBRSxPQUFPLENBQUMsSUFBSSxFQUFFLEtBQUssRUFBRSxPQUFPLEVBQUUsQ0FBQyxDQUFBO0lBQzlFLFdBQVcsQ0FBQyxLQUFLLENBQUMsYUFBYSxDQUFDLENBQUE7SUFDaEMsSUFBSSxDQUFDO1FBQ0QsTUFBTSxPQUFPLENBQUMsR0FBRyxDQUFDLGNBQWMsQ0FBQyxFQUFFLElBQUksRUFBRSxPQUFPLENBQUMsSUFBSSxFQUFFLENBQUMsRUFBRSxXQUFXLEVBQUUsR0FBRyxFQUFFLENBQ3hFLE9BQU8sQ0FBQyxLQUFLLENBQ1QsRUFBRSxHQUFHLE9BQU8sRUFBRSxHQUFHLEVBQUUsV0FBVyxFQUFFLEVBQ2hDLE9BQU8sQ0FBQyxPQUFPLEVBQ2YsT0FBTyxDQUFDLEtBQUssRUFDYixPQUFPLENBQUMsU0FBUyxFQUNqQixPQUFPLENBQUMsU0FBUyxJQUFJLFVBQVUsRUFBRSxDQUFDLFVBQVUsQ0FBQyxHQUFHLEVBQUUsRUFBRSxDQUFDLENBQ3hELENBQ0osQ0FBQTtRQUNELFdBQVcsQ0FBQyxLQUFLLENBQUMsV0FBVyxDQUFDLENBQUE7UUFDOUIsTUFBTSxPQUFPLEVBQUUsQ0FBQTtRQUNmLE9BQU8sSUFBSSxDQUFBO0lBQ2YsQ0FBQztJQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7UUFDVCxXQUFXLENBQUMsS0FBSyxDQUFDLFdBQVcsRUFBRSxDQUFDLENBQUMsQ0FBQTtRQUNqQyxPQUFPLEtBQUssQ0FBQTtJQUNoQixDQUFDO0FBQ0wsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHJhbmRvbVVVSUQgfSBmcm9tICdub2RlOmNyeXB0bydcbmltcG9ydCB7IENvbnRleHQsIG1lYXN1cmUsIHR5cGUgSnNvbiB9IGZyb20gJy4uL2NvbnRleHQuanMnXG5pbXBvcnQgeyBSb290TG9nZ2VyIH0gZnJvbSAnLi9jb250ZXh0LmpzJ1xuaW1wb3J0IHR5cGUgeyBFdmVudEhhbmRsZXIgfSBmcm9tICcuL2V2ZW50LXJlZ2lzdHJ5LmpzJ1xuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gaGFuZGxlKFxuICAgIGxvZzogUm9vdExvZ2dlcixcbiAgICBjb250ZXh0OiBPbWl0PENvbnRleHQsICdsb2cnPixcbiAgICBoYW5kbGVyOiBFdmVudEhhbmRsZXIsXG4gICAgb3B0aW9uczoge1xuICAgICAgICByZWFkb25seSBzdWJqZWN0OiBzdHJpbmdcbiAgICAgICAgcmVhZG9ubHkgZXZlbnQ6XG4gICAgICAgICAgICB8IHtcbiAgICAgICAgICAgICAgICAgIHJlYWRvbmx5IFtrZXk6IHN0cmluZ106IEpzb25cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfCB1bmRlZmluZWRcbiAgICAgICAgcmVhZG9ubHkgdGltZXN0YW1wOiBEYXRlXG4gICAgICAgIHJlYWRvbmx5IG1lc3NhZ2VJZD86IHN0cmluZ1xuICAgIH0sXG4gICAgc3VjY2VzczogKCkgPT4gUHJvbWlzZTx1bmtub3duPixcbik6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAgIGNvbnN0IGVucmljaGVkTG9nID0gbG9nLmVucmljaFJlc2VydmVkKHsgbWV0YTogY29udGV4dC5tZXRhLCBldmVudDogb3B0aW9ucyB9KVxuICAgIGVucmljaGVkTG9nLnRyYWNlKCdFdmVudCBCRUdJTicpXG4gICAgdHJ5IHtcbiAgICAgICAgYXdhaXQgbWVhc3VyZShsb2cuZW5yaWNoUmVzZXJ2ZWQoeyBtZXRhOiBjb250ZXh0Lm1ldGEgfSksICdleGVjdXRpb24nLCAoKSA9PlxuICAgICAgICAgICAgaGFuZGxlci5lbnRyeShcbiAgICAgICAgICAgICAgICB7IC4uLmNvbnRleHQsIGxvZzogZW5yaWNoZWRMb2cgfSxcbiAgICAgICAgICAgICAgICBvcHRpb25zLnN1YmplY3QsXG4gICAgICAgICAgICAgICAgb3B0aW9ucy5ldmVudCxcbiAgICAgICAgICAgICAgICBvcHRpb25zLnRpbWVzdGFtcCxcbiAgICAgICAgICAgICAgICBvcHRpb25zLm1lc3NhZ2VJZCA/PyByYW5kb21VVUlEKCkucmVwbGFjZUFsbCgnLScsICcnKSxcbiAgICAgICAgICAgICksXG4gICAgICAgIClcbiAgICAgICAgZW5yaWNoZWRMb2cuZGVidWcoJ0V2ZW50IEVORCcpXG4gICAgICAgIGF3YWl0IHN1Y2Nlc3MoKVxuICAgICAgICByZXR1cm4gdHJ1ZVxuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgZW5yaWNoZWRMb2cuZXJyb3IoJ0V2ZW50IEVORCcsIGUpXG4gICAgICAgIHJldHVybiBmYWxzZVxuICAgIH1cbn1cbiJdfQ==
17
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXZlbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJldmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sYUFBYSxDQUFBO0FBQ3hDLE9BQU8sRUFBVyxPQUFPLEVBQW1CLE1BQU0sZUFBZSxDQUFBO0FBSWpFLE1BQU0sQ0FBQyxLQUFLLFVBQVUsTUFBTSxDQUN4QixHQUFlLEVBQ2YsT0FBNkIsRUFDN0IsT0FBcUIsRUFDckIsT0FLQyxFQUNELE9BQStCO0lBRS9CLE1BQU0sV0FBVyxHQUFHLEdBQUcsQ0FBQyxjQUFjLENBQUMsRUFBRSxJQUFJLEVBQUUsT0FBTyxDQUFDLElBQUksRUFBRSxLQUFLLEVBQUUsT0FBTyxFQUFFLENBQUMsQ0FBQTtJQUM5RSxXQUFXLENBQUMsS0FBSyxDQUFDLGFBQWEsQ0FBQyxDQUFBO0lBQ2hDLElBQUksQ0FBQztRQUNELE1BQU0sT0FBTyxDQUFDLEdBQUcsQ0FBQyxjQUFjLENBQUMsRUFBRSxJQUFJLEVBQUUsT0FBTyxDQUFDLElBQUksRUFBRSxDQUFDLEVBQUUsV0FBVyxFQUFFLEdBQUcsRUFBRSxDQUN4RSxPQUFPLENBQUMsS0FBSyxDQUNULEVBQUUsR0FBRyxPQUFPLEVBQUUsR0FBRyxFQUFFLFdBQVcsRUFBRSxFQUNoQyxPQUFPLENBQUMsT0FBTyxFQUNmLE9BQU8sQ0FBQyxLQUFLLEVBQ2IsT0FBTyxDQUFDLFNBQVMsRUFDakIsT0FBTyxDQUFDLFNBQVMsSUFBSSxVQUFVLEVBQUUsQ0FBQyxVQUFVLENBQUMsR0FBRyxFQUFFLEVBQUUsQ0FBQyxDQUN4RCxDQUNKLENBQUE7UUFDRCxXQUFXLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxDQUFBO1FBQzlCLE1BQU0sT0FBTyxFQUFFLENBQUE7UUFDZixPQUFPLElBQUksQ0FBQTtJQUNmLENBQUM7SUFBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1FBQ1QsV0FBVyxDQUFDLEtBQUssQ0FBQyxXQUFXLEVBQUUsQ0FBQyxDQUFDLENBQUE7UUFDakMsT0FBTyxLQUFLLENBQUE7SUFDaEIsQ0FBQztBQUNMLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyByYW5kb21VVUlEIH0gZnJvbSAnbm9kZTpjcnlwdG8nXG5pbXBvcnQgeyBDb250ZXh0LCBtZWFzdXJlLCB0eXBlIEpzb25PYmplY3QgfSBmcm9tICcuLi9jb250ZXh0LmpzJ1xuaW1wb3J0IHsgUm9vdExvZ2dlciB9IGZyb20gJy4vY29udGV4dC5qcydcbmltcG9ydCB0eXBlIHsgRXZlbnRIYW5kbGVyIH0gZnJvbSAnLi9ldmVudC1yZWdpc3RyeS5qcydcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGhhbmRsZShcbiAgICBsb2c6IFJvb3RMb2dnZXIsXG4gICAgY29udGV4dDogT21pdDxDb250ZXh0LCAnbG9nJz4sXG4gICAgaGFuZGxlcjogRXZlbnRIYW5kbGVyLFxuICAgIG9wdGlvbnM6IHtcbiAgICAgICAgcmVhZG9ubHkgc3ViamVjdDogc3RyaW5nXG4gICAgICAgIHJlYWRvbmx5IGV2ZW50OiBKc29uT2JqZWN0IHwgdW5kZWZpbmVkXG4gICAgICAgIHJlYWRvbmx5IHRpbWVzdGFtcDogRGF0ZVxuICAgICAgICByZWFkb25seSBtZXNzYWdlSWQ/OiBzdHJpbmdcbiAgICB9LFxuICAgIHN1Y2Nlc3M6ICgpID0+IFByb21pc2U8dW5rbm93bj4sXG4pOiBQcm9taXNlPGJvb2xlYW4+IHtcbiAgICBjb25zdCBlbnJpY2hlZExvZyA9IGxvZy5lbnJpY2hSZXNlcnZlZCh7IG1ldGE6IGNvbnRleHQubWV0YSwgZXZlbnQ6IG9wdGlvbnMgfSlcbiAgICBlbnJpY2hlZExvZy50cmFjZSgnRXZlbnQgQkVHSU4nKVxuICAgIHRyeSB7XG4gICAgICAgIGF3YWl0IG1lYXN1cmUobG9nLmVucmljaFJlc2VydmVkKHsgbWV0YTogY29udGV4dC5tZXRhIH0pLCAnZXhlY3V0aW9uJywgKCkgPT5cbiAgICAgICAgICAgIGhhbmRsZXIuZW50cnkoXG4gICAgICAgICAgICAgICAgeyAuLi5jb250ZXh0LCBsb2c6IGVucmljaGVkTG9nIH0sXG4gICAgICAgICAgICAgICAgb3B0aW9ucy5zdWJqZWN0LFxuICAgICAgICAgICAgICAgIG9wdGlvbnMuZXZlbnQsXG4gICAgICAgICAgICAgICAgb3B0aW9ucy50aW1lc3RhbXAsXG4gICAgICAgICAgICAgICAgb3B0aW9ucy5tZXNzYWdlSWQgPz8gcmFuZG9tVVVJRCgpLnJlcGxhY2VBbGwoJy0nLCAnJyksXG4gICAgICAgICAgICApLFxuICAgICAgICApXG4gICAgICAgIGVucmljaGVkTG9nLmRlYnVnKCdFdmVudCBFTkQnKVxuICAgICAgICBhd2FpdCBzdWNjZXNzKClcbiAgICAgICAgcmV0dXJuIHRydWVcbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICAgIGVucmljaGVkTG9nLmVycm9yKCdFdmVudCBFTkQnLCBlKVxuICAgICAgICByZXR1cm4gZmFsc2VcbiAgICB9XG59XG4iXX0=
package/host/http.d.ts CHANGED
@@ -24,6 +24,6 @@ type JsonRequestOptions = BodylessRequestOptions & {
24
24
  };
25
25
  export declare function executeRequest(log: RootLogger, context: Omit<Context, 'log'>, handler: HttpHandler, options: RequestOptions, success: () => Promise<unknown>): Promise<Response>;
26
26
  export declare function clientFromHeaders(headers: {
27
- readonly [key: string]: string;
27
+ readonly [key: string]: string | undefined;
28
28
  } | undefined): ClientInfo;
29
29
  export {};
package/host/http.js CHANGED
@@ -1,4 +1,5 @@
1
1
  import { hash } from 'node:crypto';
2
+ import { isIP } from 'node:net';
2
3
  import { brotliCompress } from 'node:zlib';
3
4
  import { measure } from '../context.js';
4
5
  export async function executeRequest(log, context, handler, options, success) {
@@ -217,18 +218,83 @@ export function clientFromHeaders(headers) {
217
218
  if (!headers) {
218
219
  return {};
219
220
  }
220
- const address = headers['x-forwarded-for']?.split(':');
221
221
  return {
222
222
  operationId: headers['x-request-id'] ?? headers['request-id'],
223
223
  clientId: headers['x-client-id'] ??
224
224
  headers['x-installation-id'] ??
225
225
  headers['client-id'] ??
226
226
  headers['installation-id'],
227
- clientIp: address?.[0],
228
- clientPort: Number(address?.[1]) || undefined,
227
+ ...clientAddress(headers),
229
228
  userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],
230
229
  };
231
230
  }
231
+ function clientAddress(headers) {
232
+ const xff = headers['x-forwarded-for'];
233
+ if (!xff) {
234
+ return undefined;
235
+ }
236
+ for (const a of xff.split(',')) {
237
+ const valid = validClientAddress(a.trim());
238
+ if (valid) {
239
+ return valid;
240
+ }
241
+ }
242
+ return undefined;
243
+ }
244
+ function validClientAddress(s) {
245
+ const version = isIP(s);
246
+ if (version === 4) {
247
+ return { clientIp: s };
248
+ }
249
+ if (version === 6) {
250
+ return normalizedIp6(s);
251
+ }
252
+ if (s.startsWith('[') && s.endsWith(']')) {
253
+ return normalizedIp6(s.slice(1, -1));
254
+ }
255
+ if (s.startsWith(':')) {
256
+ const clientPort = Number(s.slice(1));
257
+ if (Number.isFinite(clientPort)) {
258
+ return { clientPort };
259
+ }
260
+ return undefined;
261
+ }
262
+ const url = `ip://${s}/`;
263
+ if (URL.canParse(url)) {
264
+ return addressFromUrl(new URL(url));
265
+ }
266
+ return undefined;
267
+ }
268
+ function normalizedIp6(s) {
269
+ const mapped = mappedAddress(s);
270
+ if (mapped) {
271
+ return { clientIp: mapped };
272
+ }
273
+ const url = new URL(`ip://[${s}]/`);
274
+ return { clientIp: url.hostname.slice(1, -1) };
275
+ }
276
+ function addressFromUrl(url) {
277
+ const clientIp = url.hostname.startsWith('[') && url.hostname.endsWith(']')
278
+ ? url.hostname.slice(1, -1)
279
+ : url.hostname;
280
+ const clientPort = url.port ? Number(url.port) : undefined;
281
+ if (isIP(clientIp)) {
282
+ return {
283
+ clientIp,
284
+ clientPort,
285
+ };
286
+ }
287
+ return undefined;
288
+ }
289
+ function mappedAddress(s) {
290
+ if (s.startsWith('::ffff:')) {
291
+ const ip4 = s.slice(7);
292
+ if (isIP(ip4) === 4) {
293
+ return ip4;
294
+ }
295
+ }
296
+ return undefined;
297
+ }
232
298
  function eTagged(requestHeaders, response) {
233
299
  if (response.headers.etag || !response.body) {
234
300
  return response;
@@ -269,4 +335,4 @@ function compress(body) {
269
335
  });
270
336
  });
271
337
  }
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    let enrichedLog = log.enrichReserved({ meta: context.meta, request: logRequest })\n    if (isShallow) {\n        enrichedLog.trace('Shallow request')\n        return {\n            headers: {},\n            status: 204,\n        }\n    }\n    enrichedLog.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.enrichReserved({ meta: context.meta }), 'execution', () =>\n            handler.entry({ ...context, log: enrichedLog }, 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        enrichedLog = enrichedLog.enrichReserved({\n            response: {\n                status: response.status,\n                headers: response.headers,\n                body: response.logBody,\n            },\n        })\n        if (response.status < 300) {\n            enrichedLog.debug('Request END')\n            await success()\n        } else {\n            enrichedLog.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            enrichedLog = enrichedLog.enrichReserved({ response })\n            enrichedLog.error('Request END', e)\n            return response\n        } catch (convertError) {\n            enrichedLog.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"]}
338
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http.js","sourceRoot":"","sources":["http.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,aAAa,CAAA;AAClC,OAAO,EAAE,IAAI,EAAE,MAAM,UAAU,CAAA;AAC/B,OAAO,EAAE,cAAc,EAAE,MAAM,WAAW,CAAA;AAC1C,OAAO,EAAW,OAAO,EAAE,MAAM,eAAe,CAAA;AA0BhD,MAAM,CAAC,KAAK,UAAU,cAAc,CAChC,GAAe,EACf,OAA6B,EAC7B,OAAoB,EACpB,OAAuB,EACvB,OAA+B;IAE/B,MAAM,SAAS,GACX,OAAO,CAAC,GAAG,CAAC,WAAW,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC,WAAW,CAAC,KAAK,OAAO,CAAC,GAAG,CAAC,WAAW,CAAA;IACzF,MAAM,iBAAiB,GAAG,CAAC,OAAO,CAAC,MAAM,EAAE,mBAAmB,CAAA;IAC9D,MAAM,UAAU,GAAG,iBAAiB;QAChC,CAAC,CAAC,EAAE,MAAM,EAAE,OAAO,CAAC,MAAM,EAAE,GAAG,OAAO,EAAE;QACxC,CAAC,CAAC,kBAAkB,CAAC,EAAE,MAAM,EAAE,OAAO,CAAC,MAAM,EAAE,GAAG,OAAO,EAAE,CAAC,CAAA;IAChE,IAAI,WAAW,GAAG,GAAG,CAAC,cAAc,CAAC,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,EAAE,OAAO,EAAE,UAAU,EAAE,CAAC,CAAA;IACjF,IAAI,SAAS,EAAE,CAAC;QACZ,WAAW,CAAC,KAAK,CAAC,iBAAiB,CAAC,CAAA;QACpC,OAAO;YACH,OAAO,EAAE,EAAE;YACX,MAAM,EAAE,GAAG;SACd,CAAA;IACL,CAAC;IACD,WAAW,CAAC,KAAK,CAAC,eAAe,CAAC,CAAA;IAClC,IAAI,CAAC;QACD,IAAI,SAOW,CAAA;QACf,IAAI,SAA+B,CAAA;QACnC,MAAM,GAAG,GAAG;YACR,MAAM,EAAE,OAAO,CAAC,GAAG;YACnB,IAAI,GAAG;gBACH,IAAI,SAAS,EAAE,CAAC;oBACZ,OAAO,SAAS,CAAA;gBACpB,CAAC;gBACD,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC,CAAA;gBAChC,SAAS,GAAG;oBACR,2CAA2C;oBAC3C,SAAS,EAAE,IAAI;oBACf,IAAI,EAAE,GAAG,CAAC,IAAI;oBACd,IAAI,EAAE,GAAG,CAAC,IAAI;oBACd,QAAQ,EAAE,GAAG,CAAC,QAAQ;oBACtB,IAAI,EAAE,GAAG,CAAC,IAAI;oBACd,MAAM,EAAE,GAAG,CAAC,MAAM;oBAClB,QAAQ,EAAE,GAAG,CAAC,QAAQ;oBACtB,QAAQ,EAAE,GAAG,CAAC,QAAQ;oBACtB,IAAI,EAAE,GAAG,CAAC,IAAI;oBACd,QAAQ,EAAE,GAAG,CAAC,QAAQ;oBACtB,MAAM,EAAE,GAAG,CAAC,MAAM;oBAClB,IAAI,YAAY;wBACZ,OAAO,GAAG,CAAC,YAAY,CAAA;oBAC3B,CAAC;oBACD,MAAM,EAAE,GAAG,EAAE,CAAC,GAAG,CAAC,MAAM,EAAE;oBAC1B,QAAQ,EAAE,GAAG,EAAE,CAAC,GAAG,CAAC,QAAQ,EAAE;oBAC9B,QAAQ,EAAE,GAAG,CAAC,QAAQ;oBACtB,UAAU,EAAE,CAAC,KAAa,EAAE,EAAE;wBAC1B,MAAM,KAAK,GAAG,CAAC,SAAS,KAAK,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAA;wBACrD,MAAM,IAAI,GAAG,KAAK,CAAC,KAAK,GAAG,CAAC,CAAC,CAAA;wBAC7B,IAAI,CAAC,IAAI,EAAE,CAAC;4BACR,MAAM,MAAM,CAAC,MAAM,CACf,IAAI,UAAU,CAAC,sCAAsC,KAAK,GAAG,CAAC,EAC9D;gCACI,MAAM,EAAE,IAAI,CAAC,MAAM;gCACnB,QAAQ,EAAE,GAAG,CAAC,QAAQ;gCACtB,KAAK,EAAE,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC;6BACxB,CACJ,CAAA;wBACL,CAAC;wBACD,OAAO,IAAI,CAAA;oBACf,CAAC;iBACJ,CAAA;gBACD,OAAO,SAAS,CAAA;YACpB,CAAC;YACD,IAAI,EAAE,WAAW,CAAC,OAAO,CAAC;YAC1B,OAAO,EAAE,OAAO,CAAC,OAAO,IAAI,EAAE;SACjC,CAAA;QAED,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,EAAE,CAAC,EAAE,WAAW,EAAE,GAAG,EAAE,CACvF,OAAO,CAAC,KAAK,CAAC,EAAE,GAAG,OAAO,EAAE,GAAG,EAAE,WAAW,EAAE,EAAE,GAAG,CAAC,CACvD,CAAA;QAED,MAAM,QAAQ,GAAG,gBAAgB,CAAC,MAAM,EAAE,iBAAiB,CAAC,CAAA;QAE5D,IAAI,OAAO,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACzB,QAAQ,CAAC,OAAO,GAAG;gBACf,WAAW,EAAE,GAAG;gBAChB,GAAG,QAAQ,CAAC,OAAO;aACtB,CAAA;QACL,CAAC;QAED,WAAW,GAAG,WAAW,CAAC,cAAc,CAAC;YACrC,QAAQ,EAAE;gBACN,MAAM,EAAE,QAAQ,CAAC,MAAM;gBACvB,OAAO,EAAE,QAAQ,CAAC,OAAO;gBACzB,IAAI,EAAE,QAAQ,CAAC,OAAO;aACzB;SACJ,CAAC,CAAA;QACF,IAAI,QAAQ,CAAC,MAAM,GAAG,GAAG,EAAE,CAAC;YACxB,WAAW,CAAC,KAAK,CAAC,aAAa,CAAC,CAAA;YAChC,MAAM,OAAO,EAAE,CAAA;QACnB,CAAC;aAAM,CAAC;YACJ,WAAW,CAAC,IAAI,CAAC,aAAa,CAAC,CAAA;QACnC,CAAC;QACD,OAAO,MAAM,UAAU,CAAC,GAAG,CAAC,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC,CAAA;IACxE,CAAC;IAAC,OAAO,CAAC,EAAE,CAAC;QACT,IAAI,CAAC;YACD,MAAM,QAAQ,GAAG,eAAe,CAAC,CAAC,CAAC,CAAA;YACnC,WAAW,GAAG,WAAW,CAAC,cAAc,CAAC,EAAE,QAAQ,EAAE,CAAC,CAAA;YACtD,WAAW,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC,CAAC,CAAA;YACnC,OAAO,QAAQ,CAAA;QACnB,CAAC;QAAC,OAAO,YAAY,EAAE,CAAC;YACpB,WAAW,CAAC,KAAK,CAAC,gDAAgD,EAAE,YAAY,CAAC,CAAA;YACjF,OAAO;gBACH,OAAO,EAAE,EAAE;gBACX,MAAM,EAAE,GAAG;aACd,CAAA;QACL,CAAC;IACL,CAAC;AACL,CAAC;AAED,SAAS,gBAAgB,CAAC,MAAc,EAAE,WAAoB;IAC1D,IAAI,CAAC,MAAM,EAAE,CAAC;QACV,OAAO;YACH,OAAO,EAAE,EAAE;YACX,MAAM,EAAE,GAAG;SACd,CAAA;IACL,CAAC;SAAM,IAAI,OAAO,MAAM,KAAK,QAAQ,EAAE,CAAC;QACpC,MAAM,OAAO,GAAG,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,SAAS,CAAA;QAChD,OAAO;YACH,OAAO,EAAE;gBACL,cAAc,EAAE,YAAY;aAC/B;YACD,MAAM,EAAE,GAAG;YACX,IAAI,EAAE,MAAM;YACZ,OAAO;SACV,CAAA;IACL,CAAC;SAAM,CAAC;QACJ,IAAI,MAAM,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;YAC5B,OAAO;gBACH,OAAO,EAAE,MAAM,CAAC,OAAO,IAAI,EAAE;gBAC7B,MAAM,EAAE,MAAM,CAAC,MAAM,IAAI,GAAG;aAC/B,CAAA;QACL,CAAC;aAAM,IAAI,OAAO,MAAM,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YACzC,MAAM,OAAO,GAAG,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAA;YACrD,OAAO;gBACH,OAAO,EAAE,eAAe,CAAC,MAAM,CAAC,OAAO,EAAE,YAAY,CAAC;gBACtD,MAAM,EAAE,MAAM,CAAC,MAAM,IAAI,GAAG;gBAC5B,IAAI,EAAE,MAAM,CAAC,IAAI;gBACjB,OAAO;aACV,CAAA;QACL,CAAC;aAAM,IAAI,MAAM,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC;YACtC,MAAM,OAAO,GAAG,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,SAAS,CAAA;YACxE,OAAO;gBACH,OAAO,EAAE,eAAe,CAAC,MAAM,CAAC,OAAO,EAAE,0BAA0B,CAAC;gBACpE,MAAM,EAAE,MAAM,CAAC,MAAM,IAAI,GAAG;gBAC5B,IAAI,EAAE,MAAM,CAAC,IAAI;gBACjB,OAAO;aACV,CAAA;QACL,CAAC;aAAM,CAAC;YACJ,MAAM,OAAO,GAAG,WAAW,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAA;YACrD,OAAO;gBACH,OAAO,EAAE,eAAe,CAAC,MAAM,CAAC,OAAO,EAAE,kBAAkB,CAAC;gBAC5D,MAAM,EAAE,MAAM,CAAC,MAAM,IAAI,GAAG;gBAC5B,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,IAAI,CAAC;gBACjC,OAAO;aACV,CAAA;QACL,CAAC;IACL,CAAC;AACL,CAAC;AAED,SAAS,kBAAkB,CAAC,OAA4C;IACpE,IAAI,MAAM,IAAI,OAAO,EAAE,CAAC;QACpB,MAAM,EAAE,IAAI,EAAE,GAAG,QAAQ,EAAE,GAAG,OAAO,CAAA;QACrC,OAAO,QAAQ,CAAA;IACnB,CAAC;IACD,IAAI,MAAM,IAAI,OAAO,EAAE,CAAC;QACpB,MAAM,EAAE,IAAI,EAAE,GAAG,QAAQ,EAAE,GAAG,OAAO,CAAA;QACrC,OAAO,QAAQ,CAAA;IACnB,CAAC;IACD,OAAO,OAAO,CAAA;AAClB,CAAC;AAED,SAAS,WAAW,CAAC,OAAuB;IACxC,IAAI,MAAM,IAAI,OAAO,EAAE,CAAC;QACpB,OAAO,OAAO,CAAC,IAAI,CAAA;IACvB,CAAC;IACD,IAAI,MAAM,IAAI,OAAO,EAAE,CAAC;QACpB,OAAO,OAAO,CAAC,IAAI,CAAA;IACvB,CAAC;IACD,OAAO,SAAS,CAAA;AACpB,CAAC;AAED,SAAS,eAAe,CAAC,OAAoC,EAAE,WAAmB;IAC9E,IAAI,CAAC,OAAO,EAAE,CAAC;QACX,OAAO;YACH,cAAc,EAAE,WAAW;SAC9B,CAAA;IACL,CAAC;IACD,OAAO,CAAC,cAAc,CAAC,KAAK,WAAW,CAAA;IACvC,OAAO,OAAO,CAAA;AAClB,CAAC;AAED,SAAS,eAAe,CAAC,CAAU;IAC/B,MAAM,EAAE,IAAI,EAAE,UAAU,EAAE,MAAM,EAAE,GAAG,CAA4C,CAAA;IACjF,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;QAC3B,OAAO;YACH,OAAO,EAAE;gBACL,cAAc,EAAE,YAAY;aAC/B;YACD,MAAM,EAAE,MAAM,IAAI,GAAG;YACrB,IAAI;SACP,CAAA;IACL,CAAC;SAAM,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;QAClC,OAAO;YACH,OAAO,EAAE;gBACL,cAAc,EAAE,kBAAkB;aACrC;YACD,MAAM,EAAE,MAAM,IAAI,GAAG;YACrB,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC;SAC7B,CAAA;IACL,CAAC;SAAM,CAAC;QACJ,OAAO;YACH,OAAO,EAAE,EAAE;YACX,MAAM,EAAE,MAAM,IAAI,GAAG;SACxB,CAAA;IACL,CAAC;AACL,CAAC;AAED,MAAM,UAAU,iBAAiB,CAC7B,OAAmE;IAEnE,IAAI,CAAC,OAAO,EAAE,CAAC;QACX,OAAO,EAAE,CAAA;IACb,CAAC;IACD,OAAO;QACH,WAAW,EAAE,OAAO,CAAC,cAAc,CAAC,IAAI,OAAO,CAAC,YAAY,CAAC;QAC7D,QAAQ,EACJ,OAAO,CAAC,aAAa,CAAC;YACtB,OAAO,CAAC,mBAAmB,CAAC;YAC5B,OAAO,CAAC,WAAW,CAAC;YACpB,OAAO,CAAC,iBAAiB,CAAC;QAC9B,GAAG,aAAa,CAAC,OAAO,CAAC;QACzB,SAAS,EAAE,OAAO,CAAC,4BAA4B,CAAC,IAAI,OAAO,CAAC,YAAY,CAAC;KAC5E,CAAA;AACL,CAAC;AAED,SAAS,aAAa,CAAC,OAAuD;IAC1E,MAAM,GAAG,GAAG,OAAO,CAAC,iBAAiB,CAAC,CAAA;IACtC,IAAI,CAAC,GAAG,EAAE,CAAC;QACP,OAAO,SAAS,CAAA;IACpB,CAAC;IACD,KAAK,MAAM,CAAC,IAAI,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC;QAC7B,MAAM,KAAK,GAAG,kBAAkB,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAA;QAC1C,IAAI,KAAK,EAAE,CAAC;YACR,OAAO,KAAK,CAAA;QAChB,CAAC;IACL,CAAC;IACD,OAAO,SAAS,CAAA;AACpB,CAAC;AAED,SAAS,kBAAkB,CAAC,CAAS;IACjC,MAAM,OAAO,GAAG,IAAI,CAAC,CAAC,CAAC,CAAA;IACvB,IAAI,OAAO,KAAK,CAAC,EAAE,CAAC;QAChB,OAAO,EAAE,QAAQ,EAAE,CAAC,EAAE,CAAA;IAC1B,CAAC;IACD,IAAI,OAAO,KAAK,CAAC,EAAE,CAAC;QAChB,OAAO,aAAa,CAAC,CAAC,CAAC,CAAA;IAC3B,CAAC;IACD,IAAI,CAAC,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;QACvC,OAAO,aAAa,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAA;IACxC,CAAC;IACD,IAAI,CAAC,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC;QACpB,MAAM,UAAU,GAAG,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAA;QACrC,IAAI,MAAM,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE,CAAC;YAC9B,OAAO,EAAE,UAAU,EAAE,CAAA;QACzB,CAAC;QACD,OAAO,SAAS,CAAA;IACpB,CAAC;IACD,MAAM,GAAG,GAAG,QAAQ,CAAC,GAAG,CAAA;IACxB,IAAI,GAAG,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;QACpB,OAAO,cAAc,CAAC,IAAI,GAAG,CAAC,GAAG,CAAC,CAAC,CAAA;IACvC,CAAC;IACD,OAAO,SAAS,CAAA;AACpB,CAAC;AAED,SAAS,aAAa,CAAC,CAAS;IAC5B,MAAM,MAAM,GAAG,aAAa,CAAC,CAAC,CAAC,CAAA;IAC/B,IAAI,MAAM,EAAE,CAAC;QACT,OAAO,EAAE,QAAQ,EAAE,MAAM,EAAE,CAAA;IAC/B,CAAC;IACD,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,SAAS,CAAC,IAAI,CAAC,CAAA;IACnC,OAAO,EAAE,QAAQ,EAAE,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAA;AAClD,CAAC;AAED,SAAS,cAAc,CAAC,GAAQ;IAC5B,MAAM,QAAQ,GACV,GAAG,CAAC,QAAQ,CAAC,UAAU,CAAC,GAAG,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,QAAQ,CAAC,GAAG,CAAC;QACtD,CAAC,CAAC,GAAG,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QAC3B,CAAC,CAAC,GAAG,CAAC,QAAQ,CAAA;IACtB,MAAM,UAAU,GAAG,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS,CAAA;IAC1D,IAAI,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC;QACjB,OAAO;YACH,QAAQ;YACR,UAAU;SACb,CAAA;IACL,CAAC;IACD,OAAO,SAAS,CAAA;AACpB,CAAC;AAED,SAAS,aAAa,CAAC,CAAS;IAC5B,IAAI,CAAC,CAAC,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;QAC1B,MAAM,GAAG,GAAG,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAA;QACtB,IAAI,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;YAClB,OAAO,GAAG,CAAA;QACd,CAAC;IACL,CAAC;IACD,OAAO,SAAS,CAAA;AACpB,CAAC;AAED,SAAS,OAAO,CACZ,cAAyC,EACzC,QAA2D;IAE3D,IAAI,QAAQ,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QAC1C,OAAO,QAAQ,CAAA;IACnB,CAAC;IACD,MAAM,IAAI,GAAG,IAAI,CAAC,MAAM,EAAE,QAAQ,CAAC,IAAI,EAAE,QAAQ,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAA;IAC/D,QAAQ,CAAC,OAAO,CAAC,IAAI,GAAG,IAAI,CAAA;IAC5B,IAAI,cAAc,CAAC,eAAe,CAAC,KAAK,IAAI,EAAE,CAAC;QAC3C,QAAQ,CAAC,MAAM,GAAG,GAAG,CAAA;QACrB,OAAO,QAAQ,CAAC,IAAI,CAAA;IACxB,CAAC;IACD,OAAO,QAAQ,CAAA;AACnB,CAAC;AAED,KAAK,UAAU,UAAU,CACrB,cAAyC,EACzC,QAA2D;IAE3D,IAAI,CAAC,QAAQ,CAAC,IAAI,IAAI,QAAQ,CAAC,IAAI,CAAC,MAAM,GAAG,MAAM,IAAI,QAAQ,CAAC,OAAO,CAAC,kBAAkB,CAAC,EAAE,CAAC;QAC1F,OAAO,QAAQ,CAAA;IACnB,CAAC;IACD,MAAM,SAAS,GAAG,cAAc,CAAC,iBAAiB,CAAC,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAA;IAClF,IAAI,CAAC,SAAS,EAAE,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC;QAC7B,OAAO,QAAQ,CAAA;IACnB,CAAC;IACD,OAAO;QACH,MAAM,EAAE,QAAQ,CAAC,MAAM;QACvB,OAAO,EAAE;YACL,kBAAkB,EAAE,IAAI;YACxB,GAAG,QAAQ,CAAC,OAAO;SACtB;QACD,IAAI,EAAE,MAAM,QAAQ,CAAC,QAAQ,CAAC,IAAI,CAAC;KACtC,CAAA;AACL,CAAC;AAED,SAAS,QAAQ,CAAC,IAAqB;IACnC,OAAO,IAAI,OAAO,CAAS,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;QAC3C,cAAc,CAAC,IAAI,EAAE,EAAE,EAAE,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE;YACvC,IAAI,KAAK,EAAE,CAAC;gBACR,MAAM,CAAC,KAAK,CAAC,CAAA;gBACb,OAAM;YACV,CAAC;YACD,OAAO,CAAC,MAAM,CAAC,CAAA;QACnB,CAAC,CAAC,CAAA;IACN,CAAC,CAAC,CAAA;AACN,CAAC","sourcesContent":["import { hash } from 'node:crypto'\nimport { isIP } from 'node:net'\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    let enrichedLog = log.enrichReserved({ meta: context.meta, request: logRequest })\n    if (isShallow) {\n        enrichedLog.trace('Shallow request')\n        return {\n            headers: {},\n            status: 204,\n        }\n    }\n    enrichedLog.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.enrichReserved({ meta: context.meta }), 'execution', () =>\n            handler.entry({ ...context, log: enrichedLog }, 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        enrichedLog = enrichedLog.enrichReserved({\n            response: {\n                status: response.status,\n                headers: response.headers,\n                body: response.logBody,\n            },\n        })\n        if (response.status < 300) {\n            enrichedLog.debug('Request END')\n            await success()\n        } else {\n            enrichedLog.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            enrichedLog = enrichedLog.enrichReserved({ response })\n            enrichedLog.error('Request END', e)\n            return response\n        } catch (convertError) {\n            enrichedLog.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 } | 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        ...clientAddress(headers),\n        userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],\n    }\n}\n\nfunction clientAddress(headers: { readonly [key: string]: string | undefined }) {\n    const xff = headers['x-forwarded-for']\n    if (!xff) {\n        return undefined\n    }\n    for (const a of xff.split(',')) {\n        const valid = validClientAddress(a.trim())\n        if (valid) {\n            return valid\n        }\n    }\n    return undefined\n}\n\nfunction validClientAddress(s: string) {\n    const version = isIP(s)\n    if (version === 4) {\n        return { clientIp: s }\n    }\n    if (version === 6) {\n        return normalizedIp6(s)\n    }\n    if (s.startsWith('[') && s.endsWith(']')) {\n        return normalizedIp6(s.slice(1, -1))\n    }\n    if (s.startsWith(':')) {\n        const clientPort = Number(s.slice(1))\n        if (Number.isFinite(clientPort)) {\n            return { clientPort }\n        }\n        return undefined\n    }\n    const url = `ip://${s}/`\n    if (URL.canParse(url)) {\n        return addressFromUrl(new URL(url))\n    }\n    return undefined\n}\n\nfunction normalizedIp6(s: string) {\n    const mapped = mappedAddress(s)\n    if (mapped) {\n        return { clientIp: mapped }\n    }\n    const url = new URL(`ip://[${s}]/`)\n    return { clientIp: url.hostname.slice(1, -1) }\n}\n\nfunction addressFromUrl(url: URL) {\n    const clientIp =\n        url.hostname.startsWith('[') && url.hostname.endsWith(']')\n            ? url.hostname.slice(1, -1)\n            : url.hostname\n    const clientPort = url.port ? Number(url.port) : undefined\n    if (isIP(clientIp)) {\n        return {\n            clientIp,\n            clientPort,\n        }\n    }\n    return undefined\n}\n\nfunction mappedAddress(s: string) {\n    if (s.startsWith('::ffff:')) {\n        const ip4 = s.slice(7)\n        if (isIP(ip4) === 4) {\n            return ip4\n        }\n    }\n    return undefined\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"]}
package/http.d.ts CHANGED
@@ -1,4 +1,4 @@
1
- import { Context, HandlerConfiguration, Json } from './context.js';
1
+ import { Context, HandlerConfiguration, type Json } from './context.js';
2
2
  export * from './context.js';
3
3
  export type ResponseHeaders = {
4
4
  [key: string]: string;
package/http.js CHANGED
@@ -15,4 +15,4 @@ export function patch(path, configOrHandler, fn) {
15
15
  export function del(path, configOrHandler, fn) {
16
16
  registerHttpHandler('DELETE', path, configOrHandler, fn);
17
17
  }
18
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http.js","sourceRoot":"","sources":["http.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,mBAAmB,EAAE,MAAM,yBAAyB,CAAA;AAE7D,cAAc,cAAc,CAAA;AAqN5B,MAAM,UAAU,GAAG,CACf,IAAa,EACb,eAAmD,EACnD,EAAY;IAEZ,mBAAmB,CAAC,KAAK,EAAE,IAAI,EAAE,eAAe,EAAE,EAAE,CAAC,CAAA;AACzD,CAAC;AAOD,MAAM,UAAU,IAAI,CAChB,IAAa,EACb,eAAmD,EACnD,EAAY;IAEZ,mBAAmB,CAAC,MAAM,EAAE,IAAI,EAAE,eAAe,EAAE,EAAE,CAAC,CAAA;AAC1D,CAAC;AAOD,MAAM,UAAU,GAAG,CACf,IAAa,EACb,eAAmD,EACnD,EAAY;IAEZ,mBAAmB,CAAC,KAAK,EAAE,IAAI,EAAE,eAAe,EAAE,EAAE,CAAC,CAAA;AACzD,CAAC;AAOD,MAAM,UAAU,KAAK,CACjB,IAAa,EACb,eAAmD,EACnD,EAAY;IAEZ,mBAAmB,CAAC,OAAO,EAAE,IAAI,EAAE,eAAe,EAAE,EAAE,CAAC,CAAA;AAC3D,CAAC;AAOD,MAAM,UAAU,GAAG,CACf,IAAa,EACb,eAAmD,EACnD,EAAY;IAEZ,mBAAmB,CAAC,QAAQ,EAAE,IAAI,EAAE,eAAe,EAAE,EAAE,CAAC,CAAA;AAC5D,CAAC","sourcesContent":["import { Context, HandlerConfiguration, Json } from './context.js'\nimport { registerHttpHandler } from './host/http-registry.js'\n\nexport * from './context.js'\n\nexport type ResponseHeaders = {\n    [key: string]: string\n}\n\nexport type FullResult = {\n    headers?: ResponseHeaders\n    status?: number\n    body?: unknown\n}\n\nexport type Result = void | string | FullResult\n\nexport type HttpRequest = {\n    readonly rawUrl: string\n    readonly url: ParsedUrl\n    readonly headers: Readonly<ResponseHeaders>\n    readonly body?: Json | string\n}\n\ntype Gone = 'Gone' | undefined\n\nexport type ParsedUrl = {\n    /**\n     * Gets the fragment portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/foo#bar');\n     * console.log(myURL.hash);\n     * // Prints #bar\n     * ```\n     */\n    readonly hash: string\n    /**\n     * Gets the host portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org:81/foo');\n     * console.log(myURL.host);\n     * // Prints example.org:81\n     */\n    readonly host: string\n    /**\n     * Gets the host name portion of the URL. The key difference between`url.host` and `url.hostname` is that `url.hostname` does _not_ include the\n     * port.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org:81/foo');\n     * console.log(myURL.hostname);\n     * // Prints example.org\n     */\n    readonly hostname: string\n    /**\n     * Gets the serialized URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/foo');\n     * console.log(myURL.href);\n     * // Prints https://example.org/foo\n     * ```\n     */\n    readonly href: string\n    /**\n     * Gets the serialization of the URL's origin.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/foo/bar?baz');\n     * console.log(myURL.origin);\n     * // Prints https://example.org\n     * ```\n     *\n     * ```js\n     * const idnURL = new URL('https://測試');\n     * console.log(idnURL.origin);\n     * // Prints https://xn--g6w251d\n     *\n     * console.log(idnURL.hostname);\n     * // Prints xn--g6w251d\n     * ```\n     */\n    readonly origin: string\n    /**\n     * Gets the password portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://abc:xyz@example.com');\n     * console.log(myURL.password);\n     * // Prints xyz\n     *\n     * myURL.password = '123';\n     * console.log(myURL.href);\n     * // Prints https://abc:123@example.com/\n     * ```\n     */\n    readonly password: string\n    /**\n     * Gets the path portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/abc/xyz?123');\n     * console.log(myURL.pathname);\n     * // Prints /abc/xyz\n     * ```\n     */\n    readonly pathname: string\n    /**\n     * Gets the port portion of the URL.\n     *\n     * The port value may be a number or a string containing a number in the range `0` to `65535` (inclusive). Setting the value to the default port of the `URL` objects given `protocol` will\n     * result in the `port` value becoming\n     * the empty string (`''`).\n     *\n     * The port value can be an empty string in which case the port depends on\n     * the protocol/scheme.\n     *\n     * If that string is invalid but it begins with a number, the leading number is\n     * assigned to `port`.\n     * If the number lies outside the range denoted above, it is ignored.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org:8888');\n     * console.log(myURL.port);\n     * // Prints 8888\n     * ```\n     */\n    readonly port: string\n    /**\n     * Gets the protocol portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org');\n     * console.log(myURL.protocol);\n     * // Prints https:\n     * ```\n     */\n    readonly protocol: string\n    /**\n     * Gets the serialized query portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/abc?123');\n     * console.log(myURL.search);\n     * // Prints ?123\n     * ```\n     *\n     * Any invalid URL characters appearing in the value assigned the `search`property will be `percent-encoded`.\n     */\n    readonly search: string\n    /**\n     * Gets the `URLSearchParams` object representing the query parameters of the\n     * URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/abc?foo=~bar');\n     *\n     * console.log(myURL.search);  // prints ?foo=~bar\n     * ```\n     */\n    readonly searchParams: URLSearchParams\n    /**\n     * Gets the username portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://abc:xyz@example.com');\n     * console.log(myURL.username);\n     * // Prints abc\n     * ```\n     *\n     * Any invalid URL characters appearing in the value assigned the `username` property will be `percent-encoded`.\n     */\n    readonly username: string\n\n    /**\n     * Gets a step of the path portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/abc/xyz?123');\n     * console.log(myURL.pathStepAt(0));\n     * // Prints abc\n     * console.log(myURL.pathStepAt(1));\n     * // Prints xyz\n     * ```\n     */\n    pathStepAt: (index: number) => string\n\n    /**\n     * Gone, use searchParams instead.\n     */\n    query?: Gone\n\n    /** @ignore */\n    toJSON: () => string\n}\n\nexport type HttpHandlerConfiguration = HandlerConfiguration & {\n    /**\n     * A string identifying which domains can access the endpoint cross-origin.\n     * @default undefined\n     */\n    readonly cors?: string\n}\n\nexport type Handler = (context: Context, request: HttpRequest) => Promise<Result> | Result\n\ntype Path<T> = T extends `/${string}` ? 'Error: path cannot start with slash.' : T\n\nexport function get<T extends string>(path: Path<T>, fn: Handler): void\nexport function get<T extends string>(\n    path: Path<T>,\n    config: HttpHandlerConfiguration,\n    fn: Handler,\n): void\nexport function get<T extends string>(\n    path: Path<T>,\n    configOrHandler: HttpHandlerConfiguration | Handler,\n    fn?: Handler,\n): void {\n    registerHttpHandler('GET', path, configOrHandler, fn)\n}\nexport function post<T extends string>(path: Path<T>, fn: Handler): void\nexport function post<T extends string>(\n    path: Path<T>,\n    config: HttpHandlerConfiguration,\n    fn: Handler,\n): void\nexport function post<T extends string>(\n    path: Path<T>,\n    configOrHandler: HttpHandlerConfiguration | Handler,\n    fn?: Handler,\n): void {\n    registerHttpHandler('POST', path, configOrHandler, fn)\n}\nexport function put<T extends string>(path: Path<T>, fn: Handler): void\nexport function put<T extends string>(\n    path: Path<T>,\n    config: HttpHandlerConfiguration,\n    fn: Handler,\n): void\nexport function put<T extends string>(\n    path: Path<T>,\n    configOrHandler: HttpHandlerConfiguration | Handler,\n    fn?: Handler,\n): void {\n    registerHttpHandler('PUT', path, configOrHandler, fn)\n}\nexport function patch<T extends string>(path: Path<T>, fn: Handler): void\nexport function patch<T extends string>(\n    path: Path<T>,\n    config: HttpHandlerConfiguration,\n    fn: Handler,\n): void\nexport function patch<T extends string>(\n    path: Path<T>,\n    configOrHandler: HttpHandlerConfiguration | Handler,\n    fn?: Handler,\n): void {\n    registerHttpHandler('PATCH', path, configOrHandler, fn)\n}\nexport function del<T extends string>(path: Path<T>, fn: Handler): void\nexport function del<T extends string>(\n    path: Path<T>,\n    config: HttpHandlerConfiguration,\n    fn: Handler,\n): void\nexport function del<T extends string>(\n    path: Path<T>,\n    configOrHandler: HttpHandlerConfiguration | Handler,\n    fn?: Handler,\n): void {\n    registerHttpHandler('DELETE', path, configOrHandler, fn)\n}\n"]}
18
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http.js","sourceRoot":"","sources":["http.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,mBAAmB,EAAE,MAAM,yBAAyB,CAAA;AAE7D,cAAc,cAAc,CAAA;AAqN5B,MAAM,UAAU,GAAG,CACf,IAAa,EACb,eAAmD,EACnD,EAAY;IAEZ,mBAAmB,CAAC,KAAK,EAAE,IAAI,EAAE,eAAe,EAAE,EAAE,CAAC,CAAA;AACzD,CAAC;AAOD,MAAM,UAAU,IAAI,CAChB,IAAa,EACb,eAAmD,EACnD,EAAY;IAEZ,mBAAmB,CAAC,MAAM,EAAE,IAAI,EAAE,eAAe,EAAE,EAAE,CAAC,CAAA;AAC1D,CAAC;AAOD,MAAM,UAAU,GAAG,CACf,IAAa,EACb,eAAmD,EACnD,EAAY;IAEZ,mBAAmB,CAAC,KAAK,EAAE,IAAI,EAAE,eAAe,EAAE,EAAE,CAAC,CAAA;AACzD,CAAC;AAOD,MAAM,UAAU,KAAK,CACjB,IAAa,EACb,eAAmD,EACnD,EAAY;IAEZ,mBAAmB,CAAC,OAAO,EAAE,IAAI,EAAE,eAAe,EAAE,EAAE,CAAC,CAAA;AAC3D,CAAC;AAOD,MAAM,UAAU,GAAG,CACf,IAAa,EACb,eAAmD,EACnD,EAAY;IAEZ,mBAAmB,CAAC,QAAQ,EAAE,IAAI,EAAE,eAAe,EAAE,EAAE,CAAC,CAAA;AAC5D,CAAC","sourcesContent":["import { Context, HandlerConfiguration, type Json } from './context.js'\nimport { registerHttpHandler } from './host/http-registry.js'\n\nexport * from './context.js'\n\nexport type ResponseHeaders = {\n    [key: string]: string\n}\n\nexport type FullResult = {\n    headers?: ResponseHeaders\n    status?: number\n    body?: unknown\n}\n\nexport type Result = void | string | FullResult\n\nexport type HttpRequest = {\n    readonly rawUrl: string\n    readonly url: ParsedUrl\n    readonly headers: Readonly<ResponseHeaders>\n    readonly body?: Json | string\n}\n\ntype Gone = 'Gone' | undefined\n\nexport type ParsedUrl = {\n    /**\n     * Gets the fragment portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/foo#bar');\n     * console.log(myURL.hash);\n     * // Prints #bar\n     * ```\n     */\n    readonly hash: string\n    /**\n     * Gets the host portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org:81/foo');\n     * console.log(myURL.host);\n     * // Prints example.org:81\n     */\n    readonly host: string\n    /**\n     * Gets the host name portion of the URL. The key difference between`url.host` and `url.hostname` is that `url.hostname` does _not_ include the\n     * port.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org:81/foo');\n     * console.log(myURL.hostname);\n     * // Prints example.org\n     */\n    readonly hostname: string\n    /**\n     * Gets the serialized URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/foo');\n     * console.log(myURL.href);\n     * // Prints https://example.org/foo\n     * ```\n     */\n    readonly href: string\n    /**\n     * Gets the serialization of the URL's origin.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/foo/bar?baz');\n     * console.log(myURL.origin);\n     * // Prints https://example.org\n     * ```\n     *\n     * ```js\n     * const idnURL = new URL('https://測試');\n     * console.log(idnURL.origin);\n     * // Prints https://xn--g6w251d\n     *\n     * console.log(idnURL.hostname);\n     * // Prints xn--g6w251d\n     * ```\n     */\n    readonly origin: string\n    /**\n     * Gets the password portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://abc:xyz@example.com');\n     * console.log(myURL.password);\n     * // Prints xyz\n     *\n     * myURL.password = '123';\n     * console.log(myURL.href);\n     * // Prints https://abc:123@example.com/\n     * ```\n     */\n    readonly password: string\n    /**\n     * Gets the path portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/abc/xyz?123');\n     * console.log(myURL.pathname);\n     * // Prints /abc/xyz\n     * ```\n     */\n    readonly pathname: string\n    /**\n     * Gets the port portion of the URL.\n     *\n     * The port value may be a number or a string containing a number in the range `0` to `65535` (inclusive). Setting the value to the default port of the `URL` objects given `protocol` will\n     * result in the `port` value becoming\n     * the empty string (`''`).\n     *\n     * The port value can be an empty string in which case the port depends on\n     * the protocol/scheme.\n     *\n     * If that string is invalid but it begins with a number, the leading number is\n     * assigned to `port`.\n     * If the number lies outside the range denoted above, it is ignored.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org:8888');\n     * console.log(myURL.port);\n     * // Prints 8888\n     * ```\n     */\n    readonly port: string\n    /**\n     * Gets the protocol portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org');\n     * console.log(myURL.protocol);\n     * // Prints https:\n     * ```\n     */\n    readonly protocol: string\n    /**\n     * Gets the serialized query portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/abc?123');\n     * console.log(myURL.search);\n     * // Prints ?123\n     * ```\n     *\n     * Any invalid URL characters appearing in the value assigned the `search`property will be `percent-encoded`.\n     */\n    readonly search: string\n    /**\n     * Gets the `URLSearchParams` object representing the query parameters of the\n     * URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/abc?foo=~bar');\n     *\n     * console.log(myURL.search);  // prints ?foo=~bar\n     * ```\n     */\n    readonly searchParams: URLSearchParams\n    /**\n     * Gets the username portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://abc:xyz@example.com');\n     * console.log(myURL.username);\n     * // Prints abc\n     * ```\n     *\n     * Any invalid URL characters appearing in the value assigned the `username` property will be `percent-encoded`.\n     */\n    readonly username: string\n\n    /**\n     * Gets a step of the path portion of the URL.\n     *\n     * ```js\n     * const myURL = new URL('https://example.org/abc/xyz?123');\n     * console.log(myURL.pathStepAt(0));\n     * // Prints abc\n     * console.log(myURL.pathStepAt(1));\n     * // Prints xyz\n     * ```\n     */\n    pathStepAt: (index: number) => string\n\n    /**\n     * Gone, use searchParams instead.\n     */\n    query?: Gone\n\n    /** @ignore */\n    toJSON: () => string\n}\n\nexport type HttpHandlerConfiguration = HandlerConfiguration & {\n    /**\n     * A string identifying which domains can access the endpoint cross-origin.\n     * @default undefined\n     */\n    readonly cors?: string\n}\n\nexport type Handler = (context: Context, request: HttpRequest) => Promise<Result> | Result\n\ntype Path<T> = T extends `/${string}` ? 'Error: path cannot start with slash.' : T\n\nexport function get<T extends string>(path: Path<T>, fn: Handler): void\nexport function get<T extends string>(\n    path: Path<T>,\n    config: HttpHandlerConfiguration,\n    fn: Handler,\n): void\nexport function get<T extends string>(\n    path: Path<T>,\n    configOrHandler: HttpHandlerConfiguration | Handler,\n    fn?: Handler,\n): void {\n    registerHttpHandler('GET', path, configOrHandler, fn)\n}\nexport function post<T extends string>(path: Path<T>, fn: Handler): void\nexport function post<T extends string>(\n    path: Path<T>,\n    config: HttpHandlerConfiguration,\n    fn: Handler,\n): void\nexport function post<T extends string>(\n    path: Path<T>,\n    configOrHandler: HttpHandlerConfiguration | Handler,\n    fn?: Handler,\n): void {\n    registerHttpHandler('POST', path, configOrHandler, fn)\n}\nexport function put<T extends string>(path: Path<T>, fn: Handler): void\nexport function put<T extends string>(\n    path: Path<T>,\n    config: HttpHandlerConfiguration,\n    fn: Handler,\n): void\nexport function put<T extends string>(\n    path: Path<T>,\n    configOrHandler: HttpHandlerConfiguration | Handler,\n    fn?: Handler,\n): void {\n    registerHttpHandler('PUT', path, configOrHandler, fn)\n}\nexport function patch<T extends string>(path: Path<T>, fn: Handler): void\nexport function patch<T extends string>(\n    path: Path<T>,\n    config: HttpHandlerConfiguration,\n    fn: Handler,\n): void\nexport function patch<T extends string>(\n    path: Path<T>,\n    configOrHandler: HttpHandlerConfiguration | Handler,\n    fn?: Handler,\n): void {\n    registerHttpHandler('PATCH', path, configOrHandler, fn)\n}\nexport function del<T extends string>(path: Path<T>, fn: Handler): void\nexport function del<T extends string>(\n    path: Path<T>,\n    config: HttpHandlerConfiguration,\n    fn: Handler,\n): void\nexport function del<T extends string>(\n    path: Path<T>,\n    configOrHandler: HttpHandlerConfiguration | Handler,\n    fn?: Handler,\n): void {\n    registerHttpHandler('DELETE', path, configOrHandler, fn)\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@riddance/host",
3
- "version": "0.2.4",
3
+ "version": "0.2.6",
4
4
  "type": "module",
5
5
  "license": "MIT",
6
6
  "engines": {
@@ -42,7 +42,7 @@
42
42
  "postpublish": "riddance-sync"
43
43
  },
44
44
  "devDependencies": {
45
- "@riddance/env": "0.9.3"
45
+ "@riddance/env": "0.9.7"
46
46
  },
47
- "gitHead": "2271676e1d0552abda3dd26897411d6464570468"
47
+ "gitHead": "5cd4acd5013f5fcb9598875ddcac62ecfcee4c14"
48
48
  }