whale-code 6.5.11 → 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 (613) 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/services/agent-loop-tools.js +10 -1
  20. package/dist/cli/services/agent-loop-tools.js.map +1 -1
  21. package/dist/cli/services/cli-agent-loop.js +3 -2
  22. package/dist/cli/services/cli-agent-loop.js.map +1 -1
  23. package/dist/cli/services/config-store.js +4 -3
  24. package/dist/cli/services/config-store.js.map +1 -1
  25. package/dist/cli/services/memory-manager.js +2 -2
  26. package/dist/cli/services/memory-manager.js.map +1 -1
  27. package/dist/cli/services/permission-modes.js +14 -10
  28. package/dist/cli/services/permission-modes.js.map +1 -1
  29. package/dist/cli/services/session-client.js +2 -1
  30. package/dist/cli/services/session-client.js.map +1 -1
  31. package/dist/cli/services/session-persistence.js +14 -6
  32. package/dist/cli/services/session-persistence.js.map +1 -1
  33. package/dist/cli/shared/SpinnerSlot.js +4 -1
  34. package/dist/cli/shared/SpinnerSlot.js.map +1 -1
  35. package/dist/server/handlers/browser-lifecycle.js +10 -0
  36. package/dist/server/handlers/browser-lifecycle.js.map +1 -1
  37. package/dist/server/handlers/browser.js +16 -1
  38. package/dist/server/handlers/browser.js.map +1 -1
  39. package/dist/server/handlers/campaigns.js +11 -0
  40. package/dist/server/handlers/campaigns.js.map +1 -1
  41. package/dist/server/handlers/catalog-products.js +19 -5
  42. package/dist/server/handlers/catalog-products.js.map +1 -1
  43. package/dist/server/handlers/catalog.js +42 -8
  44. package/dist/server/handlers/catalog.js.map +1 -1
  45. package/dist/server/handlers/clickhouse.js +4 -4
  46. package/dist/server/handlers/clickhouse.js.map +1 -1
  47. package/dist/server/handlers/comms-email.js +70 -8
  48. package/dist/server/handlers/comms-email.js.map +1 -1
  49. package/dist/server/handlers/comms.js +63 -21
  50. package/dist/server/handlers/comms.js.map +1 -1
  51. package/dist/server/handlers/coupons.js +141 -77
  52. package/dist/server/handlers/coupons.js.map +1 -1
  53. package/dist/server/handlers/google-ads.js +280 -8
  54. package/dist/server/handlers/google-ads.js.map +1 -1
  55. package/dist/server/handlers/remove-bg.d.ts +33 -0
  56. package/dist/server/handlers/remove-bg.js +698 -44
  57. package/dist/server/handlers/remove-bg.js.map +1 -1
  58. package/dist/server/handlers/supply-chain.js +93 -1
  59. package/dist/server/handlers/supply-chain.js.map +1 -1
  60. package/dist/server/handlers/workflow-steps-types.d.ts +1 -1
  61. package/dist/server/handlers/workflow-steps-types.js +7 -1
  62. package/dist/server/handlers/workflow-steps-types.js.map +1 -1
  63. package/dist/server/handlers/workflow-steps.js +1 -1
  64. package/dist/server/handlers/workflow-steps.js.map +1 -1
  65. package/dist/server/index.js +122 -29
  66. package/dist/server/index.js.map +1 -1
  67. package/dist/server/lib/agent-loop-turn.js +33 -3
  68. package/dist/server/lib/agent-loop-turn.js.map +1 -1
  69. package/dist/server/lib/agent-loop-types.d.ts +6 -2
  70. package/dist/server/lib/agent-loop-types.js +14 -2
  71. package/dist/server/lib/agent-loop-types.js.map +1 -1
  72. package/dist/server/lib/clickhouse-client.js +4 -2
  73. package/dist/server/lib/clickhouse-client.js.map +1 -1
  74. package/dist/server/lib/code-worker.js +4 -1
  75. package/dist/server/lib/code-worker.js.map +1 -1
  76. package/dist/server/providers/anthropic.js +103 -33
  77. package/dist/server/providers/anthropic.js.map +1 -1
  78. package/dist/server/server-chat.js +2 -2
  79. package/dist/server/server-chat.js.map +1 -1
  80. package/dist/server/server-helpers.d.ts +8 -1
  81. package/dist/server/server-helpers.js +17 -3
  82. package/dist/server/server-helpers.js.map +1 -1
  83. package/dist/server/server-persist.js +34 -21
  84. package/dist/server/server-persist.js.map +1 -1
  85. package/dist/server/server-rate-limit.d.ts +0 -1
  86. package/dist/server/server-rate-limit.js +5 -5
  87. package/dist/server/server-rate-limit.js.map +1 -1
  88. package/dist/server/server-routes-approvals.js +2 -2
  89. package/dist/server/server-routes-approvals.js.map +1 -1
  90. package/dist/server/server-routes-auth.js +2 -2
  91. package/dist/server/server-routes-auth.js.map +1 -1
  92. package/dist/server/server-routes-events.js +2 -2
  93. package/dist/server/server-routes-events.js.map +1 -1
  94. package/dist/server/server-routes-public.js +4 -4
  95. package/dist/server/server-routes-public.js.map +1 -1
  96. package/dist/server/server-routes-webchat.js +3 -3
  97. package/dist/server/server-routes-webchat.js.map +1 -1
  98. package/dist/server/server-store-circuit-breaker.js +1 -1
  99. package/dist/server/server-store-circuit-breaker.js.map +1 -1
  100. package/dist/server/tool-router.js +7 -4
  101. package/dist/server/tool-router.js.map +1 -1
  102. package/dist/server/validation.js +11 -0
  103. package/dist/server/validation.js.map +1 -1
  104. package/dist/shared/api-client.js +38 -11
  105. package/dist/shared/api-client.js.map +1 -1
  106. package/package.json +12 -10
  107. package/vendor/ink/build/ink.js +68 -24
  108. package/vendor/ink/node_modules/react-devtools-core/README.md +152 -0
  109. package/vendor/ink/node_modules/react-devtools-core/backend.js +1 -0
  110. package/vendor/ink/node_modules/react-devtools-core/dist/648.chunk.js +2 -0
  111. package/vendor/ink/node_modules/react-devtools-core/dist/648.chunk.js.map +1 -0
  112. package/vendor/ink/node_modules/react-devtools-core/dist/backend.js +15691 -0
  113. package/vendor/ink/node_modules/react-devtools-core/dist/backend.js.map +1 -0
  114. package/vendor/ink/node_modules/react-devtools-core/dist/importFile.worker.worker.js +2 -0
  115. package/vendor/ink/node_modules/react-devtools-core/dist/importFile.worker.worker.js.map +1 -0
  116. package/vendor/ink/node_modules/react-devtools-core/dist/parseSourceAndMetadata.worker.worker.js +14 -0
  117. package/vendor/ink/node_modules/react-devtools-core/dist/parseSourceAndMetadata.worker.worker.js.map +1 -0
  118. package/vendor/ink/node_modules/react-devtools-core/dist/standalone.js +2 -0
  119. package/vendor/ink/node_modules/react-devtools-core/dist/standalone.js.map +1 -0
  120. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/LICENSE +21 -0
  121. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/README.md +495 -0
  122. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/browser.js +8 -0
  123. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/index.js +10 -0
  124. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/buffer-util.js +129 -0
  125. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/constants.js +10 -0
  126. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/event-target.js +184 -0
  127. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/extension.js +223 -0
  128. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/limiter.js +55 -0
  129. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/permessage-deflate.js +518 -0
  130. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/receiver.js +607 -0
  131. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/sender.js +409 -0
  132. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/stream.js +180 -0
  133. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/validation.js +104 -0
  134. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/websocket-server.js +449 -0
  135. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/websocket.js +1197 -0
  136. package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/package.json +56 -0
  137. package/vendor/ink/node_modules/react-devtools-core/package.json +38 -0
  138. package/vendor/ink/node_modules/react-devtools-core/standalone.js +1 -0
  139. package/dist/cli/__tests__/print-mode-streaming.test.js +0 -270
  140. package/dist/cli/__tests__/print-mode.basic-output.test.js +0 -230
  141. package/dist/cli/__tests__/print-mode.session-errors.test.js +0 -252
  142. package/dist/cli/__tests__/print-mode.test.js +0 -273
  143. package/dist/cli/__tests__/serve-mode-messages.test.js +0 -338
  144. package/dist/cli/__tests__/serve-mode.messages.part2.test.js +0 -266
  145. package/dist/cli/__tests__/serve-mode.messages.test.js +0 -277
  146. package/dist/cli/__tests__/serve-mode.startup-http.test.js +0 -279
  147. package/dist/cli/__tests__/serve-mode.test.js +0 -345
  148. package/dist/cli/chat/NodeManager.d.ts +0 -30
  149. package/dist/cli/chat/NodeManager.js +0 -66
  150. package/dist/cli/chat/NodeManager.js.map +0 -1
  151. package/dist/cli/chat/chat-input-menu-handler.d.ts +0 -32
  152. package/dist/cli/chat/hooks/slash-imsg-handlers.js +0 -148
  153. package/dist/cli/chat/hooks/slash-imsg-handlers.js.map +0 -1
  154. package/dist/cli/chat/hooks/useStreamingReducer.d.ts +0 -66
  155. package/dist/cli/commands/__tests__/config-cmd.test.js +0 -270
  156. package/dist/cli/commands/__tests__/doctor.test.js +0 -257
  157. package/dist/cli/commands/__tests__/imsg-node-bridge.test.js +0 -99
  158. package/dist/cli/commands/__tests__/imsg-utils.test.js +0 -73
  159. package/dist/cli/commands/__tests__/init.test.js +0 -214
  160. package/dist/cli/commands/__tests__/mcp.test.js +0 -287
  161. package/dist/cli/commands/imsg-watcher-helpers.d.ts +0 -40
  162. package/dist/cli/commands/imsg-watcher-helpers.js +0 -184
  163. package/dist/cli/commands/imsg-watcher-helpers.js.map +0 -1
  164. package/dist/cli/commands/imsg-watcher.d.ts +0 -11
  165. package/dist/cli/commands/imsg-watcher.js +0 -230
  166. package/dist/cli/commands/imsg-watcher.js.map +0 -1
  167. package/dist/cli/services/__tests__/agent-definitions.test.js +0 -153
  168. package/dist/cli/services/__tests__/agent-events-global.test.js +0 -39
  169. package/dist/cli/services/__tests__/agent-events.part2.test.js +0 -113
  170. package/dist/cli/services/__tests__/agent-events.test.js +0 -157
  171. package/dist/cli/services/__tests__/agent-loop-auth.test.js +0 -392
  172. package/dist/cli/services/__tests__/agent-loop-budget.test.js +0 -389
  173. package/dist/cli/services/__tests__/agent-loop-tools-lifecycle.test.js +0 -430
  174. package/dist/cli/services/__tests__/agent-loop-tools-maxturns.test.js +0 -486
  175. package/dist/cli/services/__tests__/agent-loop-utils-execution.test.js +0 -528
  176. package/dist/cli/services/__tests__/agent-loop-utils-helpers.test.js +0 -466
  177. package/dist/cli/services/__tests__/agent-worker-base-execute.test.js +0 -257
  178. package/dist/cli/services/__tests__/agent-worker-base-helpers.test.js +0 -198
  179. package/dist/cli/services/__tests__/agent-worker-base.test.js +0 -278
  180. package/dist/cli/services/__tests__/auth-service-exports.test.js +0 -41
  181. package/dist/cli/services/__tests__/auth-service.part2.test.js +0 -169
  182. package/dist/cli/services/__tests__/auth-service.test.js +0 -242
  183. package/dist/cli/services/__tests__/background-processes.test.js +0 -282
  184. package/dist/cli/services/__tests__/claude-md-loader.test.js +0 -134
  185. package/dist/cli/services/__tests__/config-store.test.js +0 -247
  186. package/dist/cli/services/__tests__/debug-log.test.js +0 -199
  187. package/dist/cli/services/__tests__/edge-cases-caching.test.js +0 -174
  188. package/dist/cli/services/__tests__/edge-cases-compaction-core.test.js +0 -226
  189. package/dist/cli/services/__tests__/edge-cases-compaction-openai.test.js +0 -152
  190. package/dist/cli/services/__tests__/edge-cases-compaction-shapes.test.js +0 -53
  191. package/dist/cli/services/__tests__/edge-cases-compaction-thinking.test.js +0 -226
  192. package/dist/cli/services/__tests__/edge-cases-compaction.test.js +0 -131
  193. package/dist/cli/services/__tests__/edge-cases-paths.test.js +0 -86
  194. package/dist/cli/services/__tests__/error-logger-messages.test.js +0 -81
  195. package/dist/cli/services/__tests__/error-logger-transport.test.js +0 -119
  196. package/dist/cli/services/__tests__/error-logger.test.js +0 -264
  197. package/dist/cli/services/__tests__/file-history.test.js +0 -136
  198. package/dist/cli/services/__tests__/git-context-cache-reset.test.js +0 -223
  199. package/dist/cli/services/__tests__/git-context.test.js +0 -241
  200. package/dist/cli/services/__tests__/interactive-tools-execute.test.js +0 -166
  201. package/dist/cli/services/__tests__/interactive-tools-plan.test.js +0 -197
  202. package/dist/cli/services/__tests__/interactive-tools.part2.test.js +0 -168
  203. package/dist/cli/services/__tests__/interactive-tools.test.js +0 -179
  204. package/dist/cli/services/__tests__/keybinding-manager.test.js +0 -205
  205. package/dist/cli/services/__tests__/local-tools-dispatch.test.js +0 -404
  206. package/dist/cli/services/__tests__/local-tools.test.js +0 -238
  207. package/dist/cli/services/__tests__/lsp-manager.test.js +0 -364
  208. package/dist/cli/services/__tests__/mcp-client-connect-disconnect.test.js +0 -310
  209. package/dist/cli/services/__tests__/mcp-client.test.js +0 -93
  210. package/dist/cli/services/__tests__/memory-manager.test.js +0 -154
  211. package/dist/cli/services/__tests__/model-manager-utils.test.js +0 -154
  212. package/dist/cli/services/__tests__/model-manager.test.js +0 -175
  213. package/dist/cli/services/__tests__/permission-modes.test.js +0 -222
  214. package/dist/cli/services/__tests__/ripgrep.test.js +0 -328
  215. package/dist/cli/services/__tests__/server-tools-execute.test.js +0 -317
  216. package/dist/cli/services/__tests__/server-tools.test.js +0 -272
  217. package/dist/cli/services/__tests__/session-persistence.test.js +0 -245
  218. package/dist/cli/services/__tests__/subagent-basic.test.js +0 -489
  219. package/dist/cli/services/__tests__/subagent-edge.test.js +0 -545
  220. package/dist/cli/services/__tests__/subagent-prompts.test.js +0 -558
  221. package/dist/cli/services/__tests__/subagent-worker-errors.test.js +0 -255
  222. package/dist/cli/services/__tests__/subagent-worker.test.js +0 -242
  223. package/dist/cli/services/__tests__/system-prompt.test.js +0 -210
  224. package/dist/cli/services/__tests__/team-lead-comms-messaging.test.js +0 -250
  225. package/dist/cli/services/__tests__/team-lead-comms-result.test.js +0 -232
  226. package/dist/cli/services/__tests__/team-lead-comms-stop.test.js +0 -344
  227. package/dist/cli/services/__tests__/team-lead-comms.test.js +0 -285
  228. package/dist/cli/services/__tests__/team-lead-create.test.js +0 -327
  229. package/dist/cli/services/__tests__/team-lead-run.test.js +0 -318
  230. package/dist/cli/services/__tests__/team-lead-stop.test.js +0 -199
  231. package/dist/cli/services/__tests__/team-state-comms.test.js +0 -240
  232. package/dist/cli/services/__tests__/team-state-core.test.js +0 -230
  233. package/dist/cli/services/__tests__/team-state-tasks-complete-fail-available.test.js +0 -224
  234. package/dist/cli/services/__tests__/team-state-tasks.test.js +0 -184
  235. package/dist/cli/services/__tests__/telemetry-ai-metadata.test.js +0 -116
  236. package/dist/cli/services/__tests__/telemetry.part2.test.js +0 -195
  237. package/dist/cli/services/__tests__/telemetry.test.js +0 -176
  238. package/dist/cli/services/agent-loop-iteration.d.ts +0 -13
  239. package/dist/cli/services/agent-loop-setup.d.ts +0 -32
  240. package/dist/cli/services/agent-worker-base-api.d.ts +0 -19
  241. package/dist/cli/services/agent-worker-base-helpers.d.ts +0 -27
  242. package/dist/cli/services/agent-worker-base-tools.d.ts +0 -16
  243. package/dist/cli/services/agent-worker-base-types.d.ts +0 -81
  244. package/dist/cli/services/background-agents.d.ts +0 -26
  245. package/dist/cli/services/background-processes-ops.d.ts +0 -24
  246. package/dist/cli/services/background-tool-defs.d.ts +0 -50
  247. package/dist/cli/services/config-modules-model.test.js +0 -133
  248. package/dist/cli/services/config-modules-permission.test.js +0 -85
  249. package/dist/cli/services/config-modules-permissions.test.js +0 -85
  250. package/dist/cli/services/config-modules-session.test.js +0 -297
  251. package/dist/cli/services/format-server-response-columns.test.js +0 -265
  252. package/dist/cli/services/format-server-response-fallback.test.js +0 -65
  253. package/dist/cli/services/format-server-response-primitives-basic.test.js +0 -261
  254. package/dist/cli/services/format-server-response-primitives-nested.test.js +0 -188
  255. package/dist/cli/services/format-server-response-primitives.test.js +0 -300
  256. package/dist/cli/services/format-server-response-realworld.test.js +0 -248
  257. package/dist/cli/services/format-server-response-values.test.js +0 -247
  258. package/dist/cli/services/hooks-runners.test.js +0 -184
  259. package/dist/cli/services/hooks.glob-load.test.js +0 -233
  260. package/dist/cli/services/hooks.run-hooks.test.js +0 -184
  261. package/dist/cli/services/hooks.test.js +0 -233
  262. package/dist/cli/services/ink-incremental.d.ts +0 -19
  263. package/dist/cli/services/ink-incremental.js +0 -59
  264. package/dist/cli/services/ink-incremental.js.map +0 -1
  265. package/dist/cli/services/ink-resize-fix.d.ts +0 -18
  266. package/dist/cli/services/ink-resize-fix.js +0 -76
  267. package/dist/cli/services/ink-resize-fix.js.map +0 -1
  268. package/dist/cli/services/ink-sync-output.d.ts +0 -12
  269. package/dist/cli/services/ink-sync-output.js +0 -16
  270. package/dist/cli/services/ink-sync-output.js.map +0 -1
  271. package/dist/cli/services/interactive-tool-defs.d.ts +0 -80
  272. package/dist/cli/services/local-tools-definitions.d.ts +0 -6
  273. package/dist/cli/services/local-tools-files.test.js +0 -256
  274. package/dist/cli/services/local-tools-read-many.d.ts +0 -6
  275. package/dist/cli/services/model-router.test.js +0 -245
  276. package/dist/cli/services/rewind-rewindTo.test.js +0 -202
  277. package/dist/cli/services/rewind.test.js +0 -175
  278. package/dist/cli/services/sandbox.test.js +0 -198
  279. package/dist/cli/services/subagent-execution.d.ts +0 -12
  280. package/dist/cli/services/team-lead-auto.d.ts +0 -11
  281. package/dist/cli/services/team-lead-execution.d.ts +0 -28
  282. package/dist/cli/services/teammate-loop.js +0 -557
  283. package/dist/cli/services/teammate-loop.js.map +0 -1
  284. package/dist/cli/services/tools/__tests__/agent-tools-tasks-teams.test.js +0 -250
  285. package/dist/cli/services/tools/__tests__/agent-tools-teams.test.js +0 -200
  286. package/dist/cli/services/tools/__tests__/agent-tools.test.js +0 -340
  287. package/dist/cli/services/tools/__tests__/file-ops-cache.test.js +0 -152
  288. package/dist/cli/services/tools/__tests__/file-ops-notebook.test.js +0 -249
  289. package/dist/cli/services/tools/__tests__/file-ops-read.test.js +0 -261
  290. package/dist/cli/services/tools/__tests__/file-ops-write.test.js +0 -292
  291. package/dist/cli/services/tools/__tests__/search-tools-rg.test.js +0 -92
  292. package/dist/cli/services/tools/__tests__/search-tools.part2.test.js +0 -174
  293. package/dist/cli/services/tools/__tests__/search-tools.test.js +0 -227
  294. package/dist/cli/services/tools/__tests__/shell-exec-allowed-core.test.js +0 -163
  295. package/dist/cli/services/tools/__tests__/shell-exec-allowed-extended.test.js +0 -220
  296. package/dist/cli/services/tools/__tests__/shell-exec-allowed.part2.test.js +0 -215
  297. package/dist/cli/services/tools/__tests__/shell-exec-allowed.test.js +0 -154
  298. package/dist/cli/services/tools/__tests__/shell-exec-blocked.test.js +0 -132
  299. package/dist/cli/services/tools/__tests__/shell-exec-execution.test.js +0 -245
  300. package/dist/cli/services/tools/__tests__/task-manager-create.test.js +0 -110
  301. package/dist/cli/services/tools/__tests__/task-manager-crud.test.js +0 -339
  302. package/dist/cli/services/tools/__tests__/task-manager-list-get.test.js +0 -343
  303. package/dist/cli/services/tools/__tests__/task-manager-query.test.js +0 -346
  304. package/dist/cli/services/tools/__tests__/task-manager-routing.test.js +0 -58
  305. package/dist/cli/services/tools/__tests__/task-manager-update.test.js +0 -224
  306. package/dist/cli/services/tools/__tests__/task-manager.test.js +0 -159
  307. package/dist/cli/services/tools/__tests__/web-tools-html-search.test.js +0 -227
  308. package/dist/cli/services/tools/__tests__/web-tools.test.js +0 -285
  309. package/dist/cli/services/tools/shell-exec.test.js +0 -148
  310. package/dist/cli/shared/SharedTick.d.ts +0 -10
  311. package/dist/cli/shared/__tests__/markdown.test.js +0 -188
  312. package/dist/local-agent/__tests__/connection-disconnect.test.js +0 -201
  313. package/dist/local-agent/__tests__/connection-lifecycle.test.js +0 -289
  314. package/dist/local-agent/__tests__/connection-msghandling.test.js +0 -311
  315. package/dist/local-agent/__tests__/connection-reconnect.test.js +0 -230
  316. package/dist/local-agent/__tests__/connection-toolexec.test.js +0 -253
  317. package/dist/local-agent/__tests__/discovery.test.js +0 -328
  318. package/dist/local-agent/__tests__/executor-background.test.js +0 -219
  319. package/dist/local-agent/__tests__/executor-exec.test.js +0 -221
  320. package/dist/local-agent/__tests__/executor-jobs-sessions.test.js +0 -220
  321. package/dist/local-agent/__tests__/executor-system-info.test.js +0 -133
  322. package/dist/local-agent/__tests__/executor-systeminfo.test.js +0 -109
  323. package/dist/local-agent/__tests__/executor.test.js +0 -235
  324. package/dist/local-agent/__tests__/index.test.js +0 -139
  325. package/dist/node/__tests__/cli-channels.test.js +0 -293
  326. package/dist/node/__tests__/cli-config-edge.test.js +0 -154
  327. package/dist/node/__tests__/cli-config.test.js +0 -215
  328. package/dist/node/__tests__/config.test.js +0 -292
  329. package/dist/node/__tests__/runtime-heartbeat.test.js +0 -153
  330. package/dist/node/__tests__/runtime-lifecycle-init.test.js +0 -263
  331. package/dist/node/__tests__/runtime-lifecycle-stats.test.js +0 -180
  332. package/dist/node/__tests__/runtime-lifecycle.test.js +0 -305
  333. package/dist/node/__tests__/runtime-relay.test.js +0 -341
  334. package/dist/node/adapters/__tests__/base.test.js +0 -286
  335. package/dist/node/adapters/__tests__/discord.test.js +0 -284
  336. package/dist/node/adapters/__tests__/email-send.test.js +0 -295
  337. package/dist/node/adapters/__tests__/email.inbound-send.test.js +0 -217
  338. package/dist/node/adapters/__tests__/email.lifecycle.test.js +0 -211
  339. package/dist/node/adapters/__tests__/email.test.js +0 -290
  340. package/dist/node/adapters/__tests__/email.webhook-send.test.js +0 -251
  341. package/dist/node/adapters/__tests__/imessage-filter.test.js +0 -183
  342. package/dist/node/adapters/__tests__/imessage-lifecycle.test.js +0 -215
  343. package/dist/node/adapters/__tests__/imessage-send-restart.test.js +0 -227
  344. package/dist/node/adapters/__tests__/slack.part2.test.js +0 -135
  345. package/dist/node/adapters/__tests__/slack.test.js +0 -241
  346. package/dist/node/adapters/__tests__/sms-extras.test.js +0 -108
  347. package/dist/node/adapters/__tests__/sms-lifecycle.test.js +0 -203
  348. package/dist/node/adapters/__tests__/sms-messaging.test.js +0 -266
  349. package/dist/node/adapters/__tests__/sms.part2.test.js +0 -174
  350. package/dist/node/adapters/__tests__/sms.test.js +0 -253
  351. package/dist/node/adapters/__tests__/telegram-polling.test.js +0 -256
  352. package/dist/node/adapters/__tests__/telegram-send.test.js +0 -166
  353. package/dist/node/adapters/__tests__/webchat-inbound.test.js +0 -188
  354. package/dist/node/adapters/__tests__/webchat-outbound.test.js +0 -178
  355. package/dist/node/adapters/__tests__/whatsapp-inbound.test.js +0 -200
  356. package/dist/node/adapters/__tests__/whatsapp-send.test.js +0 -212
  357. package/dist/node/adapters/__tests__/whatsapp.test.js +0 -280
  358. package/dist/server/__tests__/gateway-fast-fail.test.js +0 -160
  359. package/dist/server/__tests__/local-agent-gateway.test.js +0 -186
  360. package/dist/server/__tests__/proxy-handlers-delegation.test.js +0 -240
  361. package/dist/server/__tests__/proxy-handlers-validation.test.js +0 -211
  362. package/dist/server/__tests__/proxy-handlers.part2.test.js +0 -240
  363. package/dist/server/__tests__/proxy-handlers.test.js +0 -213
  364. package/dist/server/__tests__/strip-base64-e2e.test.js +0 -303
  365. package/dist/server/__tests__/strip-base64.test.js +0 -256
  366. package/dist/server/__tests__/tool-router-agent-tools.test.js +0 -324
  367. package/dist/server/__tests__/tool-router-execute-core.test.js +0 -357
  368. package/dist/server/__tests__/tool-router-execute-permissions.test.js +0 -332
  369. package/dist/server/__tests__/tool-router-execute.test.js +0 -348
  370. package/dist/server/__tests__/tool-router-load.test.js +0 -432
  371. package/dist/server/__tests__/tool-router-permissions.test.js +0 -359
  372. package/dist/server/__tests__/tool-router-registry-cache.test.js +0 -383
  373. package/dist/server/__tests__/tool-router-registry-handlers.test.js +0 -272
  374. package/dist/server/__tests__/tool-router-registry.test.js +0 -331
  375. package/dist/server/__tests__/validation-inventory.test.js +0 -250
  376. package/dist/server/__tests__/validation-misc.test.js +0 -243
  377. package/dist/server/__tests__/validation-supply-chain.test.js +0 -188
  378. package/dist/server/__tests__/worker.test.js +0 -265
  379. package/dist/server/handlers/__tests__/conversation-lock.test.js +0 -117
  380. package/dist/server/handlers/__tests__/e2e/auth-cross-platform-login.e2e.test.js +0 -268
  381. package/dist/server/handlers/__tests__/e2e/auth-cross-platform-tokens.e2e.test.js +0 -264
  382. package/dist/server/handlers/__tests__/e2e/email-pipeline-send.e2e.test.js +0 -214
  383. package/dist/server/handlers/__tests__/e2e/email-pipeline-threads.e2e.test.js +0 -168
  384. package/dist/server/handlers/__tests__/e2e/error-logging-pipeline-dedup.e2e.test.js +0 -229
  385. package/dist/server/handlers/__tests__/e2e/error-logging-pipeline.e2e.test.js +0 -239
  386. package/dist/server/handlers/__tests__/e2e/error-logging-rate-limit.e2e.test.js +0 -150
  387. package/dist/server/handlers/__tests__/e2e/inventory-sync-guards.e2e.test.js +0 -177
  388. package/dist/server/handlers/__tests__/e2e/inventory-sync.e2e.test.js +0 -228
  389. package/dist/server/handlers/__tests__/e2e/inventory-sync.part2.e2e.test.js +0 -188
  390. package/dist/server/handlers/__tests__/e2e/order-lifecycle-fulfillment.e2e.test.js +0 -295
  391. package/dist/server/handlers/__tests__/e2e/order-lifecycle.e2e.test.js +0 -277
  392. package/dist/server/handlers/__tests__/e2e/order-lifecycle.fulfillment.e2e.test.js +0 -307
  393. package/dist/server/handlers/__tests__/e2e/order-lifecycle.setup.e2e.test.js +0 -177
  394. package/dist/server/handlers/__tests__/e2e/storefront-checkout-cart.e2e.test.js +0 -255
  395. package/dist/server/handlers/__tests__/e2e/storefront-checkout-webhook.e2e.test.js +0 -231
  396. package/dist/server/handlers/__tests__/e2e/workflow-execution-failures.e2e.test.js +0 -235
  397. package/dist/server/handlers/__tests__/e2e/workflow-execution.e2e.test.js +0 -294
  398. package/dist/server/handlers/__tests__/e2e/workflow-security.e2e.test.js +0 -311
  399. package/dist/server/handlers/__tests__/e2e/workflow-security.part2.e2e.test.js +0 -267
  400. package/dist/server/handlers/__tests__/workflow-cache.test.js +0 -237
  401. package/dist/server/handlers/analytics-errors-edge.test.js +0 -173
  402. package/dist/server/handlers/analytics.test.js +0 -280
  403. package/dist/server/handlers/api-docs-examples-ext.d.ts +0 -9
  404. package/dist/server/handlers/api-docs-examples-ext.js +0 -278
  405. package/dist/server/handlers/api-docs-examples-ext.js.map +0 -1
  406. package/dist/server/handlers/api-docs-examples.d.ts +0 -8
  407. package/dist/server/handlers/api-docs-examples.js +0 -221
  408. package/dist/server/handlers/api-docs-examples.js.map +0 -1
  409. package/dist/server/handlers/api-docs-sections-ext.d.ts +0 -2
  410. package/dist/server/handlers/api-docs-sections-ext.js +0 -497
  411. package/dist/server/handlers/api-docs-sections-ext.js.map +0 -1
  412. package/dist/server/handlers/api-docs-sections.d.ts +0 -21
  413. package/dist/server/handlers/api-docs-sections.js +0 -293
  414. package/dist/server/handlers/api-docs-sections.js.map +0 -1
  415. package/dist/server/handlers/api-keys.part2.test.js +0 -157
  416. package/dist/server/handlers/api-keys.test.js +0 -161
  417. package/dist/server/handlers/billing-routes.test.js +0 -123
  418. package/dist/server/handlers/billing.test.js +0 -215
  419. package/dist/server/handlers/browser-actions-errors.test.js +0 -94
  420. package/dist/server/handlers/browser-actions.part2.test.js +0 -190
  421. package/dist/server/handlers/browser-actions.test.js +0 -190
  422. package/dist/server/handlers/browser-validation.test.js +0 -257
  423. package/dist/server/handlers/catalog.test.js +0 -297
  424. package/dist/server/handlers/comms.test.js +0 -289
  425. package/dist/server/handlers/creations-advanced-collections.test.js +0 -214
  426. package/dist/server/handlers/creations-advanced-generate.test.js +0 -142
  427. package/dist/server/handlers/creations-advanced.test.js +0 -171
  428. package/dist/server/handlers/creations-collections-preview.test.js +0 -214
  429. package/dist/server/handlers/creations-crud.test.js +0 -260
  430. package/dist/server/handlers/creations-mutations.test.js +0 -197
  431. package/dist/server/handlers/crm.test.js +0 -179
  432. package/dist/server/handlers/discovery-advertise.test.js +0 -185
  433. package/dist/server/handlers/discovery-scan.test.js +0 -233
  434. package/dist/server/handlers/embeddings-embed-search.test.js +0 -196
  435. package/dist/server/handlers/embeddings-index-delete-stats.test.js +0 -140
  436. package/dist/server/handlers/embeddings-search.test.js +0 -221
  437. package/dist/server/handlers/embeddings.test.js +0 -137
  438. package/dist/server/handlers/enrichment-breach.d.ts +0 -8
  439. package/dist/server/handlers/enrichment-breach.js +0 -266
  440. package/dist/server/handlers/enrichment-breach.js.map +0 -1
  441. package/dist/server/handlers/enrichment-data.d.ts +0 -13
  442. package/dist/server/handlers/enrichment-data.js +0 -145
  443. package/dist/server/handlers/enrichment-data.js.map +0 -1
  444. package/dist/server/handlers/enrichment-mutations.test.js +0 -240
  445. package/dist/server/handlers/enrichment-queries.test.js +0 -181
  446. package/dist/server/handlers/enrichment-validation.test.js +0 -177
  447. package/dist/server/handlers/enrichment-writes.d.ts +0 -16
  448. package/dist/server/handlers/enrichment-writes.js +0 -226
  449. package/dist/server/handlers/enrichment-writes.js.map +0 -1
  450. package/dist/server/handlers/image-gen.test.js +0 -205
  451. package/dist/server/handlers/inventory.test.js +0 -380
  452. package/dist/server/handlers/kali-background.test.js +0 -222
  453. package/dist/server/handlers/kali-errors.test.js +0 -92
  454. package/dist/server/handlers/kali-validation.test.js +0 -234
  455. package/dist/server/handlers/llm-providers-actions.test.js +0 -220
  456. package/dist/server/handlers/llm-providers-anthropic.test.js +0 -239
  457. package/dist/server/handlers/llm-providers-failover.test.js +0 -232
  458. package/dist/server/handlers/llm-providers-providers.test.js +0 -300
  459. package/dist/server/handlers/llm-providers-validation.test.js +0 -239
  460. package/dist/server/handlers/local-agent-tools.test.js +0 -224
  461. package/dist/server/handlers/local-agent.test.js +0 -198
  462. package/dist/server/handlers/local-agent.tools-status.test.js +0 -204
  463. package/dist/server/handlers/local-agent.validation-exec.test.js +0 -182
  464. package/dist/server/handlers/meta-ads-audience-rules.test.js +0 -243
  465. package/dist/server/handlers/meta-ads-audience-targeting.test.js +0 -205
  466. package/dist/server/handlers/meta-ads-audiences-targeting.test.js +0 -383
  467. package/dist/server/handlers/meta-ads-crud-ads.test.js +0 -136
  468. package/dist/server/handlers/meta-ads-crud-campaigns.test.js +0 -189
  469. package/dist/server/handlers/meta-ads-crud-create.test.js +0 -303
  470. package/dist/server/handlers/meta-ads-crud-list-update.test.js +0 -259
  471. package/dist/server/handlers/meta-ads-delete-publish-sync.test.js +0 -282
  472. package/dist/server/handlers/meta-ads-insights.test.js +0 -80
  473. package/dist/server/handlers/meta-ads-list-get.test.js +0 -237
  474. package/dist/server/handlers/meta-ads-publish-delete.test.js +0 -254
  475. package/dist/server/handlers/meta-ads-publish-helpers.js +0 -117
  476. package/dist/server/handlers/meta-ads-publish-helpers.js.map +0 -1
  477. package/dist/server/handlers/meta-ads-publish-sync.test.js +0 -205
  478. package/dist/server/handlers/meta-ads-publish.test.js +0 -254
  479. package/dist/server/handlers/meta-ads-sync-insights.test.js +0 -184
  480. package/dist/server/handlers/meta-ads-update.test.js +0 -117
  481. package/dist/server/handlers/nodes-channels.test.js +0 -413
  482. package/dist/server/handlers/nodes-events.test.js +0 -131
  483. package/dist/server/handlers/nodes-list-delete.test.js +0 -171
  484. package/dist/server/handlers/nodes-messages-delivery.test.js +0 -208
  485. package/dist/server/handlers/nodes-messages.test.js +0 -211
  486. package/dist/server/handlers/nodes-register.test.js +0 -277
  487. package/dist/server/handlers/nodes.test.js +0 -353
  488. package/dist/server/handlers/operations.test.js +0 -136
  489. package/dist/server/handlers/platform-telemetry.test.js +0 -200
  490. package/dist/server/handlers/platform-websearch.test.js +0 -160
  491. package/dist/server/handlers/storefront.test.js +0 -329
  492. package/dist/server/handlers/supply-chain.test.js +0 -347
  493. package/dist/server/handlers/transcription.test.js +0 -118
  494. package/dist/server/handlers/video-gen-veo.js +0 -114
  495. package/dist/server/handlers/video-gen-veo.js.map +0 -1
  496. package/dist/server/handlers/video-gen.test.js +0 -146
  497. package/dist/server/handlers/voice.test.js +0 -153
  498. package/dist/server/handlers/workflow-steps.test.js +0 -330
  499. package/dist/server/handlers/workflows-extras.test.js +0 -65
  500. package/dist/server/handlers/workflows.part2.test.js +0 -170
  501. package/dist/server/handlers/workflows.test.js +0 -281
  502. package/dist/server/lib/__tests__/batch-client-conversion-jsonl.test.js +0 -171
  503. package/dist/server/lib/__tests__/batch-client-polling.test.js +0 -292
  504. package/dist/server/lib/__tests__/batch-client-queue.test.js +0 -270
  505. package/dist/server/lib/__tests__/clickhouse-buffer.test.js +0 -236
  506. package/dist/server/lib/__tests__/code-worker-edge-cases.test.js +0 -118
  507. package/dist/server/lib/__tests__/code-worker-pool-execute.test.js +0 -193
  508. package/dist/server/lib/__tests__/code-worker-pool-execution.test.js +0 -165
  509. package/dist/server/lib/__tests__/code-worker-pool-init.test.js +0 -131
  510. package/dist/server/lib/__tests__/code-worker-pool.test.js +0 -194
  511. package/dist/server/lib/__tests__/code-worker-sandbox-ops.test.js +0 -123
  512. package/dist/server/lib/__tests__/code-worker-sandbox.test.js +0 -217
  513. package/dist/server/lib/__tests__/code-worker.test.js +0 -179
  514. package/dist/server/lib/__tests__/compaction-service-generate.test.js +0 -229
  515. package/dist/server/lib/__tests__/compaction-service.test.js +0 -319
  516. package/dist/server/lib/__tests__/otel.test.js +0 -146
  517. package/dist/server/lib/__tests__/prompt-sanitizer-validation.test.js +0 -165
  518. package/dist/server/lib/__tests__/prompt-sanitizer.sanitize.test.js +0 -343
  519. package/dist/server/lib/__tests__/prompt-sanitizer.test.js +0 -328
  520. package/dist/server/lib/__tests__/prompt-sanitizer.validate-tool.test.js +0 -145
  521. package/dist/server/lib/__tests__/provider-capabilities.test.js +0 -263
  522. package/dist/server/lib/__tests__/provider-failover-routing.test.js +0 -145
  523. package/dist/server/lib/__tests__/provider-failover-state.test.js +0 -131
  524. package/dist/server/lib/__tests__/rate-limiter-budgets.test.js +0 -216
  525. package/dist/server/lib/__tests__/rate-limiter.budgets-tools.test.js +0 -113
  526. package/dist/server/lib/__tests__/rate-limiter.check-request.test.js +0 -141
  527. package/dist/server/lib/__tests__/rate-limiter.stats-lifecycle.test.js +0 -135
  528. package/dist/server/lib/__tests__/rate-limiter.test.js +0 -207
  529. package/dist/server/lib/__tests__/server-agent-loop-abort-conditions.test.js +0 -544
  530. package/dist/server/lib/__tests__/server-agent-loop-abort.part2.test.js +0 -504
  531. package/dist/server/lib/__tests__/server-agent-loop-abort.test.js +0 -396
  532. package/dist/server/lib/__tests__/server-agent-loop-compaction.test.js +0 -397
  533. package/dist/server/lib/__tests__/server-agent-loop-failover.test.js +0 -356
  534. package/dist/server/lib/__tests__/server-agent-loop-features-caching.test.js +0 -519
  535. package/dist/server/lib/__tests__/server-agent-loop-features-edges.test.js +0 -512
  536. package/dist/server/lib/__tests__/server-subagent-bailout.test.js +0 -194
  537. package/dist/server/lib/__tests__/server-subagent-basics.test.js +0 -348
  538. package/dist/server/lib/__tests__/server-subagent-errors-abort.test.js +0 -319
  539. package/dist/server/lib/__tests__/server-subagent-errors-progress.test.js +0 -253
  540. package/dist/server/lib/__tests__/server-subagent-errors.part2.test.js +0 -253
  541. package/dist/server/lib/__tests__/server-subagent-errors.test.js +0 -319
  542. package/dist/server/lib/__tests__/session-checkpoint-load.test.js +0 -275
  543. package/dist/server/lib/__tests__/session-checkpoint-save.test.js +0 -159
  544. package/dist/server/lib/__tests__/ssrf-guard.test.js +0 -93
  545. package/dist/server/lib/__tests__/supabase-client.test.js +0 -111
  546. package/dist/server/lib/__tests__/template-resolver.test.js +0 -317
  547. package/dist/server/lib/__tests__/utils-timeout.test.js +0 -49
  548. package/dist/server/lib/__tests__/utils.test.js +0 -322
  549. package/dist/server/providers/__tests__/anthropic-adapter.test.js +0 -228
  550. package/dist/server/providers/__tests__/anthropic-betas-toolchoice.test.js +0 -257
  551. package/dist/server/providers/__tests__/anthropic-errors.test.js +0 -262
  552. package/dist/server/providers/__tests__/anthropic-stream-core.test.js +0 -275
  553. package/dist/server/providers/__tests__/anthropic-streaming-betas.test.js +0 -247
  554. package/dist/server/providers/__tests__/anthropic-streaming-core.test.js +0 -275
  555. package/dist/server/providers/__tests__/bedrock-config.test.js +0 -177
  556. package/dist/server/providers/__tests__/bedrock-stream-behavior-streaming.test.js +0 -272
  557. package/dist/server/providers/__tests__/bedrock-stream-behavior-toolchoice.test.js +0 -214
  558. package/dist/server/providers/__tests__/bedrock-stream-behavior.part2.test.js +0 -165
  559. package/dist/server/providers/__tests__/bedrock-stream-behavior.test.js +0 -309
  560. package/dist/server/providers/__tests__/bedrock-stream-body-credentials.test.js +0 -170
  561. package/dist/server/providers/__tests__/bedrock-stream-body-extras.test.js +0 -183
  562. package/dist/server/providers/__tests__/bedrock-stream-body-request.test.js +0 -305
  563. package/dist/server/providers/__tests__/bedrock-stream-body.part2.test.js +0 -305
  564. package/dist/server/providers/__tests__/bedrock-stream-body.test.js +0 -175
  565. package/dist/server/providers/__tests__/bedrock-stream-errors.test.js +0 -165
  566. package/dist/server/providers/__tests__/gemini-config-methods.test.js +0 -182
  567. package/dist/server/providers/__tests__/gemini-config-streaming.test.js +0 -257
  568. package/dist/server/providers/__tests__/gemini-conversion-messages.test.js +0 -247
  569. package/dist/server/providers/__tests__/gemini-conversion-schema.test.js +0 -365
  570. package/dist/server/providers/__tests__/gemini-tools-choice.test.js +0 -221
  571. package/dist/server/providers/__tests__/gemini-tools-fn.test.js +0 -252
  572. package/dist/server/providers/__tests__/openai-config.test.js +0 -194
  573. package/dist/server/providers/__tests__/openai-conversion.test.js +0 -276
  574. package/dist/server/providers/__tests__/openai-messages.test.js +0 -261
  575. package/dist/server/providers/__tests__/openai-streaming.test.js +0 -394
  576. package/dist/server/providers/__tests__/openai-tools-cache.test.js +0 -227
  577. package/dist/server/providers/__tests__/registry.test.js +0 -183
  578. package/dist/server/providers/__tests__/shared.test.js +0 -297
  579. package/dist/shared/agent-core-config.test.js +0 -132
  580. package/dist/shared/agent-core-context-thinking.test.js +0 -293
  581. package/dist/shared/agent-core-loop-calls.test.js +0 -174
  582. package/dist/shared/agent-core-loop-detector-bail.test.js +0 -201
  583. package/dist/shared/agent-core-loop-detector.test.js +0 -195
  584. package/dist/shared/agent-core-loop-errors.test.js +0 -258
  585. package/dist/shared/agent-core-pricing.test.js +0 -191
  586. package/dist/shared/agent-core-sanitize-retry.test.js +0 -129
  587. package/dist/shared/api-client-build-request.test.js +0 -228
  588. package/dist/shared/api-client-build-system-caching.test.js +0 -107
  589. package/dist/shared/api-client-build.test.js +0 -223
  590. package/dist/shared/api-client-config.d.ts +0 -21
  591. package/dist/shared/api-client-helpers.d.ts +0 -57
  592. package/dist/shared/api-client-helpers.test.js +0 -261
  593. package/dist/shared/api-client-proxy-happy.test.js +0 -255
  594. package/dist/shared/api-client-proxy-retry.test.js +0 -307
  595. package/dist/shared/api-client-proxy.d.ts +0 -26
  596. package/dist/shared/api-client-proxy.test.js +0 -255
  597. package/dist/shared/api-client-retry.test.js +0 -307
  598. package/dist/shared/api-client-system-trimming.test.js +0 -261
  599. package/dist/shared/api-client-trimming.d.ts +0 -36
  600. package/dist/shared/api-client.test.js +0 -228
  601. package/dist/shared/compaction-thinking.test.js +0 -315
  602. package/dist/shared/compaction-trimming.test.js +0 -223
  603. package/dist/shared/sse-parser-callbacks.test.js +0 -422
  604. package/dist/shared/sse-parser-collect.test.js +0 -252
  605. package/dist/shared/sse-parser-e2e.test.js +0 -558
  606. package/dist/shared/sse-parser-parse.test.js +0 -253
  607. package/dist/shared/tool-dispatch-advanced-batch-build.test.js +0 -405
  608. package/dist/shared/tool-dispatch-advanced.test.js +0 -320
  609. package/dist/shared/tool-dispatch-basic.test.js +0 -278
  610. package/dist/shared/tool-dispatch-content.d.ts +0 -14
  611. package/dist/shared/tool-dispatch-parallel.test.js +0 -378
  612. package/dist/webchat/__tests__/widget-messaging.test.js +0 -323
  613. 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":"AACA,SAASA,mBAAmB,QAAQ,6BAA6B;;AAEjE;AACA;AACA;AACA;AACA,OAAO,eAAeC,gBAAgBA,CACpCC,EAAkB,EAClBC,IAA6B,EAC7BC,OAAgB,EAC+C;EAC/D,MAAMC,GAAG,GAAGD,OAAiB;EAC7B,MAAME,SAAS,GAAGC,IAAI,CAACC,GAAG,CAAC,CAAC,EAAED,IAAI,CAACE,GAAG,CAACC,MAAM,CAACP,IAAI,CAACQ,UAAU,CAAC,IAAI,EAAE,EAAE,GAAG,CAAC,CAAC;EAC3E,MAAMC,KAAK,GAAGL,IAAI,CAACC,GAAG,CAAC,CAAC,EAAED,IAAI,CAACE,GAAG,CAACC,MAAM,CAACP,IAAI,CAACS,KAAK,CAAC,IAAI,EAAE,EAAE,IAAI,CAAC,CAAC;EACnE,MAAMC,EAAE,GAAGb,mBAAmB,CAAC,CAAC;EAEhC,IAAI,CAACa,EAAE,CAACC,SAAS,EAAE;IACjB,OAAO;MAAEC,OAAO,EAAE,KAAK;MAAEC,KAAK,EAAE;IAA+B,CAAC;EAClE;EAEA,MAAMC,WAAW,GAAGZ,GAAG,GAAG,mBAAmBa,GAAG,CAACb,GAAG,CAAC,GAAG,GAAG,EAAE;EAE7D,QAAQF,IAAI,CAACgB,MAAM;IACjB;IACA,KAAK,QAAQ;MAAE;QACb,MAAMC,IAAI,GAAG,MAAMP,EAAE,CAACQ,KAAK,CAIxB;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoBJ,WAAW;AAC/B;AACA;AACA,gBAAgBL,KAAK;AACrB,OAAO,CAAC;QACF,OAAO;UAAEG,OAAO,EAAE,IAAI;UAAEK;QAAK,CAAC;MAChC;;IAEA;IACA,KAAK,cAAc;MAAE;QACnB,MAAME,OAAO,GAAGnB,IAAI,CAACoB,QAAkB;QACvC,IAAI,CAACD,OAAO,EAAE,OAAO;UAAEP,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAAuB,CAAC;QACtE,MAAMQ,KAAK,GAAG,MAAMX,EAAE,CAACQ,KAAK,CAA0B;AAC5D;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4BH,GAAG,CAACI,OAAO,CAAC;AACxC;AACA,OAAO,CAAC;QACF,OAAO;UAAEP,OAAO,EAAE,IAAI;UAAEK,IAAI,EAAE;YAAEG,QAAQ,EAAED,OAAO;YAAEE;UAAM;QAAE,CAAC;MAC9D;;IAEA;IACA,KAAK,iBAAiB;MAAE;QACtB,MAAMJ,IAAI,GAAG,MAAMP,EAAE,CAACQ,KAAK,CAA0B;AAC3D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2Cf,SAAS;AACpD,YAAYW,WAAW;AACvB;AACA,OAAO,CAAC;QACF,OAAO;UAAEF,OAAO,EAAE,IAAI;UAAEK;QAAK,CAAC;MAChC;;IAEA;IACA,KAAK,eAAe;MAAE;QACpB,MAAMK,YAAY,GAAGtB,IAAI,CAACuB,aAAuB;QACjD,IAAI,CAACD,YAAY,EAAE,OAAO;UAAEV,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAA4B,CAAC;QAChF,MAAMW,WAAW,GAAGxB,IAAI,CAACyB,KAAK,GAAG,mBAAmBV,GAAG,CAACf,IAAI,CAACyB,KAAe,CAAC,GAAG,GAAG,EAAE;QACrF,MAAMR,IAAI,GAAG,MAAMP,EAAE,CAACQ,KAAK,CAA0B;AAC3D;AACA;AACA;AACA;AACA;AACA;AACA,gCAAgCH,GAAG,CAACO,YAAY,CAAC;AACjD,YAAYR,WAAW,IAAIU,WAAW;AACtC;AACA;AACA,gBAAgBf,KAAK;AACrB,OAAO,CAAC;QACF,OAAO;UAAEG,OAAO,EAAE,IAAI;UAAEK;QAAK,CAAC;MAChC;;IAEA;IACA,KAAK,cAAc;MAAE;QACnB,IAAI,CAACf,GAAG,EAAE,OAAO;UAAEU,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAE;QAAwC,CAAC;QACnF;QACA,MAAM;UAAEI,IAAI,EAAES,OAAO;UAAEb,KAAK,EAAEc;QAAU,CAAC,GAAG,MAAM5B,EAAE,CACjD6B,IAAI,CAAC,iBAAiB,CAAC,CACvBC,MAAM,CAAC,4CAA4C,CAAC,CACpDC,EAAE,CAAC,WAAW,EAAE,IAAI,CAAC,CACrBA,EAAE,CAAC,UAAU,EAAE5B,GAAG,CAAC;QACtB,IAAIyB,SAAS,EAAE,OAAO;UAAEf,OAAO,EAAE,KAAK;UAAEC,KAAK,EAAEc,SAAS,CAACI;QAAQ,CAAC;QAElE,MAAMC,QAAQ,GAAG,CAACN,OAAO,IAAI,EAAE,EAAEO,GAAG,CAACC,CAAC,IAAI,IAAInB,GAAG,CAACmB,CAAC,CAACC,QAAQ,CAAC,GAAG,CAAC,CAACC,IAAI,CAAC,GAAG,CAAC;QAC3E,MAAMC,SAAS,GAAGL,QAAQ,GACtB,MAAMtB,EAAE,CAACQ,KAAK,CAA+C;AACvE;AACA;AACA,iCAAiCc,QAAQ;AACzC,gCAAgCjB,GAAG,CAACb,GAAG,CAAC;AACxC;AACA,WAAW,CAAC,GACF,EAAE;QAEN,MAAMoC,QAAQ,GAAG,IAAIC,GAAG,CAACF,SAAS,CAACJ,GAAG,CAACO,CAAC,IAAI,CAACA,CAAC,CAACL,QAAQ,EAAEK,CAAC,CAACC,cAAc,CAAC,CAAC,CAAC;QAC5E,MAAMxB,IAAI,GAAG,CAACS,OAAO,IAAI,EAAE,EAAEO,GAAG,CAACC,CAAC,IAAI;UACpC,MAAMQ,KAAK,GAAGJ,QAAQ,CAACK,GAAG,CAACT,CAAC,CAACC,QAAQ,CAAC,IAAI,CAAC;UAC3C,OAAO;YACLA,QAAQ,EAAED,CAAC,CAACC,QAAQ;YACpBS,aAAa,EAAEV,CAAC,CAACW,SAAS;YAC1BC,aAAa,EAAEJ,KAAK;YACpBK,YAAY,EAAEb,CAAC,CAACc,UAAU;YAC1BC,UAAU,EAAE,IAAIC,IAAI,CAAC,CAAC,CAACC,WAAW,CAAC,CAAC;YACpCC,eAAe,EAAElB,CAAC,CAACW,SAAS,GAAG,CAAC,GAAIH,KAAK,GAAGR,CAAC,CAACW,SAAS,GAAI,GAAG,GAAG,CAAC;YAClEQ,cAAc,EAAEX,KAAK,GAAGR,CAAC,CAACW;UAC5B,CAAC;QACH,CAAC,CAAC;QACF,OAAO;UAAEjC,OAAO,EAAE,IAAI;UAAEK;QAAK,CAAC;MAChC;;IAEA;IACA,KAAK,cAAc;MAAE;QACnB,MAAMA,IAAI,GAAG,MAAMP,EAAE,CAACQ,KAAK,CAGxB;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gDAAgDf,SAAS;AACzD;AACA;AACA,gBAAgBM,KAAK;AACrB,OAAO,CAAC;QACF,OAAO;UAAEG,OAAO,EAAE,IAAI;UAAEK,IAAI,EAAE;YAAEqC,MAAM,EAAErC,IAAI;YAAEsC,YAAY,EAAEtC,IAAI,CAACuC,MAAM;YAAEhD,UAAU,EAAEL;UAAU;QAAE,CAAC;MACpG;;IAEA;IACA,KAAK,eAAe;MAAE;QACpB,MAAM,CAACsD,SAAS,EAAEC,MAAM,EAAEC,UAAU,CAAC,GAAG,MAAMC,OAAO,CAACC,GAAG,CAAC,CACxDnD,EAAE,CAACQ,KAAK,CAA0B;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6CAA6Cd,IAAI,CAACE,GAAG,CAACH,SAAS,EAAE,CAAC,CAAC;AACnE,cAAcW,WAAW;AACzB;AACA,SAAS,CAAC,EACFJ,EAAE,CAACQ,KAAK,CAA0B;AAC1C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsBJ,WAAW;AACjC;AACA;AACA;AACA,SAAS,CAAC,EACFZ,GAAG,GACCJ,gBAAgB,CAACC,EAAE,EAAE;UAAEiB,MAAM,EAAE;QAAe,CAAC,EAAEd,GAAG,CAAC,GACrD0D,OAAO,CAACE,OAAO,CAAC;UAAElD,OAAO,EAAE,IAAI;UAAEK,IAAI,EAAE;QAAK,CAAC,CAAC,CACnD,CAAC;QAEF,MAAM8C,aAAa,GAAGN,SAAS,CAACO,MAAM,CAAC,CAACC,GAAG,EAAEC,CAAC,KAAKD,GAAG,IAAKC,CAAC,CAACC,cAAc,IAAe,CAAC,CAAC,EAAE,CAAC,CAAC;QAChG,MAAMC,WAAW,GAAGX,SAAS,CAACO,MAAM,CAAC,CAACC,GAAG,EAAEC,CAAC,KAAKD,GAAG,IAAKC,CAAC,CAACG,WAAW,IAAe,CAAC,CAAC,EAAE,CAAC,CAAC;QAE3F,OAAO;UACLzD,OAAO,EAAE,IAAI;UACbK,IAAI,EAAE;YACJqD,OAAO,EAAE;cACPC,cAAc,EAAEd,SAAS,CAACD,MAAM;cAChCW,cAAc,EAAEJ,aAAa;cAC7BS,YAAY,EAAEJ,WAAW;cACzBK,UAAU,EAAEV,aAAa,GAAG,CAAC,GAAG3D,IAAI,CAACsE,KAAK,CAAEN,WAAW,GAAGL,aAAa,GAAI,KAAK,CAAC,GAAG,GAAG,GAAG;YAC5F,CAAC;YACDN,SAAS;YACTkB,aAAa,EAAEjB,MAAM;YACrBkB,YAAY,EAAEjB,UAAU,CAAC1C,IAAI;YAC7BT,UAAU,EAAEL;UACd;QACF,CAAC;MACH;IAEA;MACE,OAAO;QACLS,OAAO,EAAE,KAAK;QACdC,KAAK,EAAE,8BAA8Bb,IAAI,CAACgB,MAAM;MAClD,CAAC;EACL;AACF;;AAEA;AACA,SAASD,GAAGA,CAAC8D,CAAS,EAAU;EAC9B,OAAOA,CAAC,CAACC,OAAO,CAAC,KAAK,EAAE,MAAM,CAAC,CAACA,OAAO,CAAC,IAAI,EAAE,KAAK,CAAC;AACtD","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":"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","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,