whale-code 6.5.10 → 6.6.0

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 (627) hide show
  1. package/dist/cli/chat/ChatApp.js +7 -11
  2. package/dist/cli/chat/ChatApp.js.map +1 -1
  3. package/dist/cli/chat/ChatInput.js +7 -3
  4. package/dist/cli/chat/ChatInput.js.map +1 -1
  5. package/dist/cli/chat/MessageList.js +5 -6
  6. package/dist/cli/chat/MessageList.js.map +1 -1
  7. package/dist/cli/chat/StatusBar.d.ts +2 -2
  8. package/dist/cli/chat/StatusBar.js +90 -160
  9. package/dist/cli/chat/StatusBar.js.map +1 -1
  10. package/dist/cli/chat/components/LiveArea.js +78 -115
  11. package/dist/cli/chat/components/LiveArea.js.map +1 -1
  12. package/dist/cli/chat/components/StaticMessages.js +60 -79
  13. package/dist/cli/chat/components/StaticMessages.js.map +1 -1
  14. package/dist/cli/chat/hooks/useAgentLoop.js +45 -37
  15. package/dist/cli/chat/hooks/useAgentLoop.js.map +1 -1
  16. package/dist/cli/chat/store.d.ts +12 -0
  17. package/dist/cli/chat/store.js +19 -0
  18. package/dist/cli/chat/store.js.map +1 -1
  19. package/dist/cli/commands/doctor.js +1 -6
  20. package/dist/cli/commands/doctor.js.map +1 -1
  21. package/dist/cli/services/agent-loop-tools.js +11 -2
  22. package/dist/cli/services/agent-loop-tools.js.map +1 -1
  23. package/dist/cli/services/agent-loop.js +1 -1
  24. package/dist/cli/services/agent-loop.js.map +1 -1
  25. package/dist/cli/services/cli-agent-loop.js +3 -2
  26. package/dist/cli/services/cli-agent-loop.js.map +1 -1
  27. package/dist/cli/services/config-store.d.ts +8 -10
  28. package/dist/cli/services/config-store.js +14 -13
  29. package/dist/cli/services/config-store.js.map +1 -1
  30. package/dist/cli/services/memory-manager.js +2 -2
  31. package/dist/cli/services/memory-manager.js.map +1 -1
  32. package/dist/cli/services/permission-modes.js +14 -10
  33. package/dist/cli/services/permission-modes.js.map +1 -1
  34. package/dist/cli/services/session-client.js +2 -1
  35. package/dist/cli/services/session-client.js.map +1 -1
  36. package/dist/cli/services/session-persistence.js +14 -6
  37. package/dist/cli/services/session-persistence.js.map +1 -1
  38. package/dist/cli/setup/SetupApp.d.ts +2 -2
  39. package/dist/cli/setup/SetupApp.js +91 -254
  40. package/dist/cli/setup/SetupApp.js.map +1 -1
  41. package/dist/cli/shared/SpinnerSlot.js +4 -1
  42. package/dist/cli/shared/SpinnerSlot.js.map +1 -1
  43. package/dist/cli/status/StatusApp.js +3 -3
  44. package/dist/cli/status/StatusApp.js.map +1 -1
  45. package/dist/index.js +13 -3
  46. package/dist/index.js.map +1 -1
  47. package/dist/server/handlers/browser-lifecycle.js +10 -0
  48. package/dist/server/handlers/browser-lifecycle.js.map +1 -1
  49. package/dist/server/handlers/browser.js +16 -1
  50. package/dist/server/handlers/browser.js.map +1 -1
  51. package/dist/server/handlers/campaigns.js +11 -0
  52. package/dist/server/handlers/campaigns.js.map +1 -1
  53. package/dist/server/handlers/catalog-products.js +19 -5
  54. package/dist/server/handlers/catalog-products.js.map +1 -1
  55. package/dist/server/handlers/catalog.js +42 -8
  56. package/dist/server/handlers/catalog.js.map +1 -1
  57. package/dist/server/handlers/clickhouse.js +4 -4
  58. package/dist/server/handlers/clickhouse.js.map +1 -1
  59. package/dist/server/handlers/comms-email.js +70 -8
  60. package/dist/server/handlers/comms-email.js.map +1 -1
  61. package/dist/server/handlers/comms.js +63 -21
  62. package/dist/server/handlers/comms.js.map +1 -1
  63. package/dist/server/handlers/coupons.js +141 -77
  64. package/dist/server/handlers/coupons.js.map +1 -1
  65. package/dist/server/handlers/google-ads.js +280 -8
  66. package/dist/server/handlers/google-ads.js.map +1 -1
  67. package/dist/server/handlers/remove-bg.d.ts +33 -0
  68. package/dist/server/handlers/remove-bg.js +698 -44
  69. package/dist/server/handlers/remove-bg.js.map +1 -1
  70. package/dist/server/handlers/supply-chain.js +93 -1
  71. package/dist/server/handlers/supply-chain.js.map +1 -1
  72. package/dist/server/handlers/workflow-steps-types.d.ts +1 -1
  73. package/dist/server/handlers/workflow-steps-types.js +7 -1
  74. package/dist/server/handlers/workflow-steps-types.js.map +1 -1
  75. package/dist/server/handlers/workflow-steps.js +1 -1
  76. package/dist/server/handlers/workflow-steps.js.map +1 -1
  77. package/dist/server/index.js +122 -29
  78. package/dist/server/index.js.map +1 -1
  79. package/dist/server/lib/agent-loop-turn.js +33 -3
  80. package/dist/server/lib/agent-loop-turn.js.map +1 -1
  81. package/dist/server/lib/agent-loop-types.d.ts +6 -2
  82. package/dist/server/lib/agent-loop-types.js +14 -2
  83. package/dist/server/lib/agent-loop-types.js.map +1 -1
  84. package/dist/server/lib/clickhouse-client.js +4 -2
  85. package/dist/server/lib/clickhouse-client.js.map +1 -1
  86. package/dist/server/lib/code-worker.js +4 -1
  87. package/dist/server/lib/code-worker.js.map +1 -1
  88. package/dist/server/providers/anthropic.js +103 -33
  89. package/dist/server/providers/anthropic.js.map +1 -1
  90. package/dist/server/server-chat.js +2 -2
  91. package/dist/server/server-chat.js.map +1 -1
  92. package/dist/server/server-helpers.d.ts +8 -1
  93. package/dist/server/server-helpers.js +17 -3
  94. package/dist/server/server-helpers.js.map +1 -1
  95. package/dist/server/server-persist.js +34 -21
  96. package/dist/server/server-persist.js.map +1 -1
  97. package/dist/server/server-rate-limit.d.ts +0 -1
  98. package/dist/server/server-rate-limit.js +5 -5
  99. package/dist/server/server-rate-limit.js.map +1 -1
  100. package/dist/server/server-routes-approvals.js +2 -2
  101. package/dist/server/server-routes-approvals.js.map +1 -1
  102. package/dist/server/server-routes-auth.js +2 -2
  103. package/dist/server/server-routes-auth.js.map +1 -1
  104. package/dist/server/server-routes-events.js +2 -2
  105. package/dist/server/server-routes-events.js.map +1 -1
  106. package/dist/server/server-routes-public.js +4 -4
  107. package/dist/server/server-routes-public.js.map +1 -1
  108. package/dist/server/server-routes-webchat.js +3 -3
  109. package/dist/server/server-routes-webchat.js.map +1 -1
  110. package/dist/server/server-store-circuit-breaker.js +1 -1
  111. package/dist/server/server-store-circuit-breaker.js.map +1 -1
  112. package/dist/server/tool-router.js +7 -4
  113. package/dist/server/tool-router.js.map +1 -1
  114. package/dist/server/validation.js +11 -0
  115. package/dist/server/validation.js.map +1 -1
  116. package/dist/setup.js +5 -25
  117. package/dist/setup.js.map +1 -1
  118. package/dist/shared/api-client.js +38 -11
  119. package/dist/shared/api-client.js.map +1 -1
  120. package/package.json +12 -10
  121. package/vendor/ink/build/ink.js +68 -24
  122. package/vendor/ink/node_modules/react-devtools-core/README.md +152 -0
  123. package/vendor/ink/node_modules/react-devtools-core/backend.js +1 -0
  124. package/vendor/ink/node_modules/react-devtools-core/dist/648.chunk.js +2 -0
  125. package/vendor/ink/node_modules/react-devtools-core/dist/648.chunk.js.map +1 -0
  126. package/vendor/ink/node_modules/react-devtools-core/dist/backend.js +15691 -0
  127. package/vendor/ink/node_modules/react-devtools-core/dist/backend.js.map +1 -0
  128. package/vendor/ink/node_modules/react-devtools-core/dist/importFile.worker.worker.js +2 -0
  129. package/vendor/ink/node_modules/react-devtools-core/dist/importFile.worker.worker.js.map +1 -0
  130. package/vendor/ink/node_modules/react-devtools-core/dist/parseSourceAndMetadata.worker.worker.js +14 -0
  131. package/vendor/ink/node_modules/react-devtools-core/dist/parseSourceAndMetadata.worker.worker.js.map +1 -0
  132. package/vendor/ink/node_modules/react-devtools-core/dist/standalone.js +2 -0
  133. package/vendor/ink/node_modules/react-devtools-core/dist/standalone.js.map +1 -0
  134. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/LICENSE +21 -0
  135. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/README.md +495 -0
  136. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/browser.js +8 -0
  137. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/index.js +10 -0
  138. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/buffer-util.js +129 -0
  139. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/constants.js +10 -0
  140. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/event-target.js +184 -0
  141. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/extension.js +223 -0
  142. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/limiter.js +55 -0
  143. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/permessage-deflate.js +518 -0
  144. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/receiver.js +607 -0
  145. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/sender.js +409 -0
  146. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/stream.js +180 -0
  147. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/validation.js +104 -0
  148. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/websocket-server.js +449 -0
  149. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/websocket.js +1197 -0
  150. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/package.json +56 -0
  151. package/vendor/ink/node_modules/react-devtools-core/package.json +38 -0
  152. package/vendor/ink/node_modules/react-devtools-core/standalone.js +1 -0
  153. package/dist/cli/__tests__/print-mode-streaming.test.js +0 -270
  154. package/dist/cli/__tests__/print-mode.basic-output.test.js +0 -230
  155. package/dist/cli/__tests__/print-mode.session-errors.test.js +0 -252
  156. package/dist/cli/__tests__/print-mode.test.js +0 -273
  157. package/dist/cli/__tests__/serve-mode-messages.test.js +0 -338
  158. package/dist/cli/__tests__/serve-mode.messages.part2.test.js +0 -266
  159. package/dist/cli/__tests__/serve-mode.messages.test.js +0 -277
  160. package/dist/cli/__tests__/serve-mode.startup-http.test.js +0 -279
  161. package/dist/cli/__tests__/serve-mode.test.js +0 -345
  162. package/dist/cli/chat/NodeManager.d.ts +0 -30
  163. package/dist/cli/chat/NodeManager.js +0 -66
  164. package/dist/cli/chat/NodeManager.js.map +0 -1
  165. package/dist/cli/chat/chat-input-menu-handler.d.ts +0 -32
  166. package/dist/cli/chat/hooks/slash-imsg-handlers.js +0 -148
  167. package/dist/cli/chat/hooks/slash-imsg-handlers.js.map +0 -1
  168. package/dist/cli/chat/hooks/useStreamingReducer.d.ts +0 -66
  169. package/dist/cli/commands/__tests__/config-cmd.test.js +0 -270
  170. package/dist/cli/commands/__tests__/doctor.test.js +0 -257
  171. package/dist/cli/commands/__tests__/imsg-node-bridge.test.js +0 -99
  172. package/dist/cli/commands/__tests__/imsg-utils.test.js +0 -73
  173. package/dist/cli/commands/__tests__/init.test.js +0 -214
  174. package/dist/cli/commands/__tests__/mcp.test.js +0 -287
  175. package/dist/cli/commands/imsg-watcher-helpers.d.ts +0 -40
  176. package/dist/cli/commands/imsg-watcher-helpers.js +0 -184
  177. package/dist/cli/commands/imsg-watcher-helpers.js.map +0 -1
  178. package/dist/cli/commands/imsg-watcher.d.ts +0 -11
  179. package/dist/cli/commands/imsg-watcher.js +0 -230
  180. package/dist/cli/commands/imsg-watcher.js.map +0 -1
  181. package/dist/cli/services/__tests__/agent-definitions.test.js +0 -153
  182. package/dist/cli/services/__tests__/agent-events-global.test.js +0 -39
  183. package/dist/cli/services/__tests__/agent-events.part2.test.js +0 -113
  184. package/dist/cli/services/__tests__/agent-events.test.js +0 -157
  185. package/dist/cli/services/__tests__/agent-loop-auth.test.js +0 -392
  186. package/dist/cli/services/__tests__/agent-loop-budget.test.js +0 -389
  187. package/dist/cli/services/__tests__/agent-loop-tools-lifecycle.test.js +0 -430
  188. package/dist/cli/services/__tests__/agent-loop-tools-maxturns.test.js +0 -486
  189. package/dist/cli/services/__tests__/agent-loop-utils-execution.test.js +0 -528
  190. package/dist/cli/services/__tests__/agent-loop-utils-helpers.test.js +0 -466
  191. package/dist/cli/services/__tests__/agent-worker-base-execute.test.js +0 -257
  192. package/dist/cli/services/__tests__/agent-worker-base-helpers.test.js +0 -198
  193. package/dist/cli/services/__tests__/agent-worker-base.test.js +0 -278
  194. package/dist/cli/services/__tests__/auth-service-exports.test.js +0 -41
  195. package/dist/cli/services/__tests__/auth-service.part2.test.js +0 -169
  196. package/dist/cli/services/__tests__/auth-service.test.js +0 -242
  197. package/dist/cli/services/__tests__/background-processes.test.js +0 -282
  198. package/dist/cli/services/__tests__/claude-md-loader.test.js +0 -134
  199. package/dist/cli/services/__tests__/config-store.test.js +0 -247
  200. package/dist/cli/services/__tests__/debug-log.test.js +0 -199
  201. package/dist/cli/services/__tests__/edge-cases-caching.test.js +0 -174
  202. package/dist/cli/services/__tests__/edge-cases-compaction-core.test.js +0 -226
  203. package/dist/cli/services/__tests__/edge-cases-compaction-openai.test.js +0 -152
  204. package/dist/cli/services/__tests__/edge-cases-compaction-shapes.test.js +0 -53
  205. package/dist/cli/services/__tests__/edge-cases-compaction-thinking.test.js +0 -226
  206. package/dist/cli/services/__tests__/edge-cases-compaction.test.js +0 -131
  207. package/dist/cli/services/__tests__/edge-cases-paths.test.js +0 -86
  208. package/dist/cli/services/__tests__/error-logger-messages.test.js +0 -81
  209. package/dist/cli/services/__tests__/error-logger-transport.test.js +0 -119
  210. package/dist/cli/services/__tests__/error-logger.test.js +0 -264
  211. package/dist/cli/services/__tests__/file-history.test.js +0 -136
  212. package/dist/cli/services/__tests__/git-context-cache-reset.test.js +0 -223
  213. package/dist/cli/services/__tests__/git-context.test.js +0 -241
  214. package/dist/cli/services/__tests__/interactive-tools-execute.test.js +0 -166
  215. package/dist/cli/services/__tests__/interactive-tools-plan.test.js +0 -197
  216. package/dist/cli/services/__tests__/interactive-tools.part2.test.js +0 -168
  217. package/dist/cli/services/__tests__/interactive-tools.test.js +0 -179
  218. package/dist/cli/services/__tests__/keybinding-manager.test.js +0 -205
  219. package/dist/cli/services/__tests__/local-tools-dispatch.test.js +0 -404
  220. package/dist/cli/services/__tests__/local-tools.test.js +0 -238
  221. package/dist/cli/services/__tests__/lsp-manager.test.js +0 -364
  222. package/dist/cli/services/__tests__/mcp-client-connect-disconnect.test.js +0 -310
  223. package/dist/cli/services/__tests__/mcp-client.test.js +0 -93
  224. package/dist/cli/services/__tests__/memory-manager.test.js +0 -154
  225. package/dist/cli/services/__tests__/model-manager-utils.test.js +0 -154
  226. package/dist/cli/services/__tests__/model-manager.test.js +0 -175
  227. package/dist/cli/services/__tests__/permission-modes.test.js +0 -222
  228. package/dist/cli/services/__tests__/ripgrep.test.js +0 -328
  229. package/dist/cli/services/__tests__/server-tools-execute.test.js +0 -317
  230. package/dist/cli/services/__tests__/server-tools.test.js +0 -272
  231. package/dist/cli/services/__tests__/session-persistence.test.js +0 -245
  232. package/dist/cli/services/__tests__/subagent-basic.test.js +0 -489
  233. package/dist/cli/services/__tests__/subagent-edge.test.js +0 -545
  234. package/dist/cli/services/__tests__/subagent-prompts.test.js +0 -558
  235. package/dist/cli/services/__tests__/subagent-worker-errors.test.js +0 -255
  236. package/dist/cli/services/__tests__/subagent-worker.test.js +0 -242
  237. package/dist/cli/services/__tests__/system-prompt.test.js +0 -210
  238. package/dist/cli/services/__tests__/team-lead-comms-messaging.test.js +0 -250
  239. package/dist/cli/services/__tests__/team-lead-comms-result.test.js +0 -232
  240. package/dist/cli/services/__tests__/team-lead-comms-stop.test.js +0 -344
  241. package/dist/cli/services/__tests__/team-lead-comms.test.js +0 -285
  242. package/dist/cli/services/__tests__/team-lead-create.test.js +0 -327
  243. package/dist/cli/services/__tests__/team-lead-run.test.js +0 -318
  244. package/dist/cli/services/__tests__/team-lead-stop.test.js +0 -199
  245. package/dist/cli/services/__tests__/team-state-comms.test.js +0 -240
  246. package/dist/cli/services/__tests__/team-state-core.test.js +0 -230
  247. package/dist/cli/services/__tests__/team-state-tasks-complete-fail-available.test.js +0 -224
  248. package/dist/cli/services/__tests__/team-state-tasks.test.js +0 -184
  249. package/dist/cli/services/__tests__/telemetry-ai-metadata.test.js +0 -116
  250. package/dist/cli/services/__tests__/telemetry.part2.test.js +0 -195
  251. package/dist/cli/services/__tests__/telemetry.test.js +0 -176
  252. package/dist/cli/services/agent-loop-iteration.d.ts +0 -13
  253. package/dist/cli/services/agent-loop-setup.d.ts +0 -32
  254. package/dist/cli/services/agent-worker-base-api.d.ts +0 -19
  255. package/dist/cli/services/agent-worker-base-helpers.d.ts +0 -27
  256. package/dist/cli/services/agent-worker-base-tools.d.ts +0 -16
  257. package/dist/cli/services/agent-worker-base-types.d.ts +0 -81
  258. package/dist/cli/services/background-agents.d.ts +0 -26
  259. package/dist/cli/services/background-processes-ops.d.ts +0 -24
  260. package/dist/cli/services/background-tool-defs.d.ts +0 -50
  261. package/dist/cli/services/config-modules-model.test.js +0 -133
  262. package/dist/cli/services/config-modules-permission.test.js +0 -85
  263. package/dist/cli/services/config-modules-permissions.test.js +0 -85
  264. package/dist/cli/services/config-modules-session.test.js +0 -297
  265. package/dist/cli/services/format-server-response-columns.test.js +0 -265
  266. package/dist/cli/services/format-server-response-fallback.test.js +0 -65
  267. package/dist/cli/services/format-server-response-primitives-basic.test.js +0 -261
  268. package/dist/cli/services/format-server-response-primitives-nested.test.js +0 -188
  269. package/dist/cli/services/format-server-response-primitives.test.js +0 -300
  270. package/dist/cli/services/format-server-response-realworld.test.js +0 -248
  271. package/dist/cli/services/format-server-response-values.test.js +0 -247
  272. package/dist/cli/services/hooks-runners.test.js +0 -184
  273. package/dist/cli/services/hooks.glob-load.test.js +0 -233
  274. package/dist/cli/services/hooks.run-hooks.test.js +0 -184
  275. package/dist/cli/services/hooks.test.js +0 -233
  276. package/dist/cli/services/ink-incremental.d.ts +0 -19
  277. package/dist/cli/services/ink-incremental.js +0 -59
  278. package/dist/cli/services/ink-incremental.js.map +0 -1
  279. package/dist/cli/services/ink-resize-fix.d.ts +0 -18
  280. package/dist/cli/services/ink-resize-fix.js +0 -76
  281. package/dist/cli/services/ink-resize-fix.js.map +0 -1
  282. package/dist/cli/services/ink-sync-output.d.ts +0 -12
  283. package/dist/cli/services/ink-sync-output.js +0 -16
  284. package/dist/cli/services/ink-sync-output.js.map +0 -1
  285. package/dist/cli/services/interactive-tool-defs.d.ts +0 -80
  286. package/dist/cli/services/local-tools-definitions.d.ts +0 -6
  287. package/dist/cli/services/local-tools-files.test.js +0 -256
  288. package/dist/cli/services/local-tools-read-many.d.ts +0 -6
  289. package/dist/cli/services/model-router.test.js +0 -245
  290. package/dist/cli/services/rewind-rewindTo.test.js +0 -202
  291. package/dist/cli/services/rewind.test.js +0 -175
  292. package/dist/cli/services/sandbox.test.js +0 -198
  293. package/dist/cli/services/subagent-execution.d.ts +0 -12
  294. package/dist/cli/services/team-lead-auto.d.ts +0 -11
  295. package/dist/cli/services/team-lead-execution.d.ts +0 -28
  296. package/dist/cli/services/teammate-loop.js +0 -557
  297. package/dist/cli/services/teammate-loop.js.map +0 -1
  298. package/dist/cli/services/tools/__tests__/agent-tools-tasks-teams.test.js +0 -250
  299. package/dist/cli/services/tools/__tests__/agent-tools-teams.test.js +0 -200
  300. package/dist/cli/services/tools/__tests__/agent-tools.test.js +0 -340
  301. package/dist/cli/services/tools/__tests__/file-ops-cache.test.js +0 -152
  302. package/dist/cli/services/tools/__tests__/file-ops-notebook.test.js +0 -249
  303. package/dist/cli/services/tools/__tests__/file-ops-read.test.js +0 -261
  304. package/dist/cli/services/tools/__tests__/file-ops-write.test.js +0 -292
  305. package/dist/cli/services/tools/__tests__/search-tools-rg.test.js +0 -92
  306. package/dist/cli/services/tools/__tests__/search-tools.part2.test.js +0 -174
  307. package/dist/cli/services/tools/__tests__/search-tools.test.js +0 -227
  308. package/dist/cli/services/tools/__tests__/shell-exec-allowed-core.test.js +0 -163
  309. package/dist/cli/services/tools/__tests__/shell-exec-allowed-extended.test.js +0 -220
  310. package/dist/cli/services/tools/__tests__/shell-exec-allowed.part2.test.js +0 -215
  311. package/dist/cli/services/tools/__tests__/shell-exec-allowed.test.js +0 -154
  312. package/dist/cli/services/tools/__tests__/shell-exec-blocked.test.js +0 -132
  313. package/dist/cli/services/tools/__tests__/shell-exec-execution.test.js +0 -245
  314. package/dist/cli/services/tools/__tests__/task-manager-create.test.js +0 -110
  315. package/dist/cli/services/tools/__tests__/task-manager-crud.test.js +0 -339
  316. package/dist/cli/services/tools/__tests__/task-manager-list-get.test.js +0 -343
  317. package/dist/cli/services/tools/__tests__/task-manager-query.test.js +0 -346
  318. package/dist/cli/services/tools/__tests__/task-manager-routing.test.js +0 -58
  319. package/dist/cli/services/tools/__tests__/task-manager-update.test.js +0 -224
  320. package/dist/cli/services/tools/__tests__/task-manager.test.js +0 -159
  321. package/dist/cli/services/tools/__tests__/web-tools-html-search.test.js +0 -227
  322. package/dist/cli/services/tools/__tests__/web-tools.test.js +0 -285
  323. package/dist/cli/services/tools/shell-exec.test.js +0 -148
  324. package/dist/cli/shared/SharedTick.d.ts +0 -10
  325. package/dist/cli/shared/__tests__/markdown.test.js +0 -188
  326. package/dist/local-agent/__tests__/connection-disconnect.test.js +0 -201
  327. package/dist/local-agent/__tests__/connection-lifecycle.test.js +0 -289
  328. package/dist/local-agent/__tests__/connection-msghandling.test.js +0 -311
  329. package/dist/local-agent/__tests__/connection-reconnect.test.js +0 -230
  330. package/dist/local-agent/__tests__/connection-toolexec.test.js +0 -253
  331. package/dist/local-agent/__tests__/discovery.test.js +0 -328
  332. package/dist/local-agent/__tests__/executor-background.test.js +0 -219
  333. package/dist/local-agent/__tests__/executor-exec.test.js +0 -221
  334. package/dist/local-agent/__tests__/executor-jobs-sessions.test.js +0 -220
  335. package/dist/local-agent/__tests__/executor-system-info.test.js +0 -133
  336. package/dist/local-agent/__tests__/executor-systeminfo.test.js +0 -109
  337. package/dist/local-agent/__tests__/executor.test.js +0 -235
  338. package/dist/local-agent/__tests__/index.test.js +0 -139
  339. package/dist/node/__tests__/cli-channels.test.js +0 -293
  340. package/dist/node/__tests__/cli-config-edge.test.js +0 -154
  341. package/dist/node/__tests__/cli-config.test.js +0 -215
  342. package/dist/node/__tests__/config.test.js +0 -292
  343. package/dist/node/__tests__/runtime-heartbeat.test.js +0 -153
  344. package/dist/node/__tests__/runtime-lifecycle-init.test.js +0 -263
  345. package/dist/node/__tests__/runtime-lifecycle-stats.test.js +0 -180
  346. package/dist/node/__tests__/runtime-lifecycle.test.js +0 -305
  347. package/dist/node/__tests__/runtime-relay.test.js +0 -341
  348. package/dist/node/adapters/__tests__/base.test.js +0 -286
  349. package/dist/node/adapters/__tests__/discord.test.js +0 -284
  350. package/dist/node/adapters/__tests__/email-send.test.js +0 -295
  351. package/dist/node/adapters/__tests__/email.inbound-send.test.js +0 -217
  352. package/dist/node/adapters/__tests__/email.lifecycle.test.js +0 -211
  353. package/dist/node/adapters/__tests__/email.test.js +0 -290
  354. package/dist/node/adapters/__tests__/email.webhook-send.test.js +0 -251
  355. package/dist/node/adapters/__tests__/imessage-filter.test.js +0 -183
  356. package/dist/node/adapters/__tests__/imessage-lifecycle.test.js +0 -215
  357. package/dist/node/adapters/__tests__/imessage-send-restart.test.js +0 -227
  358. package/dist/node/adapters/__tests__/slack.part2.test.js +0 -135
  359. package/dist/node/adapters/__tests__/slack.test.js +0 -241
  360. package/dist/node/adapters/__tests__/sms-extras.test.js +0 -108
  361. package/dist/node/adapters/__tests__/sms-lifecycle.test.js +0 -203
  362. package/dist/node/adapters/__tests__/sms-messaging.test.js +0 -266
  363. package/dist/node/adapters/__tests__/sms.part2.test.js +0 -174
  364. package/dist/node/adapters/__tests__/sms.test.js +0 -253
  365. package/dist/node/adapters/__tests__/telegram-polling.test.js +0 -256
  366. package/dist/node/adapters/__tests__/telegram-send.test.js +0 -166
  367. package/dist/node/adapters/__tests__/webchat-inbound.test.js +0 -188
  368. package/dist/node/adapters/__tests__/webchat-outbound.test.js +0 -178
  369. package/dist/node/adapters/__tests__/whatsapp-inbound.test.js +0 -200
  370. package/dist/node/adapters/__tests__/whatsapp-send.test.js +0 -212
  371. package/dist/node/adapters/__tests__/whatsapp.test.js +0 -280
  372. package/dist/server/__tests__/gateway-fast-fail.test.js +0 -160
  373. package/dist/server/__tests__/local-agent-gateway.test.js +0 -186
  374. package/dist/server/__tests__/proxy-handlers-delegation.test.js +0 -240
  375. package/dist/server/__tests__/proxy-handlers-validation.test.js +0 -211
  376. package/dist/server/__tests__/proxy-handlers.part2.test.js +0 -240
  377. package/dist/server/__tests__/proxy-handlers.test.js +0 -213
  378. package/dist/server/__tests__/strip-base64-e2e.test.js +0 -303
  379. package/dist/server/__tests__/strip-base64.test.js +0 -256
  380. package/dist/server/__tests__/tool-router-agent-tools.test.js +0 -324
  381. package/dist/server/__tests__/tool-router-execute-core.test.js +0 -357
  382. package/dist/server/__tests__/tool-router-execute-permissions.test.js +0 -332
  383. package/dist/server/__tests__/tool-router-execute.test.js +0 -348
  384. package/dist/server/__tests__/tool-router-load.test.js +0 -432
  385. package/dist/server/__tests__/tool-router-permissions.test.js +0 -359
  386. package/dist/server/__tests__/tool-router-registry-cache.test.js +0 -383
  387. package/dist/server/__tests__/tool-router-registry-handlers.test.js +0 -272
  388. package/dist/server/__tests__/tool-router-registry.test.js +0 -331
  389. package/dist/server/__tests__/validation-inventory.test.js +0 -250
  390. package/dist/server/__tests__/validation-misc.test.js +0 -243
  391. package/dist/server/__tests__/validation-supply-chain.test.js +0 -188
  392. package/dist/server/__tests__/worker.test.js +0 -265
  393. package/dist/server/handlers/__tests__/conversation-lock.test.js +0 -117
  394. package/dist/server/handlers/__tests__/e2e/auth-cross-platform-login.e2e.test.js +0 -268
  395. package/dist/server/handlers/__tests__/e2e/auth-cross-platform-tokens.e2e.test.js +0 -264
  396. package/dist/server/handlers/__tests__/e2e/email-pipeline-send.e2e.test.js +0 -214
  397. package/dist/server/handlers/__tests__/e2e/email-pipeline-threads.e2e.test.js +0 -168
  398. package/dist/server/handlers/__tests__/e2e/error-logging-pipeline-dedup.e2e.test.js +0 -229
  399. package/dist/server/handlers/__tests__/e2e/error-logging-pipeline.e2e.test.js +0 -239
  400. package/dist/server/handlers/__tests__/e2e/error-logging-rate-limit.e2e.test.js +0 -150
  401. package/dist/server/handlers/__tests__/e2e/inventory-sync-guards.e2e.test.js +0 -177
  402. package/dist/server/handlers/__tests__/e2e/inventory-sync.e2e.test.js +0 -228
  403. package/dist/server/handlers/__tests__/e2e/inventory-sync.part2.e2e.test.js +0 -188
  404. package/dist/server/handlers/__tests__/e2e/order-lifecycle-fulfillment.e2e.test.js +0 -295
  405. package/dist/server/handlers/__tests__/e2e/order-lifecycle.e2e.test.js +0 -277
  406. package/dist/server/handlers/__tests__/e2e/order-lifecycle.fulfillment.e2e.test.js +0 -307
  407. package/dist/server/handlers/__tests__/e2e/order-lifecycle.setup.e2e.test.js +0 -177
  408. package/dist/server/handlers/__tests__/e2e/storefront-checkout-cart.e2e.test.js +0 -255
  409. package/dist/server/handlers/__tests__/e2e/storefront-checkout-webhook.e2e.test.js +0 -231
  410. package/dist/server/handlers/__tests__/e2e/workflow-execution-failures.e2e.test.js +0 -235
  411. package/dist/server/handlers/__tests__/e2e/workflow-execution.e2e.test.js +0 -294
  412. package/dist/server/handlers/__tests__/e2e/workflow-security.e2e.test.js +0 -311
  413. package/dist/server/handlers/__tests__/e2e/workflow-security.part2.e2e.test.js +0 -267
  414. package/dist/server/handlers/__tests__/workflow-cache.test.js +0 -237
  415. package/dist/server/handlers/analytics-errors-edge.test.js +0 -173
  416. package/dist/server/handlers/analytics.test.js +0 -280
  417. package/dist/server/handlers/api-docs-examples-ext.d.ts +0 -9
  418. package/dist/server/handlers/api-docs-examples-ext.js +0 -278
  419. package/dist/server/handlers/api-docs-examples-ext.js.map +0 -1
  420. package/dist/server/handlers/api-docs-examples.d.ts +0 -8
  421. package/dist/server/handlers/api-docs-examples.js +0 -221
  422. package/dist/server/handlers/api-docs-examples.js.map +0 -1
  423. package/dist/server/handlers/api-docs-sections-ext.d.ts +0 -2
  424. package/dist/server/handlers/api-docs-sections-ext.js +0 -497
  425. package/dist/server/handlers/api-docs-sections-ext.js.map +0 -1
  426. package/dist/server/handlers/api-docs-sections.d.ts +0 -21
  427. package/dist/server/handlers/api-docs-sections.js +0 -293
  428. package/dist/server/handlers/api-docs-sections.js.map +0 -1
  429. package/dist/server/handlers/api-keys.part2.test.js +0 -157
  430. package/dist/server/handlers/api-keys.test.js +0 -161
  431. package/dist/server/handlers/billing-routes.test.js +0 -123
  432. package/dist/server/handlers/billing.test.js +0 -215
  433. package/dist/server/handlers/browser-actions-errors.test.js +0 -94
  434. package/dist/server/handlers/browser-actions.part2.test.js +0 -190
  435. package/dist/server/handlers/browser-actions.test.js +0 -190
  436. package/dist/server/handlers/browser-validation.test.js +0 -257
  437. package/dist/server/handlers/catalog.test.js +0 -297
  438. package/dist/server/handlers/comms.test.js +0 -289
  439. package/dist/server/handlers/creations-advanced-collections.test.js +0 -214
  440. package/dist/server/handlers/creations-advanced-generate.test.js +0 -142
  441. package/dist/server/handlers/creations-advanced.test.js +0 -171
  442. package/dist/server/handlers/creations-collections-preview.test.js +0 -214
  443. package/dist/server/handlers/creations-crud.test.js +0 -260
  444. package/dist/server/handlers/creations-mutations.test.js +0 -197
  445. package/dist/server/handlers/crm.test.js +0 -179
  446. package/dist/server/handlers/discovery-advertise.test.js +0 -185
  447. package/dist/server/handlers/discovery-scan.test.js +0 -233
  448. package/dist/server/handlers/embeddings-embed-search.test.js +0 -196
  449. package/dist/server/handlers/embeddings-index-delete-stats.test.js +0 -140
  450. package/dist/server/handlers/embeddings-search.test.js +0 -221
  451. package/dist/server/handlers/embeddings.test.js +0 -137
  452. package/dist/server/handlers/enrichment-breach.d.ts +0 -8
  453. package/dist/server/handlers/enrichment-breach.js +0 -266
  454. package/dist/server/handlers/enrichment-breach.js.map +0 -1
  455. package/dist/server/handlers/enrichment-data.d.ts +0 -13
  456. package/dist/server/handlers/enrichment-data.js +0 -145
  457. package/dist/server/handlers/enrichment-data.js.map +0 -1
  458. package/dist/server/handlers/enrichment-mutations.test.js +0 -240
  459. package/dist/server/handlers/enrichment-queries.test.js +0 -181
  460. package/dist/server/handlers/enrichment-validation.test.js +0 -177
  461. package/dist/server/handlers/enrichment-writes.d.ts +0 -16
  462. package/dist/server/handlers/enrichment-writes.js +0 -226
  463. package/dist/server/handlers/enrichment-writes.js.map +0 -1
  464. package/dist/server/handlers/image-gen.test.js +0 -205
  465. package/dist/server/handlers/inventory.test.js +0 -380
  466. package/dist/server/handlers/kali-background.test.js +0 -222
  467. package/dist/server/handlers/kali-errors.test.js +0 -92
  468. package/dist/server/handlers/kali-validation.test.js +0 -234
  469. package/dist/server/handlers/llm-providers-actions.test.js +0 -220
  470. package/dist/server/handlers/llm-providers-anthropic.test.js +0 -239
  471. package/dist/server/handlers/llm-providers-failover.test.js +0 -232
  472. package/dist/server/handlers/llm-providers-providers.test.js +0 -300
  473. package/dist/server/handlers/llm-providers-validation.test.js +0 -239
  474. package/dist/server/handlers/local-agent-tools.test.js +0 -224
  475. package/dist/server/handlers/local-agent.test.js +0 -198
  476. package/dist/server/handlers/local-agent.tools-status.test.js +0 -204
  477. package/dist/server/handlers/local-agent.validation-exec.test.js +0 -182
  478. package/dist/server/handlers/meta-ads-audience-rules.test.js +0 -243
  479. package/dist/server/handlers/meta-ads-audience-targeting.test.js +0 -205
  480. package/dist/server/handlers/meta-ads-audiences-targeting.test.js +0 -383
  481. package/dist/server/handlers/meta-ads-crud-ads.test.js +0 -136
  482. package/dist/server/handlers/meta-ads-crud-campaigns.test.js +0 -189
  483. package/dist/server/handlers/meta-ads-crud-create.test.js +0 -303
  484. package/dist/server/handlers/meta-ads-crud-list-update.test.js +0 -259
  485. package/dist/server/handlers/meta-ads-delete-publish-sync.test.js +0 -282
  486. package/dist/server/handlers/meta-ads-insights.test.js +0 -80
  487. package/dist/server/handlers/meta-ads-list-get.test.js +0 -237
  488. package/dist/server/handlers/meta-ads-publish-delete.test.js +0 -254
  489. package/dist/server/handlers/meta-ads-publish-helpers.js +0 -117
  490. package/dist/server/handlers/meta-ads-publish-helpers.js.map +0 -1
  491. package/dist/server/handlers/meta-ads-publish-sync.test.js +0 -205
  492. package/dist/server/handlers/meta-ads-publish.test.js +0 -254
  493. package/dist/server/handlers/meta-ads-sync-insights.test.js +0 -184
  494. package/dist/server/handlers/meta-ads-update.test.js +0 -117
  495. package/dist/server/handlers/nodes-channels.test.js +0 -413
  496. package/dist/server/handlers/nodes-events.test.js +0 -131
  497. package/dist/server/handlers/nodes-list-delete.test.js +0 -171
  498. package/dist/server/handlers/nodes-messages-delivery.test.js +0 -208
  499. package/dist/server/handlers/nodes-messages.test.js +0 -211
  500. package/dist/server/handlers/nodes-register.test.js +0 -277
  501. package/dist/server/handlers/nodes.test.js +0 -353
  502. package/dist/server/handlers/operations.test.js +0 -136
  503. package/dist/server/handlers/platform-telemetry.test.js +0 -200
  504. package/dist/server/handlers/platform-websearch.test.js +0 -160
  505. package/dist/server/handlers/storefront.test.js +0 -329
  506. package/dist/server/handlers/supply-chain.test.js +0 -347
  507. package/dist/server/handlers/transcription.test.js +0 -118
  508. package/dist/server/handlers/video-gen-veo.js +0 -114
  509. package/dist/server/handlers/video-gen-veo.js.map +0 -1
  510. package/dist/server/handlers/video-gen.test.js +0 -146
  511. package/dist/server/handlers/voice.test.js +0 -153
  512. package/dist/server/handlers/workflow-steps.test.js +0 -330
  513. package/dist/server/handlers/workflows-extras.test.js +0 -65
  514. package/dist/server/handlers/workflows.part2.test.js +0 -170
  515. package/dist/server/handlers/workflows.test.js +0 -281
  516. package/dist/server/lib/__tests__/batch-client-conversion-jsonl.test.js +0 -171
  517. package/dist/server/lib/__tests__/batch-client-polling.test.js +0 -292
  518. package/dist/server/lib/__tests__/batch-client-queue.test.js +0 -270
  519. package/dist/server/lib/__tests__/clickhouse-buffer.test.js +0 -236
  520. package/dist/server/lib/__tests__/code-worker-edge-cases.test.js +0 -118
  521. package/dist/server/lib/__tests__/code-worker-pool-execute.test.js +0 -193
  522. package/dist/server/lib/__tests__/code-worker-pool-execution.test.js +0 -165
  523. package/dist/server/lib/__tests__/code-worker-pool-init.test.js +0 -131
  524. package/dist/server/lib/__tests__/code-worker-pool.test.js +0 -194
  525. package/dist/server/lib/__tests__/code-worker-sandbox-ops.test.js +0 -123
  526. package/dist/server/lib/__tests__/code-worker-sandbox.test.js +0 -217
  527. package/dist/server/lib/__tests__/code-worker.test.js +0 -179
  528. package/dist/server/lib/__tests__/compaction-service-generate.test.js +0 -229
  529. package/dist/server/lib/__tests__/compaction-service.test.js +0 -319
  530. package/dist/server/lib/__tests__/otel.test.js +0 -146
  531. package/dist/server/lib/__tests__/prompt-sanitizer-validation.test.js +0 -165
  532. package/dist/server/lib/__tests__/prompt-sanitizer.sanitize.test.js +0 -343
  533. package/dist/server/lib/__tests__/prompt-sanitizer.test.js +0 -328
  534. package/dist/server/lib/__tests__/prompt-sanitizer.validate-tool.test.js +0 -145
  535. package/dist/server/lib/__tests__/provider-capabilities.test.js +0 -263
  536. package/dist/server/lib/__tests__/provider-failover-routing.test.js +0 -145
  537. package/dist/server/lib/__tests__/provider-failover-state.test.js +0 -131
  538. package/dist/server/lib/__tests__/rate-limiter-budgets.test.js +0 -216
  539. package/dist/server/lib/__tests__/rate-limiter.budgets-tools.test.js +0 -113
  540. package/dist/server/lib/__tests__/rate-limiter.check-request.test.js +0 -141
  541. package/dist/server/lib/__tests__/rate-limiter.stats-lifecycle.test.js +0 -135
  542. package/dist/server/lib/__tests__/rate-limiter.test.js +0 -207
  543. package/dist/server/lib/__tests__/server-agent-loop-abort-conditions.test.js +0 -544
  544. package/dist/server/lib/__tests__/server-agent-loop-abort.part2.test.js +0 -504
  545. package/dist/server/lib/__tests__/server-agent-loop-abort.test.js +0 -396
  546. package/dist/server/lib/__tests__/server-agent-loop-compaction.test.js +0 -397
  547. package/dist/server/lib/__tests__/server-agent-loop-failover.test.js +0 -356
  548. package/dist/server/lib/__tests__/server-agent-loop-features-caching.test.js +0 -519
  549. package/dist/server/lib/__tests__/server-agent-loop-features-edges.test.js +0 -512
  550. package/dist/server/lib/__tests__/server-subagent-bailout.test.js +0 -194
  551. package/dist/server/lib/__tests__/server-subagent-basics.test.js +0 -348
  552. package/dist/server/lib/__tests__/server-subagent-errors-abort.test.js +0 -319
  553. package/dist/server/lib/__tests__/server-subagent-errors-progress.test.js +0 -253
  554. package/dist/server/lib/__tests__/server-subagent-errors.part2.test.js +0 -253
  555. package/dist/server/lib/__tests__/server-subagent-errors.test.js +0 -319
  556. package/dist/server/lib/__tests__/session-checkpoint-load.test.js +0 -275
  557. package/dist/server/lib/__tests__/session-checkpoint-save.test.js +0 -159
  558. package/dist/server/lib/__tests__/ssrf-guard.test.js +0 -93
  559. package/dist/server/lib/__tests__/supabase-client.test.js +0 -111
  560. package/dist/server/lib/__tests__/template-resolver.test.js +0 -317
  561. package/dist/server/lib/__tests__/utils-timeout.test.js +0 -49
  562. package/dist/server/lib/__tests__/utils.test.js +0 -322
  563. package/dist/server/providers/__tests__/anthropic-adapter.test.js +0 -228
  564. package/dist/server/providers/__tests__/anthropic-betas-toolchoice.test.js +0 -257
  565. package/dist/server/providers/__tests__/anthropic-errors.test.js +0 -262
  566. package/dist/server/providers/__tests__/anthropic-stream-core.test.js +0 -275
  567. package/dist/server/providers/__tests__/anthropic-streaming-betas.test.js +0 -247
  568. package/dist/server/providers/__tests__/anthropic-streaming-core.test.js +0 -275
  569. package/dist/server/providers/__tests__/bedrock-config.test.js +0 -177
  570. package/dist/server/providers/__tests__/bedrock-stream-behavior-streaming.test.js +0 -272
  571. package/dist/server/providers/__tests__/bedrock-stream-behavior-toolchoice.test.js +0 -214
  572. package/dist/server/providers/__tests__/bedrock-stream-behavior.part2.test.js +0 -165
  573. package/dist/server/providers/__tests__/bedrock-stream-behavior.test.js +0 -309
  574. package/dist/server/providers/__tests__/bedrock-stream-body-credentials.test.js +0 -170
  575. package/dist/server/providers/__tests__/bedrock-stream-body-extras.test.js +0 -183
  576. package/dist/server/providers/__tests__/bedrock-stream-body-request.test.js +0 -305
  577. package/dist/server/providers/__tests__/bedrock-stream-body.part2.test.js +0 -305
  578. package/dist/server/providers/__tests__/bedrock-stream-body.test.js +0 -175
  579. package/dist/server/providers/__tests__/bedrock-stream-errors.test.js +0 -165
  580. package/dist/server/providers/__tests__/gemini-config-methods.test.js +0 -182
  581. package/dist/server/providers/__tests__/gemini-config-streaming.test.js +0 -257
  582. package/dist/server/providers/__tests__/gemini-conversion-messages.test.js +0 -247
  583. package/dist/server/providers/__tests__/gemini-conversion-schema.test.js +0 -365
  584. package/dist/server/providers/__tests__/gemini-tools-choice.test.js +0 -221
  585. package/dist/server/providers/__tests__/gemini-tools-fn.test.js +0 -252
  586. package/dist/server/providers/__tests__/openai-config.test.js +0 -194
  587. package/dist/server/providers/__tests__/openai-conversion.test.js +0 -276
  588. package/dist/server/providers/__tests__/openai-messages.test.js +0 -261
  589. package/dist/server/providers/__tests__/openai-streaming.test.js +0 -394
  590. package/dist/server/providers/__tests__/openai-tools-cache.test.js +0 -227
  591. package/dist/server/providers/__tests__/registry.test.js +0 -183
  592. package/dist/server/providers/__tests__/shared.test.js +0 -297
  593. package/dist/shared/agent-core-config.test.js +0 -132
  594. package/dist/shared/agent-core-context-thinking.test.js +0 -293
  595. package/dist/shared/agent-core-loop-calls.test.js +0 -174
  596. package/dist/shared/agent-core-loop-detector-bail.test.js +0 -201
  597. package/dist/shared/agent-core-loop-detector.test.js +0 -195
  598. package/dist/shared/agent-core-loop-errors.test.js +0 -258
  599. package/dist/shared/agent-core-pricing.test.js +0 -191
  600. package/dist/shared/agent-core-sanitize-retry.test.js +0 -129
  601. package/dist/shared/api-client-build-request.test.js +0 -228
  602. package/dist/shared/api-client-build-system-caching.test.js +0 -107
  603. package/dist/shared/api-client-build.test.js +0 -223
  604. package/dist/shared/api-client-config.d.ts +0 -21
  605. package/dist/shared/api-client-helpers.d.ts +0 -57
  606. package/dist/shared/api-client-helpers.test.js +0 -261
  607. package/dist/shared/api-client-proxy-happy.test.js +0 -255
  608. package/dist/shared/api-client-proxy-retry.test.js +0 -307
  609. package/dist/shared/api-client-proxy.d.ts +0 -26
  610. package/dist/shared/api-client-proxy.test.js +0 -255
  611. package/dist/shared/api-client-retry.test.js +0 -307
  612. package/dist/shared/api-client-system-trimming.test.js +0 -261
  613. package/dist/shared/api-client-trimming.d.ts +0 -36
  614. package/dist/shared/api-client.test.js +0 -228
  615. package/dist/shared/compaction-thinking.test.js +0 -315
  616. package/dist/shared/compaction-trimming.test.js +0 -223
  617. package/dist/shared/sse-parser-callbacks.test.js +0 -422
  618. package/dist/shared/sse-parser-collect.test.js +0 -252
  619. package/dist/shared/sse-parser-e2e.test.js +0 -558
  620. package/dist/shared/sse-parser-parse.test.js +0 -253
  621. package/dist/shared/tool-dispatch-advanced-batch-build.test.js +0 -405
  622. package/dist/shared/tool-dispatch-advanced.test.js +0 -320
  623. package/dist/shared/tool-dispatch-basic.test.js +0 -278
  624. package/dist/shared/tool-dispatch-content.d.ts +0 -14
  625. package/dist/shared/tool-dispatch-parallel.test.js +0 -378
  626. package/dist/webchat/__tests__/widget-messaging.test.js +0 -323
  627. package/dist/webchat/__tests__/widget.test.js +0 -273
