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.
Files changed (223) hide show
  1. package/README.md +14 -16
  2. package/dist/adapters/axiom.d.mts +3 -3
  3. package/dist/adapters/axiom.d.mts.map +1 -1
  4. package/dist/adapters/axiom.mjs +10 -14
  5. package/dist/adapters/axiom.mjs.map +1 -1
  6. package/dist/adapters/better-stack.d.mts +2 -2
  7. package/dist/adapters/better-stack.d.mts.map +1 -1
  8. package/dist/adapters/better-stack.mjs +9 -13
  9. package/dist/adapters/better-stack.mjs.map +1 -1
  10. package/dist/adapters/datadog.d.mts +3 -3
  11. package/dist/adapters/datadog.mjs +9 -5
  12. package/dist/adapters/datadog.mjs.map +1 -1
  13. package/dist/adapters/fs.d.mts +1 -1
  14. package/dist/adapters/fs.d.mts.map +1 -1
  15. package/dist/adapters/fs.mjs +15 -2
  16. package/dist/adapters/fs.mjs.map +1 -1
  17. package/dist/adapters/hyperdx.d.mts +2 -2
  18. package/dist/adapters/hyperdx.mjs +3 -3
  19. package/dist/adapters/hyperdx.mjs.map +1 -1
  20. package/dist/adapters/memory.d.mts +2 -3
  21. package/dist/adapters/memory.d.mts.map +1 -1
  22. package/dist/adapters/memory.mjs +2 -3
  23. package/dist/adapters/memory.mjs.map +1 -1
  24. package/dist/adapters/otlp.d.mts +4 -4
  25. package/dist/adapters/otlp.mjs +19 -11
  26. package/dist/adapters/otlp.mjs.map +1 -1
  27. package/dist/adapters/posthog.d.mts +2 -2
  28. package/dist/adapters/posthog.mjs +5 -5
  29. package/dist/adapters/posthog.mjs.map +1 -1
  30. package/dist/adapters/sentry.d.mts +3 -3
  31. package/dist/adapters/sentry.mjs +6 -6
  32. package/dist/adapters/sentry.mjs.map +1 -1
  33. package/dist/ai/index.d.mts +1 -1
  34. package/dist/{audit-BUI3af4w.mjs → audit-BQt8yAHo.mjs} +376 -116
  35. package/dist/audit-BQt8yAHo.mjs.map +1 -0
  36. package/dist/{audit-DVdkntSO.d.mts → audit-D7v6JHj0.d.mts} +113 -35
  37. package/dist/audit-D7v6JHj0.d.mts.map +1 -0
  38. package/dist/better-auth/index.d.mts +1 -1
  39. package/dist/browser.d.mts +1 -1
  40. package/dist/deferred-drain-jeajC8QF.mjs +36 -0
  41. package/dist/deferred-drain-jeajC8QF.mjs.map +1 -0
  42. package/dist/{define-D6OJdSUH.mjs → define-Bpaymi-h.mjs} +2 -1
  43. package/dist/define-Bpaymi-h.mjs.map +1 -0
  44. package/dist/{define-D-BVMf2l.d.mts → define-DTQpu4f6.d.mts} +8 -3
  45. package/dist/define-DTQpu4f6.d.mts.map +1 -0
  46. package/dist/dev-terminal-D4UaEm17.mjs +54 -0
  47. package/dist/dev-terminal-D4UaEm17.mjs.map +1 -0
  48. package/dist/{dist-H3GIh-KK.mjs → dist-DdQWiZn8.mjs} +1 -1
  49. package/dist/{dist-H3GIh-KK.mjs.map → dist-DdQWiZn8.mjs.map} +1 -1
  50. package/dist/{drain-7n3K6kPe.mjs → drain-fDb-eNwz.mjs} +35 -4
  51. package/dist/drain-fDb-eNwz.mjs.map +1 -0
  52. package/dist/elysia/index.d.mts +3 -3
  53. package/dist/elysia/index.d.mts.map +1 -1
  54. package/dist/elysia/index.mjs +8 -5
  55. package/dist/elysia/index.mjs.map +1 -1
  56. package/dist/enrich-error-stack-next.node-Dgm_rCf5.mjs +120 -0
  57. package/dist/enrich-error-stack-next.node-Dgm_rCf5.mjs.map +1 -0
  58. package/dist/{enricher-UW9npoB2.d.mts → enricher-CBRmQw6e.d.mts} +2 -2
  59. package/dist/{enricher-UW9npoB2.d.mts.map → enricher-CBRmQw6e.d.mts.map} +1 -1
  60. package/dist/{enricher-N0erZS87.mjs → enricher-DAWf2-Fx.mjs} +2 -2
  61. package/dist/{enricher-N0erZS87.mjs.map → enricher-DAWf2-Fx.mjs.map} +1 -1
  62. package/dist/enrichers.d.mts +2 -2
  63. package/dist/enrichers.mjs +2 -2
  64. package/dist/{error-CVtn5U7b.d.mts → error-CpghjrkY.d.mts} +2 -2
  65. package/dist/{error-CVtn5U7b.d.mts.map → error-CpghjrkY.d.mts.map} +1 -1
  66. package/dist/error.d.mts +1 -1
  67. package/dist/{errors-dEMNQCiL.d.mts → errors-BLU4Tfpe.d.mts} +2 -2
  68. package/dist/{errors-dEMNQCiL.d.mts.map → errors-BLU4Tfpe.d.mts.map} +1 -1
  69. package/dist/{errors-BQgyQ9xe.mjs → errors-DA0cyr8q.mjs} +1 -1
  70. package/dist/{errors-BQgyQ9xe.mjs.map → errors-DA0cyr8q.mjs.map} +1 -1
  71. package/dist/{event-1BMl7o0k.mjs → event-qwAv-7AZ.mjs} +1 -1
  72. package/dist/{event-1BMl7o0k.mjs.map → event-qwAv-7AZ.mjs.map} +1 -1
  73. package/dist/express/index.d.mts +2 -2
  74. package/dist/express/index.mjs +3 -3
  75. package/dist/fastify/index.d.mts +2 -2
  76. package/dist/fastify/index.mjs +2 -2
  77. package/dist/{fork-Bga8x-X4.mjs → fork-CgGlAaHa.mjs} +39 -14
  78. package/dist/fork-CgGlAaHa.mjs.map +1 -0
  79. package/dist/{headers-CU-QqnYg.mjs → headers-VtmnWcfn.mjs} +1 -1
  80. package/dist/{headers-CU-QqnYg.mjs.map → headers-VtmnWcfn.mjs.map} +1 -1
  81. package/dist/hono/index.d.mts +2 -2
  82. package/dist/hono/index.d.mts.map +1 -1
  83. package/dist/hono/index.mjs +10 -2
  84. package/dist/hono/index.mjs.map +1 -1
  85. package/dist/{http-B6YgAhyN.mjs → http-ChVS9GYc.mjs} +2 -2
  86. package/dist/{http-B6YgAhyN.mjs.map → http-ChVS9GYc.mjs.map} +1 -1
  87. package/dist/http.d.mts +1 -1
  88. package/dist/{index-ZSRQP_BI.d.mts → index-EvnrXvQM.d.mts} +15 -8
  89. package/dist/index-EvnrXvQM.d.mts.map +1 -0
  90. package/dist/index.d.mts +9 -9
  91. package/dist/index.mjs +9 -15
  92. package/dist/index.mjs.map +1 -1
  93. package/dist/instrumentation-create-BrjQtSKD.d.mts +115 -0
  94. package/dist/instrumentation-create-BrjQtSKD.d.mts.map +1 -0
  95. package/dist/{integration-Dhig7ae6.mjs → integration-DYp2uw8O.mjs} +3 -3
  96. package/dist/{integration-Dhig7ae6.mjs.map → integration-DYp2uw8O.mjs.map} +1 -1
  97. package/dist/{logger-CTcvd5Cc.d.mts → logger-mHIWxBhJ.d.mts} +12 -4
  98. package/dist/logger-mHIWxBhJ.d.mts.map +1 -0
  99. package/dist/logger.d.mts +2 -2
  100. package/dist/logger.mjs +2 -2
  101. package/dist/{middleware-31KhtiEF.d.mts → middleware-B_k4Mrzg.d.mts} +9 -2
  102. package/dist/middleware-B_k4Mrzg.d.mts.map +1 -0
  103. package/dist/nestjs/index.d.mts +2 -2
  104. package/dist/nestjs/index.mjs +4 -4
  105. package/dist/next/client.d.mts +1 -1
  106. package/dist/next/index.d.mts +4 -4
  107. package/dist/next/index.d.mts.map +1 -1
  108. package/dist/next/index.mjs +49 -38
  109. package/dist/next/index.mjs.map +1 -1
  110. package/dist/next/instrumentation/create.d.mts +2 -0
  111. package/dist/next/instrumentation/create.mjs +155 -0
  112. package/dist/next/instrumentation/create.mjs.map +1 -0
  113. package/dist/next/instrumentation.d.mts +2 -77
  114. package/dist/next/instrumentation.mjs +32 -81
  115. package/dist/next/instrumentation.mjs.map +1 -1
  116. package/dist/next/stream.d.mts +1 -1
  117. package/dist/next/stream.mjs +2 -2
  118. package/dist/next/stream.mjs.map +1 -1
  119. package/dist/nitro/errorHandler.mjs +23 -18
  120. package/dist/nitro/errorHandler.mjs.map +1 -1
  121. package/dist/nitro/module.d.mts +2 -2
  122. package/dist/nitro/module.d.mts.map +1 -1
  123. package/dist/nitro/module.mjs +2 -1
  124. package/dist/nitro/module.mjs.map +1 -1
  125. package/dist/nitro/plugin.mjs +74 -18
  126. package/dist/nitro/plugin.mjs.map +1 -1
  127. package/dist/nitro/v3/errorHandler.d.mts +0 -7
  128. package/dist/nitro/v3/errorHandler.mjs +13 -15
  129. package/dist/nitro/v3/errorHandler.mjs.map +1 -1
  130. package/dist/nitro/v3/index.d.mts +2 -2
  131. package/dist/nitro/v3/module.d.mts +1 -1
  132. package/dist/nitro/v3/module.d.mts.map +1 -1
  133. package/dist/nitro/v3/module.mjs +3 -3
  134. package/dist/nitro/v3/module.mjs.map +1 -1
  135. package/dist/nitro/v3/plugin.mjs +76 -44
  136. package/dist/nitro/v3/plugin.mjs.map +1 -1
  137. package/dist/nitro/v3/useLogger.d.mts +1 -1
  138. package/dist/nitro-ClRZLD1g.mjs +96 -0
  139. package/dist/nitro-ClRZLD1g.mjs.map +1 -0
  140. package/dist/{nitro-BRddgqSb.d.mts → nitro-_Hda8Deo.d.mts} +7 -2
  141. package/dist/nitro-_Hda8Deo.d.mts.map +1 -0
  142. package/dist/{nitroConfigBridge-NbFn-sIK.mjs → nitroConfigBridge-BkVWnSV3.mjs} +9 -2
  143. package/dist/nitroConfigBridge-BkVWnSV3.mjs.map +1 -0
  144. package/dist/{nodeResponse-BkkionWl.mjs → nodeResponse-CIEEbrNE.mjs} +1 -1
  145. package/dist/{nodeResponse-BkkionWl.mjs.map → nodeResponse-CIEEbrNE.mjs.map} +1 -1
  146. package/dist/nuxt/module.d.mts +6 -1
  147. package/dist/nuxt/module.d.mts.map +1 -1
  148. package/dist/nuxt/module.mjs +11 -4
  149. package/dist/nuxt/module.mjs.map +1 -1
  150. package/dist/orpc/index.d.mts +2 -2
  151. package/dist/orpc/index.d.mts.map +1 -1
  152. package/dist/orpc/index.mjs +5 -4
  153. package/dist/orpc/index.mjs.map +1 -1
  154. package/dist/{package-B23bR3tK.mjs → package-CNV_CXs8.mjs} +2 -2
  155. package/dist/package-CNV_CXs8.mjs.map +1 -0
  156. package/dist/{parseError-D4PIxEWo.d.mts → parseError-BeBXEd2V.d.mts} +2 -2
  157. package/dist/parseError-BeBXEd2V.d.mts.map +1 -0
  158. package/dist/pipeline.d.mts +0 -19
  159. package/dist/pipeline.d.mts.map +1 -1
  160. package/dist/pipeline.mjs +12 -0
  161. package/dist/pipeline.mjs.map +1 -1
  162. package/dist/pretty-error-THg0U0w9.mjs +288 -0
  163. package/dist/pretty-error-THg0U0w9.mjs.map +1 -0
  164. package/dist/pretty-error-snippet.node-itfCajBM.mjs +48 -0
  165. package/dist/pretty-error-snippet.node-itfCajBM.mjs.map +1 -0
  166. package/dist/react-router/index.d.mts +2 -2
  167. package/dist/react-router/index.mjs +5 -6
  168. package/dist/react-router/index.mjs.map +1 -1
  169. package/dist/{routes-CnIgYWf8.mjs → routes-4rMzRyTk.mjs} +1 -1
  170. package/dist/{routes-CnIgYWf8.mjs.map → routes-4rMzRyTk.mjs.map} +1 -1
  171. package/dist/runtime/client/log.d.mts +1 -1
  172. package/dist/runtime/server/routes/_evlog/ingest.post.d.mts +7 -0
  173. package/dist/runtime/server/routes/_evlog/ingest.post.mjs +39 -3
  174. package/dist/runtime/server/routes/_evlog/ingest.post.mjs.map +1 -1
  175. package/dist/runtime/server/useLogger.d.mts +1 -1
  176. package/dist/runtime/utils/parseError.d.mts +2 -2
  177. package/dist/runtime/utils/parseError.mjs +1 -1
  178. package/dist/{severity-R5Egq3qz.mjs → severity-CwXUSHt3.mjs} +1 -1
  179. package/dist/{severity-R5Egq3qz.mjs.map → severity-CwXUSHt3.mjs.map} +1 -1
  180. package/dist/{source-location-Dco0cRTz.mjs → source-location-CHOPF2nd.mjs} +2 -1
  181. package/dist/{source-location-Dco0cRTz.mjs.map → source-location-CHOPF2nd.mjs.map} +1 -1
  182. package/dist/{storage-BNubsWwz.mjs → storage-7X37OToT.mjs} +1 -1
  183. package/dist/{storage-BNubsWwz.mjs.map → storage-7X37OToT.mjs.map} +1 -1
  184. package/dist/stream.d.mts +1 -1
  185. package/dist/stream.mjs +1 -1
  186. package/dist/streamResponse-CmQ3qUbF.mjs +94 -0
  187. package/dist/streamResponse-CmQ3qUbF.mjs.map +1 -0
  188. package/dist/sveltekit/index.d.mts +2 -2
  189. package/dist/sveltekit/index.d.mts.map +1 -1
  190. package/dist/sveltekit/index.mjs +7 -8
  191. package/dist/sveltekit/index.mjs.map +1 -1
  192. package/dist/toolkit.d.mts +51 -6
  193. package/dist/toolkit.d.mts.map +1 -1
  194. package/dist/toolkit.mjs +15 -14
  195. package/dist/types.d.mts +2 -2
  196. package/dist/{useLogger-CqvH6qOf.d.mts → useLogger-Cfv8Ck8b.d.mts} +2 -2
  197. package/dist/{useLogger-CqvH6qOf.d.mts.map → useLogger-Cfv8Ck8b.d.mts.map} +1 -1
  198. package/dist/{utils-DxqvIOyR.d.mts → utils-CJJG0ZYW.d.mts} +11 -3
  199. package/dist/{utils-DxqvIOyR.d.mts.map → utils-CJJG0ZYW.d.mts.map} +1 -1
  200. package/dist/utils.d.mts +2 -2
  201. package/dist/utils.mjs +23 -9
  202. package/dist/utils.mjs.map +1 -1
  203. package/dist/vite/index.d.mts +1 -1
  204. package/dist/vite/index.mjs +1 -1
  205. package/dist/workers.d.mts +1 -1
  206. package/dist/workers.mjs +1 -1
  207. package/package.json +12 -1
  208. package/dist/audit-BUI3af4w.mjs.map +0 -1
  209. package/dist/audit-DVdkntSO.d.mts.map +0 -1
  210. package/dist/define-D-BVMf2l.d.mts.map +0 -1
  211. package/dist/define-D6OJdSUH.mjs.map +0 -1
  212. package/dist/drain-7n3K6kPe.mjs.map +0 -1
  213. package/dist/fork-Bga8x-X4.mjs.map +0 -1
  214. package/dist/index-ZSRQP_BI.d.mts.map +0 -1
  215. package/dist/logger-CTcvd5Cc.d.mts.map +0 -1
  216. package/dist/middleware-31KhtiEF.d.mts.map +0 -1
  217. package/dist/next/instrumentation.d.mts.map +0 -1
  218. package/dist/nitro-BRddgqSb.d.mts.map +0 -1
  219. package/dist/nitro-DErMq_Zj.mjs +0 -34
  220. package/dist/nitro-DErMq_Zj.mjs.map +0 -1
  221. package/dist/nitroConfigBridge-NbFn-sIK.mjs.map +0 -1
  222. package/dist/package-B23bR3tK.mjs.map +0 -1
  223. package/dist/parseError-D4PIxEWo.d.mts.map +0 -1
