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,5 +1,5 @@
1
- import { $ as RequestLogger } from "../audit-DVdkntSO.mjs";
2
- import { t as BaseEvlogOptions } from "../middleware-31KhtiEF.mjs";
1
+ import { rt as RequestLogger } from "../audit-D7v6JHj0.mjs";
2
+ import { t as BaseEvlogOptions } from "../middleware-B_k4Mrzg.mjs";
3
3
  import { Context, MiddlewareOptions, MiddlewareResult } from "@orpc/server";
4
4
 
5
5
  //#region src/orpc/index.d.ts
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.mts","names":[],"sources":["../../src/orpc/index.ts"],"mappings":";;;;;cAQiB,SAAA,sBAAS,MAAA,wBAAA,aAAA,CAAA,CAAA;;KAKd,gBAAA,GAAmB,gBAAA;;;;;;;;;AAA/B;;;;UAoBiB,gBAAA;EACf,GAAA,EAAK,aAAA;AAAA;;;;AACN;KAMI,qBAAA;EACC,OAAA;EAAe,QAAA,EAAU,QAAA;AAAA;EACzB,OAAA;EAAgB,QAAA;AAAA;;;;AAAQ;;;;;;UAWpB,oBAAA;EACR,MAAA,GACE,OAAA,EAAS,OAAA,EACT,OAAA,WACG,OAAA,CAAQ,qBAAA;AAAA;;;;;;;;AAkDf;;;;;;;;;;;;;;;;;;AA0EA;;;;;iBA1EgB,SAAA,kBAA2B,oBAAA,CAAA,CACzC,OAAA,EAAS,QAAA,EACT,OAAA,GAAS,gBAAA,GACR,QAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;iBAuEa,KAAA,kBAAuB,OAAA,CAAQ,gBAAA,IAAoB,OAAA,GAAU,gBAAA,CAAA,CAAA,IAEzE,OAAA,EAAS,iBAAA,CAAkB,QAAA,yBAC1B,OAAA,CAAQ,gBAAA,CAAiB,MAAA"}
1
+ {"version":3,"file":"index.d.mts","names":[],"sources":["../../src/orpc/index.ts"],"mappings":";;;;;cAQiB,SAAA,sBAAS,MAAA,wBAAA,aAAA,CAAA,CAAA;;KAKd,gBAAA,GAAmB,gBAAA;;;;;;;;;AAA/B;;;;UAoBiB,gBAAA;EACf,GAAA,EAAK,aAAA;AAAA;;;;AACN;KAMI,qBAAA;EACC,OAAA;EAAe,QAAA,EAAU,QAAA;AAAA;EACzB,OAAA;EAAgB,QAAA;AAAA;;;;AAAQ;;;;;;UAWpB,oBAAA;EACR,MAAA,GACE,OAAA,EAAS,OAAA,EACT,OAAA,WACG,OAAA,CAAQ,qBAAA;AAAA;;;;;;;;AAkDf;;;;;;;;;;;;;;;;;;AA6EA;;;;;iBA7EgB,SAAA,kBAA2B,oBAAA,CAAA,CACzC,OAAA,EAAS,QAAA,EACT,OAAA,GAAS,gBAAA,GACR,QAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;iBA0Ea,KAAA,kBAAuB,OAAA,CAAQ,gBAAA,IAAoB,OAAA,GAAU,gBAAA,CAAA,CAAA,IAEzE,OAAA,EAAS,iBAAA,CAAkB,QAAA,yBAC1B,OAAA,CAAQ,gBAAA,CAAiB,MAAA"}
@@ -1,7 +1,7 @@
1
1
  import { EvlogError } from "../error.mjs";
2
2
  import { parseError } from "../runtime/utils/parseError.mjs";
3
- import { t as defineFrameworkIntegration } from "../integration-Dhig7ae6.mjs";
4
- import { t as createLoggerStorage } from "../storage-BNubsWwz.mjs";
3
+ import { t as defineFrameworkIntegration } from "../integration-DYp2uw8O.mjs";
4
+ import { t as createLoggerStorage } from "../storage-7X37OToT.mjs";
5
5
  import { ORPCError } from "@orpc/server";
6
6
  //#region src/orpc/index.ts
7
7
  const { storage, useLogger } = createLoggerStorage("oRPC handler. Wrap your handler with `withEvlog()` from evlog/orpc.");
