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.
- package/dist/cli/chat/ChatApp.js +7 -11
- package/dist/cli/chat/ChatApp.js.map +1 -1
- package/dist/cli/chat/ChatInput.js +7 -3
- package/dist/cli/chat/ChatInput.js.map +1 -1
- package/dist/cli/chat/MessageList.js +5 -6
- package/dist/cli/chat/MessageList.js.map +1 -1
- package/dist/cli/chat/StatusBar.d.ts +2 -2
- package/dist/cli/chat/StatusBar.js +90 -160
- package/dist/cli/chat/StatusBar.js.map +1 -1
- package/dist/cli/chat/components/LiveArea.js +78 -115
- package/dist/cli/chat/components/LiveArea.js.map +1 -1
- package/dist/cli/chat/components/StaticMessages.js +60 -79
- package/dist/cli/chat/components/StaticMessages.js.map +1 -1
- package/dist/cli/chat/hooks/useAgentLoop.js +45 -37
- package/dist/cli/chat/hooks/useAgentLoop.js.map +1 -1
- package/dist/cli/chat/store.d.ts +12 -0
- package/dist/cli/chat/store.js +19 -0
- package/dist/cli/chat/store.js.map +1 -1
- package/dist/cli/services/agent-loop-tools.js +10 -1
- package/dist/cli/services/agent-loop-tools.js.map +1 -1
- package/dist/cli/services/cli-agent-loop.js +3 -2
- package/dist/cli/services/cli-agent-loop.js.map +1 -1
- package/dist/cli/services/config-store.js +4 -3
- package/dist/cli/services/config-store.js.map +1 -1
- package/dist/cli/services/memory-manager.js +2 -2
- package/dist/cli/services/memory-manager.js.map +1 -1
- package/dist/cli/services/permission-modes.js +14 -10
- package/dist/cli/services/permission-modes.js.map +1 -1
- package/dist/cli/services/session-client.js +2 -1
- package/dist/cli/services/session-client.js.map +1 -1
- package/dist/cli/services/session-persistence.js +14 -6
- package/dist/cli/services/session-persistence.js.map +1 -1
- package/dist/cli/shared/SpinnerSlot.js +4 -1
- package/dist/cli/shared/SpinnerSlot.js.map +1 -1
- package/dist/server/handlers/browser-lifecycle.js +10 -0
- package/dist/server/handlers/browser-lifecycle.js.map +1 -1
- package/dist/server/handlers/browser.js +16 -1
- package/dist/server/handlers/browser.js.map +1 -1
- package/dist/server/handlers/campaigns.js +11 -0
- package/dist/server/handlers/campaigns.js.map +1 -1
- package/dist/server/handlers/catalog-products.js +19 -5
- package/dist/server/handlers/catalog-products.js.map +1 -1
- package/dist/server/handlers/catalog.js +42 -8
- package/dist/server/handlers/catalog.js.map +1 -1
- package/dist/server/handlers/clickhouse.js +4 -4
- package/dist/server/handlers/clickhouse.js.map +1 -1
- package/dist/server/handlers/comms-email.js +70 -8
- package/dist/server/handlers/comms-email.js.map +1 -1
- package/dist/server/handlers/comms.js +63 -21
- package/dist/server/handlers/comms.js.map +1 -1
- package/dist/server/handlers/coupons.js +141 -77
- package/dist/server/handlers/coupons.js.map +1 -1
- package/dist/server/handlers/google-ads.js +280 -8
- package/dist/server/handlers/google-ads.js.map +1 -1
- package/dist/server/handlers/remove-bg.d.ts +33 -0
- package/dist/server/handlers/remove-bg.js +698 -44
- package/dist/server/handlers/remove-bg.js.map +1 -1
- package/dist/server/handlers/supply-chain.js +93 -1
- package/dist/server/handlers/supply-chain.js.map +1 -1
- package/dist/server/handlers/workflow-steps-types.d.ts +1 -1
- package/dist/server/handlers/workflow-steps-types.js +7 -1
- package/dist/server/handlers/workflow-steps-types.js.map +1 -1
- package/dist/server/handlers/workflow-steps.js +1 -1
- package/dist/server/handlers/workflow-steps.js.map +1 -1
- package/dist/server/index.js +122 -29
- package/dist/server/index.js.map +1 -1
- package/dist/server/lib/agent-loop-turn.js +33 -3
- package/dist/server/lib/agent-loop-turn.js.map +1 -1
- package/dist/server/lib/agent-loop-types.d.ts +6 -2
- package/dist/server/lib/agent-loop-types.js +14 -2
- package/dist/server/lib/agent-loop-types.js.map +1 -1
- package/dist/server/lib/clickhouse-client.js +4 -2
- package/dist/server/lib/clickhouse-client.js.map +1 -1
- package/dist/server/lib/code-worker.js +4 -1
- package/dist/server/lib/code-worker.js.map +1 -1
- package/dist/server/providers/anthropic.js +103 -33
- package/dist/server/providers/anthropic.js.map +1 -1
- package/dist/server/server-chat.js +2 -2
- package/dist/server/server-chat.js.map +1 -1
- package/dist/server/server-helpers.d.ts +8 -1
- package/dist/server/server-helpers.js +17 -3
- package/dist/server/server-helpers.js.map +1 -1
- package/dist/server/server-persist.js +34 -21
- package/dist/server/server-persist.js.map +1 -1
- package/dist/server/server-rate-limit.d.ts +0 -1
- package/dist/server/server-rate-limit.js +5 -5
- package/dist/server/server-rate-limit.js.map +1 -1
- package/dist/server/server-routes-approvals.js +2 -2
- package/dist/server/server-routes-approvals.js.map +1 -1
- package/dist/server/server-routes-auth.js +2 -2
- package/dist/server/server-routes-auth.js.map +1 -1
- package/dist/server/server-routes-events.js +2 -2
- package/dist/server/server-routes-events.js.map +1 -1
- package/dist/server/server-routes-public.js +4 -4
- package/dist/server/server-routes-public.js.map +1 -1
- package/dist/server/server-routes-webchat.js +3 -3
- package/dist/server/server-routes-webchat.js.map +1 -1
- package/dist/server/server-store-circuit-breaker.js +1 -1
- package/dist/server/server-store-circuit-breaker.js.map +1 -1
- package/dist/server/tool-router.js +7 -4
- package/dist/server/tool-router.js.map +1 -1
- package/dist/server/validation.js +11 -0
- package/dist/server/validation.js.map +1 -1
- package/dist/shared/api-client.js +38 -11
- package/dist/shared/api-client.js.map +1 -1
- package/package.json +12 -10
- package/vendor/ink/build/ink.js +68 -24
- package/vendor/ink/node_modules/react-devtools-core/README.md +152 -0
- package/vendor/ink/node_modules/react-devtools-core/backend.js +1 -0
- package/vendor/ink/node_modules/react-devtools-core/dist/648.chunk.js +2 -0
- package/vendor/ink/node_modules/react-devtools-core/dist/648.chunk.js.map +1 -0
- package/vendor/ink/node_modules/react-devtools-core/dist/backend.js +15691 -0
- package/vendor/ink/node_modules/react-devtools-core/dist/backend.js.map +1 -0
- package/vendor/ink/node_modules/react-devtools-core/dist/importFile.worker.worker.js +2 -0
- package/vendor/ink/node_modules/react-devtools-core/dist/importFile.worker.worker.js.map +1 -0
- package/vendor/ink/node_modules/react-devtools-core/dist/parseSourceAndMetadata.worker.worker.js +14 -0
- package/vendor/ink/node_modules/react-devtools-core/dist/parseSourceAndMetadata.worker.worker.js.map +1 -0
- package/vendor/ink/node_modules/react-devtools-core/dist/standalone.js +2 -0
- package/vendor/ink/node_modules/react-devtools-core/dist/standalone.js.map +1 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/LICENSE +21 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/README.md +495 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/browser.js +8 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/index.js +10 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/buffer-util.js +129 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/constants.js +10 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/event-target.js +184 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/extension.js +223 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/limiter.js +55 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/permessage-deflate.js +518 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/receiver.js +607 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/sender.js +409 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/stream.js +180 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/validation.js +104 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/websocket-server.js +449 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/lib/websocket.js +1197 -0
- package/vendor/ink/node_modules/react-devtools-core/node_modules/ws/package.json +56 -0
- package/vendor/ink/node_modules/react-devtools-core/package.json +38 -0
- package/vendor/ink/node_modules/react-devtools-core/standalone.js +1 -0
- package/dist/cli/__tests__/print-mode-streaming.test.js +0 -270
- package/dist/cli/__tests__/print-mode.basic-output.test.js +0 -230
- package/dist/cli/__tests__/print-mode.session-errors.test.js +0 -252
- package/dist/cli/__tests__/print-mode.test.js +0 -273
- package/dist/cli/__tests__/serve-mode-messages.test.js +0 -338
- package/dist/cli/__tests__/serve-mode.messages.part2.test.js +0 -266
- package/dist/cli/__tests__/serve-mode.messages.test.js +0 -277
- package/dist/cli/__tests__/serve-mode.startup-http.test.js +0 -279
- package/dist/cli/__tests__/serve-mode.test.js +0 -345
- package/dist/cli/chat/NodeManager.d.ts +0 -30
- package/dist/cli/chat/NodeManager.js +0 -66
- package/dist/cli/chat/NodeManager.js.map +0 -1
- package/dist/cli/chat/chat-input-menu-handler.d.ts +0 -32
- package/dist/cli/chat/hooks/slash-imsg-handlers.js +0 -148
- package/dist/cli/chat/hooks/slash-imsg-handlers.js.map +0 -1
- package/dist/cli/chat/hooks/useStreamingReducer.d.ts +0 -66
- package/dist/cli/commands/__tests__/config-cmd.test.js +0 -270
- package/dist/cli/commands/__tests__/doctor.test.js +0 -257
- package/dist/cli/commands/__tests__/imsg-node-bridge.test.js +0 -99
- package/dist/cli/commands/__tests__/imsg-utils.test.js +0 -73
- package/dist/cli/commands/__tests__/init.test.js +0 -214
- package/dist/cli/commands/__tests__/mcp.test.js +0 -287
- package/dist/cli/commands/imsg-watcher-helpers.d.ts +0 -40
- package/dist/cli/commands/imsg-watcher-helpers.js +0 -184
- package/dist/cli/commands/imsg-watcher-helpers.js.map +0 -1
- package/dist/cli/commands/imsg-watcher.d.ts +0 -11
- package/dist/cli/commands/imsg-watcher.js +0 -230
- package/dist/cli/commands/imsg-watcher.js.map +0 -1
- package/dist/cli/services/__tests__/agent-definitions.test.js +0 -153
- package/dist/cli/services/__tests__/agent-events-global.test.js +0 -39
- package/dist/cli/services/__tests__/agent-events.part2.test.js +0 -113
- package/dist/cli/services/__tests__/agent-events.test.js +0 -157
- package/dist/cli/services/__tests__/agent-loop-auth.test.js +0 -392
- package/dist/cli/services/__tests__/agent-loop-budget.test.js +0 -389
- package/dist/cli/services/__tests__/agent-loop-tools-lifecycle.test.js +0 -430
- package/dist/cli/services/__tests__/agent-loop-tools-maxturns.test.js +0 -486
- package/dist/cli/services/__tests__/agent-loop-utils-execution.test.js +0 -528
- package/dist/cli/services/__tests__/agent-loop-utils-helpers.test.js +0 -466
- package/dist/cli/services/__tests__/agent-worker-base-execute.test.js +0 -257
- package/dist/cli/services/__tests__/agent-worker-base-helpers.test.js +0 -198
- package/dist/cli/services/__tests__/agent-worker-base.test.js +0 -278
- package/dist/cli/services/__tests__/auth-service-exports.test.js +0 -41
- package/dist/cli/services/__tests__/auth-service.part2.test.js +0 -169
- package/dist/cli/services/__tests__/auth-service.test.js +0 -242
- package/dist/cli/services/__tests__/background-processes.test.js +0 -282
- package/dist/cli/services/__tests__/claude-md-loader.test.js +0 -134
- package/dist/cli/services/__tests__/config-store.test.js +0 -247
- package/dist/cli/services/__tests__/debug-log.test.js +0 -199
- package/dist/cli/services/__tests__/edge-cases-caching.test.js +0 -174
- package/dist/cli/services/__tests__/edge-cases-compaction-core.test.js +0 -226
- package/dist/cli/services/__tests__/edge-cases-compaction-openai.test.js +0 -152
- package/dist/cli/services/__tests__/edge-cases-compaction-shapes.test.js +0 -53
- package/dist/cli/services/__tests__/edge-cases-compaction-thinking.test.js +0 -226
- package/dist/cli/services/__tests__/edge-cases-compaction.test.js +0 -131
- package/dist/cli/services/__tests__/edge-cases-paths.test.js +0 -86
- package/dist/cli/services/__tests__/error-logger-messages.test.js +0 -81
- package/dist/cli/services/__tests__/error-logger-transport.test.js +0 -119
- package/dist/cli/services/__tests__/error-logger.test.js +0 -264
- package/dist/cli/services/__tests__/file-history.test.js +0 -136
- package/dist/cli/services/__tests__/git-context-cache-reset.test.js +0 -223
- package/dist/cli/services/__tests__/git-context.test.js +0 -241
- package/dist/cli/services/__tests__/interactive-tools-execute.test.js +0 -166
- package/dist/cli/services/__tests__/interactive-tools-plan.test.js +0 -197
- package/dist/cli/services/__tests__/interactive-tools.part2.test.js +0 -168
- package/dist/cli/services/__tests__/interactive-tools.test.js +0 -179
- package/dist/cli/services/__tests__/keybinding-manager.test.js +0 -205
- package/dist/cli/services/__tests__/local-tools-dispatch.test.js +0 -404
- package/dist/cli/services/__tests__/local-tools.test.js +0 -238
- package/dist/cli/services/__tests__/lsp-manager.test.js +0 -364
- package/dist/cli/services/__tests__/mcp-client-connect-disconnect.test.js +0 -310
- package/dist/cli/services/__tests__/mcp-client.test.js +0 -93
- package/dist/cli/services/__tests__/memory-manager.test.js +0 -154
- package/dist/cli/services/__tests__/model-manager-utils.test.js +0 -154
- package/dist/cli/services/__tests__/model-manager.test.js +0 -175
- package/dist/cli/services/__tests__/permission-modes.test.js +0 -222
- package/dist/cli/services/__tests__/ripgrep.test.js +0 -328
- package/dist/cli/services/__tests__/server-tools-execute.test.js +0 -317
- package/dist/cli/services/__tests__/server-tools.test.js +0 -272
- package/dist/cli/services/__tests__/session-persistence.test.js +0 -245
- package/dist/cli/services/__tests__/subagent-basic.test.js +0 -489
- package/dist/cli/services/__tests__/subagent-edge.test.js +0 -545
- package/dist/cli/services/__tests__/subagent-prompts.test.js +0 -558
- package/dist/cli/services/__tests__/subagent-worker-errors.test.js +0 -255
- package/dist/cli/services/__tests__/subagent-worker.test.js +0 -242
- package/dist/cli/services/__tests__/system-prompt.test.js +0 -210
- package/dist/cli/services/__tests__/team-lead-comms-messaging.test.js +0 -250
- package/dist/cli/services/__tests__/team-lead-comms-result.test.js +0 -232
- package/dist/cli/services/__tests__/team-lead-comms-stop.test.js +0 -344
- package/dist/cli/services/__tests__/team-lead-comms.test.js +0 -285
- package/dist/cli/services/__tests__/team-lead-create.test.js +0 -327
- package/dist/cli/services/__tests__/team-lead-run.test.js +0 -318
- package/dist/cli/services/__tests__/team-lead-stop.test.js +0 -199
- package/dist/cli/services/__tests__/team-state-comms.test.js +0 -240
- package/dist/cli/services/__tests__/team-state-core.test.js +0 -230
- package/dist/cli/services/__tests__/team-state-tasks-complete-fail-available.test.js +0 -224
- package/dist/cli/services/__tests__/team-state-tasks.test.js +0 -184
- package/dist/cli/services/__tests__/telemetry-ai-metadata.test.js +0 -116
- package/dist/cli/services/__tests__/telemetry.part2.test.js +0 -195
- package/dist/cli/services/__tests__/telemetry.test.js +0 -176
- package/dist/cli/services/agent-loop-iteration.d.ts +0 -13
- package/dist/cli/services/agent-loop-setup.d.ts +0 -32
- package/dist/cli/services/agent-worker-base-api.d.ts +0 -19
- package/dist/cli/services/agent-worker-base-helpers.d.ts +0 -27
- package/dist/cli/services/agent-worker-base-tools.d.ts +0 -16
- package/dist/cli/services/agent-worker-base-types.d.ts +0 -81
- package/dist/cli/services/background-agents.d.ts +0 -26
- package/dist/cli/services/background-processes-ops.d.ts +0 -24
- package/dist/cli/services/background-tool-defs.d.ts +0 -50
- package/dist/cli/services/config-modules-model.test.js +0 -133
- package/dist/cli/services/config-modules-permission.test.js +0 -85
- package/dist/cli/services/config-modules-permissions.test.js +0 -85
- package/dist/cli/services/config-modules-session.test.js +0 -297
- package/dist/cli/services/format-server-response-columns.test.js +0 -265
- package/dist/cli/services/format-server-response-fallback.test.js +0 -65
- package/dist/cli/services/format-server-response-primitives-basic.test.js +0 -261
- package/dist/cli/services/format-server-response-primitives-nested.test.js +0 -188
- package/dist/cli/services/format-server-response-primitives.test.js +0 -300
- package/dist/cli/services/format-server-response-realworld.test.js +0 -248
- package/dist/cli/services/format-server-response-values.test.js +0 -247
- package/dist/cli/services/hooks-runners.test.js +0 -184
- package/dist/cli/services/hooks.glob-load.test.js +0 -233
- package/dist/cli/services/hooks.run-hooks.test.js +0 -184
- package/dist/cli/services/hooks.test.js +0 -233
- package/dist/cli/services/ink-incremental.d.ts +0 -19
- package/dist/cli/services/ink-incremental.js +0 -59
- package/dist/cli/services/ink-incremental.js.map +0 -1
- package/dist/cli/services/ink-resize-fix.d.ts +0 -18
- package/dist/cli/services/ink-resize-fix.js +0 -76
- package/dist/cli/services/ink-resize-fix.js.map +0 -1
- package/dist/cli/services/ink-sync-output.d.ts +0 -12
- package/dist/cli/services/ink-sync-output.js +0 -16
- package/dist/cli/services/ink-sync-output.js.map +0 -1
- package/dist/cli/services/interactive-tool-defs.d.ts +0 -80
- package/dist/cli/services/local-tools-definitions.d.ts +0 -6
- package/dist/cli/services/local-tools-files.test.js +0 -256
- package/dist/cli/services/local-tools-read-many.d.ts +0 -6
- package/dist/cli/services/model-router.test.js +0 -245
- package/dist/cli/services/rewind-rewindTo.test.js +0 -202
- package/dist/cli/services/rewind.test.js +0 -175
- package/dist/cli/services/sandbox.test.js +0 -198
- package/dist/cli/services/subagent-execution.d.ts +0 -12
- package/dist/cli/services/team-lead-auto.d.ts +0 -11
- package/dist/cli/services/team-lead-execution.d.ts +0 -28
- package/dist/cli/services/teammate-loop.js +0 -557
- package/dist/cli/services/teammate-loop.js.map +0 -1
- package/dist/cli/services/tools/__tests__/agent-tools-tasks-teams.test.js +0 -250
- package/dist/cli/services/tools/__tests__/agent-tools-teams.test.js +0 -200
- package/dist/cli/services/tools/__tests__/agent-tools.test.js +0 -340
- package/dist/cli/services/tools/__tests__/file-ops-cache.test.js +0 -152
- package/dist/cli/services/tools/__tests__/file-ops-notebook.test.js +0 -249
- package/dist/cli/services/tools/__tests__/file-ops-read.test.js +0 -261
- package/dist/cli/services/tools/__tests__/file-ops-write.test.js +0 -292
- package/dist/cli/services/tools/__tests__/search-tools-rg.test.js +0 -92
- package/dist/cli/services/tools/__tests__/search-tools.part2.test.js +0 -174
- package/dist/cli/services/tools/__tests__/search-tools.test.js +0 -227
- package/dist/cli/services/tools/__tests__/shell-exec-allowed-core.test.js +0 -163
- package/dist/cli/services/tools/__tests__/shell-exec-allowed-extended.test.js +0 -220
- package/dist/cli/services/tools/__tests__/shell-exec-allowed.part2.test.js +0 -215
- package/dist/cli/services/tools/__tests__/shell-exec-allowed.test.js +0 -154
- package/dist/cli/services/tools/__tests__/shell-exec-blocked.test.js +0 -132
- package/dist/cli/services/tools/__tests__/shell-exec-execution.test.js +0 -245
- package/dist/cli/services/tools/__tests__/task-manager-create.test.js +0 -110
- package/dist/cli/services/tools/__tests__/task-manager-crud.test.js +0 -339
- package/dist/cli/services/tools/__tests__/task-manager-list-get.test.js +0 -343
- package/dist/cli/services/tools/__tests__/task-manager-query.test.js +0 -346
- package/dist/cli/services/tools/__tests__/task-manager-routing.test.js +0 -58
- package/dist/cli/services/tools/__tests__/task-manager-update.test.js +0 -224
- package/dist/cli/services/tools/__tests__/task-manager.test.js +0 -159
- package/dist/cli/services/tools/__tests__/web-tools-html-search.test.js +0 -227
- package/dist/cli/services/tools/__tests__/web-tools.test.js +0 -285
- package/dist/cli/services/tools/shell-exec.test.js +0 -148
- package/dist/cli/shared/SharedTick.d.ts +0 -10
- package/dist/cli/shared/__tests__/markdown.test.js +0 -188
- package/dist/local-agent/__tests__/connection-disconnect.test.js +0 -201
- package/dist/local-agent/__tests__/connection-lifecycle.test.js +0 -289
- package/dist/local-agent/__tests__/connection-msghandling.test.js +0 -311
- package/dist/local-agent/__tests__/connection-reconnect.test.js +0 -230
- package/dist/local-agent/__tests__/connection-toolexec.test.js +0 -253
- package/dist/local-agent/__tests__/discovery.test.js +0 -328
- package/dist/local-agent/__tests__/executor-background.test.js +0 -219
- package/dist/local-agent/__tests__/executor-exec.test.js +0 -221
- package/dist/local-agent/__tests__/executor-jobs-sessions.test.js +0 -220
- package/dist/local-agent/__tests__/executor-system-info.test.js +0 -133
- package/dist/local-agent/__tests__/executor-systeminfo.test.js +0 -109
- package/dist/local-agent/__tests__/executor.test.js +0 -235
- package/dist/local-agent/__tests__/index.test.js +0 -139
- package/dist/node/__tests__/cli-channels.test.js +0 -293
- package/dist/node/__tests__/cli-config-edge.test.js +0 -154
- package/dist/node/__tests__/cli-config.test.js +0 -215
- package/dist/node/__tests__/config.test.js +0 -292
- package/dist/node/__tests__/runtime-heartbeat.test.js +0 -153
- package/dist/node/__tests__/runtime-lifecycle-init.test.js +0 -263
- package/dist/node/__tests__/runtime-lifecycle-stats.test.js +0 -180
- package/dist/node/__tests__/runtime-lifecycle.test.js +0 -305
- package/dist/node/__tests__/runtime-relay.test.js +0 -341
- package/dist/node/adapters/__tests__/base.test.js +0 -286
- package/dist/node/adapters/__tests__/discord.test.js +0 -284
- package/dist/node/adapters/__tests__/email-send.test.js +0 -295
- package/dist/node/adapters/__tests__/email.inbound-send.test.js +0 -217
- package/dist/node/adapters/__tests__/email.lifecycle.test.js +0 -211
- package/dist/node/adapters/__tests__/email.test.js +0 -290
- package/dist/node/adapters/__tests__/email.webhook-send.test.js +0 -251
- package/dist/node/adapters/__tests__/imessage-filter.test.js +0 -183
- package/dist/node/adapters/__tests__/imessage-lifecycle.test.js +0 -215
- package/dist/node/adapters/__tests__/imessage-send-restart.test.js +0 -227
- package/dist/node/adapters/__tests__/slack.part2.test.js +0 -135
- package/dist/node/adapters/__tests__/slack.test.js +0 -241
- package/dist/node/adapters/__tests__/sms-extras.test.js +0 -108
- package/dist/node/adapters/__tests__/sms-lifecycle.test.js +0 -203
- package/dist/node/adapters/__tests__/sms-messaging.test.js +0 -266
- package/dist/node/adapters/__tests__/sms.part2.test.js +0 -174
- package/dist/node/adapters/__tests__/sms.test.js +0 -253
- package/dist/node/adapters/__tests__/telegram-polling.test.js +0 -256
- package/dist/node/adapters/__tests__/telegram-send.test.js +0 -166
- package/dist/node/adapters/__tests__/webchat-inbound.test.js +0 -188
- package/dist/node/adapters/__tests__/webchat-outbound.test.js +0 -178
- package/dist/node/adapters/__tests__/whatsapp-inbound.test.js +0 -200
- package/dist/node/adapters/__tests__/whatsapp-send.test.js +0 -212
- package/dist/node/adapters/__tests__/whatsapp.test.js +0 -280
- package/dist/server/__tests__/gateway-fast-fail.test.js +0 -160
- package/dist/server/__tests__/local-agent-gateway.test.js +0 -186
- package/dist/server/__tests__/proxy-handlers-delegation.test.js +0 -240
- package/dist/server/__tests__/proxy-handlers-validation.test.js +0 -211
- package/dist/server/__tests__/proxy-handlers.part2.test.js +0 -240
- package/dist/server/__tests__/proxy-handlers.test.js +0 -213
- package/dist/server/__tests__/strip-base64-e2e.test.js +0 -303
- package/dist/server/__tests__/strip-base64.test.js +0 -256
- package/dist/server/__tests__/tool-router-agent-tools.test.js +0 -324
- package/dist/server/__tests__/tool-router-execute-core.test.js +0 -357
- package/dist/server/__tests__/tool-router-execute-permissions.test.js +0 -332
- package/dist/server/__tests__/tool-router-execute.test.js +0 -348
- package/dist/server/__tests__/tool-router-load.test.js +0 -432
- package/dist/server/__tests__/tool-router-permissions.test.js +0 -359
- package/dist/server/__tests__/tool-router-registry-cache.test.js +0 -383
- package/dist/server/__tests__/tool-router-registry-handlers.test.js +0 -272
- package/dist/server/__tests__/tool-router-registry.test.js +0 -331
- package/dist/server/__tests__/validation-inventory.test.js +0 -250
- package/dist/server/__tests__/validation-misc.test.js +0 -243
- package/dist/server/__tests__/validation-supply-chain.test.js +0 -188
- package/dist/server/__tests__/worker.test.js +0 -265
- package/dist/server/handlers/__tests__/conversation-lock.test.js +0 -117
- package/dist/server/handlers/__tests__/e2e/auth-cross-platform-login.e2e.test.js +0 -268
- package/dist/server/handlers/__tests__/e2e/auth-cross-platform-tokens.e2e.test.js +0 -264
- package/dist/server/handlers/__tests__/e2e/email-pipeline-send.e2e.test.js +0 -214
- package/dist/server/handlers/__tests__/e2e/email-pipeline-threads.e2e.test.js +0 -168
- package/dist/server/handlers/__tests__/e2e/error-logging-pipeline-dedup.e2e.test.js +0 -229
- package/dist/server/handlers/__tests__/e2e/error-logging-pipeline.e2e.test.js +0 -239
- package/dist/server/handlers/__tests__/e2e/error-logging-rate-limit.e2e.test.js +0 -150
- package/dist/server/handlers/__tests__/e2e/inventory-sync-guards.e2e.test.js +0 -177
- package/dist/server/handlers/__tests__/e2e/inventory-sync.e2e.test.js +0 -228
- package/dist/server/handlers/__tests__/e2e/inventory-sync.part2.e2e.test.js +0 -188
- package/dist/server/handlers/__tests__/e2e/order-lifecycle-fulfillment.e2e.test.js +0 -295
- package/dist/server/handlers/__tests__/e2e/order-lifecycle.e2e.test.js +0 -277
- package/dist/server/handlers/__tests__/e2e/order-lifecycle.fulfillment.e2e.test.js +0 -307
- package/dist/server/handlers/__tests__/e2e/order-lifecycle.setup.e2e.test.js +0 -177
- package/dist/server/handlers/__tests__/e2e/storefront-checkout-cart.e2e.test.js +0 -255
- package/dist/server/handlers/__tests__/e2e/storefront-checkout-webhook.e2e.test.js +0 -231
- package/dist/server/handlers/__tests__/e2e/workflow-execution-failures.e2e.test.js +0 -235
- package/dist/server/handlers/__tests__/e2e/workflow-execution.e2e.test.js +0 -294
- package/dist/server/handlers/__tests__/e2e/workflow-security.e2e.test.js +0 -311
- package/dist/server/handlers/__tests__/e2e/workflow-security.part2.e2e.test.js +0 -267
- package/dist/server/handlers/__tests__/workflow-cache.test.js +0 -237
- package/dist/server/handlers/analytics-errors-edge.test.js +0 -173
- package/dist/server/handlers/analytics.test.js +0 -280
- package/dist/server/handlers/api-docs-examples-ext.d.ts +0 -9
- package/dist/server/handlers/api-docs-examples-ext.js +0 -278
- package/dist/server/handlers/api-docs-examples-ext.js.map +0 -1
- package/dist/server/handlers/api-docs-examples.d.ts +0 -8
- package/dist/server/handlers/api-docs-examples.js +0 -221
- package/dist/server/handlers/api-docs-examples.js.map +0 -1
- package/dist/server/handlers/api-docs-sections-ext.d.ts +0 -2
- package/dist/server/handlers/api-docs-sections-ext.js +0 -497
- package/dist/server/handlers/api-docs-sections-ext.js.map +0 -1
- package/dist/server/handlers/api-docs-sections.d.ts +0 -21
- package/dist/server/handlers/api-docs-sections.js +0 -293
- package/dist/server/handlers/api-docs-sections.js.map +0 -1
- package/dist/server/handlers/api-keys.part2.test.js +0 -157
- package/dist/server/handlers/api-keys.test.js +0 -161
- package/dist/server/handlers/billing-routes.test.js +0 -123
- package/dist/server/handlers/billing.test.js +0 -215
- package/dist/server/handlers/browser-actions-errors.test.js +0 -94
- package/dist/server/handlers/browser-actions.part2.test.js +0 -190
- package/dist/server/handlers/browser-actions.test.js +0 -190
- package/dist/server/handlers/browser-validation.test.js +0 -257
- package/dist/server/handlers/catalog.test.js +0 -297
- package/dist/server/handlers/comms.test.js +0 -289
- package/dist/server/handlers/creations-advanced-collections.test.js +0 -214
- package/dist/server/handlers/creations-advanced-generate.test.js +0 -142
- package/dist/server/handlers/creations-advanced.test.js +0 -171
- package/dist/server/handlers/creations-collections-preview.test.js +0 -214
- package/dist/server/handlers/creations-crud.test.js +0 -260
- package/dist/server/handlers/creations-mutations.test.js +0 -197
- package/dist/server/handlers/crm.test.js +0 -179
- package/dist/server/handlers/discovery-advertise.test.js +0 -185
- package/dist/server/handlers/discovery-scan.test.js +0 -233
- package/dist/server/handlers/embeddings-embed-search.test.js +0 -196
- package/dist/server/handlers/embeddings-index-delete-stats.test.js +0 -140
- package/dist/server/handlers/embeddings-search.test.js +0 -221
- package/dist/server/handlers/embeddings.test.js +0 -137
- package/dist/server/handlers/enrichment-breach.d.ts +0 -8
- package/dist/server/handlers/enrichment-breach.js +0 -266
- package/dist/server/handlers/enrichment-breach.js.map +0 -1
- package/dist/server/handlers/enrichment-data.d.ts +0 -13
- package/dist/server/handlers/enrichment-data.js +0 -145
- package/dist/server/handlers/enrichment-data.js.map +0 -1
- package/dist/server/handlers/enrichment-mutations.test.js +0 -240
- package/dist/server/handlers/enrichment-queries.test.js +0 -181
- package/dist/server/handlers/enrichment-validation.test.js +0 -177
- package/dist/server/handlers/enrichment-writes.d.ts +0 -16
- package/dist/server/handlers/enrichment-writes.js +0 -226
- package/dist/server/handlers/enrichment-writes.js.map +0 -1
- package/dist/server/handlers/image-gen.test.js +0 -205
- package/dist/server/handlers/inventory.test.js +0 -380
- package/dist/server/handlers/kali-background.test.js +0 -222
- package/dist/server/handlers/kali-errors.test.js +0 -92
- package/dist/server/handlers/kali-validation.test.js +0 -234
- package/dist/server/handlers/llm-providers-actions.test.js +0 -220
- package/dist/server/handlers/llm-providers-anthropic.test.js +0 -239
- package/dist/server/handlers/llm-providers-failover.test.js +0 -232
- package/dist/server/handlers/llm-providers-providers.test.js +0 -300
- package/dist/server/handlers/llm-providers-validation.test.js +0 -239
- package/dist/server/handlers/local-agent-tools.test.js +0 -224
- package/dist/server/handlers/local-agent.test.js +0 -198
- package/dist/server/handlers/local-agent.tools-status.test.js +0 -204
- package/dist/server/handlers/local-agent.validation-exec.test.js +0 -182
- package/dist/server/handlers/meta-ads-audience-rules.test.js +0 -243
- package/dist/server/handlers/meta-ads-audience-targeting.test.js +0 -205
- package/dist/server/handlers/meta-ads-audiences-targeting.test.js +0 -383
- package/dist/server/handlers/meta-ads-crud-ads.test.js +0 -136
- package/dist/server/handlers/meta-ads-crud-campaigns.test.js +0 -189
- package/dist/server/handlers/meta-ads-crud-create.test.js +0 -303
- package/dist/server/handlers/meta-ads-crud-list-update.test.js +0 -259
- package/dist/server/handlers/meta-ads-delete-publish-sync.test.js +0 -282
- package/dist/server/handlers/meta-ads-insights.test.js +0 -80
- package/dist/server/handlers/meta-ads-list-get.test.js +0 -237
- package/dist/server/handlers/meta-ads-publish-delete.test.js +0 -254
- package/dist/server/handlers/meta-ads-publish-helpers.js +0 -117
- package/dist/server/handlers/meta-ads-publish-helpers.js.map +0 -1
- package/dist/server/handlers/meta-ads-publish-sync.test.js +0 -205
- package/dist/server/handlers/meta-ads-publish.test.js +0 -254
- package/dist/server/handlers/meta-ads-sync-insights.test.js +0 -184
- package/dist/server/handlers/meta-ads-update.test.js +0 -117
- package/dist/server/handlers/nodes-channels.test.js +0 -413
- package/dist/server/handlers/nodes-events.test.js +0 -131
- package/dist/server/handlers/nodes-list-delete.test.js +0 -171
- package/dist/server/handlers/nodes-messages-delivery.test.js +0 -208
- package/dist/server/handlers/nodes-messages.test.js +0 -211
- package/dist/server/handlers/nodes-register.test.js +0 -277
- package/dist/server/handlers/nodes.test.js +0 -353
- package/dist/server/handlers/operations.test.js +0 -136
- package/dist/server/handlers/platform-telemetry.test.js +0 -200
- package/dist/server/handlers/platform-websearch.test.js +0 -160
- package/dist/server/handlers/storefront.test.js +0 -329
- package/dist/server/handlers/supply-chain.test.js +0 -347
- package/dist/server/handlers/transcription.test.js +0 -118
- package/dist/server/handlers/video-gen-veo.js +0 -114
- package/dist/server/handlers/video-gen-veo.js.map +0 -1
- package/dist/server/handlers/video-gen.test.js +0 -146
- package/dist/server/handlers/voice.test.js +0 -153
- package/dist/server/handlers/workflow-steps.test.js +0 -330
- package/dist/server/handlers/workflows-extras.test.js +0 -65
- package/dist/server/handlers/workflows.part2.test.js +0 -170
- package/dist/server/handlers/workflows.test.js +0 -281
- package/dist/server/lib/__tests__/batch-client-conversion-jsonl.test.js +0 -171
- package/dist/server/lib/__tests__/batch-client-polling.test.js +0 -292
- package/dist/server/lib/__tests__/batch-client-queue.test.js +0 -270
- package/dist/server/lib/__tests__/clickhouse-buffer.test.js +0 -236
- package/dist/server/lib/__tests__/code-worker-edge-cases.test.js +0 -118
- package/dist/server/lib/__tests__/code-worker-pool-execute.test.js +0 -193
- package/dist/server/lib/__tests__/code-worker-pool-execution.test.js +0 -165
- package/dist/server/lib/__tests__/code-worker-pool-init.test.js +0 -131
- package/dist/server/lib/__tests__/code-worker-pool.test.js +0 -194
- package/dist/server/lib/__tests__/code-worker-sandbox-ops.test.js +0 -123
- package/dist/server/lib/__tests__/code-worker-sandbox.test.js +0 -217
- package/dist/server/lib/__tests__/code-worker.test.js +0 -179
- package/dist/server/lib/__tests__/compaction-service-generate.test.js +0 -229
- package/dist/server/lib/__tests__/compaction-service.test.js +0 -319
- package/dist/server/lib/__tests__/otel.test.js +0 -146
- package/dist/server/lib/__tests__/prompt-sanitizer-validation.test.js +0 -165
- package/dist/server/lib/__tests__/prompt-sanitizer.sanitize.test.js +0 -343
- package/dist/server/lib/__tests__/prompt-sanitizer.test.js +0 -328
- package/dist/server/lib/__tests__/prompt-sanitizer.validate-tool.test.js +0 -145
- package/dist/server/lib/__tests__/provider-capabilities.test.js +0 -263
- package/dist/server/lib/__tests__/provider-failover-routing.test.js +0 -145
- package/dist/server/lib/__tests__/provider-failover-state.test.js +0 -131
- package/dist/server/lib/__tests__/rate-limiter-budgets.test.js +0 -216
- package/dist/server/lib/__tests__/rate-limiter.budgets-tools.test.js +0 -113
- package/dist/server/lib/__tests__/rate-limiter.check-request.test.js +0 -141
- package/dist/server/lib/__tests__/rate-limiter.stats-lifecycle.test.js +0 -135
- package/dist/server/lib/__tests__/rate-limiter.test.js +0 -207
- package/dist/server/lib/__tests__/server-agent-loop-abort-conditions.test.js +0 -544
- package/dist/server/lib/__tests__/server-agent-loop-abort.part2.test.js +0 -504
- package/dist/server/lib/__tests__/server-agent-loop-abort.test.js +0 -396
- package/dist/server/lib/__tests__/server-agent-loop-compaction.test.js +0 -397
- package/dist/server/lib/__tests__/server-agent-loop-failover.test.js +0 -356
- package/dist/server/lib/__tests__/server-agent-loop-features-caching.test.js +0 -519
- package/dist/server/lib/__tests__/server-agent-loop-features-edges.test.js +0 -512
- package/dist/server/lib/__tests__/server-subagent-bailout.test.js +0 -194
- package/dist/server/lib/__tests__/server-subagent-basics.test.js +0 -348
- package/dist/server/lib/__tests__/server-subagent-errors-abort.test.js +0 -319
- package/dist/server/lib/__tests__/server-subagent-errors-progress.test.js +0 -253
- package/dist/server/lib/__tests__/server-subagent-errors.part2.test.js +0 -253
- package/dist/server/lib/__tests__/server-subagent-errors.test.js +0 -319
- package/dist/server/lib/__tests__/session-checkpoint-load.test.js +0 -275
- package/dist/server/lib/__tests__/session-checkpoint-save.test.js +0 -159
- package/dist/server/lib/__tests__/ssrf-guard.test.js +0 -93
- package/dist/server/lib/__tests__/supabase-client.test.js +0 -111
- package/dist/server/lib/__tests__/template-resolver.test.js +0 -317
- package/dist/server/lib/__tests__/utils-timeout.test.js +0 -49
- package/dist/server/lib/__tests__/utils.test.js +0 -322
- package/dist/server/providers/__tests__/anthropic-adapter.test.js +0 -228
- package/dist/server/providers/__tests__/anthropic-betas-toolchoice.test.js +0 -257
- package/dist/server/providers/__tests__/anthropic-errors.test.js +0 -262
- package/dist/server/providers/__tests__/anthropic-stream-core.test.js +0 -275
- package/dist/server/providers/__tests__/anthropic-streaming-betas.test.js +0 -247
- package/dist/server/providers/__tests__/anthropic-streaming-core.test.js +0 -275
- package/dist/server/providers/__tests__/bedrock-config.test.js +0 -177
- package/dist/server/providers/__tests__/bedrock-stream-behavior-streaming.test.js +0 -272
- package/dist/server/providers/__tests__/bedrock-stream-behavior-toolchoice.test.js +0 -214
- package/dist/server/providers/__tests__/bedrock-stream-behavior.part2.test.js +0 -165
- package/dist/server/providers/__tests__/bedrock-stream-behavior.test.js +0 -309
- package/dist/server/providers/__tests__/bedrock-stream-body-credentials.test.js +0 -170
- package/dist/server/providers/__tests__/bedrock-stream-body-extras.test.js +0 -183
- package/dist/server/providers/__tests__/bedrock-stream-body-request.test.js +0 -305
- package/dist/server/providers/__tests__/bedrock-stream-body.part2.test.js +0 -305
- package/dist/server/providers/__tests__/bedrock-stream-body.test.js +0 -175
- package/dist/server/providers/__tests__/bedrock-stream-errors.test.js +0 -165
- package/dist/server/providers/__tests__/gemini-config-methods.test.js +0 -182
- package/dist/server/providers/__tests__/gemini-config-streaming.test.js +0 -257
- package/dist/server/providers/__tests__/gemini-conversion-messages.test.js +0 -247
- package/dist/server/providers/__tests__/gemini-conversion-schema.test.js +0 -365
- package/dist/server/providers/__tests__/gemini-tools-choice.test.js +0 -221
- package/dist/server/providers/__tests__/gemini-tools-fn.test.js +0 -252
- package/dist/server/providers/__tests__/openai-config.test.js +0 -194
- package/dist/server/providers/__tests__/openai-conversion.test.js +0 -276
- package/dist/server/providers/__tests__/openai-messages.test.js +0 -261
- package/dist/server/providers/__tests__/openai-streaming.test.js +0 -394
- package/dist/server/providers/__tests__/openai-tools-cache.test.js +0 -227
- package/dist/server/providers/__tests__/registry.test.js +0 -183
- package/dist/server/providers/__tests__/shared.test.js +0 -297
- package/dist/shared/agent-core-config.test.js +0 -132
- package/dist/shared/agent-core-context-thinking.test.js +0 -293
- package/dist/shared/agent-core-loop-calls.test.js +0 -174
- package/dist/shared/agent-core-loop-detector-bail.test.js +0 -201
- package/dist/shared/agent-core-loop-detector.test.js +0 -195
- package/dist/shared/agent-core-loop-errors.test.js +0 -258
- package/dist/shared/agent-core-pricing.test.js +0 -191
- package/dist/shared/agent-core-sanitize-retry.test.js +0 -129
- package/dist/shared/api-client-build-request.test.js +0 -228
- package/dist/shared/api-client-build-system-caching.test.js +0 -107
- package/dist/shared/api-client-build.test.js +0 -223
- package/dist/shared/api-client-config.d.ts +0 -21
- package/dist/shared/api-client-helpers.d.ts +0 -57
- package/dist/shared/api-client-helpers.test.js +0 -261
- package/dist/shared/api-client-proxy-happy.test.js +0 -255
- package/dist/shared/api-client-proxy-retry.test.js +0 -307
- package/dist/shared/api-client-proxy.d.ts +0 -26
- package/dist/shared/api-client-proxy.test.js +0 -255
- package/dist/shared/api-client-retry.test.js +0 -307
- package/dist/shared/api-client-system-trimming.test.js +0 -261
- package/dist/shared/api-client-trimming.d.ts +0 -36
- package/dist/shared/api-client.test.js +0 -228
- package/dist/shared/compaction-thinking.test.js +0 -315
- package/dist/shared/compaction-trimming.test.js +0 -223
- package/dist/shared/sse-parser-callbacks.test.js +0 -422
- package/dist/shared/sse-parser-collect.test.js +0 -252
- package/dist/shared/sse-parser-e2e.test.js +0 -558
- package/dist/shared/sse-parser-parse.test.js +0 -253
- package/dist/shared/tool-dispatch-advanced-batch-build.test.js +0 -405
- package/dist/shared/tool-dispatch-advanced.test.js +0 -320
- package/dist/shared/tool-dispatch-basic.test.js +0 -278
- package/dist/shared/tool-dispatch-content.d.ts +0 -14
- package/dist/shared/tool-dispatch-parallel.test.js +0 -378
- package/dist/webchat/__tests__/widget-messaging.test.js +0 -323
- package/dist/webchat/__tests__/widget.test.js +0 -273
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"clickhouse.js","names":["getClickHouseClient","handleClickHouse","sb","args","storeId","sid","hoursBack","hours_back","limit","Math","min","ch","isEnabled","success","error","storeFilter","esc","action","data","query","traceId","trace_id","spans","functionName","function_name","levelFilter","level","budgets","budgetErr","from","select","eq","message","agentIds","map","b","agent_id","join","spendRows","spendMap","Map","r","total_cost_usd","spend","get","budget_amount","limit_usd","current_spend","period_start","created_at","period_end","Date","toISOString","utilization_pct","is_over_budget","groups","total_groups","length","functions","traces","budgetData","Promise","all","resolve","totalRequests","reduce","sum","f","total_requests","totalErrors","error_count","summary","function_count","total_errors","error_rate","round","recent_traces","cost_budgets","s","replace"],"sources":["../../../src/server/handlers/clickhouse.ts"],"sourcesContent":["import type { SupabaseClient } from \"@supabase/supabase-js\";\nimport { getClickHouseClient } from \"../lib/clickhouse-client.js\";\n\n/**\n * ClickHouse observability handler — platform-level metrics via direct HTTP queries.\n * Replaces FDW RPCs with direct ClickHouse queries for lower latency.\n */\nexport async function handleClickHouse(\n sb: SupabaseClient,\n args: Record<string, unknown>,\n storeId?: string,\n): Promise<{ success: boolean; data?: unknown; error?: string }> {\n const sid = storeId as string;\n const hoursBack = (args.hours_back as number) || 24;\n const limit = Math.min((args.limit as number) || 50, 200);\n const ch = getClickHouseClient();\n\n if (!ch.isEnabled) {\n return { success: false, error: \"ClickHouse is not configured\" };\n }\n\n const storeFilter = sid ? `AND store_id = '${esc(sid)}'` : \"\";\n\n switch (args.action) {\n // ---- traces: List recent traces ----\n case \"traces\": {\n const data = await ch.query<{\n trace_id: string; root_service: string; root_operation: string;\n duration: number; span_count: number; error_count: number;\n started_at: string; status: string;\n }>(`\n SELECT\n trace_id,\n arrayElement(groupArrayIf(service_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_service,\n arrayElement(groupArrayIf(operation_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_operation,\n toInt32(dateDiff('millisecond', min(started_at), max(ended_at))) AS duration,\n count() AS span_count,\n countIf(status_code = 'ERROR') AS error_count,\n min(started_at) AS started_at,\n if(countIf(status_code = 'ERROR') > 0, 'error', 'ok') AS status\n FROM ai_spans\n WHERE 1=1 ${storeFilter}\n GROUP BY trace_id\n ORDER BY min(started_at) DESC\n LIMIT ${limit}\n `);\n return { success: true, data };\n }\n\n // ---- trace_detail: Full trace waterfall ----\n case \"trace_detail\": {\n const traceId = args.trace_id as string;\n if (!traceId) return { success: false, error: \"trace_id is required\" };\n const spans = await ch.query<Record<string, unknown>>(`\n SELECT\n span_id, parent_span_id, trace_id, operation_name,\n service_name, duration_ms AS duration,\n started_at, ended_at,\n if(status_code = 'ERROR', 'error', 'ok') AS status,\n attributes, events\n FROM ai_spans\n WHERE trace_id = '${esc(traceId)}'\n ORDER BY started_at ASC\n `);\n return { success: true, data: { trace_id: traceId, spans } };\n }\n\n // ---- function_health: Service health metrics ----\n case \"function_health\": {\n const data = await ch.query<Record<string, unknown>>(`\n SELECT\n service_name AS function_name,\n sum(total_requests) AS total_reqs,\n sum(error_count) AS total_errors,\n sum(client_error_count) AS client_errors,\n if(sum(total_requests) > 0, sum(error_count) / sum(total_requests), 0) AS error_rate,\n if(sum(latency_count) > 0, sum(latency_sum_ms) / sum(latency_count), 0) AS avg_latency_ms,\n max(latency_max_ms) AS p95_latency_ms,\n min(latency_min_ms) AS min_latency_ms,\n max(latency_max_ms) AS max_latency_ms,\n max(bucket) AS last_request_at\n FROM function_health\n WHERE bucket >= now() - INTERVAL ${hoursBack} HOUR\n ${storeFilter}\n GROUP BY service_name\n `);\n return { success: true, data };\n }\n\n // ---- function_logs: Request logs ----\n case \"function_logs\": {\n const functionName = args.function_name as string;\n if (!functionName) return { success: false, error: \"function_name is required\" };\n const levelFilter = args.level ? `AND severity = '${esc(args.level as string)}'` : \"\";\n const data = await ch.query<Record<string, unknown>>(`\n SELECT\n request_id, service_name AS function_name, http_status AS status,\n duration_ms, http_method AS method, http_path AS path,\n store_id, status_code = 'ERROR' AS has_error,\n started_at AS first_seen, events AS lines\n FROM ai_spans\n WHERE service_name = '${esc(functionName)}'\n ${storeFilter} ${levelFilter}\n AND span_kind = 'SERVER'\n ORDER BY started_at DESC\n LIMIT ${limit}\n `);\n return { success: true, data };\n }\n\n // ---- cost_budgets: Budget utilization ----\n case \"cost_budgets\": {\n if (!sid) return { success: false, error: \"store_id is required for cost_budgets\" };\n // Cost budgets require Postgres for ai_cost_budgets table + ClickHouse for spend\n const { data: budgets, error: budgetErr } = await sb\n .from(\"ai_cost_budgets\")\n .select(\"agent_id, limit_usd, is_active, created_at\")\n .eq(\"is_active\", true)\n .eq(\"store_id\", sid);\n if (budgetErr) return { success: false, error: budgetErr.message };\n\n const agentIds = (budgets || []).map(b => `'${esc(b.agent_id)}'`).join(\",\");\n const spendRows = agentIds\n ? await ch.query<{ agent_id: string; total_cost_usd: number }>(`\n SELECT agent_id, sum(total_cost_usd) AS total_cost_usd\n FROM token_usage_hourly\n WHERE agent_id IN (${agentIds})\n AND store_id = '${esc(sid)}'\n GROUP BY agent_id\n `)\n : [];\n\n const spendMap = new Map(spendRows.map(r => [r.agent_id, r.total_cost_usd]));\n const data = (budgets || []).map(b => {\n const spend = spendMap.get(b.agent_id) || 0;\n return {\n agent_id: b.agent_id,\n budget_amount: b.limit_usd,\n current_spend: spend,\n period_start: b.created_at,\n period_end: new Date().toISOString(),\n utilization_pct: b.limit_usd > 0 ? (spend / b.limit_usd) * 100 : 0,\n is_over_budget: spend > b.limit_usd,\n };\n });\n return { success: true, data };\n }\n\n // ---- error_groups: Error fingerprints grouped ----\n case \"error_groups\": {\n const data = await ch.query<{\n fingerprint: string; error_type: string; message: string;\n severity: string; service: string; count: number; last_seen: string;\n }>(`\n SELECT\n fingerprint,\n any(error_type) AS error_type,\n substring(any(error_message), 1, 200) AS message,\n any(severity) AS severity,\n any(service_name) AS service,\n count() AS count,\n max(occurred_at) AS last_seen\n FROM error_events\n WHERE occurred_at >= now() - INTERVAL ${hoursBack} HOUR\n GROUP BY fingerprint\n ORDER BY count DESC\n LIMIT ${limit}\n `);\n return { success: true, data: { groups: data, total_groups: data.length, hours_back: hoursBack } };\n }\n\n // ---- system_status: Combined overview ----\n case \"system_status\": {\n const [functions, traces, budgetData] = await Promise.all([\n ch.query<Record<string, unknown>>(`\n SELECT\n service_name AS function_name,\n sum(total_requests) AS total_reqs,\n sum(error_count) AS total_errors,\n if(sum(total_requests) > 0, sum(error_count) / sum(total_requests), 0) AS error_rate,\n max(bucket) AS last_request_at\n FROM function_health\n WHERE bucket >= now() - INTERVAL ${Math.min(hoursBack, 1)} HOUR\n ${storeFilter}\n GROUP BY service_name\n `),\n ch.query<Record<string, unknown>>(`\n SELECT\n trace_id,\n arrayElement(groupArrayIf(service_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_service,\n arrayElement(groupArrayIf(operation_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_operation,\n toInt32(dateDiff('millisecond', min(started_at), max(ended_at))) AS duration,\n count() AS span_count,\n countIf(status_code = 'ERROR') AS error_count,\n min(started_at) AS started_at,\n if(countIf(status_code = 'ERROR') > 0, 'error', 'ok') AS status\n FROM ai_spans\n WHERE 1=1 ${storeFilter}\n GROUP BY trace_id\n ORDER BY min(started_at) DESC\n LIMIT 10\n `),\n sid\n ? handleClickHouse(sb, { action: \"cost_budgets\" }, sid)\n : Promise.resolve({ success: true, data: null }),\n ]);\n\n const totalRequests = functions.reduce((sum, f) => sum + ((f.total_requests as number) || 0), 0);\n const totalErrors = functions.reduce((sum, f) => sum + ((f.error_count as number) || 0), 0);\n\n return {\n success: true,\n data: {\n summary: {\n function_count: functions.length,\n total_requests: totalRequests,\n total_errors: totalErrors,\n error_rate: totalRequests > 0 ? Math.round((totalErrors / totalRequests) * 10000) / 100 : 0,\n },\n functions,\n recent_traces: traces,\n cost_budgets: budgetData.data,\n hours_back: hoursBack,\n },\n };\n }\n\n default:\n return {\n success: false,\n error: `Unknown clickhouse action: ${args.action}. Available: traces, trace_detail, function_health, function_logs, cost_budgets, error_groups, system_status`,\n };\n }\n}\n\n/** Escape single quotes for ClickHouse SQL */\nfunction esc(s: string): string {\n return s.replace(/'/g, \"\\\\'\");\n}\n"],"mappings":"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","ignoreList":[]}
|
|
1
|
+
{"version":3,"file":"clickhouse.js","names":["getClickHouseClient","handleClickHouse","sb","args","storeId","sid","hoursBack","Math","max","min","Number","hours_back","limit","ch","isEnabled","success","error","storeFilter","esc","action","data","query","traceId","trace_id","spans","functionName","function_name","levelFilter","level","budgets","budgetErr","from","select","eq","message","agentIds","map","b","agent_id","join","spendRows","spendMap","Map","r","total_cost_usd","spend","get","budget_amount","limit_usd","current_spend","period_start","created_at","period_end","Date","toISOString","utilization_pct","is_over_budget","groups","total_groups","length","functions","traces","budgetData","Promise","all","resolve","totalRequests","reduce","sum","f","total_requests","totalErrors","error_count","summary","function_count","total_errors","error_rate","round","recent_traces","cost_budgets","s","replace"],"sources":["../../../src/server/handlers/clickhouse.ts"],"sourcesContent":["import type { SupabaseClient } from \"@supabase/supabase-js\";\nimport { getClickHouseClient } from \"../lib/clickhouse-client.js\";\n\n/**\n * ClickHouse observability handler — platform-level metrics via direct HTTP queries.\n * Replaces FDW RPCs with direct ClickHouse queries for lower latency.\n */\nexport async function handleClickHouse(\n sb: SupabaseClient,\n args: Record<string, unknown>,\n storeId?: string,\n): Promise<{ success: boolean; data?: unknown; error?: string }> {\n const sid = storeId as string;\n const hoursBack = Math.max(1, Math.min(Number(args.hours_back) || 24, 720));\n const limit = Math.max(1, Math.min(Number(args.limit) || 50, 1000));\n const ch = getClickHouseClient();\n\n if (!ch.isEnabled) {\n return { success: false, error: \"ClickHouse is not configured\" };\n }\n\n const storeFilter = sid ? `AND store_id = '${esc(sid)}'` : \"\";\n\n switch (args.action) {\n // ---- traces: List recent traces ----\n case \"traces\": {\n const data = await ch.query<{\n trace_id: string; root_service: string; root_operation: string;\n duration: number; span_count: number; error_count: number;\n started_at: string; status: string;\n }>(`\n SELECT\n trace_id,\n arrayElement(groupArrayIf(service_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_service,\n arrayElement(groupArrayIf(operation_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_operation,\n toInt32(dateDiff('millisecond', min(started_at), max(ended_at))) AS duration,\n count() AS span_count,\n countIf(status_code = 'ERROR') AS error_count,\n min(started_at) AS started_at,\n if(countIf(status_code = 'ERROR') > 0, 'error', 'ok') AS status\n FROM ai_spans\n WHERE 1=1 ${storeFilter}\n GROUP BY trace_id\n ORDER BY min(started_at) DESC\n LIMIT ${limit}\n `);\n return { success: true, data };\n }\n\n // ---- trace_detail: Full trace waterfall ----\n case \"trace_detail\": {\n const traceId = args.trace_id as string;\n if (!traceId) return { success: false, error: \"trace_id is required\" };\n const spans = await ch.query<Record<string, unknown>>(`\n SELECT\n span_id, parent_span_id, trace_id, operation_name,\n service_name, duration_ms AS duration,\n started_at, ended_at,\n if(status_code = 'ERROR', 'error', 'ok') AS status,\n attributes, events\n FROM ai_spans\n WHERE trace_id = '${esc(traceId)}'\n ORDER BY started_at ASC\n `);\n return { success: true, data: { trace_id: traceId, spans } };\n }\n\n // ---- function_health: Service health metrics ----\n case \"function_health\": {\n const data = await ch.query<Record<string, unknown>>(`\n SELECT\n service_name AS function_name,\n sum(total_requests) AS total_reqs,\n sum(error_count) AS total_errors,\n sum(client_error_count) AS client_errors,\n if(sum(total_requests) > 0, sum(error_count) / sum(total_requests), 0) AS error_rate,\n if(sum(latency_count) > 0, sum(latency_sum_ms) / sum(latency_count), 0) AS avg_latency_ms,\n max(latency_max_ms) AS p95_latency_ms,\n min(latency_min_ms) AS min_latency_ms,\n max(latency_max_ms) AS max_latency_ms,\n max(bucket) AS last_request_at\n FROM function_health\n WHERE bucket >= now() - INTERVAL ${hoursBack} HOUR\n ${storeFilter}\n GROUP BY service_name\n `);\n return { success: true, data };\n }\n\n // ---- function_logs: Request logs ----\n case \"function_logs\": {\n const functionName = args.function_name as string;\n if (!functionName) return { success: false, error: \"function_name is required\" };\n const levelFilter = args.level ? `AND severity = '${esc(args.level as string)}'` : \"\";\n const data = await ch.query<Record<string, unknown>>(`\n SELECT\n request_id, service_name AS function_name, http_status AS status,\n duration_ms, http_method AS method, http_path AS path,\n store_id, status_code = 'ERROR' AS has_error,\n started_at AS first_seen, events AS lines\n FROM ai_spans\n WHERE service_name = '${esc(functionName)}'\n ${storeFilter} ${levelFilter}\n AND span_kind = 'SERVER'\n ORDER BY started_at DESC\n LIMIT ${limit}\n `);\n return { success: true, data };\n }\n\n // ---- cost_budgets: Budget utilization ----\n case \"cost_budgets\": {\n if (!sid) return { success: false, error: \"store_id is required for cost_budgets\" };\n // Cost budgets require Postgres for ai_cost_budgets table + ClickHouse for spend\n const { data: budgets, error: budgetErr } = await sb\n .from(\"ai_cost_budgets\")\n .select(\"agent_id, limit_usd, is_active, created_at\")\n .eq(\"is_active\", true)\n .eq(\"store_id\", sid);\n if (budgetErr) return { success: false, error: budgetErr.message };\n\n const agentIds = (budgets || []).map(b => `'${esc(b.agent_id)}'`).join(\",\");\n const spendRows = agentIds\n ? await ch.query<{ agent_id: string; total_cost_usd: number }>(`\n SELECT agent_id, sum(total_cost_usd) AS total_cost_usd\n FROM token_usage_hourly\n WHERE agent_id IN (${agentIds})\n AND store_id = '${esc(sid)}'\n GROUP BY agent_id\n `)\n : [];\n\n const spendMap = new Map(spendRows.map(r => [r.agent_id, r.total_cost_usd]));\n const data = (budgets || []).map(b => {\n const spend = spendMap.get(b.agent_id) || 0;\n return {\n agent_id: b.agent_id,\n budget_amount: b.limit_usd,\n current_spend: spend,\n period_start: b.created_at,\n period_end: new Date().toISOString(),\n utilization_pct: b.limit_usd > 0 ? (spend / b.limit_usd) * 100 : 0,\n is_over_budget: spend > b.limit_usd,\n };\n });\n return { success: true, data };\n }\n\n // ---- error_groups: Error fingerprints grouped ----\n case \"error_groups\": {\n const data = await ch.query<{\n fingerprint: string; error_type: string; message: string;\n severity: string; service: string; count: number; last_seen: string;\n }>(`\n SELECT\n fingerprint,\n any(error_type) AS error_type,\n substring(any(error_message), 1, 200) AS message,\n any(severity) AS severity,\n any(service_name) AS service,\n count() AS count,\n max(occurred_at) AS last_seen\n FROM error_events\n WHERE occurred_at >= now() - INTERVAL ${hoursBack} HOUR\n GROUP BY fingerprint\n ORDER BY count DESC\n LIMIT ${limit}\n `);\n return { success: true, data: { groups: data, total_groups: data.length, hours_back: hoursBack } };\n }\n\n // ---- system_status: Combined overview ----\n case \"system_status\": {\n const [functions, traces, budgetData] = await Promise.all([\n ch.query<Record<string, unknown>>(`\n SELECT\n service_name AS function_name,\n sum(total_requests) AS total_reqs,\n sum(error_count) AS total_errors,\n if(sum(total_requests) > 0, sum(error_count) / sum(total_requests), 0) AS error_rate,\n max(bucket) AS last_request_at\n FROM function_health\n WHERE bucket >= now() - INTERVAL ${Math.min(hoursBack, 1)} HOUR\n ${storeFilter}\n GROUP BY service_name\n `),\n ch.query<Record<string, unknown>>(`\n SELECT\n trace_id,\n arrayElement(groupArrayIf(service_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_service,\n arrayElement(groupArrayIf(operation_name, parent_span_id = '' OR parent_span_id IS NULL), 1) AS root_operation,\n toInt32(dateDiff('millisecond', min(started_at), max(ended_at))) AS duration,\n count() AS span_count,\n countIf(status_code = 'ERROR') AS error_count,\n min(started_at) AS started_at,\n if(countIf(status_code = 'ERROR') > 0, 'error', 'ok') AS status\n FROM ai_spans\n WHERE 1=1 ${storeFilter}\n GROUP BY trace_id\n ORDER BY min(started_at) DESC\n LIMIT 10\n `),\n sid\n ? handleClickHouse(sb, { action: \"cost_budgets\" }, sid)\n : Promise.resolve({ success: true, data: null }),\n ]);\n\n const totalRequests = functions.reduce((sum, f) => sum + ((f.total_requests as number) || 0), 0);\n const totalErrors = functions.reduce((sum, f) => sum + ((f.error_count as number) || 0), 0);\n\n return {\n success: true,\n data: {\n summary: {\n function_count: functions.length,\n total_requests: totalRequests,\n total_errors: totalErrors,\n error_rate: totalRequests > 0 ? Math.round((totalErrors / totalRequests) * 10000) / 100 : 0,\n },\n functions,\n recent_traces: traces,\n cost_budgets: budgetData.data,\n hours_back: hoursBack,\n },\n };\n }\n\n default:\n return {\n success: false,\n error: `Unknown clickhouse action: ${args.action}. Available: traces, trace_detail, function_health, function_logs, cost_budgets, error_groups, system_status`,\n };\n }\n}\n\n/** Escape backslashes then single quotes for ClickHouse SQL */\nfunction esc(s: string): string {\n return s.replace(/\\\\/g, \"\\\\\\\\\").replace(/'/g, \"\\\\'\");\n}\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -181,9 +181,27 @@ export async function handleEmail(sb, args, storeId) {
|
|
|
181
181
|
attachments: resolvedAttachments
|
|
182
182
|
} : {})
|
|
183
183
|
};
|
|
184
|
-
if (args.cc)
|
|
185
|
-
|
|
186
|
-
|
|
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
|
|
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
|
-
|
|
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 {
|
|
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
|
|
30
|
+
/** Generate a PNG thumbnail for a PDF document and store it in Supabase. */
|
|
13
31
|
async function generateThumbnail(sb, docId, pdfUrl) {
|
|
14
|
-
|
|
15
|
-
const
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
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
|
|
21
|
-
|
|
22
|
-
|
|
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(
|
|
25
|
-
contentType:
|
|
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(
|
|
31
|
-
await sb.from(
|
|
68
|
+
} = sb.storage.from("screenshots").getPublicUrl(thumbPath);
|
|
69
|
+
await sb.from("store_documents").update({
|
|
32
70
|
thumbnail_url: thumbUrlData.publicUrl
|
|
33
|
-
}).eq(
|
|
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
|
|
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
|
|
1402
|
+
const resolverUrl = `https://${qrDomain}/q/${qrCode}`;
|
|
1361
1403
|
sb.from("qr_codes").insert({
|
|
1362
1404
|
store_id: sid,
|
|
1363
1405
|
code: qrCode,
|