@@ -1 +1 @@
1
- {"version":3,"file":"clickhouse.js","names":["getClickHouseClient","handleClickHouse","sb","args","storeId","sid","hoursBack","hours_back","limit","Math","min","ch","isEnabled","success","error","storeFilter","esc","action","data","query","traceId","trace_id","spans","functionName","function_name","levelFilter","level","budgets","budgetErr","from","select","eq","message","agentIds","map","b","agent_id","join","spendRows","spendMap","Map","r","total_cost_usd","spend","get","budget_amount","limit_usd","current_spend","period_start","created_at","period_end","Date","toISOString","utilization_pct","is_over_budget","groups","total_groups","length","functions","traces","budgetData","Promise","all","resolve","totalRequests","reduce","sum","f","total_requests","totalErrors","error_count","summary","function_count","total_errors","error_rate","round","recent_traces","cost_budgets","s","replace"],"sources":["../../../src/server/handlers/clickhouse.ts"],"sourcesContent":["import type { SupabaseClient } from \"@supabase/supabase-js\";\nimport { getClickHouseClient } from \"../lib/clickhouse-client.js\";\n\n/**\n * ClickHouse observability handler — platform-level metrics via direct HTTP queries.\n * Replaces FDW RPCs with direct ClickHouse queries for lower latency.\n */\nexport async function handleClickHouse(\n sb: SupabaseClient,\n args: Record<string, unknown>,\n storeId?: string,\n): Promise<{ success: boolean; data?: unknown; error?: string }> {\n const sid = storeId as string;\n const hoursBack = (args.hours_back as number) || 24;\n const limit = Math.min((args.limit as number) || 50, 200);\n const ch = getClickHouseClient();\n\n if (!ch.isEnabled) {\n return { success: false, error: \"ClickHouse is not configured\" };\n }\n\n const storeFilter = sid ? `AND store_id = '${esc(sid)}'` : \"\";\n\n switch (args.action) {\n // ---- traces: List recent traces ----\n case \"traces\": {\n const data = await ch.query<{\n trace_id: string; root_service: string; root_operation: string;\n duration: number; span_count: number; error_count: number;\n started_at: string; status: string;\n }>(`\n SELECT\n trace_id,\n arrayElement(groupArrayIf(service_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_service,\n arrayElement(groupArrayIf(operation_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_operation,\n toInt32(dateDiff('millisecond', min(started_at), max(ended_at))) AS duration,\n count() AS span_count,\n countIf(status_code = 'ERROR') AS error_count,\n min(started_at) AS started_at,\n if(countIf(status_code = 'ERROR') > 0, 'error', 'ok') AS status\n FROM ai_spans\n WHERE 1=1 ${storeFilter}\n GROUP BY trace_id\n ORDER BY min(started_at) DESC\n LIMIT ${limit}\n `);\n return { success: true, data };\n }\n\n // ---- trace_detail: Full trace waterfall ----\n case \"trace_detail\": {\n const traceId = args.trace_id as string;\n if (!traceId) return { success: false, error: \"trace_id is required\" };\n const spans = await ch.query<Record<string, unknown>>(`\n SELECT\n span_id, parent_span_id, trace_id, operation_name,\n service_name, duration_ms AS duration,\n started_at, ended_at,\n if(status_code = 'ERROR', 'error', 'ok') AS status,\n attributes, events\n FROM ai_spans\n WHERE trace_id = '${esc(traceId)}'\n ORDER BY started_at ASC\n `);\n return { success: true, data: { trace_id: traceId, spans } };\n }\n\n // ---- function_health: Service health metrics ----\n case \"function_health\": {\n const data = await ch.query<Record<string, unknown>>(`\n SELECT\n service_name AS function_name,\n sum(total_requests) AS total_reqs,\n sum(error_count) AS total_errors,\n sum(client_error_count) AS client_errors,\n if(sum(total_requests) > 0, sum(error_count) / sum(total_requests), 0) AS error_rate,\n if(sum(latency_count) > 0, sum(latency_sum_ms) / sum(latency_count), 0) AS avg_latency_ms,\n max(latency_max_ms) AS p95_latency_ms,\n min(latency_min_ms) AS min_latency_ms,\n max(latency_max_ms) AS max_latency_ms,\n max(bucket) AS last_request_at\n FROM function_health\n WHERE bucket >= now() - INTERVAL ${hoursBack} HOUR\n ${storeFilter}\n GROUP BY service_name\n `);\n return { success: true, data };\n }\n\n // ---- function_logs: Request logs ----\n case \"function_logs\": {\n const functionName = args.function_name as string;\n if (!functionName) return { success: false, error: \"function_name is required\" };\n const levelFilter = args.level ? `AND severity = '${esc(args.level as string)}'` : \"\";\n const data = await ch.query<Record<string, unknown>>(`\n SELECT\n request_id, service_name AS function_name, http_status AS status,\n duration_ms, http_method AS method, http_path AS path,\n store_id, status_code = 'ERROR' AS has_error,\n started_at AS first_seen, events AS lines\n FROM ai_spans\n WHERE service_name = '${esc(functionName)}'\n ${storeFilter} ${levelFilter}\n AND span_kind = 'SERVER'\n ORDER BY started_at DESC\n LIMIT ${limit}\n `);\n return { success: true, data };\n }\n\n // ---- cost_budgets: Budget utilization ----\n case \"cost_budgets\": {\n if (!sid) return { success: false, error: \"store_id is required for cost_budgets\" };\n // Cost budgets require Postgres for ai_cost_budgets table + ClickHouse for spend\n const { data: budgets, error: budgetErr } = await sb\n .from(\"ai_cost_budgets\")\n .select(\"agent_id, limit_usd, is_active, created_at\")\n .eq(\"is_active\", true)\n .eq(\"store_id\", sid);\n if (budgetErr) return { success: false, error: budgetErr.message };\n\n const agentIds = (budgets || []).map(b => `'${esc(b.agent_id)}'`).join(\",\");\n const spendRows = agentIds\n ? await ch.query<{ agent_id: string; total_cost_usd: number }>(`\n SELECT agent_id, sum(total_cost_usd) AS total_cost_usd\n FROM token_usage_hourly\n WHERE agent_id IN (${agentIds})\n AND store_id = '${esc(sid)}'\n GROUP BY agent_id\n `)\n : [];\n\n const spendMap = new Map(spendRows.map(r => [r.agent_id, r.total_cost_usd]));\n const data = (budgets || []).map(b => {\n const spend = spendMap.get(b.agent_id) || 0;\n return {\n agent_id: b.agent_id,\n budget_amount: b.limit_usd,\n current_spend: spend,\n period_start: b.created_at,\n period_end: new Date().toISOString(),\n utilization_pct: b.limit_usd > 0 ? (spend / b.limit_usd) * 100 : 0,\n is_over_budget: spend > b.limit_usd,\n };\n });\n return { success: true, data };\n }\n\n // ---- error_groups: Error fingerprints grouped ----\n case \"error_groups\": {\n const data = await ch.query<{\n fingerprint: string; error_type: string; message: string;\n severity: string; service: string; count: number; last_seen: string;\n }>(`\n SELECT\n fingerprint,\n any(error_type) AS error_type,\n substring(any(error_message), 1, 200) AS message,\n any(severity) AS severity,\n any(service_name) AS service,\n count() AS count,\n max(occurred_at) AS last_seen\n FROM error_events\n WHERE occurred_at >= now() - INTERVAL ${hoursBack} HOUR\n GROUP BY fingerprint\n ORDER BY count DESC\n LIMIT ${limit}\n `);\n return { success: true, data: { groups: data, total_groups: data.length, hours_back: hoursBack } };\n }\n\n // ---- system_status: Combined overview ----\n case \"system_status\": {\n const [functions, traces, budgetData] = await Promise.all([\n ch.query<Record<string, unknown>>(`\n SELECT\n service_name AS function_name,\n sum(total_requests) AS total_reqs,\n sum(error_count) AS total_errors,\n if(sum(total_requests) > 0, sum(error_count) / sum(total_requests), 0) AS error_rate,\n max(bucket) AS last_request_at\n FROM function_health\n WHERE bucket >= now() - INTERVAL ${Math.min(hoursBack, 1)} HOUR\n ${storeFilter}\n GROUP BY service_name\n `),\n ch.query<Record<string, unknown>>(`\n SELECT\n trace_id,\n arrayElement(groupArrayIf(service_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_service,\n arrayElement(groupArrayIf(operation_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_operation,\n toInt32(dateDiff('millisecond', min(started_at), max(ended_at))) AS duration,\n count() AS span_count,\n countIf(status_code = 'ERROR') AS error_count,\n min(started_at) AS started_at,\n if(countIf(status_code = 'ERROR') > 0, 'error', 'ok') AS status\n FROM ai_spans\n WHERE 1=1 ${storeFilter}\n GROUP BY trace_id\n ORDER BY min(started_at) DESC\n LIMIT 10\n `),\n sid\n ? handleClickHouse(sb, { action: \"cost_budgets\" }, sid)\n : Promise.resolve({ success: true, data: null }),\n ]);\n\n const totalRequests = functions.reduce((sum, f) => sum + ((f.total_requests as number) || 0), 0);\n const totalErrors = functions.reduce((sum, f) => sum + ((f.error_count as number) || 0), 0);\n\n return {\n success: true,\n data: {\n summary: {\n function_count: functions.length,\n total_requests: totalRequests,\n total_errors: totalErrors,\n error_rate: totalRequests > 0 ? Math.round((totalErrors / totalRequests) * 10000) / 100 : 0,\n },\n functions,\n recent_traces: traces,\n cost_budgets: budgetData.data,\n hours_back: hoursBack,\n },\n };\n }\n\n default:\n return {\n success: false,\n error: `Unknown clickhouse action: ${args.action}. Available: traces, trace_detail, function_health, function_logs, cost_budgets, error_groups, system_status`,\n };\n }\n}\n\n/** Escape single quotes for ClickHouse SQL */\nfunction esc(s: string): string {\n return s.replace(/'/g, \"\\\\'\");\n}\n"],"mappings":"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","ignoreList":[]}
1
+ {"version":3,"file":"clickhouse.js","names":["getClickHouseClient","handleClickHouse","sb","args","storeId","sid","hoursBack","Math","max","min","Number","hours_back","limit","ch","isEnabled","success","error","storeFilter","esc","action","data","query","traceId","trace_id","spans","functionName","function_name","levelFilter","level","budgets","budgetErr","from","select","eq","message","agentIds","map","b","agent_id","join","spendRows","spendMap","Map","r","total_cost_usd","spend","get","budget_amount","limit_usd","current_spend","period_start","created_at","period_end","Date","toISOString","utilization_pct","is_over_budget","groups","total_groups","length","functions","traces","budgetData","Promise","all","resolve","totalRequests","reduce","sum","f","total_requests","totalErrors","error_count","summary","function_count","total_errors","error_rate","round","recent_traces","cost_budgets","s","replace"],"sources":["../../../src/server/handlers/clickhouse.ts"],"sourcesContent":["import type { SupabaseClient } from \"@supabase/supabase-js\";\nimport { getClickHouseClient } from \"../lib/clickhouse-client.js\";\n\n/**\n * ClickHouse observability handler — platform-level metrics via direct HTTP queries.\n * Replaces FDW RPCs with direct ClickHouse queries for lower latency.\n */\nexport async function handleClickHouse(\n sb: SupabaseClient,\n args: Record<string, unknown>,\n storeId?: string,\n): Promise<{ success: boolean; data?: unknown; error?: string }> {\n const sid = storeId as string;\n const hoursBack = Math.max(1, Math.min(Number(args.hours_back) || 24, 720));\n const limit = Math.max(1, Math.min(Number(args.limit) || 50, 1000));\n const ch = getClickHouseClient();\n\n if (!ch.isEnabled) {\n return { success: false, error: \"ClickHouse is not configured\" };\n }\n\n const storeFilter = sid ? `AND store_id = '${esc(sid)}'` : \"\";\n\n switch (args.action) {\n // ---- traces: List recent traces ----\n case \"traces\": {\n const data = await ch.query<{\n trace_id: string; root_service: string; root_operation: string;\n duration: number; span_count: number; error_count: number;\n started_at: string; status: string;\n }>(`\n SELECT\n trace_id,\n arrayElement(groupArrayIf(service_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_service,\n arrayElement(groupArrayIf(operation_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_operation,\n toInt32(dateDiff('millisecond', min(started_at), max(ended_at))) AS duration,\n count() AS span_count,\n countIf(status_code = 'ERROR') AS error_count,\n min(started_at) AS started_at,\n if(countIf(status_code = 'ERROR') > 0, 'error', 'ok') AS status\n FROM ai_spans\n WHERE 1=1 ${storeFilter}\n GROUP BY trace_id\n ORDER BY min(started_at) DESC\n LIMIT ${limit}\n `);\n return { success: true, data };\n }\n\n // ---- trace_detail: Full trace waterfall ----\n case \"trace_detail\": {\n const traceId = args.trace_id as string;\n if (!traceId) return { success: false, error: \"trace_id is required\" };\n const spans = await ch.query<Record<string, unknown>>(`\n SELECT\n span_id, parent_span_id, trace_id, operation_name,\n service_name, duration_ms AS duration,\n started_at, ended_at,\n if(status_code = 'ERROR', 'error', 'ok') AS status,\n attributes, events\n FROM ai_spans\n WHERE trace_id = '${esc(traceId)}'\n ORDER BY started_at ASC\n `);\n return { success: true, data: { trace_id: traceId, spans } };\n }\n\n // ---- function_health: Service health metrics ----\n case \"function_health\": {\n const data = await ch.query<Record<string, unknown>>(`\n SELECT\n service_name AS function_name,\n sum(total_requests) AS total_reqs,\n sum(error_count) AS total_errors,\n sum(client_error_count) AS client_errors,\n if(sum(total_requests) > 0, sum(error_count) / sum(total_requests), 0) AS error_rate,\n if(sum(latency_count) > 0, sum(latency_sum_ms) / sum(latency_count), 0) AS avg_latency_ms,\n max(latency_max_ms) AS p95_latency_ms,\n min(latency_min_ms) AS min_latency_ms,\n max(latency_max_ms) AS max_latency_ms,\n max(bucket) AS last_request_at\n FROM function_health\n WHERE bucket >= now() - INTERVAL ${hoursBack} HOUR\n ${storeFilter}\n GROUP BY service_name\n `);\n return { success: true, data };\n }\n\n // ---- function_logs: Request logs ----\n case \"function_logs\": {\n const functionName = args.function_name as string;\n if (!functionName) return { success: false, error: \"function_name is required\" };\n const levelFilter = args.level ? `AND severity = '${esc(args.level as string)}'` : \"\";\n const data = await ch.query<Record<string, unknown>>(`\n SELECT\n request_id, service_name AS function_name, http_status AS status,\n duration_ms, http_method AS method, http_path AS path,\n store_id, status_code = 'ERROR' AS has_error,\n started_at AS first_seen, events AS lines\n FROM ai_spans\n WHERE service_name = '${esc(functionName)}'\n ${storeFilter} ${levelFilter}\n AND span_kind = 'SERVER'\n ORDER BY started_at DESC\n LIMIT ${limit}\n `);\n return { success: true, data };\n }\n\n // ---- cost_budgets: Budget utilization ----\n case \"cost_budgets\": {\n if (!sid) return { success: false, error: \"store_id is required for cost_budgets\" };\n // Cost budgets require Postgres for ai_cost_budgets table + ClickHouse for spend\n const { data: budgets, error: budgetErr } = await sb\n .from(\"ai_cost_budgets\")\n .select(\"agent_id, limit_usd, is_active, created_at\")\n .eq(\"is_active\", true)\n .eq(\"store_id\", sid);\n if (budgetErr) return { success: false, error: budgetErr.message };\n\n const agentIds = (budgets || []).map(b => `'${esc(b.agent_id)}'`).join(\",\");\n const spendRows = agentIds\n ? await ch.query<{ agent_id: string; total_cost_usd: number }>(`\n SELECT agent_id, sum(total_cost_usd) AS total_cost_usd\n FROM token_usage_hourly\n WHERE agent_id IN (${agentIds})\n AND store_id = '${esc(sid)}'\n GROUP BY agent_id\n `)\n : [];\n\n const spendMap = new Map(spendRows.map(r => [r.agent_id, r.total_cost_usd]));\n const data = (budgets || []).map(b => {\n const spend = spendMap.get(b.agent_id) || 0;\n return {\n agent_id: b.agent_id,\n budget_amount: b.limit_usd,\n current_spend: spend,\n period_start: b.created_at,\n period_end: new Date().toISOString(),\n utilization_pct: b.limit_usd > 0 ? (spend / b.limit_usd) * 100 : 0,\n is_over_budget: spend > b.limit_usd,\n };\n });\n return { success: true, data };\n }\n\n // ---- error_groups: Error fingerprints grouped ----\n case \"error_groups\": {\n const data = await ch.query<{\n fingerprint: string; error_type: string; message: string;\n severity: string; service: string; count: number; last_seen: string;\n }>(`\n SELECT\n fingerprint,\n any(error_type) AS error_type,\n substring(any(error_message), 1, 200) AS message,\n any(severity) AS severity,\n any(service_name) AS service,\n count() AS count,\n max(occurred_at) AS last_seen\n FROM error_events\n WHERE occurred_at >= now() - INTERVAL ${hoursBack} HOUR\n GROUP BY fingerprint\n ORDER BY count DESC\n LIMIT ${limit}\n `);\n return { success: true, data: { groups: data, total_groups: data.length, hours_back: hoursBack } };\n }\n\n // ---- system_status: Combined overview ----\n case \"system_status\": {\n const [functions, traces, budgetData] = await Promise.all([\n ch.query<Record<string, unknown>>(`\n SELECT\n service_name AS function_name,\n sum(total_requests) AS total_reqs,\n sum(error_count) AS total_errors,\n if(sum(total_requests) > 0, sum(error_count) / sum(total_requests), 0) AS error_rate,\n max(bucket) AS last_request_at\n FROM function_health\n WHERE bucket >= now() - INTERVAL ${Math.min(hoursBack, 1)} HOUR\n ${storeFilter}\n GROUP BY service_name\n `),\n ch.query<Record<string, unknown>>(`\n SELECT\n trace_id,\n arrayElement(groupArrayIf(service_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_service,\n arrayElement(groupArrayIf(operation_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_operation,\n toInt32(dateDiff('millisecond', min(started_at), max(ended_at))) AS duration,\n count() AS span_count,\n countIf(status_code = 'ERROR') AS error_count,\n min(started_at) AS started_at,\n if(countIf(status_code = 'ERROR') > 0, 'error', 'ok') AS status\n FROM ai_spans\n WHERE 1=1 ${storeFilter}\n GROUP BY trace_id\n ORDER BY min(started_at) DESC\n LIMIT 10\n `),\n sid\n ? handleClickHouse(sb, { action: \"cost_budgets\" }, sid)\n : Promise.resolve({ success: true, data: null }),\n ]);\n\n const totalRequests = functions.reduce((sum, f) => sum + ((f.total_requests as number) || 0), 0);\n const totalErrors = functions.reduce((sum, f) => sum + ((f.error_count as number) || 0), 0);\n\n return {\n success: true,\n data: {\n summary: {\n function_count: functions.length,\n total_requests: totalRequests,\n total_errors: totalErrors,\n error_rate: totalRequests > 0 ? Math.round((totalErrors / totalRequests) * 10000) / 100 : 0,\n },\n functions,\n recent_traces: traces,\n cost_budgets: budgetData.data,\n hours_back: hoursBack,\n },\n };\n }\n\n default:\n return {\n success: false,\n error: `Unknown clickhouse action: ${args.action}. Available: traces, trace_detail, function_health, function_logs, cost_budgets, error_groups, system_status`,\n };\n }\n}\n\n/** Escape backslashes then single quotes for ClickHouse SQL */\nfunction esc(s: string): string {\n return s.replace(/\\\\/g, \"\\\\\\\\\").replace(/'/g, \"\\\\'\");\n}\n"],"mappings":"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","ignoreList":[]}
@@ -181,9 +181,27 @@ export async function handleEmail(sb, args, storeId) {
181
181
  attachments: resolvedAttachments
182
182
  } : {})
