autotel-adapters 0.3.8 → 0.3.10
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/dist/cloudflare.cjs +60 -124
- package/dist/cloudflare.cjs.map +1 -1
- package/dist/cloudflare.d.cts +24 -24
- package/dist/cloudflare.d.cts.map +1 -0
- package/dist/cloudflare.d.ts +24 -24
- package/dist/cloudflare.d.ts.map +1 -0
- package/dist/cloudflare.js +69 -3
- package/dist/cloudflare.js.map +1 -1
- package/dist/core.cjs +94 -74
- package/dist/core.cjs.map +1 -1
- package/dist/core.d.cts +20 -19
- package/dist/core.d.cts.map +1 -0
- package/dist/core.d.ts +20 -19
- package/dist/core.d.ts.map +1 -0
- package/dist/core.js +110 -2
- package/dist/core.js.map +1 -1
- package/dist/express.cjs +67 -127
- package/dist/express.cjs.map +1 -1
- package/dist/express.d.cts +24 -24
- package/dist/express.d.cts.map +1 -0
- package/dist/express.d.ts +24 -24
- package/dist/express.d.ts.map +1 -0
- package/dist/express.js +78 -3
- package/dist/express.js.map +1 -1
- package/dist/fastify.cjs +58 -118
- package/dist/fastify.cjs.map +1 -1
- package/dist/fastify.d.cts +24 -24
- package/dist/fastify.d.cts.map +1 -0
- package/dist/fastify.d.ts +24 -24
- package/dist/fastify.d.ts.map +1 -0
- package/dist/fastify.js +69 -3
- package/dist/fastify.js.map +1 -1
- package/dist/hono.cjs +18 -48
- package/dist/hono.cjs.map +1 -1
- package/dist/hono.d.cts +6 -6
- package/dist/hono.d.cts.map +1 -0
- package/dist/hono.d.ts +6 -6
- package/dist/hono.d.ts.map +1 -0
- package/dist/hono.js +22 -3
- package/dist/hono.js.map +1 -1
- package/dist/index.cjs +27 -455
- package/dist/index.d.cts +9 -11
- package/dist/index.d.ts +9 -11
- package/dist/index.js +10 -10
- package/dist/next.cjs +56 -119
- package/dist/next.cjs.map +1 -1
- package/dist/next.d.cts +21 -21
- package/dist/next.d.cts.map +1 -0
- package/dist/next.d.ts +21 -21
- package/dist/next.d.ts.map +1 -0
- package/dist/next.js +65 -3
- package/dist/next.js.map +1 -1
- package/dist/nitro.cjs +50 -93
- package/dist/nitro.cjs.map +1 -1
- package/dist/nitro.d.cts +19 -19
- package/dist/nitro.d.cts.map +1 -0
- package/dist/nitro.d.ts +19 -19
- package/dist/nitro.d.ts.map +1 -0
- package/dist/nitro.js +62 -3
- package/dist/nitro.js.map +1 -1
- package/dist/tanstack.cjs +16 -46
- package/dist/tanstack.cjs.map +1 -1
- package/dist/tanstack.d.cts +9 -9
- package/dist/tanstack.d.cts.map +1 -0
- package/dist/tanstack.d.ts +9 -9
- package/dist/tanstack.d.ts.map +1 -0
- package/dist/tanstack.js +20 -3
- package/dist/tanstack.js.map +1 -1
- package/package.json +5 -5
- package/dist/chunk-2YPL66HM.js +0 -79
- package/dist/chunk-2YPL66HM.js.map +0 -1
- package/dist/chunk-6TOW47TB.js +0 -70
- package/dist/chunk-6TOW47TB.js.map +0 -1
- package/dist/chunk-DJ2OU3S6.js +0 -90
- package/dist/chunk-DJ2OU3S6.js.map +0 -1
- package/dist/chunk-FAHH33UI.js +0 -76
- package/dist/chunk-FAHH33UI.js.map +0 -1
- package/dist/chunk-HTZFHBTN.js +0 -82
- package/dist/chunk-HTZFHBTN.js.map +0 -1
- package/dist/chunk-JH5ZTTLF.js +0 -23
- package/dist/chunk-JH5ZTTLF.js.map +0 -1
- package/dist/chunk-U3U4WH42.js +0 -21
- package/dist/chunk-U3U4WH42.js.map +0 -1
- package/dist/chunk-VJZDW2DS.js +0 -87
- package/dist/chunk-VJZDW2DS.js.map +0 -1
- package/dist/index.cjs.map +0 -1
- package/dist/index.js.map +0 -1
package/dist/cloudflare.cjs
CHANGED
|
@@ -1,141 +1,77 @@
|
|
|
1
|
-
'
|
|
1
|
+
Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
|
|
2
|
+
const require_core = require('./core.cjs');
|
|
3
|
+
let autotel = require("autotel");
|
|
2
4
|
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
// src/cloudflare.ts
|
|
6
|
-
function createUseLogger(options) {
|
|
7
|
-
return function useLogger2(context, requestLoggerOptions) {
|
|
8
|
-
let logger;
|
|
9
|
-
try {
|
|
10
|
-
logger = autotel.getRequestLogger(void 0, requestLoggerOptions);
|
|
11
|
-
} catch {
|
|
12
|
-
throw new Error(
|
|
13
|
-
`[autotel-adapters/${options.adapterName}] No active trace context. Wrap your handler with autotel trace instrumentation before calling useLogger().`
|
|
14
|
-
);
|
|
15
|
-
}
|
|
16
|
-
if (context && options.enrich) {
|
|
17
|
-
const extra = options.enrich(context);
|
|
18
|
-
if (extra && Object.keys(extra).length > 0) {
|
|
19
|
-
logger.set(extra);
|
|
20
|
-
}
|
|
21
|
-
}
|
|
22
|
-
return logger;
|
|
23
|
-
};
|
|
24
|
-
}
|
|
25
|
-
function createAdapterToolkit(options) {
|
|
26
|
-
return {
|
|
27
|
-
useLogger: createUseLogger(options),
|
|
28
|
-
parseError: autotel.parseError,
|
|
29
|
-
createStructuredError: autotel.createStructuredError,
|
|
30
|
-
createDrainPipeline: autotel.createDrainPipeline
|
|
31
|
-
};
|
|
32
|
-
}
|
|
33
|
-
function getHeader(headers, name) {
|
|
34
|
-
if (!headers) return void 0;
|
|
35
|
-
if ("get" in headers && typeof headers.get === "function") {
|
|
36
|
-
return headers.get(name) ?? void 0;
|
|
37
|
-
}
|
|
38
|
-
const dictionary = headers;
|
|
39
|
-
const value = dictionary[name] ?? dictionary[name.toLowerCase()];
|
|
40
|
-
return typeof value === "string" ? value : void 0;
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
// src/cloudflare.ts
|
|
44
|
-
var requestLoggers = /* @__PURE__ */ new WeakMap();
|
|
5
|
+
//#region src/cloudflare.ts
|
|
6
|
+
const requestLoggers = /* @__PURE__ */ new WeakMap();
|
|
45
7
|
function enrichFromRequest(request) {
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
...request.cf?.city ? { "cloudflare.city": request.cf.city } : {}
|
|
64
|
-
};
|
|
8
|
+
if (!request) return void 0;
|
|
9
|
+
let route = "/";
|
|
10
|
+
if (request.url) try {
|
|
11
|
+
route = new URL(request.url).pathname;
|
|
12
|
+
} catch {
|
|
13
|
+
route = request.url;
|
|
14
|
+
}
|
|
15
|
+
const requestId = require_core.getHeader(request.headers, "x-request-id") ?? require_core.getHeader(request.headers, "cf-ray");
|
|
16
|
+
return {
|
|
17
|
+
...request.method ? { "http.request.method": request.method } : {},
|
|
18
|
+
...request.url ? { "url.full": request.url } : {},
|
|
19
|
+
...route ? { "http.route": route } : {},
|
|
20
|
+
...requestId ? { "http.request.id": requestId } : {},
|
|
21
|
+
...request.cf?.country ? { "cloudflare.country": request.cf.country } : {},
|
|
22
|
+
...request.cf?.colo ? { "cloudflare.colo": request.cf.colo } : {},
|
|
23
|
+
...request.cf?.city ? { "cloudflare.city": request.cf.city } : {}
|
|
24
|
+
};
|
|
65
25
|
}
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
26
|
+
const baseUseLogger = require_core.createUseLogger({
|
|
27
|
+
adapterName: "cloudflare",
|
|
28
|
+
enrich: enrichFromRequest
|
|
69
29
|
});
|
|
70
30
|
function useLogger(request, requestLoggerOptions) {
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
31
|
+
if (request) {
|
|
32
|
+
const existing = requestLoggers.get(request);
|
|
33
|
+
if (existing) return existing;
|
|
34
|
+
}
|
|
35
|
+
return baseUseLogger(request, requestLoggerOptions);
|
|
76
36
|
}
|
|
77
37
|
function withAutotelFetch(handler, options) {
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
log.set(custom);
|
|
95
|
-
}
|
|
96
|
-
requestLoggers.set(innerRequest, log);
|
|
97
|
-
try {
|
|
98
|
-
return await handler(innerRequest, innerEnv, innerExecutionContext);
|
|
99
|
-
} finally {
|
|
100
|
-
if (options?.autoEmit !== false) {
|
|
101
|
-
log.emitNow();
|
|
102
|
-
}
|
|
103
|
-
requestLoggers.delete(innerRequest);
|
|
104
|
-
}
|
|
105
|
-
}
|
|
106
|
-
);
|
|
107
|
-
return await wrapped(request, env, executionContext);
|
|
108
|
-
};
|
|
38
|
+
return async (request, env, executionContext) => {
|
|
39
|
+
return await (0, autotel.trace)({ name: typeof options?.spanName === "function" ? options.spanName(request, env) : options?.spanName ?? `cloudflare.${request.method ?? "request"}` }, (ctx) => async (innerRequest, innerEnv, innerExecutionContext) => {
|
|
40
|
+
const log = (0, autotel.getRequestLogger)(ctx, options?.requestLoggerOptions);
|
|
41
|
+
const auto = enrichFromRequest(innerRequest);
|
|
42
|
+
if (auto && Object.keys(auto).length > 0) log.set(auto);
|
|
43
|
+
const custom = options?.enrich?.(innerRequest, innerEnv, innerExecutionContext);
|
|
44
|
+
if (custom && Object.keys(custom).length > 0) log.set(custom);
|
|
45
|
+
requestLoggers.set(innerRequest, log);
|
|
46
|
+
try {
|
|
47
|
+
return await handler(innerRequest, innerEnv, innerExecutionContext);
|
|
48
|
+
} finally {
|
|
49
|
+
if (options?.autoEmit !== false) log.emitNow();
|
|
50
|
+
requestLoggers.delete(innerRequest);
|
|
51
|
+
}
|
|
52
|
+
})(request, env, executionContext);
|
|
53
|
+
};
|
|
109
54
|
}
|
|
110
55
|
function createCloudflareAdapter(options) {
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
56
|
+
return {
|
|
57
|
+
withAutotelFetch: (handler) => withAutotelFetch(handler, options),
|
|
58
|
+
useLogger,
|
|
59
|
+
parseError: (error) => (0, autotel.parseError)(error),
|
|
60
|
+
createStructuredError: (input) => (0, autotel.createStructuredError)(input),
|
|
61
|
+
createDrainPipeline: (drainOptions) => (0, autotel.createDrainPipeline)(drainOptions)
|
|
62
|
+
};
|
|
118
63
|
}
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
64
|
+
const cloudflareToolkit = require_core.createAdapterToolkit({
|
|
65
|
+
adapterName: "cloudflare",
|
|
66
|
+
enrich: enrichFromRequest
|
|
122
67
|
});
|
|
123
68
|
|
|
124
|
-
|
|
125
|
-
enumerable: true,
|
|
126
|
-
get: function () { return autotel.createDrainPipeline; }
|
|
127
|
-
});
|
|
128
|
-
Object.defineProperty(exports, "createStructuredError", {
|
|
129
|
-
enumerable: true,
|
|
130
|
-
get: function () { return autotel.createStructuredError; }
|
|
131
|
-
});
|
|
132
|
-
Object.defineProperty(exports, "parseError", {
|
|
133
|
-
enumerable: true,
|
|
134
|
-
get: function () { return autotel.parseError; }
|
|
135
|
-
});
|
|
69
|
+
//#endregion
|
|
136
70
|
exports.cloudflareToolkit = cloudflareToolkit;
|
|
137
71
|
exports.createCloudflareAdapter = createCloudflareAdapter;
|
|
72
|
+
exports.createDrainPipeline = autotel.createDrainPipeline;
|
|
73
|
+
exports.createStructuredError = autotel.createStructuredError;
|
|
74
|
+
exports.parseError = autotel.parseError;
|
|
138
75
|
exports.useLogger = useLogger;
|
|
139
76
|
exports.withAutotelFetch = withAutotelFetch;
|
|
140
|
-
//# sourceMappingURL=cloudflare.cjs.map
|
|
141
77
|
//# sourceMappingURL=cloudflare.cjs.map
|
package/dist/cloudflare.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/core.ts","../src/cloudflare.ts"],"names":["useLogger","getRequestLogger","parseError","createStructuredError","createDrainPipeline","trace"],"mappings":"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","file":"cloudflare.cjs","sourcesContent":["import type { AsyncLocalStorage } from 'node:async_hooks';\nimport {\n createDrainPipeline,\n createStructuredError,\n getRequestLogger,\n parseError,\n trace,\n type ParsedError,\n type RequestLogger,\n type RequestLoggerOptions,\n type RequestLogSnapshot,\n type DrainPipelineOptions,\n type PipelineDrainFn,\n type StructuredError,\n type StructuredErrorInput,\n} from 'autotel';\n\nexport interface AdapterUseLoggerOptions<TContext> {\n adapterName: string;\n enrich?: (context: TContext) => Record<string, unknown> | undefined;\n}\n\nexport interface AdapterToolkit<TContext> {\n useLogger: (\n context?: TContext,\n options?: RequestLoggerOptions,\n ) => RequestLogger;\n parseError: (error: unknown) => ParsedError;\n createStructuredError: (input: StructuredErrorInput) => StructuredError;\n createDrainPipeline: <T = unknown>(\n options?: DrainPipelineOptions<T>,\n ) => (drain: (batch: T[]) => void | Promise<void>) => PipelineDrainFn<T>;\n}\n\nexport function createUseLogger<TContext = unknown>(\n options: AdapterUseLoggerOptions<TContext>,\n) {\n return function useLogger(\n context?: TContext,\n requestLoggerOptions?: RequestLoggerOptions,\n ): RequestLogger {\n let logger: RequestLogger;\n try {\n logger = getRequestLogger(undefined, requestLoggerOptions);\n } catch {\n throw new Error(\n `[autotel-adapters/${options.adapterName}] No active trace context. ` +\n `Wrap your handler with autotel trace instrumentation before calling useLogger().`,\n );\n }\n\n if (context && options.enrich) {\n const extra = options.enrich(context);\n if (extra && Object.keys(extra).length > 0) {\n logger.set(extra);\n }\n }\n\n return logger;\n };\n}\n\nexport interface RequestRunnerOptions {\n requestLoggerOptions?: RequestLoggerOptions;\n /** Emit one wide event automatically when the handler settles. Default `true`. */\n autoEmit?: boolean;\n /** Fields merged into the wide event at emit time (e.g. response status). */\n finalize?: () => Record<string, unknown> | undefined;\n}\n\n/**\n * Build a request runner bound to one framework's logger storage. The returned\n * function opens a span, creates a request logger, runs `handler` inside the\n * storage so `useLogger()` resolves it, records thrown errors, and emits one\n * wide event when the handler settles (unless `autoEmit` is `false`).\n */\nexport function createRequestRunner(storage: AsyncLocalStorage<RequestLogger>) {\n return function runRequest<T>(\n spanName: string,\n enrich: (log: RequestLogger) => void,\n handler: () => T | Promise<T>,\n options?: RequestRunnerOptions,\n ): Promise<T> {\n const wrapped = trace(\n { name: spanName },\n (ctx) => async (): Promise<T> => {\n const log = getRequestLogger(ctx, options?.requestLoggerOptions);\n enrich(log);\n try {\n return await storage.run(log, () => handler());\n } catch (error) {\n log.error(error instanceof Error ? error : new Error(String(error)));\n throw error;\n } finally {\n if (options?.autoEmit !== false) {\n log.emitNow(options?.finalize?.());\n }\n }\n },\n );\n return wrapped();\n };\n}\n\nexport function createAdapterToolkit<TContext = unknown>(\n options: AdapterUseLoggerOptions<TContext>,\n): AdapterToolkit<TContext> {\n return {\n useLogger: createUseLogger(options),\n parseError,\n createStructuredError,\n createDrainPipeline,\n };\n}\n\n/**\n * Description of a single adapter config field. `env` is the ordered list of\n * environment variables to fall back to.\n */\nexport interface ConfigField<T> {\n key: keyof T & string;\n env?: string[];\n}\n\nfunction resolveEnv(envKeys?: string[]): string | undefined {\n if (!envKeys) return undefined;\n for (const key of envKeys) {\n const value = process.env[key];\n if (value) return value;\n }\n return undefined;\n}\n\n/**\n * Returns true when at least one env-backed field is not provided via\n * `overrides`, meaning runtime config may still contribute and should be\n * probed to preserve precedence (`overrides > runtime > env`).\n *\n * @example\n * ```ts\n * const FIELDS: ConfigField<MyAdapterConfig>[] = [\n * { key: 'token', env: ['MY_ADAPTER_TOKEN'] },\n * { key: 'endpoint', env: ['MY_ADAPTER_URL'] },\n * ]\n *\n * if (shouldProbeRuntime(FIELDS, overrides)) {\n * runtimeConfig = await loadRuntimeConfig()\n * }\n * ```\n */\nexport function shouldProbeRuntime<T>(\n fields: ConfigField<T>[],\n overrides?: Partial<T>,\n): boolean {\n return fields.some(({ key, env }) => {\n if (!env || env.length === 0) return false;\n if (overrides?.[key] !== undefined) return false;\n return true;\n });\n}\n\n/**\n * Resolve adapter configuration with the standard priority chain:\n *\n * 1. `overrides` passed to the adapter factory\n * 2. `runtimeConfig.autotel.{namespace}.{key}` (if a probe was performed)\n * 3. `runtimeConfig.{namespace}.{key}` (if a probe was performed)\n * 4. `process.env[envKey]` for each env in `field.env`\n *\n * Pass an async `probe` to defer the runtime config lookup so it is only\n * invoked when runtime resolution is needed (i.e. at least one env-backed\n * field is not set by overrides). Adapters that have no probe target may pass\n * `() => Promise.resolve(undefined)`.\n */\nexport async function resolveAdapterConfig<T>(\n namespace: string,\n fields: ConfigField<T>[],\n overrides: Partial<T> | undefined,\n probe: () => Promise<Record<string, any> | undefined>,\n): Promise<Partial<T>> {\n const runtimeConfig = shouldProbeRuntime(fields, overrides)\n ? await probe()\n : undefined;\n const autotelNs = runtimeConfig?.autotel?.[namespace];\n const rootNs = runtimeConfig?.[namespace];\n\n const config: Record<string, unknown> = {};\n for (const { key, env } of fields) {\n config[key] =\n overrides?.[key] ??\n autotelNs?.[key] ??\n rootNs?.[key] ??\n resolveEnv(env);\n }\n\n return config as Partial<T>;\n}\n\nexport type HeadersLike =\n | { get(name: string): string | null }\n | Record<string, string | undefined>;\n\nexport function getHeader(\n headers: HeadersLike | undefined,\n name: string,\n): string | undefined {\n if (!headers) return undefined;\n if ('get' in headers && typeof headers.get === 'function') {\n return headers.get(name) ?? undefined;\n }\n const dictionary = headers as Record<string, string | undefined>;\n const value = dictionary[name] ?? dictionary[name.toLowerCase()];\n return typeof value === 'string' ? value : undefined;\n}\n\nexport type {\n RequestLogger,\n RequestLoggerOptions,\n RequestLogSnapshot,\n ParsedError,\n StructuredError,\n StructuredErrorInput,\n DrainPipelineOptions,\n PipelineDrainFn,\n};\n","import {\n createDrainPipeline,\n createStructuredError,\n getRequestLogger,\n parseError,\n trace,\n type DrainPipelineOptions,\n type ParsedError,\n type PipelineDrainFn,\n type RequestLogger,\n type RequestLoggerOptions,\n type StructuredError,\n type StructuredErrorInput,\n} from 'autotel';\nimport { createAdapterToolkit, createUseLogger, getHeader } from './core';\n\nexport interface CloudflareRequestLike {\n method?: string;\n url?: string;\n headers?:\n | { get(name: string): string | null }\n | Record<string, string | undefined>;\n cf?: Record<string, unknown>;\n}\n\nexport interface CloudflareExecutionContextLike {\n waitUntil?: (promise: Promise<unknown>) => void;\n passThroughOnException?: () => void;\n}\n\nexport interface CloudflareWithAutotelOptions<TEnv = unknown> {\n spanName?: string | ((request: CloudflareRequestLike, env: TEnv) => string);\n requestLoggerOptions?: RequestLoggerOptions;\n enrich?: (\n request: CloudflareRequestLike,\n env: TEnv,\n ctx: CloudflareExecutionContextLike,\n ) => Record<string, unknown> | undefined;\n /** Emit one wide event automatically when the handler settles. Default `true`. */\n autoEmit?: boolean;\n}\n\nconst requestLoggers = new WeakMap<object, RequestLogger>();\n\nfunction enrichFromRequest(\n request?: CloudflareRequestLike,\n): Record<string, unknown> | undefined {\n if (!request) return undefined;\n\n let route = '/';\n if (request.url) {\n try {\n route = new URL(request.url).pathname;\n } catch {\n route = request.url;\n }\n }\n\n const requestId =\n getHeader(request.headers, 'x-request-id') ??\n getHeader(request.headers, 'cf-ray');\n\n return {\n ...(request.method ? { 'http.request.method': request.method } : {}),\n ...(request.url ? { 'url.full': request.url } : {}),\n ...(route ? { 'http.route': route } : {}),\n ...(requestId ? { 'http.request.id': requestId } : {}),\n ...(request.cf?.country ? { 'cloudflare.country': request.cf.country } : {}),\n ...(request.cf?.colo ? { 'cloudflare.colo': request.cf.colo } : {}),\n ...(request.cf?.city ? { 'cloudflare.city': request.cf.city } : {}),\n };\n}\n\nconst baseUseLogger = createUseLogger<CloudflareRequestLike>({\n adapterName: 'cloudflare',\n enrich: enrichFromRequest,\n});\n\nexport function useLogger(\n request?: CloudflareRequestLike,\n requestLoggerOptions?: RequestLoggerOptions,\n): RequestLogger {\n if (request) {\n const existing = requestLoggers.get(request as object);\n if (existing) return existing;\n }\n return baseUseLogger(request, requestLoggerOptions);\n}\n\nexport function withAutotelFetch<\n TEnv,\n TRequest extends CloudflareRequestLike,\n TContext extends CloudflareExecutionContextLike,\n TReturn,\n>(\n handler: (\n request: TRequest,\n env: TEnv,\n ctx: TContext,\n ) => TReturn | Promise<TReturn>,\n options?: CloudflareWithAutotelOptions<TEnv>,\n): (\n request: TRequest,\n env: TEnv,\n ctx: TContext,\n) => Promise<TReturn> {\n return async (\n request: TRequest,\n env: TEnv,\n executionContext: TContext,\n ): Promise<TReturn> => {\n const spanName =\n typeof options?.spanName === 'function'\n ? options.spanName(request, env)\n : (options?.spanName ?? `cloudflare.${request.method ?? 'request'}`);\n\n const wrapped = trace(\n { name: spanName },\n (ctx) => async (\n innerRequest: TRequest,\n innerEnv: TEnv,\n innerExecutionContext: TContext,\n ) => {\n const log = getRequestLogger(ctx, options?.requestLoggerOptions);\n const auto = enrichFromRequest(innerRequest);\n if (auto && Object.keys(auto).length > 0) {\n log.set(auto);\n }\n const custom = options?.enrich?.(\n innerRequest,\n innerEnv,\n innerExecutionContext,\n );\n if (custom && Object.keys(custom).length > 0) {\n log.set(custom);\n }\n\n requestLoggers.set(innerRequest as object, log);\n try {\n return await handler(innerRequest, innerEnv, innerExecutionContext);\n } finally {\n if (options?.autoEmit !== false) {\n log.emitNow();\n }\n requestLoggers.delete(innerRequest as object);\n }\n },\n );\n\n return await wrapped(request, env, executionContext);\n };\n}\n\nexport function createCloudflareAdapter<TEnv = unknown>(\n options?: CloudflareWithAutotelOptions<TEnv>,\n) {\n return {\n withAutotelFetch: <\n TRequest extends CloudflareRequestLike,\n TContext extends CloudflareExecutionContextLike,\n TReturn,\n >(\n handler: (\n request: TRequest,\n env: TEnv,\n ctx: TContext,\n ) => TReturn | Promise<TReturn>,\n ) => withAutotelFetch(handler, options),\n useLogger,\n parseError: (error: unknown): ParsedError => parseError(error),\n createStructuredError: (\n input: StructuredErrorInput,\n ): StructuredError => createStructuredError(input),\n createDrainPipeline: <T = unknown>(\n drainOptions?: DrainPipelineOptions<T>,\n ): ((batchDrain: (batch: T[]) => void | Promise<void>) => PipelineDrainFn<T>) =>\n createDrainPipeline(drainOptions),\n };\n}\n\nexport const cloudflareToolkit = createAdapterToolkit<CloudflareRequestLike>({\n adapterName: 'cloudflare',\n enrich: enrichFromRequest,\n});\n\nexport { parseError, createDrainPipeline, createStructuredError };\n"]}
|
|
1
|
+
{"version":3,"file":"cloudflare.cjs","names":["getHeader","createUseLogger","createAdapterToolkit"],"sources":["../src/cloudflare.ts"],"sourcesContent":["import {\n createDrainPipeline,\n createStructuredError,\n getRequestLogger,\n parseError,\n trace,\n type DrainPipelineOptions,\n type ParsedError,\n type PipelineDrainFn,\n type RequestLogger,\n type RequestLoggerOptions,\n type StructuredError,\n type StructuredErrorInput,\n} from 'autotel';\nimport { createAdapterToolkit, createUseLogger, getHeader } from './core';\n\nexport interface CloudflareRequestLike {\n method?: string;\n url?: string;\n headers?:\n | { get(name: string): string | null }\n | Record<string, string | undefined>;\n cf?: Record<string, unknown>;\n}\n\nexport interface CloudflareExecutionContextLike {\n waitUntil?: (promise: Promise<unknown>) => void;\n passThroughOnException?: () => void;\n}\n\nexport interface CloudflareWithAutotelOptions<TEnv = unknown> {\n spanName?: string | ((request: CloudflareRequestLike, env: TEnv) => string);\n requestLoggerOptions?: RequestLoggerOptions;\n enrich?: (\n request: CloudflareRequestLike,\n env: TEnv,\n ctx: CloudflareExecutionContextLike,\n ) => Record<string, unknown> | undefined;\n /** Emit one wide event automatically when the handler settles. Default `true`. */\n autoEmit?: boolean;\n}\n\nconst requestLoggers = new WeakMap<object, RequestLogger>();\n\nfunction enrichFromRequest(\n request?: CloudflareRequestLike,\n): Record<string, unknown> | undefined {\n if (!request) return undefined;\n\n let route = '/';\n if (request.url) {\n try {\n route = new URL(request.url).pathname;\n } catch {\n route = request.url;\n }\n }\n\n const requestId =\n getHeader(request.headers, 'x-request-id') ??\n getHeader(request.headers, 'cf-ray');\n\n return {\n ...(request.method ? { 'http.request.method': request.method } : {}),\n ...(request.url ? { 'url.full': request.url } : {}),\n ...(route ? { 'http.route': route } : {}),\n ...(requestId ? { 'http.request.id': requestId } : {}),\n ...(request.cf?.country ? { 'cloudflare.country': request.cf.country } : {}),\n ...(request.cf?.colo ? { 'cloudflare.colo': request.cf.colo } : {}),\n ...(request.cf?.city ? { 'cloudflare.city': request.cf.city } : {}),\n };\n}\n\nconst baseUseLogger = createUseLogger<CloudflareRequestLike>({\n adapterName: 'cloudflare',\n enrich: enrichFromRequest,\n});\n\nexport function useLogger(\n request?: CloudflareRequestLike,\n requestLoggerOptions?: RequestLoggerOptions,\n): RequestLogger {\n if (request) {\n const existing = requestLoggers.get(request as object);\n if (existing) return existing;\n }\n return baseUseLogger(request, requestLoggerOptions);\n}\n\nexport function withAutotelFetch<\n TEnv,\n TRequest extends CloudflareRequestLike,\n TContext extends CloudflareExecutionContextLike,\n TReturn,\n>(\n handler: (\n request: TRequest,\n env: TEnv,\n ctx: TContext,\n ) => TReturn | Promise<TReturn>,\n options?: CloudflareWithAutotelOptions<TEnv>,\n): (\n request: TRequest,\n env: TEnv,\n ctx: TContext,\n) => Promise<TReturn> {\n return async (\n request: TRequest,\n env: TEnv,\n executionContext: TContext,\n ): Promise<TReturn> => {\n const spanName =\n typeof options?.spanName === 'function'\n ? options.spanName(request, env)\n : (options?.spanName ?? `cloudflare.${request.method ?? 'request'}`);\n\n const wrapped = trace(\n { name: spanName },\n (ctx) => async (\n innerRequest: TRequest,\n innerEnv: TEnv,\n innerExecutionContext: TContext,\n ) => {\n const log = getRequestLogger(ctx, options?.requestLoggerOptions);\n const auto = enrichFromRequest(innerRequest);\n if (auto && Object.keys(auto).length > 0) {\n log.set(auto);\n }\n const custom = options?.enrich?.(\n innerRequest,\n innerEnv,\n innerExecutionContext,\n );\n if (custom && Object.keys(custom).length > 0) {\n log.set(custom);\n }\n\n requestLoggers.set(innerRequest as object, log);\n try {\n return await handler(innerRequest, innerEnv, innerExecutionContext);\n } finally {\n if (options?.autoEmit !== false) {\n log.emitNow();\n }\n requestLoggers.delete(innerRequest as object);\n }\n },\n );\n\n return await wrapped(request, env, executionContext);\n };\n}\n\nexport function createCloudflareAdapter<TEnv = unknown>(\n options?: CloudflareWithAutotelOptions<TEnv>,\n) {\n return {\n withAutotelFetch: <\n TRequest extends CloudflareRequestLike,\n TContext extends CloudflareExecutionContextLike,\n TReturn,\n >(\n handler: (\n request: TRequest,\n env: TEnv,\n ctx: TContext,\n ) => TReturn | Promise<TReturn>,\n ) => withAutotelFetch(handler, options),\n useLogger,\n parseError: (error: unknown): ParsedError => parseError(error),\n createStructuredError: (\n input: StructuredErrorInput,\n ): StructuredError => createStructuredError(input),\n createDrainPipeline: <T = unknown>(\n drainOptions?: DrainPipelineOptions<T>,\n ): ((batchDrain: (batch: T[]) => void | Promise<void>) => PipelineDrainFn<T>) =>\n createDrainPipeline(drainOptions),\n };\n}\n\nexport const cloudflareToolkit = createAdapterToolkit<CloudflareRequestLike>({\n adapterName: 'cloudflare',\n enrich: enrichFromRequest,\n});\n\nexport { parseError, createDrainPipeline, createStructuredError };\n"],"mappings":";;;;;AA0CA,MAAM,iCAAiB,IAAI,QAA+B;AAE1D,SAAS,kBACP,SACqC;CACrC,IAAI,CAAC,SAAS,OAAO;CAErB,IAAI,QAAQ;CACZ,IAAI,QAAQ,KACV,IAAI;EACF,QAAQ,IAAI,IAAI,QAAQ,GAAG,CAAC,CAAC;CAC/B,QAAQ;EACN,QAAQ,QAAQ;CAClB;CAGF,MAAM,YACJA,uBAAU,QAAQ,SAAS,cAAc,KACzCA,uBAAU,QAAQ,SAAS,QAAQ;CAErC,OAAO;EACL,GAAI,QAAQ,SAAS,EAAE,uBAAuB,QAAQ,OAAO,IAAI,CAAC;EAClE,GAAI,QAAQ,MAAM,EAAE,YAAY,QAAQ,IAAI,IAAI,CAAC;EACjD,GAAI,QAAQ,EAAE,cAAc,MAAM,IAAI,CAAC;EACvC,GAAI,YAAY,EAAE,mBAAmB,UAAU,IAAI,CAAC;EACpD,GAAI,QAAQ,IAAI,UAAU,EAAE,sBAAsB,QAAQ,GAAG,QAAQ,IAAI,CAAC;EAC1E,GAAI,QAAQ,IAAI,OAAO,EAAE,mBAAmB,QAAQ,GAAG,KAAK,IAAI,CAAC;EACjE,GAAI,QAAQ,IAAI,OAAO,EAAE,mBAAmB,QAAQ,GAAG,KAAK,IAAI,CAAC;CACnE;AACF;AAEA,MAAM,gBAAgBC,6BAAuC;CAC3D,aAAa;CACb,QAAQ;AACV,CAAC;AAED,SAAgB,UACd,SACA,sBACe;CACf,IAAI,SAAS;EACX,MAAM,WAAW,eAAe,IAAI,OAAiB;EACrD,IAAI,UAAU,OAAO;CACvB;CACA,OAAO,cAAc,SAAS,oBAAoB;AACpD;AAEA,SAAgB,iBAMd,SAKA,SAKoB;CACpB,OAAO,OACL,SACA,KACA,qBACqB;EAuCrB,OAAO,yBAhCL,EAAE,MALF,OAAO,SAAS,aAAa,aACzB,QAAQ,SAAS,SAAS,GAAG,IAC5B,SAAS,YAAY,cAAc,QAAQ,UAAU,YAGzC,IAChB,QAAQ,OACP,cACA,UACA,0BACG;GACH,MAAM,oCAAuB,KAAK,SAAS,oBAAoB;GAC/D,MAAM,OAAO,kBAAkB,YAAY;GAC3C,IAAI,QAAQ,OAAO,KAAK,IAAI,CAAC,CAAC,SAAS,GACrC,IAAI,IAAI,IAAI;GAEd,MAAM,SAAS,SAAS,SACtB,cACA,UACA,qBACF;GACA,IAAI,UAAU,OAAO,KAAK,MAAM,CAAC,CAAC,SAAS,GACzC,IAAI,IAAI,MAAM;GAGhB,eAAe,IAAI,cAAwB,GAAG;GAC9C,IAAI;IACF,OAAO,MAAM,QAAQ,cAAc,UAAU,qBAAqB;GACpE,UAAU;IACR,IAAI,SAAS,aAAa,OACxB,IAAI,QAAQ;IAEd,eAAe,OAAO,YAAsB;GAC9C;EACF,CAGiB,CAAC,CAAC,SAAS,KAAK,gBAAgB;CACrD;AACF;AAEA,SAAgB,wBACd,SACA;CACA,OAAO;EACL,mBAKE,YAKG,iBAAiB,SAAS,OAAO;EACtC;EACA,aAAa,kCAA2C,KAAK;EAC7D,wBACE,6CAC0C,KAAK;EACjD,sBACE,kDAEoB,YAAY;CACpC;AACF;AAEA,MAAa,oBAAoBC,kCAA4C;CAC3E,aAAa;CACb,QAAQ;AACV,CAAC"}
|
package/dist/cloudflare.d.cts
CHANGED
|
@@ -1,36 +1,36 @@
|
|
|
1
|
-
import { AdapterToolkit } from
|
|
2
|
-
import {
|
|
3
|
-
export { createDrainPipeline, createStructuredError, parseError } from 'autotel';
|
|
4
|
-
import 'node:async_hooks';
|
|
1
|
+
import { AdapterToolkit } from "./core.cjs";
|
|
2
|
+
import { DrainPipelineOptions, ParsedError, PipelineDrainFn, RequestLogger, RequestLoggerOptions, StructuredError, StructuredErrorInput, createDrainPipeline, createStructuredError, parseError } from "autotel";
|
|
5
3
|
|
|
4
|
+
//#region src/cloudflare.d.ts
|
|
6
5
|
interface CloudflareRequestLike {
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
6
|
+
method?: string;
|
|
7
|
+
url?: string;
|
|
8
|
+
headers?: {
|
|
9
|
+
get(name: string): string | null;
|
|
10
|
+
} | Record<string, string | undefined>;
|
|
11
|
+
cf?: Record<string, unknown>;
|
|
13
12
|
}
|
|
14
13
|
interface CloudflareExecutionContextLike {
|
|
15
|
-
|
|
16
|
-
|
|
14
|
+
waitUntil?: (promise: Promise<unknown>) => void;
|
|
15
|
+
passThroughOnException?: () => void;
|
|
17
16
|
}
|
|
18
17
|
interface CloudflareWithAutotelOptions<TEnv = unknown> {
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
18
|
+
spanName?: string | ((request: CloudflareRequestLike, env: TEnv) => string);
|
|
19
|
+
requestLoggerOptions?: RequestLoggerOptions;
|
|
20
|
+
enrich?: (request: CloudflareRequestLike, env: TEnv, ctx: CloudflareExecutionContextLike) => Record<string, unknown> | undefined;
|
|
21
|
+
/** Emit one wide event automatically when the handler settles. Default `true`. */
|
|
22
|
+
autoEmit?: boolean;
|
|
24
23
|
}
|
|
25
24
|
declare function useLogger(request?: CloudflareRequestLike, requestLoggerOptions?: RequestLoggerOptions): RequestLogger;
|
|
26
25
|
declare function withAutotelFetch<TEnv, TRequest extends CloudflareRequestLike, TContext extends CloudflareExecutionContextLike, TReturn>(handler: (request: TRequest, env: TEnv, ctx: TContext) => TReturn | Promise<TReturn>, options?: CloudflareWithAutotelOptions<TEnv>): (request: TRequest, env: TEnv, ctx: TContext) => Promise<TReturn>;
|
|
27
26
|
declare function createCloudflareAdapter<TEnv = unknown>(options?: CloudflareWithAutotelOptions<TEnv>): {
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
27
|
+
withAutotelFetch: <TRequest extends CloudflareRequestLike, TContext extends CloudflareExecutionContextLike, TReturn>(handler: (request: TRequest, env: TEnv, ctx: TContext) => TReturn | Promise<TReturn>) => (request: TRequest, env: TEnv, ctx: TContext) => Promise<TReturn>;
|
|
28
|
+
useLogger: typeof useLogger;
|
|
29
|
+
parseError: (error: unknown) => ParsedError;
|
|
30
|
+
createStructuredError: (input: StructuredErrorInput) => StructuredError;
|
|
31
|
+
createDrainPipeline: <T = unknown>(drainOptions?: DrainPipelineOptions<T>) => ((batchDrain: (batch: T[]) => void | Promise<void>) => PipelineDrainFn<T>);
|
|
33
32
|
};
|
|
34
33
|
declare const cloudflareToolkit: AdapterToolkit<CloudflareRequestLike>;
|
|
35
|
-
|
|
36
|
-
export {
|
|
34
|
+
//#endregion
|
|
35
|
+
export { CloudflareExecutionContextLike, CloudflareRequestLike, CloudflareWithAutotelOptions, cloudflareToolkit, createCloudflareAdapter, createDrainPipeline, createStructuredError, parseError, useLogger, withAutotelFetch };
|
|
36
|
+
//# sourceMappingURL=cloudflare.d.cts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cloudflare.d.cts","names":[],"sources":["../src/cloudflare.ts"],"mappings":";;;;UAgBiB,qBAAA;EACf,MAAA;EACA,GAAA;EACA,OAAA;IACM,GAAA,CAAI,IAAA;EAAA,IACN,MAAA;EACJ,EAAA,GAAK,MAAM;AAAA;AAAA,UAGI,8BAAA;EACf,SAAA,IAAa,OAAA,EAAS,OAAO;EAC7B,sBAAA;AAAA;AAAA,UAGe,4BAAA;EACf,QAAA,cAAsB,OAAA,EAAS,qBAAA,EAAuB,GAAA,EAAK,IAAA;EAC3D,oBAAA,GAAuB,oBAAA;EACvB,MAAA,IACE,OAAA,EAAS,qBAAA,EACT,GAAA,EAAK,IAAA,EACL,GAAA,EAAK,8BAAA,KACF,MAAA;EAfM;EAiBX,QAAA;AAAA;AAAA,iBAuCc,SAAA,CACd,OAAA,GAAU,qBAAA,EACV,oBAAA,GAAuB,oBAAA,GACtB,aAAA;AAAA,iBAQa,gBAAA,wBAEG,qBAAA,mBACA,8BAAA,WAGjB,OAAA,GACE,OAAA,EAAS,QAAA,EACT,GAAA,EAAK,IAAA,EACL,GAAA,EAAK,QAAA,KACF,OAAA,GAAU,OAAA,CAAQ,OAAA,GACvB,OAAA,GAAU,4BAAA,CAA6B,IAAA,KAEvC,OAAA,EAAS,QAAA,EACT,GAAA,EAAK,IAAA,EACL,GAAA,EAAK,QAAA,KACF,OAAA,CAAQ,OAAA;AAAA,iBAgDG,uBAAA,iBACd,OAAA,GAAU,4BAAA,CAA6B,IAAA;sCAIlB,qBAAA,mBACA,8BAAA,WACV,OAAA,GAGL,OAAA,EAAS,QAAA,EACT,GAAA,EAAK,IAAA,EACL,GAAA,EAAK,QAAA,KACF,OAAA,GAAU,OAAA,CAAQ,OAAA,OAAQ,OAAA,EAAA,QAAA,EAAA,GAAA,EAAA,IAAA,EAAA,GAAA,EAAA,QAAA,KAAA,OAAA,CAAA,OAAA;;kCAGH,WAAA;iCAErB,oBAAA,KACN,eAAA;qCACoB,YAAA,GACN,oBAAA,CAAqB,CAAA,QACjC,UAAA,GAAa,KAAA,EAAO,CAAA,cAAe,OAAA,WAAkB,eAAA,CAAgB,CAAA;AAAA;AAAA,cAKjE,iBAAA,EAAiB,cAAA,CAAA,qBAAA"}
|
package/dist/cloudflare.d.ts
CHANGED
|
@@ -1,36 +1,36 @@
|
|
|
1
|
-
import { AdapterToolkit } from
|
|
2
|
-
import {
|
|
3
|
-
export { createDrainPipeline, createStructuredError, parseError } from 'autotel';
|
|
4
|
-
import 'node:async_hooks';
|
|
1
|
+
import { AdapterToolkit } from "./core.js";
|
|
2
|
+
import { DrainPipelineOptions, ParsedError, PipelineDrainFn, RequestLogger, RequestLoggerOptions, StructuredError, StructuredErrorInput, createDrainPipeline, createStructuredError, parseError } from "autotel";
|
|
5
3
|
|
|
4
|
+
//#region src/cloudflare.d.ts
|
|
6
5
|
interface CloudflareRequestLike {
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
6
|
+
method?: string;
|
|
7
|
+
url?: string;
|
|
8
|
+
headers?: {
|
|
9
|
+
get(name: string): string | null;
|
|
10
|
+
} | Record<string, string | undefined>;
|
|
11
|
+
cf?: Record<string, unknown>;
|
|
13
12
|
}
|
|
14
13
|
interface CloudflareExecutionContextLike {
|
|
15
|
-
|
|
16
|
-
|
|
14
|
+
waitUntil?: (promise: Promise<unknown>) => void;
|
|
15
|
+
passThroughOnException?: () => void;
|
|
17
16
|
}
|
|
18
17
|
interface CloudflareWithAutotelOptions<TEnv = unknown> {
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
18
|
+
spanName?: string | ((request: CloudflareRequestLike, env: TEnv) => string);
|
|
19
|
+
requestLoggerOptions?: RequestLoggerOptions;
|
|
20
|
+
enrich?: (request: CloudflareRequestLike, env: TEnv, ctx: CloudflareExecutionContextLike) => Record<string, unknown> | undefined;
|
|
21
|
+
/** Emit one wide event automatically when the handler settles. Default `true`. */
|
|
22
|
+
autoEmit?: boolean;
|
|
24
23
|
}
|
|
25
24
|
declare function useLogger(request?: CloudflareRequestLike, requestLoggerOptions?: RequestLoggerOptions): RequestLogger;
|
|
26
25
|
declare function withAutotelFetch<TEnv, TRequest extends CloudflareRequestLike, TContext extends CloudflareExecutionContextLike, TReturn>(handler: (request: TRequest, env: TEnv, ctx: TContext) => TReturn | Promise<TReturn>, options?: CloudflareWithAutotelOptions<TEnv>): (request: TRequest, env: TEnv, ctx: TContext) => Promise<TReturn>;
|
|
27
26
|
declare function createCloudflareAdapter<TEnv = unknown>(options?: CloudflareWithAutotelOptions<TEnv>): {
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
27
|
+
withAutotelFetch: <TRequest extends CloudflareRequestLike, TContext extends CloudflareExecutionContextLike, TReturn>(handler: (request: TRequest, env: TEnv, ctx: TContext) => TReturn | Promise<TReturn>) => (request: TRequest, env: TEnv, ctx: TContext) => Promise<TReturn>;
|
|
28
|
+
useLogger: typeof useLogger;
|
|
29
|
+
parseError: (error: unknown) => ParsedError;
|
|
30
|
+
createStructuredError: (input: StructuredErrorInput) => StructuredError;
|
|
31
|
+
createDrainPipeline: <T = unknown>(drainOptions?: DrainPipelineOptions<T>) => ((batchDrain: (batch: T[]) => void | Promise<void>) => PipelineDrainFn<T>);
|
|
33
32
|
};
|
|
34
33
|
declare const cloudflareToolkit: AdapterToolkit<CloudflareRequestLike>;
|
|
35
|
-
|
|
36
|
-
export {
|
|
34
|
+
//#endregion
|
|
35
|
+
export { CloudflareExecutionContextLike, CloudflareRequestLike, CloudflareWithAutotelOptions, cloudflareToolkit, createCloudflareAdapter, createDrainPipeline, createStructuredError, parseError, useLogger, withAutotelFetch };
|
|
36
|
+
//# sourceMappingURL=cloudflare.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cloudflare.d.ts","names":[],"sources":["../src/cloudflare.ts"],"mappings":";;;;UAgBiB,qBAAA;EACf,MAAA;EACA,GAAA;EACA,OAAA;IACM,GAAA,CAAI,IAAA;EAAA,IACN,MAAA;EACJ,EAAA,GAAK,MAAM;AAAA;AAAA,UAGI,8BAAA;EACf,SAAA,IAAa,OAAA,EAAS,OAAO;EAC7B,sBAAA;AAAA;AAAA,UAGe,4BAAA;EACf,QAAA,cAAsB,OAAA,EAAS,qBAAA,EAAuB,GAAA,EAAK,IAAA;EAC3D,oBAAA,GAAuB,oBAAA;EACvB,MAAA,IACE,OAAA,EAAS,qBAAA,EACT,GAAA,EAAK,IAAA,EACL,GAAA,EAAK,8BAAA,KACF,MAAA;EAfM;EAiBX,QAAA;AAAA;AAAA,iBAuCc,SAAA,CACd,OAAA,GAAU,qBAAA,EACV,oBAAA,GAAuB,oBAAA,GACtB,aAAA;AAAA,iBAQa,gBAAA,wBAEG,qBAAA,mBACA,8BAAA,WAGjB,OAAA,GACE,OAAA,EAAS,QAAA,EACT,GAAA,EAAK,IAAA,EACL,GAAA,EAAK,QAAA,KACF,OAAA,GAAU,OAAA,CAAQ,OAAA,GACvB,OAAA,GAAU,4BAAA,CAA6B,IAAA,KAEvC,OAAA,EAAS,QAAA,EACT,GAAA,EAAK,IAAA,EACL,GAAA,EAAK,QAAA,KACF,OAAA,CAAQ,OAAA;AAAA,iBAgDG,uBAAA,iBACd,OAAA,GAAU,4BAAA,CAA6B,IAAA;sCAIlB,qBAAA,mBACA,8BAAA,WACV,OAAA,GAGL,OAAA,EAAS,QAAA,EACT,GAAA,EAAK,IAAA,EACL,GAAA,EAAK,QAAA,KACF,OAAA,GAAU,OAAA,CAAQ,OAAA,OAAQ,OAAA,EAAA,QAAA,EAAA,GAAA,EAAA,IAAA,EAAA,GAAA,EAAA,QAAA,KAAA,OAAA,CAAA,OAAA;;kCAGH,WAAA;iCAErB,oBAAA,KACN,eAAA;qCACoB,YAAA,GACN,oBAAA,CAAqB,CAAA,QACjC,UAAA,GAAa,KAAA,EAAO,CAAA,cAAe,OAAA,WAAkB,eAAA,CAAgB,CAAA;AAAA;AAAA,cAKjE,iBAAA,EAAiB,cAAA,CAAA,qBAAA"}
|
package/dist/cloudflare.js
CHANGED
|
@@ -1,4 +1,70 @@
|
|
|
1
|
-
|
|
2
|
-
import
|
|
3
|
-
|
|
1
|
+
import { createAdapterToolkit, createUseLogger, getHeader } from "./core.js";
|
|
2
|
+
import { createDrainPipeline, createStructuredError, getRequestLogger, parseError, trace } from "autotel";
|
|
3
|
+
|
|
4
|
+
//#region src/cloudflare.ts
|
|
5
|
+
const requestLoggers = /* @__PURE__ */ new WeakMap();
|
|
6
|
+
function enrichFromRequest(request) {
|
|
7
|
+
if (!request) return void 0;
|
|
8
|
+
let route = "/";
|
|
9
|
+
if (request.url) try {
|
|
10
|
+
route = new URL(request.url).pathname;
|
|
11
|
+
} catch {
|
|
12
|
+
route = request.url;
|
|
13
|
+
}
|
|
14
|
+
const requestId = getHeader(request.headers, "x-request-id") ?? getHeader(request.headers, "cf-ray");
|
|
15
|
+
return {
|
|
16
|
+
...request.method ? { "http.request.method": request.method } : {},
|
|
17
|
+
...request.url ? { "url.full": request.url } : {},
|
|
18
|
+
...route ? { "http.route": route } : {},
|
|
19
|
+
...requestId ? { "http.request.id": requestId } : {},
|
|
20
|
+
...request.cf?.country ? { "cloudflare.country": request.cf.country } : {},
|
|
21
|
+
...request.cf?.colo ? { "cloudflare.colo": request.cf.colo } : {},
|
|
22
|
+
...request.cf?.city ? { "cloudflare.city": request.cf.city } : {}
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
const baseUseLogger = createUseLogger({
|
|
26
|
+
adapterName: "cloudflare",
|
|
27
|
+
enrich: enrichFromRequest
|
|
28
|
+
});
|
|
29
|
+
function useLogger(request, requestLoggerOptions) {
|
|
30
|
+
if (request) {
|
|
31
|
+
const existing = requestLoggers.get(request);
|
|
32
|
+
if (existing) return existing;
|
|
33
|
+
}
|
|
34
|
+
return baseUseLogger(request, requestLoggerOptions);
|
|
35
|
+
}
|
|
36
|
+
function withAutotelFetch(handler, options) {
|
|
37
|
+
return async (request, env, executionContext) => {
|
|
38
|
+
return await trace({ name: typeof options?.spanName === "function" ? options.spanName(request, env) : options?.spanName ?? `cloudflare.${request.method ?? "request"}` }, (ctx) => async (innerRequest, innerEnv, innerExecutionContext) => {
|
|
39
|
+
const log = getRequestLogger(ctx, options?.requestLoggerOptions);
|
|
40
|
+
const auto = enrichFromRequest(innerRequest);
|
|
41
|
+
if (auto && Object.keys(auto).length > 0) log.set(auto);
|
|
42
|
+
const custom = options?.enrich?.(innerRequest, innerEnv, innerExecutionContext);
|
|
43
|
+
if (custom && Object.keys(custom).length > 0) log.set(custom);
|
|
44
|
+
requestLoggers.set(innerRequest, log);
|
|
45
|
+
try {
|
|
46
|
+
return await handler(innerRequest, innerEnv, innerExecutionContext);
|
|
47
|
+
} finally {
|
|
48
|
+
if (options?.autoEmit !== false) log.emitNow();
|
|
49
|
+
requestLoggers.delete(innerRequest);
|
|
50
|
+
}
|
|
51
|
+
})(request, env, executionContext);
|
|
52
|
+
};
|
|
53
|
+
}
|
|
54
|
+
function createCloudflareAdapter(options) {
|
|
55
|
+
return {
|
|
56
|
+
withAutotelFetch: (handler) => withAutotelFetch(handler, options),
|
|
57
|
+
useLogger,
|
|
58
|
+
parseError: (error) => parseError(error),
|
|
59
|
+
createStructuredError: (input) => createStructuredError(input),
|
|
60
|
+
createDrainPipeline: (drainOptions) => createDrainPipeline(drainOptions)
|
|
61
|
+
};
|
|
62
|
+
}
|
|
63
|
+
const cloudflareToolkit = createAdapterToolkit({
|
|
64
|
+
adapterName: "cloudflare",
|
|
65
|
+
enrich: enrichFromRequest
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
//#endregion
|
|
69
|
+
export { cloudflareToolkit, createCloudflareAdapter, createDrainPipeline, createStructuredError, parseError, useLogger, withAutotelFetch };
|
|
4
70
|
//# sourceMappingURL=cloudflare.js.map
|
package/dist/cloudflare.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"
|
|
1
|
+
{"version":3,"file":"cloudflare.js","names":[],"sources":["../src/cloudflare.ts"],"sourcesContent":["import {\n createDrainPipeline,\n createStructuredError,\n getRequestLogger,\n parseError,\n trace,\n type DrainPipelineOptions,\n type ParsedError,\n type PipelineDrainFn,\n type RequestLogger,\n type RequestLoggerOptions,\n type StructuredError,\n type StructuredErrorInput,\n} from 'autotel';\nimport { createAdapterToolkit, createUseLogger, getHeader } from './core';\n\nexport interface CloudflareRequestLike {\n method?: string;\n url?: string;\n headers?:\n | { get(name: string): string | null }\n | Record<string, string | undefined>;\n cf?: Record<string, unknown>;\n}\n\nexport interface CloudflareExecutionContextLike {\n waitUntil?: (promise: Promise<unknown>) => void;\n passThroughOnException?: () => void;\n}\n\nexport interface CloudflareWithAutotelOptions<TEnv = unknown> {\n spanName?: string | ((request: CloudflareRequestLike, env: TEnv) => string);\n requestLoggerOptions?: RequestLoggerOptions;\n enrich?: (\n request: CloudflareRequestLike,\n env: TEnv,\n ctx: CloudflareExecutionContextLike,\n ) => Record<string, unknown> | undefined;\n /** Emit one wide event automatically when the handler settles. Default `true`. */\n autoEmit?: boolean;\n}\n\nconst requestLoggers = new WeakMap<object, RequestLogger>();\n\nfunction enrichFromRequest(\n request?: CloudflareRequestLike,\n): Record<string, unknown> | undefined {\n if (!request) return undefined;\n\n let route = '/';\n if (request.url) {\n try {\n route = new URL(request.url).pathname;\n } catch {\n route = request.url;\n }\n }\n\n const requestId =\n getHeader(request.headers, 'x-request-id') ??\n getHeader(request.headers, 'cf-ray');\n\n return {\n ...(request.method ? { 'http.request.method': request.method } : {}),\n ...(request.url ? { 'url.full': request.url } : {}),\n ...(route ? { 'http.route': route } : {}),\n ...(requestId ? { 'http.request.id': requestId } : {}),\n ...(request.cf?.country ? { 'cloudflare.country': request.cf.country } : {}),\n ...(request.cf?.colo ? { 'cloudflare.colo': request.cf.colo } : {}),\n ...(request.cf?.city ? { 'cloudflare.city': request.cf.city } : {}),\n };\n}\n\nconst baseUseLogger = createUseLogger<CloudflareRequestLike>({\n adapterName: 'cloudflare',\n enrich: enrichFromRequest,\n});\n\nexport function useLogger(\n request?: CloudflareRequestLike,\n requestLoggerOptions?: RequestLoggerOptions,\n): RequestLogger {\n if (request) {\n const existing = requestLoggers.get(request as object);\n if (existing) return existing;\n }\n return baseUseLogger(request, requestLoggerOptions);\n}\n\nexport function withAutotelFetch<\n TEnv,\n TRequest extends CloudflareRequestLike,\n TContext extends CloudflareExecutionContextLike,\n TReturn,\n>(\n handler: (\n request: TRequest,\n env: TEnv,\n ctx: TContext,\n ) => TReturn | Promise<TReturn>,\n options?: CloudflareWithAutotelOptions<TEnv>,\n): (\n request: TRequest,\n env: TEnv,\n ctx: TContext,\n) => Promise<TReturn> {\n return async (\n request: TRequest,\n env: TEnv,\n executionContext: TContext,\n ): Promise<TReturn> => {\n const spanName =\n typeof options?.spanName === 'function'\n ? options.spanName(request, env)\n : (options?.spanName ?? `cloudflare.${request.method ?? 'request'}`);\n\n const wrapped = trace(\n { name: spanName },\n (ctx) => async (\n innerRequest: TRequest,\n innerEnv: TEnv,\n innerExecutionContext: TContext,\n ) => {\n const log = getRequestLogger(ctx, options?.requestLoggerOptions);\n const auto = enrichFromRequest(innerRequest);\n if (auto && Object.keys(auto).length > 0) {\n log.set(auto);\n }\n const custom = options?.enrich?.(\n innerRequest,\n innerEnv,\n innerExecutionContext,\n );\n if (custom && Object.keys(custom).length > 0) {\n log.set(custom);\n }\n\n requestLoggers.set(innerRequest as object, log);\n try {\n return await handler(innerRequest, innerEnv, innerExecutionContext);\n } finally {\n if (options?.autoEmit !== false) {\n log.emitNow();\n }\n requestLoggers.delete(innerRequest as object);\n }\n },\n );\n\n return await wrapped(request, env, executionContext);\n };\n}\n\nexport function createCloudflareAdapter<TEnv = unknown>(\n options?: CloudflareWithAutotelOptions<TEnv>,\n) {\n return {\n withAutotelFetch: <\n TRequest extends CloudflareRequestLike,\n TContext extends CloudflareExecutionContextLike,\n TReturn,\n >(\n handler: (\n request: TRequest,\n env: TEnv,\n ctx: TContext,\n ) => TReturn | Promise<TReturn>,\n ) => withAutotelFetch(handler, options),\n useLogger,\n parseError: (error: unknown): ParsedError => parseError(error),\n createStructuredError: (\n input: StructuredErrorInput,\n ): StructuredError => createStructuredError(input),\n createDrainPipeline: <T = unknown>(\n drainOptions?: DrainPipelineOptions<T>,\n ): ((batchDrain: (batch: T[]) => void | Promise<void>) => PipelineDrainFn<T>) =>\n createDrainPipeline(drainOptions),\n };\n}\n\nexport const cloudflareToolkit = createAdapterToolkit<CloudflareRequestLike>({\n adapterName: 'cloudflare',\n enrich: enrichFromRequest,\n});\n\nexport { parseError, createDrainPipeline, createStructuredError };\n"],"mappings":";;;;AA0CA,MAAM,iCAAiB,IAAI,QAA+B;AAE1D,SAAS,kBACP,SACqC;CACrC,IAAI,CAAC,SAAS,OAAO;CAErB,IAAI,QAAQ;CACZ,IAAI,QAAQ,KACV,IAAI;EACF,QAAQ,IAAI,IAAI,QAAQ,GAAG,CAAC,CAAC;CAC/B,QAAQ;EACN,QAAQ,QAAQ;CAClB;CAGF,MAAM,YACJ,UAAU,QAAQ,SAAS,cAAc,KACzC,UAAU,QAAQ,SAAS,QAAQ;CAErC,OAAO;EACL,GAAI,QAAQ,SAAS,EAAE,uBAAuB,QAAQ,OAAO,IAAI,CAAC;EAClE,GAAI,QAAQ,MAAM,EAAE,YAAY,QAAQ,IAAI,IAAI,CAAC;EACjD,GAAI,QAAQ,EAAE,cAAc,MAAM,IAAI,CAAC;EACvC,GAAI,YAAY,EAAE,mBAAmB,UAAU,IAAI,CAAC;EACpD,GAAI,QAAQ,IAAI,UAAU,EAAE,sBAAsB,QAAQ,GAAG,QAAQ,IAAI,CAAC;EAC1E,GAAI,QAAQ,IAAI,OAAO,EAAE,mBAAmB,QAAQ,GAAG,KAAK,IAAI,CAAC;EACjE,GAAI,QAAQ,IAAI,OAAO,EAAE,mBAAmB,QAAQ,GAAG,KAAK,IAAI,CAAC;CACnE;AACF;AAEA,MAAM,gBAAgB,gBAAuC;CAC3D,aAAa;CACb,QAAQ;AACV,CAAC;AAED,SAAgB,UACd,SACA,sBACe;CACf,IAAI,SAAS;EACX,MAAM,WAAW,eAAe,IAAI,OAAiB;EACrD,IAAI,UAAU,OAAO;CACvB;CACA,OAAO,cAAc,SAAS,oBAAoB;AACpD;AAEA,SAAgB,iBAMd,SAKA,SAKoB;CACpB,OAAO,OACL,SACA,KACA,qBACqB;EAuCrB,OAAO,MAjCS,MACd,EAAE,MALF,OAAO,SAAS,aAAa,aACzB,QAAQ,SAAS,SAAS,GAAG,IAC5B,SAAS,YAAY,cAAc,QAAQ,UAAU,YAGzC,IAChB,QAAQ,OACP,cACA,UACA,0BACG;GACH,MAAM,MAAM,iBAAiB,KAAK,SAAS,oBAAoB;GAC/D,MAAM,OAAO,kBAAkB,YAAY;GAC3C,IAAI,QAAQ,OAAO,KAAK,IAAI,CAAC,CAAC,SAAS,GACrC,IAAI,IAAI,IAAI;GAEd,MAAM,SAAS,SAAS,SACtB,cACA,UACA,qBACF;GACA,IAAI,UAAU,OAAO,KAAK,MAAM,CAAC,CAAC,SAAS,GACzC,IAAI,IAAI,MAAM;GAGhB,eAAe,IAAI,cAAwB,GAAG;GAC9C,IAAI;IACF,OAAO,MAAM,QAAQ,cAAc,UAAU,qBAAqB;GACpE,UAAU;IACR,IAAI,SAAS,aAAa,OACxB,IAAI,QAAQ;IAEd,eAAe,OAAO,YAAsB;GAC9C;EACF,CAGiB,CAAC,CAAC,SAAS,KAAK,gBAAgB;CACrD;AACF;AAEA,SAAgB,wBACd,SACA;CACA,OAAO;EACL,mBAKE,YAKG,iBAAiB,SAAS,OAAO;EACtC;EACA,aAAa,UAAgC,WAAW,KAAK;EAC7D,wBACE,UACoB,sBAAsB,KAAK;EACjD,sBACE,iBAEA,oBAAoB,YAAY;CACpC;AACF;AAEA,MAAa,oBAAoB,qBAA4C;CAC3E,aAAa;CACb,QAAQ;AACV,CAAC"}
|