evlog 2.18.1 → 2.19.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +14 -16
- package/dist/adapters/axiom.d.mts +3 -3
- package/dist/adapters/axiom.d.mts.map +1 -1
- package/dist/adapters/axiom.mjs +10 -14
- package/dist/adapters/axiom.mjs.map +1 -1
- package/dist/adapters/better-stack.d.mts +2 -2
- package/dist/adapters/better-stack.d.mts.map +1 -1
- package/dist/adapters/better-stack.mjs +9 -13
- package/dist/adapters/better-stack.mjs.map +1 -1
- package/dist/adapters/datadog.d.mts +3 -3
- package/dist/adapters/datadog.mjs +9 -5
- package/dist/adapters/datadog.mjs.map +1 -1
- package/dist/adapters/fs.d.mts +1 -1
- package/dist/adapters/fs.d.mts.map +1 -1
- package/dist/adapters/fs.mjs +15 -2
- package/dist/adapters/fs.mjs.map +1 -1
- package/dist/adapters/hyperdx.d.mts +2 -2
- package/dist/adapters/hyperdx.mjs +3 -3
- package/dist/adapters/hyperdx.mjs.map +1 -1
- package/dist/adapters/memory.d.mts +2 -3
- package/dist/adapters/memory.d.mts.map +1 -1
- package/dist/adapters/memory.mjs +2 -3
- package/dist/adapters/memory.mjs.map +1 -1
- package/dist/adapters/otlp.d.mts +4 -4
- package/dist/adapters/otlp.mjs +19 -11
- package/dist/adapters/otlp.mjs.map +1 -1
- package/dist/adapters/posthog.d.mts +2 -2
- package/dist/adapters/posthog.mjs +5 -5
- package/dist/adapters/posthog.mjs.map +1 -1
- package/dist/adapters/sentry.d.mts +3 -3
- package/dist/adapters/sentry.mjs +6 -6
- package/dist/adapters/sentry.mjs.map +1 -1
- package/dist/ai/index.d.mts +1 -1
- package/dist/{audit-BUI3af4w.mjs → audit-BQt8yAHo.mjs} +376 -116
- package/dist/audit-BQt8yAHo.mjs.map +1 -0
- package/dist/{audit-DVdkntSO.d.mts → audit-D7v6JHj0.d.mts} +113 -35
- package/dist/audit-D7v6JHj0.d.mts.map +1 -0
- package/dist/better-auth/index.d.mts +1 -1
- package/dist/browser.d.mts +1 -1
- package/dist/deferred-drain-jeajC8QF.mjs +36 -0
- package/dist/deferred-drain-jeajC8QF.mjs.map +1 -0
- package/dist/{define-D6OJdSUH.mjs → define-Bpaymi-h.mjs} +2 -1
- package/dist/define-Bpaymi-h.mjs.map +1 -0
- package/dist/{define-D-BVMf2l.d.mts → define-DTQpu4f6.d.mts} +8 -3
- package/dist/define-DTQpu4f6.d.mts.map +1 -0
- package/dist/dev-terminal-D4UaEm17.mjs +54 -0
- package/dist/dev-terminal-D4UaEm17.mjs.map +1 -0
- package/dist/{dist-H3GIh-KK.mjs → dist-DdQWiZn8.mjs} +1 -1
- package/dist/{dist-H3GIh-KK.mjs.map → dist-DdQWiZn8.mjs.map} +1 -1
- package/dist/{drain-7n3K6kPe.mjs → drain-fDb-eNwz.mjs} +35 -4
- package/dist/drain-fDb-eNwz.mjs.map +1 -0
- package/dist/elysia/index.d.mts +3 -3
- package/dist/elysia/index.d.mts.map +1 -1
- package/dist/elysia/index.mjs +8 -5
- package/dist/elysia/index.mjs.map +1 -1
- package/dist/enrich-error-stack-next.node-Dgm_rCf5.mjs +120 -0
- package/dist/enrich-error-stack-next.node-Dgm_rCf5.mjs.map +1 -0
- package/dist/{enricher-UW9npoB2.d.mts → enricher-CBRmQw6e.d.mts} +2 -2
- package/dist/{enricher-UW9npoB2.d.mts.map → enricher-CBRmQw6e.d.mts.map} +1 -1
- package/dist/{enricher-N0erZS87.mjs → enricher-DAWf2-Fx.mjs} +2 -2
- package/dist/{enricher-N0erZS87.mjs.map → enricher-DAWf2-Fx.mjs.map} +1 -1
- package/dist/enrichers.d.mts +2 -2
- package/dist/enrichers.mjs +2 -2
- package/dist/{error-CVtn5U7b.d.mts → error-CpghjrkY.d.mts} +2 -2
- package/dist/{error-CVtn5U7b.d.mts.map → error-CpghjrkY.d.mts.map} +1 -1
- package/dist/error.d.mts +1 -1
- package/dist/{errors-dEMNQCiL.d.mts → errors-BLU4Tfpe.d.mts} +2 -2
- package/dist/{errors-dEMNQCiL.d.mts.map → errors-BLU4Tfpe.d.mts.map} +1 -1
- package/dist/{errors-BQgyQ9xe.mjs → errors-DA0cyr8q.mjs} +1 -1
- package/dist/{errors-BQgyQ9xe.mjs.map → errors-DA0cyr8q.mjs.map} +1 -1
- package/dist/{event-1BMl7o0k.mjs → event-qwAv-7AZ.mjs} +1 -1
- package/dist/{event-1BMl7o0k.mjs.map → event-qwAv-7AZ.mjs.map} +1 -1
- package/dist/express/index.d.mts +2 -2
- package/dist/express/index.mjs +3 -3
- package/dist/fastify/index.d.mts +2 -2
- package/dist/fastify/index.mjs +2 -2
- package/dist/{fork-Bga8x-X4.mjs → fork-CgGlAaHa.mjs} +39 -14
- package/dist/fork-CgGlAaHa.mjs.map +1 -0
- package/dist/{headers-CU-QqnYg.mjs → headers-VtmnWcfn.mjs} +1 -1
- package/dist/{headers-CU-QqnYg.mjs.map → headers-VtmnWcfn.mjs.map} +1 -1
- package/dist/hono/index.d.mts +2 -2
- package/dist/hono/index.d.mts.map +1 -1
- package/dist/hono/index.mjs +10 -2
- package/dist/hono/index.mjs.map +1 -1
- package/dist/{http-B6YgAhyN.mjs → http-ChVS9GYc.mjs} +2 -2
- package/dist/{http-B6YgAhyN.mjs.map → http-ChVS9GYc.mjs.map} +1 -1
- package/dist/http.d.mts +1 -1
- package/dist/{index-ZSRQP_BI.d.mts → index-EvnrXvQM.d.mts} +15 -8
- package/dist/index-EvnrXvQM.d.mts.map +1 -0
- package/dist/index.d.mts +9 -9
- package/dist/index.mjs +9 -15
- package/dist/index.mjs.map +1 -1
- package/dist/instrumentation-create-BrjQtSKD.d.mts +115 -0
- package/dist/instrumentation-create-BrjQtSKD.d.mts.map +1 -0
- package/dist/{integration-Dhig7ae6.mjs → integration-DYp2uw8O.mjs} +3 -3
- package/dist/{integration-Dhig7ae6.mjs.map → integration-DYp2uw8O.mjs.map} +1 -1
- package/dist/{logger-CTcvd5Cc.d.mts → logger-mHIWxBhJ.d.mts} +12 -4
- package/dist/logger-mHIWxBhJ.d.mts.map +1 -0
- package/dist/logger.d.mts +2 -2
- package/dist/logger.mjs +2 -2
- package/dist/{middleware-31KhtiEF.d.mts → middleware-B_k4Mrzg.d.mts} +9 -2
- package/dist/middleware-B_k4Mrzg.d.mts.map +1 -0
- package/dist/nestjs/index.d.mts +2 -2
- package/dist/nestjs/index.mjs +4 -4
- package/dist/next/client.d.mts +1 -1
- package/dist/next/index.d.mts +4 -4
- package/dist/next/index.d.mts.map +1 -1
- package/dist/next/index.mjs +49 -38
- package/dist/next/index.mjs.map +1 -1
- package/dist/next/instrumentation/create.d.mts +2 -0
- package/dist/next/instrumentation/create.mjs +155 -0
- package/dist/next/instrumentation/create.mjs.map +1 -0
- package/dist/next/instrumentation.d.mts +2 -77
- package/dist/next/instrumentation.mjs +32 -81
- package/dist/next/instrumentation.mjs.map +1 -1
- package/dist/next/stream.d.mts +1 -1
- package/dist/next/stream.mjs +2 -2
- package/dist/next/stream.mjs.map +1 -1
- package/dist/nitro/errorHandler.mjs +23 -18
- package/dist/nitro/errorHandler.mjs.map +1 -1
- package/dist/nitro/module.d.mts +2 -2
- package/dist/nitro/module.d.mts.map +1 -1
- package/dist/nitro/module.mjs +2 -1
- package/dist/nitro/module.mjs.map +1 -1
- package/dist/nitro/plugin.mjs +74 -18
- package/dist/nitro/plugin.mjs.map +1 -1
- package/dist/nitro/v3/errorHandler.d.mts +0 -7
- package/dist/nitro/v3/errorHandler.mjs +13 -15
- package/dist/nitro/v3/errorHandler.mjs.map +1 -1
- package/dist/nitro/v3/index.d.mts +2 -2
- package/dist/nitro/v3/module.d.mts +1 -1
- package/dist/nitro/v3/module.d.mts.map +1 -1
- package/dist/nitro/v3/module.mjs +3 -3
- package/dist/nitro/v3/module.mjs.map +1 -1
- package/dist/nitro/v3/plugin.mjs +76 -44
- package/dist/nitro/v3/plugin.mjs.map +1 -1
- package/dist/nitro/v3/useLogger.d.mts +1 -1
- package/dist/nitro-ClRZLD1g.mjs +96 -0
- package/dist/nitro-ClRZLD1g.mjs.map +1 -0
- package/dist/{nitro-BRddgqSb.d.mts → nitro-_Hda8Deo.d.mts} +7 -2
- package/dist/nitro-_Hda8Deo.d.mts.map +1 -0
- package/dist/{nitroConfigBridge-NbFn-sIK.mjs → nitroConfigBridge-BkVWnSV3.mjs} +9 -2
- package/dist/nitroConfigBridge-BkVWnSV3.mjs.map +1 -0
- package/dist/{nodeResponse-BkkionWl.mjs → nodeResponse-CIEEbrNE.mjs} +1 -1
- package/dist/{nodeResponse-BkkionWl.mjs.map → nodeResponse-CIEEbrNE.mjs.map} +1 -1
- package/dist/nuxt/module.d.mts +6 -1
- package/dist/nuxt/module.d.mts.map +1 -1
- package/dist/nuxt/module.mjs +11 -4
- package/dist/nuxt/module.mjs.map +1 -1
- package/dist/orpc/index.d.mts +2 -2
- package/dist/orpc/index.d.mts.map +1 -1
- package/dist/orpc/index.mjs +5 -4
- package/dist/orpc/index.mjs.map +1 -1
- package/dist/{package-B23bR3tK.mjs → package-CNV_CXs8.mjs} +2 -2
- package/dist/package-CNV_CXs8.mjs.map +1 -0
- package/dist/{parseError-D4PIxEWo.d.mts → parseError-BeBXEd2V.d.mts} +2 -2
- package/dist/parseError-BeBXEd2V.d.mts.map +1 -0
- package/dist/pipeline.d.mts +0 -19
- package/dist/pipeline.d.mts.map +1 -1
- package/dist/pipeline.mjs +12 -0
- package/dist/pipeline.mjs.map +1 -1
- package/dist/pretty-error-THg0U0w9.mjs +288 -0
- package/dist/pretty-error-THg0U0w9.mjs.map +1 -0
- package/dist/pretty-error-snippet.node-itfCajBM.mjs +48 -0
- package/dist/pretty-error-snippet.node-itfCajBM.mjs.map +1 -0
- package/dist/react-router/index.d.mts +2 -2
- package/dist/react-router/index.mjs +5 -6
- package/dist/react-router/index.mjs.map +1 -1
- package/dist/{routes-CnIgYWf8.mjs → routes-4rMzRyTk.mjs} +1 -1
- package/dist/{routes-CnIgYWf8.mjs.map → routes-4rMzRyTk.mjs.map} +1 -1
- package/dist/runtime/client/log.d.mts +1 -1
- package/dist/runtime/server/routes/_evlog/ingest.post.d.mts +7 -0
- package/dist/runtime/server/routes/_evlog/ingest.post.mjs +39 -3
- package/dist/runtime/server/routes/_evlog/ingest.post.mjs.map +1 -1
- package/dist/runtime/server/useLogger.d.mts +1 -1
- package/dist/runtime/utils/parseError.d.mts +2 -2
- package/dist/runtime/utils/parseError.mjs +1 -1
- package/dist/{severity-R5Egq3qz.mjs → severity-CwXUSHt3.mjs} +1 -1
- package/dist/{severity-R5Egq3qz.mjs.map → severity-CwXUSHt3.mjs.map} +1 -1
- package/dist/{source-location-Dco0cRTz.mjs → source-location-CHOPF2nd.mjs} +2 -1
- package/dist/{source-location-Dco0cRTz.mjs.map → source-location-CHOPF2nd.mjs.map} +1 -1
- package/dist/{storage-BNubsWwz.mjs → storage-7X37OToT.mjs} +1 -1
- package/dist/{storage-BNubsWwz.mjs.map → storage-7X37OToT.mjs.map} +1 -1
- package/dist/stream.d.mts +1 -1
- package/dist/stream.mjs +1 -1
- package/dist/streamResponse-CmQ3qUbF.mjs +94 -0
- package/dist/streamResponse-CmQ3qUbF.mjs.map +1 -0
- package/dist/sveltekit/index.d.mts +2 -2
- package/dist/sveltekit/index.d.mts.map +1 -1
- package/dist/sveltekit/index.mjs +7 -8
- package/dist/sveltekit/index.mjs.map +1 -1
- package/dist/toolkit.d.mts +51 -6
- package/dist/toolkit.d.mts.map +1 -1
- package/dist/toolkit.mjs +15 -14
- package/dist/types.d.mts +2 -2
- package/dist/{useLogger-CqvH6qOf.d.mts → useLogger-Cfv8Ck8b.d.mts} +2 -2
- package/dist/{useLogger-CqvH6qOf.d.mts.map → useLogger-Cfv8Ck8b.d.mts.map} +1 -1
- package/dist/{utils-DxqvIOyR.d.mts → utils-CJJG0ZYW.d.mts} +11 -3
- package/dist/{utils-DxqvIOyR.d.mts.map → utils-CJJG0ZYW.d.mts.map} +1 -1
- package/dist/utils.d.mts +2 -2
- package/dist/utils.mjs +23 -9
- package/dist/utils.mjs.map +1 -1
- package/dist/vite/index.d.mts +1 -1
- package/dist/vite/index.mjs +1 -1
- package/dist/workers.d.mts +1 -1
- package/dist/workers.mjs +1 -1
- package/package.json +12 -1
- package/dist/audit-BUI3af4w.mjs.map +0 -1
- package/dist/audit-DVdkntSO.d.mts.map +0 -1
- package/dist/define-D-BVMf2l.d.mts.map +0 -1
- package/dist/define-D6OJdSUH.mjs.map +0 -1
- package/dist/drain-7n3K6kPe.mjs.map +0 -1
- package/dist/fork-Bga8x-X4.mjs.map +0 -1
- package/dist/index-ZSRQP_BI.d.mts.map +0 -1
- package/dist/logger-CTcvd5Cc.d.mts.map +0 -1
- package/dist/middleware-31KhtiEF.d.mts.map +0 -1
- package/dist/next/instrumentation.d.mts.map +0 -1
- package/dist/nitro-BRddgqSb.d.mts.map +0 -1
- package/dist/nitro-DErMq_Zj.mjs +0 -34
- package/dist/nitro-DErMq_Zj.mjs.map +0 -1
- package/dist/nitroConfigBridge-NbFn-sIK.mjs.map +0 -1
- package/dist/package-B23bR3tK.mjs.map +0 -1
- package/dist/parseError-D4PIxEWo.d.mts.map +0 -1
package/dist/browser.d.mts
CHANGED
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
//#region src/shared/enrich-error-stack.node.ts
|
|
2
|
+
function shouldEnrichStackFromConfig() {
|
|
3
|
+
try {
|
|
4
|
+
const raw = process.env.__EVLOG_CONFIG;
|
|
5
|
+
if (raw) return JSON.parse(raw).pretty ?? process.env.NODE_ENV !== "production";
|
|
6
|
+
} catch {}
|
|
7
|
+
return process.env.NODE_ENV !== "production";
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* Rewrite `error.stack` with source-mapped frames when the Nitro dev runtime is available.
|
|
11
|
+
* Matches Nitro's Youch output (e.g. `server/api/foo.ts:100` instead of `.nuxt/dev/index.mjs`).
|
|
12
|
+
*/
|
|
13
|
+
async function enrichErrorStackForDev(error, options = {}) {
|
|
14
|
+
if (process.env.NODE_ENV === "production") return;
|
|
15
|
+
if (!(options.pretty ?? shouldEnrichStackFromConfig())) return;
|
|
16
|
+
for (const specifier of ["nitropack/runtime/internal/error/dev", "nitro/runtime/internal/error/dev"]) try {
|
|
17
|
+
const mod = await import(specifier);
|
|
18
|
+
if (typeof mod.loadStackTrace === "function") {
|
|
19
|
+
await mod.loadStackTrace(error).catch(() => {});
|
|
20
|
+
return;
|
|
21
|
+
}
|
|
22
|
+
} catch {}
|
|
23
|
+
}
|
|
24
|
+
//#endregion
|
|
25
|
+
//#region src/nitro/deferred-drain.ts
|
|
26
|
+
/** @internal Extend drain lifetime on Cloudflare without blocking Nitro Node responses. */
|
|
27
|
+
function extendDeferredDrain(drainPromise, waitUntil) {
|
|
28
|
+
drainPromise.catch((err) => {
|
|
29
|
+
console.error("[evlog] background drain failed:", err);
|
|
30
|
+
});
|
|
31
|
+
if (typeof waitUntil === "function") waitUntil(drainPromise);
|
|
32
|
+
}
|
|
33
|
+
//#endregion
|
|
34
|
+
export { enrichErrorStackForDev as n, extendDeferredDrain as t };
|
|
35
|
+
|
|
36
|
+
//# sourceMappingURL=deferred-drain-jeajC8QF.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"deferred-drain-jeajC8QF.mjs","names":[],"sources":["../src/shared/enrich-error-stack.node.ts","../src/nitro/deferred-drain.ts"],"sourcesContent":["/** Options for {@link enrichErrorStackForDev}. */\nexport interface EnrichErrorStackOptions {\n /** When false, skip Nitro source-map stack enrichment. @default true in dev when pretty is enabled */\n pretty?: boolean\n}\n\nfunction shouldEnrichStackFromConfig(): boolean {\n try {\n const raw = process.env.__EVLOG_CONFIG\n if (raw) {\n const config = JSON.parse(raw) as { pretty?: boolean }\n return config.pretty ?? process.env.NODE_ENV !== 'production'\n }\n } catch {\n // ignore malformed config\n }\n return process.env.NODE_ENV !== 'production'\n}\n\n/**\n * Rewrite `error.stack` with source-mapped frames when the Nitro dev runtime is available.\n * Matches Nitro's Youch output (e.g. `server/api/foo.ts:100` instead of `.nuxt/dev/index.mjs`).\n */\nexport async function enrichErrorStackForDev(\n error: Error,\n options: EnrichErrorStackOptions = {},\n): Promise<void> {\n if (process.env.NODE_ENV === 'production') return\n const pretty = options.pretty ?? shouldEnrichStackFromConfig()\n if (!pretty) return\n\n const specifiers = [\n 'nitropack/runtime/internal/error/dev',\n 'nitro/runtime/internal/error/dev',\n ]\n for (const specifier of specifiers) {\n try {\n const mod = await import(specifier)\n if (typeof mod.loadStackTrace === 'function') {\n await mod.loadStackTrace(error).catch(() => {})\n return\n }\n } catch {\n // try next runtime\n }\n }\n}\n","/** @internal Extend drain lifetime on Cloudflare without blocking Nitro Node responses. */\nexport function extendDeferredDrain(\n drainPromise: Promise<unknown>,\n waitUntil?: (promise: Promise<unknown>) => void,\n): void {\n void drainPromise.catch((err) => {\n console.error('[evlog] background drain failed:', err)\n })\n if (typeof waitUntil === 'function') {\n waitUntil(drainPromise)\n }\n}\n"],"mappings":";AAMA,SAAS,8BAAuC;AAC9C,KAAI;EACF,MAAM,MAAM,QAAQ,IAAI;AACxB,MAAI,IAEF,QADe,KAAK,MAAM,IACb,CAAC,UAAU,QAAQ,IAAI,aAAa;SAE7C;AAGR,QAAO,QAAQ,IAAI,aAAa;;;;;;AAOlC,eAAsB,uBACpB,OACA,UAAmC,EAAE,EACtB;AACf,KAAI,QAAQ,IAAI,aAAa,aAAc;AAE3C,KAAI,EADW,QAAQ,UAAU,6BAA6B,EACjD;AAMb,MAAK,MAAM,aAAa,CAHtB,wCACA,mCAEgC,CAChC,KAAI;EACF,MAAM,MAAM,MAAM,OAAO;AACzB,MAAI,OAAO,IAAI,mBAAmB,YAAY;AAC5C,SAAM,IAAI,eAAe,MAAM,CAAC,YAAY,GAAG;AAC/C;;SAEI;;;;;ACzCZ,SAAgB,oBACd,cACA,WACM;AACD,cAAa,OAAO,QAAQ;AAC/B,UAAQ,MAAM,oCAAoC,IAAI;GACtD;AACF,KAAI,OAAO,cAAc,WACvB,WAAU,aAAa"}
|
|
@@ -35,6 +35,7 @@ function toLoggerConfig(config) {
|
|
|
35
35
|
if (env) out.env = env;
|
|
36
36
|
if (config.enabled !== void 0) out.enabled = config.enabled;
|
|
37
37
|
if (config.pretty !== void 0) out.pretty = config.pretty;
|
|
38
|
+
if (config.dev !== void 0) out.dev = config.dev;
|
|
38
39
|
if (config.sampling !== void 0) out.sampling = config.sampling;
|
|
39
40
|
if (config.minLevel !== void 0) out.minLevel = config.minLevel;
|
|
40
41
|
if (config.stringify !== void 0) out.stringify = config.stringify;
|
|
@@ -60,4 +61,4 @@ function toMiddlewareOptions(config) {
|
|
|
60
61
|
//#endregion
|
|
61
62
|
export { toLoggerConfig as n, toMiddlewareOptions as r, defineEvlog as t };
|
|
62
63
|
|
|
63
|
-
//# sourceMappingURL=define-
|
|
64
|
+
//# sourceMappingURL=define-Bpaymi-h.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"define-Bpaymi-h.mjs","names":[],"sources":["../src/shared/define.ts"],"sourcesContent":["import type { EnvironmentContext, LoggerConfig, SamplingConfig } from '../types'\nimport type { DevTerminalInput } from './dev-terminal'\nimport type { BaseEvlogOptions } from './middleware'\n\n/**\n * Single-config shape accepted everywhere evlog is bootstrapped: at\n * `initLogger`, in framework middleware, and in the Nuxt module. Authored\n * with {@link defineEvlog} and split via {@link toLoggerConfig} /\n * {@link toMiddlewareOptions}.\n */\nexport interface EvlogConfig extends BaseEvlogOptions {\n service?: string\n environment?: string\n /** Full environment context override (advanced). */\n env?: Partial<EnvironmentContext>\n /** Enable or disable all logging globally. */\n enabled?: boolean\n /** Auto-detected from `NODE_ENV` when omitted. */\n pretty?: boolean\n /**\n * Dev terminal output: preset or explicit overlay + pretty-error settings.\n * @default 'evlog' when pretty in development\n */\n dev?: DevTerminalInput\n sampling?: SamplingConfig\n /** Suppress built-in console output (useful when drains own the channel). */\n silent?: boolean\n /** Emit JSON strings (default) or raw objects in non-pretty mode. */\n stringify?: boolean\n /** Minimum severity for the global `log` API. */\n minLevel?: LoggerConfig['minLevel']\n}\n\n/**\n * Identity helper for authoring an evlog configuration once and sharing it\n * across `initLogger` and framework middleware.\n *\n * @example\n * ```ts\n * export const config = defineEvlog({\n * service: 'checkout',\n * redact: true,\n * sampling: { rates: { info: 25 } },\n * enrich: createDefaultEnrichers(),\n * plugins: [drainPlugin('axiom', createAxiomDrain())],\n * })\n *\n * initLogger(toLoggerConfig(config))\n * app.use(evlogMiddleware(config))\n * ```\n */\nexport function defineEvlog<T extends EvlogConfig>(config: T): T {\n return config\n}\n\n/**\n * Project an {@link EvlogConfig} onto the surface accepted by `initLogger`.\n * Strips middleware-only fields (`include`, `exclude`, `routes`, `enrich`,\n * `keep`); drains and plugins are preserved.\n */\nexport function toLoggerConfig(config: EvlogConfig): LoggerConfig {\n const env: Partial<EnvironmentContext> | undefined = config.env\n ? { ...config.env }\n : config.service || config.environment\n ? {}\n : undefined\n if (env) {\n if (config.service) env.service = config.service\n if (config.environment) env.environment = config.environment\n }\n const out: LoggerConfig = {}\n if (env) out.env = env\n if (config.enabled !== undefined) out.enabled = config.enabled\n if (config.pretty !== undefined) out.pretty = config.pretty\n if (config.dev !== undefined) out.dev = config.dev\n if (config.sampling !== undefined) out.sampling = config.sampling\n if (config.minLevel !== undefined) out.minLevel = config.minLevel\n if (config.stringify !== undefined) out.stringify = config.stringify\n if (config.silent !== undefined) out.silent = config.silent\n if (config.redact !== undefined) out.redact = config.redact\n if (config.drain) out.drain = config.drain\n if (config.plugins) out.plugins = config.plugins\n return out\n}\n\n/** Project an {@link EvlogConfig} onto the surface accepted by framework middleware. */\nexport function toMiddlewareOptions<T extends BaseEvlogOptions>(config: EvlogConfig): T {\n const out: BaseEvlogOptions = {}\n if (config.include) out.include = config.include\n if (config.exclude) out.exclude = config.exclude\n if (config.routes) out.routes = config.routes\n if (config.drain) out.drain = config.drain\n if (config.enrich) out.enrich = config.enrich\n if (config.keep) out.keep = config.keep\n if (config.redact !== undefined) out.redact = config.redact\n if (config.plugins) out.plugins = config.plugins\n return out as T\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;AAmDA,SAAgB,YAAmC,QAAc;AAC/D,QAAO;;;;;;;AAQT,SAAgB,eAAe,QAAmC;CAChE,MAAM,MAA+C,OAAO,MACxD,EAAE,GAAG,OAAO,KAAK,GACjB,OAAO,WAAW,OAAO,cACvB,EAAE,GACF,KAAA;AACN,KAAI,KAAK;AACP,MAAI,OAAO,QAAS,KAAI,UAAU,OAAO;AACzC,MAAI,OAAO,YAAa,KAAI,cAAc,OAAO;;CAEnD,MAAM,MAAoB,EAAE;AAC5B,KAAI,IAAK,KAAI,MAAM;AACnB,KAAI,OAAO,YAAY,KAAA,EAAW,KAAI,UAAU,OAAO;AACvD,KAAI,OAAO,WAAW,KAAA,EAAW,KAAI,SAAS,OAAO;AACrD,KAAI,OAAO,QAAQ,KAAA,EAAW,KAAI,MAAM,OAAO;AAC/C,KAAI,OAAO,aAAa,KAAA,EAAW,KAAI,WAAW,OAAO;AACzD,KAAI,OAAO,aAAa,KAAA,EAAW,KAAI,WAAW,OAAO;AACzD,KAAI,OAAO,cAAc,KAAA,EAAW,KAAI,YAAY,OAAO;AAC3D,KAAI,OAAO,WAAW,KAAA,EAAW,KAAI,SAAS,OAAO;AACrD,KAAI,OAAO,WAAW,KAAA,EAAW,KAAI,SAAS,OAAO;AACrD,KAAI,OAAO,MAAO,KAAI,QAAQ,OAAO;AACrC,KAAI,OAAO,QAAS,KAAI,UAAU,OAAO;AACzC,QAAO;;;AAIT,SAAgB,oBAAgD,QAAwB;CACtF,MAAM,MAAwB,EAAE;AAChC,KAAI,OAAO,QAAS,KAAI,UAAU,OAAO;AACzC,KAAI,OAAO,QAAS,KAAI,UAAU,OAAO;AACzC,KAAI,OAAO,OAAQ,KAAI,SAAS,OAAO;AACvC,KAAI,OAAO,MAAO,KAAI,QAAQ,OAAO;AACrC,KAAI,OAAO,OAAQ,KAAI,SAAS,OAAO;AACvC,KAAI,OAAO,KAAM,KAAI,OAAO,OAAO;AACnC,KAAI,OAAO,WAAW,KAAA,EAAW,KAAI,SAAS,OAAO;AACrD,KAAI,OAAO,QAAS,KAAI,UAAU,OAAO;AACzC,QAAO"}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import { t as BaseEvlogOptions } from "./middleware-
|
|
1
|
+
import { H as EnvironmentContext, Z as LoggerConfig, ot as SamplingConfig, pt as DevTerminalInput } from "./audit-D7v6JHj0.mjs";
|
|
2
|
+
import { t as BaseEvlogOptions } from "./middleware-B_k4Mrzg.mjs";
|
|
3
3
|
|
|
4
4
|
//#region src/shared/define.d.ts
|
|
5
5
|
/**
|
|
@@ -17,6 +17,11 @@ interface EvlogConfig extends BaseEvlogOptions {
|
|
|
17
17
|
enabled?: boolean;
|
|
18
18
|
/** Auto-detected from `NODE_ENV` when omitted. */
|
|
19
19
|
pretty?: boolean;
|
|
20
|
+
/**
|
|
21
|
+
* Dev terminal output: preset or explicit overlay + pretty-error settings.
|
|
22
|
+
* @default 'evlog' when pretty in development
|
|
23
|
+
*/
|
|
24
|
+
dev?: DevTerminalInput;
|
|
20
25
|
sampling?: SamplingConfig;
|
|
21
26
|
/** Suppress built-in console output (useful when drains own the channel). */
|
|
22
27
|
silent?: boolean;
|
|
@@ -54,4 +59,4 @@ declare function toLoggerConfig(config: EvlogConfig): LoggerConfig;
|
|
|
54
59
|
declare function toMiddlewareOptions<T extends BaseEvlogOptions>(config: EvlogConfig): T;
|
|
55
60
|
//#endregion
|
|
56
61
|
export { toMiddlewareOptions as i, defineEvlog as n, toLoggerConfig as r, EvlogConfig as t };
|
|
57
|
-
//# sourceMappingURL=define-
|
|
62
|
+
//# sourceMappingURL=define-DTQpu4f6.d.mts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"define-DTQpu4f6.d.mts","names":[],"sources":["../src/shared/define.ts"],"mappings":";;;;;AAUA;;;;;UAAiB,WAAA,SAAoB,gBAAA;EACnC,OAAA;EACA,WAAA;EAFmC;EAInC,GAAA,GAAM,OAAA,CAAQ,kBAAA;EAJqC;EAMnD,OAAA;EALA;EAOA,MAAA;EAJA;;;;EASA,GAAA,GAAM,gBAAA;EACN,QAAA,GAAW,cAAA;EADL;EAGN,MAAA;EAFW;EAIX,SAAA;EAAA;EAEA,QAAA,GAAW,YAAA;AAAA;;;AAqBb;;;;;;;;;;;;;;;AASA;iBATgB,WAAA,WAAsB,WAAA,CAAA,CAAa,MAAA,EAAQ,CAAA,GAAI,CAAA;;;;;;iBAS/C,cAAA,CAAe,MAAA,EAAQ,WAAA,GAAc,YAAA;;iBA0BrC,mBAAA,WAA8B,gBAAA,CAAA,CAAkB,MAAA,EAAQ,WAAA,GAAc,CAAA"}
|
|
@@ -0,0 +1,54 @@
|
|
|
1
|
+
import { isDev } from "./utils.mjs";
|
|
2
|
+
//#region src/shared/dev-terminal.ts
|
|
3
|
+
const DEV_PRESETS = {
|
|
4
|
+
evlog: {
|
|
5
|
+
frameworkOverlay: false,
|
|
6
|
+
detail: "full"
|
|
7
|
+
},
|
|
8
|
+
nitro: {
|
|
9
|
+
frameworkOverlay: true,
|
|
10
|
+
detail: "guidance"
|
|
11
|
+
},
|
|
12
|
+
both: {
|
|
13
|
+
frameworkOverlay: true,
|
|
14
|
+
detail: "full"
|
|
15
|
+
}
|
|
16
|
+
};
|
|
17
|
+
function finalizePrettyError(partial, frameworkOverlay, pretty, inDev) {
|
|
18
|
+
const compact = partial.compact ?? (inDev && pretty);
|
|
19
|
+
const detail = partial.detail ?? (frameworkOverlay ? "guidance" : "full");
|
|
20
|
+
const stackDepth = detail === "guidance" ? 0 : partial.stackDepth ?? (compact ? 2 : 3);
|
|
21
|
+
return {
|
|
22
|
+
snippet: partial.snippet ?? (detail === "full" && pretty && inDev),
|
|
23
|
+
stackDepth,
|
|
24
|
+
compact,
|
|
25
|
+
detail
|
|
26
|
+
};
|
|
27
|
+
}
|
|
28
|
+
/**
|
|
29
|
+
* Resolve dev terminal settings from `dev` presets or explicit objects.
|
|
30
|
+
*/
|
|
31
|
+
function resolveDevTerminal(input = {}) {
|
|
32
|
+
const pretty = input.pretty ?? isDev();
|
|
33
|
+
const inDev = isDev();
|
|
34
|
+
let frameworkOverlay;
|
|
35
|
+
let prettyError = {};
|
|
36
|
+
if (typeof input.dev === "string" && input.dev in DEV_PRESETS) {
|
|
37
|
+
const { frameworkOverlay: presetOverlay, detail } = DEV_PRESETS[input.dev];
|
|
38
|
+
frameworkOverlay = presetOverlay;
|
|
39
|
+
prettyError = { detail };
|
|
40
|
+
} else if (input.dev && typeof input.dev === "object") {
|
|
41
|
+
const { frameworkOverlay: devOverlay, prettyError: devPrettyError } = input.dev;
|
|
42
|
+
frameworkOverlay = devOverlay;
|
|
43
|
+
if (devPrettyError) prettyError = devPrettyError;
|
|
44
|
+
}
|
|
45
|
+
if (frameworkOverlay === void 0) frameworkOverlay = !(pretty && inDev);
|
|
46
|
+
return {
|
|
47
|
+
frameworkOverlay,
|
|
48
|
+
prettyError: finalizePrettyError(prettyError, frameworkOverlay, pretty, inDev)
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
//#endregion
|
|
52
|
+
export { resolveDevTerminal as t };
|
|
53
|
+
|
|
54
|
+
//# sourceMappingURL=dev-terminal-D4UaEm17.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"dev-terminal-D4UaEm17.mjs","names":[],"sources":["../src/shared/dev-terminal.ts"],"sourcesContent":["import { isDev } from '../utils'\n\n/** Dev terminal preset — shorthand for common overlay + pretty-error combinations. */\nexport type DevTerminalPreset = 'evlog' | 'nitro' | 'both'\n\n/** How much stack detail evlog prints inside the wide-event error block. */\nexport type DevPrettyErrorDetail = 'full' | 'guidance'\n\n/** Pretty-print options for the `error:` block in dev wide events. */\nexport interface DevPrettyErrorConfig {\n snippet?: boolean\n stackDepth?: number\n compact?: boolean\n detail?: DevPrettyErrorDetail\n}\n\n/** Resolved pretty-error settings used at runtime. */\nexport type ResolvedPrettyError = Required<DevPrettyErrorConfig>\n\n/** Resolved dev terminal object (alternative to preset strings). */\nexport interface DevTerminalConfigObject {\n /** Show Nitro `[request error]` + Youch in the terminal. @default false when pretty in dev. */\n frameworkOverlay?: boolean\n prettyError?: DevPrettyErrorConfig\n}\n\n/** User-facing dev terminal config: preset string or explicit object. */\nexport type DevTerminalInput = DevTerminalPreset | DevTerminalConfigObject\n\n/** Resolved dev terminal settings used at runtime. */\nexport interface ResolvedDevTerminal {\n frameworkOverlay: boolean\n prettyError: ResolvedPrettyError\n}\n\n/** Config surface accepted by {@link resolveDevTerminal}. */\nexport interface DevTerminalResolveInput {\n pretty?: boolean\n dev?: DevTerminalInput\n}\n\nconst DEV_PRESETS: Record<DevTerminalPreset, { frameworkOverlay: boolean; detail: DevPrettyErrorDetail }> = {\n evlog: { frameworkOverlay: false, detail: 'full' },\n nitro: { frameworkOverlay: true, detail: 'guidance' },\n both: { frameworkOverlay: true, detail: 'full' },\n}\n\nfunction finalizePrettyError(\n partial: DevPrettyErrorConfig,\n frameworkOverlay: boolean,\n pretty: boolean,\n inDev: boolean,\n): ResolvedPrettyError {\n const compact = partial.compact ?? (inDev && pretty)\n const detail = partial.detail ?? (frameworkOverlay ? 'guidance' : 'full')\n const stackDepth = detail === 'guidance'\n ? 0\n : (partial.stackDepth ?? (compact ? 2 : 3))\n const snippet = partial.snippet ?? (detail === 'full' && pretty && inDev)\n\n return { snippet, stackDepth, compact, detail }\n}\n\n/**\n * Resolve dev terminal settings from `dev` presets or explicit objects.\n */\nexport function resolveDevTerminal(input: DevTerminalResolveInput = {}): ResolvedDevTerminal {\n const pretty = input.pretty ?? isDev()\n const inDev = isDev()\n\n let frameworkOverlay: boolean | undefined\n let prettyError: DevPrettyErrorConfig = {}\n\n if (typeof input.dev === 'string' && input.dev in DEV_PRESETS) {\n const { frameworkOverlay: presetOverlay, detail } = DEV_PRESETS[input.dev]\n frameworkOverlay = presetOverlay\n prettyError = { detail }\n } else if (input.dev && typeof input.dev === 'object') {\n const { frameworkOverlay: devOverlay, prettyError: devPrettyError } = input.dev\n frameworkOverlay = devOverlay\n if (devPrettyError) {\n prettyError = devPrettyError\n }\n }\n\n if (frameworkOverlay === undefined) {\n frameworkOverlay = !(pretty && inDev)\n }\n\n return {\n frameworkOverlay,\n prettyError: finalizePrettyError(prettyError, frameworkOverlay, pretty, inDev),\n }\n}\n\n/**\n * Whether Nitro's dev Youch overlay should print to the terminal.\n * @internal\n */\nexport function shouldShowFrameworkOverlay(input: DevTerminalResolveInput = {}): boolean {\n return resolveDevTerminal(input).frameworkOverlay\n}\n"],"mappings":";;AAyCA,MAAM,cAAsG;CAC1G,OAAO;EAAE,kBAAkB;EAAO,QAAQ;EAAQ;CAClD,OAAO;EAAE,kBAAkB;EAAM,QAAQ;EAAY;CACrD,MAAM;EAAE,kBAAkB;EAAM,QAAQ;EAAQ;CACjD;AAED,SAAS,oBACP,SACA,kBACA,QACA,OACqB;CACrB,MAAM,UAAU,QAAQ,YAAY,SAAS;CAC7C,MAAM,SAAS,QAAQ,WAAW,mBAAmB,aAAa;CAClE,MAAM,aAAa,WAAW,aAC1B,IACC,QAAQ,eAAe,UAAU,IAAI;AAG1C,QAAO;EAAE,SAFO,QAAQ,YAAY,WAAW,UAAU,UAAU;EAEjD;EAAY;EAAS;EAAQ;;;;;AAMjD,SAAgB,mBAAmB,QAAiC,EAAE,EAAuB;CAC3F,MAAM,SAAS,MAAM,UAAU,OAAO;CACtC,MAAM,QAAQ,OAAO;CAErB,IAAI;CACJ,IAAI,cAAoC,EAAE;AAE1C,KAAI,OAAO,MAAM,QAAQ,YAAY,MAAM,OAAO,aAAa;EAC7D,MAAM,EAAE,kBAAkB,eAAe,WAAW,YAAY,MAAM;AACtE,qBAAmB;AACnB,gBAAc,EAAE,QAAQ;YACf,MAAM,OAAO,OAAO,MAAM,QAAQ,UAAU;EACrD,MAAM,EAAE,kBAAkB,YAAY,aAAa,mBAAmB,MAAM;AAC5E,qBAAmB;AACnB,MAAI,eACF,eAAc;;AAIlB,KAAI,qBAAqB,KAAA,EACvB,oBAAmB,EAAE,UAAU;AAGjC,QAAO;EACL;EACA,aAAa,oBAAoB,aAAa,kBAAkB,QAAQ,MAAM;EAC/E"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"dist-H3GIh-KK.mjs","names":[],"sources":["../../../node_modules/.pnpm/ufo@1.6.4/node_modules/ufo/dist/index.mjs"],"sourcesContent":["const n = /[^\\0-\\x7E]/;\nconst t = /[\\x2E\\u3002\\uFF0E\\uFF61]/g;\nconst o = {\n overflow: \"Overflow Error\",\n \"not-basic\": \"Illegal Input\",\n \"invalid-input\": \"Invalid Input\"\n};\nconst e = Math.floor;\nconst r = String.fromCharCode;\nfunction s(n2) {\n throw new RangeError(o[n2]);\n}\nconst c = function(n2, t2) {\n return n2 + 22 + 75 * (n2 < 26) - ((t2 != 0) << 5);\n};\nconst u = function(n2, t2, o2) {\n let r2 = 0;\n for (n2 = o2 ? e(n2 / 700) : n2 >> 1, n2 += e(n2 / t2); n2 > 455; r2 += 36) {\n n2 = e(n2 / 35);\n }\n return e(r2 + 36 * n2 / (n2 + 38));\n};\nfunction toASCII(o2) {\n return (function(n2, o3) {\n const e2 = n2.split(\"@\");\n let r2 = \"\";\n e2.length > 1 && (r2 = e2[0] + \"@\", n2 = e2[1]);\n const s2 = (function(n3, t2) {\n const o4 = [];\n let e3 = n3.length;\n for (; e3--; ) {\n o4[e3] = t2(n3[e3]);\n }\n return o4;\n })((n2 = n2.replace(t, \".\")).split(\".\"), o3).join(\".\");\n return r2 + s2;\n })(o2, function(t2) {\n return n.test(t2) ? \"xn--\" + (function(n2) {\n const t3 = [];\n const o3 = (n2 = (function(n3) {\n const t4 = [];\n let o4 = 0;\n const e2 = n3.length;\n for (; o4 < e2; ) {\n const r2 = n3.charCodeAt(o4++);\n if (r2 >= 55296 && r2 <= 56319 && o4 < e2) {\n const e3 = n3.charCodeAt(o4++);\n (64512 & e3) == 56320 ? t4.push(((1023 & r2) << 10) + (1023 & e3) + 65536) : (t4.push(r2), o4--);\n } else {\n t4.push(r2);\n }\n }\n return t4;\n })(n2)).length;\n let f = 128;\n let i = 0;\n let l = 72;\n for (const o4 of n2) {\n o4 < 128 && t3.push(r(o4));\n }\n const h = t3.length;\n let p = h;\n for (h && t3.push(\"-\"); p < o3; ) {\n let o4 = 2147483647;\n for (const t4 of n2) {\n t4 >= f && t4 < o4 && (o4 = t4);\n }\n const a = p + 1;\n o4 - f > e((2147483647 - i) / a) && s(\"overflow\"), i += (o4 - f) * a, f = o4;\n for (const o5 of n2) {\n if (o5 < f && ++i > 2147483647 && s(\"overflow\"), o5 == f) {\n let n3 = i;\n for (let o6 = 36; ; o6 += 36) {\n const s2 = o6 <= l ? 1 : o6 >= l + 26 ? 26 : o6 - l;\n if (n3 < s2) {\n break;\n }\n const u2 = n3 - s2;\n const f2 = 36 - s2;\n t3.push(r(c(s2 + u2 % f2, 0))), n3 = e(u2 / f2);\n }\n t3.push(r(c(n3, 0))), l = u(i, a, p == h), i = 0, ++p;\n }\n }\n ++i, ++f;\n }\n return t3.join(\"\");\n })(t2) : t2;\n });\n}\n\nconst HASH_RE = /#/g;\nconst AMPERSAND_RE = /&/g;\nconst SLASH_RE = /\\//g;\nconst EQUAL_RE = /=/g;\nconst IM_RE = /\\?/g;\nconst PLUS_RE = /\\+/g;\nconst ENC_CARET_RE = /%5e/gi;\nconst ENC_BACKTICK_RE = /%60/gi;\nconst ENC_CURLY_OPEN_RE = /%7b/gi;\nconst ENC_PIPE_RE = /%7c/gi;\nconst ENC_CURLY_CLOSE_RE = /%7d/gi;\nconst ENC_SPACE_RE = /%20/gi;\nconst ENC_SLASH_RE = /%2f/gi;\nconst ENC_ENC_SLASH_RE = /%252f/gi;\nfunction encode(text) {\n return encodeURI(\"\" + text).replace(ENC_PIPE_RE, \"|\");\n}\nfunction encodeHash(text) {\n return encode(text).replace(ENC_CURLY_OPEN_RE, \"{\").replace(ENC_CURLY_CLOSE_RE, \"}\").replace(ENC_CARET_RE, \"^\");\n}\nfunction encodeQueryValue(input) {\n return encode(typeof input === \"string\" ? input : JSON.stringify(input)).replace(PLUS_RE, \"%2B\").replace(ENC_SPACE_RE, \"+\").replace(HASH_RE, \"%23\").replace(AMPERSAND_RE, \"%26\").replace(ENC_BACKTICK_RE, \"`\").replace(ENC_CARET_RE, \"^\").replace(SLASH_RE, \"%2F\");\n}\nfunction encodeQueryKey(text) {\n return encodeQueryValue(text).replace(EQUAL_RE, \"%3D\");\n}\nfunction encodePath(text) {\n return encode(text).replace(HASH_RE, \"%23\").replace(IM_RE, \"%3F\").replace(ENC_ENC_SLASH_RE, \"%2F\").replace(AMPERSAND_RE, \"%26\").replace(PLUS_RE, \"%2B\");\n}\nfunction encodeParam(text) {\n return encodePath(text).replace(SLASH_RE, \"%2F\");\n}\nfunction decode(text = \"\") {\n try {\n return decodeURIComponent(\"\" + text);\n } catch {\n return \"\" + text;\n }\n}\nfunction decodePath(text) {\n return decode(text.replace(ENC_SLASH_RE, \"%252F\"));\n}\nfunction decodeQueryKey(text) {\n return decode(text.replace(PLUS_RE, \" \"));\n}\nfunction decodeQueryValue(text) {\n return decode(text.replace(PLUS_RE, \" \"));\n}\nfunction encodeHost(name = \"\") {\n return toASCII(name);\n}\n\nfunction parseQuery(parametersString = \"\") {\n const object = /* @__PURE__ */ Object.create(null);\n if (parametersString[0] === \"?\") {\n parametersString = parametersString.slice(1);\n }\n for (const parameter of parametersString.split(\"&\")) {\n const s = parameter.match(/([^=]+)=?(.*)/) || [];\n if (s.length < 2) {\n continue;\n }\n const key = decodeQueryKey(s[1]);\n if (key === \"__proto__\" || key === \"constructor\") {\n continue;\n }\n const value = decodeQueryValue(s[2] || \"\");\n if (object[key] === void 0) {\n object[key] = value;\n } else if (Array.isArray(object[key])) {\n object[key].push(value);\n } else {\n object[key] = [object[key], value];\n }\n }\n return object;\n}\nfunction encodeQueryItem(key, value) {\n if (typeof value === \"number\" || typeof value === \"boolean\") {\n value = String(value);\n }\n if (!value) {\n return encodeQueryKey(key);\n }\n if (Array.isArray(value)) {\n return value.map(\n (_value) => `${encodeQueryKey(key)}=${encodeQueryValue(_value)}`\n ).join(\"&\");\n }\n return `${encodeQueryKey(key)}=${encodeQueryValue(value)}`;\n}\nfunction stringifyQuery(query) {\n return Object.keys(query).filter((k) => query[k] !== void 0).map((k) => encodeQueryItem(k, query[k])).filter(Boolean).join(\"&\");\n}\n\nconst PROTOCOL_STRICT_REGEX = /^[\\s\\w\\0+.-]{2,}:([/\\\\]{1,2})/;\nconst PROTOCOL_REGEX = /^[\\s\\w\\0+.-]{2,}:([/\\\\]{2})?/;\nconst PROTOCOL_RELATIVE_REGEX = /^([/\\\\]\\s*){2,}[^/\\\\]/;\nconst PROTOCOL_SCRIPT_RE = /^[\\s\\0]*(blob|data|javascript|vbscript):$/i;\nconst TRAILING_SLASH_RE = /\\/$|\\/\\?|\\/#/;\nconst JOIN_LEADING_SLASH_RE = /^\\.?\\//;\nfunction isRelative(inputString) {\n return [\"./\", \"../\"].some((string_) => inputString.startsWith(string_));\n}\nfunction hasProtocol(inputString, opts = {}) {\n if (typeof opts === \"boolean\") {\n opts = { acceptRelative: opts };\n }\n if (opts.strict) {\n return PROTOCOL_STRICT_REGEX.test(inputString);\n }\n return PROTOCOL_REGEX.test(inputString) || (opts.acceptRelative ? PROTOCOL_RELATIVE_REGEX.test(inputString) : false);\n}\nfunction isScriptProtocol(protocol) {\n return !!protocol && PROTOCOL_SCRIPT_RE.test(protocol);\n}\nfunction hasTrailingSlash(input = \"\", respectQueryAndFragment) {\n if (!respectQueryAndFragment) {\n return input.endsWith(\"/\");\n }\n return TRAILING_SLASH_RE.test(input);\n}\nfunction withoutTrailingSlash(input = \"\", respectQueryAndFragment) {\n if (!respectQueryAndFragment) {\n return (hasTrailingSlash(input) ? input.slice(0, -1) : input) || \"/\";\n }\n if (!hasTrailingSlash(input, true)) {\n return input || \"/\";\n }\n let path = input;\n let fragment = \"\";\n const fragmentIndex = input.indexOf(\"#\");\n if (fragmentIndex !== -1) {\n path = input.slice(0, fragmentIndex);\n fragment = input.slice(fragmentIndex);\n }\n const [s0, ...s] = path.split(\"?\");\n const cleanPath = s0.endsWith(\"/\") ? s0.slice(0, -1) : s0;\n return (cleanPath || \"/\") + (s.length > 0 ? `?${s.join(\"?\")}` : \"\") + fragment;\n}\nfunction withTrailingSlash(input = \"\", respectQueryAndFragment) {\n if (!respectQueryAndFragment) {\n return input.endsWith(\"/\") ? input : input + \"/\";\n }\n if (hasTrailingSlash(input, true)) {\n return input || \"/\";\n }\n let path = input;\n let fragment = \"\";\n const fragmentIndex = input.indexOf(\"#\");\n if (fragmentIndex !== -1) {\n path = input.slice(0, fragmentIndex);\n fragment = input.slice(fragmentIndex);\n if (!path) {\n return fragment;\n }\n }\n const [s0, ...s] = path.split(\"?\");\n return s0 + \"/\" + (s.length > 0 ? `?${s.join(\"?\")}` : \"\") + fragment;\n}\nfunction hasLeadingSlash(input = \"\") {\n return input.startsWith(\"/\");\n}\nfunction withoutLeadingSlash(input = \"\") {\n return (hasLeadingSlash(input) ? input.slice(1) : input) || \"/\";\n}\nfunction withLeadingSlash(input = \"\") {\n return hasLeadingSlash(input) ? input : \"/\" + input;\n}\nfunction cleanDoubleSlashes(input = \"\") {\n return input.split(\"://\").map((string_) => string_.replace(/\\/{2,}/g, \"/\")).join(\"://\");\n}\nfunction withBase(input, base) {\n if (isEmptyURL(base) || hasProtocol(input)) {\n return input;\n }\n const _base = withoutTrailingSlash(base);\n if (input.startsWith(_base)) {\n const nextChar = input[_base.length];\n if (!nextChar || nextChar === \"/\" || nextChar === \"?\") {\n return input;\n }\n }\n return joinURL(_base, input);\n}\nfunction withoutBase(input, base) {\n if (isEmptyURL(base)) {\n return input;\n }\n const _base = withoutTrailingSlash(base);\n if (!input.startsWith(_base)) {\n return input;\n }\n const nextChar = input[_base.length];\n if (nextChar && nextChar !== \"/\" && nextChar !== \"?\") {\n return input;\n }\n const trimmed = input.slice(_base.length).replace(/^\\/+/, \"\");\n return \"/\" + trimmed;\n}\nfunction withQuery(input, query) {\n const parsed = parseURL(input);\n const mergedQuery = { ...parseQuery(parsed.search), ...query };\n parsed.search = stringifyQuery(mergedQuery);\n return stringifyParsedURL(parsed);\n}\nfunction filterQuery(input, predicate) {\n if (!input.includes(\"?\")) {\n return input;\n }\n const parsed = parseURL(input);\n const query = parseQuery(parsed.search);\n const filteredQuery = Object.fromEntries(\n Object.entries(query).filter(([key, value]) => predicate(key, value))\n );\n parsed.search = stringifyQuery(filteredQuery);\n return stringifyParsedURL(parsed);\n}\nfunction getQuery(input) {\n return parseQuery(parseURL(input).search);\n}\nfunction isEmptyURL(url) {\n return !url || url === \"/\";\n}\nfunction isNonEmptyURL(url) {\n return url && url !== \"/\";\n}\nfunction joinURL(base, ...input) {\n let url = base || \"\";\n for (const segment of input.filter((url2) => isNonEmptyURL(url2))) {\n if (url) {\n const _segment = segment.replace(JOIN_LEADING_SLASH_RE, \"\");\n url = withTrailingSlash(url) + _segment;\n } else {\n url = segment;\n }\n }\n return url;\n}\nfunction joinRelativeURL(..._input) {\n const JOIN_SEGMENT_SPLIT_RE = /\\/(?!\\/)/;\n const input = _input.filter(Boolean);\n const segments = [];\n let segmentsDepth = 0;\n for (const i of input) {\n if (!i || i === \"/\") {\n continue;\n }\n for (const [sindex, s] of i.split(JOIN_SEGMENT_SPLIT_RE).entries()) {\n if (!s || s === \".\") {\n continue;\n }\n if (s === \"..\") {\n if (segments.length === 1 && hasProtocol(segments[0])) {\n continue;\n }\n segments.pop();\n segmentsDepth--;\n continue;\n }\n if (sindex === 1 && segments[segments.length - 1]?.endsWith(\":/\")) {\n segments[segments.length - 1] += \"/\" + s;\n continue;\n }\n segments.push(s);\n segmentsDepth++;\n }\n }\n let url = segments.join(\"/\");\n if (segmentsDepth >= 0) {\n if (input[0]?.startsWith(\"/\") && !url.startsWith(\"/\")) {\n url = \"/\" + url;\n } else if (input[0]?.startsWith(\"./\") && !url.startsWith(\"./\")) {\n url = \"./\" + url;\n }\n } else {\n url = \"../\".repeat(-1 * segmentsDepth) + url;\n }\n if (input[input.length - 1]?.endsWith(\"/\") && !url.endsWith(\"/\")) {\n url += \"/\";\n }\n return url;\n}\nfunction withHttp(input) {\n return withProtocol(input, \"http://\");\n}\nfunction withHttps(input) {\n return withProtocol(input, \"https://\");\n}\nfunction withoutProtocol(input) {\n return withProtocol(input, \"\");\n}\nfunction withProtocol(input, protocol) {\n let match = input.match(PROTOCOL_REGEX);\n if (!match) {\n match = input.match(/^\\/{2,}/);\n }\n if (!match) {\n return protocol + input;\n }\n return protocol + input.slice(match[0].length);\n}\nfunction normalizeURL(input) {\n const parsed = parseURL(input);\n parsed.pathname = encodePath(decodePath(parsed.pathname));\n parsed.hash = encodeHash(decode(parsed.hash));\n parsed.host = encodeHost(decode(parsed.host));\n parsed.search = stringifyQuery(parseQuery(parsed.search));\n return stringifyParsedURL(parsed);\n}\nfunction resolveURL(base = \"\", ...inputs) {\n if (typeof base !== \"string\") {\n throw new TypeError(\n `URL input should be string received ${typeof base} (${base})`\n );\n }\n const filteredInputs = inputs.filter((input) => isNonEmptyURL(input));\n if (filteredInputs.length === 0) {\n return base;\n }\n const url = parseURL(base);\n for (const inputSegment of filteredInputs) {\n const urlSegment = parseURL(inputSegment);\n if (urlSegment.pathname) {\n url.pathname = withTrailingSlash(url.pathname) + withoutLeadingSlash(urlSegment.pathname);\n }\n if (urlSegment.hash && urlSegment.hash !== \"#\") {\n url.hash = urlSegment.hash;\n }\n if (urlSegment.search && urlSegment.search !== \"?\") {\n if (url.search && url.search !== \"?\") {\n const queryString = stringifyQuery({\n ...parseQuery(url.search),\n ...parseQuery(urlSegment.search)\n });\n url.search = queryString.length > 0 ? \"?\" + queryString : \"\";\n } else {\n url.search = urlSegment.search;\n }\n }\n }\n return stringifyParsedURL(url);\n}\nfunction isSamePath(p1, p2) {\n return decode(withoutTrailingSlash(p1)) === decode(withoutTrailingSlash(p2));\n}\nfunction isEqual(a, b, options = {}) {\n if (!options.trailingSlash) {\n a = withTrailingSlash(a);\n b = withTrailingSlash(b);\n }\n if (!options.leadingSlash) {\n a = withLeadingSlash(a);\n b = withLeadingSlash(b);\n }\n if (!options.encoding) {\n a = decode(a);\n b = decode(b);\n }\n return a === b;\n}\nfunction withFragment(input, hash) {\n if (!hash || hash === \"#\") {\n return input;\n }\n const parsed = parseURL(input);\n parsed.hash = hash === \"\" ? \"\" : \"#\" + encodeHash(hash);\n return stringifyParsedURL(parsed);\n}\nfunction withoutFragment(input) {\n return stringifyParsedURL({ ...parseURL(input), hash: \"\" });\n}\nfunction withoutHost(input) {\n const parsed = parseURL(input);\n return (parsed.pathname || \"/\") + parsed.search + parsed.hash;\n}\n\nconst protocolRelative = Symbol.for(\"ufo:protocolRelative\");\nfunction parseURL(input = \"\", defaultProto) {\n const _specialProtoMatch = input.match(\n /^[\\s\\0]*(blob:|data:|javascript:|vbscript:)(.*)/i\n );\n if (_specialProtoMatch) {\n const [, _proto, _pathname = \"\"] = _specialProtoMatch;\n return {\n protocol: _proto.toLowerCase(),\n pathname: _pathname,\n href: _proto + _pathname,\n auth: \"\",\n host: \"\",\n search: \"\",\n hash: \"\"\n };\n }\n if (!hasProtocol(input, { acceptRelative: true })) {\n return defaultProto ? parseURL(defaultProto + input) : parsePath(input);\n }\n const [, protocol = \"\", auth, hostAndPath = \"\"] = input.replace(/\\\\/g, \"/\").match(/^[\\s\\0]*([\\w+.-]{2,}:)?\\/\\/([^/@]+@)?(.*)/) || [];\n let [, host = \"\", path = \"\"] = hostAndPath.match(/([^#/?]*)(.*)?/) || [];\n if (protocol === \"file:\") {\n path = path.replace(/\\/(?=[A-Za-z]:)/, \"\");\n }\n const { pathname, search, hash } = parsePath(path);\n return {\n protocol: protocol.toLowerCase(),\n auth: auth ? auth.slice(0, Math.max(0, auth.length - 1)) : \"\",\n host,\n pathname,\n search,\n hash,\n [protocolRelative]: !protocol\n };\n}\nfunction parsePath(input = \"\") {\n const [pathname = \"\", search = \"\", hash = \"\"] = (input.match(/([^#?]*)(\\?[^#]*)?(#.*)?/) || []).splice(1);\n return {\n pathname,\n search,\n hash\n };\n}\nfunction parseAuth(input = \"\") {\n const [username, password] = input.split(\":\");\n return {\n username: decode(username),\n password: decode(password)\n };\n}\nfunction parseHost(input = \"\") {\n const [hostname, port] = (input.match(/([^/:]*):?(\\d+)?/) || []).splice(1);\n return {\n hostname: decode(hostname),\n port\n };\n}\nfunction stringifyParsedURL(parsed) {\n const pathname = parsed.pathname || \"\";\n const search = parsed.search ? (parsed.search.startsWith(\"?\") ? \"\" : \"?\") + parsed.search : \"\";\n const hash = parsed.hash || \"\";\n const auth = parsed.auth ? parsed.auth + \"@\" : \"\";\n const host = parsed.host || \"\";\n const proto = parsed.protocol || parsed[protocolRelative] ? (parsed.protocol || \"\") + \"//\" : \"\";\n return proto + auth + host + pathname + search + hash;\n}\nconst FILENAME_STRICT_REGEX = /\\/([^/]+\\.[^/]+)$/;\nconst FILENAME_REGEX = /\\/([^/]+)$/;\nfunction parseFilename(input = \"\", opts) {\n const { pathname } = parseURL(input);\n const matches = opts?.strict ? pathname.match(FILENAME_STRICT_REGEX) : pathname.match(FILENAME_REGEX);\n return matches ? matches[1] : void 0;\n}\n\nclass $URL {\n protocol;\n host;\n auth;\n pathname;\n query = {};\n hash;\n constructor(input = \"\") {\n if (typeof input !== \"string\") {\n throw new TypeError(\n `URL input should be string received ${typeof input} (${input})`\n );\n }\n const parsed = parseURL(input);\n this.protocol = decode(parsed.protocol);\n this.host = decode(parsed.host);\n this.auth = decode(parsed.auth);\n this.pathname = decodePath(parsed.pathname);\n this.query = parseQuery(parsed.search);\n this.hash = decode(parsed.hash);\n }\n get hostname() {\n return parseHost(this.host).hostname;\n }\n get port() {\n return parseHost(this.host).port || \"\";\n }\n get username() {\n return parseAuth(this.auth).username;\n }\n get password() {\n return parseAuth(this.auth).password || \"\";\n }\n get hasProtocol() {\n return this.protocol.length;\n }\n get isAbsolute() {\n return this.hasProtocol || this.pathname[0] === \"/\";\n }\n get search() {\n const q = stringifyQuery(this.query);\n return q.length > 0 ? \"?\" + q : \"\";\n }\n get searchParams() {\n const p = new URLSearchParams();\n for (const name in this.query) {\n const value = this.query[name];\n if (Array.isArray(value)) {\n for (const v of value) {\n p.append(name, v);\n }\n } else {\n p.append(\n name,\n typeof value === \"string\" ? value : JSON.stringify(value)\n );\n }\n }\n return p;\n }\n get origin() {\n return (this.protocol ? this.protocol + \"//\" : \"\") + encodeHost(this.host);\n }\n get fullpath() {\n return encodePath(this.pathname) + this.search + encodeHash(this.hash);\n }\n get encodedAuth() {\n if (!this.auth) {\n return \"\";\n }\n const { username, password } = parseAuth(this.auth);\n return encodeURIComponent(username) + (password ? \":\" + encodeURIComponent(password) : \"\");\n }\n get href() {\n const auth = this.encodedAuth;\n const originWithAuth = (this.protocol ? this.protocol + \"//\" : \"\") + (auth ? auth + \"@\" : \"\") + encodeHost(this.host);\n return this.hasProtocol && this.isAbsolute ? originWithAuth + this.fullpath : this.fullpath;\n }\n append(url) {\n if (url.hasProtocol) {\n throw new Error(\"Cannot append a URL with protocol\");\n }\n Object.assign(this.query, url.query);\n if (url.pathname) {\n this.pathname = withTrailingSlash(this.pathname) + withoutLeadingSlash(url.pathname);\n }\n if (url.hash) {\n this.hash = url.hash;\n }\n }\n toJSON() {\n return this.href;\n }\n toString() {\n return this.href;\n }\n}\nfunction createURL(input) {\n return new $URL(input);\n}\n\nexport { $URL, cleanDoubleSlashes, createURL, decode, decodePath, decodeQueryKey, decodeQueryValue, encode, encodeHash, encodeHost, encodeParam, encodePath, encodeQueryItem, encodeQueryKey, encodeQueryValue, filterQuery, getQuery, hasLeadingSlash, hasProtocol, hasTrailingSlash, isEmptyURL, isEqual, isNonEmptyURL, isRelative, isSamePath, isScriptProtocol, joinRelativeURL, joinURL, normalizeURL, parseAuth, parseFilename, parseHost, parsePath, parseQuery, parseURL, resolveURL, stringifyParsedURL, stringifyQuery, withBase, withFragment, withHttp, withHttps, withLeadingSlash, withProtocol, withQuery, withTrailingSlash, withoutBase, withoutFragment, withoutHost, withoutLeadingSlash, withoutProtocol, withoutTrailingSlash };\n"],"x_google_ignoreList":[0],"mappings":"AAQU,OAAO;AAkLjB,MAAM,wBAAwB;AAC9B,MAAM,iBAAiB;AACvB,MAAM,0BAA0B;AAOhC,SAAS,YAAY,aAAa,OAAO,EAAE,EAAE;AAC3C,KAAI,OAAO,SAAS,UAClB,QAAO,EAAE,gBAAgB,MAAM;AAEjC,KAAI,KAAK,OACP,QAAO,sBAAsB,KAAK,YAAY;AAEhD,QAAO,eAAe,KAAK,YAAY,KAAK,KAAK,iBAAiB,wBAAwB,KAAK,YAAY,GAAG;;AA0QhH,MAAM,mBAAmB,OAAO,IAAI,uBAAuB;AAC3D,SAAS,SAAS,QAAQ,IAAI,cAAc;CAC1C,MAAM,qBAAqB,MAAM,MAC/B,mDACD;AACD,KAAI,oBAAoB;EACtB,MAAM,GAAG,QAAQ,YAAY,MAAM;AACnC,SAAO;GACL,UAAU,OAAO,aAAa;GAC9B,UAAU;GACV,MAAM,SAAS;GACf,MAAM;GACN,MAAM;GACN,QAAQ;GACR,MAAM;GACP;;AAEH,KAAI,CAAC,YAAY,OAAO,EAAE,gBAAgB,MAAM,CAAC,CAC/C,QAAO,eAAe,SAAS,eAAe,MAAM,GAAG,UAAU,MAAM;CAEzE,MAAM,GAAG,WAAW,IAAI,MAAM,cAAc,MAAM,MAAM,QAAQ,OAAO,IAAI,CAAC,MAAM,4CAA4C,IAAI,EAAE;CACpI,IAAI,GAAG,OAAO,IAAI,OAAO,MAAM,YAAY,MAAM,iBAAiB,IAAI,EAAE;AACxE,KAAI,aAAa,QACf,QAAO,KAAK,QAAQ,mBAAmB,GAAG;CAE5C,MAAM,EAAE,UAAU,QAAQ,SAAS,UAAU,KAAK;AAClD,QAAO;EACL,UAAU,SAAS,aAAa;EAChC,MAAM,OAAO,KAAK,MAAM,GAAG,KAAK,IAAI,GAAG,KAAK,SAAS,EAAE,CAAC,GAAG;EAC3D;EACA;EACA;EACA;GACC,mBAAmB,CAAC;EACtB;;AAEH,SAAS,UAAU,QAAQ,IAAI;CAC7B,MAAM,CAAC,WAAW,IAAI,SAAS,IAAI,OAAO,OAAO,MAAM,MAAM,2BAA2B,IAAI,EAAE,EAAE,OAAO,EAAE;AACzG,QAAO;EACL;EACA;EACA;EACD"}
|
|
1
|
+
{"version":3,"file":"dist-DdQWiZn8.mjs","names":[],"sources":["../../../node_modules/.pnpm/ufo@1.6.4/node_modules/ufo/dist/index.mjs"],"sourcesContent":["const n = /[^\\0-\\x7E]/;\nconst t = /[\\x2E\\u3002\\uFF0E\\uFF61]/g;\nconst o = {\n overflow: \"Overflow Error\",\n \"not-basic\": \"Illegal Input\",\n \"invalid-input\": \"Invalid Input\"\n};\nconst e = Math.floor;\nconst r = String.fromCharCode;\nfunction s(n2) {\n throw new RangeError(o[n2]);\n}\nconst c = function(n2, t2) {\n return n2 + 22 + 75 * (n2 < 26) - ((t2 != 0) << 5);\n};\nconst u = function(n2, t2, o2) {\n let r2 = 0;\n for (n2 = o2 ? e(n2 / 700) : n2 >> 1, n2 += e(n2 / t2); n2 > 455; r2 += 36) {\n n2 = e(n2 / 35);\n }\n return e(r2 + 36 * n2 / (n2 + 38));\n};\nfunction toASCII(o2) {\n return (function(n2, o3) {\n const e2 = n2.split(\"@\");\n let r2 = \"\";\n e2.length > 1 && (r2 = e2[0] + \"@\", n2 = e2[1]);\n const s2 = (function(n3, t2) {\n const o4 = [];\n let e3 = n3.length;\n for (; e3--; ) {\n o4[e3] = t2(n3[e3]);\n }\n return o4;\n })((n2 = n2.replace(t, \".\")).split(\".\"), o3).join(\".\");\n return r2 + s2;\n })(o2, function(t2) {\n return n.test(t2) ? \"xn--\" + (function(n2) {\n const t3 = [];\n const o3 = (n2 = (function(n3) {\n const t4 = [];\n let o4 = 0;\n const e2 = n3.length;\n for (; o4 < e2; ) {\n const r2 = n3.charCodeAt(o4++);\n if (r2 >= 55296 && r2 <= 56319 && o4 < e2) {\n const e3 = n3.charCodeAt(o4++);\n (64512 & e3) == 56320 ? t4.push(((1023 & r2) << 10) + (1023 & e3) + 65536) : (t4.push(r2), o4--);\n } else {\n t4.push(r2);\n }\n }\n return t4;\n })(n2)).length;\n let f = 128;\n let i = 0;\n let l = 72;\n for (const o4 of n2) {\n o4 < 128 && t3.push(r(o4));\n }\n const h = t3.length;\n let p = h;\n for (h && t3.push(\"-\"); p < o3; ) {\n let o4 = 2147483647;\n for (const t4 of n2) {\n t4 >= f && t4 < o4 && (o4 = t4);\n }\n const a = p + 1;\n o4 - f > e((2147483647 - i) / a) && s(\"overflow\"), i += (o4 - f) * a, f = o4;\n for (const o5 of n2) {\n if (o5 < f && ++i > 2147483647 && s(\"overflow\"), o5 == f) {\n let n3 = i;\n for (let o6 = 36; ; o6 += 36) {\n const s2 = o6 <= l ? 1 : o6 >= l + 26 ? 26 : o6 - l;\n if (n3 < s2) {\n break;\n }\n const u2 = n3 - s2;\n const f2 = 36 - s2;\n t3.push(r(c(s2 + u2 % f2, 0))), n3 = e(u2 / f2);\n }\n t3.push(r(c(n3, 0))), l = u(i, a, p == h), i = 0, ++p;\n }\n }\n ++i, ++f;\n }\n return t3.join(\"\");\n })(t2) : t2;\n });\n}\n\nconst HASH_RE = /#/g;\nconst AMPERSAND_RE = /&/g;\nconst SLASH_RE = /\\//g;\nconst EQUAL_RE = /=/g;\nconst IM_RE = /\\?/g;\nconst PLUS_RE = /\\+/g;\nconst ENC_CARET_RE = /%5e/gi;\nconst ENC_BACKTICK_RE = /%60/gi;\nconst ENC_CURLY_OPEN_RE = /%7b/gi;\nconst ENC_PIPE_RE = /%7c/gi;\nconst ENC_CURLY_CLOSE_RE = /%7d/gi;\nconst ENC_SPACE_RE = /%20/gi;\nconst ENC_SLASH_RE = /%2f/gi;\nconst ENC_ENC_SLASH_RE = /%252f/gi;\nfunction encode(text) {\n return encodeURI(\"\" + text).replace(ENC_PIPE_RE, \"|\");\n}\nfunction encodeHash(text) {\n return encode(text).replace(ENC_CURLY_OPEN_RE, \"{\").replace(ENC_CURLY_CLOSE_RE, \"}\").replace(ENC_CARET_RE, \"^\");\n}\nfunction encodeQueryValue(input) {\n return encode(typeof input === \"string\" ? input : JSON.stringify(input)).replace(PLUS_RE, \"%2B\").replace(ENC_SPACE_RE, \"+\").replace(HASH_RE, \"%23\").replace(AMPERSAND_RE, \"%26\").replace(ENC_BACKTICK_RE, \"`\").replace(ENC_CARET_RE, \"^\").replace(SLASH_RE, \"%2F\");\n}\nfunction encodeQueryKey(text) {\n return encodeQueryValue(text).replace(EQUAL_RE, \"%3D\");\n}\nfunction encodePath(text) {\n return encode(text).replace(HASH_RE, \"%23\").replace(IM_RE, \"%3F\").replace(ENC_ENC_SLASH_RE, \"%2F\").replace(AMPERSAND_RE, \"%26\").replace(PLUS_RE, \"%2B\");\n}\nfunction encodeParam(text) {\n return encodePath(text).replace(SLASH_RE, \"%2F\");\n}\nfunction decode(text = \"\") {\n try {\n return decodeURIComponent(\"\" + text);\n } catch {\n return \"\" + text;\n }\n}\nfunction decodePath(text) {\n return decode(text.replace(ENC_SLASH_RE, \"%252F\"));\n}\nfunction decodeQueryKey(text) {\n return decode(text.replace(PLUS_RE, \" \"));\n}\nfunction decodeQueryValue(text) {\n return decode(text.replace(PLUS_RE, \" \"));\n}\nfunction encodeHost(name = \"\") {\n return toASCII(name);\n}\n\nfunction parseQuery(parametersString = \"\") {\n const object = /* @__PURE__ */ Object.create(null);\n if (parametersString[0] === \"?\") {\n parametersString = parametersString.slice(1);\n }\n for (const parameter of parametersString.split(\"&\")) {\n const s = parameter.match(/([^=]+)=?(.*)/) || [];\n if (s.length < 2) {\n continue;\n }\n const key = decodeQueryKey(s[1]);\n if (key === \"__proto__\" || key === \"constructor\") {\n continue;\n }\n const value = decodeQueryValue(s[2] || \"\");\n if (object[key] === void 0) {\n object[key] = value;\n } else if (Array.isArray(object[key])) {\n object[key].push(value);\n } else {\n object[key] = [object[key], value];\n }\n }\n return object;\n}\nfunction encodeQueryItem(key, value) {\n if (typeof value === \"number\" || typeof value === \"boolean\") {\n value = String(value);\n }\n if (!value) {\n return encodeQueryKey(key);\n }\n if (Array.isArray(value)) {\n return value.map(\n (_value) => `${encodeQueryKey(key)}=${encodeQueryValue(_value)}`\n ).join(\"&\");\n }\n return `${encodeQueryKey(key)}=${encodeQueryValue(value)}`;\n}\nfunction stringifyQuery(query) {\n return Object.keys(query).filter((k) => query[k] !== void 0).map((k) => encodeQueryItem(k, query[k])).filter(Boolean).join(\"&\");\n}\n\nconst PROTOCOL_STRICT_REGEX = /^[\\s\\w\\0+.-]{2,}:([/\\\\]{1,2})/;\nconst PROTOCOL_REGEX = /^[\\s\\w\\0+.-]{2,}:([/\\\\]{2})?/;\nconst PROTOCOL_RELATIVE_REGEX = /^([/\\\\]\\s*){2,}[^/\\\\]/;\nconst PROTOCOL_SCRIPT_RE = /^[\\s\\0]*(blob|data|javascript|vbscript):$/i;\nconst TRAILING_SLASH_RE = /\\/$|\\/\\?|\\/#/;\nconst JOIN_LEADING_SLASH_RE = /^\\.?\\//;\nfunction isRelative(inputString) {\n return [\"./\", \"../\"].some((string_) => inputString.startsWith(string_));\n}\nfunction hasProtocol(inputString, opts = {}) {\n if (typeof opts === \"boolean\") {\n opts = { acceptRelative: opts };\n }\n if (opts.strict) {\n return PROTOCOL_STRICT_REGEX.test(inputString);\n }\n return PROTOCOL_REGEX.test(inputString) || (opts.acceptRelative ? PROTOCOL_RELATIVE_REGEX.test(inputString) : false);\n}\nfunction isScriptProtocol(protocol) {\n return !!protocol && PROTOCOL_SCRIPT_RE.test(protocol);\n}\nfunction hasTrailingSlash(input = \"\", respectQueryAndFragment) {\n if (!respectQueryAndFragment) {\n return input.endsWith(\"/\");\n }\n return TRAILING_SLASH_RE.test(input);\n}\nfunction withoutTrailingSlash(input = \"\", respectQueryAndFragment) {\n if (!respectQueryAndFragment) {\n return (hasTrailingSlash(input) ? input.slice(0, -1) : input) || \"/\";\n }\n if (!hasTrailingSlash(input, true)) {\n return input || \"/\";\n }\n let path = input;\n let fragment = \"\";\n const fragmentIndex = input.indexOf(\"#\");\n if (fragmentIndex !== -1) {\n path = input.slice(0, fragmentIndex);\n fragment = input.slice(fragmentIndex);\n }\n const [s0, ...s] = path.split(\"?\");\n const cleanPath = s0.endsWith(\"/\") ? s0.slice(0, -1) : s0;\n return (cleanPath || \"/\") + (s.length > 0 ? `?${s.join(\"?\")}` : \"\") + fragment;\n}\nfunction withTrailingSlash(input = \"\", respectQueryAndFragment) {\n if (!respectQueryAndFragment) {\n return input.endsWith(\"/\") ? input : input + \"/\";\n }\n if (hasTrailingSlash(input, true)) {\n return input || \"/\";\n }\n let path = input;\n let fragment = \"\";\n const fragmentIndex = input.indexOf(\"#\");\n if (fragmentIndex !== -1) {\n path = input.slice(0, fragmentIndex);\n fragment = input.slice(fragmentIndex);\n if (!path) {\n return fragment;\n }\n }\n const [s0, ...s] = path.split(\"?\");\n return s0 + \"/\" + (s.length > 0 ? `?${s.join(\"?\")}` : \"\") + fragment;\n}\nfunction hasLeadingSlash(input = \"\") {\n return input.startsWith(\"/\");\n}\nfunction withoutLeadingSlash(input = \"\") {\n return (hasLeadingSlash(input) ? input.slice(1) : input) || \"/\";\n}\nfunction withLeadingSlash(input = \"\") {\n return hasLeadingSlash(input) ? input : \"/\" + input;\n}\nfunction cleanDoubleSlashes(input = \"\") {\n return input.split(\"://\").map((string_) => string_.replace(/\\/{2,}/g, \"/\")).join(\"://\");\n}\nfunction withBase(input, base) {\n if (isEmptyURL(base) || hasProtocol(input)) {\n return input;\n }\n const _base = withoutTrailingSlash(base);\n if (input.startsWith(_base)) {\n const nextChar = input[_base.length];\n if (!nextChar || nextChar === \"/\" || nextChar === \"?\") {\n return input;\n }\n }\n return joinURL(_base, input);\n}\nfunction withoutBase(input, base) {\n if (isEmptyURL(base)) {\n return input;\n }\n const _base = withoutTrailingSlash(base);\n if (!input.startsWith(_base)) {\n return input;\n }\n const nextChar = input[_base.length];\n if (nextChar && nextChar !== \"/\" && nextChar !== \"?\") {\n return input;\n }\n const trimmed = input.slice(_base.length).replace(/^\\/+/, \"\");\n return \"/\" + trimmed;\n}\nfunction withQuery(input, query) {\n const parsed = parseURL(input);\n const mergedQuery = { ...parseQuery(parsed.search), ...query };\n parsed.search = stringifyQuery(mergedQuery);\n return stringifyParsedURL(parsed);\n}\nfunction filterQuery(input, predicate) {\n if (!input.includes(\"?\")) {\n return input;\n }\n const parsed = parseURL(input);\n const query = parseQuery(parsed.search);\n const filteredQuery = Object.fromEntries(\n Object.entries(query).filter(([key, value]) => predicate(key, value))\n );\n parsed.search = stringifyQuery(filteredQuery);\n return stringifyParsedURL(parsed);\n}\nfunction getQuery(input) {\n return parseQuery(parseURL(input).search);\n}\nfunction isEmptyURL(url) {\n return !url || url === \"/\";\n}\nfunction isNonEmptyURL(url) {\n return url && url !== \"/\";\n}\nfunction joinURL(base, ...input) {\n let url = base || \"\";\n for (const segment of input.filter((url2) => isNonEmptyURL(url2))) {\n if (url) {\n const _segment = segment.replace(JOIN_LEADING_SLASH_RE, \"\");\n url = withTrailingSlash(url) + _segment;\n } else {\n url = segment;\n }\n }\n return url;\n}\nfunction joinRelativeURL(..._input) {\n const JOIN_SEGMENT_SPLIT_RE = /\\/(?!\\/)/;\n const input = _input.filter(Boolean);\n const segments = [];\n let segmentsDepth = 0;\n for (const i of input) {\n if (!i || i === \"/\") {\n continue;\n }\n for (const [sindex, s] of i.split(JOIN_SEGMENT_SPLIT_RE).entries()) {\n if (!s || s === \".\") {\n continue;\n }\n if (s === \"..\") {\n if (segments.length === 1 && hasProtocol(segments[0])) {\n continue;\n }\n segments.pop();\n segmentsDepth--;\n continue;\n }\n if (sindex === 1 && segments[segments.length - 1]?.endsWith(\":/\")) {\n segments[segments.length - 1] += \"/\" + s;\n continue;\n }\n segments.push(s);\n segmentsDepth++;\n }\n }\n let url = segments.join(\"/\");\n if (segmentsDepth >= 0) {\n if (input[0]?.startsWith(\"/\") && !url.startsWith(\"/\")) {\n url = \"/\" + url;\n } else if (input[0]?.startsWith(\"./\") && !url.startsWith(\"./\")) {\n url = \"./\" + url;\n }\n } else {\n url = \"../\".repeat(-1 * segmentsDepth) + url;\n }\n if (input[input.length - 1]?.endsWith(\"/\") && !url.endsWith(\"/\")) {\n url += \"/\";\n }\n return url;\n}\nfunction withHttp(input) {\n return withProtocol(input, \"http://\");\n}\nfunction withHttps(input) {\n return withProtocol(input, \"https://\");\n}\nfunction withoutProtocol(input) {\n return withProtocol(input, \"\");\n}\nfunction withProtocol(input, protocol) {\n let match = input.match(PROTOCOL_REGEX);\n if (!match) {\n match = input.match(/^\\/{2,}/);\n }\n if (!match) {\n return protocol + input;\n }\n return protocol + input.slice(match[0].length);\n}\nfunction normalizeURL(input) {\n const parsed = parseURL(input);\n parsed.pathname = encodePath(decodePath(parsed.pathname));\n parsed.hash = encodeHash(decode(parsed.hash));\n parsed.host = encodeHost(decode(parsed.host));\n parsed.search = stringifyQuery(parseQuery(parsed.search));\n return stringifyParsedURL(parsed);\n}\nfunction resolveURL(base = \"\", ...inputs) {\n if (typeof base !== \"string\") {\n throw new TypeError(\n `URL input should be string received ${typeof base} (${base})`\n );\n }\n const filteredInputs = inputs.filter((input) => isNonEmptyURL(input));\n if (filteredInputs.length === 0) {\n return base;\n }\n const url = parseURL(base);\n for (const inputSegment of filteredInputs) {\n const urlSegment = parseURL(inputSegment);\n if (urlSegment.pathname) {\n url.pathname = withTrailingSlash(url.pathname) + withoutLeadingSlash(urlSegment.pathname);\n }\n if (urlSegment.hash && urlSegment.hash !== \"#\") {\n url.hash = urlSegment.hash;\n }\n if (urlSegment.search && urlSegment.search !== \"?\") {\n if (url.search && url.search !== \"?\") {\n const queryString = stringifyQuery({\n ...parseQuery(url.search),\n ...parseQuery(urlSegment.search)\n });\n url.search = queryString.length > 0 ? \"?\" + queryString : \"\";\n } else {\n url.search = urlSegment.search;\n }\n }\n }\n return stringifyParsedURL(url);\n}\nfunction isSamePath(p1, p2) {\n return decode(withoutTrailingSlash(p1)) === decode(withoutTrailingSlash(p2));\n}\nfunction isEqual(a, b, options = {}) {\n if (!options.trailingSlash) {\n a = withTrailingSlash(a);\n b = withTrailingSlash(b);\n }\n if (!options.leadingSlash) {\n a = withLeadingSlash(a);\n b = withLeadingSlash(b);\n }\n if (!options.encoding) {\n a = decode(a);\n b = decode(b);\n }\n return a === b;\n}\nfunction withFragment(input, hash) {\n if (!hash || hash === \"#\") {\n return input;\n }\n const parsed = parseURL(input);\n parsed.hash = hash === \"\" ? \"\" : \"#\" + encodeHash(hash);\n return stringifyParsedURL(parsed);\n}\nfunction withoutFragment(input) {\n return stringifyParsedURL({ ...parseURL(input), hash: \"\" });\n}\nfunction withoutHost(input) {\n const parsed = parseURL(input);\n return (parsed.pathname || \"/\") + parsed.search + parsed.hash;\n}\n\nconst protocolRelative = Symbol.for(\"ufo:protocolRelative\");\nfunction parseURL(input = \"\", defaultProto) {\n const _specialProtoMatch = input.match(\n /^[\\s\\0]*(blob:|data:|javascript:|vbscript:)(.*)/i\n );\n if (_specialProtoMatch) {\n const [, _proto, _pathname = \"\"] = _specialProtoMatch;\n return {\n protocol: _proto.toLowerCase(),\n pathname: _pathname,\n href: _proto + _pathname,\n auth: \"\",\n host: \"\",\n search: \"\",\n hash: \"\"\n };\n }\n if (!hasProtocol(input, { acceptRelative: true })) {\n return defaultProto ? parseURL(defaultProto + input) : parsePath(input);\n }\n const [, protocol = \"\", auth, hostAndPath = \"\"] = input.replace(/\\\\/g, \"/\").match(/^[\\s\\0]*([\\w+.-]{2,}:)?\\/\\/([^/@]+@)?(.*)/) || [];\n let [, host = \"\", path = \"\"] = hostAndPath.match(/([^#/?]*)(.*)?/) || [];\n if (protocol === \"file:\") {\n path = path.replace(/\\/(?=[A-Za-z]:)/, \"\");\n }\n const { pathname, search, hash } = parsePath(path);\n return {\n protocol: protocol.toLowerCase(),\n auth: auth ? auth.slice(0, Math.max(0, auth.length - 1)) : \"\",\n host,\n pathname,\n search,\n hash,\n [protocolRelative]: !protocol\n };\n}\nfunction parsePath(input = \"\") {\n const [pathname = \"\", search = \"\", hash = \"\"] = (input.match(/([^#?]*)(\\?[^#]*)?(#.*)?/) || []).splice(1);\n return {\n pathname,\n search,\n hash\n };\n}\nfunction parseAuth(input = \"\") {\n const [username, password] = input.split(\":\");\n return {\n username: decode(username),\n password: decode(password)\n };\n}\nfunction parseHost(input = \"\") {\n const [hostname, port] = (input.match(/([^/:]*):?(\\d+)?/) || []).splice(1);\n return {\n hostname: decode(hostname),\n port\n };\n}\nfunction stringifyParsedURL(parsed) {\n const pathname = parsed.pathname || \"\";\n const search = parsed.search ? (parsed.search.startsWith(\"?\") ? \"\" : \"?\") + parsed.search : \"\";\n const hash = parsed.hash || \"\";\n const auth = parsed.auth ? parsed.auth + \"@\" : \"\";\n const host = parsed.host || \"\";\n const proto = parsed.protocol || parsed[protocolRelative] ? (parsed.protocol || \"\") + \"//\" : \"\";\n return proto + auth + host + pathname + search + hash;\n}\nconst FILENAME_STRICT_REGEX = /\\/([^/]+\\.[^/]+)$/;\nconst FILENAME_REGEX = /\\/([^/]+)$/;\nfunction parseFilename(input = \"\", opts) {\n const { pathname } = parseURL(input);\n const matches = opts?.strict ? pathname.match(FILENAME_STRICT_REGEX) : pathname.match(FILENAME_REGEX);\n return matches ? matches[1] : void 0;\n}\n\nclass $URL {\n protocol;\n host;\n auth;\n pathname;\n query = {};\n hash;\n constructor(input = \"\") {\n if (typeof input !== \"string\") {\n throw new TypeError(\n `URL input should be string received ${typeof input} (${input})`\n );\n }\n const parsed = parseURL(input);\n this.protocol = decode(parsed.protocol);\n this.host = decode(parsed.host);\n this.auth = decode(parsed.auth);\n this.pathname = decodePath(parsed.pathname);\n this.query = parseQuery(parsed.search);\n this.hash = decode(parsed.hash);\n }\n get hostname() {\n return parseHost(this.host).hostname;\n }\n get port() {\n return parseHost(this.host).port || \"\";\n }\n get username() {\n return parseAuth(this.auth).username;\n }\n get password() {\n return parseAuth(this.auth).password || \"\";\n }\n get hasProtocol() {\n return this.protocol.length;\n }\n get isAbsolute() {\n return this.hasProtocol || this.pathname[0] === \"/\";\n }\n get search() {\n const q = stringifyQuery(this.query);\n return q.length > 0 ? \"?\" + q : \"\";\n }\n get searchParams() {\n const p = new URLSearchParams();\n for (const name in this.query) {\n const value = this.query[name];\n if (Array.isArray(value)) {\n for (const v of value) {\n p.append(name, v);\n }\n } else {\n p.append(\n name,\n typeof value === \"string\" ? value : JSON.stringify(value)\n );\n }\n }\n return p;\n }\n get origin() {\n return (this.protocol ? this.protocol + \"//\" : \"\") + encodeHost(this.host);\n }\n get fullpath() {\n return encodePath(this.pathname) + this.search + encodeHash(this.hash);\n }\n get encodedAuth() {\n if (!this.auth) {\n return \"\";\n }\n const { username, password } = parseAuth(this.auth);\n return encodeURIComponent(username) + (password ? \":\" + encodeURIComponent(password) : \"\");\n }\n get href() {\n const auth = this.encodedAuth;\n const originWithAuth = (this.protocol ? this.protocol + \"//\" : \"\") + (auth ? auth + \"@\" : \"\") + encodeHost(this.host);\n return this.hasProtocol && this.isAbsolute ? originWithAuth + this.fullpath : this.fullpath;\n }\n append(url) {\n if (url.hasProtocol) {\n throw new Error(\"Cannot append a URL with protocol\");\n }\n Object.assign(this.query, url.query);\n if (url.pathname) {\n this.pathname = withTrailingSlash(this.pathname) + withoutLeadingSlash(url.pathname);\n }\n if (url.hash) {\n this.hash = url.hash;\n }\n }\n toJSON() {\n return this.href;\n }\n toString() {\n return this.href;\n }\n}\nfunction createURL(input) {\n return new $URL(input);\n}\n\nexport { $URL, cleanDoubleSlashes, createURL, decode, decodePath, decodeQueryKey, decodeQueryValue, encode, encodeHash, encodeHost, encodeParam, encodePath, encodeQueryItem, encodeQueryKey, encodeQueryValue, filterQuery, getQuery, hasLeadingSlash, hasProtocol, hasTrailingSlash, isEmptyURL, isEqual, isNonEmptyURL, isRelative, isSamePath, isScriptProtocol, joinRelativeURL, joinURL, normalizeURL, parseAuth, parseFilename, parseHost, parsePath, parseQuery, parseURL, resolveURL, stringifyParsedURL, stringifyQuery, withBase, withFragment, withHttp, withHttps, withLeadingSlash, withProtocol, withQuery, withTrailingSlash, withoutBase, withoutFragment, withoutHost, withoutLeadingSlash, withoutProtocol, withoutTrailingSlash };\n"],"x_google_ignoreList":[0],"mappings":"AAQU,OAAO;AAkLjB,MAAM,wBAAwB;AAC9B,MAAM,iBAAiB;AACvB,MAAM,0BAA0B;AAOhC,SAAS,YAAY,aAAa,OAAO,EAAE,EAAE;AAC3C,KAAI,OAAO,SAAS,UAClB,QAAO,EAAE,gBAAgB,MAAM;AAEjC,KAAI,KAAK,OACP,QAAO,sBAAsB,KAAK,YAAY;AAEhD,QAAO,eAAe,KAAK,YAAY,KAAK,KAAK,iBAAiB,wBAAwB,KAAK,YAAY,GAAG;;AA0QhH,MAAM,mBAAmB,OAAO,IAAI,uBAAuB;AAC3D,SAAS,SAAS,QAAQ,IAAI,cAAc;CAC1C,MAAM,qBAAqB,MAAM,MAC/B,mDACD;AACD,KAAI,oBAAoB;EACtB,MAAM,GAAG,QAAQ,YAAY,MAAM;AACnC,SAAO;GACL,UAAU,OAAO,aAAa;GAC9B,UAAU;GACV,MAAM,SAAS;GACf,MAAM;GACN,MAAM;GACN,QAAQ;GACR,MAAM;GACP;;AAEH,KAAI,CAAC,YAAY,OAAO,EAAE,gBAAgB,MAAM,CAAC,CAC/C,QAAO,eAAe,SAAS,eAAe,MAAM,GAAG,UAAU,MAAM;CAEzE,MAAM,GAAG,WAAW,IAAI,MAAM,cAAc,MAAM,MAAM,QAAQ,OAAO,IAAI,CAAC,MAAM,4CAA4C,IAAI,EAAE;CACpI,IAAI,GAAG,OAAO,IAAI,OAAO,MAAM,YAAY,MAAM,iBAAiB,IAAI,EAAE;AACxE,KAAI,aAAa,QACf,QAAO,KAAK,QAAQ,mBAAmB,GAAG;CAE5C,MAAM,EAAE,UAAU,QAAQ,SAAS,UAAU,KAAK;AAClD,QAAO;EACL,UAAU,SAAS,aAAa;EAChC,MAAM,OAAO,KAAK,MAAM,GAAG,KAAK,IAAI,GAAG,KAAK,SAAS,EAAE,CAAC,GAAG;EAC3D;EACA;EACA;EACA;GACC,mBAAmB,CAAC;EACtB;;AAEH,SAAS,UAAU,QAAQ,IAAI;CAC7B,MAAM,CAAC,WAAW,IAAI,SAAS,IAAI,OAAO,OAAO,MAAM,MAAM,2BAA2B,IAAI,EAAE,EAAE,OAAO,EAAE;AACzG,QAAO;EACL;EACA;EACA;EACD"}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import { t as getNitroRuntimeConfigRecord } from "./nitroConfigBridge-
|
|
2
|
-
import { r as httpPost } from "./http-
|
|
1
|
+
import { t as getNitroRuntimeConfigRecord } from "./nitroConfigBridge-BkVWnSV3.mjs";
|
|
2
|
+
import { r as httpPost } from "./http-ChVS9GYc.mjs";
|
|
3
3
|
//#region src/shared/config.ts
|
|
4
4
|
/** Read the full Nitro `useRuntimeConfig()` record (or `undefined` outside Nitro). */
|
|
5
5
|
function getRuntimeConfig() {
|
|
@@ -21,6 +21,37 @@ async function resolveAdapterConfig(namespace, fields, overrides) {
|
|
|
21
21
|
for (const { key, env } of fields) config[key] = overrides?.[key] ?? evlogNs?.[key] ?? rootNs?.[key] ?? resolveEnv(env);
|
|
22
22
|
return config;
|
|
23
23
|
}
|
|
24
|
+
/**
|
|
25
|
+
* Format environment variable names for user-facing messages.
|
|
26
|
+
* Drops internal `NUXT_*` aliases; joins alternatives with `, ` and fields with `/`.
|
|
27
|
+
*/
|
|
28
|
+
function formatPublicEnvKeys(...envLists) {
|
|
29
|
+
return envLists.map((list) => {
|
|
30
|
+
const all = (list ?? []).filter((k) => !!k);
|
|
31
|
+
const publicKeys = all.filter((k) => !k.startsWith("NUXT_"));
|
|
32
|
+
return [...new Set(publicKeys.length > 0 ? publicKeys : all)].join(", ");
|
|
33
|
+
}).filter((part) => part.length > 0).join("/");
|
|
34
|
+
}
|
|
35
|
+
const warnedDeprecatedAliases = /* @__PURE__ */ new Set();
|
|
36
|
+
/**
|
|
37
|
+
* Copy a deprecated config field onto its replacement when the replacement is
|
|
38
|
+
* unset, warning once per adapter/field pair.
|
|
39
|
+
*/
|
|
40
|
+
function applyDeprecatedAlias(config, opts) {
|
|
41
|
+
const record = config;
|
|
42
|
+
if (record[opts.to] === void 0 || record[opts.to] === null) {
|
|
43
|
+
const fromValue = record[opts.from];
|
|
44
|
+
if (fromValue !== void 0 && fromValue !== null) {
|
|
45
|
+
const warnKey = `${opts.adapter}:${opts.from}`;
|
|
46
|
+
if (!warnedDeprecatedAliases.has(warnKey)) {
|
|
47
|
+
warnedDeprecatedAliases.add(warnKey);
|
|
48
|
+
console.warn(`[evlog/${opts.adapter}] \`${opts.from}\` is deprecated, use \`${opts.to}\` instead.`);
|
|
49
|
+
}
|
|
50
|
+
record[opts.to] = fromValue;
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
return config;
|
|
54
|
+
}
|
|
24
55
|
function shouldProbeRuntimeConfig(fields, overrides) {
|
|
25
56
|
return fields.some(({ key, env }) => {
|
|
26
57
|
if (overrides?.[key] !== void 0) return false;
|
|
@@ -112,6 +143,6 @@ function defineHttpDrain(options) {
|
|
|
112
143
|
});
|
|
113
144
|
}
|
|
114
145
|
//#endregion
|
|
115
|
-
export {
|
|
146
|
+
export { getRuntimeConfig as a, formatPublicEnvKeys as i, defineHttpDrain as n, resolveAdapterConfig as o, applyDeprecatedAlias as r, defineDrain as t };
|
|
116
147
|
|
|
117
|
-
//# sourceMappingURL=drain-
|
|
148
|
+
//# sourceMappingURL=drain-fDb-eNwz.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"drain-fDb-eNwz.mjs","names":[],"sources":["../src/shared/config.ts","../src/shared/drain.ts"],"sourcesContent":["import { getNitroRuntimeConfigRecord } from './nitroConfigBridge'\n\n/** Read the full Nitro `useRuntimeConfig()` record (or `undefined` outside Nitro). */\nexport function getRuntimeConfig(): Promise<Record<string, any> | undefined> {\n return getNitroRuntimeConfigRecord()\n}\n\n/**\n * Description of a single adapter config field. `env` is the ordered list of\n * environment variables to fall back to, e.g. `['NUXT_AXIOM_API_KEY', 'AXIOM_API_KEY']`.\n */\nexport interface ConfigField<T> {\n key: keyof T & string\n env?: string[]\n}\n\n/**\n * Resolve adapter configuration with the standard priority chain:\n *\n * 1. `overrides` passed to the drain factory\n * 2. `runtimeConfig.evlog.{namespace}.{key}` (Nitro)\n * 3. `runtimeConfig.{namespace}.{key}` (Nitro)\n * 4. `process.env[envKey]` for each env in `field.env`\n */\nexport async function resolveAdapterConfig<T>(\n namespace: string,\n fields: ConfigField<T>[],\n overrides?: Partial<T>,\n): Promise<Partial<T>> {\n const runtimeConfig = shouldProbeRuntimeConfig(fields, overrides)\n ? await getRuntimeConfig()\n : undefined\n const evlogNs = runtimeConfig?.evlog?.[namespace]\n const rootNs = runtimeConfig?.[namespace]\n\n const config: Record<string, unknown> = {}\n\n for (const { key, env } of fields) {\n config[key] =\n overrides?.[key]\n ?? evlogNs?.[key]\n ?? rootNs?.[key]\n ?? resolveEnv(env)\n }\n\n return config as Partial<T>\n}\n\n/**\n * Format environment variable names for user-facing messages.\n * Drops internal `NUXT_*` aliases; joins alternatives with `, ` and fields with `/`.\n */\nexport function formatPublicEnvKeys(...envLists: (string[] | undefined)[]): string {\n return envLists\n .map((list) => {\n const all = (list ?? []).filter((k): k is string => !!k)\n const publicKeys = all.filter(k => !k.startsWith('NUXT_'))\n const keys = [...new Set(publicKeys.length > 0 ? publicKeys : all)]\n return keys.join(', ')\n })\n .filter(part => part.length > 0)\n .join('/')\n}\n\nconst warnedDeprecatedAliases = new Set<string>()\n\n/**\n * Copy a deprecated config field onto its replacement when the replacement is\n * unset, warning once per adapter/field pair.\n */\nexport function applyDeprecatedAlias<T extends object>(\n config: T,\n opts: {\n adapter: string\n from: keyof T & string\n to: keyof T & string\n },\n): T {\n const record = config as Record<string, unknown>\n if (record[opts.to] === undefined || record[opts.to] === null) {\n const fromValue = record[opts.from]\n if (fromValue !== undefined && fromValue !== null) {\n const warnKey = `${opts.adapter}:${opts.from}`\n if (!warnedDeprecatedAliases.has(warnKey)) {\n warnedDeprecatedAliases.add(warnKey)\n console.warn(`[evlog/${opts.adapter}] \\`${opts.from}\\` is deprecated, use \\`${opts.to}\\` instead.`)\n }\n record[opts.to] = fromValue\n }\n }\n return config\n}\n\n// Avoid the Nitro virtual-module import when env/overrides already resolve\n// every env-backed field — optional tuning fields (timeout, retries) should\n// not trigger a runtime probe in non-Nitro runtimes.\nfunction shouldProbeRuntimeConfig<T>(\n fields: ConfigField<T>[],\n overrides?: Partial<T>,\n): boolean {\n return fields.some(({ key, env }) => {\n if (overrides?.[key] !== undefined) return false\n if (!env) return false\n return resolveEnv(env) === undefined\n })\n}\n\nfunction resolveEnv(envKeys?: string[]): string | undefined {\n if (!envKeys) return undefined\n for (const key of envKeys) {\n const val = process.env[key]\n if (val) return val\n }\n return undefined\n}\n","import type { DrainContext, WideEvent } from '../types'\nimport { httpPost } from './http'\n\n/**\n * Drain definition backed by an arbitrary `send` function. Use this for\n * non-HTTP transports (filesystem, in-memory queue, native SDK). For HTTP\n * backends, use `defineHttpDrain` instead.\n */\nexport interface DrainOptions<TConfig> {\n /** Stable identifier used in error logs. */\n name: string\n /** Return `null` to skip draining (e.g. missing API key in dev). */\n resolve: () => TConfig | null | Promise<TConfig | null>\n send: (events: WideEvent[], config: TConfig) => Promise<void>\n}\n\n/**\n * Build a drain callback. Errors raised by `send` are logged with the drain\n * name and swallowed, so a failing drain never breaks the request pipeline.\n *\n * @example\n * ```ts\n * export function createMyDrain(overrides?: Partial<MyConfig>) {\n * return defineDrain<MyConfig>({\n * name: 'my-drain',\n * resolve: () => ({ url: process.env.MY_URL ?? null }),\n * send: async (events, config) => { ... },\n * })\n * }\n * ```\n */\nexport function defineDrain<TConfig>(options: DrainOptions<TConfig>): (ctx: DrainContext | DrainContext[]) => Promise<void> {\n return async (ctx: DrainContext | DrainContext[]) => {\n const contexts = Array.isArray(ctx) ? ctx : [ctx]\n if (contexts.length === 0) return\n\n const config = await options.resolve()\n if (!config) return\n\n try {\n await options.send(contexts.map(c => c.event), config)\n } catch (error) {\n console.error(`[evlog/${options.name}] Failed to send events:`, error)\n }\n }\n}\n\nexport interface HttpDrainRequest {\n url: string\n /** Caller is responsible for `Content-Type`. */\n headers: Record<string, string>\n body: string\n}\n\n/** Adapters only need to ship config + `encode()` — no manual `fetch`. */\nexport interface HttpDrainOptions<TConfig> {\n /** Stable identifier used in error logs. */\n name: string\n /** Return `null` to skip draining (e.g. missing API key in dev). */\n resolve: () => TConfig | null | Promise<TConfig | null>\n /** Return `null` to skip the batch without raising. */\n encode: (events: WideEvent[], config: TConfig) => HttpDrainRequest | null\n /** @default 5000 */\n timeout?: number\n /** @default 2 */\n retries?: number\n /** Read the timeout off the resolved config (falls back to `timeout`). */\n resolveTimeout?: (config: TConfig) => number | undefined\n /** Read the retry count off the resolved config (falls back to `retries`). */\n resolveRetries?: (config: TConfig) => number | undefined\n}\n\nconst DEFAULT_HTTP_TIMEOUT = 5000\n\n/**\n * Build an HTTP drain. Timeouts/retries are resolved from the config (with\n * overrides via `resolveTimeout` / `resolveRetries`) and forwarded to\n * {@link httpPost}.\n *\n * @example\n * ```ts\n * export function createMyDrain(overrides?: Partial<MyConfig>) {\n * return defineHttpDrain<MyConfig>({\n * name: 'my',\n * resolve: async () => {\n * const cfg = await resolveAdapterConfig<MyConfig>('my', FIELDS, overrides)\n * return cfg.apiKey ? cfg as MyConfig : null\n * },\n * encode: (events, config) => ({\n * url: `${config.endpoint ?? 'https://api.my.com'}/ingest`,\n * headers: { 'Content-Type': 'application/json', Authorization: `Bearer ${config.apiKey}` },\n * body: JSON.stringify(events),\n * }),\n * })\n * }\n * ```\n */\nexport function defineHttpDrain<TConfig>(options: HttpDrainOptions<TConfig>): (ctx: DrainContext | DrainContext[]) => Promise<void> {\n return defineDrain<TConfig>({\n name: options.name,\n resolve: options.resolve,\n send: async (events, config) => {\n if (events.length === 0) return\n const request = options.encode(events, config)\n if (!request) return\n const timeout = options.resolveTimeout?.(config)\n ?? (config as { timeout?: number }).timeout\n ?? options.timeout\n ?? DEFAULT_HTTP_TIMEOUT\n const retries = options.resolveRetries?.(config)\n ?? (config as { retries?: number }).retries\n ?? options.retries\n await httpPost({\n url: request.url,\n headers: request.headers,\n body: request.body,\n timeout,\n retries,\n label: options.name,\n source: options.name,\n })\n },\n })\n}\n"],"mappings":"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"}
|
package/dist/elysia/index.d.mts
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import { t as BaseEvlogOptions } from "../middleware-
|
|
1
|
+
import { rt as RequestLogger } from "../audit-D7v6JHj0.mjs";
|
|
2
|
+
import { t as BaseEvlogOptions } from "../middleware-B_k4Mrzg.mjs";
|
|
3
3
|
import { Elysia } from "elysia";
|
|
4
4
|
|
|
5
5
|
//#region src/elysia/index.d.ts
|
|
@@ -50,7 +50,7 @@ declare function evlog(options?: EvlogElysiaOptions): Elysia<"", {
|
|
|
50
50
|
decorator: {};
|
|
51
51
|
store: {};
|
|
52
52
|
derive: {
|
|
53
|
-
readonly log: RequestLogger
|
|
53
|
+
readonly log: RequestLogger;
|
|
54
54
|
};
|
|
55
55
|
resolve: {};
|
|
56
56
|
}, {
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.mts","names":[],"sources":["../../src/elysia/index.ts"],"mappings":";;;;;KAWY,kBAAA,GAAqB,gBAAA;;AAAjC;;;;;AAoBA;;;;;;;;;;;;iBAAgB,SAAA,oBAA6B,MAAA,kBAAA,CAAA,GAA4B,aAAA,CAAc,CAAA;;AAuEvF;;;;;;;;;;;;;;;;;;;;;;iBAAgB,KAAA,CAAM,OAAA,GAAS,kBAAA,GAAuB,MAAA"}
|
|
1
|
+
{"version":3,"file":"index.d.mts","names":[],"sources":["../../src/elysia/index.ts"],"mappings":";;;;;KAWY,kBAAA,GAAqB,gBAAA;;AAAjC;;;;;AAoBA;;;;;;;;;;;;iBAAgB,SAAA,oBAA6B,MAAA,kBAAA,CAAA,GAA4B,aAAA,CAAc,CAAA;;AAuEvF;;;;;;;;;;;;;;;;;;;;;;iBAAgB,KAAA,CAAM,OAAA,GAAS,kBAAA,GAAuB,MAAA;;;;kBAcG,aAAA;EAAA"}
|
package/dist/elysia/index.mjs
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
import { t as attachForkToLogger } from "../fork-
|
|
2
|
-
import { t as defineFrameworkIntegration } from "../integration-
|
|
1
|
+
import { t as attachForkToLogger } from "../fork-CgGlAaHa.mjs";
|
|
2
|
+
import { t as defineFrameworkIntegration } from "../integration-DYp2uw8O.mjs";
|
|
3
3
|
import { AsyncLocalStorage } from "node:async_hooks";
|
|
4
4
|
import { Elysia } from "elysia";
|
|
5
5
|
//#region src/elysia/index.ts
|
|
@@ -78,10 +78,12 @@ const integration = defineFrameworkIntegration({
|
|
|
78
78
|
function evlog(options = {}) {
|
|
79
79
|
const emitted = /* @__PURE__ */ new WeakSet();
|
|
80
80
|
const requestState = /* @__PURE__ */ new WeakMap();
|
|
81
|
-
return new Elysia({ name: "evlog" }).
|
|
81
|
+
return new Elysia({ name: "evlog" }).onRequest(({ request }) => {
|
|
82
|
+
const url = new URL(request.url);
|
|
83
|
+
const headers = request.headers.toJSON?.() ?? Object.fromEntries(request.headers.entries());
|
|
82
84
|
const ctx = {
|
|
83
85
|
request,
|
|
84
|
-
path,
|
|
86
|
+
path: url.pathname,
|
|
85
87
|
headers
|
|
86
88
|
};
|
|
87
89
|
const { logger, finish, skipped } = integration.start(ctx, options);
|
|
@@ -91,7 +93,8 @@ function evlog(options = {}) {
|
|
|
91
93
|
skipped,
|
|
92
94
|
logger
|
|
93
95
|
});
|
|
94
|
-
|
|
96
|
+
}).derive({ as: "global" }, ({ request }) => {
|
|
97
|
+
return { log: requestState.get(request)?.logger };
|
|
95
98
|
}).onAfterResponse({ as: "global" }, async ({ request, set }) => {
|
|
96
99
|
const state = requestState.get(request);
|
|
97
100
|
if (!state || state.skipped || emitted.has(request)) return;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.mjs","names":[],"sources":["../../src/elysia/index.ts"],"sourcesContent":["import { AsyncLocalStorage } from 'node:async_hooks'\nimport { Elysia } from 'elysia'\nimport type { RequestLogger } from '../types'\nimport { defineFrameworkIntegration } from '../shared/integration'\nimport type { BaseEvlogOptions } from '../shared/middleware'\nimport { attachForkToLogger } from '../shared/fork'\n\nconst storage = new AsyncLocalStorage<RequestLogger>()\n\nconst activeLoggers = new WeakSet<RequestLogger>()\n\nexport type EvlogElysiaOptions = BaseEvlogOptions\n\n/**\n * Get the request-scoped logger from anywhere in the call stack.\n * Must be called inside a request handled by the `evlog()` plugin.\n *\n * Unlike other frameworks, Elysia uses `storage.enterWith()` which persists\n * beyond the request lifecycle. This accessor additionally checks `activeLoggers`\n * to ensure the logger belongs to an in-flight request.\n *\n * @example\n * ```ts\n * import { useLogger } from 'evlog/elysia'\n *\n * function findUser(id: string) {\n * const log = useLogger()\n * log.set({ user: { id } })\n * }\n * ```\n */\nexport function useLogger<T extends object = Record<string, unknown>>(): RequestLogger<T> {\n const logger = storage.getStore()\n if (!logger || !activeLoggers.has(logger)) {\n throw new Error(\n '[evlog] useLogger() was called outside of an evlog plugin context. '\n + 'Make sure app.use(evlog()) is registered before your routes.',\n )\n }\n return logger as RequestLogger<T>\n}\n\ninterface ElysiaContext {\n request: Request\n path: string\n headers: Record<string, string>\n}\n\nconst integration = defineFrameworkIntegration<ElysiaContext>({\n name: 'elysia',\n extractRequest: ({ request, path, headers }) => ({\n method: request.method,\n path,\n headers,\n requestId: headers['x-request-id'],\n }),\n attachLogger: ({ request, path, headers }, logger) => {\n attachForkToLogger(storage, logger, {\n method: request.method,\n path,\n requestId: headers['x-request-id'],\n }, {\n onChildEnter: (child) => {\n activeLoggers.add(child)\n },\n onChildExit: (child) => {\n activeLoggers.delete(child)\n },\n })\n activeLoggers.add(logger)\n },\n})\n\ninterface RequestState {\n finish: (opts?: { status?: number; error?: Error }) => Promise<unknown>\n skipped: boolean\n logger: RequestLogger\n}\n\n/**\n * Create an evlog plugin for Elysia.\n *\n * @example\n * ```ts\n * import { Elysia } from 'elysia'\n * import { evlog } from 'evlog/elysia'\n * import { createAxiomDrain } from 'evlog/axiom'\n *\n * const app = new Elysia()\n * .use(evlog({\n * drain: createAxiomDrain(),\n * enrich: (ctx) => {\n * ctx.event.region = process.env.FLY_REGION\n * },\n * }))\n * .get('/health', ({ log }) => {\n * log.set({ route: 'health' })\n * return { ok: true }\n * })\n * .listen(3000)\n * ```\n */\nexport function evlog(options: EvlogElysiaOptions = {}) {\n const emitted = new WeakSet<Request>()\n const requestState = new WeakMap<Request, RequestState>()\n\n return new Elysia({ name: 'evlog' })\n .
|
|
1
|
+
{"version":3,"file":"index.mjs","names":[],"sources":["../../src/elysia/index.ts"],"sourcesContent":["import { AsyncLocalStorage } from 'node:async_hooks'\nimport { Elysia } from 'elysia'\nimport type { RequestLogger } from '../types'\nimport { defineFrameworkIntegration } from '../shared/integration'\nimport type { BaseEvlogOptions } from '../shared/middleware'\nimport { attachForkToLogger } from '../shared/fork'\n\nconst storage = new AsyncLocalStorage<RequestLogger>()\n\nconst activeLoggers = new WeakSet<RequestLogger>()\n\nexport type EvlogElysiaOptions = BaseEvlogOptions\n\n/**\n * Get the request-scoped logger from anywhere in the call stack.\n * Must be called inside a request handled by the `evlog()` plugin.\n *\n * Unlike other frameworks, Elysia uses `storage.enterWith()` which persists\n * beyond the request lifecycle. This accessor additionally checks `activeLoggers`\n * to ensure the logger belongs to an in-flight request.\n *\n * @example\n * ```ts\n * import { useLogger } from 'evlog/elysia'\n *\n * function findUser(id: string) {\n * const log = useLogger()\n * log.set({ user: { id } })\n * }\n * ```\n */\nexport function useLogger<T extends object = Record<string, unknown>>(): RequestLogger<T> {\n const logger = storage.getStore()\n if (!logger || !activeLoggers.has(logger)) {\n throw new Error(\n '[evlog] useLogger() was called outside of an evlog plugin context. '\n + 'Make sure app.use(evlog()) is registered before your routes.',\n )\n }\n return logger as RequestLogger<T>\n}\n\ninterface ElysiaContext {\n request: Request\n path: string\n headers: Record<string, string>\n}\n\nconst integration = defineFrameworkIntegration<ElysiaContext>({\n name: 'elysia',\n extractRequest: ({ request, path, headers }) => ({\n method: request.method,\n path,\n headers,\n requestId: headers['x-request-id'],\n }),\n attachLogger: ({ request, path, headers }, logger) => {\n attachForkToLogger(storage, logger, {\n method: request.method,\n path,\n requestId: headers['x-request-id'],\n }, {\n onChildEnter: (child) => {\n activeLoggers.add(child)\n },\n onChildExit: (child) => {\n activeLoggers.delete(child)\n },\n })\n activeLoggers.add(logger)\n },\n})\n\ninterface RequestState {\n finish: (opts?: { status?: number; error?: Error }) => Promise<unknown>\n skipped: boolean\n logger: RequestLogger\n}\n\n/**\n * Create an evlog plugin for Elysia.\n *\n * @example\n * ```ts\n * import { Elysia } from 'elysia'\n * import { evlog } from 'evlog/elysia'\n * import { createAxiomDrain } from 'evlog/axiom'\n *\n * const app = new Elysia()\n * .use(evlog({\n * drain: createAxiomDrain(),\n * enrich: (ctx) => {\n * ctx.event.region = process.env.FLY_REGION\n * },\n * }))\n * .get('/health', ({ log }) => {\n * log.set({ route: 'health' })\n * return { ok: true }\n * })\n * .listen(3000)\n * ```\n */\nexport function evlog(options: EvlogElysiaOptions = {}) {\n const emitted = new WeakSet<Request>()\n const requestState = new WeakMap<Request, RequestState>()\n\n return new Elysia({ name: 'evlog' })\n .onRequest(({ request }) => {\n const url = new URL(request.url)\n const headers = (request.headers).toJSON?.() ?? Object.fromEntries(request.headers.entries())\n const ctx: ElysiaContext = { request, path: url.pathname, headers }\n const { logger, finish, skipped } = integration.start(ctx, options)\n storage.enterWith(logger)\n requestState.set(request, { finish, skipped, logger })\n })\n .derive({ as: 'global' }, ({ request }) => {\n return { log: requestState.get(request)?.logger as RequestLogger }\n })\n .onAfterResponse({ as: 'global' }, async ({ request, set }) => {\n const state = requestState.get(request)\n if (!state || state.skipped || emitted.has(request)) return\n emitted.add(request)\n await state.finish({ status: set.status as number || 200 })\n activeLoggers.delete(state.logger)\n storage.enterWith(undefined as unknown as RequestLogger)\n })\n .onError({ as: 'global' }, async ({ request, error }) => {\n const state = requestState.get(request)\n if (!state || state.skipped || emitted.has(request)) return\n emitted.add(request)\n const err = error instanceof Error ? error : new Error(String(error))\n state.logger.error(err)\n await state.finish({ error: err })\n activeLoggers.delete(state.logger)\n storage.enterWith(undefined as unknown as RequestLogger)\n })\n}\n"],"mappings":";;;;;AAOA,MAAM,UAAU,IAAI,mBAAkC;AAEtD,MAAM,gCAAgB,IAAI,SAAwB;;;;;;;;;;;;;;;;;;;AAsBlD,SAAgB,YAA0E;CACxF,MAAM,SAAS,QAAQ,UAAU;AACjC,KAAI,CAAC,UAAU,CAAC,cAAc,IAAI,OAAO,CACvC,OAAM,IAAI,MACR,kIAED;AAEH,QAAO;;AAST,MAAM,cAAc,2BAA0C;CAC5D,MAAM;CACN,iBAAiB,EAAE,SAAS,MAAM,eAAe;EAC/C,QAAQ,QAAQ;EAChB;EACA;EACA,WAAW,QAAQ;EACpB;CACD,eAAe,EAAE,SAAS,MAAM,WAAW,WAAW;AACpD,qBAAmB,SAAS,QAAQ;GAClC,QAAQ,QAAQ;GAChB;GACA,WAAW,QAAQ;GACpB,EAAE;GACD,eAAe,UAAU;AACvB,kBAAc,IAAI,MAAM;;GAE1B,cAAc,UAAU;AACtB,kBAAc,OAAO,MAAM;;GAE9B,CAAC;AACF,gBAAc,IAAI,OAAO;;CAE5B,CAAC;;;;;;;;;;;;;;;;;;;;;;;;AA+BF,SAAgB,MAAM,UAA8B,EAAE,EAAE;CACtD,MAAM,0BAAU,IAAI,SAAkB;CACtC,MAAM,+BAAe,IAAI,SAAgC;AAEzD,QAAO,IAAI,OAAO,EAAE,MAAM,SAAS,CAAC,CACjC,WAAW,EAAE,cAAc;EAC1B,MAAM,MAAM,IAAI,IAAI,QAAQ,IAAI;EAChC,MAAM,UAAW,QAAQ,QAAS,UAAU,IAAI,OAAO,YAAY,QAAQ,QAAQ,SAAS,CAAC;EAC7F,MAAM,MAAqB;GAAE;GAAS,MAAM,IAAI;GAAU;GAAS;EACnE,MAAM,EAAE,QAAQ,QAAQ,YAAY,YAAY,MAAM,KAAK,QAAQ;AACnE,UAAQ,UAAU,OAAO;AACzB,eAAa,IAAI,SAAS;GAAE;GAAQ;GAAS;GAAQ,CAAC;GACtD,CACD,OAAO,EAAE,IAAI,UAAU,GAAG,EAAE,cAAc;AACzC,SAAO,EAAE,KAAK,aAAa,IAAI,QAAQ,EAAE,QAAyB;GAClE,CACD,gBAAgB,EAAE,IAAI,UAAU,EAAE,OAAO,EAAE,SAAS,UAAU;EAC7D,MAAM,QAAQ,aAAa,IAAI,QAAQ;AACvC,MAAI,CAAC,SAAS,MAAM,WAAW,QAAQ,IAAI,QAAQ,CAAE;AACrD,UAAQ,IAAI,QAAQ;AACpB,QAAM,MAAM,OAAO,EAAE,QAAQ,IAAI,UAAoB,KAAK,CAAC;AAC3D,gBAAc,OAAO,MAAM,OAAO;AAClC,UAAQ,UAAU,KAAA,EAAsC;GACxD,CACD,QAAQ,EAAE,IAAI,UAAU,EAAE,OAAO,EAAE,SAAS,YAAY;EACvD,MAAM,QAAQ,aAAa,IAAI,QAAQ;AACvC,MAAI,CAAC,SAAS,MAAM,WAAW,QAAQ,IAAI,QAAQ,CAAE;AACrD,UAAQ,IAAI,QAAQ;EACpB,MAAM,MAAM,iBAAiB,QAAQ,QAAQ,IAAI,MAAM,OAAO,MAAM,CAAC;AACrE,QAAM,OAAO,MAAM,IAAI;AACvB,QAAM,MAAM,OAAO,EAAE,OAAO,KAAK,CAAC;AAClC,gBAAc,OAAO,MAAM,OAAO;AAClC,UAAQ,UAAU,KAAA,EAAsC;GACxD"}
|
|
@@ -0,0 +1,120 @@
|
|
|
1
|
+
import { a as isFrameworkRuntimePath } from "./pretty-error-THg0U0w9.mjs";
|
|
2
|
+
import { createRequire } from "node:module";
|
|
3
|
+
import { isAbsolute, relative } from "node:path";
|
|
4
|
+
import { fileURLToPath, pathToFileURL } from "node:url";
|
|
5
|
+
import { existsSync, readFileSync } from "node:fs";
|
|
6
|
+
//#region src/shared/enrich-error-stack-next.node.ts
|
|
7
|
+
const require = createRequire(import.meta.url);
|
|
8
|
+
function formatMappedFrame(methodName, sourceURL, line1, column1) {
|
|
9
|
+
let sourceLocation = line1 !== null ? `:${line1}` : "";
|
|
10
|
+
if (column1 !== null && sourceLocation !== "") sourceLocation += `:${column1}`;
|
|
11
|
+
let fileLocation;
|
|
12
|
+
if (sourceURL !== null && sourceURL.startsWith("file://") && URL.canParse(sourceURL)) fileLocation = relative(process.cwd(), fileURLToPath(sourceURL));
|
|
13
|
+
else if (sourceURL !== null && sourceURL.startsWith("/")) fileLocation = relative(process.cwd(), sourceURL);
|
|
14
|
+
else fileLocation = sourceURL ?? "unknown";
|
|
15
|
+
return methodName ? ` at ${methodName} (${fileLocation}${sourceLocation})` : ` at ${fileLocation}${sourceLocation}`;
|
|
16
|
+
}
|
|
17
|
+
function shouldSkipMappedSource(source) {
|
|
18
|
+
const normalized = source.replace(/\\/g, "/");
|
|
19
|
+
return normalized.includes("node_modules") || normalized.includes("/packages/evlog/") || isFrameworkRuntimePath(normalized);
|
|
20
|
+
}
|
|
21
|
+
function resolveFrameFile(frame) {
|
|
22
|
+
if (!frame.file) return null;
|
|
23
|
+
if (frame.file.startsWith("file://")) try {
|
|
24
|
+
return fileURLToPath(frame.file);
|
|
25
|
+
} catch {
|
|
26
|
+
return frame.file;
|
|
27
|
+
}
|
|
28
|
+
if (isAbsolute(frame.file)) return frame.file;
|
|
29
|
+
return null;
|
|
30
|
+
}
|
|
31
|
+
function getSourceMapConsumer(frameFile, cache) {
|
|
32
|
+
const cached = cache.get(frameFile);
|
|
33
|
+
if (cached !== void 0) return cached;
|
|
34
|
+
const mapPath = `${frameFile}.map`;
|
|
35
|
+
if (!existsSync(mapPath)) {
|
|
36
|
+
cache.set(frameFile, null);
|
|
37
|
+
return null;
|
|
38
|
+
}
|
|
39
|
+
try {
|
|
40
|
+
const sourceMapModule = require("next/dist/compiled/source-map");
|
|
41
|
+
const payload = JSON.parse(readFileSync(mapPath, "utf8"));
|
|
42
|
+
const chunkUrl = pathToFileURL(frameFile).href;
|
|
43
|
+
const consumer = new sourceMapModule.SourceMapConsumer(payload, `${chunkUrl}.map`);
|
|
44
|
+
cache.set(frameFile, consumer);
|
|
45
|
+
return consumer;
|
|
46
|
+
} catch {
|
|
47
|
+
cache.set(frameFile, null);
|
|
48
|
+
return null;
|
|
49
|
+
}
|
|
50
|
+
}
|
|
51
|
+
function mapFrame(frame, cache) {
|
|
52
|
+
if (frame.file?.startsWith("node:")) return {
|
|
53
|
+
frame,
|
|
54
|
+
skipped: true
|
|
55
|
+
};
|
|
56
|
+
const frameFile = resolveFrameFile(frame);
|
|
57
|
+
if (!frameFile || frame.line1 === null) return {
|
|
58
|
+
frame,
|
|
59
|
+
skipped: false
|
|
60
|
+
};
|
|
61
|
+
const consumer = getSourceMapConsumer(frameFile, cache);
|
|
62
|
+
if (!consumer) {
|
|
63
|
+
if (frameFile.includes(".next/")) return {
|
|
64
|
+
frame,
|
|
65
|
+
skipped: true
|
|
66
|
+
};
|
|
67
|
+
return {
|
|
68
|
+
frame,
|
|
69
|
+
skipped: false
|
|
70
|
+
};
|
|
71
|
+
}
|
|
72
|
+
const sourcePosition = consumer.originalPositionFor({
|
|
73
|
+
line: frame.line1,
|
|
74
|
+
column: (frame.column1 ?? 1) - 1
|
|
75
|
+
});
|
|
76
|
+
if (!sourcePosition.source || sourcePosition.line === null) return {
|
|
77
|
+
frame,
|
|
78
|
+
skipped: frameFile.includes(".next/")
|
|
79
|
+
};
|
|
80
|
+
if (shouldSkipMappedSource(sourcePosition.source)) return {
|
|
81
|
+
frame,
|
|
82
|
+
skipped: true
|
|
83
|
+
};
|
|
84
|
+
return {
|
|
85
|
+
frame: {
|
|
86
|
+
...frame,
|
|
87
|
+
file: sourcePosition.source,
|
|
88
|
+
line1: sourcePosition.line,
|
|
89
|
+
column1: sourcePosition.column === null ? null : sourcePosition.column + 1,
|
|
90
|
+
methodName: sourcePosition.name ?? frame.methodName
|
|
91
|
+
},
|
|
92
|
+
skipped: false
|
|
93
|
+
};
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Rewrite `error.stack` with Turbopack/Webpack source-mapped frames in Next.js dev.
|
|
97
|
+
* Reads sibling `.map` files for `.next` chunks (same resolution as the dev overlay).
|
|
98
|
+
*/
|
|
99
|
+
function enrichErrorStackFromNextDev(error) {
|
|
100
|
+
if (process.env.NODE_ENV === "production") return;
|
|
101
|
+
if (!error.stack) return;
|
|
102
|
+
try {
|
|
103
|
+
const { parseStack } = require("next/dist/server/lib/parse-stack");
|
|
104
|
+
const frames = parseStack(error.stack);
|
|
105
|
+
if (frames.length === 0) return;
|
|
106
|
+
const cache = /* @__PURE__ */ new Map();
|
|
107
|
+
const mappedLines = [];
|
|
108
|
+
for (const frame of frames) {
|
|
109
|
+
const { frame: mapped, skipped } = mapFrame(frame, cache);
|
|
110
|
+
if (skipped) continue;
|
|
111
|
+
mappedLines.push(formatMappedFrame(mapped.methodName, mapped.file, mapped.line1, mapped.column1));
|
|
112
|
+
}
|
|
113
|
+
if (mappedLines.length === 0) return;
|
|
114
|
+
error.stack = `${error.name || "Error"}: ${error.message}\n${mappedLines.join("\n")}`;
|
|
115
|
+
} catch {}
|
|
116
|
+
}
|
|
117
|
+
//#endregion
|
|
118
|
+
export { enrichErrorStackFromNextDev };
|
|
119
|
+
|
|
120
|
+
//# sourceMappingURL=enrich-error-stack-next.node-Dgm_rCf5.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"enrich-error-stack-next.node-Dgm_rCf5.mjs","names":[],"sources":["../src/shared/enrich-error-stack-next.node.ts"],"sourcesContent":["import { existsSync, readFileSync } from 'node:fs'\nimport { createRequire } from 'node:module'\nimport { isAbsolute, relative } from 'node:path'\nimport { pathToFileURL, fileURLToPath } from 'node:url'\nimport { isFrameworkRuntimePath } from './pretty-error'\n\n/** Parsed stack frame from Next.js `parseStack`. */\ninterface NextParsedFrame {\n file: string | null\n line1: number | null\n column1: number | null\n methodName: string | null\n arguments: string[]\n}\n\ntype SourceMapConsumerInstance = {\n originalPositionFor: (pos: { line: number, column: number }) => {\n source: string | null\n line: number | null\n column: number | null\n name: string | null\n }\n}\n\nconst require = createRequire(import.meta.url)\n\nfunction formatMappedFrame(\n methodName: string | null,\n sourceURL: string | null,\n line1: number | null,\n column1: number | null,\n): string {\n let sourceLocation = line1 !== null ? `:${line1}` : ''\n if (column1 !== null && sourceLocation !== '') {\n sourceLocation += `:${column1}`\n }\n\n let fileLocation: string\n if (sourceURL !== null && sourceURL.startsWith('file://') && URL.canParse(sourceURL)) {\n fileLocation = relative(process.cwd(), fileURLToPath(sourceURL))\n } else if (sourceURL !== null && sourceURL.startsWith('/')) {\n fileLocation = relative(process.cwd(), sourceURL)\n } else {\n fileLocation = sourceURL ?? 'unknown'\n }\n\n return methodName\n ? ` at ${methodName} (${fileLocation}${sourceLocation})`\n : ` at ${fileLocation}${sourceLocation}`\n}\n\nfunction shouldSkipMappedSource(source: string): boolean {\n const normalized = source.replace(/\\\\/g, '/')\n return normalized.includes('node_modules')\n || normalized.includes('/packages/evlog/')\n || isFrameworkRuntimePath(normalized)\n}\n\nfunction resolveFrameFile(frame: NextParsedFrame): string | null {\n if (!frame.file) return null\n if (frame.file.startsWith('file://')) {\n try {\n return fileURLToPath(frame.file)\n } catch {\n return frame.file\n }\n }\n if (isAbsolute(frame.file)) return frame.file\n return null\n}\n\nfunction getSourceMapConsumer(\n frameFile: string,\n cache: Map<string, SourceMapConsumerInstance | null>,\n): SourceMapConsumerInstance | null {\n const cached = cache.get(frameFile)\n if (cached !== undefined) return cached\n\n const mapPath = `${frameFile}.map`\n if (!existsSync(mapPath)) {\n cache.set(frameFile, null)\n return null\n }\n\n try {\n const sourceMapModule = require('next/dist/compiled/source-map') as {\n SourceMapConsumer: new(payload: unknown, sourceMapURL: string) => SourceMapConsumerInstance\n }\n const payload = JSON.parse(readFileSync(mapPath, 'utf8')) as unknown\n const chunkUrl = pathToFileURL(frameFile).href\n const consumer = new sourceMapModule.SourceMapConsumer(payload, `${chunkUrl}.map`)\n cache.set(frameFile, consumer)\n return consumer\n } catch {\n cache.set(frameFile, null)\n return null\n }\n}\n\nfunction mapFrame(\n frame: NextParsedFrame,\n cache: Map<string, SourceMapConsumerInstance | null>,\n): { frame: NextParsedFrame, skipped: boolean } {\n if (frame.file?.startsWith('node:')) {\n return { frame, skipped: true }\n }\n\n const frameFile = resolveFrameFile(frame)\n if (!frameFile || frame.line1 === null) {\n return { frame, skipped: false }\n }\n\n const consumer = getSourceMapConsumer(frameFile, cache)\n if (!consumer) {\n if (frameFile.includes('.next/')) {\n return { frame, skipped: true }\n }\n return { frame, skipped: false }\n }\n\n const sourcePosition = consumer.originalPositionFor({\n line: frame.line1,\n column: (frame.column1 ?? 1) - 1,\n })\n\n if (!sourcePosition.source || sourcePosition.line === null) {\n return { frame, skipped: frameFile.includes('.next/') }\n }\n\n if (shouldSkipMappedSource(sourcePosition.source)) {\n return { frame, skipped: true }\n }\n\n return {\n frame: {\n ...frame,\n file: sourcePosition.source,\n line1: sourcePosition.line,\n column1: sourcePosition.column === null ? null : sourcePosition.column + 1,\n methodName: sourcePosition.name ?? frame.methodName,\n },\n skipped: false,\n }\n}\n\n/**\n * Rewrite `error.stack` with Turbopack/Webpack source-mapped frames in Next.js dev.\n * Reads sibling `.map` files for `.next` chunks (same resolution as the dev overlay).\n */\nexport function enrichErrorStackFromNextDev(error: Error): void {\n if (process.env.NODE_ENV === 'production') return\n if (!error.stack) return\n\n try {\n const { parseStack } = require('next/dist/server/lib/parse-stack') as {\n parseStack: (stack: string, distDir?: string) => NextParsedFrame[]\n }\n\n const frames = parseStack(error.stack)\n if (frames.length === 0) return\n\n const cache = new Map<string, SourceMapConsumerInstance | null>()\n const mappedLines: string[] = []\n\n for (const frame of frames) {\n const { frame: mapped, skipped } = mapFrame(frame, cache)\n if (skipped) continue\n mappedLines.push(formatMappedFrame(mapped.methodName, mapped.file, mapped.line1, mapped.column1))\n }\n\n if (mappedLines.length === 0) return\n\n error.stack = `${error.name || 'Error'}: ${error.message}\\n${mappedLines.join('\\n')}`\n } catch {\n // Next internals unavailable — keep the original stack\n }\n}\n"],"mappings":"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"}
|