183
183
  };
184
- if (args.cc) payload.cc = args.cc;
185
- if (args.bcc) payload.bcc = args.bcc;
186
- if (args.reply_to) payload.reply_to = args.reply_to;
184
+ if (args.cc) {
185
+ if (/[\r\n]/.test(String(args.cc))) return {
186
+ success: false,
187
+ error: "cc contains invalid characters"
188
+ };
189
+ payload.cc = args.cc;
190
+ }
191
+ if (args.bcc) {
192
+ if (/[\r\n]/.test(String(args.bcc))) return {
193
+ success: false,
194
+ error: "bcc contains invalid characters"
195
+ };
196
+ payload.bcc = args.bcc;
197
+ }
198
+ if (args.reply_to) {
199
+ if (/[\r\n]/.test(String(args.reply_to))) return {
200
+ success: false,
201
+ error: "reply_to contains invalid characters"
202
+ };
203
+ payload.reply_to = args.reply_to;
204
+ }
187
205
  const resp = await fetch("https://api.resend.com/emails", {
188
206
  method: "POST",
189
207
  headers: {
@@ -235,7 +253,8 @@ export async function handleEmail(sb, args, storeId) {
235
253
  let textContent = tpl.text_content || "";
236
254
  let subjectLine = tpl.subject || "";
237
255
  for (const [key, val] of Object.entries(tData)) {
238
- const re = new RegExp(`\\{\\{\\s*${key}\\s*\\}\\}`, "g");
256
+ const escapedKey = key.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
257
+ const re = new RegExp(`\\{\\{\\s*${escapedKey}\\s*\\}\\}`, "g");
239
258
  htmlContent = htmlContent.replace(re, val);
240
259
  textContent = textContent.replace(re, val);
241
260
  subjectLine = subjectLine.replace(re, val);
@@ -310,6 +329,39 @@ export async function handleEmail(sb, args, storeId) {
310
329
  success: false,
311
330
  error: "body (reply text) required"
312
331
  };
332
+
333
+ // CRLF injection checks
334
+ if (args.to && /[\r\n]/.test(String(args.to))) return {
335
+ success: false,
336
+ error: "Invalid recipient email address"
337
+ };
338
+ if (args.subject && /[\r\n]/.test(String(args.subject))) return {
339
+ success: false,
340
+ error: "Subject contains invalid characters"
341
+ };
342
+ if (args.from && /[\r\n]/.test(String(args.from))) return {
343
+ success: false,
344
+ error: "from contains invalid characters"
345
+ };
346
+ if (args.in_reply_to && /[\r\n]/.test(String(args.in_reply_to))) return {
347
+ success: false,
348
+ error: "in_reply_to contains invalid characters"
349
+ };
350
+
351
+ // Per-store email rate limit (100 emails/hour) — same as send action
352
+ const replyOneHourAgo = new Date(Date.now() - 3600_000).toISOString();
353
+ const {
354
+ count: replyRecentSends
355
+ } = await sb.from("email_sends").select("id", {
356
+ count: "exact",
357
+ head: true
358
+ }).eq("store_id", sid).gte("created_at", replyOneHourAgo);
359
+ if ((replyRecentSends ?? 0) >= 100) {
360
+ return {
361
+ success: false,
362
+ error: "Rate limit exceeded: maximum 100 emails per hour per store"
363
+ };
364
+ }
313
365
  const {
314
366
  data: store
315
367
  } = await sb.from("stores").select("resend_api_key").eq("id", sid).single();
@@ -337,13 +389,23 @@ export async function handleEmail(sb, args, storeId) {
337
389
  })
338
390
  });
339
391
  const result = await resp.json();
340
- return resp.ok ? {
341
- success: true,
342
- data: result
343
- } : {
392
+ if (!resp.ok) return {
344
393
  success: false,
345
394
  error: result.error || "Reply failed"
346
395
  };
396
+
397
+ // Track send for rate limiting
398
+ const replySubject = args.subject || `Re: Thread ${args.thread_id}`;
399
+ await sb.from("email_sends").insert({
400
+ store_id: sid,
401
+ to_email: args.to,
402
+ subject: replySubject,
403
+ resend_id: result.id
404
+ }).then(() => {}, () => {});
405
+ return {
406
+ success: true,
407
+ data: result
408
+ };
347
409
  } catch (err) {
348
410
  return {
349
411
  success: false,
@@ -1 +1 @@
1
- {"version":3,"file":"comms-email.js","names":["sanitizeFilterValue","groupBy","validateUrl","MAX_ATTACHMENT_BYTES","MAX_TOTAL_ATTACHMENT_BYTES","MAX_ATTACHMENT_COUNT","handleEmail","sb","args","storeId","success","error","sid","action","q","from","select","eq","order","ascending","limit","status","mailbox","priority","data","message","flattened","map","row","latest_message","rest","subject","from_email","thread_id","single","messages","thread","lines","msg","sort","a","b","Date","created_at","getTime","push","to_email","slice","body","body_text","length","join","oneHourAgo","now","toISOString","count","recentSends","head","gte","to","test","bodyHtml","body_html","bodyText","attachments","att","url","urlCheck","store","resend_api_key","resolvedAttachments","totalBytes","resp","fetch","ok","buf","Buffer","arrayBuffer","filename","content","toString","fromEmail","email","store_name","toLowerCase","replace","payload","html","text","cc","bcc","reply_to","method","headers","Authorization","JSON","stringify","result","json","insert","store_id","resend_id","id","then","templateSlug","template","tpl","tplErr","or","tData","template_data","htmlContent","html_content","textContent","text_content","subjectLine","key","val","Object","entries","re","RegExp","st","ilike","email_id","in_reply_to","err","updates","intent","ai_intent","ai_summary","update","stats","total","by_status","by_mailbox","by_priority","name","slug","category","description","preview_text","is_active","template_id","undefined","keys","deleted"],"sources":["../../../src/server/handlers/comms-email.ts"],"sourcesContent":["// comms-email.ts — Email handler (inbox, send, templates)\nimport type { SupabaseClient } from \"@supabase/supabase-js\";\nimport { sanitizeFilterValue, groupBy } from \"../lib/utils.js\";\nimport { validateUrl } from \"../lib/ssrf-guard.js\";\n\nconst MAX_ATTACHMENT_BYTES = 10 * 1024 * 1024; // 10MB per attachment\nconst MAX_TOTAL_ATTACHMENT_BYTES = 25 * 1024 * 1024; // 25MB total\nconst MAX_ATTACHMENT_COUNT = 10;\n\nexport async function handleEmail(sb: SupabaseClient, args: Record<string, unknown>, storeId?: string) {\n if (!storeId) return { success: false, error: \"store_id required\" };\n const sid = storeId as string;\n switch (args.action) {\n case \"inbox\": {\n let q = sb.from(\"email_threads\").select(\"*, latest_message:email_inbox(subject, from_email, created_at)\")\n .eq(\"store_id\", sid).order(\"updated_at\", { ascending: false }).limit(args.limit as number || 25);\n if (args.status) q = q.eq(\"status\", args.status as string);\n if (args.mailbox) q = q.eq(\"mailbox\", args.mailbox as string);\n if (args.priority) q = q.eq(\"priority\", args.priority as string);\n const { data, error } = await q;\n if (error) return { success: false, error: error.message };\n // Flatten latest_message join so subject/from appear as table columns\n const flattened = (data || []).map((row: any) => {\n const { latest_message, ...rest } = row;\n return { ...rest, subject: latest_message?.subject || null, from_email: latest_message?.from_email || null };\n });\n return { success: true, data: flattened };\n }\n case \"inbox_get\": {\n const { data, error } = await sb.from(\"email_threads\")\n .select(\"*, messages:email_inbox(id, subject, from_email, to_email, body_text, created_at)\").eq(\"id\", args.thread_id as string).eq(\"store_id\", sid).single();\n if (error) return { success: false, error: error.message };\n // Pre-format so messages are visible (formatter handles sub-tables but body gets truncated)\n const { messages, ...thread } = data as any;\n const lines: string[] = [\n `## Email Thread`,\n `**Status**: ${thread.status || \"\\u2014\"} | **Mailbox**: ${thread.mailbox || \"\\u2014\"} | **Priority**: ${thread.priority || \"\\u2014\"}`,\n \"\",\n ];\n for (const msg of (messages || []).sort((a: any, b: any) => new Date(a.created_at).getTime() - new Date(b.created_at).getTime())) {\n lines.push(`### ${msg.from_email} \\u2192 ${msg.to_email || \"\\u2014\"} (${msg.created_at?.slice(0, 16) || \"\\u2014\"})`);\n lines.push(`**Subject**: ${msg.subject || \"(no subject)\"}`);\n const body = (msg.body_text || \"\").slice(0, 500);\n lines.push(body + (msg.body_text?.length > 500 ? \"...\" : \"\"));\n lines.push(\"\");\n }\n return { success: true, data: lines.join(\"\\n\") };\n }\n case \"send\": {\n // P0 FIX: Per-store email rate limit (100 emails/hour)\n const oneHourAgo = new Date(Date.now() - 3600_000).toISOString();\n const { count: recentSends } = await sb.from(\"email_sends\")\n .select(\"id\", { count: \"exact\", head: true })\n .eq(\"store_id\", sid)\n .gte(\"created_at\", oneHourAgo);\n if ((recentSends ?? 0) >= 100) {\n return { success: false, error: \"Rate limit exceeded: maximum 100 emails per hour per store\" };\n }\n\n const to = args.to as string;\n if (!to) return { success: false, error: \"to (email address) is required\" };\n // P0 FIX: Sanitize email recipient to prevent header injection\n if (/[\\r\\n]/.test(to)) return { success: false, error: \"Invalid recipient email address\" };\n const subject = (args.subject as string) || \"No Subject\";\n // P0 FIX: Sanitize subject to prevent header injection\n if (/[\\r\\n]/.test(subject)) return { success: false, error: \"Subject contains invalid characters\" };\n const bodyHtml = (args.body_html as string) || \"\";\n const bodyText = (args.body_text as string) || \"\";\n\n // P0 FIX: Validate attachments — reject oversized or too-many attachments\n const attachments = (args.attachments as { filename: string; url: string }[]) || [];\n if (attachments.length > MAX_ATTACHMENT_COUNT) {\n return { success: false, error: `Too many attachments (max ${MAX_ATTACHMENT_COUNT})` };\n }\n // Validate all attachment URLs are public (no SSRF into internal services)\n for (const att of attachments) {\n if (att.url) {\n const urlCheck = await validateUrl(att.url);\n if (urlCheck) return { success: false, error: `Invalid attachment URL \"${att.url}\": ${urlCheck}` };\n }\n }\n\n // Fetch store Resend API key\n const { data: store } = await sb.from(\"stores\")\n .select(\"resend_api_key, store_name, email\")\n .eq(\"id\", sid).single();\n if (!store?.resend_api_key) return { success: false, error: \"Store has no Resend API key configured\" };\n\n // Download and validate attachment sizes (best-effort — streams through Resend)\n const resolvedAttachments: Array<{ filename: string; content: string }> = [];\n let totalBytes = 0;\n for (const att of attachments) {\n if (att.url) {\n const resp = await fetch(att.url);\n if (!resp.ok) return { success: false, error: `Failed to fetch attachment: ${att.url}` };\n const buf = Buffer.from(await resp.arrayBuffer());\n if (buf.length > MAX_ATTACHMENT_BYTES) {\n return { success: false, error: `Attachment \"${att.filename}\" exceeds ${MAX_ATTACHMENT_BYTES / 1024 / 1024}MB limit` };\n }\n totalBytes += buf.length;\n if (totalBytes > MAX_TOTAL_ATTACHMENT_BYTES) {\n return { success: false, error: `Total attachment size exceeds ${MAX_TOTAL_ATTACHMENT_BYTES / 1024 / 1024}MB limit` };\n }\n resolvedAttachments.push({ filename: att.filename || \"attachment\", content: buf.toString(\"base64\") });\n }\n }\n\n const fromEmail = (args.from as string) || store.email || `${store.store_name?.toLowerCase().replace(/\\s+/g, \"\")}@mail.floradistro.com`;\n const payload: Record<string, unknown> = {\n from: fromEmail, to, subject,\n ...(bodyHtml ? { html: bodyHtml } : {}),\n ...(bodyText ? { text: bodyText } : {}),\n ...(resolvedAttachments.length ? { attachments: resolvedAttachments } : {}),\n };\n if (args.cc) payload.cc = args.cc;\n if (args.bcc) payload.bcc = args.bcc;\n if (args.reply_to) payload.reply_to = args.reply_to;\n\n const resp = await fetch(\"https://api.resend.com/emails\", {\n method: \"POST\",\n headers: { Authorization: `Bearer ${store.resend_api_key}`, \"Content-Type\": \"application/json\" },\n body: JSON.stringify(payload),\n });\n const result = await resp.json();\n if (!resp.ok) return { success: false, error: `Email send failed: ${result.message || JSON.stringify(result)}` };\n\n // Track send for rate limiting\n await sb.from(\"email_sends\").insert({ store_id: sid, to_email: to, subject, resend_id: result.id }).then(() => {}, () => {});\n\n return { success: true, data: { id: result.id, to, subject, status: \"sent\" } };\n }\n case \"send_template\": {\n const templateSlug = args.template as string;\n if (!templateSlug) return { success: false, error: \"template slug required\" };\n\n const { data: tpl, error: tplErr } = await sb.from(\"email_templates\")\n .select(\"*\")\n .eq(\"slug\", templateSlug)\n .or(`store_id.eq.${sid},store_id.is.null`)\n .eq(\"is_active\", true)\n .limit(1)\n .single();\n if (tplErr || !tpl) return { success: false, error: `Template \"${templateSlug}\" not found` };\n\n const tData = (args.template_data as Record<string, string>) || {};\n let htmlContent = tpl.html_content as string || \"\";\n let textContent = tpl.text_content as string || \"\";\n let subjectLine = tpl.subject as string || \"\";\n for (const [key, val] of Object.entries(tData)) {\n const re = new RegExp(`\\\\{\\\\{\\\\s*${key}\\\\s*\\\\}\\\\}`, \"g\");\n htmlContent = htmlContent.replace(re, val);\n textContent = textContent.replace(re, val);\n subjectLine = subjectLine.replace(re, val);\n }\n\n return handleEmail(sb, {\n action: \"send\",\n to: args.to,\n subject: subjectLine,\n body_html: htmlContent,\n body_text: textContent,\n from: args.from,\n cc: args.cc,\n bcc: args.bcc,\n }, storeId);\n }\n case \"list\": {\n let q = sb.from(\"email_sends\")\n .select(\"id, to_email, subject, status, resend_id, created_at\")\n .eq(\"store_id\", sid).order(\"created_at\", { ascending: false });\n if (args.to) { const st = sanitizeFilterValue(args.to as string); q = q.ilike(\"to_email\", `%${st}%`); }\n const { data, error } = await q.limit(args.limit as number || 25);\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n case \"get\": {\n const { data, error } = await sb.from(\"email_sends\")\n .select(\"*\").eq(\"id\", args.email_id as string).eq(\"store_id\", sid).single();\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n case \"templates\": {\n const { data, error } = await sb.from(\"email_templates\")\n .select(\"id, name, slug, subject, category, description, preview_text, is_active, created_at\")\n .or(`store_id.eq.${sid},store_id.is.null`)\n .eq(\"is_active\", true)\n .order(\"name\");\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n case \"inbox_reply\": {\n if (!args.thread_id) return { success: false, error: \"thread_id required\" };\n if (!args.body) return { success: false, error: \"body (reply text) required\" };\n const { data: store } = await sb.from(\"stores\")\n .select(\"resend_api_key\").eq(\"id\", sid).single();\n if (!store?.resend_api_key) return { success: false, error: \"Store has no Resend API key\" };\n try {\n const resp = await fetch(\"https://api.resend.com/emails\", {\n method: \"POST\",\n headers: { Authorization: `Bearer ${store.resend_api_key}`, \"Content-Type\": \"application/json\" },\n body: JSON.stringify({\n from: args.from || `support@mail.floradistro.com`,\n to: args.to as string,\n subject: args.subject || `Re: Thread ${args.thread_id}`,\n html: args.body as string,\n ...(args.in_reply_to ? { headers: { \"In-Reply-To\": args.in_reply_to } } : {}),\n }),\n });\n const result = await resp.json();\n return resp.ok ? { success: true, data: result } : { success: false, error: result.error || \"Reply failed\" };\n } catch (err) {\n return { success: false, error: `Reply failed: ${err}` };\n }\n }\n case \"inbox_update\": {\n const updates: Record<string, unknown> = {};\n if (args.status) updates.status = args.status;\n if (args.priority) updates.priority = args.priority;\n if (args.intent) updates.ai_intent = args.intent;\n if (args.ai_summary) updates.ai_summary = args.ai_summary;\n const { data, error } = await sb.from(\"email_threads\")\n .update(updates).eq(\"id\", args.thread_id as string).eq(\"store_id\", sid).select().single();\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n case \"inbox_stats\": {\n const { data, error } = await sb.from(\"email_threads\")\n .select(\"status, mailbox, priority\").eq(\"store_id\", sid).limit(1000);\n if (error) return { success: false, error: error.message };\n const stats = {\n total: data.length,\n by_status: groupBy(data, \"status\"),\n by_mailbox: groupBy(data, \"mailbox\"),\n by_priority: groupBy(data, \"priority\")\n };\n return { success: true, data: stats };\n }\n case \"create_template\": {\n if (!args.name) return { success: false, error: \"name is required\" };\n if (!args.subject) return { success: false, error: \"subject is required\" };\n const slug = (args.slug as string) || (args.name as string).toLowerCase().replace(/[^a-z0-9]+/g, \"-\").replace(/^-|-$/g, \"\");\n const { data, error } = await sb.from(\"email_templates\").insert({\n store_id: sid,\n name: args.name as string,\n slug,\n subject: args.subject as string,\n category: (args.category as string) || \"general\",\n description: (args.description as string) || null,\n html_content: (args.html_content as string) || null,\n preview_text: (args.preview_text as string) || null,\n text_content: (args.text_content as string) || null,\n is_active: true,\n }).select(\"id, name, slug, subject, category, created_at\").single();\n if (error) return { success: false, error: error.message };\n return { success: true, data };\n }\n case \"update_template\": {\n if (!args.template_id) return { success: false, error: \"template_id is required\" };\n const updates: Record<string, unknown> = {};\n if (args.name !== undefined) updates.name = args.name;\n if (args.slug !== undefined) updates.slug = args.slug;\n if (args.subject !== undefined) updates.subject = args.subject;\n if (args.category !== undefined) updates.category = args.category;\n if (args.description !== undefined) updates.description = args.description;\n if (args.html_content !== undefined) updates.html_content = args.html_content;\n if (args.preview_text !== undefined) updates.preview_text = args.preview_text;\n if (args.text_content !== undefined) updates.text_content = args.text_content;\n if (Object.keys(updates).length === 0) return { success: false, error: \"No fields to update\" };\n const { data, error } = await sb.from(\"email_templates\")\n .update(updates).eq(\"id\", args.template_id as string).eq(\"store_id\", sid)\n .select(\"id, name, slug, subject, category, updated_at\").single();\n if (error) return { success: false, error: error.message };\n return { success: true, data };\n }\n case \"delete_template\": {\n if (!args.template_id) return { success: false, error: \"template_id is required\" };\n const { data, error } = await sb.from(\"email_templates\")\n .update({ is_active: false }).eq(\"id\", args.template_id as string).eq(\"store_id\", sid)\n .select(\"id, name\").single();\n if (error) return { success: false, error: error.message };\n return { success: true, data: { ...data, deleted: true } };\n }\n default:\n return { success: false, error: `Unknown email action: ${args.action}. Valid: send, send_template, list, get, templates, inbox, inbox_get, inbox_reply, inbox_update, inbox_stats, create_template, update_template, delete_template` };\n }\n}\n"],"mappings":"AAAA;;AAEA,SAASA,mBAAmB,EAAEC,OAAO,QAAQ,iBAAiB;AAC9D,SAASC,WAAW,QAAQ,sBAAsB;AAElD,MAAMC,oBAAoB,GAAG,EAAE,GAAG,IAAI,GAAG,IAAI,CAAC,CAAC;AAC/C,MAAMC,0BAA0B,GAAG,EAAE,GAAG,IAAI,GAAG,IAAI,CAAC,CAAC;AACrD,MAAMC,oBAAoB,GAAG,EAAE;AAE/B,OAAO,eAAeC,WAAWA,CAACC,EAAkB,EAAEC,IAA6B,EAAEC,OAAgB,EAAE;EACrG,IAAI,CAACA,OAAO,EAAE,OAAO;IAAEC,OAAO,EAAE,KAAK;IAAEC,KAAK,EAAE;EAAoB,CAAC;EACnE,MAAMC,GAAG,GAAGH,OAAiB;EAC7B,QAAQD,IAAI,CAACK,MAAM;IACjB,KAAK,OAAO;MAAE;QACZ,IAAIC,CAAC,GAAGP,EAAE,CAACQ,IAAI,CAAC,eAAe,CAAC,CAACC,MAAM,CAAC,gEAAgE,CAAC,CACtGC,EAAE,CAAC,UAAU,EAAEL,GAAG,CAAC,CAACM,KAAK,CAAC,YAAY,EAAE;UAAEC,SAAS,EAAE;QAAM,CAAC,CAAC,CAACC,KAAK,CAACZ,IAAI,CAACY,KAAK,IAAc,EAAE,CAAC;QAClG,IAAIZ,IAAI,CAACa,MAAM,EAAEP,CAAC,GAAGA,CAAC,CAACG,EAAE,CAAC,QAAQ,EAAET,IAAI,CAACa,MAAgB,CAAC;QAC1D,IAAIb,IAAI,CAACc,OAAO,EAAER,CAAC,GAAGA,CAAC,CAACG,EAAE,CAAC,SAAS,EAAET,IAAI,CAACc,OAAiB,CAAC;QAC7D,IAAId,IAAI,CAACe,QAAQ,EAAET,CAAC,GAAGA,CAAC,CAACG,EAAE,CAAC,UAAU,EAAET,IAAI,CAACe,QAAkB,CAAC;QAChE,MAAM;UAAEC,IAAI;UAAEb;QAAM,CAAC,GAAG,MAAMG,CAAC;QAC/B,IAAIH,KAAK,EAAE,OAAO;UAAED,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAEA,KAAK,CAACc;QAAQ,CAAC;QAC1D;QACA,MAAMC,SAAS,GAAG,CAACF,IAAI,IAAI,EAAE,EAAEG,GAAG,CAAEC,GAAQ,IAAK;UAC/C,MAAM;YAAEC,cAAc;YAAE,GAAGC;UAAK,CAAC,GAAGF,GAAG;UACvC,OAAO;YAAE,GAAGE,IAAI;YAAEC,OAAO,EAAEF,cAAc,EAAEE,OAAO,IAAI,IAAI;YAAEC,UAAU,EAAEH,cAAc,EAAEG,UAAU,IAAI;UAAK,CAAC;QAC9G,CAAC,CAAC;QACF,OAAO;UAAEtB,OAAO,EAAE,IAAI;UAAEc,IAAI,EAAEE;QAAU,CAAC;MAC3C;IACA,KAAK,WAAW;MAAE;QAChB,MAAM;UAAEF,IAAI;UAAEb;QAAM,CAAC,GAAG,MAAMJ,EAAE,CAACQ,IAAI,CAAC,eAAe,CAAC,CACnDC,MAAM,CAAC,mFAAmF,CAAC,CAACC,EAAE,CAAC,IAAI,EAAET,IAAI,CAACyB,SAAmB,CAAC,CAAChB,EAAE,CAAC,UAAU,EAAEL,GAAG,CAAC,CAACsB,MAAM,CAAC,CAAC;QAC9J,IAAIvB,KAAK,EAAE,OAAO;UAAED,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAEA,KAAK,CAACc;QAAQ,CAAC;QAC1D;QACA,MAAM;UAAEU,QAAQ;UAAE,GAAGC;QAAO,CAAC,GAAGZ,IAAW;QAC3C,MAAMa,KAAe,GAAG,CACtB,iBAAiB,EACjB,eAAeD,MAAM,CAACf,MAAM,IAAI,QAAQ,mBAAmBe,MAAM,CAACd,OAAO,IAAI,QAAQ,oBAAoBc,MAAM,CAACb,QAAQ,IAAI,QAAQ,EAAE,EACtI,EAAE,CACH;QACD,KAAK,MAAMe,GAAG,IAAI,CAACH,QAAQ,IAAI,EAAE,EAAEI,IAAI,CAAC,CAACC,CAAM,EAAEC,CAAM,KAAK,IAAIC,IAAI,CAACF,CAAC,CAACG,UAAU,CAAC,CAACC,OAAO,CAAC,CAAC,GAAG,IAAIF,IAAI,CAACD,CAAC,CAACE,UAAU,CAAC,CAACC,OAAO,CAAC,CAAC,CAAC,EAAE;UAChIP,KAAK,CAACQ,IAAI,CAAC,OAAOP,GAAG,CAACN,UAAU,WAAWM,GAAG,CAACQ,QAAQ,IAAI,QAAQ,KAAKR,GAAG,CAACK,UAAU,EAAEI,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,QAAQ,GAAG,CAAC;UACpHV,KAAK,CAACQ,IAAI,CAAC,gBAAgBP,GAAG,CAACP,OAAO,IAAI,cAAc,EAAE,CAAC;UAC3D,MAAMiB,IAAI,GAAG,CAACV,GAAG,CAACW,SAAS,IAAI,EAAE,EAAEF,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC;UAChDV,KAAK,CAACQ,IAAI,CAACG,IAAI,IAAIV,GAAG,CAACW,SAAS,EAAEC,MAAM,GAAG,GAAG,GAAG,KAAK,GAAG,EAAE,CAAC,CAAC;UAC7Db,KAAK,CAACQ,IAAI,CAAC,EAAE,CAAC;QAChB;QACA,OAAO;UAAEnC,OAAO,EAAE,IAAI;UAAEc,IAAI,EAAEa,KAAK,CAACc,IAAI,CAAC,IAAI;QAAE,CAAC;MAClD;IACA,KAAK,MAAM;MAAE;QACX;QACA,MAAMC,UAAU,GAAG,IAAIV,IAAI,CAACA,IAAI,CAACW,GAAG,CAAC,CAAC,GAAG,QAAQ,CAAC,CAACC,WAAW,CAAC,CAAC;QAChE,MAAM;UAAEC,KAAK,EAAEC;QAAY,CAAC,GAAG,MAAMjD,EAAE,CAACQ,IAAI,CAAC,aAAa,CAAC,CACxDC,MAAM,CAAC,IAAI,EAAE;UAAEuC,KAAK,EAAE,OAAO;UAAEE,IAAI,EAAE;QAAK,CAAC,CAAC,CAC5CxC,EAAE,CAAC,UAAU,EAAEL,GAAG,CAAC,CACnB8C,GAAG,CAAC,YAAY,EAAEN,UAAU,CAAC;QAChC,IAAI,CAACI,WAAW,IAAI,CAAC,KAAK,GAAG,EAAE;UAC7B,OAAO;YAAE9C,OAAO,EAAE,KAAK;YAAEC,KAAK,EAAE;UAA6D,CAAC;QAChG;QAEA,MAAMgD,EAAE,GAAGnD,IAAI,CAACmD,EAAY;QAC5B,IAAI,CAACA,EAAE,EAAE,OAAO;UAAEjD,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAAiC,CAAC;QAC3E;QACA,IAAI,QAAQ,CAACiD,IAAI,CAACD,EAAE,CAAC,EAAE,OAAO;UAAEjD,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAAkC,CAAC;QAC1F,MAAMoB,OAAO,GAAIvB,IAAI,CAACuB,OAAO,IAAe,YAAY;QACxD;QACA,IAAI,QAAQ,CAAC6B,IAAI,CAAC7B,OAAO,CAAC,EAAE,OAAO;UAAErB,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAAsC,CAAC;QACnG,MAAMkD,QAAQ,GAAIrD,IAAI,CAACsD,SAAS,IAAe,EAAE;QACjD,MAAMC,QAAQ,GAAIvD,IAAI,CAACyC,SAAS,IAAe,EAAE;;QAEjD;QACA,MAAMe,WAAW,GAAIxD,IAAI,CAACwD,WAAW,IAA4C,EAAE;QACnF,IAAIA,WAAW,CAACd,MAAM,GAAG7C,oBAAoB,EAAE;UAC7C,OAAO;YAAEK,OAAO,EAAE,KAAK;YAAEC,KAAK,EAAE,6BAA6BN,oBAAoB;UAAI,CAAC;QACxF;QACA;QACA,KAAK,MAAM4D,GAAG,IAAID,WAAW,EAAE;UAC7B,IAAIC,GAAG,CAACC,GAAG,EAAE;YACX,MAAMC,QAAQ,GAAG,MAAMjE,WAAW,CAAC+D,GAAG,CAACC,GAAG,CAAC;YAC3C,IAAIC,QAAQ,EAAE,OAAO;cAAEzD,OAAO,EAAE,KAAK;cAAEC,KAAK,EAAE,2BAA2BsD,GAAG,CAACC,GAAG,MAAMC,QAAQ;YAAG,CAAC;UACpG;QACF;;QAEA;QACA,MAAM;UAAE3C,IAAI,EAAE4C;QAAM,CAAC,GAAG,MAAM7D,EAAE,CAACQ,IAAI,CAAC,QAAQ,CAAC,CAC5CC,MAAM,CAAC,mCAAmC,CAAC,CAC3CC,EAAE,CAAC,IAAI,EAAEL,GAAG,CAAC,CAACsB,MAAM,CAAC,CAAC;QACzB,IAAI,CAACkC,KAAK,EAAEC,cAAc,EAAE,OAAO;UAAE3D,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAAyC,CAAC;;QAEtG;QACA,MAAM2D,mBAAiE,GAAG,EAAE;QAC5E,IAAIC,UAAU,GAAG,CAAC;QAClB,KAAK,MAAMN,GAAG,IAAID,WAAW,EAAE;UAC7B,IAAIC,GAAG,CAACC,GAAG,EAAE;YACX,MAAMM,IAAI,GAAG,MAAMC,KAAK,CAACR,GAAG,CAACC,GAAG,CAAC;YACjC,IAAI,CAACM,IAAI,CAACE,EAAE,EAAE,OAAO;cAAEhE,OAAO,EAAE,KAAK;cAAEC,KAAK,EAAE,+BAA+BsD,GAAG,CAACC,GAAG;YAAG,CAAC;YACxF,MAAMS,GAAG,GAAGC,MAAM,CAAC7D,IAAI,CAAC,MAAMyD,IAAI,CAACK,WAAW,CAAC,CAAC,CAAC;YACjD,IAAIF,GAAG,CAACzB,MAAM,GAAG/C,oBAAoB,EAAE;cACrC,OAAO;gBAAEO,OAAO,EAAE,KAAK;gBAAEC,KAAK,EAAE,eAAesD,GAAG,CAACa,QAAQ,aAAa3E,oBAAoB,GAAG,IAAI,GAAG,IAAI;cAAW,CAAC;YACxH;YACAoE,UAAU,IAAII,GAAG,CAACzB,MAAM;YACxB,IAAIqB,UAAU,GAAGnE,0BAA0B,EAAE;cAC3C,OAAO;gBAAEM,OAAO,EAAE,KAAK;gBAAEC,KAAK,EAAE,iCAAiCP,0BAA0B,GAAG,IAAI,GAAG,IAAI;cAAW,CAAC;YACvH;YACAkE,mBAAmB,CAACzB,IAAI,CAAC;cAAEiC,QAAQ,EAAEb,GAAG,CAACa,QAAQ,IAAI,YAAY;cAAEC,OAAO,EAAEJ,GAAG,CAACK,QAAQ,CAAC,QAAQ;YAAE,CAAC,CAAC;UACvG;QACF;QAEA,MAAMC,SAAS,GAAIzE,IAAI,CAACO,IAAI,IAAeqD,KAAK,CAACc,KAAK,IAAI,GAAGd,KAAK,CAACe,UAAU,EAAEC,WAAW,CAAC,CAAC,CAACC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC,uBAAuB;QACvI,MAAMC,OAAgC,GAAG;UACvCvE,IAAI,EAAEkE,SAAS;UAAEtB,EAAE;UAAE5B,OAAO;UAC5B,IAAI8B,QAAQ,GAAG;YAAE0B,IAAI,EAAE1B;UAAS,CAAC,GAAG,CAAC,CAAC,CAAC;UACvC,IAAIE,QAAQ,GAAG;YAAEyB,IAAI,EAAEzB;UAAS,CAAC,GAAG,CAAC,CAAC,CAAC;UACvC,IAAIO,mBAAmB,CAACpB,MAAM,GAAG;YAAEc,WAAW,EAAEM;UAAoB,CAAC,GAAG,CAAC,CAAC;QAC5E,CAAC;QACD,IAAI9D,IAAI,CAACiF,EAAE,EAAEH,OAAO,CAACG,EAAE,GAAGjF,IAAI,CAACiF,EAAE;QACjC,IAAIjF,IAAI,CAACkF,GAAG,EAAEJ,OAAO,CAACI,GAAG,GAAGlF,IAAI,CAACkF,GAAG;QACpC,IAAIlF,IAAI,CAACmF,QAAQ,EAAEL,OAAO,CAACK,QAAQ,GAAGnF,IAAI,CAACmF,QAAQ;QAEnD,MAAMnB,IAAI,GAAG,MAAMC,KAAK,CAAC,+BAA+B,EAAE;UACxDmB,MAAM,EAAE,MAAM;UACdC,OAAO,EAAE;YAAEC,aAAa,EAAE,UAAU1B,KAAK,CAACC,cAAc,EAAE;YAAE,cAAc,EAAE;UAAmB,CAAC;UAChGrB,IAAI,EAAE+C,IAAI,CAACC,SAAS,CAACV,OAAO;QAC9B,CAAC,CAAC;QACF,MAAMW,MAAM,GAAG,MAAMzB,IAAI,CAAC0B,IAAI,CAAC,CAAC;QAChC,IAAI,CAAC1B,IAAI,CAACE,EAAE,EAAE,OAAO;UAAEhE,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE,sBAAsBsF,MAAM,CAACxE,OAAO,IAAIsE,IAAI,CAACC,SAAS,CAACC,MAAM,CAAC;QAAG,CAAC;;QAEhH;QACA,MAAM1F,EAAE,CAACQ,IAAI,CAAC,aAAa,CAAC,CAACoF,MAAM,CAAC;UAAEC,QAAQ,EAAExF,GAAG;UAAEkC,QAAQ,EAAEa,EAAE;UAAE5B,OAAO;UAAEsE,SAAS,EAAEJ,MAAM,CAACK;QAAG,CAAC,CAAC,CAACC,IAAI,CAAC,MAAM,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC,CAAC;QAE5H,OAAO;UAAE7F,OAAO,EAAE,IAAI;UAAEc,IAAI,EAAE;YAAE8E,EAAE,EAAEL,MAAM,CAACK,EAAE;YAAE3C,EAAE;YAAE5B,OAAO;YAAEV,MAAM,EAAE;UAAO;QAAE,CAAC;MAChF;IACA,KAAK,eAAe;MAAE;QACpB,MAAMmF,YAAY,GAAGhG,IAAI,CAACiG,QAAkB;QAC5C,IAAI,CAACD,YAAY,EAAE,OAAO;UAAE9F,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAAyB,CAAC;QAE7E,MAAM;UAAEa,IAAI,EAAEkF,GAAG;UAAE/F,KAAK,EAAEgG;QAAO,CAAC,GAAG,MAAMpG,EAAE,CAACQ,IAAI,CAAC,iBAAiB,CAAC,CAClEC,MAAM,CAAC,GAAG,CAAC,CACXC,EAAE,CAAC,MAAM,EAAEuF,YAAY,CAAC,CACxBI,EAAE,CAAC,eAAehG,GAAG,mBAAmB,CAAC,CACzCK,EAAE,CAAC,WAAW,EAAE,IAAI,CAAC,CACrBG,KAAK,CAAC,CAAC,CAAC,CACRc,MAAM,CAAC,CAAC;QACX,IAAIyE,MAAM,IAAI,CAACD,GAAG,EAAE,OAAO;UAAEhG,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE,aAAa6F,YAAY;QAAc,CAAC;QAE5F,MAAMK,KAAK,GAAIrG,IAAI,CAACsG,aAAa,IAA+B,CAAC,CAAC;QAClE,IAAIC,WAAW,GAAGL,GAAG,CAACM,YAAY,IAAc,EAAE;QAClD,IAAIC,WAAW,GAAGP,GAAG,CAACQ,YAAY,IAAc,EAAE;QAClD,IAAIC,WAAW,GAAGT,GAAG,CAAC3E,OAAO,IAAc,EAAE;QAC7C,KAAK,MAAM,CAACqF,GAAG,EAAEC,GAAG,CAAC,IAAIC,MAAM,CAACC,OAAO,CAACV,KAAK,CAAC,EAAE;UAC9C,MAAMW,EAAE,GAAG,IAAIC,MAAM,CAAC,aAAaL,GAAG,YAAY,EAAE,GAAG,CAAC;UACxDL,WAAW,GAAGA,WAAW,CAAC1B,OAAO,CAACmC,EAAE,EAAEH,GAAG,CAAC;UAC1CJ,WAAW,GAAGA,WAAW,CAAC5B,OAAO,CAACmC,EAAE,EAAEH,GAAG,CAAC;UAC1CF,WAAW,GAAGA,WAAW,CAAC9B,OAAO,CAACmC,EAAE,EAAEH,GAAG,CAAC;QAC5C;QAEA,OAAO/G,WAAW,CAACC,EAAE,EAAE;UACrBM,MAAM,EAAE,MAAM;UACd8C,EAAE,EAAEnD,IAAI,CAACmD,EAAE;UACX5B,OAAO,EAAEoF,WAAW;UACpBrD,SAAS,EAAEiD,WAAW;UACtB9D,SAAS,EAAEgE,WAAW;UACtBlG,IAAI,EAAEP,IAAI,CAACO,IAAI;UACf0E,EAAE,EAAEjF,IAAI,CAACiF,EAAE;UACXC,GAAG,EAAElF,IAAI,CAACkF;QACZ,CAAC,EAAEjF,OAAO,CAAC;MACb;IACA,KAAK,MAAM;MAAE;QACX,IAAIK,CAAC,GAAGP,EAAE,CAACQ,IAAI,CAAC,aAAa,CAAC,CAC3BC,MAAM,CAAC,sDAAsD,CAAC,CAC9DC,EAAE,CAAC,UAAU,EAAEL,GAAG,CAAC,CAACM,KAAK,CAAC,YAAY,EAAE;UAAEC,SAAS,EAAE;QAAM,CAAC,CAAC;QAChE,IAAIX,IAAI,CAACmD,EAAE,EAAE;UAAE,MAAM+D,EAAE,GAAG1H,mBAAmB,CAACQ,IAAI,CAACmD,EAAY,CAAC;UAAE7C,CAAC,GAAGA,CAAC,CAAC6G,KAAK,CAAC,UAAU,EAAE,IAAID,EAAE,GAAG,CAAC;QAAE;QACtG,MAAM;UAAElG,IAAI;UAAEb;QAAM,CAAC,GAAG,MAAMG,CAAC,CAACM,KAAK,CAACZ,IAAI,CAACY,KAAK,IAAc,EAAE,CAAC;QACjE,OAAOT,KAAK,GAAG;UAAED,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAEA,KAAK,CAACc;QAAQ,CAAC,GAAG;UAAEf,OAAO,EAAE,IAAI;UAAEc;QAAK,CAAC;MACnF;IACA,KAAK,KAAK;MAAE;QACV,MAAM;UAAEA,IAAI;UAAEb;QAAM,CAAC,GAAG,MAAMJ,EAAE,CAACQ,IAAI,CAAC,aAAa,CAAC,CACjDC,MAAM,CAAC,GAAG,CAAC,CAACC,EAAE,CAAC,IAAI,EAAET,IAAI,CAACoH,QAAkB,CAAC,CAAC3G,EAAE,CAAC,UAAU,EAAEL,GAAG,CAAC,CAACsB,MAAM,CAAC,CAAC;QAC7E,OAAOvB,KAAK,GAAG;UAAED,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAEA,KAAK,CAACc;QAAQ,CAAC,GAAG;UAAEf,OAAO,EAAE,IAAI;UAAEc;QAAK,CAAC;MACnF;IACA,KAAK,WAAW;MAAE;QAChB,MAAM;UAAEA,IAAI;UAAEb;QAAM,CAAC,GAAG,MAAMJ,EAAE,CAACQ,IAAI,CAAC,iBAAiB,CAAC,CACrDC,MAAM,CAAC,qFAAqF,CAAC,CAC7F4F,EAAE,CAAC,eAAehG,GAAG,mBAAmB,CAAC,CACzCK,EAAE,CAAC,WAAW,EAAE,IAAI,CAAC,CACrBC,KAAK,CAAC,MAAM,CAAC;QAChB,OAAOP,KAAK,GAAG;UAAED,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAEA,KAAK,CAACc;QAAQ,CAAC,GAAG;UAAEf,OAAO,EAAE,IAAI;UAAEc;QAAK,CAAC;MACnF;IACA,KAAK,aAAa;MAAE;QAClB,IAAI,CAAChB,IAAI,CAACyB,SAAS,EAAE,OAAO;UAAEvB,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAAqB,CAAC;QAC3E,IAAI,CAACH,IAAI,CAACwC,IAAI,EAAE,OAAO;UAAEtC,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAA6B,CAAC;QAC9E,MAAM;UAAEa,IAAI,EAAE4C;QAAM,CAAC,GAAG,MAAM7D,EAAE,CAACQ,IAAI,CAAC,QAAQ,CAAC,CAC5CC,MAAM,CAAC,gBAAgB,CAAC,CAACC,EAAE,CAAC,IAAI,EAAEL,GAAG,CAAC,CAACsB,MAAM,CAAC,CAAC;QAClD,IAAI,CAACkC,KAAK,EAAEC,cAAc,EAAE,OAAO;UAAE3D,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAA8B,CAAC;QAC3F,IAAI;UACF,MAAM6D,IAAI,GAAG,MAAMC,KAAK,CAAC,+BAA+B,EAAE;YACxDmB,MAAM,EAAE,MAAM;YACdC,OAAO,EAAE;cAAEC,aAAa,EAAE,UAAU1B,KAAK,CAACC,cAAc,EAAE;cAAE,cAAc,EAAE;YAAmB,CAAC;YAChGrB,IAAI,EAAE+C,IAAI,CAACC,SAAS,CAAC;cACnBjF,IAAI,EAAEP,IAAI,CAACO,IAAI,IAAI,8BAA8B;cACjD4C,EAAE,EAAEnD,IAAI,CAACmD,EAAY;cACrB5B,OAAO,EAAEvB,IAAI,CAACuB,OAAO,IAAI,cAAcvB,IAAI,CAACyB,SAAS,EAAE;cACvDsD,IAAI,EAAE/E,IAAI,CAACwC,IAAc;cACzB,IAAIxC,IAAI,CAACqH,WAAW,GAAG;gBAAEhC,OAAO,EAAE;kBAAE,aAAa,EAAErF,IAAI,CAACqH;gBAAY;cAAE,CAAC,GAAG,CAAC,CAAC;YAC9E,CAAC;UACH,CAAC,CAAC;UACF,MAAM5B,MAAM,GAAG,MAAMzB,IAAI,CAAC0B,IAAI,CAAC,CAAC;UAChC,OAAO1B,IAAI,CAACE,EAAE,GAAG;YAAEhE,OAAO,EAAE,IAAI;YAAEc,IAAI,EAAEyE;UAAO,CAAC,GAAG;YAAEvF,OAAO,EAAE,KAAK;YAAEC,KAAK,EAAEsF,MAAM,CAACtF,KAAK,IAAI;UAAe,CAAC;QAC9G,CAAC,CAAC,OAAOmH,GAAG,EAAE;UACZ,OAAO;YAAEpH,OAAO,EAAE,KAAK;YAAEC,KAAK,EAAE,iBAAiBmH,GAAG;UAAG,CAAC;QAC1D;MACF;IACA,KAAK,cAAc;MAAE;QACnB,MAAMC,OAAgC,GAAG,CAAC,CAAC;QAC3C,IAAIvH,IAAI,CAACa,MAAM,EAAE0G,OAAO,CAAC1G,MAAM,GAAGb,IAAI,CAACa,MAAM;QAC7C,IAAIb,IAAI,CAACe,QAAQ,EAAEwG,OAAO,CAACxG,QAAQ,GAAGf,IAAI,CAACe,QAAQ;QACnD,IAAIf,IAAI,CAACwH,MAAM,EAAED,OAAO,CAACE,SAAS,GAAGzH,IAAI,CAACwH,MAAM;QAChD,IAAIxH,IAAI,CAAC0H,UAAU,EAAEH,OAAO,CAACG,UAAU,GAAG1H,IAAI,CAAC0H,UAAU;QACzD,MAAM;UAAE1G,IAAI;UAAEb;QAAM,CAAC,GAAG,MAAMJ,EAAE,CAACQ,IAAI,CAAC,eAAe,CAAC,CACnDoH,MAAM,CAACJ,OAAO,CAAC,CAAC9G,EAAE,CAAC,IAAI,EAAET,IAAI,CAACyB,SAAmB,CAAC,CAAChB,EAAE,CAAC,UAAU,EAAEL,GAAG,CAAC,CAACI,MAAM,CAAC,CAAC,CAACkB,MAAM,CAAC,CAAC;QAC3F,OAAOvB,KAAK,GAAG;UAAED,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAEA,KAAK,CAACc;QAAQ,CAAC,GAAG;UAAEf,OAAO,EAAE,IAAI;UAAEc;QAAK,CAAC;MACnF;IACA,KAAK,aAAa;MAAE;QAClB,MAAM;UAAEA,IAAI;UAAEb;QAAM,CAAC,GAAG,MAAMJ,EAAE,CAACQ,IAAI,CAAC,eAAe,CAAC,CACnDC,MAAM,CAAC,2BAA2B,CAAC,CAACC,EAAE,CAAC,UAAU,EAAEL,GAAG,CAAC,CAACQ,KAAK,CAAC,IAAI,CAAC;QACtE,IAAIT,KAAK,EAAE,OAAO;UAAED,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAEA,KAAK,CAACc;QAAQ,CAAC;QAC1D,MAAM2G,KAAK,GAAG;UACZC,KAAK,EAAE7G,IAAI,CAAC0B,MAAM;UAClBoF,SAAS,EAAErI,OAAO,CAACuB,IAAI,EAAE,QAAQ,CAAC;UAClC+G,UAAU,EAAEtI,OAAO,CAACuB,IAAI,EAAE,SAAS,CAAC;UACpCgH,WAAW,EAAEvI,OAAO,CAACuB,IAAI,EAAE,UAAU;QACvC,CAAC;QACD,OAAO;UAAEd,OAAO,EAAE,IAAI;UAAEc,IAAI,EAAE4G;QAAM,CAAC;MACvC;IACA,KAAK,iBAAiB;MAAE;QACtB,IAAI,CAAC5H,IAAI,CAACiI,IAAI,EAAE,OAAO;UAAE/H,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAAmB,CAAC;QACpE,IAAI,CAACH,IAAI,CAACuB,OAAO,EAAE,OAAO;UAAErB,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAAsB,CAAC;QAC1E,MAAM+H,IAAI,GAAIlI,IAAI,CAACkI,IAAI,IAAgBlI,IAAI,CAACiI,IAAI,CAAYrD,WAAW,CAAC,CAAC,CAACC,OAAO,CAAC,aAAa,EAAE,GAAG,CAAC,CAACA,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC;QAC3H,MAAM;UAAE7D,IAAI;UAAEb;QAAM,CAAC,GAAG,MAAMJ,EAAE,CAACQ,IAAI,CAAC,iBAAiB,CAAC,CAACoF,MAAM,CAAC;UAC9DC,QAAQ,EAAExF,GAAG;UACb6H,IAAI,EAAEjI,IAAI,CAACiI,IAAc;UACzBC,IAAI;UACJ3G,OAAO,EAAEvB,IAAI,CAACuB,OAAiB;UAC/B4G,QAAQ,EAAGnI,IAAI,CAACmI,QAAQ,IAAe,SAAS;UAChDC,WAAW,EAAGpI,IAAI,CAACoI,WAAW,IAAe,IAAI;UACjD5B,YAAY,EAAGxG,IAAI,CAACwG,YAAY,IAAe,IAAI;UACnD6B,YAAY,EAAGrI,IAAI,CAACqI,YAAY,IAAe,IAAI;UACnD3B,YAAY,EAAG1G,IAAI,CAAC0G,YAAY,IAAe,IAAI;UACnD4B,SAAS,EAAE;QACb,CAAC,CAAC,CAAC9H,MAAM,CAAC,+CAA+C,CAAC,CAACkB,MAAM,CAAC,CAAC;QACnE,IAAIvB,KAAK,EAAE,OAAO;UAAED,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAEA,KAAK,CAACc;QAAQ,CAAC;QAC1D,OAAO;UAAEf,OAAO,EAAE,IAAI;UAAEc;QAAK,CAAC;MAChC;IACA,KAAK,iBAAiB;MAAE;QACtB,IAAI,CAAChB,IAAI,CAACuI,WAAW,EAAE,OAAO;UAAErI,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAA0B,CAAC;QAClF,MAAMoH,OAAgC,GAAG,CAAC,CAAC;QAC3C,IAAIvH,IAAI,CAACiI,IAAI,KAAKO,SAAS,EAAEjB,OAAO,CAACU,IAAI,GAAGjI,IAAI,CAACiI,IAAI;QACrD,IAAIjI,IAAI,CAACkI,IAAI,KAAKM,SAAS,EAAEjB,OAAO,CAACW,IAAI,GAAGlI,IAAI,CAACkI,IAAI;QACrD,IAAIlI,IAAI,CAACuB,OAAO,KAAKiH,SAAS,EAAEjB,OAAO,CAAChG,OAAO,GAAGvB,IAAI,CAACuB,OAAO;QAC9D,IAAIvB,IAAI,CAACmI,QAAQ,KAAKK,SAAS,EAAEjB,OAAO,CAACY,QAAQ,GAAGnI,IAAI,CAACmI,QAAQ;QACjE,IAAInI,IAAI,CAACoI,WAAW,KAAKI,SAAS,EAAEjB,OAAO,CAACa,WAAW,GAAGpI,IAAI,CAACoI,WAAW;QAC1E,IAAIpI,IAAI,CAACwG,YAAY,KAAKgC,SAAS,EAAEjB,OAAO,CAACf,YAAY,GAAGxG,IAAI,CAACwG,YAAY;QAC7E,IAAIxG,IAAI,CAACqI,YAAY,KAAKG,SAAS,EAAEjB,OAAO,CAACc,YAAY,GAAGrI,IAAI,CAACqI,YAAY;QAC7E,IAAIrI,IAAI,CAAC0G,YAAY,KAAK8B,SAAS,EAAEjB,OAAO,CAACb,YAAY,GAAG1G,IAAI,CAAC0G,YAAY;QAC7E,IAAII,MAAM,CAAC2B,IAAI,CAAClB,OAAO,CAAC,CAAC7E,MAAM,KAAK,CAAC,EAAE,OAAO;UAAExC,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAAsB,CAAC;QAC9F,MAAM;UAAEa,IAAI;UAAEb;QAAM,CAAC,GAAG,MAAMJ,EAAE,CAACQ,IAAI,CAAC,iBAAiB,CAAC,CACrDoH,MAAM,CAACJ,OAAO,CAAC,CAAC9G,EAAE,CAAC,IAAI,EAAET,IAAI,CAACuI,WAAqB,CAAC,CAAC9H,EAAE,CAAC,UAAU,EAAEL,GAAG,CAAC,CACxEI,MAAM,CAAC,+CAA+C,CAAC,CAACkB,MAAM,CAAC,CAAC;QACnE,IAAIvB,KAAK,EAAE,OAAO;UAAED,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAEA,KAAK,CAACc;QAAQ,CAAC;QAC1D,OAAO;UAAEf,OAAO,EAAE,IAAI;UAAEc;QAAK,CAAC;MAChC;IACA,KAAK,iBAAiB;MAAE;QACtB,IAAI,CAAChB,IAAI,CAACuI,WAAW,EAAE,OAAO;UAAErI,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAA0B,CAAC;QAClF,MAAM;UAAEa,IAAI;UAAEb;QAAM,CAAC,GAAG,MAAMJ,EAAE,CAACQ,IAAI,CAAC,iBAAiB,CAAC,CACrDoH,MAAM,CAAC;UAAEW,SAAS,EAAE;QAAM,CAAC,CAAC,CAAC7H,EAAE,CAAC,IAAI,EAAET,IAAI,CAACuI,WAAqB,CAAC,CAAC9H,EAAE,CAAC,UAAU,EAAEL,GAAG,CAAC,CACrFI,MAAM,CAAC,UAAU,CAAC,CAACkB,MAAM,CAAC,CAAC;QAC9B,IAAIvB,KAAK,EAAE,OAAO;UAAED,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAEA,KAAK,CAACc;QAAQ,CAAC;QAC1D,OAAO;UAAEf,OAAO,EAAE,IAAI;UAAEc,IAAI,EAAE;YAAE,GAAGA,IAAI;YAAE0H,OAAO,EAAE;UAAK;QAAE,CAAC;MAC5D;IACA;MACE,OAAO;QAAExI,OAAO,EAAE,KAAK;QAAEC,KAAK,EAAE,yBAAyBH,IAAI,CAACK,MAAM;MAAkK,CAAC;EAC3O;AACF","ignoreList":[]}
1
+ {"version":3,"file":"comms-email.js","names":["sanitizeFilterValue","groupBy","validateUrl","MAX_ATTACHMENT_BYTES","MAX_TOTAL_ATTACHMENT_BYTES","MAX_ATTACHMENT_COUNT","handleEmail","sb","args","storeId","success","error","sid","action","q","from","select","eq","order","ascending","limit","status","mailbox","priority","data","message","flattened","map","row","latest_message","rest","subject","from_email","thread_id","single","messages","thread","lines","msg","sort","a","b","Date","created_at","getTime","push","to_email","slice","body","body_text","length","join","oneHourAgo","now","toISOString","count","recentSends","head","gte","to","test","bodyHtml","body_html","bodyText","attachments","att","url","urlCheck","store","resend_api_key","resolvedAttachments","totalBytes","resp","fetch","ok","buf","Buffer","arrayBuffer","filename","content","toString","fromEmail","email","store_name","toLowerCase","replace","payload","html","text","cc","String","bcc","reply_to","method","headers","Authorization","JSON","stringify","result","json","insert","store_id","resend_id","id","then","templateSlug","template","tpl","tplErr","or","tData","template_data","htmlContent","html_content","textContent","text_content","subjectLine","key","val","Object","entries","escapedKey","re","RegExp","st","ilike","email_id","in_reply_to","replyOneHourAgo","replyRecentSends","replySubject","err","updates","intent","ai_intent","ai_summary","update","stats","total","by_status","by_mailbox","by_priority","name","slug","category","description","preview_text","is_active","template_id","undefined","keys","deleted"],"sources":["../../../src/server/handlers/comms-email.ts"],"sourcesContent":["// comms-email.ts — Email handler (inbox, send, templates)\nimport type { SupabaseClient } from \"@supabase/supabase-js\";\nimport { sanitizeFilterValue, groupBy } from \"../lib/utils.js\";\nimport { validateUrl } from \"../lib/ssrf-guard.js\";\n\nconst MAX_ATTACHMENT_BYTES = 10 * 1024 * 1024; // 10MB per attachment\nconst MAX_TOTAL_ATTACHMENT_BYTES = 25 * 1024 * 1024; // 25MB total\nconst MAX_ATTACHMENT_COUNT = 10;\n\nexport async function handleEmail(sb: SupabaseClient, args: Record<string, unknown>, storeId?: string) {\n if (!storeId) return { success: false, error: \"store_id required\" };\n const sid = storeId as string;\n switch (args.action) {\n case \"inbox\": {\n let q = sb.from(\"email_threads\").select(\"*, latest_message:email_inbox(subject, from_email, created_at)\")\n .eq(\"store_id\", sid).order(\"updated_at\", { ascending: false }).limit(args.limit as number || 25);\n if (args.status) q = q.eq(\"status\", args.status as string);\n if (args.mailbox) q = q.eq(\"mailbox\", args.mailbox as string);\n if (args.priority) q = q.eq(\"priority\", args.priority as string);\n const { data, error } = await q;\n if (error) return { success: false, error: error.message };\n // Flatten latest_message join so subject/from appear as table columns\n const flattened = (data || []).map((row: any) => {\n const { latest_message, ...rest } = row;\n return { ...rest, subject: latest_message?.subject || null, from_email: latest_message?.from_email || null };\n });\n return { success: true, data: flattened };\n }\n case \"inbox_get\": {\n const { data, error } = await sb.from(\"email_threads\")\n .select(\"*, messages:email_inbox(id, subject, from_email, to_email, body_text, created_at)\").eq(\"id\", args.thread_id as string).eq(\"store_id\", sid).single();\n if (error) return { success: false, error: error.message };\n // Pre-format so messages are visible (formatter handles sub-tables but body gets truncated)\n const { messages, ...thread } = data as any;\n const lines: string[] = [\n `## Email Thread`,\n `**Status**: ${thread.status || \"\\u2014\"} | **Mailbox**: ${thread.mailbox || \"\\u2014\"} | **Priority**: ${thread.priority || \"\\u2014\"}`,\n \"\",\n ];\n for (const msg of (messages || []).sort((a: any, b: any) => new Date(a.created_at).getTime() - new Date(b.created_at).getTime())) {\n lines.push(`### ${msg.from_email} \\u2192 ${msg.to_email || \"\\u2014\"} (${msg.created_at?.slice(0, 16) || \"\\u2014\"})`);\n lines.push(`**Subject**: ${msg.subject || \"(no subject)\"}`);\n const body = (msg.body_text || \"\").slice(0, 500);\n lines.push(body + (msg.body_text?.length > 500 ? \"...\" : \"\"));\n lines.push(\"\");\n }\n return { success: true, data: lines.join(\"\\n\") };\n }\n case \"send\": {\n // P0 FIX: Per-store email rate limit (100 emails/hour)\n const oneHourAgo = new Date(Date.now() - 3600_000).toISOString();\n const { count: recentSends } = await sb.from(\"email_sends\")\n .select(\"id\", { count: \"exact\", head: true })\n .eq(\"store_id\", sid)\n .gte(\"created_at\", oneHourAgo);\n if ((recentSends ?? 0) >= 100) {\n return { success: false, error: \"Rate limit exceeded: maximum 100 emails per hour per store\" };\n }\n\n const to = args.to as string;\n if (!to) return { success: false, error: \"to (email address) is required\" };\n // P0 FIX: Sanitize email recipient to prevent header injection\n if (/[\\r\\n]/.test(to)) return { success: false, error: \"Invalid recipient email address\" };\n const subject = (args.subject as string) || \"No Subject\";\n // P0 FIX: Sanitize subject to prevent header injection\n if (/[\\r\\n]/.test(subject)) return { success: false, error: \"Subject contains invalid characters\" };\n const bodyHtml = (args.body_html as string) || \"\";\n const bodyText = (args.body_text as string) || \"\";\n\n // P0 FIX: Validate attachments — reject oversized or too-many attachments\n const attachments = (args.attachments as { filename: string; url: string }[]) || [];\n if (attachments.length > MAX_ATTACHMENT_COUNT) {\n return { success: false, error: `Too many attachments (max ${MAX_ATTACHMENT_COUNT})` };\n }\n // Validate all attachment URLs are public (no SSRF into internal services)\n for (const att of attachments) {\n if (att.url) {\n const urlCheck = await validateUrl(att.url);\n if (urlCheck) return { success: false, error: `Invalid attachment URL \"${att.url}\": ${urlCheck}` };\n }\n }\n\n // Fetch store Resend API key\n const { data: store } = await sb.from(\"stores\")\n .select(\"resend_api_key, store_name, email\")\n .eq(\"id\", sid).single();\n if (!store?.resend_api_key) return { success: false, error: \"Store has no Resend API key configured\" };\n\n // Download and validate attachment sizes (best-effort — streams through Resend)\n const resolvedAttachments: Array<{ filename: string; content: string }> = [];\n let totalBytes = 0;\n for (const att of attachments) {\n if (att.url) {\n const resp = await fetch(att.url);\n if (!resp.ok) return { success: false, error: `Failed to fetch attachment: ${att.url}` };\n const buf = Buffer.from(await resp.arrayBuffer());\n if (buf.length > MAX_ATTACHMENT_BYTES) {\n return { success: false, error: `Attachment \"${att.filename}\" exceeds ${MAX_ATTACHMENT_BYTES / 1024 / 1024}MB limit` };\n }\n totalBytes += buf.length;\n if (totalBytes > MAX_TOTAL_ATTACHMENT_BYTES) {\n return { success: false, error: `Total attachment size exceeds ${MAX_TOTAL_ATTACHMENT_BYTES / 1024 / 1024}MB limit` };\n }\n resolvedAttachments.push({ filename: att.filename || \"attachment\", content: buf.toString(\"base64\") });\n }\n }\n\n const fromEmail = (args.from as string) || store.email || `${store.store_name?.toLowerCase().replace(/\\s+/g, \"\")}@mail.floradistro.com`;\n const payload: Record<string, unknown> = {\n from: fromEmail, to, subject,\n ...(bodyHtml ? { html: bodyHtml } : {}),\n ...(bodyText ? { text: bodyText } : {}),\n ...(resolvedAttachments.length ? { attachments: resolvedAttachments } : {}),\n };\n if (args.cc) {\n if (/[\\r\\n]/.test(String(args.cc))) return { success: false, error: \"cc contains invalid characters\" };\n payload.cc = args.cc;\n }\n if (args.bcc) {\n if (/[\\r\\n]/.test(String(args.bcc))) return { success: false, error: \"bcc contains invalid characters\" };\n payload.bcc = args.bcc;\n }\n if (args.reply_to) {\n if (/[\\r\\n]/.test(String(args.reply_to))) return { success: false, error: \"reply_to contains invalid characters\" };\n payload.reply_to = args.reply_to;\n }\n\n const resp = await fetch(\"https://api.resend.com/emails\", {\n method: \"POST\",\n headers: { Authorization: `Bearer ${store.resend_api_key}`, \"Content-Type\": \"application/json\" },\n body: JSON.stringify(payload),\n });\n const result = await resp.json();\n if (!resp.ok) return { success: false, error: `Email send failed: ${result.message || JSON.stringify(result)}` };\n\n // Track send for rate limiting\n await sb.from(\"email_sends\").insert({ store_id: sid, to_email: to, subject, resend_id: result.id }).then(() => {}, () => {});\n\n return { success: true, data: { id: result.id, to, subject, status: \"sent\" } };\n }\n case \"send_template\": {\n const templateSlug = args.template as string;\n if (!templateSlug) return { success: false, error: \"template slug required\" };\n\n const { data: tpl, error: tplErr } = await sb.from(\"email_templates\")\n .select(\"*\")\n .eq(\"slug\", templateSlug)\n .or(`store_id.eq.${sid},store_id.is.null`)\n .eq(\"is_active\", true)\n .limit(1)\n .single();\n if (tplErr || !tpl) return { success: false, error: `Template \"${templateSlug}\" not found` };\n\n const tData = (args.template_data as Record<string, string>) || {};\n let htmlContent = tpl.html_content as string || \"\";\n let textContent = tpl.text_content as string || \"\";\n let subjectLine = tpl.subject as string || \"\";\n for (const [key, val] of Object.entries(tData)) {\n const escapedKey = key.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n const re = new RegExp(`\\\\{\\\\{\\\\s*${escapedKey}\\\\s*\\\\}\\\\}`, \"g\");\n htmlContent = htmlContent.replace(re, val);\n textContent = textContent.replace(re, val);\n subjectLine = subjectLine.replace(re, val);\n }\n\n return handleEmail(sb, {\n action: \"send\",\n to: args.to,\n subject: subjectLine,\n body_html: htmlContent,\n body_text: textContent,\n from: args.from,\n cc: args.cc,\n bcc: args.bcc,\n }, storeId);\n }\n case \"list\": {\n let q = sb.from(\"email_sends\")\n .select(\"id, to_email, subject, status, resend_id, created_at\")\n .eq(\"store_id\", sid).order(\"created_at\", { ascending: false });\n if (args.to) { const st = sanitizeFilterValue(args.to as string); q = q.ilike(\"to_email\", `%${st}%`); }\n const { data, error } = await q.limit(args.limit as number || 25);\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n case \"get\": {\n const { data, error } = await sb.from(\"email_sends\")\n .select(\"*\").eq(\"id\", args.email_id as string).eq(\"store_id\", sid).single();\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n case \"templates\": {\n const { data, error } = await sb.from(\"email_templates\")\n .select(\"id, name, slug, subject, category, description, preview_text, is_active, created_at\")\n .or(`store_id.eq.${sid},store_id.is.null`)\n .eq(\"is_active\", true)\n .order(\"name\");\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n case \"inbox_reply\": {\n if (!args.thread_id) return { success: false, error: \"thread_id required\" };\n if (!args.body) return { success: false, error: \"body (reply text) required\" };\n\n // CRLF injection checks\n if (args.to && /[\\r\\n]/.test(String(args.to))) return { success: false, error: \"Invalid recipient email address\" };\n if (args.subject && /[\\r\\n]/.test(String(args.subject))) return { success: false, error: \"Subject contains invalid characters\" };\n if (args.from && /[\\r\\n]/.test(String(args.from))) return { success: false, error: \"from contains invalid characters\" };\n if (args.in_reply_to && /[\\r\\n]/.test(String(args.in_reply_to))) return { success: false, error: \"in_reply_to contains invalid characters\" };\n\n // Per-store email rate limit (100 emails/hour) — same as send action\n const replyOneHourAgo = new Date(Date.now() - 3600_000).toISOString();\n const { count: replyRecentSends } = await sb.from(\"email_sends\")\n .select(\"id\", { count: \"exact\", head: true })\n .eq(\"store_id\", sid)\n .gte(\"created_at\", replyOneHourAgo);\n if ((replyRecentSends ?? 0) >= 100) {\n return { success: false, error: \"Rate limit exceeded: maximum 100 emails per hour per store\" };\n }\n\n const { data: store } = await sb.from(\"stores\")\n .select(\"resend_api_key\").eq(\"id\", sid).single();\n if (!store?.resend_api_key) return { success: false, error: \"Store has no Resend API key\" };\n try {\n const resp = await fetch(\"https://api.resend.com/emails\", {\n method: \"POST\",\n headers: { Authorization: `Bearer ${store.resend_api_key}`, \"Content-Type\": \"application/json\" },\n body: JSON.stringify({\n from: args.from || `support@mail.floradistro.com`,\n to: args.to as string,\n subject: args.subject || `Re: Thread ${args.thread_id}`,\n html: args.body as string,\n ...(args.in_reply_to ? { headers: { \"In-Reply-To\": args.in_reply_to } } : {}),\n }),\n });\n const result = await resp.json();\n if (!resp.ok) return { success: false, error: result.error || \"Reply failed\" };\n\n // Track send for rate limiting\n const replySubject = (args.subject as string) || `Re: Thread ${args.thread_id}`;\n await sb.from(\"email_sends\").insert({ store_id: sid, to_email: args.to as string, subject: replySubject, resend_id: result.id }).then(() => {}, () => {});\n\n return { success: true, data: result };\n } catch (err) {\n return { success: false, error: `Reply failed: ${err}` };\n }\n }\n case \"inbox_update\": {\n const updates: Record<string, unknown> = {};\n if (args.status) updates.status = args.status;\n if (args.priority) updates.priority = args.priority;\n if (args.intent) updates.ai_intent = args.intent;\n if (args.ai_summary) updates.ai_summary = args.ai_summary;\n const { data, error } = await sb.from(\"email_threads\")\n .update(updates).eq(\"id\", args.thread_id as string).eq(\"store_id\", sid).select().single();\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n case \"inbox_stats\": {\n const { data, error } = await sb.from(\"email_threads\")\n .select(\"status, mailbox, priority\").eq(\"store_id\", sid).limit(1000);\n if (error) return { success: false, error: error.message };\n const stats = {\n total: data.length,\n by_status: groupBy(data, \"status\"),\n by_mailbox: groupBy(data, \"mailbox\"),\n by_priority: groupBy(data, \"priority\")\n };\n return { success: true, data: stats };\n }\n case \"create_template\": {\n if (!args.name) return { success: false, error: \"name is required\" };\n if (!args.subject) return { success: false, error: \"subject is required\" };\n const slug = (args.slug as string) || (args.name as string).toLowerCase().replace(/[^a-z0-9]+/g, \"-\").replace(/^-|-$/g, \"\");\n const { data, error } = await sb.from(\"email_templates\").insert({\n store_id: sid,\n name: args.name as string,\n slug,\n subject: args.subject as string,\n category: (args.category as string) || \"general\",\n description: (args.description as string) || null,\n html_content: (args.html_content as string) || null,\n preview_text: (args.preview_text as string) || null,\n text_content: (args.text_content as string) || null,\n is_active: true,\n }).select(\"id, name, slug, subject, category, created_at\").single();\n if (error) return { success: false, error: error.message };\n return { success: true, data };\n }\n case \"update_template\": {\n if (!args.template_id) return { success: false, error: \"template_id is required\" };\n const updates: Record<string, unknown> = {};\n if (args.name !== undefined) updates.name = args.name;\n if (args.slug !== undefined) updates.slug = args.slug;\n if (args.subject !== undefined) updates.subject = args.subject;\n if (args.category !== undefined) updates.category = args.category;\n if (args.description !== undefined) updates.description = args.description;\n if (args.html_content !== undefined) updates.html_content = args.html_content;\n if (args.preview_text !== undefined) updates.preview_text = args.preview_text;\n if (args.text_content !== undefined) updates.text_content = args.text_content;\n if (Object.keys(updates).length === 0) return { success: false, error: \"No fields to update\" };\n const { data, error } = await sb.from(\"email_templates\")\n .update(updates).eq(\"id\", args.template_id as string).eq(\"store_id\", sid)\n .select(\"id, name, slug, subject, category, updated_at\").single();\n if (error) return { success: false, error: error.message };\n return { success: true, data };\n }\n case \"delete_template\": {\n if (!args.template_id) return { success: false, error: \"template_id is required\" };\n const { data, error } = await sb.from(\"email_templates\")\n .update({ is_active: false }).eq(\"id\", args.template_id as string).eq(\"store_id\", sid)\n .select(\"id, name\").single();\n if (error) return { success: false, error: error.message };\n return { success: true, data: { ...data, deleted: true } };\n }\n default:\n return { success: false, error: `Unknown email action: ${args.action}. Valid: send, send_template, list, get, templates, inbox, inbox_get, inbox_reply, inbox_update, inbox_stats, create_template, update_template, delete_template` };\n }\n}\n"],"mappings":"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","ignoreList":[]}
@@ -4,33 +4,71 @@ import { applyGenerationRules, generateCannabinoidData, applyCalculations, runVa
4
4
  import { renderLayoutToPdf, renderHtmlToPdf, renderLabelToPdf } from "../lib/react-pdf-layout.js";
5
5
  import { renderCOAToPdf } from "../lib/coa-renderer.js";
6
6
  import QRCode from "qrcode";
7
- import { handleBrowser } from "./browser.js";
7
+ import { createCanvas } from "@napi-rs/canvas";
8
8
  const MAX_ATTACHMENT_BYTES = 10 * 1024 * 1024; // 10MB per attachment
9
9
  const MAX_TOTAL_ATTACHMENT_BYTES = 25 * 1024 * 1024; // 25MB total
10
10
  const MAX_ATTACHMENT_COUNT = 10;
11
+ const THUMB_HEIGHT = 400;
12
+ class NodeCanvasFactory {
13
+ create(w, h) {
14
+ const canvas = createCanvas(w, h);
15
+ return {
16
+ canvas,
17
+ context: canvas.getContext("2d")
18
+ };
19
+ }
20
+ reset(cc, w, h) {
21
+ cc.canvas.width = w;
22
+ cc.canvas.height = h;
23
+ }
24
+ destroy(cc) {
25
+ cc.canvas.width = 0;
26
+ cc.canvas.height = 0;
27
+ }
28
+ }
11
29
 
12
- /** Generate a PNG thumbnail for a PDF document via Playwright screenshot and store it in Supabase. */
30
+ /** Generate a PNG thumbnail for a PDF document and store it in Supabase. */
13
31
  async function generateThumbnail(sb, docId, pdfUrl) {
14
- // Wrap in Google Docs viewer — headless Chromium downloads raw PDFs instead of rendering them
15
- const viewerUrl = `https://docs.google.com/gview?url=${encodeURIComponent(pdfUrl)}&embedded=true`;
16
- const result = await handleBrowser(sb, {
17
- action: "screenshot",
18
- url: viewerUrl
32
+ const pdfjs = await import("pdfjs-dist/legacy/build/pdf.mjs");
33
+ const resp = await fetch(pdfUrl);
34
+ if (!resp.ok) return;
35
+ const data = new Uint8Array(await resp.arrayBuffer());
36
+ const pdf = await pdfjs.getDocument({
37
+ data,
38
+ useSystemFonts: true,
39
+ disableFontFace: true
40
+ }).promise;
41
+ const page = await pdf.getPage(1);
42
+ const vp = page.getViewport({
43
+ scale: 1
19
44
  });
20
- const b64 = result.data?.screenshot_base64;
21
- if (!b64) return;
22
- const thumbBuffer = Buffer.from(b64, 'base64');
45
+ const scale = THUMB_HEIGHT / vp.height;
46
+ const svp = page.getViewport({
47
+ scale
48
+ });
49
+ const factory = new NodeCanvasFactory();
50
+ const {
51
+ canvas,
52
+ context
53
+ } = factory.create(Math.round(svp.width), Math.round(svp.height));
54
+ await page.render({
55
+ canvasContext: context,
56
+ viewport: svp,
57
+ canvasFactory: factory
58
+ }).promise;
59
+ const thumbBuffer = Buffer.from(canvas.toBuffer("image/png"));
60
+ pdf.destroy();
23
61
  const thumbPath = `thumbs/${docId}.png`;
24
- await sb.storage.from('screenshots').upload(thumbPath, thumbBuffer, {
25
- contentType: 'image/png',
62
+ await sb.storage.from("screenshots").upload(thumbPath, thumbBuffer, {
63
+ contentType: "image/png",
26
64
  upsert: true
27
65
  });
28
66
  const {
29
67
  data: thumbUrlData
30
- } = sb.storage.from('screenshots').getPublicUrl(thumbPath);
31
- await sb.from('store_documents').update({
68
+ } = sb.storage.from("screenshots").getPublicUrl(thumbPath);
69
+ await sb.from("store_documents").update({
32
70
  thumbnail_url: thumbUrlData.publicUrl
33
- }).eq('id', docId);
71
+ }).eq("id", docId);
34
72
  }
35
73
  export async function handleEmail(sb, args, storeId) {
36
74
  if (!storeId) return {
@@ -586,7 +624,7 @@ export async function handleDocuments(sb, args, storeId) {
586
624
  success: false,
587
625
  error: insertErr.message
588
626
  };
589
- if (fileUrl.endsWith('.pdf')) await generateThumbnail(sb, record.id, fileUrl).catch(() => {});
627
+ if (fileUrl.endsWith('.pdf')) await generateThumbnail(sb, record.id, fileUrl).catch(err => console.error("[thumbnail]", err.message));
590
628
  return {
591
629
  success: true,
592
630
  data: {
@@ -829,7 +867,7 @@ export async function handleDocuments(sb, args, storeId) {
829
867
  success: false,
830
868
  error: insertErr.message
831
869
  };
832
- if (urlData.publicUrl.endsWith('.pdf')) await generateThumbnail(sb, record.id, urlData.publicUrl).catch(() => {});
870
+ if (urlData.publicUrl.endsWith('.pdf')) await generateThumbnail(sb, record.id, urlData.publicUrl).catch(err => console.error("[thumbnail]", err.message));
833
871
  return {
834
872
  success: true,
835
873
  data: {
@@ -966,7 +1004,7 @@ export async function handleDocuments(sb, args, storeId) {
966
1004
  error: "template_id or template_slug required"
967
1005
  };
968
1006
 
969
- // 1. Load pdf_template
1007
+ // 1. Load pdf_template + store's custom QR domain
970
1008
  let tplQuery = sb.from("pdf_templates").select("*").eq("is_active", true);
971
1009
  if (templateId) tplQuery = tplQuery.eq("id", templateId);else tplQuery = tplQuery.eq("slug", templateSlug);
972
1010
  tplQuery = tplQuery.or(`store_id.eq.${sid},store_id.is.null`);
@@ -978,6 +1016,10 @@ export async function handleDocuments(sb, args, storeId) {
978
1016
  success: false,
979
1017
  error: `Template not found: ${templateId || templateSlug}`
980
1018
  };
1019
+ const {
1020
+ data: storeRow
1021
+ } = await sb.from("stores").select("qr_domain").eq("id", sid).single();
1022
+ const qrDomain = storeRow?.qr_domain || "whale-gateway.fly.dev";
981
1023
 
982
1024
  // 2. Optionally load document_profile + its client store
983
1025
  let profileConfig = {};
@@ -1173,7 +1215,7 @@ export async function handleDocuments(sb, args, storeId) {
1173
1215
  // Generate a predictable QR code — will be registered in qr_codes table after document insert
1174
1216
  const _qrCodePrefix = `DOC-${Date.now().toString(36)}`;
1175
1217
  mergedData._qrCodePrefix = _qrCodePrefix;
1176
- const resolverUrl = `https://whale-gateway.fly.dev/q/${_qrCodePrefix}`;
1218
+ const resolverUrl = `https://${qrDomain}/q/${_qrCodePrefix}`;
1177
1219
  mergedData._resolverUrl = resolverUrl;
1178
1220
  try {
1179
1221
  mergedData.qrCodeDataUrl = await QRCode.toDataURL(resolverUrl, {
@@ -1352,12 +1394,12 @@ export async function handleDocuments(sb, args, storeId) {
1352
1394
  success: false,
1353
1395
  error: insertErr.message
1354
1396
  };
1355
- await generateThumbnail(sb, record.id, urlData.publicUrl).catch(() => {});
1397
+ await generateThumbnail(sb, record.id, urlData.publicUrl).catch(err => console.error("[thumbnail]", err.message));
1356
1398
 
1357
1399
  // 14. Create unified qr_codes record for scan tracking
1358
1400
  if (mergedData._coaVerificationUrl && mergedData._qrCodePrefix) {
1359
1401
  const qrCode = mergedData._qrCodePrefix;
1360
- const resolverUrl = `https://whale-gateway.fly.dev/q/${qrCode}`;
1402
+ const resolverUrl = `https://${qrDomain}/q/${qrCode}`;
1361
1403
  sb.from("qr_codes").insert({
1362
1404
  store_id: sid,
1363
1405
  code: qrCode,