@@ -51,7 +51,7 @@ const integration = defineFrameworkIntegration({
51
51
  */
52
52
  function withEvlog(handler, options = {}) {
53
53
  const handle = async (request, callOptions) => {
54
- const { skipped, finish, runWith, logger } = integration.start({ request }, options);
54
+ const { skipped, finish, finishResponse, runWith, logger } = integration.start({ request }, options);
55
55
  const initialContext = callOptions?.context ?? {};
56
56
  const finalOptions = {
57
57
  ...callOptions,
@@ -63,7 +63,8 @@ function withEvlog(handler, options = {}) {
63
63
  if (skipped) return handler.handle(request, finalOptions);
64
64
  try {
65
65
  const result = await runWith(() => handler.handle(request, finalOptions));
66
- await finish({ status: result.matched ? result.response.status : 404 });
66
+ if (result.matched) result.response = await finishResponse(result.response, { status: result.response.status });
67
+ else await finish({ status: 404 });
67
68
  return result;
68
69
  } catch (error) {
69
70
  await finish({ error });
@@ -1 +1 @@
1
- {"version":3,"file":"index.mjs","names":[],"sources":["../../src/orpc/index.ts"],"sourcesContent":["import { ORPCError, type Context, type MiddlewareOptions, type MiddlewareResult } from '@orpc/server'\nimport type { RequestLogger } from '../types'\nimport { EvlogError } from '../error'\nimport { parseError } from '../runtime/utils/parseError'\nimport { defineFrameworkIntegration } from '../shared/integration'\nimport type { BaseEvlogOptions } from '../shared/middleware'\nimport { createLoggerStorage } from '../shared/storage'\n\nconst { storage, useLogger } = createLoggerStorage(\n 'oRPC handler. Wrap your handler with `withEvlog()` from evlog/orpc.',\n)\n\n/** Options accepted by {@link withEvlog} for oRPC request instrumentation. */\nexport type EvlogOrpcOptions = BaseEvlogOptions\n\n/**\n * Access the current request-scoped logger outside oRPC context callbacks.\n * Requires {@link withEvlog} on the handler.\n */\nexport { useLogger }\n\n/**\n * Inject this type into your oRPC initial context to access `context.log`\n * inside procedures.\n *\n * @example\n * ```ts\n * import { os } from '@orpc/server'\n * import { evlog, type EvlogOrpcContext } from 'evlog/orpc'\n *\n * const base = os.$context<EvlogOrpcContext>().use(evlog())\n * ```\n */\nexport interface EvlogOrpcContext {\n log: RequestLogger\n}\n\n/**\n * Result shape of `handler.handle()` for oRPC's fetch adapter\n * ({@link https://orpc.dev/docs/adapters/http RPCHandler / OpenAPIHandler}).\n */\ntype OrpcFetchHandleResult =\n | { matched: true, response: Response }\n | { matched: false, response: undefined }\n\n/**\n * Minimal subset of oRPC's `FetchHandler` that we need to wrap. Anything\n * compatible (RPCHandler, OpenAPIHandler, custom handler) plugs in.\n *\n * `options` is intentionally typed loosely so this matches both the\n * `(req, opts)` and `(req, opts?)` overloads that oRPC produces depending on\n * whether the router declares a non-empty initial context. The wrapper just\n * splats the original options through and injects `log` into `context`.\n */\ninterface OrpcFetchHandlerLike {\n handle: (\n request: Request,\n options?: any,\n ) => Promise<OrpcFetchHandleResult>\n}\n\nconst integration = defineFrameworkIntegration<{ request: Request }>({\n name: 'orpc',\n extractRequest: ({ request }) => {\n const url = new URL(request.url)\n return {\n method: request.method,\n path: url.pathname,\n headers: request.headers,\n requestId: request.headers.get('x-request-id') ?? undefined,\n }\n },\n attachLogger: () => {\n /* logger is injected into the oRPC context inside withEvlog() */\n },\n storage,\n})\n\n/**\n * Wrap an oRPC handler so each matched request emits a single wide event.\n * Works with any handler that exposes `.handle(request, options)` from\n * `@orpc/server/fetch` (RPCHandler, OpenAPIHandler, custom handlers).\n *\n * The returned proxy preserves the original handler's identity (instance\n * methods, plugins, etc.) and only intercepts `handle`. Inside procedures,\n * the request logger is exposed as `context.log` — pair this with\n * `os.use(evlog())` to also accumulate `operation` (`path.join('.')`) on the\n * wide event.\n *\n * Routes that are filtered out by `include`/`exclude` are passed straight to\n * the underlying handler with no instrumentation.\n *\n * @example\n * ```ts\n * import { RPCHandler } from '@orpc/server/fetch'\n * import { withEvlog } from 'evlog/orpc'\n * import { router } from './router'\n *\n * const handler = withEvlog(new RPCHandler(router), {\n * include: ['/rpc/**'],\n * })\n *\n * export default async function fetch(request: Request) {\n * const { matched, response } = await handler.handle(request, { prefix: '/rpc' })\n * return matched ? response : new Response('Not Found', { status: 404 })\n * }\n * ```\n */\nexport function withEvlog<THandler extends OrpcFetchHandlerLike>(\n handler: THandler,\n options: EvlogOrpcOptions = {},\n): THandler {\n const handle: THandler['handle'] = async (request, callOptions) => {\n const { skipped, finish, runWith, logger } = integration.start({ request }, options)\n\n const initialContext = (callOptions as { context?: Record<string, unknown> } | undefined)?.context ?? {}\n const finalOptions = {\n ...callOptions,\n context: { ...initialContext, log: logger },\n } as Parameters<THandler['handle']>[1]\n\n if (skipped) {\n return handler.handle(request, finalOptions)\n }\n\n try {\n const result = await runWith(() => handler.handle(request, finalOptions))\n const status = result.matched ? result.response.status : 404\n await finish({ status })\n return result\n } catch (error) {\n await finish({ error: error as Error })\n throw error\n }\n }\n\n return new Proxy(handler, {\n get(target, prop, receiver) {\n if (prop === 'handle') return handle\n return Reflect.get(target, prop, receiver)\n },\n })\n}\n\nfunction isEvlogError(error: unknown): error is EvlogError {\n return error instanceof EvlogError || (error instanceof Error && error.name === 'EvlogError')\n}\n\n/**\n * Procedure-level middleware. Three responsibilities:\n *\n * 1. Adds `operation` (the procedure path joined by `.`) to the wide event,\n * so consumers can group events by procedure without parsing URLs.\n * 2. Captures errors thrown by the procedure on the wide event so the level\n * is promoted to `error`.\n * 3. Converts {@link EvlogError} (from `createError()` / `defineErrorCatalog`)\n * into a structurally-equivalent {@link ORPCError} before re-throwing, so\n * the wire response carries the catalog `code`, status, message, and the\n * `why` / `fix` / `link` guidance under `data` — instead of being wrapped\n * as `INTERNAL_SERVER_ERROR` by oRPC's default handler. The catalog and\n * `createError()` stay the canonical evlog way to author errors;\n * `evlog/orpc` is the bridge.\n *\n * Requires `withEvlog()` to be wrapped around the handler — the request\n * logger flows in via `context.log`. Declare {@link EvlogOrpcContext} on\n * your oRPC base for typed access.\n *\n * @example\n * ```ts\n * import { os } from '@orpc/server'\n * import { evlog, type EvlogOrpcContext } from 'evlog/orpc'\n *\n * const base = os.$context<EvlogOrpcContext>().use(evlog())\n *\n * export const getUser = base\n * .input(z.object({ id: z.string() }))\n * .handler(async ({ input, context }) => {\n * context.log.set({ user: { id: input.id } })\n * return await db.user.findUnique(input)\n * })\n * ```\n */\nexport function evlog<TContext extends Partial<EvlogOrpcContext> & Context = EvlogOrpcContext>() {\n return async function evlogMiddleware(\n options: MiddlewareOptions<TContext, unknown, any, any>,\n ): Promise<MiddlewareResult<Record<never, never>, unknown>> {\n const { context: { log }, path, next } = options\n if (log && path.length > 0) {\n log.set({ operation: path.join('.') })\n }\n try {\n return await next()\n } catch (error) {\n if (log) log.error(error as Error)\n if (isEvlogError(error)) {\n throw toOrpcError(error)\n }\n throw error\n }\n }\n}\n\nfunction toOrpcError(error: EvlogError): ORPCError<string, Record<string, unknown>> {\n const parsed = parseError(error)\n const data: Record<string, unknown> = {}\n if (parsed.why !== undefined) data.why = parsed.why\n if (parsed.fix !== undefined) data.fix = parsed.fix\n if (parsed.link !== undefined) data.link = parsed.link\n return new ORPCError(parsed.code ?? 'EVLOG_ERROR', {\n status: parsed.status,\n message: parsed.message,\n data,\n cause: error,\n })\n}\n"],"mappings":";;;;;;AAQA,MAAM,EAAE,SAAS,cAAc,oBAC7B,sEACD;AAmDD,MAAM,cAAc,2BAAiD;CACnE,MAAM;CACN,iBAAiB,EAAE,cAAc;EAC/B,MAAM,MAAM,IAAI,IAAI,QAAQ,IAAI;AAChC,SAAO;GACL,QAAQ,QAAQ;GAChB,MAAM,IAAI;GACV,SAAS,QAAQ;GACjB,WAAW,QAAQ,QAAQ,IAAI,eAAe,IAAI,KAAA;GACnD;;CAEH,oBAAoB;CAGpB;CACD,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAgCF,SAAgB,UACd,SACA,UAA4B,EAAE,EACpB;CACV,MAAM,SAA6B,OAAO,SAAS,gBAAgB;EACjE,MAAM,EAAE,SAAS,QAAQ,SAAS,WAAW,YAAY,MAAM,EAAE,SAAS,EAAE,QAAQ;EAEpF,MAAM,iBAAkB,aAAmE,WAAW,EAAE;EACxG,MAAM,eAAe;GACnB,GAAG;GACH,SAAS;IAAE,GAAG;IAAgB,KAAK;IAAQ;GAC5C;AAED,MAAI,QACF,QAAO,QAAQ,OAAO,SAAS,aAAa;AAG9C,MAAI;GACF,MAAM,SAAS,MAAM,cAAc,QAAQ,OAAO,SAAS,aAAa,CAAC;AAEzE,SAAM,OAAO,EAAE,QADA,OAAO,UAAU,OAAO,SAAS,SAAS,KAClC,CAAC;AACxB,UAAO;WACA,OAAO;AACd,SAAM,OAAO,EAAS,OAAgB,CAAC;AACvC,SAAM;;;AAIV,QAAO,IAAI,MAAM,SAAS,EACxB,IAAI,QAAQ,MAAM,UAAU;AAC1B,MAAI,SAAS,SAAU,QAAO;AAC9B,SAAO,QAAQ,IAAI,QAAQ,MAAM,SAAS;IAE7C,CAAC;;AAGJ,SAAS,aAAa,OAAqC;AACzD,QAAO,iBAAiB,cAAe,iBAAiB,SAAS,MAAM,SAAS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAqClF,SAAgB,QAAiF;AAC/F,QAAO,eAAe,gBACpB,SAC0D;EAC1D,MAAM,EAAE,SAAS,EAAE,OAAO,MAAM,SAAS;AACzC,MAAI,OAAO,KAAK,SAAS,EACvB,KAAI,IAAI,EAAE,WAAW,KAAK,KAAK,IAAI,EAAE,CAAC;AAExC,MAAI;AACF,UAAO,MAAM,MAAM;WACZ,OAAO;AACd,OAAI,IAAK,KAAI,MAAM,MAAe;AAClC,OAAI,aAAa,MAAM,CACrB,OAAM,YAAY,MAAM;AAE1B,SAAM;;;;AAKZ,SAAS,YAAY,OAA+D;CAClF,MAAM,SAAS,WAAW,MAAM;CAChC,MAAM,OAAgC,EAAE;AACxC,KAAI,OAAO,QAAQ,KAAA,EAAW,MAAK,MAAM,OAAO;AAChD,KAAI,OAAO,QAAQ,KAAA,EAAW,MAAK,MAAM,OAAO;AAChD,KAAI,OAAO,SAAS,KAAA,EAAW,MAAK,OAAO,OAAO;AAClD,QAAO,IAAI,UAAU,OAAO,QAAQ,eAAe;EACjD,QAAQ,OAAO;EACf,SAAS,OAAO;EAChB;EACA,OAAO;EACR,CAAC"}
1
+ {"version":3,"file":"index.mjs","names":[],"sources":["../../src/orpc/index.ts"],"sourcesContent":["import { ORPCError, type Context, type MiddlewareOptions, type MiddlewareResult } from '@orpc/server'\nimport type { RequestLogger } from '../types'\nimport { EvlogError } from '../error'\nimport { parseError } from '../runtime/utils/parseError'\nimport { defineFrameworkIntegration } from '../shared/integration'\nimport type { BaseEvlogOptions } from '../shared/middleware'\nimport { createLoggerStorage } from '../shared/storage'\n\nconst { storage, useLogger } = createLoggerStorage(\n 'oRPC handler. Wrap your handler with `withEvlog()` from evlog/orpc.',\n)\n\n/** Options accepted by {@link withEvlog} for oRPC request instrumentation. */\nexport type EvlogOrpcOptions = BaseEvlogOptions\n\n/**\n * Access the current request-scoped logger outside oRPC context callbacks.\n * Requires {@link withEvlog} on the handler.\n */\nexport { useLogger }\n\n/**\n * Inject this type into your oRPC initial context to access `context.log`\n * inside procedures.\n *\n * @example\n * ```ts\n * import { os } from '@orpc/server'\n * import { evlog, type EvlogOrpcContext } from 'evlog/orpc'\n *\n * const base = os.$context<EvlogOrpcContext>().use(evlog())\n * ```\n */\nexport interface EvlogOrpcContext {\n log: RequestLogger\n}\n\n/**\n * Result shape of `handler.handle()` for oRPC's fetch adapter\n * ({@link https://orpc.dev/docs/adapters/http RPCHandler / OpenAPIHandler}).\n */\ntype OrpcFetchHandleResult =\n | { matched: true, response: Response }\n | { matched: false, response: undefined }\n\n/**\n * Minimal subset of oRPC's `FetchHandler` that we need to wrap. Anything\n * compatible (RPCHandler, OpenAPIHandler, custom handler) plugs in.\n *\n * `options` is intentionally typed loosely so this matches both the\n * `(req, opts)` and `(req, opts?)` overloads that oRPC produces depending on\n * whether the router declares a non-empty initial context. The wrapper just\n * splats the original options through and injects `log` into `context`.\n */\ninterface OrpcFetchHandlerLike {\n handle: (\n request: Request,\n options?: any,\n ) => Promise<OrpcFetchHandleResult>\n}\n\nconst integration = defineFrameworkIntegration<{ request: Request }>({\n name: 'orpc',\n extractRequest: ({ request }) => {\n const url = new URL(request.url)\n return {\n method: request.method,\n path: url.pathname,\n headers: request.headers,\n requestId: request.headers.get('x-request-id') ?? undefined,\n }\n },\n attachLogger: () => {\n /* logger is injected into the oRPC context inside withEvlog() */\n },\n storage,\n})\n\n/**\n * Wrap an oRPC handler so each matched request emits a single wide event.\n * Works with any handler that exposes `.handle(request, options)` from\n * `@orpc/server/fetch` (RPCHandler, OpenAPIHandler, custom handlers).\n *\n * The returned proxy preserves the original handler's identity (instance\n * methods, plugins, etc.) and only intercepts `handle`. Inside procedures,\n * the request logger is exposed as `context.log` — pair this with\n * `os.use(evlog())` to also accumulate `operation` (`path.join('.')`) on the\n * wide event.\n *\n * Routes that are filtered out by `include`/`exclude` are passed straight to\n * the underlying handler with no instrumentation.\n *\n * @example\n * ```ts\n * import { RPCHandler } from '@orpc/server/fetch'\n * import { withEvlog } from 'evlog/orpc'\n * import { router } from './router'\n *\n * const handler = withEvlog(new RPCHandler(router), {\n * include: ['/rpc/**'],\n * })\n *\n * export default async function fetch(request: Request) {\n * const { matched, response } = await handler.handle(request, { prefix: '/rpc' })\n * return matched ? response : new Response('Not Found', { status: 404 })\n * }\n * ```\n */\nexport function withEvlog<THandler extends OrpcFetchHandlerLike>(\n handler: THandler,\n options: EvlogOrpcOptions = {},\n): THandler {\n const handle: THandler['handle'] = async (request, callOptions) => {\n const { skipped, finish, finishResponse, runWith, logger } = integration.start({ request }, options)\n\n const initialContext = (callOptions as { context?: Record<string, unknown> } | undefined)?.context ?? {}\n const finalOptions = {\n ...callOptions,\n context: { ...initialContext, log: logger },\n } as Parameters<THandler['handle']>[1]\n\n if (skipped) {\n return handler.handle(request, finalOptions)\n }\n\n try {\n const result = await runWith(() => handler.handle(request, finalOptions))\n if (result.matched) {\n result.response = await finishResponse(result.response, { status: result.response.status })\n } else {\n await finish({ status: 404 })\n }\n return result\n } catch (error) {\n await finish({ error: error as Error })\n throw error\n }\n }\n\n return new Proxy(handler, {\n get(target, prop, receiver) {\n if (prop === 'handle') return handle\n return Reflect.get(target, prop, receiver)\n },\n })\n}\n\nfunction isEvlogError(error: unknown): error is EvlogError {\n return error instanceof EvlogError || (error instanceof Error && error.name === 'EvlogError')\n}\n\n/**\n * Procedure-level middleware. Three responsibilities:\n *\n * 1. Adds `operation` (the procedure path joined by `.`) to the wide event,\n * so consumers can group events by procedure without parsing URLs.\n * 2. Captures errors thrown by the procedure on the wide event so the level\n * is promoted to `error`.\n * 3. Converts {@link EvlogError} (from `createError()` / `defineErrorCatalog`)\n * into a structurally-equivalent {@link ORPCError} before re-throwing, so\n * the wire response carries the catalog `code`, status, message, and the\n * `why` / `fix` / `link` guidance under `data` — instead of being wrapped\n * as `INTERNAL_SERVER_ERROR` by oRPC's default handler. The catalog and\n * `createError()` stay the canonical evlog way to author errors;\n * `evlog/orpc` is the bridge.\n *\n * Requires `withEvlog()` to be wrapped around the handler — the request\n * logger flows in via `context.log`. Declare {@link EvlogOrpcContext} on\n * your oRPC base for typed access.\n *\n * @example\n * ```ts\n * import { os } from '@orpc/server'\n * import { evlog, type EvlogOrpcContext } from 'evlog/orpc'\n *\n * const base = os.$context<EvlogOrpcContext>().use(evlog())\n *\n * export const getUser = base\n * .input(z.object({ id: z.string() }))\n * .handler(async ({ input, context }) => {\n * context.log.set({ user: { id: input.id } })\n * return await db.user.findUnique(input)\n * })\n * ```\n */\nexport function evlog<TContext extends Partial<EvlogOrpcContext> & Context = EvlogOrpcContext>() {\n return async function evlogMiddleware(\n options: MiddlewareOptions<TContext, unknown, any, any>,\n ): Promise<MiddlewareResult<Record<never, never>, unknown>> {\n const { context: { log }, path, next } = options\n if (log && path.length > 0) {\n log.set({ operation: path.join('.') })\n }\n try {\n return await next()\n } catch (error) {\n if (log) log.error(error as Error)\n if (isEvlogError(error)) {\n throw toOrpcError(error)\n }\n throw error\n }\n }\n}\n\nfunction toOrpcError(error: EvlogError): ORPCError<string, Record<string, unknown>> {\n const parsed = parseError(error)\n const data: Record<string, unknown> = {}\n if (parsed.why !== undefined) data.why = parsed.why\n if (parsed.fix !== undefined) data.fix = parsed.fix\n if (parsed.link !== undefined) data.link = parsed.link\n return new ORPCError(parsed.code ?? 'EVLOG_ERROR', {\n status: parsed.status,\n message: parsed.message,\n data,\n cause: error,\n })\n}\n"],"mappings":";;;;;;AAQA,MAAM,EAAE,SAAS,cAAc,oBAC7B,sEACD;AAmDD,MAAM,cAAc,2BAAiD;CACnE,MAAM;CACN,iBAAiB,EAAE,cAAc;EAC/B,MAAM,MAAM,IAAI,IAAI,QAAQ,IAAI;AAChC,SAAO;GACL,QAAQ,QAAQ;GAChB,MAAM,IAAI;GACV,SAAS,QAAQ;GACjB,WAAW,QAAQ,QAAQ,IAAI,eAAe,IAAI,KAAA;GACnD;;CAEH,oBAAoB;CAGpB;CACD,CAAC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAgCF,SAAgB,UACd,SACA,UAA4B,EAAE,EACpB;CACV,MAAM,SAA6B,OAAO,SAAS,gBAAgB;EACjE,MAAM,EAAE,SAAS,QAAQ,gBAAgB,SAAS,WAAW,YAAY,MAAM,EAAE,SAAS,EAAE,QAAQ;EAEpG,MAAM,iBAAkB,aAAmE,WAAW,EAAE;EACxG,MAAM,eAAe;GACnB,GAAG;GACH,SAAS;IAAE,GAAG;IAAgB,KAAK;IAAQ;GAC5C;AAED,MAAI,QACF,QAAO,QAAQ,OAAO,SAAS,aAAa;AAG9C,MAAI;GACF,MAAM,SAAS,MAAM,cAAc,QAAQ,OAAO,SAAS,aAAa,CAAC;AACzE,OAAI,OAAO,QACT,QAAO,WAAW,MAAM,eAAe,OAAO,UAAU,EAAE,QAAQ,OAAO,SAAS,QAAQ,CAAC;OAE3F,OAAM,OAAO,EAAE,QAAQ,KAAK,CAAC;AAE/B,UAAO;WACA,OAAO;AACd,SAAM,OAAO,EAAS,OAAgB,CAAC;AACvC,SAAM;;;AAIV,QAAO,IAAI,MAAM,SAAS,EACxB,IAAI,QAAQ,MAAM,UAAU;AAC1B,MAAI,SAAS,SAAU,QAAO;AAC9B,SAAO,QAAQ,IAAI,QAAQ,MAAM,SAAS;IAE7C,CAAC;;AAGJ,SAAS,aAAa,OAAqC;AACzD,QAAO,iBAAiB,cAAe,iBAAiB,SAAS,MAAM,SAAS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAqClF,SAAgB,QAAiF;AAC/F,QAAO,eAAe,gBACpB,SAC0D;EAC1D,MAAM,EAAE,SAAS,EAAE,OAAO,MAAM,SAAS;AACzC,MAAI,OAAO,KAAK,SAAS,EACvB,KAAI,IAAI,EAAE,WAAW,KAAK,KAAK,IAAI,EAAE,CAAC;AAExC,MAAI;AACF,UAAO,MAAM,MAAM;WACZ,OAAO;AACd,OAAI,IAAK,KAAI,MAAM,MAAe;AAClC,OAAI,aAAa,MAAM,CACrB,OAAM,YAAY,MAAM;AAE1B,SAAM;;;;AAKZ,SAAS,YAAY,OAA+D;CAClF,MAAM,SAAS,WAAW,MAAM;CAChC,MAAM,OAAgC,EAAE;AACxC,KAAI,OAAO,QAAQ,KAAA,EAAW,MAAK,MAAM,OAAO;AAChD,KAAI,OAAO,QAAQ,KAAA,EAAW,MAAK,MAAM,OAAO;AAChD,KAAI,OAAO,SAAS,KAAA,EAAW,MAAK,OAAO,OAAO;AAClD,QAAO,IAAI,UAAU,OAAO,QAAQ,eAAe;EACjD,QAAQ,OAAO;EACf,SAAS,OAAO;EAChB;EACA,OAAO;EACR,CAAC"}
@@ -1,7 +1,7 @@
1
1
  //#region package.json
2
2
  var name = "evlog";
3
- var version = "2.18.1";
3
+ var version = "2.19.1";
4
4
  //#endregion
5
5
  export { version as n, name as t };
6
6
 
7
- //# sourceMappingURL=package-B23bR3tK.mjs.map
7
+ //# sourceMappingURL=package-CNV_CXs8.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"package-CNV_CXs8.mjs","names":[],"sources":["../package.json"],"sourcesContent":[""],"mappings":""}
@@ -1,7 +1,7 @@
1
- import { J as ParsedError } from "./audit-DVdkntSO.mjs";
1
+ import { Q as ParsedError } from "./audit-D7v6JHj0.mjs";
2
2
 
3
3
  //#region src/runtime/utils/parseError.d.ts
4
4
  declare function parseError(error: unknown): ParsedError;
5
5
  //#endregion
6
6
  export { parseError as t };
7
- //# sourceMappingURL=parseError-D4PIxEWo.d.mts.map
7
+ //# sourceMappingURL=parseError-BeBXEd2V.d.mts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"parseError-BeBXEd2V.d.mts","names":[],"sources":["../src/runtime/utils/parseError.ts"],"mappings":";;;iBAcgB,UAAA,CAAW,KAAA,YAAiB,WAAA"}
@@ -21,25 +21,6 @@ interface PipelineDrainFn<T> {
21
21
  flush: () => Promise<void>;
22
22
  readonly pending: number;
23
23
  }
24
- /**
25
- * Create a drain pipeline that batches events, retries on failure, and manages buffer overflow.
26
- *
27
- * Returns a higher-order function: pass your drain adapter to get a hook-compatible function.
28
- *
29
- * @example
30
- * ```ts
31
- * const pipeline = createDrainPipeline({ batch: { size: 50 } })
32
- * const drain = pipeline(async (batch) => {
33
- * await sendToBackend(batch)
34
- * })
35
- *
36
- * // Use as a hook
37
- * nitroApp.hooks.hook('evlog:drain', drain)
38
- *
39
- * // Flush on shutdown
40
- * nitroApp.hooks.hook('close', () => drain.flush())
41
- * ```
42
- */
43
24
  declare function createDrainPipeline<T = unknown>(options?: DrainPipelineOptions<T>): (drain: (batch: T[]) => void | Promise<void>) => PipelineDrainFn<T>;
44
25
  //#endregion
45
26
  export { DrainPipelineOptions, PipelineDrainFn, createDrainPipeline };
@@ -1 +1 @@
1
- {"version":3,"file":"pipeline.d.mts","names":[],"sources":["../src/pipeline.ts"],"mappings":";UAAiB,oBAAA;EACf,KAAA;IADmC,iFAGjC,IAAA,WAiBqC;IAfrC,UAAA;EAAA;EAEF,KAAA;IAFE,iGAIA,WAAA,WAAA;IAEA,OAAA,uCAEA;IAAA,cAAA,WAKF;IAHE,UAAA;EAAA;EAKW;EAFb,aAAA;EAE0B;EAA1B,SAAA,IAAa,MAAA,EAAQ,CAAA,IAAK,KAAA,GAAQ,KAAA;AAAA;AAAA,UAGnB,eAAA;EAAA,CACd,GAAA,EAAK,CAAA;EADwB;EAG9B,KAAA,QAAa,OAAA;EAAA,SACJ,OAAA;AAAA;;;;;;;AAsBX;;;;;;;;;;;;;iBAAgB,mBAAA,aAAA,CAAiC,OAAA,GAAU,oBAAA,CAAqB,CAAA,KAAM,KAAA,GAAQ,KAAA,EAAO,CAAA,cAAe,OAAA,WAAkB,eAAA,CAAgB,CAAA"}
1
+ {"version":3,"file":"pipeline.d.mts","names":[],"sources":["../src/pipeline.ts"],"mappings":";UAAiB,oBAAA;EACf,KAAA;IADmC,iFAGjC,IAAA,WAiBqC;IAfrC,UAAA;EAAA;EAEF,KAAA;IAFE,iGAIA,WAAA,WAAA;IAEA,OAAA,uCAEA;IAAA,cAAA,WAKF;IAHE,UAAA;EAAA;EAKW;EAFb,aAAA;EAE0B;EAA1B,SAAA,IAAa,MAAA,EAAQ,CAAA,IAAK,KAAA,GAAQ,KAAA;AAAA;AAAA,UAGnB,eAAA;EAAA,CACd,GAAA,EAAK,CAAA;EADwB;EAG9B,KAAA,QAAa,OAAA;EAAA,SACJ,OAAA;AAAA;AAAA,iBAkCK,mBAAA,aAAA,CAAiC,OAAA,GAAU,oBAAA,CAAqB,CAAA,KAAM,KAAA,GAAQ,KAAA,EAAO,CAAA,cAAe,OAAA,WAAkB,eAAA,CAAgB,CAAA"}
package/dist/pipeline.mjs CHANGED
@@ -18,6 +18,17 @@
18
18
  * nitroApp.hooks.hook('close', () => drain.flush())
19
19
  * ```
20
20
  */
21
+ /**
22
+ * Unref a timer on runtimes that support it (Node, Bun) so an idle flush
23
+ * scheduling timer never holds the process open. Buffered events are delivered
24
+ * on shutdown via the documented `flush()` contract, not by keeping timers
25
+ * alive. Retry backoff timers are intentionally left ref'd: an in-flight batch
26
+ * is active work, and an unref'd timer awaited by `flush()` would let the
27
+ * process exit mid-retry.
28
+ */
29
+ function unrefTimer(timer) {
30
+ timer.unref?.();
31
+ }
21
32
  function createDrainPipeline(options) {
22
33
  const batchSize = options?.batch?.size ?? 50;
23
34
  const intervalMs = options?.batch?.intervalMs ?? 5e3;
@@ -49,6 +60,7 @@ function createDrainPipeline(options) {
49
60
  timer = null;
50
61
  if (!activeFlush) startFlush();
51
62
  }, intervalMs);
63
+ unrefTimer(timer);
52
64
  }
53
65
  function getRetryDelay(attempt) {
54
66
  let delay;
@@ -1 +1 @@
1
- {"version":3,"file":"pipeline.mjs","names":[],"sources":["../src/pipeline.ts"],"sourcesContent":["export interface DrainPipelineOptions<T = unknown> {\n batch?: {\n /** Maximum number of events per batch sent to the drain function. @default 50 */\n size?: number\n /** Maximum time (ms) an event can stay buffered before a flush is triggered, even if the batch is not full. @default 5000 */\n intervalMs?: number\n }\n retry?: {\n /** Total number of attempts (including the initial one) before dropping the batch. @default 3 */\n maxAttempts?: number\n /** Delay strategy between retry attempts. @default 'exponential' */\n backoff?: 'exponential' | 'linear' | 'fixed'\n /** Base delay (ms) for the first retry. Scaled by the backoff strategy on subsequent retries. @default 1000 */\n initialDelayMs?: number\n /** Upper bound (ms) for any single retry delay. @default 30000 */\n maxDelayMs?: number\n }\n /** Maximum number of events held in the buffer. When exceeded, the oldest event is dropped. @default 1000 */\n maxBufferSize?: number\n /** Called when a batch is dropped after all retry attempts are exhausted, or when the buffer overflows. */\n onDropped?: (events: T[], error?: Error) => void\n}\n\nexport interface PipelineDrainFn<T> {\n (ctx: T): void\n /** Flush all buffered events. Call on server shutdown. */\n flush: () => Promise<void>\n readonly pending: number\n}\n\n/**\n * Create a drain pipeline that batches events, retries on failure, and manages buffer overflow.\n *\n * Returns a higher-order function: pass your drain adapter to get a hook-compatible function.\n *\n * @example\n * ```ts\n * const pipeline = createDrainPipeline({ batch: { size: 50 } })\n * const drain = pipeline(async (batch) => {\n * await sendToBackend(batch)\n * })\n *\n * // Use as a hook\n * nitroApp.hooks.hook('evlog:drain', drain)\n *\n * // Flush on shutdown\n * nitroApp.hooks.hook('close', () => drain.flush())\n * ```\n */\nexport function createDrainPipeline<T = unknown>(options?: DrainPipelineOptions<T>): (drain: (batch: T[]) => void | Promise<void>) => PipelineDrainFn<T> {\n const batchSize = options?.batch?.size ?? 50\n const intervalMs = options?.batch?.intervalMs ?? 5000\n const maxBufferSize = options?.maxBufferSize ?? 1000\n const maxAttempts = options?.retry?.maxAttempts ?? 3\n const backoffStrategy = options?.retry?.backoff ?? 'exponential'\n const initialDelayMs = options?.retry?.initialDelayMs ?? 1000\n const maxDelayMs = options?.retry?.maxDelayMs ?? 30000\n const onDropped = options?.onDropped\n\n if (batchSize <= 0 || !Number.isFinite(batchSize)) {\n throw new Error(`[evlog/pipeline] batch.size must be a positive finite number, got: ${batchSize}`)\n }\n if (intervalMs <= 0 || !Number.isFinite(intervalMs)) {\n throw new Error(`[evlog/pipeline] batch.intervalMs must be a positive finite number, got: ${intervalMs}`)\n }\n if (maxBufferSize <= 0 || !Number.isFinite(maxBufferSize)) {\n throw new Error(`[evlog/pipeline] maxBufferSize must be a positive finite number, got: ${maxBufferSize}`)\n }\n if (maxAttempts <= 0 || !Number.isFinite(maxAttempts)) {\n throw new Error(`[evlog/pipeline] retry.maxAttempts must be a positive finite number, got: ${maxAttempts}`)\n }\n if (initialDelayMs < 0 || !Number.isFinite(initialDelayMs)) {\n throw new Error(`[evlog/pipeline] retry.initialDelayMs must be a non-negative finite number, got: ${initialDelayMs}`)\n }\n if (maxDelayMs < 0 || !Number.isFinite(maxDelayMs)) {\n throw new Error(`[evlog/pipeline] retry.maxDelayMs must be a non-negative finite number, got: ${maxDelayMs}`)\n }\n\n return (drain: (batch: T[]) => void | Promise<void>): PipelineDrainFn<T> => {\n const buffer: T[] = []\n let timer: ReturnType<typeof setTimeout> | null = null\n let activeFlush: Promise<void> | null = null\n\n function clearTimer(): void {\n if (timer !== null) {\n clearTimeout(timer)\n timer = null\n }\n }\n\n function scheduleFlush(): void {\n if (timer !== null || activeFlush) return\n timer = setTimeout(() => {\n timer = null\n if (!activeFlush) startFlush()\n }, intervalMs)\n }\n\n function getRetryDelay(attempt: number): number {\n let delay: number\n switch (backoffStrategy) {\n case 'linear':\n delay = initialDelayMs * attempt\n break\n case 'fixed':\n delay = initialDelayMs\n break\n case 'exponential':\n default:\n delay = initialDelayMs * 2 ** (attempt - 1)\n break\n }\n return Math.min(delay, maxDelayMs)\n }\n\n async function sendWithRetry(batch: T[]): Promise<void> {\n let lastError: Error | undefined\n for (let attempt = 1; attempt <= maxAttempts; attempt++) {\n try {\n await drain(batch)\n return\n } catch (error) {\n lastError = error instanceof Error ? error : new Error(String(error))\n if (attempt < maxAttempts) {\n await new Promise<void>(r => setTimeout(r, getRetryDelay(attempt)))\n }\n }\n }\n onDropped?.(batch, lastError)\n }\n\n async function drainBuffer(): Promise<void> {\n while (buffer.length > 0) {\n const batch = buffer.splice(0, batchSize)\n await sendWithRetry(batch)\n }\n }\n\n function startFlush(): void {\n if (activeFlush) return\n activeFlush = drainBuffer().finally(() => {\n activeFlush = null\n if (buffer.length >= batchSize) {\n startFlush()\n } else if (buffer.length > 0) {\n scheduleFlush()\n }\n })\n }\n\n function push(ctx: T): void {\n if (buffer.length >= maxBufferSize) {\n const dropped = buffer.splice(0, 1)\n onDropped?.(dropped)\n }\n buffer.push(ctx)\n\n if (buffer.length >= batchSize) {\n clearTimer()\n startFlush()\n } else if (!activeFlush) {\n scheduleFlush()\n }\n }\n\n async function flush(): Promise<void> {\n clearTimer()\n if (activeFlush) {\n await activeFlush\n }\n // Snapshot the buffer length to avoid infinite loop if push() is called during flush\n const snapshot = buffer.length\n if (snapshot > 0) {\n const toFlush = buffer.splice(0, snapshot)\n while (toFlush.length > 0) {\n const batch = toFlush.splice(0, batchSize)\n await sendWithRetry(batch)\n }\n }\n }\n\n const hookFn = push as PipelineDrainFn<T>\n hookFn.flush = flush\n Object.defineProperty(hookFn, 'pending', {\n get: () => buffer.length,\n enumerable: true,\n })\n\n return hookFn\n }\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"pipeline.mjs","names":[],"sources":["../src/pipeline.ts"],"sourcesContent":["export interface DrainPipelineOptions<T = unknown> {\n batch?: {\n /** Maximum number of events per batch sent to the drain function. @default 50 */\n size?: number\n /** Maximum time (ms) an event can stay buffered before a flush is triggered, even if the batch is not full. @default 5000 */\n intervalMs?: number\n }\n retry?: {\n /** Total number of attempts (including the initial one) before dropping the batch. @default 3 */\n maxAttempts?: number\n /** Delay strategy between retry attempts. @default 'exponential' */\n backoff?: 'exponential' | 'linear' | 'fixed'\n /** Base delay (ms) for the first retry. Scaled by the backoff strategy on subsequent retries. @default 1000 */\n initialDelayMs?: number\n /** Upper bound (ms) for any single retry delay. @default 30000 */\n maxDelayMs?: number\n }\n /** Maximum number of events held in the buffer. When exceeded, the oldest event is dropped. @default 1000 */\n maxBufferSize?: number\n /** Called when a batch is dropped after all retry attempts are exhausted, or when the buffer overflows. */\n onDropped?: (events: T[], error?: Error) => void\n}\n\nexport interface PipelineDrainFn<T> {\n (ctx: T): void\n /** Flush all buffered events. Call on server shutdown. */\n flush: () => Promise<void>\n readonly pending: number\n}\n\n/**\n * Create a drain pipeline that batches events, retries on failure, and manages buffer overflow.\n *\n * Returns a higher-order function: pass your drain adapter to get a hook-compatible function.\n *\n * @example\n * ```ts\n * const pipeline = createDrainPipeline({ batch: { size: 50 } })\n * const drain = pipeline(async (batch) => {\n * await sendToBackend(batch)\n * })\n *\n * // Use as a hook\n * nitroApp.hooks.hook('evlog:drain', drain)\n *\n * // Flush on shutdown\n * nitroApp.hooks.hook('close', () => drain.flush())\n * ```\n */\n/**\n * Unref a timer on runtimes that support it (Node, Bun) so an idle flush\n * scheduling timer never holds the process open. Buffered events are delivered\n * on shutdown via the documented `flush()` contract, not by keeping timers\n * alive. Retry backoff timers are intentionally left ref'd: an in-flight batch\n * is active work, and an unref'd timer awaited by `flush()` would let the\n * process exit mid-retry.\n */\nfunction unrefTimer(timer: ReturnType<typeof setTimeout>): void {\n (timer as { unref?: () => void }).unref?.()\n}\n\nexport function createDrainPipeline<T = unknown>(options?: DrainPipelineOptions<T>): (drain: (batch: T[]) => void | Promise<void>) => PipelineDrainFn<T> {\n const batchSize = options?.batch?.size ?? 50\n const intervalMs = options?.batch?.intervalMs ?? 5000\n const maxBufferSize = options?.maxBufferSize ?? 1000\n const maxAttempts = options?.retry?.maxAttempts ?? 3\n const backoffStrategy = options?.retry?.backoff ?? 'exponential'\n const initialDelayMs = options?.retry?.initialDelayMs ?? 1000\n const maxDelayMs = options?.retry?.maxDelayMs ?? 30000\n const onDropped = options?.onDropped\n\n if (batchSize <= 0 || !Number.isFinite(batchSize)) {\n throw new Error(`[evlog/pipeline] batch.size must be a positive finite number, got: ${batchSize}`)\n }\n if (intervalMs <= 0 || !Number.isFinite(intervalMs)) {\n throw new Error(`[evlog/pipeline] batch.intervalMs must be a positive finite number, got: ${intervalMs}`)\n }\n if (maxBufferSize <= 0 || !Number.isFinite(maxBufferSize)) {\n throw new Error(`[evlog/pipeline] maxBufferSize must be a positive finite number, got: ${maxBufferSize}`)\n }\n if (maxAttempts <= 0 || !Number.isFinite(maxAttempts)) {\n throw new Error(`[evlog/pipeline] retry.maxAttempts must be a positive finite number, got: ${maxAttempts}`)\n }\n if (initialDelayMs < 0 || !Number.isFinite(initialDelayMs)) {\n throw new Error(`[evlog/pipeline] retry.initialDelayMs must be a non-negative finite number, got: ${initialDelayMs}`)\n }\n if (maxDelayMs < 0 || !Number.isFinite(maxDelayMs)) {\n throw new Error(`[evlog/pipeline] retry.maxDelayMs must be a non-negative finite number, got: ${maxDelayMs}`)\n }\n\n return (drain: (batch: T[]) => void | Promise<void>): PipelineDrainFn<T> => {\n const buffer: T[] = []\n let timer: ReturnType<typeof setTimeout> | null = null\n let activeFlush: Promise<void> | null = null\n\n function clearTimer(): void {\n if (timer !== null) {\n clearTimeout(timer)\n timer = null\n }\n }\n\n function scheduleFlush(): void {\n if (timer !== null || activeFlush) return\n timer = setTimeout(() => {\n timer = null\n if (!activeFlush) startFlush()\n }, intervalMs)\n unrefTimer(timer)\n }\n\n function getRetryDelay(attempt: number): number {\n let delay: number\n switch (backoffStrategy) {\n case 'linear':\n delay = initialDelayMs * attempt\n break\n case 'fixed':\n delay = initialDelayMs\n break\n case 'exponential':\n default:\n delay = initialDelayMs * 2 ** (attempt - 1)\n break\n }\n return Math.min(delay, maxDelayMs)\n }\n\n async function sendWithRetry(batch: T[]): Promise<void> {\n let lastError: Error | undefined\n for (let attempt = 1; attempt <= maxAttempts; attempt++) {\n try {\n await drain(batch)\n return\n } catch (error) {\n lastError = error instanceof Error ? error : new Error(String(error))\n if (attempt < maxAttempts) {\n await new Promise<void>(r => setTimeout(r, getRetryDelay(attempt)))\n }\n }\n }\n onDropped?.(batch, lastError)\n }\n\n async function drainBuffer(): Promise<void> {\n while (buffer.length > 0) {\n const batch = buffer.splice(0, batchSize)\n await sendWithRetry(batch)\n }\n }\n\n function startFlush(): void {\n if (activeFlush) return\n activeFlush = drainBuffer().finally(() => {\n activeFlush = null\n if (buffer.length >= batchSize) {\n startFlush()\n } else if (buffer.length > 0) {\n scheduleFlush()\n }\n })\n }\n\n function push(ctx: T): void {\n if (buffer.length >= maxBufferSize) {\n const dropped = buffer.splice(0, 1)\n onDropped?.(dropped)\n }\n buffer.push(ctx)\n\n if (buffer.length >= batchSize) {\n clearTimer()\n startFlush()\n } else if (!activeFlush) {\n scheduleFlush()\n }\n }\n\n async function flush(): Promise<void> {\n clearTimer()\n if (activeFlush) {\n await activeFlush\n }\n // Snapshot the buffer length to avoid infinite loop if push() is called during flush\n const snapshot = buffer.length\n if (snapshot > 0) {\n const toFlush = buffer.splice(0, snapshot)\n while (toFlush.length > 0) {\n const batch = toFlush.splice(0, batchSize)\n await sendWithRetry(batch)\n }\n }\n }\n\n const hookFn = push as PipelineDrainFn<T>\n hookFn.flush = flush\n Object.defineProperty(hookFn, 'pending', {\n get: () => buffer.length,\n enumerable: true,\n })\n\n return hookFn\n }\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;AAyDA,SAAS,WAAW,OAA4C;AAC7D,OAAiC,SAAS;;AAG7C,SAAgB,oBAAiC,SAAwG;CACvJ,MAAM,YAAY,SAAS,OAAO,QAAQ;CAC1C,MAAM,aAAa,SAAS,OAAO,cAAc;CACjD,MAAM,gBAAgB,SAAS,iBAAiB;CAChD,MAAM,cAAc,SAAS,OAAO,eAAe;CACnD,MAAM,kBAAkB,SAAS,OAAO,WAAW;CACnD,MAAM,iBAAiB,SAAS,OAAO,kBAAkB;CACzD,MAAM,aAAa,SAAS,OAAO,cAAc;CACjD,MAAM,YAAY,SAAS;AAE3B,KAAI,aAAa,KAAK,CAAC,OAAO,SAAS,UAAU,CAC/C,OAAM,IAAI,MAAM,sEAAsE,YAAY;AAEpG,KAAI,cAAc,KAAK,CAAC,OAAO,SAAS,WAAW,CACjD,OAAM,IAAI,MAAM,4EAA4E,aAAa;AAE3G,KAAI,iBAAiB,KAAK,CAAC,OAAO,SAAS,cAAc,CACvD,OAAM,IAAI,MAAM,yEAAyE,gBAAgB;AAE3G,KAAI,eAAe,KAAK,CAAC,OAAO,SAAS,YAAY,CACnD,OAAM,IAAI,MAAM,6EAA6E,cAAc;AAE7G,KAAI,iBAAiB,KAAK,CAAC,OAAO,SAAS,eAAe,CACxD,OAAM,IAAI,MAAM,oFAAoF,iBAAiB;AAEvH,KAAI,aAAa,KAAK,CAAC,OAAO,SAAS,WAAW,CAChD,OAAM,IAAI,MAAM,gFAAgF,aAAa;AAG/G,SAAQ,UAAoE;EAC1E,MAAM,SAAc,EAAE;EACtB,IAAI,QAA8C;EAClD,IAAI,cAAoC;EAExC,SAAS,aAAmB;AAC1B,OAAI,UAAU,MAAM;AAClB,iBAAa,MAAM;AACnB,YAAQ;;;EAIZ,SAAS,gBAAsB;AAC7B,OAAI,UAAU,QAAQ,YAAa;AACnC,WAAQ,iBAAiB;AACvB,YAAQ;AACR,QAAI,CAAC,YAAa,aAAY;MAC7B,WAAW;AACd,cAAW,MAAM;;EAGnB,SAAS,cAAc,SAAyB;GAC9C,IAAI;AACJ,WAAQ,iBAAR;IACE,KAAK;AACH,aAAQ,iBAAiB;AACzB;IACF,KAAK;AACH,aAAQ;AACR;IAEF;AACE,aAAQ,iBAAiB,MAAM,UAAU;AACzC;;AAEJ,UAAO,KAAK,IAAI,OAAO,WAAW;;EAGpC,eAAe,cAAc,OAA2B;GACtD,IAAI;AACJ,QAAK,IAAI,UAAU,GAAG,WAAW,aAAa,UAC5C,KAAI;AACF,UAAM,MAAM,MAAM;AAClB;YACO,OAAO;AACd,gBAAY,iBAAiB,QAAQ,QAAQ,IAAI,MAAM,OAAO,MAAM,CAAC;AACrE,QAAI,UAAU,YACZ,OAAM,IAAI,SAAc,MAAK,WAAW,GAAG,cAAc,QAAQ,CAAC,CAAC;;AAIzE,eAAY,OAAO,UAAU;;EAG/B,eAAe,cAA6B;AAC1C,UAAO,OAAO,SAAS,EAErB,OAAM,cADQ,OAAO,OAAO,GAAG,UACN,CAAC;;EAI9B,SAAS,aAAmB;AAC1B,OAAI,YAAa;AACjB,iBAAc,aAAa,CAAC,cAAc;AACxC,kBAAc;AACd,QAAI,OAAO,UAAU,UACnB,aAAY;aACH,OAAO,SAAS,EACzB,gBAAe;KAEjB;;EAGJ,SAAS,KAAK,KAAc;AAC1B,OAAI,OAAO,UAAU,eAAe;IAClC,MAAM,UAAU,OAAO,OAAO,GAAG,EAAE;AACnC,gBAAY,QAAQ;;AAEtB,UAAO,KAAK,IAAI;AAEhB,OAAI,OAAO,UAAU,WAAW;AAC9B,gBAAY;AACZ,gBAAY;cACH,CAAC,YACV,gBAAe;;EAInB,eAAe,QAAuB;AACpC,eAAY;AACZ,OAAI,YACF,OAAM;GAGR,MAAM,WAAW,OAAO;AACxB,OAAI,WAAW,GAAG;IAChB,MAAM,UAAU,OAAO,OAAO,GAAG,SAAS;AAC1C,WAAO,QAAQ,SAAS,EAEtB,OAAM,cADQ,QAAQ,OAAO,GAAG,UACP,CAAC;;;EAKhC,MAAM,SAAS;AACf,SAAO,QAAQ;AACf,SAAO,eAAe,QAAQ,WAAW;GACvC,WAAW,OAAO;GAClB,YAAY;GACb,CAAC;AAEF,SAAO"}
@@ -0,0 +1,288 @@
1
+ import { colors, isBrowser, isDev } from "./utils.mjs";
2
+ //#region src/shared/pretty-error.ts
3
+ let snippetReader = null;
4
+ /**
5
+ * Register a disk-backed snippet reader (Node.js integrations only).
6
+ * @internal
7
+ */
8
+ function registerPrettyErrorSnippetReader(reader) {
9
+ snippetReader = reader;
10
+ }
11
+ /** Tree-only breathing line (connector without content). */
12
+ const PRETTY_ERROR_TREE_SPACER = "__EVLOG_TREE_SPACER__";
13
+ function pushTreeSpacer(children) {
14
+ children.push(PRETTY_ERROR_TREE_SPACER);
15
+ }
16
+ const SKIP_PATH_RE = /(?:^|[/\\])(?:node_modules|\.nuxt|\.next|\.output)(?:[/\\]|$)/;
17
+ const SKIP_FRAME_PATH_RE = /(?:^|[/\\])(?:packages[/\\]evlog|evlog[/\\](?:dist|src))(?:[/\\]|$)/;
18
+ const SKIP_FRAME_FN_RE = /^(?:createError|EvlogError|new EvlogError)$/;
19
+ function isPlainObject(val) {
20
+ return val !== null && typeof val === "object" && !Array.isArray(val);
21
+ }
22
+ function pickString(obj, key) {
23
+ const val = obj[key];
24
+ return typeof val === "string" && val.length > 0 ? val : void 0;
25
+ }
26
+ function pickNumber(obj, key) {
27
+ const val = obj[key];
28
+ return typeof val === "number" ? val : void 0;
29
+ }
30
+ function extractGuidance(data) {
31
+ return {
32
+ code: pickString(data, "code"),
33
+ why: pickString(data, "why"),
34
+ fix: pickString(data, "fix"),
35
+ link: pickString(data, "link")
36
+ };
37
+ }
38
+ /**
39
+ * Extract structured error fields from a wide-event `error` value.
40
+ */
41
+ function normalizeErrorContext(error) {
42
+ if (error === null || error === void 0) return null;
43
+ if (typeof error === "string") return { message: error };
44
+ if (!isPlainObject(error)) return { message: String(error) };
45
+ const result = {
46
+ message: pickString(error, "message") ?? pickString(error, "statusText") ?? pickString(error, "statusMessage") ?? "Unknown error",
47
+ name: pickString(error, "name"),
48
+ code: pickString(error, "code"),
49
+ why: pickString(error, "why"),
50
+ fix: pickString(error, "fix"),
51
+ link: pickString(error, "link"),
52
+ status: pickNumber(error, "status") ?? pickNumber(error, "statusCode"),
53
+ stack: pickString(error, "stack")
54
+ };
55
+ const { data, cause } = error;
56
+ if (isPlainObject(data)) {
57
+ const guidance = extractGuidance(data);
58
+ if (!result.code) result.code = guidance.code;
59
+ if (!result.why) result.why = guidance.why;
60
+ if (!result.fix) result.fix = guidance.fix;
61
+ if (!result.link) result.link = guidance.link;
62
+ }
63
+ if (cause instanceof Error) result.cause = cause.message;
64
+ else if (isPlainObject(cause) && pickString(cause, "message")) result.cause = pickString(cause, "message");
65
+ return result;
66
+ }
67
+ /** Decode a `file://` URL or path for display and snippet lookup. */
68
+ function decodeFileUrl(file) {
69
+ if (file.startsWith("file://")) try {
70
+ return decodeURIComponent(new URL(file).pathname);
71
+ } catch {
72
+ return file.slice(7);
73
+ }
74
+ return file;
75
+ }
76
+ function isAppPath(file) {
77
+ const normalized = decodeFileUrl(file).replace(/\\/g, "/");
78
+ if (normalized.startsWith("node:")) return false;
79
+ if (SKIP_PATH_RE.test(normalized)) return false;
80
+ if (normalized.includes("/node_modules/")) return false;
81
+ if (isFrameworkRuntimePath(normalized)) return false;
82
+ return true;
83
+ }
84
+ function formatDisplayPath(file, cwd) {
85
+ const decoded = decodeFileUrl(file).replace(/\\/g, "/");
86
+ const cwdNorm = cwd.replace(/\\/g, "/").replace(/\/$/, "");
87
+ if (cwdNorm && decoded.startsWith(`${cwdNorm}/`)) {
88
+ const rel = decoded.slice(cwdNorm.length + 1);
89
+ return rel.startsWith("./") ? rel.slice(2) : rel;
90
+ }
91
+ const serverIdx = decoded.indexOf("/server/");
92
+ if (serverIdx >= 0) return decoded.slice(serverIdx + 1);
93
+ const srcIdx = decoded.indexOf("/src/");
94
+ if (srcIdx >= 0) return decoded.slice(srcIdx + 1);
95
+ return decoded;
96
+ }
97
+ /**
98
+ * Compact a stack trace for wide-event storage (drains, NDJSON).
99
+ * Drops node_modules, build output, and evlog internals; keeps up to five useful frames.
100
+ * Returns the stack unchanged when no app frame survives the filter.
101
+ */
102
+ function compactStackForStorage(stack, maxFrames = 5) {
103
+ if (!stack) return void 0;
104
+ const head = stack.split("\n")[0] ?? "";
105
+ const useful = parseStackFrames(stack).filter((f) => f.file && f.line && f.isApp && !isInternalErrorFrame(f)).slice(0, maxFrames);
106
+ if (useful.length === 0) return stack;
107
+ return [head, ...useful.map((f) => f.raw)].join("\n");
108
+ }
109
+ /**
110
+ * Parse a V8 stack trace string into frames.
111
+ */
112
+ function parseStackFrames(stack) {
113
+ if (!stack) return [];
114
+ const lines = stack.split("\n");
115
+ const frames = [];
116
+ for (const line of lines) {
117
+ const trimmed = line.trim();
118
+ if (!trimmed.startsWith("at ")) continue;
119
+ const withFn = trimmed.match(/^at (.+?) \((.+):(\d+):(\d+)\)$/);
120
+ if (withFn) {
121
+ const [, fn, file, lineStr, colStr] = withFn;
122
+ frames.push({
123
+ raw: trimmed,
124
+ fn,
125
+ file,
126
+ line: Number(lineStr),
127
+ column: Number(colStr),
128
+ isApp: isAppPath(file)
129
+ });
130
+ continue;
131
+ }
132
+ const withoutFn = trimmed.match(/^at (.+):(\d+):(\d+)$/);
133
+ if (withoutFn) {
134
+ const [, file, lineStr, colStr] = withoutFn;
135
+ frames.push({
136
+ raw: trimmed,
137
+ file,
138
+ line: Number(lineStr),
139
+ column: Number(colStr),
140
+ isApp: isAppPath(file)
141
+ });
142
+ continue;
143
+ }
144
+ const asyncFn = trimmed.match(/^at async (.+?) \((.+):(\d+):(\d+)\)$/);
145
+ if (asyncFn) {
146
+ const [, fn, file, lineStr, colStr] = asyncFn;
147
+ frames.push({
148
+ raw: trimmed,
149
+ fn: `async ${fn}`,
150
+ file,
151
+ line: Number(lineStr),
152
+ column: Number(colStr),
153
+ isApp: isAppPath(file)
154
+ });
155
+ }
156
+ }
157
+ return frames;
158
+ }
159
+ /** True for Next.js runtime internals (route-modules, next/dist, bundled next-server). */
160
+ function isFrameworkRuntimePath(path) {
161
+ const normalized = path.replace(/\\/g, "/");
162
+ return normalized.includes("route-modules/") || normalized.includes("webpack://next/") || normalized.includes("/next/dist/") || normalized.includes("/compiled/next-server/");
163
+ }
164
+ function isFrameworkRuntimeFrame(frame) {
165
+ if (!frame.file) return false;
166
+ return isFrameworkRuntimePath(decodeFileUrl(frame.file));
167
+ }
168
+ function isInternalErrorFrame(frame) {
169
+ if (isFrameworkRuntimeFrame(frame)) return true;
170
+ if (frame.fn) {
171
+ const fn = frame.fn.replace(/^async /, "");
172
+ if (SKIP_FRAME_FN_RE.test(fn)) return true;
173
+ }
174
+ if (!frame.file) return true;
175
+ const path = decodeFileUrl(frame.file).replace(/\\/g, "/");
176
+ if (SKIP_FRAME_PATH_RE.test(path)) return true;
177
+ if (path.includes(".nuxt/")) return true;
178
+ if (path.includes(".next/")) return true;
179
+ return false;
180
+ }
181
+ /**
182
+ * Pick the most useful frame for code snippets (topmost app throw site).
183
+ */
184
+ function pickPrimaryFrame(frames) {
185
+ for (const frame of frames) if (frame.isApp && frame.file && frame.line && !isInternalErrorFrame(frame)) return frame;
186
+ }
187
+ /**
188
+ * Read source lines around a stack frame when a server snippet reader is registered.
189
+ */
190
+ function readCodeSnippet(file, line, contextLines = 2) {
191
+ if (!isDev() || isBrowser() || !snippetReader) return null;
192
+ const path = decodeFileUrl(file).replace(/\\/g, "/");
193
+ if (path.includes(".next/") || path.includes(".nuxt/") || path.includes(".output/")) return null;
194
+ return snippetReader(file, line, contextLines);
195
+ }
196
+ function formatSnippetLines(snippet) {
197
+ const width = String(snippet[snippet.length - 1]?.line ?? 0).length;
198
+ return snippet.map(({ line, content, isErrorLine }) => {
199
+ const marker = isErrorLine ? `${colors.red}❯${colors.reset}` : `${colors.dim} ${colors.reset}`;
200
+ const numColor = isErrorLine ? colors.red : colors.gray;
201
+ const trimmed = content.length > 120 ? `${content.slice(0, 117)}…` : content;
202
+ return `${marker} ${numColor}${String(line).padStart(width, " ")}${colors.reset} ${colors.dim}┃${colors.reset} ${colors.dim}${trimmed}${colors.reset}`;
203
+ });
204
+ }
205
+ function formatFrameLocation(frame, cwd) {
206
+ const file = frame.file ? formatDisplayPath(frame.file, cwd) : "unknown";
207
+ const loc = frame.line ? `${file}:${frame.line}` : file;
208
+ const fn = frame.fn && frame.fn !== "<unknown>" ? frame.fn : void 0;
209
+ return fn ? `at ${fn} (${loc})` : `at ${loc}`;
210
+ }
211
+ function formatCollapsedFrame(frame, cwd) {
212
+ const file = frame.file ? formatDisplayPath(frame.file, cwd) : "unknown";
213
+ const loc = frame.line ? `${file}:${frame.line}` : file;
214
+ const prefix = frame.fn?.startsWith("async") ? "at async " : "at ";
215
+ const fn = frame.fn?.replace(/^async /, "");
216
+ if (fn && fn !== "<unknown>" && fn !== loc) return `${prefix}${fn} (${loc})`;
217
+ return `${prefix}${loc}`;
218
+ }
219
+ const GUIDANCE_WRAP_WIDTH = 76;
220
+ const GUIDANCE_CONTINUATION = " ";
221
+ /** Wrap guidance text with hanging indent for long Why/Fix lines. */
222
+ function formatGuidanceLine(label, text, labelColor) {
223
+ const prefix = `${labelColor}${label}:${colors.reset} `;
224
+ const lines = [];
225
+ let remaining = text.trim();
226
+ let first = true;
227
+ while (remaining.length > 0) {
228
+ const budget = first ? Math.max(24, GUIDANCE_WRAP_WIDTH - prefix.length) : Math.max(24, GUIDANCE_WRAP_WIDTH - 5);
229
+ if (remaining.length <= budget) {
230
+ lines.push(first ? `${prefix}${remaining}` : `${GUIDANCE_CONTINUATION}${remaining}`);
231
+ break;
232
+ }
233
+ let split = remaining.lastIndexOf(" ", budget);
234
+ if (split <= 0) split = budget;
235
+ const chunk = remaining.slice(0, split).trimEnd();
236
+ lines.push(first ? `${prefix}${chunk}` : `${GUIDANCE_CONTINUATION}${chunk}`);
237
+ remaining = remaining.slice(split).trimStart();
238
+ first = false;
239
+ }
240
+ return lines;
241
+ }
242
+ /**
243
+ * Build pretty-print tree entries for a wide-event `error` field.
244
+ */
245
+ function buildErrorEntries(error, options = {}) {
246
+ const normalized = normalizeErrorContext(error);
247
+ if (!normalized) return [];
248
+ const cwd = options.cwd ?? (typeof process !== "undefined" && typeof process.cwd === "function" ? process.cwd() : ".");
249
+ const compact = options.compact ?? isDev();
250
+ const guidanceOnly = (options.detail ?? "full") === "guidance";
251
+ const showFrames = !guidanceOnly && !isBrowser() && (options.snippet ?? isDev());
252
+ const stackDepth = guidanceOnly ? 0 : options.stackDepth ?? (compact ? 2 : 3);
253
+ const snippetContextLines = compact ? 1 : 2;
254
+ const children = [];
255
+ const frames = guidanceOnly ? [] : parseStackFrames(normalized.stack);
256
+ const primary = guidanceOnly ? void 0 : pickPrimaryFrame(frames);
257
+ if (!guidanceOnly && primary?.file && primary.line) {
258
+ pushTreeSpacer(children);
259
+ if (showFrames) {
260
+ const snippet = readCodeSnippet(primary.file, primary.line, snippetContextLines);
261
+ children.push(`${colors.dim} ${formatFrameLocation(primary, cwd)}${colors.reset}`);
262
+ if (snippet) children.push(...formatSnippetLines(snippet));
263
+ } else children.push(`${colors.dim} ${formatFrameLocation(primary, cwd)}${colors.reset}`);
264
+ }
265
+ if (normalized.code) children.push(`${colors.dim}Code:${colors.reset} ${normalized.code}`);
266
+ if (Boolean(normalized.why || normalized.fix || normalized.link)) pushTreeSpacer(children);
267
+ if (normalized.why) children.push(...formatGuidanceLine("Why", normalized.why, colors.yellow));
268
+ if (normalized.fix) children.push(...formatGuidanceLine("Fix", normalized.fix, colors.cyan));
269
+ if (normalized.link) children.push(`${colors.dim}More:${colors.reset} ${normalized.link}`);
270
+ if (normalized.cause && normalized.cause !== normalized.message) children.push(`${colors.dim}Caused by:${colors.reset} ${normalized.cause}`);
271
+ const hiddenCount = guidanceOnly ? 0 : frames.filter((f) => !f.isApp || isInternalErrorFrame(f)).length;
272
+ const tailFrames = stackDepth > 0 ? frames.filter((f) => f !== primary && !isInternalErrorFrame(f)).slice(0, stackDepth) : [];
273
+ if (!guidanceOnly && (hiddenCount > 0 || tailFrames.length > 0)) {
274
+ pushTreeSpacer(children);
275
+ if (hiddenCount > 0) children.push(`${colors.gray}stack (${hiddenCount} frame${hiddenCount === 1 ? "" : "s"} hidden in node_modules)${colors.reset}`);
276
+ else children.push(`${colors.gray}stack${colors.reset}`);
277
+ for (const frame of tailFrames) children.push(`${colors.gray} ${formatCollapsedFrame(frame, cwd)}${colors.reset}`);
278
+ }
279
+ return [{
280
+ key: "error",
281
+ value: `${colors.red}${colors.bold}${normalized.message}${colors.reset}`,
282
+ children: children.length > 0 ? children : void 0
283
+ }];
284
+ }
285
+ //#endregion
286
+ export { isFrameworkRuntimePath as a, decodeFileUrl as i, buildErrorEntries as n, registerPrettyErrorSnippetReader as o, compactStackForStorage as r, PRETTY_ERROR_TREE_SPACER as t };
287
+
288
+ //# sourceMappingURL=pretty-error-THg0U0w9.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pretty-error-THg0U0w9.mjs","names":[],"sources":["../src/shared/pretty-error.ts"],"sourcesContent":["import { colors, isBrowser, isDev } from '../utils'\nimport type { ResolvedPrettyError } from './dev-terminal'\n\n/** @internal Server-only snippet reader registered by Nitro plugin or initLogger. */\ntype SnippetReader = (file: string, line: number, contextLines?: number) => CodeSnippetLine[] | null\n\nlet snippetReader: SnippetReader | null = null\n\n/**\n * Register a disk-backed snippet reader (Node.js integrations only).\n * @internal\n */\nexport function registerPrettyErrorSnippetReader(reader: SnippetReader | null): void {\n snippetReader = reader\n}\n\n/** Tree-only breathing line (connector without content). */\nexport const PRETTY_ERROR_TREE_SPACER = '__EVLOG_TREE_SPACER__'\n\nfunction pushTreeSpacer(children: string[]) {\n children.push(PRETTY_ERROR_TREE_SPACER)\n}\n\n/** Pretty-print tree node for error sections. */\nexport interface PrettyErrorTreeEntry {\n key: string\n value: string\n /** Optional ANSI color for the value (server only). */\n valueColor?: string\n children?: string[]\n}\n\n/** Normalized error fields extracted from wide-event `error` context. */\nexport interface NormalizedErrorContext {\n message: string\n name?: string\n code?: string\n why?: string\n fix?: string\n link?: string\n status?: number\n stack?: string\n cause?: string\n}\n\n/** Parsed V8 stack frame. */\nexport interface StackFrame {\n raw: string\n file?: string\n line?: number\n column?: number\n fn?: string\n /** True for application source (not node_modules / build output). */\n isApp: boolean\n}\n\n/** Options for {@link buildErrorEntries}. */\nexport type PrettyErrorOptions = Partial<ResolvedPrettyError> & {\n /** Project root for relative paths in snippets. @default process.cwd() */\n cwd?: string\n}\n\nexport interface CodeSnippetLine {\n line: number\n content: string\n isErrorLine: boolean\n}\n\nconst SKIP_PATH_RE = /(?:^|[/\\\\])(?:node_modules|\\.nuxt|\\.next|\\.output)(?:[/\\\\]|$)/\nconst SKIP_FRAME_PATH_RE = /(?:^|[/\\\\])(?:packages[/\\\\]evlog|evlog[/\\\\](?:dist|src))(?:[/\\\\]|$)/\nconst SKIP_FRAME_FN_RE = /^(?:createError|EvlogError|new EvlogError)$/\n\nfunction isPlainObject(val: unknown): val is Record<string, unknown> {\n return val !== null && typeof val === 'object' && !Array.isArray(val)\n}\n\nfunction pickString(obj: Record<string, unknown>, key: string): string | undefined {\n const val = obj[key]\n return typeof val === 'string' && val.length > 0 ? val : undefined\n}\n\nfunction pickNumber(obj: Record<string, unknown>, key: string): number | undefined {\n const val = obj[key]\n return typeof val === 'number' ? val : undefined\n}\n\nfunction extractGuidance(data: Record<string, unknown>): Pick<NormalizedErrorContext, 'code' | 'why' | 'fix' | 'link'> {\n return {\n code: pickString(data, 'code'),\n why: pickString(data, 'why'),\n fix: pickString(data, 'fix'),\n link: pickString(data, 'link'),\n }\n}\n\n/**\n * Extract structured error fields from a wide-event `error` value.\n */\nexport function normalizeErrorContext(error: unknown): NormalizedErrorContext | null {\n if (error === null || error === undefined) return null\n\n if (typeof error === 'string') {\n return { message: error }\n }\n\n if (!isPlainObject(error)) {\n return { message: String(error) }\n }\n\n const message = pickString(error, 'message')\n ?? pickString(error, 'statusText')\n ?? pickString(error, 'statusMessage')\n ?? 'Unknown error'\n\n const result: NormalizedErrorContext = {\n message,\n name: pickString(error, 'name'),\n code: pickString(error, 'code'),\n why: pickString(error, 'why'),\n fix: pickString(error, 'fix'),\n link: pickString(error, 'link'),\n status: pickNumber(error, 'status') ?? pickNumber(error, 'statusCode'),\n stack: pickString(error, 'stack'),\n }\n\n const { data, cause } = error\n if (isPlainObject(data)) {\n const guidance = extractGuidance(data)\n if (!result.code) result.code = guidance.code\n if (!result.why) result.why = guidance.why\n if (!result.fix) result.fix = guidance.fix\n if (!result.link) result.link = guidance.link\n }\n\n if (cause instanceof Error) {\n result.cause = cause.message\n } else if (isPlainObject(cause) && pickString(cause, 'message')) {\n result.cause = pickString(cause, 'message')\n }\n\n return result\n}\n\n/** Decode a `file://` URL or path for display and snippet lookup. */\nexport function decodeFileUrl(file: string): string {\n if (file.startsWith('file://')) {\n try {\n return decodeURIComponent(new URL(file).pathname)\n } catch {\n return file.slice('file://'.length)\n }\n }\n return file\n}\n\nfunction isAppPath(file: string): boolean {\n const normalized = decodeFileUrl(file).replace(/\\\\/g, '/')\n if (normalized.startsWith('node:')) return false\n if (SKIP_PATH_RE.test(normalized)) return false\n if (normalized.includes('/node_modules/')) return false\n if (isFrameworkRuntimePath(normalized)) return false\n return true\n}\n\nfunction formatDisplayPath(file: string, cwd: string): string {\n const decoded = decodeFileUrl(file).replace(/\\\\/g, '/')\n const cwdNorm = cwd.replace(/\\\\/g, '/').replace(/\\/$/, '')\n if (cwdNorm && decoded.startsWith(`${cwdNorm}/`)) {\n const rel = decoded.slice(cwdNorm.length + 1)\n return rel.startsWith('./') ? rel.slice(2) : rel\n }\n const serverIdx = decoded.indexOf('/server/')\n if (serverIdx >= 0) return decoded.slice(serverIdx + 1)\n const srcIdx = decoded.indexOf('/src/')\n if (srcIdx >= 0) return decoded.slice(srcIdx + 1)\n return decoded\n}\n\n/**\n * Compact a stack trace for wide-event storage (drains, NDJSON).\n * Drops node_modules, build output, and evlog internals; keeps up to five useful frames.\n * Returns the stack unchanged when no app frame survives the filter.\n */\nexport function compactStackForStorage(stack: string | undefined, maxFrames = 5): string | undefined {\n if (!stack) return undefined\n\n const head = stack.split('\\n')[0] ?? ''\n const frames = parseStackFrames(stack)\n const useful = frames.filter(f => f.file && f.line && f.isApp && !isInternalErrorFrame(f)).slice(0, maxFrames)\n\n if (useful.length === 0) return stack\n\n return [head, ...useful.map(f => f.raw)].join('\\n')\n}\n\n/**\n * Parse a V8 stack trace string into frames.\n */\nexport function parseStackFrames(stack: string | undefined): StackFrame[] {\n if (!stack) return []\n\n const lines = stack.split('\\n')\n const frames: StackFrame[] = []\n\n for (const line of lines) {\n const trimmed = line.trim()\n if (!trimmed.startsWith('at ')) continue\n\n const withFn = trimmed.match(/^at (.+?) \\((.+):(\\d+):(\\d+)\\)$/)\n if (withFn) {\n const [, fn, file, lineStr, colStr] = withFn\n frames.push({\n raw: trimmed,\n fn,\n file,\n line: Number(lineStr),\n column: Number(colStr),\n isApp: isAppPath(file!),\n })\n continue\n }\n\n const withoutFn = trimmed.match(/^at (.+):(\\d+):(\\d+)$/)\n if (withoutFn) {\n const [, file, lineStr, colStr] = withoutFn\n frames.push({\n raw: trimmed,\n file,\n line: Number(lineStr),\n column: Number(colStr),\n isApp: isAppPath(file!),\n })\n continue\n }\n\n const asyncFn = trimmed.match(/^at async (.+?) \\((.+):(\\d+):(\\d+)\\)$/)\n if (asyncFn) {\n const [, fn, file, lineStr, colStr] = asyncFn\n frames.push({\n raw: trimmed,\n fn: `async ${fn}`,\n file,\n line: Number(lineStr),\n column: Number(colStr),\n isApp: isAppPath(file!),\n })\n }\n }\n\n return frames\n}\n\n/** True for Next.js runtime internals (route-modules, next/dist, bundled next-server). */\nexport function isFrameworkRuntimePath(path: string): boolean {\n const normalized = path.replace(/\\\\/g, '/')\n return normalized.includes('route-modules/')\n || normalized.includes('webpack://next/')\n || normalized.includes('/next/dist/')\n || normalized.includes('/compiled/next-server/')\n}\n\nfunction isFrameworkRuntimeFrame(frame: StackFrame): boolean {\n if (!frame.file) return false\n return isFrameworkRuntimePath(decodeFileUrl(frame.file))\n}\n\nfunction isInternalErrorFrame(frame: StackFrame): boolean {\n if (isFrameworkRuntimeFrame(frame)) return true\n if (frame.fn) {\n const fn = frame.fn.replace(/^async /, '')\n if (SKIP_FRAME_FN_RE.test(fn)) return true\n }\n if (!frame.file) return true\n const path = decodeFileUrl(frame.file).replace(/\\\\/g, '/')\n if (SKIP_FRAME_PATH_RE.test(path)) return true\n if (path.includes('.nuxt/')) return true\n if (path.includes('.next/')) return true\n return false\n}\n\n/**\n * Pick the most useful frame for code snippets (topmost app throw site).\n */\nexport function pickPrimaryFrame(frames: StackFrame[]): StackFrame | undefined {\n for (const frame of frames) {\n if (frame.isApp && frame.file && frame.line && !isInternalErrorFrame(frame)) {\n return frame\n }\n }\n return undefined\n}\n\n/**\n * Read source lines around a stack frame when a server snippet reader is registered.\n */\nexport function readCodeSnippet(\n file: string,\n line: number,\n contextLines = 2,\n): CodeSnippetLine[] | null {\n if (!isDev() || isBrowser() || !snippetReader) return null\n const path = decodeFileUrl(file).replace(/\\\\/g, '/')\n if (path.includes('.next/') || path.includes('.nuxt/') || path.includes('.output/')) return null\n return snippetReader(file, line, contextLines)\n}\n\nfunction formatSnippetLines(snippet: CodeSnippetLine[]): string[] {\n const width = String(snippet[snippet.length - 1]?.line ?? 0).length\n return snippet.map(({ line, content, isErrorLine }) => {\n const marker = isErrorLine ? `${colors.red}❯${colors.reset}` : `${colors.dim} ${colors.reset}`\n const numColor = isErrorLine ? colors.red : colors.gray\n const trimmed = content.length > 120 ? `${content.slice(0, 117)}…` : content\n return `${marker} ${numColor}${String(line).padStart(width, ' ')}${colors.reset} ${colors.dim}┃${colors.reset} ${colors.dim}${trimmed}${colors.reset}`\n })\n}\n\nfunction formatFrameLocation(frame: StackFrame, cwd: string): string {\n const file = frame.file ? formatDisplayPath(frame.file, cwd) : 'unknown'\n const loc = frame.line ? `${file}:${frame.line}` : file\n const fn = frame.fn && frame.fn !== '<unknown>' ? frame.fn : undefined\n return fn ? `at ${fn} (${loc})` : `at ${loc}`\n}\n\nfunction formatCollapsedFrame(frame: StackFrame, cwd: string): string {\n const file = frame.file ? formatDisplayPath(frame.file, cwd) : 'unknown'\n const loc = frame.line ? `${file}:${frame.line}` : file\n const prefix = frame.fn?.startsWith('async') ? 'at async ' : 'at '\n const fn = frame.fn?.replace(/^async /, '')\n if (fn && fn !== '<unknown>' && fn !== loc) {\n return `${prefix}${fn} (${loc})`\n }\n return `${prefix}${loc}`\n}\n\nconst GUIDANCE_WRAP_WIDTH = 76\nconst GUIDANCE_CONTINUATION = ' '\n\n/** Wrap guidance text with hanging indent for long Why/Fix lines. */\nfunction formatGuidanceLine(label: string, text: string, labelColor: string): string[] {\n const prefix = `${labelColor}${label}:${colors.reset} `\n const lines: string[] = []\n let remaining = text.trim()\n let first = true\n\n while (remaining.length > 0) {\n const budget = first\n ? Math.max(24, GUIDANCE_WRAP_WIDTH - prefix.length)\n : Math.max(24, GUIDANCE_WRAP_WIDTH - GUIDANCE_CONTINUATION.length)\n if (remaining.length <= budget) {\n lines.push(first ? `${prefix}${remaining}` : `${GUIDANCE_CONTINUATION}${remaining}`)\n break\n }\n let split = remaining.lastIndexOf(' ', budget)\n if (split <= 0) split = budget\n const chunk = remaining.slice(0, split).trimEnd()\n lines.push(first ? `${prefix}${chunk}` : `${GUIDANCE_CONTINUATION}${chunk}`)\n remaining = remaining.slice(split).trimStart()\n first = false\n }\n\n return lines\n}\n\n/**\n * Build pretty-print tree entries for a wide-event `error` field.\n */\nexport function buildErrorEntries(\n error: unknown,\n options: PrettyErrorOptions = {},\n): PrettyErrorTreeEntry[] {\n const normalized = normalizeErrorContext(error)\n if (!normalized) return []\n\n const cwd = options.cwd ?? (typeof process !== 'undefined' && typeof process.cwd === 'function' ? process.cwd() : '.')\n const compact = options.compact ?? isDev()\n const detail = options.detail ?? 'full'\n const guidanceOnly = detail === 'guidance'\n const showFrames = !guidanceOnly && !isBrowser() && (options.snippet ?? isDev())\n const stackDepth = guidanceOnly\n ? 0\n : (options.stackDepth ?? (compact ? 2 : 3))\n const snippetContextLines = compact ? 1 : 2\n\n const children: string[] = []\n const frames = guidanceOnly ? [] : parseStackFrames(normalized.stack)\n const primary = guidanceOnly ? undefined : pickPrimaryFrame(frames)\n\n if (!guidanceOnly && primary?.file && primary.line) {\n pushTreeSpacer(children)\n if (showFrames) {\n const snippet = readCodeSnippet(primary.file, primary.line, snippetContextLines)\n children.push(`${colors.dim} ${formatFrameLocation(primary, cwd)}${colors.reset}`)\n if (snippet) {\n children.push(...formatSnippetLines(snippet))\n }\n } else {\n children.push(`${colors.dim} ${formatFrameLocation(primary, cwd)}${colors.reset}`)\n }\n }\n\n if (normalized.code) {\n children.push(`${colors.dim}Code:${colors.reset} ${normalized.code}`)\n }\n\n const hasGuidance = Boolean(normalized.why || normalized.fix || normalized.link)\n if (hasGuidance) {\n pushTreeSpacer(children)\n }\n\n if (normalized.why) {\n children.push(...formatGuidanceLine('Why', normalized.why, colors.yellow))\n }\n if (normalized.fix) {\n children.push(...formatGuidanceLine('Fix', normalized.fix, colors.cyan))\n }\n if (normalized.link) {\n children.push(`${colors.dim}More:${colors.reset} ${normalized.link}`)\n }\n\n if (normalized.cause && normalized.cause !== normalized.message) {\n children.push(`${colors.dim}Caused by:${colors.reset} ${normalized.cause}`)\n }\n\n const hiddenCount = guidanceOnly ? 0 : frames.filter(f => !f.isApp || isInternalErrorFrame(f)).length\n const tailFrames = stackDepth > 0\n ? frames.filter(f => f !== primary && !isInternalErrorFrame(f)).slice(0, stackDepth)\n : []\n\n if (!guidanceOnly && (hiddenCount > 0 || tailFrames.length > 0)) {\n pushTreeSpacer(children)\n if (hiddenCount > 0) {\n children.push(`${colors.gray}stack (${hiddenCount} frame${hiddenCount === 1 ? '' : 's'} hidden in node_modules)${colors.reset}`)\n } else {\n children.push(`${colors.gray}stack${colors.reset}`)\n }\n for (const frame of tailFrames) {\n children.push(`${colors.gray} ${formatCollapsedFrame(frame, cwd)}${colors.reset}`)\n }\n }\n\n return [\n {\n key: 'error',\n value: `${colors.red}${colors.bold}${normalized.message}${colors.reset}`,\n children: children.length > 0 ? children : undefined,\n },\n ]\n}\n"],"mappings":"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"}
@@ -0,0 +1,48 @@
1
+ import { i as decodeFileUrl } from "./pretty-error-THg0U0w9.mjs";
2
+ import "node:module";
3
+ import { resolve } from "node:path";
4
+ import { readFileSync } from "node:fs";
5
+ //#region \0rolldown/runtime.js
6
+ var __defProp = Object.defineProperty;
7
+ var __exportAll = (all, no_symbols) => {
8
+ let target = {};
9
+ for (var name in all) __defProp(target, name, {
10
+ get: all[name],
11
+ enumerable: true
12
+ });
13
+ if (!no_symbols) __defProp(target, Symbol.toStringTag, { value: "Module" });
14
+ return target;
15
+ };
16
+ //#endregion
17
+ //#region src/shared/pretty-error-snippet.node.ts
18
+ var pretty_error_snippet_node_exports = /* @__PURE__ */ __exportAll({ readCodeSnippetFromDisk: () => readCodeSnippetFromDisk });
19
+ /**
20
+ * Read source lines around a stack frame from disk (Node.js only).
21
+ */
22
+ function readCodeSnippetFromDisk(file, line, contextLines = 2) {
23
+ const decoded = decodeFileUrl(file);
24
+ let content;
25
+ try {
26
+ content = readFileSync(decoded, "utf8");
27
+ } catch {
28
+ try {
29
+ content = readFileSync(resolve(process.cwd(), decoded), "utf8");
30
+ } catch {
31
+ return null;
32
+ }
33
+ }
34
+ const lines = content.split("\n");
35
+ const start = Math.max(0, line - contextLines - 1);
36
+ const end = Math.min(lines.length, line + contextLines);
37
+ const snippet = [];
38
+ for (let i = start; i < end; i++) snippet.push({
39
+ line: i + 1,
40
+ content: lines[i] ?? "",
41
+ isErrorLine: i + 1 === line
42
+ });
43
+ return snippet.length > 0 ? snippet : null;
44
+ }
45
+ //#endregion
46
+ export { readCodeSnippetFromDisk as n, pretty_error_snippet_node_exports as t };
47
+
48
+ //# sourceMappingURL=pretty-error-snippet.node-itfCajBM.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"pretty-error-snippet.node-itfCajBM.mjs","names":[],"sources":["../src/shared/pretty-error-snippet.node.ts"],"sourcesContent":["import { readFileSync } from 'node:fs'\nimport { resolve } from 'node:path'\nimport type { CodeSnippetLine } from './pretty-error'\nimport { decodeFileUrl } from './pretty-error'\n\n/**\n * Read source lines around a stack frame from disk (Node.js only).\n */\nexport function readCodeSnippetFromDisk(\n file: string,\n line: number,\n contextLines = 2,\n): CodeSnippetLine[] | null {\n const decoded = decodeFileUrl(file)\n let content: string\n try {\n content = readFileSync(decoded, 'utf8')\n } catch {\n try {\n content = readFileSync(resolve(process.cwd(), decoded), 'utf8')\n } catch {\n return null\n }\n }\n\n const lines = content.split('\\n')\n const start = Math.max(0, line - contextLines - 1)\n const end = Math.min(lines.length, line + contextLines)\n const snippet: CodeSnippetLine[] = []\n\n for (let i = start; i < end; i++) {\n snippet.push({\n line: i + 1,\n content: lines[i] ?? '',\n isErrorLine: i + 1 === line,\n })\n }\n\n return snippet.length > 0 ? snippet : null\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;AAQA,SAAgB,wBACd,MACA,MACA,eAAe,GACW;CAC1B,MAAM,UAAU,cAAc,KAAK;CACnC,IAAI;AACJ,KAAI;AACF,YAAU,aAAa,SAAS,OAAO;SACjC;AACN,MAAI;AACF,aAAU,aAAa,QAAQ,QAAQ,KAAK,EAAE,QAAQ,EAAE,OAAO;UACzD;AACN,UAAO;;;CAIX,MAAM,QAAQ,QAAQ,MAAM,KAAK;CACjC,MAAM,QAAQ,KAAK,IAAI,GAAG,OAAO,eAAe,EAAE;CAClD,MAAM,MAAM,KAAK,IAAI,MAAM,QAAQ,OAAO,aAAa;CACvD,MAAM,UAA6B,EAAE;AAErC,MAAK,IAAI,IAAI,OAAO,IAAI,KAAK,IAC3B,SAAQ,KAAK;EACX,MAAM,IAAI;EACV,SAAS,MAAM,MAAM;EACrB,aAAa,IAAI,MAAM;EACxB,CAAC;AAGJ,QAAO,QAAQ,SAAS,IAAI,UAAU"}
@@ -1,5 +1,5 @@
1
- import { $ as RequestLogger } from "../audit-DVdkntSO.mjs";
2
- import { t as BaseEvlogOptions } from "../middleware-31KhtiEF.mjs";
1
+ import { rt as RequestLogger } from "../audit-D7v6JHj0.mjs";
2
+ import { t as BaseEvlogOptions } from "../middleware-B_k4Mrzg.mjs";
3
3
  import * as _$react_router0 from "react-router";
4
4
 
5
5
  //#region src/react-router/index.d.ts