@@ -1,9 +1,20 @@
1
- import { t as extractErrorStatus } from "../errors-BQgyQ9xe.mjs";
2
- import { n as serializeEvlogErrorResponse, t as resolveEvlogError } from "../nitro-DErMq_Zj.mjs";
3
- import { getRequestURL, send, setResponseHeader, setResponseStatus } from "h3";
1
+ import { t as extractErrorStatus } from "../errors-DA0cyr8q.mjs";
2
+ import { a as serializeEvlogErrorResponse, i as resolveEvlogError, n as markH3ErrorHandled, o as shouldSuppressNitroDevOverlay, s as suppressNitroDevOverlay, t as buildPlainNitroErrorBody } from "../nitro-ClRZLD1g.mjs";
3
+ import { getRequestURL, setResponseHeader, setResponseStatus } from "h3";
4
4
  import { defineNitroErrorHandler } from "nitropack/runtime/internal/error/utils";
5
5
  //#region src/nitro/errorHandler.ts
6
6
  /**
7
+ * Flush the error response by ending the Node response directly.
8
+ *
9
+ * h3 v1's `send()` is a no-op once the event is marked handled, and the event
10
+ * must be marked handled *before* responding so Nitro's chained dev handler
11
+ * (Youch overlay) does not run after us. Ending the Node response directly
12
+ * resolves that tension: the flush is synchronous and unconditional (#374).
13
+ */
14
+ function endNodeResponse(event, body) {
15
+ if (!event.node.res.writableEnded) event.node.res.end(body);
16
+ }
17
+ /**
7
18
  * Custom Nitro error handler that properly serializes EvlogError.
8
19
  * This ensures that 'data' (containing 'why', 'fix', 'link') is preserved
9
20
  * in the JSON response regardless of the underlying HTTP framework.
@@ -11,29 +22,23 @@ import { defineNitroErrorHandler } from "nitropack/runtime/internal/error/utils"
11
22
  * For non-EvlogError, it preserves Nitro's default response shape while
12
23
  * sanitizing internal error details in production for 5xx errors.
13
24
  */
