@riddance/host 0.0.4 → 0.0.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/host/context.js CHANGED
@@ -34,8 +34,7 @@ export function createContext(clientInfo, loggers, eventTransport, timeouts, out
34
34
  const innerController = new AbortController();
35
35
  // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
36
36
  const logTransport = loggers.length === 1 ? loggers[0] : new LogMulticaster(loggers);
37
- const logger = makeLogger(logTransport, meta?.config?.minimumLogLevel, outerController.signal);
38
- logger.enrichReserved({
37
+ const logger = makeLogger(logTransport, meta?.config?.minimumLogLevel, outerController.signal).enrichReserved({
39
38
  operationId: clientInfo.operationId,
40
39
  client: {
41
40
  id: clientInfo.clientId,
@@ -100,4 +99,4 @@ process.on('uncaughtException', err => {
100
99
  process.on('unhandledRejection', reason => {
101
100
  globalLogger?.fatal('Unhandled rejection.', reason, undefined);
102
101
  });
103
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":";;;;;;;;;;;;AACA,OAAO,EAAE,YAAY,EAAE,MAAM,cAAc,CAAA;AAC3C,OAAO,EAAE,UAAU,EAAE,MAAM,cAAc,CAAA;AA6CzC,MAAM,cAAc;IAIhB,YAAY,UAA0B;QAHtC,6CAA2B;QAIvB,uBAAA,IAAI,8BAAe,UAAU,MAAA,CAAA;QAC7B,IAAI,CAAC,WAAW,GAAG,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,IAAI,MAAM,CAAC,gBAAgB,CAAA;IAC9F,CAAC;IAED,WAAW,CAAC,OAAmB,EAAE,MAAmB;QAChD,MAAM,QAAQ,GAAG,uBAAA,IAAI,kCAAY,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAA;QAC3F,IAAI,QAAQ,CAAC,MAAM,KAAK,CAAC,EAAE;YACvB,OAAM;SACT;QACD,OAAO,OAAO,CAAC,GAAG,CAAC,QAAQ,CAA6B,CAAA;IAC5D,CAAC;CACJ;;AAOD,MAAM,UAAU,aAAa,CACzB,UAAsB,EACtB,OAAuB,EACvB,cAA8B,EAC9B,QAA2C,EAC3C,eAAgC,EAChC,IAAe,EACf,WAAqC,EACrC,GAA8B;IAO9B,MAAM,OAAO,GACT,CAAC,QAAQ,CAAC,GAAG;QACT,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,IAAI,QAAQ,CAAC,OAAO,EAAE,QAAQ,CAAC,GAAG,CAAC;QACnE,CAAC,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,IAAI,QAAQ,CAAC,OAAO,CAAC,GAAG,IAAI,CAAA;IAC3D,MAAM,eAAe,GAAG,IAAI,eAAe,EAAE,CAAA;IAC7C,oEAAoE;IACpE,MAAM,YAAY,GAAG,OAAO,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAE,CAAC,CAAC,CAAC,IAAI,cAAc,CAAC,OAAO,CAAC,CAAA;IACrF,MAAM,MAAM,GAAG,UAAU,CACrB,YAAY,EACZ,IAAI,EAAE,MAAM,EAAE,eAAe,EAC7B,eAAe,CAAC,MAAqB,CACxC,CAAA;IACD,MAAM,CAAC,cAAc,CAAC;QAClB,WAAW,EAAE,UAAU,CAAC,WAAW;QACnC,MAAM,EAAE;YACJ,EAAE,EAAE,UAAU,CAAC,QAAQ;YACvB,EAAE,EAAE,UAAU,CAAC,QAAQ;YACvB,IAAI,EAAE,UAAU,CAAC,UAAU;YAC3B,SAAS,EAAE,UAAU,CAAC,SAAS;SAClC;KACJ,CAAC,CAAA;IACF,YAAY,GAAG,MAAM,CAAA;IACrB,MAAM,OAAO,GAAG,IAAI,YAAY,CAC5B,cAAc,EACd,MAAM,EACN,UAAU,EACV,OAAO,EACP,eAAe,CAAC,MAAqB,CACxC,CAAA;IACD,MAAM,eAAe,GAAmC,EAAE,CAAA;IAC1D,MAAM,GAAG,GAAG;QACR,GAAG,EAAE,WAAW,IAAK,OAAO,CAAC,GAAmB;QAChD,MAAM,EAAE,eAAe,CAAC,MAAqB;QAC7C,GAAG,EAAE,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC,IAAI,IAAI,EAAE,CAAC;QAC9B,WAAW,EAAE,UAAU,CAAC,WAAW;QACnC,MAAM,EAAE;YACJ,EAAE,EAAE,UAAU,CAAC,QAAQ;YACvB,EAAE,EAAE,UAAU,CAAC,QAAQ;YACvB,IAAI,EAAE,UAAU,CAAC,UAAU;YAC3B,SAAS,EAAE,UAAU,CAAC,SAAS;SAClC;QACD,IAAI,EAAE,IAAI;YACN,CAAC,CAAC;gBACI,WAAW,EAAE,IAAI,CAAC,WAAW;gBAC7B,QAAQ,EAAE,IAAI,CAAC,QAAQ;gBACvB,QAAQ,EAAE,IAAI,CAAC,QAAQ;aAC1B;YACH,CAAC,CAAC,SAAS;QACf,IAAI,EAAE,CAAC,KAAa,EAAE,IAAY,EAAE,OAAe,EAAE,IAAW,EAAE,SAAkB,EAAE,EAAE,CACpF,OAAO,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,OAAO,EAAE,EAAE,EAAE,SAAS,EAAE,EAAE,IAAI,CAAC;QAC/D,YAAY,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,KAAK,EAAE;QACnC,SAAS,EAAE,CAAC,EAA8B,EAAE,EAAE,CAAC,eAAe,CAAC,IAAI,CAAC,EAAE,CAAC;KAC1E,CAAA;IACD,MAAM,aAAa,GAAG,UAAU,CAAC,GAAG,EAAE;QAClC,MAAM,CAAC,KAAK,CAAC,UAAU,EAAE,SAAS,EAAE,SAAS,CAAC,CAAA;QAC9C,eAAe,CAAC,KAAK,EAAE,CAAA;QACvB,mCAAmC;QACnC,KAAK,MAAM,CAAC,KAAK,EAAE,CAAA;QACnB,mCAAmC;QACnC,KAAK,OAAO,CAAC,KAAK,EAAE,CAAA;IACxB,CAAC,EAAE,OAAO,CAAC,CAAA;IACX,MAAM,WAAW,GAAG,UAAU,CAAC,GAAG,EAAE;QAChC,MAAM,CAAC,KAAK,CAAC,iBAAiB,EAAE,SAAS,EAAE,SAAS,CAAC,CAAA;QACrD,eAAe,CAAC,KAAK,EAAE,CAAA;IAC3B,CAAC,EAAE,OAAO,GAAG,KAAK,CAAC,CAAA;IACnB,OAAO;QACH,GAAG,EAAE,MAAM;QACX,OAAO,EAAE,GAAG;QACZ,OAAO,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,eAAe,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC;QAC3D,KAAK,EAAE,KAAK,IAAI,EAAE;YACd,YAAY,CAAC,aAAa,CAAC,CAAA;YAC3B,MAAM,OAAO,CAAC,KAAK,EAAE,CAAA;YACrB,MAAM,MAAM,CAAC,KAAK,EAAE,CAAA;YACpB,YAAY,CAAC,WAAW,CAAC,CAAA;QAC7B,CAAC;KACJ,CAAA;AACL,CAAC;AAED,IAAI,YAAgC,CAAA;AAEpC,OAAO,CAAC,EAAE,CAAC,mBAAmB,EAAE,GAAG,CAAC,EAAE;IAClC,YAAY,EAAE,KAAK,CAAC,qBAAqB,EAAE,GAAG,EAAE,SAAS,CAAC,CAAA;AAC9D,CAAC,CAAC,CAAA;AACF,OAAO,CAAC,EAAE,CAAC,oBAAoB,EAAE,MAAM,CAAC,EAAE;IACtC,YAAY,EAAE,KAAK,CAAC,sBAAsB,EAAE,MAAM,EAAE,SAAS,CAAC,CAAA;AAClE,CAAC,CAAC,CAAA","sourcesContent":["import { AbortSignal, Context, Environment, Json, Logger } from '../context.js'\r\nimport { EventEmitter } from './emitter.js'\r\nimport { makeLogger } from './logging.js'\r\nimport { Metadata } from './registry.js'\r\n\r\nexport interface ClientInfo {\r\n    readonly operationId?: string\r\n    readonly clientId?: string\r\n    readonly clientIp?: string\r\n    readonly clientPort?: number\r\n    readonly userAgent?: string\r\n}\r\n\r\nexport interface EventMetadata {\r\n    topic: string\r\n    type: string\r\n    subject: string\r\n    id?: string\r\n}\r\n\r\nexport interface BufferedEvent {\r\n    eventTime: Date\r\n    meta: Omit<EventMetadata, 'topic'>\r\n    ids: ClientInfo\r\n    json?: string\r\n}\r\n\r\nexport interface EventTransport {\r\n    readonly publishRate: number\r\n    sendEvents(topic: string, events: BufferedEvent[], signal: AbortSignal): Promise<void>\r\n}\r\n\r\nexport type LogLevel = 'trace' | 'debug' | 'info' | 'warning' | 'error' | 'fatal'\r\n\r\nexport interface LogEntry {\r\n    readonly level: LogLevel\r\n    readonly timestamp: number\r\n    readonly message: string\r\n    readonly error: unknown\r\n    readonly json: string\r\n}\r\n\r\nexport interface LogTransport {\r\n    readonly publishRate?: number\r\n    sendEntries(entries: LogEntry[], signal: AbortSignal): Promise<void> | undefined\r\n}\r\n\r\nclass LogMulticaster implements LogTransport {\r\n    #transports: LogTransport[]\r\n    readonly publishRate: number\r\n\r\n    constructor(transports: LogTransport[]) {\r\n        this.#transports = transports\r\n        this.publishRate = transports.map(t => t.publishRate).sort()[0] ?? Number.MAX_SAFE_INTEGER\r\n    }\r\n\r\n    sendEntries(entries: LogEntry[], signal: AbortSignal) {\r\n        const promises = this.#transports.map(t => t.sendEntries(entries, signal)).filter(p => !!p)\r\n        if (promises.length === 0) {\r\n            return\r\n        }\r\n        return Promise.all(promises) as unknown as Promise<void>\r\n    }\r\n}\r\n\r\nexport interface RootLogger extends Logger {\r\n    enrichReserved(fields: object): RootLogger\r\n    flush(): Promise<void>\r\n}\r\n\r\nexport function createContext(\r\n    clientInfo: ClientInfo,\r\n    loggers: LogTransport[],\r\n    eventTransport: EventTransport,\r\n    timeouts: { default: number; cap?: number },\r\n    outerController: AbortController,\r\n    meta?: Metadata,\r\n    environment?: Environment | undefined,\r\n    now?: (() => Date) | undefined,\r\n): {\r\n    log: RootLogger\r\n    context: Omit<Context, 'log'>\r\n    success: () => Promise<unknown>\r\n    flush: () => Promise<void>\r\n} {\r\n    const timeout =\r\n        (timeouts.cap\r\n            ? Math.min(meta?.config?.timeout ?? timeouts.default, timeouts.cap)\r\n            : meta?.config?.timeout ?? timeouts.default) * 1000\r\n    const innerController = new AbortController()\r\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\r\n    const logTransport = loggers.length === 1 ? loggers[0]! : new LogMulticaster(loggers)\r\n    const logger = makeLogger(\r\n        logTransport,\r\n        meta?.config?.minimumLogLevel,\r\n        outerController.signal as AbortSignal,\r\n    )\r\n    logger.enrichReserved({\r\n        operationId: clientInfo.operationId,\r\n        client: {\r\n            id: clientInfo.clientId,\r\n            ip: clientInfo.clientIp,\r\n            port: clientInfo.clientPort,\r\n            userAgent: clientInfo.userAgent,\r\n        },\r\n    })\r\n    globalLogger = logger\r\n    const emitter = new EventEmitter(\r\n        eventTransport,\r\n        logger,\r\n        clientInfo,\r\n        timeout,\r\n        outerController.signal as AbortSignal,\r\n    )\r\n    const successHandlers: (() => Promise<void> | void)[] = []\r\n    const ctx = {\r\n        env: environment ?? (process.env as Environment),\r\n        signal: innerController.signal as AbortSignal,\r\n        now: now ?? (() => new Date()),\r\n        operationId: clientInfo.operationId,\r\n        client: {\r\n            id: clientInfo.clientId,\r\n            ip: clientInfo.clientIp,\r\n            port: clientInfo.clientPort,\r\n            userAgent: clientInfo.userAgent,\r\n        },\r\n        meta: meta\r\n            ? {\r\n                  packageName: meta.packageName,\r\n                  fileName: meta.fileName,\r\n                  revision: meta.revision,\r\n              }\r\n            : undefined,\r\n        emit: (topic: string, type: string, subject: string, data?: Json, messageId?: string) =>\r\n            emitter.emit({ topic, type, subject, id: messageId }, data),\r\n        eventBarrier: () => emitter.flush(),\r\n        onSuccess: (fn: () => Promise<void> | void) => successHandlers.push(fn),\r\n    }\r\n    const timeoutHandle = setTimeout(() => {\r\n        logger.error('Timeout.', undefined, undefined)\r\n        innerController.abort()\r\n        // eslint-disable-next-line no-void\r\n        void logger.flush()\r\n        // eslint-disable-next-line no-void\r\n        void emitter.flush()\r\n    }, timeout)\r\n    const flushHandle = setTimeout(() => {\r\n        logger.error('Aborting flush.', undefined, undefined)\r\n        outerController.abort()\r\n    }, timeout + 15000)\r\n    return {\r\n        log: logger,\r\n        context: ctx,\r\n        success: () => Promise.all(successHandlers.map(fn => fn())),\r\n        flush: async () => {\r\n            clearTimeout(timeoutHandle)\r\n            await emitter.flush()\r\n            await logger.flush()\r\n            clearTimeout(flushHandle)\r\n        },\r\n    }\r\n}\r\n\r\nlet globalLogger: Logger | undefined\r\n\r\nprocess.on('uncaughtException', err => {\r\n    globalLogger?.fatal('Uncaught exception.', err, undefined)\r\n})\r\nprocess.on('unhandledRejection', reason => {\r\n    globalLogger?.fatal('Unhandled rejection.', reason, undefined)\r\n})\r\n"]}
102
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbortSignal, Context, Environment, Json, Logger } from '../context.js'\r\nimport { EventEmitter } from './emitter.js'\r\nimport { makeLogger } from './logging.js'\r\nimport { Metadata } from './registry.js'\r\n\r\nexport interface ClientInfo {\r\n    readonly operationId?: string\r\n    readonly clientId?: string\r\n    readonly clientIp?: string\r\n    readonly clientPort?: number\r\n    readonly userAgent?: string\r\n}\r\n\r\nexport interface EventMetadata {\r\n    topic: string\r\n    type: string\r\n    subject: string\r\n    id?: string\r\n}\r\n\r\nexport interface BufferedEvent {\r\n    eventTime: Date\r\n    meta: Omit<EventMetadata, 'topic'>\r\n    ids: ClientInfo\r\n    json?: string\r\n}\r\n\r\nexport interface EventTransport {\r\n    readonly publishRate: number\r\n    sendEvents(topic: string, events: BufferedEvent[], signal: AbortSignal): Promise<void>\r\n}\r\n\r\nexport type LogLevel = 'trace' | 'debug' | 'info' | 'warning' | 'error' | 'fatal'\r\n\r\nexport interface LogEntry {\r\n    readonly level: LogLevel\r\n    readonly timestamp: number\r\n    readonly message: string\r\n    readonly error: unknown\r\n    readonly json: string\r\n}\r\n\r\nexport interface LogTransport {\r\n    readonly publishRate?: number\r\n    sendEntries(entries: LogEntry[], signal: AbortSignal): Promise<void> | undefined\r\n}\r\n\r\nclass LogMulticaster implements LogTransport {\r\n    #transports: LogTransport[]\r\n    readonly publishRate: number\r\n\r\n    constructor(transports: LogTransport[]) {\r\n        this.#transports = transports\r\n        this.publishRate = transports.map(t => t.publishRate).sort()[0] ?? Number.MAX_SAFE_INTEGER\r\n    }\r\n\r\n    sendEntries(entries: LogEntry[], signal: AbortSignal) {\r\n        const promises = this.#transports.map(t => t.sendEntries(entries, signal)).filter(p => !!p)\r\n        if (promises.length === 0) {\r\n            return\r\n        }\r\n        return Promise.all(promises) as unknown as Promise<void>\r\n    }\r\n}\r\n\r\nexport interface RootLogger extends Logger {\r\n    enrichReserved(fields: object): RootLogger\r\n    flush(): Promise<void>\r\n}\r\n\r\nexport function createContext(\r\n    clientInfo: ClientInfo,\r\n    loggers: LogTransport[],\r\n    eventTransport: EventTransport,\r\n    timeouts: { default: number; cap?: number },\r\n    outerController: AbortController,\r\n    meta?: Metadata,\r\n    environment?: Environment | undefined,\r\n    now?: (() => Date) | undefined,\r\n): {\r\n    log: RootLogger\r\n    context: Omit<Context, 'log'>\r\n    success: () => Promise<unknown>\r\n    flush: () => Promise<void>\r\n} {\r\n    const timeout =\r\n        (timeouts.cap\r\n            ? Math.min(meta?.config?.timeout ?? timeouts.default, timeouts.cap)\r\n            : meta?.config?.timeout ?? timeouts.default) * 1000\r\n    const innerController = new AbortController()\r\n    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\r\n    const logTransport = loggers.length === 1 ? loggers[0]! : new LogMulticaster(loggers)\r\n    const logger = makeLogger(\r\n        logTransport,\r\n        meta?.config?.minimumLogLevel,\r\n        outerController.signal as AbortSignal,\r\n    ).enrichReserved({\r\n        operationId: clientInfo.operationId,\r\n        client: {\r\n            id: clientInfo.clientId,\r\n            ip: clientInfo.clientIp,\r\n            port: clientInfo.clientPort,\r\n            userAgent: clientInfo.userAgent,\r\n        },\r\n    })\r\n    globalLogger = logger\r\n    const emitter = new EventEmitter(\r\n        eventTransport,\r\n        logger,\r\n        clientInfo,\r\n        timeout,\r\n        outerController.signal as AbortSignal,\r\n    )\r\n    const successHandlers: (() => Promise<void> | void)[] = []\r\n    const ctx = {\r\n        env: environment ?? (process.env as Environment),\r\n        signal: innerController.signal as AbortSignal,\r\n        now: now ?? (() => new Date()),\r\n        operationId: clientInfo.operationId,\r\n        client: {\r\n            id: clientInfo.clientId,\r\n            ip: clientInfo.clientIp,\r\n            port: clientInfo.clientPort,\r\n            userAgent: clientInfo.userAgent,\r\n        },\r\n        meta: meta\r\n            ? {\r\n                  packageName: meta.packageName,\r\n                  fileName: meta.fileName,\r\n                  revision: meta.revision,\r\n              }\r\n            : undefined,\r\n        emit: (topic: string, type: string, subject: string, data?: Json, messageId?: string) =>\r\n            emitter.emit({ topic, type, subject, id: messageId }, data),\r\n        eventBarrier: () => emitter.flush(),\r\n        onSuccess: (fn: () => Promise<void> | void) => successHandlers.push(fn),\r\n    }\r\n    const timeoutHandle = setTimeout(() => {\r\n        logger.error('Timeout.', undefined, undefined)\r\n        innerController.abort()\r\n        // eslint-disable-next-line no-void\r\n        void logger.flush()\r\n        // eslint-disable-next-line no-void\r\n        void emitter.flush()\r\n    }, timeout)\r\n    const flushHandle = setTimeout(() => {\r\n        logger.error('Aborting flush.', undefined, undefined)\r\n        outerController.abort()\r\n    }, timeout + 15000)\r\n    return {\r\n        log: logger,\r\n        context: ctx,\r\n        success: () => Promise.all(successHandlers.map(fn => fn())),\r\n        flush: async () => {\r\n            clearTimeout(timeoutHandle)\r\n            await emitter.flush()\r\n            await logger.flush()\r\n            clearTimeout(flushHandle)\r\n        },\r\n    }\r\n}\r\n\r\nlet globalLogger: Logger | undefined\r\n\r\nprocess.on('uncaughtException', err => {\r\n    globalLogger?.fatal('Uncaught exception.', err, undefined)\r\n})\r\nprocess.on('unhandledRejection', reason => {\r\n    globalLogger?.fatal('Unhandled rejection.', reason, undefined)\r\n})\r\n"]}
package/host/http.js CHANGED
@@ -25,7 +25,7 @@ export async function executeRequest(log, context, handler, options, success) {
25
25
  ...parse(this.rawUrl, true),
26
26
  pathStepAt: (index) => {
27
27
  const steps = (pathSteps ??= parsedUrl.pathname?.split('/') ?? []);
28
- const step = steps[index];
28
+ const step = steps[index - 1];
29
29
  if (!step) {
30
30
  throw new RangeError(`Path does not have a step at index ${index}.`);
31
31
  }
@@ -202,4 +202,4 @@ export function clientFromHeaders(headers) {
202
202
  userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],
203
203
  };
204
204
  }
205
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http.js","sourceRoot":"","sources":["http.ts"],"names":[],"mappings":"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","sourcesContent":["import { parse, UrlWithParsedQuery } from 'node:url'\r\nimport { Context, measure } from '../context.js'\r\nimport type { Json, ResponseHeaders, Result } from '../http.js'\r\nimport { ClientInfo, RootLogger } from './context.js'\r\nimport type { HttpHandler } from './registry.js'\r\n\r\nexport interface Response {\r\n    headers: { readonly [key: string]: string }\r\n    status: number\r\n    body?: string | Buffer\r\n}\r\n\r\ntype RequestOptions = BodylessRequestOptions | StringRequestOptions | JsonRequestOptions\r\n\r\ninterface BodylessRequestOptions {\r\n    uri: string\r\n    headers?: { readonly [key: string]: string }\r\n}\r\n\r\ninterface StringRequestOptions extends BodylessRequestOptions {\r\n    body: string\r\n}\r\n\r\ninterface JsonRequestOptions extends BodylessRequestOptions {\r\n    json: Json\r\n}\r\n\r\nexport async function executeRequest(\r\n    log: RootLogger,\r\n    context: Omit<Context, 'log'>,\r\n    handler: HttpHandler,\r\n    options: RequestOptions,\r\n    success: () => Promise<unknown>,\r\n): Promise<Response> {\r\n    const isShallow =\r\n        context.env.SHALLOW_KEY && options.headers?.['x-shallow'] === context.env.SHALLOW_KEY\r\n    const includeBodyInLogs = !handler.config?.excludeBodyFromLogs\r\n    const logRequest = includeBodyInLogs\r\n        ? { method: handler.method, ...options }\r\n        : withoutRequestBody({ method: handler.method, ...options })\r\n    log = log.enrichReserved({ meta: context.meta, request: logRequest })\r\n    if (isShallow) {\r\n        log.trace('Shallow request')\r\n        return {\r\n            headers: {},\r\n            status: 204,\r\n        }\r\n    }\r\n    log.trace('Request BEGIN')\r\n    try {\r\n        let parsedUrl: UrlWithParsedQuery & { pathStepAt: (index: number) => string }\r\n        let pathSteps: string[]\r\n        const req = {\r\n            rawUrl: options.uri,\r\n            get url() {\r\n                return (parsedUrl ??= {\r\n                    ...parse(this.rawUrl, true),\r\n                    pathStepAt: (index: number) => {\r\n                        const steps = (pathSteps ??= parsedUrl.pathname?.split('/') ?? [])\r\n                        const step = steps[index]\r\n                        if (!step) {\r\n                            throw new RangeError(`Path does not have a step at index ${index}.`)\r\n                        }\r\n                        return step\r\n                    },\r\n                })\r\n            },\r\n            body: requestBody(options),\r\n            headers: options.headers ?? {},\r\n        }\r\n\r\n        const result = await measure(log, 'execution', () =>\r\n            handler.entry({ ...context, log }, req),\r\n        )\r\n\r\n        const response = resultToResponse(result, includeBodyInLogs)\r\n\r\n        if (context.signal.aborted) {\r\n            response.headers = {\r\n                'x-timeout': '1',\r\n                ...response.headers,\r\n            }\r\n        }\r\n\r\n        log = log.enrichReserved({ request: logRequest })\r\n        if (response.status < 300) {\r\n            log.debug('Request END')\r\n            await success()\r\n        } else {\r\n            log.warn('Request END')\r\n        }\r\n        return response\r\n    } catch (e) {\r\n        try {\r\n            const response = errorToResponse(e)\r\n            log = log.enrichReserved({ response })\r\n            log.error('Request END', e)\r\n            return response\r\n        } catch (convertError) {\r\n            log.error('Could not convert exception to error response.', convertError)\r\n            return {\r\n                headers: {},\r\n                status: 500,\r\n            }\r\n        }\r\n    }\r\n}\r\n\r\nfunction resultToResponse(result: Result, withLogBody: boolean): Response & { logBody?: unknown } {\r\n    if (!result) {\r\n        return {\r\n            headers: {},\r\n            status: 204,\r\n        }\r\n    } else if (typeof result === 'string') {\r\n        const logBody = withLogBody ? result : undefined\r\n        return {\r\n            headers: {\r\n                'content-type': 'text/plain',\r\n            },\r\n            status: 200,\r\n            body: result,\r\n            logBody,\r\n        }\r\n    } else {\r\n        if (result.body === undefined) {\r\n            return {\r\n                headers: result.headers ?? {},\r\n                status: result.status ?? 200,\r\n            }\r\n        } else if (typeof result.body === 'string') {\r\n            const logBody = withLogBody ? result.body : undefined\r\n            return {\r\n                headers: withContentType(result.headers, 'text/plain'),\r\n                status: result.status ?? 200,\r\n                body: result.body,\r\n                logBody,\r\n            }\r\n        } else if (Buffer.isBuffer(result.body)) {\r\n            const logBody = withLogBody ? result.body.toString('base64') : undefined\r\n            return {\r\n                headers: withContentType(result.headers, 'application/octet-stream'),\r\n                status: result.status ?? 200,\r\n                body: result.body,\r\n                logBody,\r\n            }\r\n        } else {\r\n            const logBody = withLogBody ? result.body : undefined\r\n            return {\r\n                headers: withContentType(result.headers, 'application/json'),\r\n                status: result.status ?? 200,\r\n                body: JSON.stringify(result.body),\r\n                logBody,\r\n            }\r\n        }\r\n    }\r\n}\r\n\r\nfunction withoutRequestBody(options: RequestOptions & { method: string }) {\r\n    if (hasJsonBody(options)) {\r\n        const { json, ...bodyless } = options\r\n        return bodyless\r\n    }\r\n    if (hasStringBody(options)) {\r\n        const { body, ...bodyless } = options\r\n        return bodyless\r\n    }\r\n    return options\r\n}\r\n\r\nfunction requestBody(options: RequestOptions): Json | string | undefined {\r\n    if (hasJsonBody(options)) {\r\n        return options.json\r\n    }\r\n    if (hasStringBody(options)) {\r\n        return options.body\r\n    }\r\n    return undefined\r\n}\r\n\r\nfunction hasJsonBody(options: RequestOptions): options is JsonRequestOptions {\r\n    return (options as { json?: unknown }).json !== undefined\r\n}\r\n\r\nfunction hasStringBody(options: RequestOptions): options is StringRequestOptions {\r\n    return (options as { body?: unknown }).body !== undefined\r\n}\r\n\r\nfunction withContentType(headers: ResponseHeaders | undefined, contentType: string) {\r\n    if (!headers) {\r\n        return {\r\n            'content-type': contentType,\r\n        }\r\n    }\r\n    if (!headers['content-type']) {\r\n        headers['content-type'] = contentType\r\n    }\r\n    return headers\r\n}\r\n\r\nfunction errorToResponse(e: unknown): Response {\r\n    const { body, statusCode: status } = e as { body?: unknown; statusCode?: number }\r\n    if (typeof body === 'string') {\r\n        return {\r\n            headers: {\r\n                'content-type': 'text/plain',\r\n            },\r\n            status: status ?? 500,\r\n            body,\r\n        }\r\n    } else if (typeof body === 'object') {\r\n        return {\r\n            headers: {\r\n                'content-type': 'application/json',\r\n            },\r\n            status: status ?? 500,\r\n            body: JSON.stringify(body),\r\n        }\r\n    } else {\r\n        return {\r\n            headers: {},\r\n            status: status ?? 500,\r\n        }\r\n    }\r\n}\r\n\r\nexport function clientFromHeaders(\r\n    headers: { readonly [key: string]: string } | undefined,\r\n): ClientInfo {\r\n    if (!headers) {\r\n        return {}\r\n    }\r\n    return {\r\n        operationId: headers['x-request-id'] ?? headers['request-id'],\r\n        clientId:\r\n            headers['x-client-id'] ??\r\n            headers['x-installation-id'] ??\r\n            headers['client-id'] ??\r\n            headers['installation-id'],\r\n        clientIp: headers['x-forwarded-for'],\r\n        userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],\r\n    }\r\n}\r\n"]}
205
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"http.js","sourceRoot":"","sources":["http.ts"],"names":[],"mappings":"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","sourcesContent":["import { parse, UrlWithParsedQuery } from 'node:url'\r\nimport { Context, measure } from '../context.js'\r\nimport type { Json, ResponseHeaders, Result } from '../http.js'\r\nimport { ClientInfo, RootLogger } from './context.js'\r\nimport type { HttpHandler } from './registry.js'\r\n\r\nexport interface Response {\r\n    headers: { readonly [key: string]: string }\r\n    status: number\r\n    body?: string | Buffer\r\n}\r\n\r\ntype RequestOptions = BodylessRequestOptions | StringRequestOptions | JsonRequestOptions\r\n\r\ninterface BodylessRequestOptions {\r\n    uri: string\r\n    headers?: { readonly [key: string]: string }\r\n}\r\n\r\ninterface StringRequestOptions extends BodylessRequestOptions {\r\n    body: string\r\n}\r\n\r\ninterface JsonRequestOptions extends BodylessRequestOptions {\r\n    json: Json\r\n}\r\n\r\nexport async function executeRequest(\r\n    log: RootLogger,\r\n    context: Omit<Context, 'log'>,\r\n    handler: HttpHandler,\r\n    options: RequestOptions,\r\n    success: () => Promise<unknown>,\r\n): Promise<Response> {\r\n    const isShallow =\r\n        context.env.SHALLOW_KEY && options.headers?.['x-shallow'] === context.env.SHALLOW_KEY\r\n    const includeBodyInLogs = !handler.config?.excludeBodyFromLogs\r\n    const logRequest = includeBodyInLogs\r\n        ? { method: handler.method, ...options }\r\n        : withoutRequestBody({ method: handler.method, ...options })\r\n    log = log.enrichReserved({ meta: context.meta, request: logRequest })\r\n    if (isShallow) {\r\n        log.trace('Shallow request')\r\n        return {\r\n            headers: {},\r\n            status: 204,\r\n        }\r\n    }\r\n    log.trace('Request BEGIN')\r\n    try {\r\n        let parsedUrl: UrlWithParsedQuery & { pathStepAt: (index: number) => string }\r\n        let pathSteps: string[]\r\n        const req = {\r\n            rawUrl: options.uri,\r\n            get url() {\r\n                return (parsedUrl ??= {\r\n                    ...parse(this.rawUrl, true),\r\n                    pathStepAt: (index: number) => {\r\n                        const steps = (pathSteps ??= parsedUrl.pathname?.split('/') ?? [])\r\n                        const step = steps[index - 1]\r\n                        if (!step) {\r\n                            throw new RangeError(`Path does not have a step at index ${index}.`)\r\n                        }\r\n                        return step\r\n                    },\r\n                })\r\n            },\r\n            body: requestBody(options),\r\n            headers: options.headers ?? {},\r\n        }\r\n\r\n        const result = await measure(log, 'execution', () =>\r\n            handler.entry({ ...context, log }, req),\r\n        )\r\n\r\n        const response = resultToResponse(result, includeBodyInLogs)\r\n\r\n        if (context.signal.aborted) {\r\n            response.headers = {\r\n                'x-timeout': '1',\r\n                ...response.headers,\r\n            }\r\n        }\r\n\r\n        log = log.enrichReserved({ request: logRequest })\r\n        if (response.status < 300) {\r\n            log.debug('Request END')\r\n            await success()\r\n        } else {\r\n            log.warn('Request END')\r\n        }\r\n        return response\r\n    } catch (e) {\r\n        try {\r\n            const response = errorToResponse(e)\r\n            log = log.enrichReserved({ response })\r\n            log.error('Request END', e)\r\n            return response\r\n        } catch (convertError) {\r\n            log.error('Could not convert exception to error response.', convertError)\r\n            return {\r\n                headers: {},\r\n                status: 500,\r\n            }\r\n        }\r\n    }\r\n}\r\n\r\nfunction resultToResponse(result: Result, withLogBody: boolean): Response & { logBody?: unknown } {\r\n    if (!result) {\r\n        return {\r\n            headers: {},\r\n            status: 204,\r\n        }\r\n    } else if (typeof result === 'string') {\r\n        const logBody = withLogBody ? result : undefined\r\n        return {\r\n            headers: {\r\n                'content-type': 'text/plain',\r\n            },\r\n            status: 200,\r\n            body: result,\r\n            logBody,\r\n        }\r\n    } else {\r\n        if (result.body === undefined) {\r\n            return {\r\n                headers: result.headers ?? {},\r\n                status: result.status ?? 200,\r\n            }\r\n        } else if (typeof result.body === 'string') {\r\n            const logBody = withLogBody ? result.body : undefined\r\n            return {\r\n                headers: withContentType(result.headers, 'text/plain'),\r\n                status: result.status ?? 200,\r\n                body: result.body,\r\n                logBody,\r\n            }\r\n        } else if (Buffer.isBuffer(result.body)) {\r\n            const logBody = withLogBody ? result.body.toString('base64') : undefined\r\n            return {\r\n                headers: withContentType(result.headers, 'application/octet-stream'),\r\n                status: result.status ?? 200,\r\n                body: result.body,\r\n                logBody,\r\n            }\r\n        } else {\r\n            const logBody = withLogBody ? result.body : undefined\r\n            return {\r\n                headers: withContentType(result.headers, 'application/json'),\r\n                status: result.status ?? 200,\r\n                body: JSON.stringify(result.body),\r\n                logBody,\r\n            }\r\n        }\r\n    }\r\n}\r\n\r\nfunction withoutRequestBody(options: RequestOptions & { method: string }) {\r\n    if (hasJsonBody(options)) {\r\n        const { json, ...bodyless } = options\r\n        return bodyless\r\n    }\r\n    if (hasStringBody(options)) {\r\n        const { body, ...bodyless } = options\r\n        return bodyless\r\n    }\r\n    return options\r\n}\r\n\r\nfunction requestBody(options: RequestOptions): Json | string | undefined {\r\n    if (hasJsonBody(options)) {\r\n        return options.json\r\n    }\r\n    if (hasStringBody(options)) {\r\n        return options.body\r\n    }\r\n    return undefined\r\n}\r\n\r\nfunction hasJsonBody(options: RequestOptions): options is JsonRequestOptions {\r\n    return (options as { json?: unknown }).json !== undefined\r\n}\r\n\r\nfunction hasStringBody(options: RequestOptions): options is StringRequestOptions {\r\n    return (options as { body?: unknown }).body !== undefined\r\n}\r\n\r\nfunction withContentType(headers: ResponseHeaders | undefined, contentType: string) {\r\n    if (!headers) {\r\n        return {\r\n            'content-type': contentType,\r\n        }\r\n    }\r\n    if (!headers['content-type']) {\r\n        headers['content-type'] = contentType\r\n    }\r\n    return headers\r\n}\r\n\r\nfunction errorToResponse(e: unknown): Response {\r\n    const { body, statusCode: status } = e as { body?: unknown; statusCode?: number }\r\n    if (typeof body === 'string') {\r\n        return {\r\n            headers: {\r\n                'content-type': 'text/plain',\r\n            },\r\n            status: status ?? 500,\r\n            body,\r\n        }\r\n    } else if (typeof body === 'object') {\r\n        return {\r\n            headers: {\r\n                'content-type': 'application/json',\r\n            },\r\n            status: status ?? 500,\r\n            body: JSON.stringify(body),\r\n        }\r\n    } else {\r\n        return {\r\n            headers: {},\r\n            status: status ?? 500,\r\n        }\r\n    }\r\n}\r\n\r\nexport function clientFromHeaders(\r\n    headers: { readonly [key: string]: string } | undefined,\r\n): ClientInfo {\r\n    if (!headers) {\r\n        return {}\r\n    }\r\n    return {\r\n        operationId: headers['x-request-id'] ?? headers['request-id'],\r\n        clientId:\r\n            headers['x-client-id'] ??\r\n            headers['x-installation-id'] ??\r\n            headers['client-id'] ??\r\n            headers['installation-id'],\r\n        clientIp: headers['x-forwarded-for'],\r\n        userAgent: headers['x-forwarded-for-user-agent'] ?? headers['user-agent'],\r\n    }\r\n}\r\n"]}
package/host/logging.js CHANGED
@@ -161,13 +161,13 @@ class EnrichingLogger {
161
161
  if (__classPrivateFieldGet(this, _EnrichingLogger_level, "f") < 3) {
162
162
  return;
163
163
  }
164
- __classPrivateFieldGet(this, _EnrichingLogger_buffer, "f").collect('debug', 3, message, error, fields, __classPrivateFieldGet(this, _EnrichingLogger_reservedEnrichment, "f"), __classPrivateFieldGet(this, _EnrichingLogger_customEnrichment, "f"));
164
+ __classPrivateFieldGet(this, _EnrichingLogger_buffer, "f").collect('info', 3, message, error, fields, __classPrivateFieldGet(this, _EnrichingLogger_reservedEnrichment, "f"), __classPrivateFieldGet(this, _EnrichingLogger_customEnrichment, "f"));
165
165
  }
166
166
  warn(message, error, fields) {
167
167
  if (__classPrivateFieldGet(this, _EnrichingLogger_level, "f") < 2) {
168
168
  return;
169
169
  }
170
- __classPrivateFieldGet(this, _EnrichingLogger_buffer, "f").collect('debug', 2, message, error, fields, __classPrivateFieldGet(this, _EnrichingLogger_reservedEnrichment, "f"), __classPrivateFieldGet(this, _EnrichingLogger_customEnrichment, "f"));
170
+ __classPrivateFieldGet(this, _EnrichingLogger_buffer, "f").collect('warning', 2, message, error, fields, __classPrivateFieldGet(this, _EnrichingLogger_reservedEnrichment, "f"), __classPrivateFieldGet(this, _EnrichingLogger_customEnrichment, "f"));
171
171
  }
172
172
  error(message, error, fields) {
173
173
  if (__classPrivateFieldGet(this, _EnrichingLogger_level, "f") < 1) {
@@ -202,4 +202,4 @@ function errorAsJson(error) {
202
202
  name: typeof error,
203
203
  };
204
204
  }
205
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logging.js","sourceRoot":"","sources":["logging.ts"],"names":[],"mappings":"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","sourcesContent":["import { performance } from 'perf_hooks'\r\nimport { AbortSignal, Json, Logger } from '../context.js'\r\nimport { LogEntry, LogLevel, LogTransport, RootLogger } from './context.js'\r\n\r\nexport function makeLogger(\r\n    transport: LogTransport,\r\n    minimumLogLevel: LogLevel | undefined,\r\n    signal: AbortSignal,\r\n): RootLogger {\r\n    return new EnrichingLogger(\r\n        new LogBuffer(transport, signal),\r\n        minimumLogLevel\r\n            ? ['fatal', 'error', 'warning', 'info', 'debug', 'trace'].indexOf(minimumLogLevel)\r\n            : 5,\r\n        signal,\r\n    )\r\n}\r\n\r\nconst performanceTimeOrigin100ns = Math.round(performance.timeOrigin * 10000)\r\n\r\nexport function highPrecisionISODate(performanceNow: number) {\r\n    const now100ns = performanceTimeOrigin100ns + Math.round(performanceNow * 10000)\r\n    return (\r\n        new Date(now100ns / 10000).toISOString().substring(0, 20) +\r\n        (now100ns % 10000000).toString().padStart(7, '0') +\r\n        'Z'\r\n    )\r\n}\r\n\r\nclass LogBuffer {\r\n    readonly #transport: LogTransport\r\n    #entries: LogEntry[] = []\r\n    #size = 0\r\n    #flusher?: Promise<void> | undefined\r\n    readonly #signal: AbortSignal\r\n    #asyncTransport: boolean | undefined\r\n    #timeout: NodeJS.Timeout | undefined\r\n\r\n    constructor(transport: LogTransport, signal: AbortSignal) {\r\n        this.#transport = transport\r\n        this.#signal = signal\r\n    }\r\n\r\n    collect(\r\n        level: LogLevel,\r\n        numericLogLevel: number,\r\n        message: string,\r\n        error: unknown,\r\n        fields: object | undefined,\r\n        reservedEnrichment: object | undefined,\r\n        customEnrichment: object | undefined,\r\n    ) {\r\n        const offset = performance.now()\r\n        const json = JSON.stringify({\r\n            timestamp: highPrecisionISODate(offset),\r\n            level,\r\n            message,\r\n            error: errorAsJson(error),\r\n            ...reservedEnrichment,\r\n            ...((fields || customEnrichment) && {\r\n                fields: customEnrichment\r\n                    ? fields\r\n                        ? { ...customEnrichment, ...fields }\r\n                        : customEnrichment\r\n                    : fields,\r\n            }),\r\n        })\r\n        this.#entries.push({\r\n            timestamp: offset,\r\n            level,\r\n            message,\r\n            error,\r\n            json,\r\n        })\r\n        this.#size += json.length\r\n\r\n        if (this.#asyncTransport === false) {\r\n            // eslint-disable-next-line no-void\r\n            void this.#transport.sendEntries(this.#entries, this.#signal)\r\n            this.#entries = []\r\n            this.#size = 0\r\n        } else if (this.#asyncTransport === undefined) {\r\n            this.#asyncTransport = true\r\n            setImmediate(() => {\r\n                if (this.#flusher) {\r\n                    return\r\n                }\r\n                const sendResult = this.#transport.sendEntries(this.#entries, this.#signal)\r\n                this.#entries = []\r\n                this.#size = 0\r\n                if (sendResult) {\r\n                    this.#flusher = sendResult\r\n                } else {\r\n                    this.#asyncTransport = false\r\n                }\r\n            })\r\n        } else {\r\n            if (numericLogLevel < 2 || this.#entries.length > 8 || this.#size > 64000) {\r\n                // eslint-disable-next-line no-void\r\n                void this.flush()\r\n            } else {\r\n                this.#timeout = setTimeout(() => {\r\n                    // eslint-disable-next-line no-void\r\n                    void this.flush()\r\n                    this.#timeout = undefined\r\n                }, 2000)\r\n            }\r\n        }\r\n    }\r\n\r\n    async flush(): Promise<void> {\r\n        if (this.#entries.length === 0) {\r\n            return\r\n        }\r\n        this.#startFlush(this.#entries)\r\n        this.#entries = []\r\n        this.#size = 0\r\n        if (this.#timeout) {\r\n            clearTimeout(this.#timeout)\r\n            this.#timeout = undefined\r\n        }\r\n        return await this.#flusher\r\n    }\r\n\r\n    #startFlush(entries: LogEntry[]) {\r\n        if (this.#flusher) {\r\n            this.#flusher = this.#flusher.then(() =>\r\n                this.#transport.sendEntries(entries, this.#signal),\r\n            )\r\n        } else {\r\n            this.#flusher = this.#transport.sendEntries(entries, this.#signal)\r\n        }\r\n    }\r\n}\r\n\r\nclass EnrichingLogger implements Logger {\r\n    readonly #buffer: LogBuffer\r\n    readonly #reservedEnrichment?: object\r\n    readonly #customEnrichment?: object\r\n    readonly #level: number\r\n\r\n    constructor(\r\n        buffer: LogBuffer,\r\n        level: number,\r\n        reservedEnrichment?: object,\r\n        customEnrichment?: object,\r\n    ) {\r\n        this.#buffer = buffer\r\n        this.#level = level\r\n        this.#reservedEnrichment = reservedEnrichment\r\n        this.#customEnrichment = customEnrichment\r\n    }\r\n\r\n    enrich(fields: object): Logger {\r\n        return new EnrichingLogger(this.#buffer, this.#level, this.#reservedEnrichment, {\r\n            ...(this.#customEnrichment ?? {}),\r\n            ...fields,\r\n        })\r\n    }\r\n\r\n    flush() {\r\n        return this.#buffer.flush()\r\n    }\r\n\r\n    enrichReserved(fields: object): EnrichingLogger {\r\n        return new EnrichingLogger(\r\n            this.#buffer,\r\n            this.#level,\r\n            {\r\n                ...(this.#reservedEnrichment ?? {}),\r\n                ...fields,\r\n            },\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n\r\n    trace(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 5) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'trace',\r\n            5,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    debug(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 4) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'debug',\r\n            4,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    info(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 3) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'debug',\r\n            3,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    warn(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 2) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'debug',\r\n            2,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    error(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 1) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'error',\r\n            1,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    fatal(message: string, error: unknown, fields: object | undefined): void {\r\n        this.#buffer.collect(\r\n            'fatal',\r\n            0,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n}\r\n\r\nfunction errorAsJson(error: unknown): Json | undefined {\r\n    if (error === undefined || error === null) {\r\n        return undefined\r\n    }\r\n    if (error instanceof Error) {\r\n        return {\r\n            message: error.message,\r\n            name: error.name,\r\n            stack: error.stack,\r\n            ...(error as unknown as { [key: string]: unknown }),\r\n        } as Json\r\n    }\r\n    if (error instanceof Object) {\r\n        return {\r\n            ...error,\r\n        } as Json\r\n    }\r\n    return {\r\n        message: (error as object | undefined)?.toString(),\r\n        name: typeof error,\r\n    } as Json\r\n}\r\n"]}
205
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"logging.js","sourceRoot":"","sources":["logging.ts"],"names":[],"mappings":"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","sourcesContent":["import { performance } from 'perf_hooks'\r\nimport { AbortSignal, Json, Logger } from '../context.js'\r\nimport { LogEntry, LogLevel, LogTransport, RootLogger } from './context.js'\r\n\r\nexport function makeLogger(\r\n    transport: LogTransport,\r\n    minimumLogLevel: LogLevel | undefined,\r\n    signal: AbortSignal,\r\n): RootLogger {\r\n    return new EnrichingLogger(\r\n        new LogBuffer(transport, signal),\r\n        minimumLogLevel\r\n            ? ['fatal', 'error', 'warning', 'info', 'debug', 'trace'].indexOf(minimumLogLevel)\r\n            : 5,\r\n        signal,\r\n    )\r\n}\r\n\r\nconst performanceTimeOrigin100ns = Math.round(performance.timeOrigin * 10000)\r\n\r\nexport function highPrecisionISODate(performanceNow: number) {\r\n    const now100ns = performanceTimeOrigin100ns + Math.round(performanceNow * 10000)\r\n    return (\r\n        new Date(now100ns / 10000).toISOString().substring(0, 20) +\r\n        (now100ns % 10000000).toString().padStart(7, '0') +\r\n        'Z'\r\n    )\r\n}\r\n\r\nclass LogBuffer {\r\n    readonly #transport: LogTransport\r\n    #entries: LogEntry[] = []\r\n    #size = 0\r\n    #flusher?: Promise<void> | undefined\r\n    readonly #signal: AbortSignal\r\n    #asyncTransport: boolean | undefined\r\n    #timeout: NodeJS.Timeout | undefined\r\n\r\n    constructor(transport: LogTransport, signal: AbortSignal) {\r\n        this.#transport = transport\r\n        this.#signal = signal\r\n    }\r\n\r\n    collect(\r\n        level: LogLevel,\r\n        numericLogLevel: number,\r\n        message: string,\r\n        error: unknown,\r\n        fields: object | undefined,\r\n        reservedEnrichment: object | undefined,\r\n        customEnrichment: object | undefined,\r\n    ) {\r\n        const offset = performance.now()\r\n        const json = JSON.stringify({\r\n            timestamp: highPrecisionISODate(offset),\r\n            level,\r\n            message,\r\n            error: errorAsJson(error),\r\n            ...reservedEnrichment,\r\n            ...((fields || customEnrichment) && {\r\n                fields: customEnrichment\r\n                    ? fields\r\n                        ? { ...customEnrichment, ...fields }\r\n                        : customEnrichment\r\n                    : fields,\r\n            }),\r\n        })\r\n        this.#entries.push({\r\n            timestamp: offset,\r\n            level,\r\n            message,\r\n            error,\r\n            json,\r\n        })\r\n        this.#size += json.length\r\n\r\n        if (this.#asyncTransport === false) {\r\n            // eslint-disable-next-line no-void\r\n            void this.#transport.sendEntries(this.#entries, this.#signal)\r\n            this.#entries = []\r\n            this.#size = 0\r\n        } else if (this.#asyncTransport === undefined) {\r\n            this.#asyncTransport = true\r\n            setImmediate(() => {\r\n                if (this.#flusher) {\r\n                    return\r\n                }\r\n                const sendResult = this.#transport.sendEntries(this.#entries, this.#signal)\r\n                this.#entries = []\r\n                this.#size = 0\r\n                if (sendResult) {\r\n                    this.#flusher = sendResult\r\n                } else {\r\n                    this.#asyncTransport = false\r\n                }\r\n            })\r\n        } else {\r\n            if (numericLogLevel < 2 || this.#entries.length > 8 || this.#size > 64000) {\r\n                // eslint-disable-next-line no-void\r\n                void this.flush()\r\n            } else {\r\n                this.#timeout = setTimeout(() => {\r\n                    // eslint-disable-next-line no-void\r\n                    void this.flush()\r\n                    this.#timeout = undefined\r\n                }, 2000)\r\n            }\r\n        }\r\n    }\r\n\r\n    async flush(): Promise<void> {\r\n        if (this.#entries.length === 0) {\r\n            return\r\n        }\r\n        this.#startFlush(this.#entries)\r\n        this.#entries = []\r\n        this.#size = 0\r\n        if (this.#timeout) {\r\n            clearTimeout(this.#timeout)\r\n            this.#timeout = undefined\r\n        }\r\n        return await this.#flusher\r\n    }\r\n\r\n    #startFlush(entries: LogEntry[]) {\r\n        if (this.#flusher) {\r\n            this.#flusher = this.#flusher.then(() =>\r\n                this.#transport.sendEntries(entries, this.#signal),\r\n            )\r\n        } else {\r\n            this.#flusher = this.#transport.sendEntries(entries, this.#signal)\r\n        }\r\n    }\r\n}\r\n\r\nclass EnrichingLogger implements Logger {\r\n    readonly #buffer: LogBuffer\r\n    readonly #reservedEnrichment?: object\r\n    readonly #customEnrichment?: object\r\n    readonly #level: number\r\n\r\n    constructor(\r\n        buffer: LogBuffer,\r\n        level: number,\r\n        reservedEnrichment?: object,\r\n        customEnrichment?: object,\r\n    ) {\r\n        this.#buffer = buffer\r\n        this.#level = level\r\n        this.#reservedEnrichment = reservedEnrichment\r\n        this.#customEnrichment = customEnrichment\r\n    }\r\n\r\n    enrich(fields: object): Logger {\r\n        return new EnrichingLogger(this.#buffer, this.#level, this.#reservedEnrichment, {\r\n            ...(this.#customEnrichment ?? {}),\r\n            ...fields,\r\n        })\r\n    }\r\n\r\n    flush() {\r\n        return this.#buffer.flush()\r\n    }\r\n\r\n    enrichReserved(fields: object): EnrichingLogger {\r\n        return new EnrichingLogger(\r\n            this.#buffer,\r\n            this.#level,\r\n            {\r\n                ...(this.#reservedEnrichment ?? {}),\r\n                ...fields,\r\n            },\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n\r\n    trace(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 5) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'trace',\r\n            5,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    debug(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 4) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'debug',\r\n            4,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    info(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 3) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'info',\r\n            3,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    warn(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 2) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'warning',\r\n            2,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    error(message: string, error: unknown, fields: object | undefined): void {\r\n        if (this.#level < 1) {\r\n            return\r\n        }\r\n        this.#buffer.collect(\r\n            'error',\r\n            1,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n    fatal(message: string, error: unknown, fields: object | undefined): void {\r\n        this.#buffer.collect(\r\n            'fatal',\r\n            0,\r\n            message,\r\n            error,\r\n            fields,\r\n            this.#reservedEnrichment,\r\n            this.#customEnrichment,\r\n        )\r\n    }\r\n}\r\n\r\nfunction errorAsJson(error: unknown): Json | undefined {\r\n    if (error === undefined || error === null) {\r\n        return undefined\r\n    }\r\n    if (error instanceof Error) {\r\n        return {\r\n            message: error.message,\r\n            name: error.name,\r\n            stack: error.stack,\r\n            ...(error as unknown as { [key: string]: unknown }),\r\n        } as Json\r\n    }\r\n    if (error instanceof Object) {\r\n        return {\r\n            ...error,\r\n        } as Json\r\n    }\r\n    return {\r\n        message: (error as object | undefined)?.toString(),\r\n        name: typeof error,\r\n    } as Json\r\n}\r\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@riddance/host",
3
- "version": "0.0.4",
3
+ "version": "0.0.6",
4
4
  "type": "module",
5
5
  "license": "MIT",
6
6
  "engines": {
@@ -29,7 +29,7 @@
29
29
  "./http": "./host/http.js"
30
30
  },
31
31
  "devDependencies": {
32
- "@riddance/env": "0.0.2"
32
+ "@riddance/env": "0.0.3"
33
33
  },
34
34
  "scripts": {
35
35
  "prepare": "tsc",