14
- var errorHandler_default = defineNitroErrorHandler((error, event) => {
25
+ var errorHandler_default = defineNitroErrorHandler(async (error, event, ctx) => {
26
+ const suppressOverlay = shouldSuppressNitroDevOverlay();
27
+ if (!suppressOverlay && ctx?.defaultHandler) await ctx.defaultHandler(error, event, { silent: false });
28
+ markH3ErrorHandled(event);
29
+ if (suppressOverlay) suppressNitroDevOverlay(error);
15
30
  const evlogError = resolveEvlogError(error);
16
31
  const isDev = process.env.NODE_ENV === "development";
17
32
  const url = getRequestURL(event, { xForwardedHost: true }).pathname;
18
33
  if (!evlogError) {
19
- const status = extractErrorStatus(error);
20
- const rawMessage = (error.statusText ?? error.statusMessage ?? error.message) || "Internal Server Error";
21
- const message = isDev ? rawMessage : status >= 500 ? "Internal Server Error" : rawMessage;
22
- setResponseStatus(event, status);
34
+ const body = buildPlainNitroErrorBody(error, url, isDev);
35
+ setResponseStatus(event, body.status);
23
36
  setResponseHeader(event, "Content-Type", "application/json");
24
- return send(event, JSON.stringify({
25
- url,
26
- status,
27
- statusCode: status,
28
- statusText: message,
29
- statusMessage: message,
30
- message,
31
- error: true
32
- }));
37
+ return endNodeResponse(event, JSON.stringify(body));
33
38
  }
34
39
  setResponseStatus(event, extractErrorStatus(evlogError));
35
40
  setResponseHeader(event, "Content-Type", "application/json");
36
- return send(event, JSON.stringify(serializeEvlogErrorResponse(evlogError, url)));
41
+ return endNodeResponse(event, JSON.stringify(serializeEvlogErrorResponse(evlogError, url)));
37
42
  });
38
43
  //#endregion
39
44
  export { errorHandler_default as default };
@@ -1 +1 @@
1
- {"version":3,"file":"errorHandler.mjs","names":[],"sources":["../../src/nitro/errorHandler.ts"],"sourcesContent":["// Import from specific subpath — the barrel 'nitropack/runtime' re-exports from\n// internal/app.mjs which imports virtual modules that crash outside rollup builds.\nimport { defineNitroErrorHandler } from 'nitropack/runtime/internal/error/utils'\nimport { getRequestURL, setResponseHeader, setResponseStatus, send } from 'h3'\nimport { resolveEvlogError, extractErrorStatus, serializeEvlogErrorResponse } from '../nitro'\n\n/**\n * Custom Nitro error handler that properly serializes EvlogError.\n * This ensures that 'data' (containing 'why', 'fix', 'link') is preserved\n * in the JSON response regardless of the underlying HTTP framework.\n *\n * For non-EvlogError, it preserves Nitro's default response shape while\n * sanitizing internal error details in production for 5xx errors.\n */\nexport default defineNitroErrorHandler((error, event) => {\n const evlogError = resolveEvlogError(error)\n\n const isDev = process.env.NODE_ENV === 'development'\n const url = getRequestURL(event, { xForwardedHost: true }).pathname\n\n // For non-EvlogError, preserve Nitro's default response shape\n if (!evlogError) {\n const status = extractErrorStatus(error)\n\n // Derive message from statusText/statusMessage/message for cross-version compatibility\n const rawMessage = ((error as { statusText?: string }).statusText\n ?? (error as { statusMessage?: string }).statusMessage\n ?? error.message) || 'Internal Server Error'\n\n // Sanitize internal error details in production for 5xx errors\n const message = isDev\n ? rawMessage\n : (status >= 500 ? 'Internal Server Error' : rawMessage)\n\n setResponseStatus(event, status)\n setResponseHeader(event, 'Content-Type', 'application/json')\n\n return send(event, JSON.stringify({\n url,\n status,\n statusCode: status,\n statusText: message,\n statusMessage: message,\n message,\n error: true,\n }))\n }\n\n const status = extractErrorStatus(evlogError)\n\n setResponseStatus(event, status)\n setResponseHeader(event, 'Content-Type', 'application/json')\n\n return send(event, JSON.stringify(serializeEvlogErrorResponse(evlogError, url)))\n})\n"],"mappings":";;;;;;;;;;;;;AAcA,IAAA,uBAAe,yBAAyB,OAAO,UAAU;CACvD,MAAM,aAAa,kBAAkB,MAAM;CAE3C,MAAM,QAAQ,QAAQ,IAAI,aAAa;CACvC,MAAM,MAAM,cAAc,OAAO,EAAE,gBAAgB,MAAM,CAAC,CAAC;AAG3D,KAAI,CAAC,YAAY;EACf,MAAM,SAAS,mBAAmB,MAAM;EAGxC,MAAM,cAAe,MAAkC,cACjD,MAAqC,iBACtC,MAAM,YAAY;EAGvB,MAAM,UAAU,QACZ,aACC,UAAU,MAAM,0BAA0B;AAE/C,oBAAkB,OAAO,OAAO;AAChC,oBAAkB,OAAO,gBAAgB,mBAAmB;AAE5D,SAAO,KAAK,OAAO,KAAK,UAAU;GAChC;GACA;GACA,YAAY;GACZ,YAAY;GACZ,eAAe;GACf;GACA,OAAO;GACR,CAAC,CAAC;;AAKL,mBAAkB,OAFH,mBAAmB,WAEH,CAAC;AAChC,mBAAkB,OAAO,gBAAgB,mBAAmB;AAE5D,QAAO,KAAK,OAAO,KAAK,UAAU,4BAA4B,YAAY,IAAI,CAAC,CAAC;EAChF"}
1
+ {"version":3,"file":"errorHandler.mjs","names":[],"sources":["../../src/nitro/errorHandler.ts"],"sourcesContent":["// Import from specific subpath — the barrel 'nitropack/runtime' re-exports from\n// internal/app.mjs which imports virtual modules that crash outside rollup builds.\nimport { defineNitroErrorHandler } from 'nitropack/runtime/internal/error/utils'\nimport { getRequestURL, setResponseHeader, setResponseStatus } from 'h3'\nimport type { H3Event } from 'h3'\nimport {\n resolveEvlogError,\n extractErrorStatus,\n buildPlainNitroErrorBody,\n serializeEvlogErrorResponse,\n markH3ErrorHandled,\n shouldSuppressNitroDevOverlay,\n suppressNitroDevOverlay,\n} from '../nitro'\n\n/**\n * Flush the error response by ending the Node response directly.\n *\n * h3 v1's `send()` is a no-op once the event is marked handled, and the event\n * must be marked handled *before* responding so Nitro's chained dev handler\n * (Youch overlay) does not run after us. Ending the Node response directly\n * resolves that tension: the flush is synchronous and unconditional (#374).\n */\nfunction endNodeResponse(event: H3Event, body: string): void {\n if (!event.node.res.writableEnded) {\n event.node.res.end(body)\n }\n}\n\n/**\n * Custom Nitro error handler that properly serializes EvlogError.\n * This ensures that 'data' (containing 'why', 'fix', 'link') is preserved\n * in the JSON response regardless of the underlying HTTP framework.\n *\n * For non-EvlogError, it preserves Nitro's default response shape while\n * sanitizing internal error details in production for 5xx errors.\n */\nexport default defineNitroErrorHandler(async (error, event, ctx) => {\n const suppressOverlay = shouldSuppressNitroDevOverlay()\n\n // Nitro v2 always passes `ctx`, but a missing context (e.g. the handler\n // invoked directly) must degrade to a flushed response, not a crash.\n if (!suppressOverlay && ctx?.defaultHandler) {\n await ctx.defaultHandler(error, event, { silent: false })\n }\n\n markH3ErrorHandled(event)\n if (suppressOverlay) {\n suppressNitroDevOverlay(error)\n }\n\n const evlogError = resolveEvlogError(error)\n\n const isDev = process.env.NODE_ENV === 'development'\n const url = getRequestURL(event, { xForwardedHost: true }).pathname\n\n if (!evlogError) {\n const body = buildPlainNitroErrorBody(error, url, isDev)\n setResponseStatus(event, body.status as number)\n setResponseHeader(event, 'Content-Type', 'application/json')\n return endNodeResponse(event, JSON.stringify(body))\n }\n\n const status = extractErrorStatus(evlogError)\n\n setResponseStatus(event, status)\n setResponseHeader(event, 'Content-Type', 'application/json')\n\n return endNodeResponse(event, JSON.stringify(serializeEvlogErrorResponse(evlogError, url)))\n})\n"],"mappings":";;;;;;;;;;;;;AAuBA,SAAS,gBAAgB,OAAgB,MAAoB;AAC3D,KAAI,CAAC,MAAM,KAAK,IAAI,cAClB,OAAM,KAAK,IAAI,IAAI,KAAK;;;;;;;;;;AAY5B,IAAA,uBAAe,wBAAwB,OAAO,OAAO,OAAO,QAAQ;CAClE,MAAM,kBAAkB,+BAA+B;AAIvD,KAAI,CAAC,mBAAmB,KAAK,eAC3B,OAAM,IAAI,eAAe,OAAO,OAAO,EAAE,QAAQ,OAAO,CAAC;AAG3D,oBAAmB,MAAM;AACzB,KAAI,gBACF,yBAAwB,MAAM;CAGhC,MAAM,aAAa,kBAAkB,MAAM;CAE3C,MAAM,QAAQ,QAAQ,IAAI,aAAa;CACvC,MAAM,MAAM,cAAc,OAAO,EAAE,gBAAgB,MAAM,CAAC,CAAC;AAE3D,KAAI,CAAC,YAAY;EACf,MAAM,OAAO,yBAAyB,OAAO,KAAK,MAAM;AACxD,oBAAkB,OAAO,KAAK,OAAiB;AAC/C,oBAAkB,OAAO,gBAAgB,mBAAmB;AAC5D,SAAO,gBAAgB,OAAO,KAAK,UAAU,KAAK,CAAC;;AAKrD,mBAAkB,OAFH,mBAAmB,WAEH,CAAC;AAChC,mBAAkB,OAAO,gBAAgB,mBAAmB;AAE5D,QAAO,gBAAgB,OAAO,KAAK,UAAU,4BAA4B,YAAY,IAAI,CAAC,CAAC;EAC3F"}
@@ -1,5 +1,5 @@
1
- import { t as useLogger } from "../useLogger-CqvH6qOf.mjs";
2
- import { t as NitroModuleOptions } from "../nitro-BRddgqSb.mjs";
1
+ import { t as useLogger } from "../useLogger-Cfv8Ck8b.mjs";
2
+ import { t as NitroModuleOptions } from "../nitro-_Hda8Deo.mjs";
3
3
  import { Nitro } from "nitropack";
4
4
 
5
5
  //#region src/nitro/module.d.ts
@@ -1 +1 @@
1
- {"version":3,"file":"module.d.mts","names":[],"sources":["../../src/nitro/module.ts"],"mappings":";;;;;iBAiBwB,KAAA,CAAM,OAAA,GAAU,kBAAA;;eAGvB,KAAA;AAAA"}
1
+ {"version":3,"file":"module.d.mts","names":[],"sources":["../../src/nitro/module.ts"],"mappings":";;;;;iBAkBwB,KAAA,CAAM,OAAA,GAAU,kBAAA;;eAGvB,KAAA;AAAA"}
@@ -1,4 +1,5 @@
1
1
  import { useLogger } from "../runtime/server/useLogger.mjs";
2
+ import { r as prependNitroErrorHandler } from "../nitro-ClRZLD1g.mjs";
2
3
  import { dirname, resolve } from "node:path";
3
4
  import { fileURLToPath } from "node:url";
4
5
  //#region src/nitro/module.ts
@@ -12,7 +13,7 @@ function evlog(options) {
12
13
  setup(nitro) {
13
14
  nitro.options.plugins = nitro.options.plugins || [];
14
15
  nitro.options.plugins.push(resolveModulePath("plugin"));
15
- if (!nitro.options.errorHandler) nitro.options.errorHandler = resolveModulePath("errorHandler");
16
+ nitro.options.errorHandler = prependNitroErrorHandler(nitro.options.errorHandler, resolveModulePath("errorHandler"));
16
17
  nitro.options.noExternals = true;
17
18
  nitro.options.runtimeConfig = nitro.options.runtimeConfig || {};
18
19
  nitro.options.runtimeConfig.evlog = options || {};
@@ -1 +1 @@
1
- {"version":3,"file":"module.mjs","names":[],"sources":["../../src/nitro/module.ts"],"sourcesContent":["import { dirname, resolve } from 'node:path'\nimport { fileURLToPath } from 'node:url'\nimport type { Nitro } from 'nitropack'\nimport type { NitroModuleOptions } from '../nitro'\n\nexport type { NitroModuleOptions }\n\nconst _dir = dirname(fileURLToPath(import.meta.url))\n\n// Nitro raw-interpolates these paths into JS string literals when generating\n// the #nitro/virtual/plugins and #nitro/virtual/error-handler modules, so\n// Windows backslashes would be parsed as escape sequences (\\n, \\v, …) and\n// break module resolution. Normalize to POSIX separators.\nfunction resolveModulePath(name: string): string {\n return resolve(_dir, name).replace(/\\\\/g, '/')\n}\n\nexport default function evlog(options?: NitroModuleOptions) {\n return {\n name: 'evlog',\n setup(nitro: Nitro) {\n // Push the plugin (no extension — Nitro's bundler resolves it)\n nitro.options.plugins = nitro.options.plugins || []\n nitro.options.plugins.push(resolveModulePath('plugin'))\n\n // Set error handler only if not already configured by user\n if (!nitro.options.errorHandler) {\n nitro.options.errorHandler = resolveModulePath('errorHandler')\n }\n\n // explicitly tell nitro to bundle evlog's files to correctly resolve nitro dependencies\n // in nitro v2 we can only disable externals globally\n\n nitro.options.noExternals = true\n\n // Inject config into runtimeConfig — works in production where the\n // plugin is bundled through Nitro's builder and the virtual\n // runtime-config module resolves correctly.\n nitro.options.runtimeConfig = nitro.options.runtimeConfig || {}\n nitro.options.runtimeConfig.evlog = options || {}\n\n // Bake the config into the bundle as a literal so the plugin never has\n // to do a runtime `import('nitropack/runtime/internal/config')` to\n // discover it. The dynamic probe transitively imports a build-only\n // virtual module; on Vercel + Bun the missing virtual triggers Bun's\n // auto-installer and crashes with `ReadOnlyFileSystem` (issue #312).\n nitro.options.replace = nitro.options.replace || {}\n nitro.options.replace.__EVLOG_CONFIG__ = JSON.stringify(options || {})\n\n // In dev mode, Nitro loads plugins externally (not bundled), so the\n // virtual runtime-config module is unreachable and useRuntimeConfig()\n // returns a stub without our values. process.env is inherited by the\n // Worker Threads that run the dev server, making it a reliable bridge.\n // The plugin reads: useRuntimeConfig().evlog ?? process.env.__EVLOG_CONFIG\n process.env.__EVLOG_CONFIG = JSON.stringify(options || {})\n },\n }\n}\n\nexport { useLogger } from '../runtime/server/useLogger'\n"],"mappings":";;;;AAOA,MAAM,OAAO,QAAQ,cAAc,OAAO,KAAK,IAAI,CAAC;AAMpD,SAAS,kBAAkB,MAAsB;AAC/C,QAAO,QAAQ,MAAM,KAAK,CAAC,QAAQ,OAAO,IAAI;;AAGhD,SAAwB,MAAM,SAA8B;AAC1D,QAAO;EACL,MAAM;EACN,MAAM,OAAc;AAElB,SAAM,QAAQ,UAAU,MAAM,QAAQ,WAAW,EAAE;AACnD,SAAM,QAAQ,QAAQ,KAAK,kBAAkB,SAAS,CAAC;AAGvD,OAAI,CAAC,MAAM,QAAQ,aACjB,OAAM,QAAQ,eAAe,kBAAkB,eAAe;AAMhE,SAAM,QAAQ,cAAc;AAK5B,SAAM,QAAQ,gBAAgB,MAAM,QAAQ,iBAAiB,EAAE;AAC/D,SAAM,QAAQ,cAAc,QAAQ,WAAW,EAAE;AAOjD,SAAM,QAAQ,UAAU,MAAM,QAAQ,WAAW,EAAE;AACnD,SAAM,QAAQ,QAAQ,mBAAmB,KAAK,UAAU,WAAW,EAAE,CAAC;AAOtE,WAAQ,IAAI,iBAAiB,KAAK,UAAU,WAAW,EAAE,CAAC;;EAE7D"}
1
+ {"version":3,"file":"module.mjs","names":[],"sources":["../../src/nitro/module.ts"],"sourcesContent":["import { dirname, resolve } from 'node:path'\nimport { fileURLToPath } from 'node:url'\nimport type { Nitro } from 'nitropack'\nimport type { NitroModuleOptions } from '../nitro'\nimport { prependNitroErrorHandler } from '../nitro'\n\nexport type { NitroModuleOptions }\n\nconst _dir = dirname(fileURLToPath(import.meta.url))\n\n// Nitro raw-interpolates these paths into JS string literals when generating\n// the #nitro/virtual/plugins and #nitro/virtual/error-handler modules, so\n// Windows backslashes would be parsed as escape sequences (\\n, \\v, …) and\n// break module resolution. Normalize to POSIX separators.\nfunction resolveModulePath(name: string): string {\n return resolve(_dir, name).replace(/\\\\/g, '/')\n}\n\nexport default function evlog(options?: NitroModuleOptions) {\n return {\n name: 'evlog',\n setup(nitro: Nitro) {\n // Push the plugin (no extension — Nitro's bundler resolves it)\n nitro.options.plugins = nitro.options.plugins || []\n nitro.options.plugins.push(resolveModulePath('plugin'))\n\n // Prepend so evlog runs before any framework handler (Nuxt registers its own).\n nitro.options.errorHandler = prependNitroErrorHandler(\n nitro.options.errorHandler,\n resolveModulePath('errorHandler'),\n )\n\n // explicitly tell nitro to bundle evlog's files to correctly resolve nitro dependencies\n // in nitro v2 we can only disable externals globally\n\n nitro.options.noExternals = true\n\n // Inject config into runtimeConfig — works in production where the\n // plugin is bundled through Nitro's builder and the virtual\n // runtime-config module resolves correctly.\n nitro.options.runtimeConfig = nitro.options.runtimeConfig || {}\n nitro.options.runtimeConfig.evlog = options || {}\n\n // Bake the config into the bundle as a literal so the plugin never has\n // to do a runtime `import('nitropack/runtime/internal/config')` to\n // discover it. The dynamic probe transitively imports a build-only\n // virtual module; on Vercel + Bun the missing virtual triggers Bun's\n // auto-installer and crashes with `ReadOnlyFileSystem` (issue #312).\n nitro.options.replace = nitro.options.replace || {}\n nitro.options.replace.__EVLOG_CONFIG__ = JSON.stringify(options || {})\n\n // In dev mode, Nitro loads plugins externally (not bundled), so the\n // virtual runtime-config module is unreachable and useRuntimeConfig()\n // returns a stub without our values. process.env is inherited by the\n // Worker Threads that run the dev server, making it a reliable bridge.\n // The plugin reads: useRuntimeConfig().evlog ?? process.env.__EVLOG_CONFIG\n process.env.__EVLOG_CONFIG = JSON.stringify(options || {})\n },\n }\n}\n\nexport { useLogger } from '../runtime/server/useLogger'\n"],"mappings":";;;;;AAQA,MAAM,OAAO,QAAQ,cAAc,OAAO,KAAK,IAAI,CAAC;AAMpD,SAAS,kBAAkB,MAAsB;AAC/C,QAAO,QAAQ,MAAM,KAAK,CAAC,QAAQ,OAAO,IAAI;;AAGhD,SAAwB,MAAM,SAA8B;AAC1D,QAAO;EACL,MAAM;EACN,MAAM,OAAc;AAElB,SAAM,QAAQ,UAAU,MAAM,QAAQ,WAAW,EAAE;AACnD,SAAM,QAAQ,QAAQ,KAAK,kBAAkB,SAAS,CAAC;AAGvD,SAAM,QAAQ,eAAe,yBAC3B,MAAM,QAAQ,cACd,kBAAkB,eAAe,CAClC;AAKD,SAAM,QAAQ,cAAc;AAK5B,SAAM,QAAQ,gBAAgB,MAAM,QAAQ,iBAAiB,EAAE;AAC/D,SAAM,QAAQ,cAAc,QAAQ,WAAW,EAAE;AAOjD,SAAM,QAAQ,UAAU,MAAM,QAAQ,WAAW,EAAE;AACnD,SAAM,QAAQ,QAAQ,mBAAmB,KAAK,UAAU,WAAW,EAAE,CAAC;AAOtE,WAAQ,IAAI,iBAAiB,KAAK,UAAU,WAAW,EAAE,CAAC;;EAE7D"}
@@ -1,13 +1,17 @@
1
- import { b as initLogger, g as createRequestLogger, j as normalizeRedactConfig, x as isEnabled, y as getGlobalPluginRunner } from "../audit-BUI3af4w.mjs";
1
+ import { n as readCodeSnippetFromDisk } from "../pretty-error-snippet.node-itfCajBM.mjs";
2
2
  import { filterSafeHeaders } from "../utils.mjs";
3
- import { t as extractErrorStatus } from "../errors-BQgyQ9xe.mjs";
4
- import { n as shouldLog, t as getServiceForPath } from "../routes-CnIgYWf8.mjs";
5
- import { n as resolveEvlogConfigForNitroPlugin, r as setActiveNitroRuntime } from "../nitroConfigBridge-NbFn-sIK.mjs";
3
+ import { N as normalizeRedactConfig, b as initLogger, g as createRequestLogger, x as isEnabled, y as getGlobalPluginRunner } from "../audit-BQt8yAHo.mjs";
4
+ import { o as registerPrettyErrorSnippetReader } from "../pretty-error-THg0U0w9.mjs";
5
+ import { t as extractErrorStatus } from "../errors-DA0cyr8q.mjs";
6
+ import { i as setActiveNitroRuntime, r as resolveEvlogConfigForNitroPlugin } from "../nitroConfigBridge-BkVWnSV3.mjs";
7
+ import { n as shouldLog, t as getServiceForPath } from "../routes-4rMzRyTk.mjs";
8
+ import { n as enrichErrorStackForDev, t as extendDeferredDrain } from "../deferred-drain-jeajC8QF.mjs";
9
+ import { r as shouldDeferEmitForResponse, t as bindStreamingResponseLifecycle } from "../streamResponse-CmQ3qUbF.mjs";
6
10
  import { startStreamServer } from "../stream.mjs";
7
11
  import { defineNitroPlugin } from "nitropack/runtime/internal/plugin";
8
12
  import { getHeaders } from "h3";
9
- //#region src/nitro/plugin.ts
10
- function getSafeHeaders(event) {
13
+ //#region src/nitro/enrich-drain.ts
14
+ function getSafeHeaders$1(event) {
11
15
  return filterSafeHeaders(getHeaders(event));
12
16
  }
13
17
  function getSafeResponseHeaders(event) {
@@ -23,7 +27,7 @@ function getSafeResponseHeaders(event) {
23
27
  if (Object.keys(headers).length === 0) return void 0;
24
28
  return filterSafeHeaders(headers);
25
29
  }
26
- function getResponseStatus(event) {
30
+ function getResponseStatus$1(event) {
27
31
  if (event.node?.res?.statusCode) return event.node.res.statusCode;
28
32
  if (event.response?.status) return event.response.status;
29
33
  if (typeof event.context.status === "number") return event.context.status;
@@ -36,14 +40,23 @@ function buildHookContext(event) {
36
40
  method: event.method,
37
41
  path: event.path
38
42
  },
39
- headers: getSafeHeaders(event),
43
+ headers: getSafeHeaders$1(event),
40
44
  response: {
41
- status: getResponseStatus(event),
45
+ status: getResponseStatus$1(event),
42
46
  headers: responseHeaders
43
47
  }
44
48
  };
45
49
  }
46
- async function callEnrichAndDrain(nitroApp, emittedEvent, event) {
50
+ function resolveDeferredWaitUntil(event) {
51
+ if (globalThis.navigator?.userAgent !== "Cloudflare-Workers") return void 0;
52
+ const waitUntilCtx = event.context.cloudflare?.context ?? event.context;
53
+ if (typeof waitUntilCtx?.waitUntil === "function") return waitUntilCtx.waitUntil.bind(waitUntilCtx);
54
+ }
55
+ /**
56
+ * Run evlog enrich + drain hooks for an emitted wide event.
57
+ * @internal Exported for Nitro plugin tests.
58
+ */
59
+ async function callEnrichAndDrain(nitroApp, emittedEvent, event, options) {
47
60
  if (!emittedEvent) return;
48
61
  const hookContext = buildHookContext(event);
49
62
  const enrichCtx = {
@@ -56,7 +69,11 @@ async function callEnrichAndDrain(nitroApp, emittedEvent, event) {
56
69
  } catch (err) {
57
70
  console.error("[evlog] enrich failed:", err);
58
71
  }
59
- if (runner.hasEnrich) await runner.runEnrich(enrichCtx);
72
+ if (runner.hasEnrich) try {
73
+ await runner.runEnrich(enrichCtx);
74
+ } catch (err) {
75
+ console.error("[evlog] enrich failed:", err);
76
+ }
60
77
  const drainCtx = {
61
78
  event: emittedEvent,
62
79
  request: hookContext.request,
@@ -65,20 +82,39 @@ async function callEnrichAndDrain(nitroApp, emittedEvent, event) {
65
82
  const drainTasks = [nitroApp.hooks.callHook("evlog:drain", drainCtx).catch((err) => {
66
83
  console.error("[evlog] drain failed:", err);
67
84
  })];
68
- if (runner.hasDrain) drainTasks.push(runner.runDrain(drainCtx));
85
+ if (runner.hasDrain) drainTasks.push(runner.runDrain(drainCtx).catch((err) => {
86
+ console.error("[evlog] drain failed:", err);
87
+ }));
69
88
  const drainPromise = Promise.all(drainTasks);
89
+ if (options?.deferDrain) {
90
+ extendDeferredDrain(drainPromise, resolveDeferredWaitUntil(event));
91
+ return;
92
+ }
70
93
  const waitUntilCtx = event.context.cloudflare?.context ?? event.context;
71
94
  if (typeof waitUntilCtx?.waitUntil === "function") waitUntilCtx.waitUntil(drainPromise);
72
95
  else await drainPromise;
73
96
  }
97
+ //#endregion
98
+ //#region src/nitro/plugin.ts
99
+ function getSafeHeaders(event) {
100
+ return filterSafeHeaders(getHeaders(event));
101
+ }
102
+ function getResponseStatus(event) {
103
+ if (event.node?.res?.statusCode) return event.node.res.statusCode;
104
+ if (event.response?.status) return event.response.status;
105
+ if (typeof event.context.status === "number") return event.context.status;
106
+ return 200;
107
+ }
74
108
  var plugin_default = defineNitroPlugin(async (nitroApp) => {
75
109
  setActiveNitroRuntime("v2");
76
110
  const evlogConfig = await resolveEvlogConfigForNitroPlugin();
77
111
  const redact = normalizeRedactConfig(evlogConfig?.redact);
112
+ registerPrettyErrorSnippetReader(readCodeSnippetFromDisk);
78
113
  initLogger({
79
114
  enabled: evlogConfig?.enabled,
80
115
  env: evlogConfig?.env,
81
116
  pretty: evlogConfig?.pretty,
117
+ dev: evlogConfig?.dev,
82
118
  silent: evlogConfig?.silent,
83
119
  sampling: evlogConfig?.sampling,
84
120
  minLevel: evlogConfig?.minLevel,
@@ -132,8 +168,12 @@ var plugin_default = defineNitroPlugin(async (nitroApp) => {
132
168
  if (!e) return;
133
169
  if (!e.context._evlogShouldEmit) return;
134
170
  const requestLog = e.context.log;
135
- if (requestLog) {
136
- requestLog.error(error);
171
+ if (!requestLog) return;
172
+ e.context._evlogEmitting = true;
173
+ try {
174
+ const err = error;
175
+ enrichErrorStackForDev(err, { pretty: evlogConfig?.pretty });
176
+ requestLog.error(err);
137
177
  const errorStatus = extractErrorStatus(error);
138
178
  requestLog.set({ status: errorStatus });
139
179
  const startTime = e.context._evlogStartTime;
@@ -148,15 +188,23 @@ var plugin_default = defineNitroPlugin(async (nitroApp) => {
148
188
  await nitroApp.hooks.callHook("evlog:emit:keep", tailCtx);
149
189
  const runner = getGlobalPluginRunner();
150
190
  if (runner.hasKeep) await runner.runKeep(tailCtx);
151
- e.context._evlogEmitted = true;
152
- await callEnrichAndDrain(nitroApp, requestLog.emit({ _forceKeep: tailCtx.shouldKeep }), e);
191
+ const emittedEvent = requestLog.emit({ _forceKeep: tailCtx.shouldKeep });
192
+ if (emittedEvent) {
193
+ e.context._evlogEmitted = true;
194
+ callEnrichAndDrain(nitroApp, emittedEvent, e, { deferDrain: true }).catch((err) => {
195
+ console.error("[evlog] background enrich/drain failed:", err);
196
+ });
197
+ }
198
+ } finally {
199
+ delete e.context._evlogEmitting;
153
200
  }
154
201
  });
155
202
  nitroApp.hooks.hook("afterResponse", async (event) => {
156
203
  const e = event;
157
- if (e.context._evlogEmitted || !e.context._evlogShouldEmit) return;
204
+ if (e.context._evlogEmitted || e.context._evlogEmitting || !e.context._evlogShouldEmit) return;
158
205
  const requestLog = e.context.log;
159
- if (requestLog) {
206
+ if (!requestLog) return;
207
+ const emitSuccessResponse = async () => {
160
208
  const status = getResponseStatus(e);
161
209
  requestLog.set({ status });
162
210
  const startTime = e.context._evlogStartTime;
@@ -172,7 +220,15 @@ var plugin_default = defineNitroPlugin(async (nitroApp) => {
172
220
  const runner = getGlobalPluginRunner();
173
221
  if (runner.hasKeep) await runner.runKeep(tailCtx);
174
222
  await callEnrichAndDrain(nitroApp, requestLog.emit({ _forceKeep: tailCtx.shouldKeep }), e);
223
+ };
224
+ if (e.response && shouldDeferEmitForResponse(e.response)) {
225
+ e.response = bindStreamingResponseLifecycle(e.response, async (meta) => {
226
+ if (meta.error) requestLog.error(meta.error);
227
+ await emitSuccessResponse();
228
+ });
229
+ return;
175
230
  }
231
+ await emitSuccessResponse();
176
232
  });
177
233
  });
178
234
  //#endregion
@@ -1 +1 @@
1
- {"version":3,"file":"plugin.mjs","names":[],"sources":["../../src/nitro/plugin.ts"],"sourcesContent":["import type { NitroApp } from 'nitropack/types'\n// Import from specific subpaths to avoid the barrel 'nitropack/runtime' which\n// re-exports from internal/app.mjs — that file imports #nitro-internal-virtual/*\n// modules that only exist inside rollup builds and crash when loaded externally\n// (nitropack dev loads plugins outside the bundle via Worker threads).\nimport { defineNitroPlugin } from 'nitropack/runtime/internal/plugin'\nimport { getHeaders } from 'h3'\nimport { createRequestLogger, getGlobalPluginRunner, initLogger, isEnabled } from '../logger'\nimport { shouldLog, getServiceForPath, extractErrorStatus } from '../nitro'\nimport { normalizeRedactConfig } from '../redact'\nimport { resolveEvlogConfigForNitroPlugin, setActiveNitroRuntime } from '../shared/nitroConfigBridge'\nimport { startStreamServer, type StreamServerOptions } from '../stream'\nimport type { EnrichContext, RequestLogger, ServerEvent, TailSamplingContext, WideEvent } from '../types'\nimport { filterSafeHeaders } from '../utils'\n\nfunction getSafeHeaders(event: ServerEvent): Record<string, string> {\n const allHeaders = getHeaders(event as Parameters<typeof getHeaders>[0])\n return filterSafeHeaders(allHeaders)\n}\n\nfunction getSafeResponseHeaders(event: ServerEvent): Record<string, string> | undefined {\n const headers: Record<string, string> = {}\n const nodeRes = event.node?.res as { getHeaders?: () => Record<string, unknown> } | undefined\n\n if (nodeRes?.getHeaders) {\n for (const [key, value] of Object.entries(nodeRes.getHeaders())) {\n if (value === undefined) continue\n headers[key] = Array.isArray(value) ? value.join(', ') : String(value)\n }\n }\n\n if (event.response?.headers) {\n event.response.headers.forEach((value, key) => {\n headers[key] = value\n })\n }\n\n if (Object.keys(headers).length === 0) return undefined\n return filterSafeHeaders(headers)\n}\n\nfunction getResponseStatus(event: ServerEvent): number {\n // Node.js style\n if (event.node?.res?.statusCode) {\n return event.node.res.statusCode\n }\n\n // Web Standard\n if (event.response?.status) {\n return event.response.status\n }\n\n // Context-based\n if (typeof event.context.status === 'number') {\n return event.context.status\n }\n\n return 200\n}\n\nfunction buildHookContext(event: ServerEvent): Omit<EnrichContext, 'event'> {\n const responseHeaders = getSafeResponseHeaders(event)\n return {\n request: { method: event.method, path: event.path },\n headers: getSafeHeaders(event),\n response: {\n status: getResponseStatus(event),\n headers: responseHeaders,\n },\n }\n}\n\nasync function callEnrichAndDrain(\n nitroApp: NitroApp,\n emittedEvent: WideEvent | null,\n event: ServerEvent,\n): Promise<void> {\n if (!emittedEvent) return\n\n const hookContext = buildHookContext(event)\n const enrichCtx: EnrichContext = { event: emittedEvent, ...hookContext }\n const runner = getGlobalPluginRunner()\n\n try {\n await nitroApp.hooks.callHook('evlog:enrich', enrichCtx)\n } catch (err) {\n console.error('[evlog] enrich failed:', err)\n }\n if (runner.hasEnrich) {\n await runner.runEnrich(enrichCtx)\n }\n\n const drainCtx = {\n event: emittedEvent,\n request: hookContext.request,\n headers: hookContext.headers,\n }\n const drainTasks: Array<Promise<unknown>> = [\n nitroApp.hooks.callHook('evlog:drain', drainCtx).catch((err) => {\n console.error('[evlog] drain failed:', err)\n }),\n ]\n if (runner.hasDrain) {\n drainTasks.push(runner.runDrain(drainCtx))\n }\n const drainPromise = Promise.all(drainTasks)\n\n // Use waitUntil if available (Cloudflare Workers, Vercel Edge, etc.)\n // This keeps the runtime alive for background work without blocking the response\n const waitUntilCtx = event.context.cloudflare?.context ?? event.context\n if (typeof waitUntilCtx?.waitUntil === 'function') {\n waitUntilCtx.waitUntil(drainPromise)\n } else {\n // Fallback: await drain to prevent lost logs in serverless environments\n // (e.g. Vercel Fluid Compute). On the normal path this runs from\n // afterResponse (response already sent); on the error path it may run\n // before the error response is finalized.\n await drainPromise\n }\n}\n\nexport default defineNitroPlugin(async (nitroApp) => {\n setActiveNitroRuntime('v2')\n const evlogConfig = await resolveEvlogConfigForNitroPlugin()\n\n const redact = normalizeRedactConfig(evlogConfig?.redact as boolean | Record<string, unknown> | undefined)\n\n initLogger({\n enabled: evlogConfig?.enabled,\n env: evlogConfig?.env,\n pretty: evlogConfig?.pretty,\n silent: evlogConfig?.silent,\n sampling: evlogConfig?.sampling,\n minLevel: evlogConfig?.minLevel,\n redact,\n _suppressDrainWarning: true,\n })\n\n // When `evlog.stream` is set (or auto-on in dev), boot the mini stream\n // server and hook every drained event into it. The server runs on its\n // own ephemeral port — the user's API surface is untouched.\n const streamSetting = (evlogConfig as { stream?: boolean | StreamServerOptions } | undefined)?.stream\n if (streamSetting === true || (streamSetting && typeof streamSetting === 'object')) {\n const streamOpts: StreamServerOptions = streamSetting === true ? {} : streamSetting\n startStreamServer(streamOpts).then((server) => {\n nitroApp.hooks.hook('evlog:drain', (ctx) => {\n if (ctx?.event) server.drain(ctx)\n })\n }).catch((err) => {\n console.error('[evlog] failed to start stream server:', err)\n })\n }\n\n // When globally disabled, createRequestLogger returns a no-op logger — still\n // attach it so handlers can call useLogger(event) without throwing.\n if (!isEnabled()) {\n nitroApp.hooks.hook('request', (event) => {\n const e = event as ServerEvent\n let requestIdOverride: string | undefined\n if (globalThis.navigator?.userAgent === 'Cloudflare-Workers') {\n const cfRay = getSafeHeaders(e)?.['cf-ray']\n if (cfRay) requestIdOverride = cfRay\n }\n e.context.log = createRequestLogger({\n method: e.method,\n path: e.path,\n requestId: requestIdOverride || e.context.requestId || crypto.randomUUID(),\n }, { _deferDrain: true })\n })\n return\n }\n\n nitroApp.hooks.hook('request', (event) => {\n const e = event as ServerEvent\n\n // Evaluate route filtering but always create the logger so that server\n // middleware (which runs for every request) can call useLogger(event)\n // without throwing. Filtering is enforced at emit time instead.\n e.context._evlogShouldEmit = shouldLog(e.path, evlogConfig?.include, evlogConfig?.exclude)\n\n // Store start time for duration calculation in tail sampling\n e.context._evlogStartTime = Date.now()\n\n let requestIdOverride: string | undefined = undefined\n if (globalThis.navigator?.userAgent === 'Cloudflare-Workers') {\n const cfRay = getSafeHeaders(e)?.['cf-ray']\n if (cfRay) requestIdOverride = cfRay\n }\n\n const requestLog = createRequestLogger({\n method: e.method,\n path: e.path,\n requestId: requestIdOverride || e.context.requestId || crypto.randomUUID(),\n }, { _deferDrain: true })\n\n // Apply route-based service configuration if a matching route is found\n const routeService = getServiceForPath(e.path, evlogConfig?.routes)\n if (routeService) {\n requestLog.set({ service: routeService })\n }\n\n e.context.log = requestLog\n })\n\n nitroApp.hooks.hook('error', async (error, { event }) => {\n const e = event as ServerEvent | undefined\n if (!e) return\n if (!e.context._evlogShouldEmit) return\n\n const requestLog = e.context.log as RequestLogger | undefined\n if (requestLog) {\n requestLog.error(error as Error)\n\n const errorStatus = extractErrorStatus(error)\n requestLog.set({ status: errorStatus })\n\n // Build tail sampling context\n const startTime = e.context._evlogStartTime as number | undefined\n const durationMs = startTime ? Date.now() - startTime : undefined\n\n const tailCtx: TailSamplingContext = {\n status: errorStatus,\n duration: durationMs,\n path: e.path,\n method: e.method,\n context: requestLog.getContext(),\n shouldKeep: false,\n }\n\n // Call evlog:emit:keep hook + plugin runner keep hook\n await nitroApp.hooks.callHook('evlog:emit:keep', tailCtx)\n const runner = getGlobalPluginRunner()\n if (runner.hasKeep) await runner.runKeep(tailCtx)\n\n e.context._evlogEmitted = true\n\n const emittedEvent = requestLog.emit({ _forceKeep: tailCtx.shouldKeep })\n await callEnrichAndDrain(nitroApp, emittedEvent, e)\n }\n })\n\n nitroApp.hooks.hook('afterResponse', async (event) => {\n const e = event as ServerEvent\n // Skip if already emitted by error hook or route was filtered out\n if (e.context._evlogEmitted || !e.context._evlogShouldEmit) return\n\n const requestLog = e.context.log as RequestLogger | undefined\n if (requestLog) {\n const status = getResponseStatus(e)\n requestLog.set({ status })\n\n const startTime = e.context._evlogStartTime as number | undefined\n const durationMs = startTime ? Date.now() - startTime : undefined\n\n const tailCtx: TailSamplingContext = {\n status,\n duration: durationMs,\n path: e.path,\n method: e.method,\n context: requestLog.getContext(),\n shouldKeep: false,\n }\n\n await nitroApp.hooks.callHook('evlog:emit:keep', tailCtx)\n const runner = getGlobalPluginRunner()\n if (runner.hasKeep) await runner.runKeep(tailCtx)\n\n const emittedEvent = requestLog.emit({ _forceKeep: tailCtx.shouldKeep })\n await callEnrichAndDrain(nitroApp, emittedEvent, e)\n }\n })\n})\n"],"mappings":"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"}
1
+ {"version":3,"file":"plugin.mjs","names":["getSafeHeaders","getResponseStatus"],"sources":["../../src/nitro/enrich-drain.ts","../../src/nitro/plugin.ts"],"sourcesContent":["import type { NitroApp } from 'nitropack/types'\nimport { getHeaders } from 'h3'\nimport { getGlobalPluginRunner } from '../logger'\nimport type { EnrichContext, ServerEvent, WideEvent } from '../types'\nimport { filterSafeHeaders } from '../utils'\nimport { extendDeferredDrain } from './deferred-drain'\n\nfunction getSafeHeaders(event: ServerEvent): Record<string, string> {\n const allHeaders = getHeaders(event as Parameters<typeof getHeaders>[0])\n return filterSafeHeaders(allHeaders)\n}\n\nfunction getSafeResponseHeaders(event: ServerEvent): Record<string, string> | undefined {\n const headers: Record<string, string> = {}\n const nodeRes = event.node?.res as { getHeaders?: () => Record<string, unknown> } | undefined\n\n if (nodeRes?.getHeaders) {\n for (const [key, value] of Object.entries(nodeRes.getHeaders())) {\n if (value === undefined) continue\n headers[key] = Array.isArray(value) ? value.join(', ') : String(value)\n }\n }\n\n if (event.response?.headers) {\n event.response.headers.forEach((value, key) => {\n headers[key] = value\n })\n }\n\n if (Object.keys(headers).length === 0) return undefined\n return filterSafeHeaders(headers)\n}\n\nfunction getResponseStatus(event: ServerEvent): number {\n if (event.node?.res?.statusCode) {\n return event.node.res.statusCode\n }\n if (event.response?.status) {\n return event.response.status\n }\n if (typeof event.context.status === 'number') {\n return event.context.status\n }\n return 200\n}\n\nfunction buildHookContext(event: ServerEvent): Omit<EnrichContext, 'event'> {\n const responseHeaders = getSafeResponseHeaders(event)\n return {\n request: { method: event.method, path: event.path },\n headers: getSafeHeaders(event),\n response: {\n status: getResponseStatus(event),\n headers: responseHeaders,\n },\n }\n}\n\nfunction resolveDeferredWaitUntil(event: ServerEvent): ((promise: Promise<unknown>) => void) | undefined {\n if (globalThis.navigator?.userAgent !== 'Cloudflare-Workers') return undefined\n const waitUntilCtx = event.context.cloudflare?.context ?? event.context\n if (typeof waitUntilCtx?.waitUntil === 'function') {\n return waitUntilCtx.waitUntil.bind(waitUntilCtx)\n }\n return undefined\n}\n\n/**\n * Run evlog enrich + drain hooks for an emitted wide event.\n * @internal Exported for Nitro plugin tests.\n */\nexport async function callEnrichAndDrain(\n nitroApp: NitroApp,\n emittedEvent: WideEvent | null,\n event: ServerEvent,\n options?: { deferDrain?: boolean },\n): Promise<void> {\n if (!emittedEvent) return\n\n const hookContext = buildHookContext(event)\n const enrichCtx: EnrichContext = { event: emittedEvent, ...hookContext }\n const runner = getGlobalPluginRunner()\n\n try {\n await nitroApp.hooks.callHook('evlog:enrich', enrichCtx)\n } catch (err) {\n console.error('[evlog] enrich failed:', err)\n }\n if (runner.hasEnrich) {\n try {\n await runner.runEnrich(enrichCtx)\n } catch (err) {\n console.error('[evlog] enrich failed:', err)\n }\n }\n\n const drainCtx = {\n event: emittedEvent,\n request: hookContext.request,\n headers: hookContext.headers,\n }\n const drainTasks: Array<Promise<unknown>> = [\n nitroApp.hooks.callHook('evlog:drain', drainCtx).catch((err) => {\n console.error('[evlog] drain failed:', err)\n }),\n ]\n if (runner.hasDrain) {\n drainTasks.push(\n runner.runDrain(drainCtx).catch((err) => {\n console.error('[evlog] drain failed:', err)\n }),\n )\n }\n const drainPromise = Promise.all(drainTasks)\n\n // deferDrain: never block the HTTP error response on Nitro Node (h3 2.13+ waitUntil\n // queues work before send). On Cloudflare, register waitUntil so drains survive.\n if (options?.deferDrain) {\n extendDeferredDrain(drainPromise, resolveDeferredWaitUntil(event))\n return\n }\n\n const waitUntilCtx = event.context.cloudflare?.context ?? event.context\n if (typeof waitUntilCtx?.waitUntil === 'function') {\n waitUntilCtx.waitUntil(drainPromise)\n } else {\n await drainPromise\n }\n}\n","// Import from specific subpaths to avoid the barrel 'nitropack/runtime' which\n// re-exports from internal/app.mjs — that file imports #nitro-internal-virtual/*\n// modules that only exist inside rollup builds and crash when loaded externally\n// (nitropack dev loads plugins outside the bundle via Worker threads).\nimport { defineNitroPlugin } from 'nitropack/runtime/internal/plugin'\nimport { getHeaders } from 'h3'\nimport { createRequestLogger, getGlobalPluginRunner, initLogger, isEnabled, markWideEventDrainStarted } from '../logger'\nimport { registerPrettyErrorSnippetReader } from '../shared/pretty-error'\nimport { readCodeSnippetFromDisk } from '../shared/pretty-error-snippet.node'\nimport { enrichErrorStackForDev } from '../shared/enrich-error-stack.node'\nimport { shouldLog, getServiceForPath, extractErrorStatus } from '../nitro'\nimport { normalizeRedactConfig } from '../redact'\nimport { resolveEvlogConfigForNitroPlugin, setActiveNitroRuntime } from '../shared/nitroConfigBridge'\nimport { bindStreamingResponseLifecycle, shouldDeferEmitForResponse } from '../shared/streamResponse'\nimport { startStreamServer, type StreamServerOptions } from '../stream'\nimport type { RequestLogger, ServerEvent, TailSamplingContext } from '../types'\nimport { filterSafeHeaders } from '../utils'\nimport { callEnrichAndDrain } from './enrich-drain'\n\nfunction getSafeHeaders(event: ServerEvent): Record<string, string> {\n const allHeaders = getHeaders(event as Parameters<typeof getHeaders>[0])\n return filterSafeHeaders(allHeaders)\n}\n\nfunction getResponseStatus(event: ServerEvent): number {\n // Node.js style\n if (event.node?.res?.statusCode) {\n return event.node.res.statusCode\n }\n\n // Web Standard\n if (event.response?.status) {\n return event.response.status\n }\n\n // Context-based\n if (typeof event.context.status === 'number') {\n return event.context.status\n }\n\n return 200\n}\n\n\nexport default defineNitroPlugin(async (nitroApp) => {\n setActiveNitroRuntime('v2')\n const evlogConfig = await resolveEvlogConfigForNitroPlugin()\n\n const redact = normalizeRedactConfig(evlogConfig?.redact as boolean | Record<string, unknown> | undefined)\n\n registerPrettyErrorSnippetReader(readCodeSnippetFromDisk)\n\n initLogger({\n enabled: evlogConfig?.enabled,\n env: evlogConfig?.env,\n pretty: evlogConfig?.pretty,\n dev: evlogConfig?.dev,\n silent: evlogConfig?.silent,\n sampling: evlogConfig?.sampling,\n minLevel: evlogConfig?.minLevel,\n redact,\n _suppressDrainWarning: true,\n })\n\n // When `evlog.stream` is set (or auto-on in dev), boot the mini stream\n // server and hook every drained event into it. The server runs on its\n // own ephemeral port — the user's API surface is untouched.\n const streamSetting = (evlogConfig as { stream?: boolean | StreamServerOptions } | undefined)?.stream\n if (streamSetting === true || (streamSetting && typeof streamSetting === 'object')) {\n const streamOpts: StreamServerOptions = streamSetting === true ? {} : streamSetting\n startStreamServer(streamOpts).then((server) => {\n nitroApp.hooks.hook('evlog:drain', (ctx) => {\n if (ctx?.event) server.drain(ctx)\n })\n }).catch((err) => {\n console.error('[evlog] failed to start stream server:', err)\n })\n }\n\n // When globally disabled, createRequestLogger returns a no-op logger — still\n // attach it so handlers can call useLogger(event) without throwing.\n if (!isEnabled()) {\n nitroApp.hooks.hook('request', (event) => {\n const e = event as ServerEvent\n let requestIdOverride: string | undefined\n if (globalThis.navigator?.userAgent === 'Cloudflare-Workers') {\n const cfRay = getSafeHeaders(e)?.['cf-ray']\n if (cfRay) requestIdOverride = cfRay\n }\n e.context.log = createRequestLogger({\n method: e.method,\n path: e.path,\n requestId: requestIdOverride || e.context.requestId || crypto.randomUUID(),\n }, { _deferDrain: true })\n })\n return\n }\n\n nitroApp.hooks.hook('request', (event) => {\n const e = event as ServerEvent\n\n // Evaluate route filtering but always create the logger so that server\n // middleware (which runs for every request) can call useLogger(event)\n // without throwing. Filtering is enforced at emit time instead.\n e.context._evlogShouldEmit = shouldLog(e.path, evlogConfig?.include, evlogConfig?.exclude)\n\n // Store start time for duration calculation in tail sampling\n e.context._evlogStartTime = Date.now()\n\n let requestIdOverride: string | undefined = undefined\n if (globalThis.navigator?.userAgent === 'Cloudflare-Workers') {\n const cfRay = getSafeHeaders(e)?.['cf-ray']\n if (cfRay) requestIdOverride = cfRay\n }\n\n const requestLog = createRequestLogger({\n method: e.method,\n path: e.path,\n requestId: requestIdOverride || e.context.requestId || crypto.randomUUID(),\n }, { _deferDrain: true })\n\n // Apply route-based service configuration if a matching route is found\n const routeService = getServiceForPath(e.path, evlogConfig?.routes)\n if (routeService) {\n requestLog.set({ service: routeService })\n }\n\n e.context.log = requestLog\n })\n\n nitroApp.hooks.hook('error', async (error, { event }) => {\n const e = event as ServerEvent | undefined\n if (!e) return\n if (!e.context._evlogShouldEmit) return\n\n const requestLog = e.context.log as RequestLogger | undefined\n if (!requestLog) return\n\n e.context._evlogEmitting = true\n try {\n const err = error as Error\n void enrichErrorStackForDev(err, { pretty: evlogConfig?.pretty })\n requestLog.error(err)\n\n const errorStatus = extractErrorStatus(error)\n requestLog.set({ status: errorStatus })\n\n const startTime = e.context._evlogStartTime as number | undefined\n const durationMs = startTime ? Date.now() - startTime : undefined\n\n const tailCtx: TailSamplingContext = {\n status: errorStatus,\n duration: durationMs,\n path: e.path,\n method: e.method,\n context: requestLog.getContext(),\n shouldKeep: false,\n }\n\n await nitroApp.hooks.callHook('evlog:emit:keep', tailCtx)\n const runner = getGlobalPluginRunner()\n if (runner.hasKeep) await runner.runKeep(tailCtx)\n\n const emittedEvent = requestLog.emit({ _forceKeep: tailCtx.shouldKeep })\n if (emittedEvent) {\n e.context._evlogEmitted = true\n void callEnrichAndDrain(nitroApp, emittedEvent, e, { deferDrain: true }).catch((err) => {\n console.error('[evlog] background enrich/drain failed:', err)\n })\n }\n } finally {\n delete e.context._evlogEmitting\n }\n })\n\n nitroApp.hooks.hook('afterResponse', async (event) => {\n const e = event as ServerEvent\n if (e.context._evlogEmitted || e.context._evlogEmitting || !e.context._evlogShouldEmit) return\n\n const requestLog = e.context.log as RequestLogger | undefined\n if (!requestLog) return\n\n const emitSuccessResponse = async () => {\n const status = getResponseStatus(e)\n requestLog.set({ status })\n\n const startTime = e.context._evlogStartTime as number | undefined\n const durationMs = startTime ? Date.now() - startTime : undefined\n\n const tailCtx: TailSamplingContext = {\n status,\n duration: durationMs,\n path: e.path,\n method: e.method,\n context: requestLog.getContext(),\n shouldKeep: false,\n }\n\n await nitroApp.hooks.callHook('evlog:emit:keep', tailCtx)\n const runner = getGlobalPluginRunner()\n if (runner.hasKeep) await runner.runKeep(tailCtx)\n\n const emittedEvent = requestLog.emit({ _forceKeep: tailCtx.shouldKeep })\n await callEnrichAndDrain(nitroApp, emittedEvent, e)\n }\n\n if (e.response && shouldDeferEmitForResponse(e.response)) {\n e.response = bindStreamingResponseLifecycle(e.response, async (meta) => {\n if (meta.error) {\n requestLog.error(meta.error)\n }\n await emitSuccessResponse()\n })\n return\n }\n\n await emitSuccessResponse()\n })\n})\n"],"mappings":"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"}
@@ -6,16 +6,9 @@ import * as _$nitro_types0 from "nitro/types";
6
6
  * This ensures that 'data' (containing 'why', 'fix', 'link') is preserved
7
7
  * in the JSON response regardless of the underlying HTTP framework.
8
8
  *
9
- * For non-EvlogError, returns undefined to let Nitro's default handler take over.
10
- *
11
9
  * Usage in nitro.config.ts:
12
10
  * ```ts
13
- * // errorHandler.ts
14
11
  * export { default } from 'evlog/nitro/v3/errorHandler'
15
- * // nitro.config.ts
16
- * export default defineConfig({
17
- * errorHandler: './errorHandler',
18
- * })
19
12
  * ```
20
13
  */
21
14
  declare const _default: _$nitro_types0.NitroErrorHandler;
@@ -1,6 +1,6 @@
1
- import { t as extractErrorStatus } from "../../errors-BQgyQ9xe.mjs";
2
- import { n as serializeEvlogErrorResponse, t as resolveEvlogError } from "../../nitro-DErMq_Zj.mjs";
3
- import { t as parseURL } from "../../dist-H3GIh-KK.mjs";
1
+ import { t as extractErrorStatus } from "../../errors-DA0cyr8q.mjs";
2
+ import { a as serializeEvlogErrorResponse, i as resolveEvlogError, n as markH3ErrorHandled, o as shouldSuppressNitroDevOverlay, s as suppressNitroDevOverlay, t as buildPlainNitroErrorBody } from "../../nitro-ClRZLD1g.mjs";
3
+ import { t as parseURL } from "../../dist-DdQWiZn8.mjs";
4
4
  import { defineErrorHandler } from "nitro";
5
5
  //#region src/nitro-v3/errorHandler.ts
6
6
  /**
@@ -8,24 +8,22 @@ import { defineErrorHandler } from "nitro";
8
8
  * This ensures that 'data' (containing 'why', 'fix', 'link') is preserved
9
9
  * in the JSON response regardless of the underlying HTTP framework.
10
10
  *
11
- * For non-EvlogError, returns undefined to let Nitro's default handler take over.
12
- *
13
11
  * Usage in nitro.config.ts:
14
12
  * ```ts
15
- * // errorHandler.ts
16
13
  * export { default } from 'evlog/nitro/v3/errorHandler'
17
- * // nitro.config.ts
18
- * export default defineConfig({
19
- * errorHandler: './errorHandler',
20
- * })
21
14
  * ```
22
15
  */
23
- var errorHandler_default = defineErrorHandler((error, event) => {
24
- const evlogError = resolveEvlogError(error);
25
- if (!evlogError) return;
16
+ var errorHandler_default = defineErrorHandler(async (error, event, ctx) => {
17
+ const suppressOverlay = shouldSuppressNitroDevOverlay();
18
+ if (!suppressOverlay) await ctx.defaultHandler(error, event, { silent: false });
19
+ markH3ErrorHandled(event);
20
+ if (suppressOverlay) suppressNitroDevOverlay(error);
26
21
  const url = parseURL(event.req.url).pathname;
27
- const status = extractErrorStatus(evlogError);
28
- return new Response(JSON.stringify(serializeEvlogErrorResponse(evlogError, url)), {
22
+ const isDev = process.env.NODE_ENV === "development";
23
+ const evlogError = resolveEvlogError(error);
24
+ const body = evlogError ? serializeEvlogErrorResponse(evlogError, url) : buildPlainNitroErrorBody(error, url, isDev);
25
+ const status = extractErrorStatus(evlogError ?? error);
26
+ return new Response(JSON.stringify(body), {
29
27
  status,
30
28
  headers: { "Content-Type": "application/json" }
31
29
  });
@@ -1 +1 @@
1
- {"version":3,"file":"errorHandler.mjs","names":[],"sources":["../../../src/nitro-v3/errorHandler.ts"],"sourcesContent":["import { parseURL } from 'ufo'\nimport { defineErrorHandler } from 'nitro'\nimport { resolveEvlogError, extractErrorStatus, serializeEvlogErrorResponse } from '../nitro'\n\n/**\n * Custom Nitro v3 error handler that properly serializes EvlogError.\n * This ensures that 'data' (containing 'why', 'fix', 'link') is preserved\n * in the JSON response regardless of the underlying HTTP framework.\n *\n * For non-EvlogError, returns undefined to let Nitro's default handler take over.\n *\n * Usage in nitro.config.ts:\n * ```ts\n * // errorHandler.ts\n * export { default } from 'evlog/nitro/v3/errorHandler'\n * // nitro.config.ts\n * export default defineConfig({\n * errorHandler: './errorHandler',\n * })\n * ```\n */\nexport default defineErrorHandler((error, event) => {\n const evlogError = resolveEvlogError(error)\n\n if (!evlogError) return\n\n const url = parseURL(event.req.url).pathname\n const status = extractErrorStatus(evlogError)\n\n return new Response(JSON.stringify(serializeEvlogErrorResponse(evlogError, url)), {\n status,\n headers: { 'Content-Type': 'application/json' },\n })\n})\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;AAqBA,IAAA,uBAAe,oBAAoB,OAAO,UAAU;CAClD,MAAM,aAAa,kBAAkB,MAAM;AAE3C,KAAI,CAAC,WAAY;CAEjB,MAAM,MAAM,SAAS,MAAM,IAAI,IAAI,CAAC;CACpC,MAAM,SAAS,mBAAmB,WAAW;AAE7C,QAAO,IAAI,SAAS,KAAK,UAAU,4BAA4B,YAAY,IAAI,CAAC,EAAE;EAChF;EACA,SAAS,EAAE,gBAAgB,oBAAoB;EAChD,CAAC;EACF"}
1
+ {"version":3,"file":"errorHandler.mjs","names":[],"sources":["../../../src/nitro-v3/errorHandler.ts"],"sourcesContent":["import { parseURL } from 'ufo'\nimport { defineErrorHandler } from 'nitro'\nimport {\n resolveEvlogError,\n extractErrorStatus,\n buildPlainNitroErrorBody,\n serializeEvlogErrorResponse,\n shouldSuppressNitroDevOverlay,\n suppressNitroDevOverlay,\n markH3ErrorHandled,\n} from '../nitro'\nimport type { NitroErrorHandlerContext } from '../shared/nitro-types'\n\n/**\n * Custom Nitro v3 error handler that properly serializes EvlogError.\n * This ensures that 'data' (containing 'why', 'fix', 'link') is preserved\n * in the JSON response regardless of the underlying HTTP framework.\n *\n * Usage in nitro.config.ts:\n * ```ts\n * export { default } from 'evlog/nitro/v3/errorHandler'\n * ```\n */\nexport default defineErrorHandler(async (error, event, ctx: NitroErrorHandlerContext) => {\n const suppressOverlay = shouldSuppressNitroDevOverlay()\n\n if (!suppressOverlay) {\n await ctx.defaultHandler(error, event, { silent: false })\n }\n\n markH3ErrorHandled(event)\n\n if (suppressOverlay) {\n suppressNitroDevOverlay(error)\n }\n\n const url = parseURL(event.req.url).pathname\n const isDev = process.env.NODE_ENV === 'development'\n const evlogError = resolveEvlogError(error)\n\n const body = evlogError\n ? serializeEvlogErrorResponse(evlogError, url)\n : buildPlainNitroErrorBody(error, url, isDev)\n const status = extractErrorStatus(evlogError ?? error)\n\n return new Response(JSON.stringify(body), {\n status,\n headers: { 'Content-Type': 'application/json' },\n })\n})\n"],"mappings":";;;;;;;;;;;;;;;AAuBA,IAAA,uBAAe,mBAAmB,OAAO,OAAO,OAAO,QAAkC;CACvF,MAAM,kBAAkB,+BAA+B;AAEvD,KAAI,CAAC,gBACH,OAAM,IAAI,eAAe,OAAO,OAAO,EAAE,QAAQ,OAAO,CAAC;AAG3D,oBAAmB,MAAM;AAEzB,KAAI,gBACF,yBAAwB,MAAM;CAGhC,MAAM,MAAM,SAAS,MAAM,IAAI,IAAI,CAAC;CACpC,MAAM,QAAQ,QAAQ,IAAI,aAAa;CACvC,MAAM,aAAa,kBAAkB,MAAM;CAE3C,MAAM,OAAO,aACT,4BAA4B,YAAY,IAAI,GAC5C,yBAAyB,OAAO,KAAK,MAAM;CAC/C,MAAM,SAAS,mBAAmB,cAAc,MAAM;AAEtD,QAAO,IAAI,SAAS,KAAK,UAAU,KAAK,EAAE;EACxC;EACA,SAAS,EAAE,gBAAgB,oBAAoB;EAChD,CAAC;EACF"}
@@ -1,5 +1,5 @@
1
- import { n as createError, t as EvlogError } from "../../error-CVtn5U7b.mjs";
2
- import { t as parseError } from "../../parseError-D4PIxEWo.mjs";
1
+ import { n as createError, t as EvlogError } from "../../error-CpghjrkY.mjs";
2
+ import { t as parseError } from "../../parseError-BeBXEd2V.mjs";
3
3
  import evlog from "./module.mjs";
4
4
  import { useLogger } from "./useLogger.mjs";
5
5
  import { evlogErrorHandler } from "./middleware.mjs";
@@ -1,4 +1,4 @@
1
- import { t as NitroModuleOptions } from "../../nitro-BRddgqSb.mjs";
1
+ import { t as NitroModuleOptions } from "../../nitro-_Hda8Deo.mjs";
2
2
  import { Nitro } from "nitro/types";
3
3
 
4
4
  //#region src/nitro-v3/module.d.ts
@@ -1 +1 @@
1
- {"version":3,"file":"module.d.mts","names":[],"sources":["../../../src/nitro-v3/module.ts"],"mappings":";;;;iBAiBwB,KAAA,CAAM,OAAA,GAAU,kBAAA;;eAGvB,KAAA;AAAA"}
1
+ {"version":3,"file":"module.d.mts","names":[],"sources":["../../../src/nitro-v3/module.ts"],"mappings":";;;;iBAkBwB,KAAA,CAAM,OAAA,GAAU,kBAAA;;eAGvB,KAAA;AAAA"}
@@ -1,3 +1,4 @@
1
+ import { r as prependNitroErrorHandler } from "../../nitro-ClRZLD1g.mjs";
1
2
  import { dirname, resolve } from "node:path";
2
3
  import { fileURLToPath } from "node:url";
3
4
  //#region src/nitro-v3/module.ts
@@ -13,9 +14,8 @@ function evlog(options) {
13
14
  nitro.options.plugins.push(resolveModulePath("plugin"));
14
15
  if (!nitro.options.noExternals) nitro.options.noExternals = ["evlog"];
15
16
  else if (Array.isArray(nitro.options.noExternals)) nitro.options.noExternals.push("evlog");
16
- if (!nitro.options.errorHandler) nitro.options.errorHandler = [resolveModulePath("errorHandler")];
17
- else if (Array.isArray(nitro.options.errorHandler)) nitro.options.errorHandler.unshift(resolveModulePath("errorHandler"));
18
- else if (typeof nitro.options.errorHandler === "string") nitro.options.errorHandler = [resolveModulePath("errorHandler"), nitro.options.errorHandler];
17
+ const handlers = prependNitroErrorHandler(nitro.options.errorHandler, resolveModulePath("errorHandler"));
18
+ nitro.options.errorHandler = Array.isArray(handlers) ? handlers : [handlers];
19
19
  nitro.options.runtimeConfig = nitro.options.runtimeConfig || {};
20
20
  nitro.options.runtimeConfig.evlog = options || {};
21
21
  nitro.options.replace = nitro.options.replace || {};
@@ -1 +1 @@
1
- {"version":3,"file":"module.mjs","names":[],"sources":["../../../src/nitro-v3/module.ts"],"sourcesContent":["import { dirname, resolve } from 'node:path'\nimport { fileURLToPath } from 'node:url'\nimport type { Nitro } from 'nitro/types'\nimport type { NitroModuleOptions } from '../nitro'\n\nexport type { NitroModuleOptions }\n\nconst _dir = dirname(fileURLToPath(import.meta.url))\n\n// Nitro raw-interpolates these paths into JS string literals when generating\n// the #nitro/virtual/plugins and #nitro/virtual/error-handler modules, so\n// Windows backslashes would be parsed as escape sequences (\\n, \\v, …) and\n// break module resolution. Normalize to POSIX separators.\nfunction resolveModulePath(name: string): string {\n return resolve(_dir, name).replace(/\\\\/g, '/')\n}\n\nexport default function evlog(options?: NitroModuleOptions) {\n return {\n name: 'evlog',\n setup(nitro: Nitro) {\n // Push the plugin (no extension — Nitro's bundler resolves it)\n nitro.options.plugins = nitro.options.plugins || []\n nitro.options.plugins.push(resolveModulePath('plugin'))\n\n // explicitly tell nitro to bundle evlog's files to correctly resolve nitro dependencies\n if (!nitro.options.noExternals) {\n nitro.options.noExternals = ['evlog']\n } else if (Array.isArray(nitro.options.noExternals)) {\n nitro.options.noExternals.push('evlog')\n }\n\n\n // Set error handler only if not already configured by user\n if (!nitro.options.errorHandler) {\n nitro.options.errorHandler = [resolveModulePath('errorHandler')]\n } else if (Array.isArray(nitro.options.errorHandler)) {\n nitro.options.errorHandler.unshift(resolveModulePath('errorHandler'))\n } else if (typeof nitro.options.errorHandler === 'string') {\n nitro.options.errorHandler = [resolveModulePath('errorHandler'), nitro.options.errorHandler]\n }\n\n // Inject config into runtimeConfig — works in production where the\n // plugin is bundled through Nitro's builder and the virtual\n // runtime-config module resolves correctly.\n nitro.options.runtimeConfig = nitro.options.runtimeConfig || {}\n nitro.options.runtimeConfig.evlog = options || {}\n\n // Bake the config into the bundle as a literal so the plugin never has\n // to do a runtime `import('nitro/runtime-config')` to discover it. That\n // dynamic import resolves to a module which itself imports the build-only\n // virtual `#nitro/virtual/runtime-config` — fine inside the Nitro build,\n // but on Vercel + Bun the missing virtual triggers Bun's auto-installer\n // and crashes with `ReadOnlyFileSystem` (see issue #312).\n nitro.options.replace = nitro.options.replace || {}\n nitro.options.replace.__EVLOG_CONFIG__ = JSON.stringify(options || {})\n\n // In dev mode, Nitro loads plugins externally (not bundled), so the\n // virtual runtime-config module is unreachable and useRuntimeConfig()\n // returns a stub without our values. process.env is inherited by the\n // Worker Threads that run the dev server, making it a reliable bridge.\n // The plugin reads: useRuntimeConfig().evlog ?? process.env.__EVLOG_CONFIG\n process.env.__EVLOG_CONFIG = JSON.stringify(options || {})\n },\n }\n}\n"],"mappings":";;;AAOA,MAAM,OAAO,QAAQ,cAAc,OAAO,KAAK,IAAI,CAAC;AAMpD,SAAS,kBAAkB,MAAsB;AAC/C,QAAO,QAAQ,MAAM,KAAK,CAAC,QAAQ,OAAO,IAAI;;AAGhD,SAAwB,MAAM,SAA8B;AAC1D,QAAO;EACL,MAAM;EACN,MAAM,OAAc;AAElB,SAAM,QAAQ,UAAU,MAAM,QAAQ,WAAW,EAAE;AACnD,SAAM,QAAQ,QAAQ,KAAK,kBAAkB,SAAS,CAAC;AAGvD,OAAI,CAAC,MAAM,QAAQ,YACjB,OAAM,QAAQ,cAAc,CAAC,QAAQ;YAC5B,MAAM,QAAQ,MAAM,QAAQ,YAAY,CACjD,OAAM,QAAQ,YAAY,KAAK,QAAQ;AAKzC,OAAI,CAAC,MAAM,QAAQ,aACjB,OAAM,QAAQ,eAAe,CAAC,kBAAkB,eAAe,CAAC;YACvD,MAAM,QAAQ,MAAM,QAAQ,aAAa,CAClD,OAAM,QAAQ,aAAa,QAAQ,kBAAkB,eAAe,CAAC;YAC5D,OAAO,MAAM,QAAQ,iBAAiB,SAC/C,OAAM,QAAQ,eAAe,CAAC,kBAAkB,eAAe,EAAE,MAAM,QAAQ,aAAa;AAM9F,SAAM,QAAQ,gBAAgB,MAAM,QAAQ,iBAAiB,EAAE;AAC/D,SAAM,QAAQ,cAAc,QAAQ,WAAW,EAAE;AAQjD,SAAM,QAAQ,UAAU,MAAM,QAAQ,WAAW,EAAE;AACnD,SAAM,QAAQ,QAAQ,mBAAmB,KAAK,UAAU,WAAW,EAAE,CAAC;AAOtE,WAAQ,IAAI,iBAAiB,KAAK,UAAU,WAAW,EAAE,CAAC;;EAE7D"}
1
+ {"version":3,"file":"module.mjs","names":[],"sources":["../../../src/nitro-v3/module.ts"],"sourcesContent":["import { dirname, resolve } from 'node:path'\nimport { fileURLToPath } from 'node:url'\nimport type { Nitro } from 'nitro/types'\nimport type { NitroModuleOptions } from '../nitro'\nimport { prependNitroErrorHandler } from '../nitro'\n\nexport type { NitroModuleOptions }\n\nconst _dir = dirname(fileURLToPath(import.meta.url))\n\n// Nitro raw-interpolates these paths into JS string literals when generating\n// the #nitro/virtual/plugins and #nitro/virtual/error-handler modules, so\n// Windows backslashes would be parsed as escape sequences (\\n, \\v, …) and\n// break module resolution. Normalize to POSIX separators.\nfunction resolveModulePath(name: string): string {\n return resolve(_dir, name).replace(/\\\\/g, '/')\n}\n\nexport default function evlog(options?: NitroModuleOptions) {\n return {\n name: 'evlog',\n setup(nitro: Nitro) {\n // Push the plugin (no extension — Nitro's bundler resolves it)\n nitro.options.plugins = nitro.options.plugins || []\n nitro.options.plugins.push(resolveModulePath('plugin'))\n\n // explicitly tell nitro to bundle evlog's files to correctly resolve nitro dependencies\n if (!nitro.options.noExternals) {\n nitro.options.noExternals = ['evlog']\n } else if (Array.isArray(nitro.options.noExternals)) {\n nitro.options.noExternals.push('evlog')\n }\n\n\n const handlers = prependNitroErrorHandler(\n nitro.options.errorHandler,\n resolveModulePath('errorHandler'),\n )\n nitro.options.errorHandler = Array.isArray(handlers) ? handlers : [handlers]\n\n // Inject config into runtimeConfig — works in production where the\n // plugin is bundled through Nitro's builder and the virtual\n // runtime-config module resolves correctly.\n nitro.options.runtimeConfig = nitro.options.runtimeConfig || {}\n nitro.options.runtimeConfig.evlog = options || {}\n\n // Bake the config into the bundle as a literal so the plugin never has\n // to do a runtime `import('nitro/runtime-config')` to discover it. That\n // dynamic import resolves to a module which itself imports the build-only\n // virtual `#nitro/virtual/runtime-config` — fine inside the Nitro build,\n // but on Vercel + Bun the missing virtual triggers Bun's auto-installer\n // and crashes with `ReadOnlyFileSystem` (see issue #312).\n nitro.options.replace = nitro.options.replace || {}\n nitro.options.replace.__EVLOG_CONFIG__ = JSON.stringify(options || {})\n\n // In dev mode, Nitro loads plugins externally (not bundled), so the\n // virtual runtime-config module is unreachable and useRuntimeConfig()\n // returns a stub without our values. process.env is inherited by the\n // Worker Threads that run the dev server, making it a reliable bridge.\n // The plugin reads: useRuntimeConfig().evlog ?? process.env.__EVLOG_CONFIG\n process.env.__EVLOG_CONFIG = JSON.stringify(options || {})\n },\n }\n}\n"],"mappings":";;;;AAQA,MAAM,OAAO,QAAQ,cAAc,OAAO,KAAK,IAAI,CAAC;AAMpD,SAAS,kBAAkB,MAAsB;AAC/C,QAAO,QAAQ,MAAM,KAAK,CAAC,QAAQ,OAAO,IAAI;;AAGhD,SAAwB,MAAM,SAA8B;AAC1D,QAAO;EACL,MAAM;EACN,MAAM,OAAc;AAElB,SAAM,QAAQ,UAAU,MAAM,QAAQ,WAAW,EAAE;AACnD,SAAM,QAAQ,QAAQ,KAAK,kBAAkB,SAAS,CAAC;AAGvD,OAAI,CAAC,MAAM,QAAQ,YACjB,OAAM,QAAQ,cAAc,CAAC,QAAQ;YAC5B,MAAM,QAAQ,MAAM,QAAQ,YAAY,CACjD,OAAM,QAAQ,YAAY,KAAK,QAAQ;GAIzC,MAAM,WAAW,yBACf,MAAM,QAAQ,cACd,kBAAkB,eAAe,CAClC;AACD,SAAM,QAAQ,eAAe,MAAM,QAAQ,SAAS,GAAG,WAAW,CAAC,SAAS;AAK5E,SAAM,QAAQ,gBAAgB,MAAM,QAAQ,iBAAiB,EAAE;AAC/D,SAAM,QAAQ,cAAc,QAAQ,WAAW,EAAE;AAQjD,SAAM,QAAQ,UAAU,MAAM,QAAQ,WAAW,EAAE;AACnD,SAAM,QAAQ,QAAQ,mBAAmB,KAAK,UAAU,WAAW,EAAE,CAAC;AAOtE,WAAQ,IAAI,iBAAiB,KAAK,UAAU,WAAW,EAAE,CAAC;;EAE7D"}