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":"browser-lifecycle.js","names":["chromium","detectAndSolveCaptcha","browserInstance","idleTimer","IDLE_TIMEOUT_MS","PAGE_TIMEOUT_MS","MAX_RESULT_BYTES","MAX_TEXT_CHARS","resetIdleTimer","clearTimeout","setTimeout","close","console","log","getBrowser","isConnected","launch","headless","executablePath","process","env","PLAYWRIGHT_CHROMIUM_EXECUTABLE_PATH","undefined","args","on","withPage","url","waitFor","fn","browser","context","newContext","userAgent","viewport","width","height","ignoreHTTPSErrors","setDefaultTimeout","page","newPage","goto","waitUntil","timeout","captchaResult","solved","type","waitForTimeout","waitForSelector","catch","isBlockedUrl","parsed","URL","protocol","host","hostname","toLowerCase","endsWith","test","startsWith","includes","stripScriptsAndStyles","html","replace","trim","truncate","text","maxChars","length","substring","truncateResult","data","json","JSON","stringify","parse"],"sources":["../../../src/server/handlers/browser-lifecycle.ts"],"sourcesContent":["// browser-lifecycle.ts — Browser instance management, page context, SSRF protection, text helpers\n\nimport { chromium, type Browser, type Page } from \"playwright-core\";\nimport { detectAndSolveCaptcha } from \"./browser-captcha.js\";\n\n// ============================================================================\n// BROWSER INSTANCE LIFECYCLE\n// ============================================================================\n\nlet browserInstance: Browser | null = null;\nlet idleTimer: ReturnType<typeof setTimeout> | null = null;\nconst IDLE_TIMEOUT_MS = 5 * 60 * 1000; // 5 minutes\nexport const PAGE_TIMEOUT_MS = 60_000; // 60 seconds per page operation (increased for CAPTCHA solving)\nexport const MAX_RESULT_BYTES = 500 * 1024; // 500KB safety cap — context_management handles limits\nexport const MAX_TEXT_CHARS = 200_000; // 200K chars for text content\n\nfunction resetIdleTimer(): void {\n if (idleTimer) clearTimeout(idleTimer);\n idleTimer = setTimeout(async () => {\n if (browserInstance) {\n try {\n await browserInstance.close();\n } catch { /* browser may already be closed */ }\n browserInstance = null;\n console.log(\"[browser] Closed after idle timeout\");\n }\n }, IDLE_TIMEOUT_MS);\n}\n\nasync function getBrowser(): Promise<Browser> {\n if (browserInstance && browserInstance.isConnected()) {\n resetIdleTimer();\n return browserInstance;\n }\n\n browserInstance = await chromium.launch({\n headless: true,\n executablePath: process.env.PLAYWRIGHT_CHROMIUM_EXECUTABLE_PATH || undefined,\n args: [\n \"--no-sandbox\",\n \"--disable-setuid-sandbox\",\n \"--disable-dev-shm-usage\",\n \"--disable-gpu\",\n ],\n });\n\n browserInstance.on(\"disconnected\", () => {\n browserInstance = null;\n if (idleTimer) {\n clearTimeout(idleTimer);\n idleTimer = null;\n }\n });\n\n resetIdleTimer();\n console.log(\"[browser] Launched new instance\");\n return browserInstance;\n}\n\nexport async function withPage<T>(url: string, waitFor: string | undefined, fn: (page: Page) => Promise<T>): Promise<T> {\n const browser = await getBrowser();\n const context = await browser.newContext({\n userAgent: \"WhaleBot/1.0 (https://whale.app)\",\n viewport: { width: 1280, height: 720 },\n ignoreHTTPSErrors: true,\n });\n context.setDefaultTimeout(PAGE_TIMEOUT_MS);\n\n const page = await context.newPage();\n try {\n await page.goto(url, { waitUntil: \"domcontentloaded\", timeout: PAGE_TIMEOUT_MS });\n\n // Auto-detect and solve CAPTCHAs after page load\n const captchaResult = await detectAndSolveCaptcha(page);\n if (captchaResult.solved) {\n console.log(`[browser] Solved ${captchaResult.type} CAPTCHA on ${url}`);\n // Wait for page to update after CAPTCHA solution\n await page.waitForTimeout(2000);\n }\n\n if (waitFor) {\n await page.waitForSelector(waitFor, { timeout: PAGE_TIMEOUT_MS });\n }\n\n return await fn(page);\n } finally {\n await page.close().catch(() => {});\n await context.close().catch(() => {});\n }\n}\n\n// ============================================================================\n// SSRF PROTECTION\n// ============================================================================\n\nexport function isBlockedUrl(url: string): string | null {\n let parsed: URL;\n try {\n parsed = new URL(url);\n } catch {\n return \"Invalid URL\";\n }\n\n if (parsed.protocol !== \"http:\" && parsed.protocol !== \"https:\") {\n return `Blocked protocol: ${parsed.protocol}`;\n }\n\n const host = parsed.hostname.toLowerCase();\n\n // Block localhost variants\n if (host === \"localhost\" || host === \"127.0.0.1\" || host === \"0.0.0.0\" ||\n host === \"::1\" || host === \"[::1]\") {\n return \"Blocked: localhost\";\n }\n\n // Block private/internal domains\n if (host.endsWith(\".internal\") || host.endsWith(\".local\") ||\n host === \"metadata.google.internal\") {\n return \"Blocked: internal/local domain\";\n }\n\n // Block private IP ranges\n if (/^10\\./.test(host) ||\n /^172\\.(1[6-9]|2\\d|3[01])\\./.test(host) ||\n /^192\\.168\\./.test(host) ||\n /^169\\.254\\./.test(host)) {\n return \"Blocked: private IP range\";\n }\n\n // Block IPv6 private\n if (host.startsWith(\"fd\") || host.startsWith(\"fc00:\") ||\n host.startsWith(\"fe80:\") || host.includes(\"::ffff:\")) {\n return \"Blocked: IPv6 private range\";\n }\n\n return null;\n}\n\n// ============================================================================\n// TEXT CLEANUP\n// ============================================================================\n\nexport function stripScriptsAndStyles(html: string): string {\n return html\n .replace(/<script[\\s\\S]*?<\\/script>/gi, \"\")\n .replace(/<style[\\s\\S]*?<\\/style>/gi, \"\")\n .replace(/<[^>]+>/g, \" \")\n .replace(/\\s+/g, \" \")\n .trim();\n}\n\nexport function truncate(text: string, maxChars: number): string {\n if (text.length <= maxChars) return text;\n return text.substring(0, maxChars) + \"\\n...[truncated]\";\n}\n\nexport function truncateResult(data: unknown): unknown {\n const json = JSON.stringify(data);\n if (json.length <= MAX_RESULT_BYTES) return data;\n // If too large, stringify and truncate\n return JSON.parse(truncate(json, MAX_RESULT_BYTES));\n}\n"],"mappings":"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","ignoreList":[]}
|
|
1
|
+
{"version":3,"file":"browser-lifecycle.js","names":["chromium","detectAndSolveCaptcha","validateUrl","browserInstance","idleTimer","IDLE_TIMEOUT_MS","PAGE_TIMEOUT_MS","MAX_RESULT_BYTES","MAX_TEXT_CHARS","resetIdleTimer","clearTimeout","setTimeout","close","console","log","getBrowser","isConnected","launch","headless","executablePath","process","env","PLAYWRIGHT_CHROMIUM_EXECUTABLE_PATH","undefined","args","on","withPage","url","waitFor","fn","blocked","isBlockedUrl","Error","dnsBlocked","browser","context","newContext","userAgent","viewport","width","height","ignoreHTTPSErrors","setDefaultTimeout","page","newPage","goto","waitUntil","timeout","captchaResult","solved","type","waitForTimeout","waitForSelector","catch","parsed","URL","protocol","host","hostname","toLowerCase","endsWith","test","startsWith","includes","stripScriptsAndStyles","html","replace","trim","truncate","text","maxChars","length","substring","truncateResult","data","json","JSON","stringify","parse"],"sources":["../../../src/server/handlers/browser-lifecycle.ts"],"sourcesContent":["// browser-lifecycle.ts — Browser instance management, page context, SSRF protection, text helpers\n\nimport { chromium, type Browser, type Page } from \"playwright-core\";\nimport { detectAndSolveCaptcha } from \"./browser-captcha.js\";\nimport { validateUrl } from \"../lib/ssrf-guard.js\";\n\n// ============================================================================\n// BROWSER INSTANCE LIFECYCLE\n// ============================================================================\n\nlet browserInstance: Browser | null = null;\nlet idleTimer: ReturnType<typeof setTimeout> | null = null;\nconst IDLE_TIMEOUT_MS = 5 * 60 * 1000; // 5 minutes\nexport const PAGE_TIMEOUT_MS = 60_000; // 60 seconds per page operation (increased for CAPTCHA solving)\nexport const MAX_RESULT_BYTES = 500 * 1024; // 500KB safety cap — context_management handles limits\nexport const MAX_TEXT_CHARS = 200_000; // 200K chars for text content\n\nfunction resetIdleTimer(): void {\n if (idleTimer) clearTimeout(idleTimer);\n idleTimer = setTimeout(async () => {\n if (browserInstance) {\n try {\n await browserInstance.close();\n } catch { /* browser may already be closed */ }\n browserInstance = null;\n console.log(\"[browser] Closed after idle timeout\");\n }\n }, IDLE_TIMEOUT_MS);\n}\n\nasync function getBrowser(): Promise<Browser> {\n if (browserInstance && browserInstance.isConnected()) {\n resetIdleTimer();\n return browserInstance;\n }\n\n browserInstance = await chromium.launch({\n headless: true,\n executablePath: process.env.PLAYWRIGHT_CHROMIUM_EXECUTABLE_PATH || undefined,\n args: [\n \"--no-sandbox\",\n \"--disable-setuid-sandbox\",\n \"--disable-dev-shm-usage\",\n \"--disable-gpu\",\n ],\n });\n\n browserInstance.on(\"disconnected\", () => {\n browserInstance = null;\n if (idleTimer) {\n clearTimeout(idleTimer);\n idleTimer = null;\n }\n });\n\n resetIdleTimer();\n console.log(\"[browser] Launched new instance\");\n return browserInstance;\n}\n\nexport async function withPage<T>(url: string, waitFor: string | undefined, fn: (page: Page) => Promise<T>): Promise<T> {\n // SSRF check — fast synchronous pre-check, then DNS-based validation\n const blocked = isBlockedUrl(url);\n if (blocked) {\n throw new Error(blocked);\n }\n const dnsBlocked = await validateUrl(url);\n if (dnsBlocked) {\n throw new Error(dnsBlocked);\n }\n\n const browser = await getBrowser();\n const context = await browser.newContext({\n userAgent: \"WhaleBot/1.0 (https://whale.app)\",\n viewport: { width: 1280, height: 720 },\n ignoreHTTPSErrors: true,\n });\n context.setDefaultTimeout(PAGE_TIMEOUT_MS);\n\n const page = await context.newPage();\n try {\n await page.goto(url, { waitUntil: \"domcontentloaded\", timeout: PAGE_TIMEOUT_MS });\n\n // Auto-detect and solve CAPTCHAs after page load\n const captchaResult = await detectAndSolveCaptcha(page);\n if (captchaResult.solved) {\n console.log(`[browser] Solved ${captchaResult.type} CAPTCHA on ${url}`);\n // Wait for page to update after CAPTCHA solution\n await page.waitForTimeout(2000);\n }\n\n if (waitFor) {\n await page.waitForSelector(waitFor, { timeout: PAGE_TIMEOUT_MS });\n }\n\n return await fn(page);\n } finally {\n await page.close().catch(() => {});\n await context.close().catch(() => {});\n }\n}\n\n// ============================================================================\n// SSRF PROTECTION\n// ============================================================================\n\nexport function isBlockedUrl(url: string): string | null {\n let parsed: URL;\n try {\n parsed = new URL(url);\n } catch {\n return \"Invalid URL\";\n }\n\n if (parsed.protocol !== \"http:\" && parsed.protocol !== \"https:\") {\n return `Blocked protocol: ${parsed.protocol}`;\n }\n\n const host = parsed.hostname.toLowerCase();\n\n // Block localhost variants\n if (host === \"localhost\" || host === \"127.0.0.1\" || host === \"0.0.0.0\" ||\n host === \"::1\" || host === \"[::1]\") {\n return \"Blocked: localhost\";\n }\n\n // Block private/internal domains\n if (host.endsWith(\".internal\") || host.endsWith(\".local\") ||\n host === \"metadata.google.internal\") {\n return \"Blocked: internal/local domain\";\n }\n\n // Block private IP ranges\n if (/^10\\./.test(host) ||\n /^172\\.(1[6-9]|2\\d|3[01])\\./.test(host) ||\n /^192\\.168\\./.test(host) ||\n /^169\\.254\\./.test(host)) {\n return \"Blocked: private IP range\";\n }\n\n // Block IPv6 private\n if (host.startsWith(\"fd\") || host.startsWith(\"fc00:\") ||\n host.startsWith(\"fe80:\") || host.includes(\"::ffff:\")) {\n return \"Blocked: IPv6 private range\";\n }\n\n return null;\n}\n\n// ============================================================================\n// TEXT CLEANUP\n// ============================================================================\n\nexport function stripScriptsAndStyles(html: string): string {\n return html\n .replace(/<script[\\s\\S]*?<\\/script>/gi, \"\")\n .replace(/<style[\\s\\S]*?<\\/style>/gi, \"\")\n .replace(/<[^>]+>/g, \" \")\n .replace(/\\s+/g, \" \")\n .trim();\n}\n\nexport function truncate(text: string, maxChars: number): string {\n if (text.length <= maxChars) return text;\n return text.substring(0, maxChars) + \"\\n...[truncated]\";\n}\n\nexport function truncateResult(data: unknown): unknown {\n const json = JSON.stringify(data);\n if (json.length <= MAX_RESULT_BYTES) return data;\n // If too large, stringify and truncate\n return JSON.parse(truncate(json, MAX_RESULT_BYTES));\n}\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -2,6 +2,7 @@
|
|
|
2
2
|
// Supports: navigate, screenshot, extract, click, fill, evaluate, pdf
|
|
3
3
|
|
|
4
4
|
import { chromium } from "playwright-core";
|
|
5
|
+
import { validateUrl } from "../lib/ssrf-guard.js";
|
|
5
6
|
|
|
6
7
|
// ============================================================================
|
|
7
8
|
// BROWSER INSTANCE LIFECYCLE
|
|
@@ -501,7 +502,7 @@ export async function handleBrowser(_sb, args, _storeId) {
|
|
|
501
502
|
};
|
|
502
503
|
}
|
|
503
504
|
|
|
504
|
-
// SSRF check
|
|
505
|
+
// SSRF check — fast synchronous pre-check, then DNS-based validation
|
|
505
506
|
const blocked = isBlockedUrl(url);
|
|
506
507
|
if (blocked) {
|
|
507
508
|
return {
|
|
@@ -509,6 +510,20 @@ export async function handleBrowser(_sb, args, _storeId) {
|
|
|
509
510
|
error: blocked
|
|
510
511
|
};
|
|
511
512
|
}
|
|
513
|
+
try {
|
|
514
|
+
const dnsBlocked = await validateUrl(url);
|
|
515
|
+
if (dnsBlocked) {
|
|
516
|
+
return {
|
|
517
|
+
success: false,
|
|
518
|
+
error: dnsBlocked
|
|
519
|
+
};
|
|
520
|
+
}
|
|
521
|
+
} catch {
|
|
522
|
+
return {
|
|
523
|
+
success: false,
|
|
524
|
+
error: "URL validation failed"
|
|
525
|
+
};
|
|
526
|
+
}
|
|
512
527
|
try {
|
|
513
528
|
switch (action) {
|
|
514
529
|
case "navigate":
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"browser.js","names":["chromium","browserInstance","idleTimer","IDLE_TIMEOUT_MS","PAGE_TIMEOUT_MS","MAX_RESULT_BYTES","MAX_TEXT_CHARS","resetIdleTimer","clearTimeout","setTimeout","close","console","log","getBrowser","isConnected","launch","headless","executablePath","process","env","PLAYWRIGHT_CHROMIUM_EXECUTABLE_PATH","undefined","args","on","withPage","url","waitFor","fn","browser","context","newContext","userAgent","viewport","width","height","ignoreHTTPSErrors","setDefaultTimeout","page","newPage","goto","waitUntil","timeout","captchaResult","detectAndSolveCaptcha","solved","type","waitForTimeout","waitForSelector","catch","isBlockedUrl","parsed","URL","protocol","host","hostname","toLowerCase","endsWith","test","startsWith","includes","stripScriptsAndStyles","html","replace","trim","truncate","text","maxChars","length","substring","truncateResult","data","json","JSON","stringify","parse","actionNavigate","title","content","textContent","success","text_length","actionScreenshot","buffer","screenshot","fullPage","base64","toString","screenshot_base64","format","size_bytes","actionExtract","selector","error","elements","$$eval","els","map","el","tag","tagName","innerHTML","attributes","Object","fromEntries","Array","from","a","name","value","count","slice","actionClick","click","clicked","actionFill","fill","filled","actionEvaluate","script","result","evaluate","actionPdf","pdf","printBackground","margin","top","bottom","left","right","pdf_base64","callCapMonster","taskType","params","apiKey","CAPMONSTER_API_KEY","Error","createResp","fetch","method","headers","body","clientKey","task","createData","errorId","errorDescription","taskId","i","Promise","r","resultResp","resultData","status","solution","gRecaptchaResponse","token","cf_clearance","pageUrl","recaptchaKey","document","querySelector","classList","contains","getAttribute","iframe","src","match","websiteURL","websiteKey","t","textarea","getElementById","style","display","win","window","cfg","___grecaptcha_cfg","clients","firstKey","keys","first","findCallback","obj","depth","val","values","found","cb","hcaptchaKey","hcaptchaCallback","turnstileKey","input","turnstileCallback","err","message","String","handleBrowser","_sb","_storeId","action","wait_for","blocked","cleanMessage"],"sources":["../../../src/server/handlers/browser.ts"],"sourcesContent":["// server/handlers/browser.ts — Browser automation via Playwright\n// Supports: navigate, screenshot, extract, click, fill, evaluate, pdf\n\nimport type { SupabaseClient } from \"@supabase/supabase-js\";\nimport { chromium, type Browser, type Page } from \"playwright-core\";\n\n// ============================================================================\n// BROWSER INSTANCE LIFECYCLE\n// ============================================================================\n\nlet browserInstance: Browser | null = null;\nlet idleTimer: ReturnType<typeof setTimeout> | null = null;\nconst IDLE_TIMEOUT_MS = 5 * 60 * 1000; // 5 minutes\nconst PAGE_TIMEOUT_MS = 60_000; // 60 seconds per page operation (increased for CAPTCHA solving)\nconst MAX_RESULT_BYTES = 500 * 1024; // 500KB safety cap — context_management handles limits\nconst MAX_TEXT_CHARS = 200_000; // 200K chars for text content\n\nfunction resetIdleTimer(): void {\n if (idleTimer) clearTimeout(idleTimer);\n idleTimer = setTimeout(async () => {\n if (browserInstance) {\n try {\n await browserInstance.close();\n } catch { /* browser may already be closed */ }\n browserInstance = null;\n console.log(\"[browser] Closed after idle timeout\");\n }\n }, IDLE_TIMEOUT_MS);\n}\n\nasync function getBrowser(): Promise<Browser> {\n if (browserInstance && browserInstance.isConnected()) {\n resetIdleTimer();\n return browserInstance;\n }\n\n browserInstance = await chromium.launch({\n headless: true,\n executablePath: process.env.PLAYWRIGHT_CHROMIUM_EXECUTABLE_PATH || undefined,\n args: [\n \"--no-sandbox\",\n \"--disable-setuid-sandbox\",\n \"--disable-dev-shm-usage\",\n \"--disable-gpu\",\n ],\n });\n\n browserInstance.on(\"disconnected\", () => {\n browserInstance = null;\n if (idleTimer) {\n clearTimeout(idleTimer);\n idleTimer = null;\n }\n });\n\n resetIdleTimer();\n console.log(\"[browser] Launched new instance\");\n return browserInstance;\n}\n\nasync function withPage<T>(url: string, waitFor: string | undefined, fn: (page: Page) => Promise<T>): Promise<T> {\n const browser = await getBrowser();\n const context = await browser.newContext({\n userAgent: \"WhaleBot/1.0 (https://whale.app)\",\n viewport: { width: 1280, height: 720 },\n ignoreHTTPSErrors: true,\n });\n context.setDefaultTimeout(PAGE_TIMEOUT_MS);\n\n const page = await context.newPage();\n try {\n await page.goto(url, { waitUntil: \"domcontentloaded\", timeout: PAGE_TIMEOUT_MS });\n\n // Auto-detect and solve CAPTCHAs after page load\n const captchaResult = await detectAndSolveCaptcha(page);\n if (captchaResult.solved) {\n console.log(`[browser] Solved ${captchaResult.type} CAPTCHA on ${url}`);\n // Wait for page to update after CAPTCHA solution\n await page.waitForTimeout(2000);\n }\n\n if (waitFor) {\n await page.waitForSelector(waitFor, { timeout: PAGE_TIMEOUT_MS });\n }\n\n return await fn(page);\n } finally {\n await page.close().catch(() => {});\n await context.close().catch(() => {});\n }\n}\n\n// ============================================================================\n// SSRF PROTECTION\n// ============================================================================\n\nfunction isBlockedUrl(url: string): string | null {\n let parsed: URL;\n try {\n parsed = new URL(url);\n } catch {\n return \"Invalid URL\";\n }\n\n if (parsed.protocol !== \"http:\" && parsed.protocol !== \"https:\") {\n return `Blocked protocol: ${parsed.protocol}`;\n }\n\n const host = parsed.hostname.toLowerCase();\n\n // Block localhost variants\n if (host === \"localhost\" || host === \"127.0.0.1\" || host === \"0.0.0.0\" ||\n host === \"::1\" || host === \"[::1]\") {\n return \"Blocked: localhost\";\n }\n\n // Block private/internal domains\n if (host.endsWith(\".internal\") || host.endsWith(\".local\") ||\n host === \"metadata.google.internal\") {\n return \"Blocked: internal/local domain\";\n }\n\n // Block private IP ranges\n if (/^10\\./.test(host) ||\n /^172\\.(1[6-9]|2\\d|3[01])\\./.test(host) ||\n /^192\\.168\\./.test(host) ||\n /^169\\.254\\./.test(host)) {\n return \"Blocked: private IP range\";\n }\n\n // Block IPv6 private\n if (host.startsWith(\"fd\") || host.startsWith(\"fc00:\") ||\n host.startsWith(\"fe80:\") || host.includes(\"::ffff:\")) {\n return \"Blocked: IPv6 private range\";\n }\n\n return null;\n}\n\n// ============================================================================\n// TEXT CLEANUP\n// ============================================================================\n\nfunction stripScriptsAndStyles(html: string): string {\n return html\n .replace(/<script[\\s\\S]*?<\\/script>/gi, \"\")\n .replace(/<style[\\s\\S]*?<\\/style>/gi, \"\")\n .replace(/<[^>]+>/g, \" \")\n .replace(/\\s+/g, \" \")\n .trim();\n}\n\nfunction truncate(text: string, maxChars: number): string {\n if (text.length <= maxChars) return text;\n return text.substring(0, maxChars) + \"\\n...[truncated]\";\n}\n\nfunction truncateResult(data: unknown): unknown {\n const json = JSON.stringify(data);\n if (json.length <= MAX_RESULT_BYTES) return data;\n // If too large, stringify and truncate\n return JSON.parse(truncate(json, MAX_RESULT_BYTES));\n}\n\n// ============================================================================\n// ACTION HANDLERS\n// ============================================================================\n\nasync function actionNavigate(url: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n return withPage(url, waitFor, async (page) => {\n const title = await page.title();\n const html = await page.content();\n const textContent = stripScriptsAndStyles(html);\n\n return {\n success: true,\n data: {\n url: page.url(),\n title,\n text: truncate(textContent, MAX_TEXT_CHARS),\n text_length: textContent.length,\n },\n };\n });\n}\n\nasync function actionScreenshot(url: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n return withPage(url, waitFor, async (page) => {\n const buffer = await page.screenshot({ type: \"png\", fullPage: false });\n const base64 = buffer.toString(\"base64\");\n const title = await page.title();\n\n return {\n success: true,\n data: {\n url: page.url(),\n title,\n screenshot_base64: base64,\n format: \"png\",\n size_bytes: buffer.length,\n },\n };\n });\n}\n\nasync function actionExtract(url: string, selector: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n if (!selector) {\n return { success: false, error: \"selector is required for extract action\" };\n }\n\n return withPage(url, waitFor, async (page) => {\n const elements = await page.$$eval(selector, (els) =>\n els.map((el) => ({\n tag: el.tagName.toLowerCase(),\n text: el.textContent?.trim() || \"\",\n html: el.innerHTML.substring(0, 2000),\n attributes: Object.fromEntries(\n Array.from(el.attributes).map((a) => [a.name, a.value])\n ),\n }))\n );\n\n return {\n success: true,\n data: {\n url: page.url(),\n selector,\n count: elements.length,\n elements: elements.slice(0, 50), // Cap at 50 elements\n },\n };\n });\n}\n\nasync function actionClick(url: string, selector: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n if (!selector) {\n return { success: false, error: \"selector is required for click action\" };\n }\n\n return withPage(url, waitFor, async (page) => {\n await page.click(selector, { timeout: PAGE_TIMEOUT_MS });\n\n // Wait a moment for any navigation or dynamic content\n await page.waitForTimeout(1000);\n\n const title = await page.title();\n const html = await page.content();\n const textContent = stripScriptsAndStyles(html);\n\n return {\n success: true,\n data: {\n url: page.url(),\n title,\n clicked: selector,\n text: truncate(textContent, MAX_TEXT_CHARS),\n },\n };\n });\n}\n\nasync function actionFill(url: string, selector: string, value: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n if (!selector) {\n return { success: false, error: \"selector is required for fill action\" };\n }\n if (value === undefined || value === null) {\n return { success: false, error: \"value is required for fill action\" };\n }\n\n return withPage(url, waitFor, async (page) => {\n await page.fill(selector, value, { timeout: PAGE_TIMEOUT_MS });\n\n return {\n success: true,\n data: {\n url: page.url(),\n filled: { selector, value },\n },\n };\n });\n}\n\nasync function actionEvaluate(url: string, script: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n if (!script) {\n return { success: false, error: \"script is required for evaluate action\" };\n }\n\n return withPage(url, waitFor, async (page) => {\n const result = await page.evaluate(script);\n\n return {\n success: true,\n data: {\n url: page.url(),\n result: truncateResult(result),\n },\n };\n });\n}\n\nasync function actionPdf(url: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n return withPage(url, waitFor, async (page) => {\n const buffer = await page.pdf({\n format: \"A4\",\n printBackground: true,\n margin: { top: \"1cm\", bottom: \"1cm\", left: \"1cm\", right: \"1cm\" },\n });\n\n const base64 = buffer.toString(\"base64\");\n const title = await page.title();\n\n return {\n success: true,\n data: {\n url: page.url(),\n title,\n pdf_base64: base64,\n format: \"A4\",\n size_bytes: buffer.length,\n },\n };\n });\n}\n\n// ============================================================================\n// CAPTCHA DETECTION & SOLVING (via CapMonster Cloud)\n// ============================================================================\n\ninterface CaptchaResult {\n solved: boolean;\n type?: string;\n}\n\nasync function callCapMonster(taskType: string, params: Record<string, unknown>): Promise<string> {\n const apiKey = process.env.CAPMONSTER_API_KEY;\n if (!apiKey) throw new Error(\"CAPMONSTER_API_KEY not configured\");\n\n const createResp = await fetch(\"https://api.capmonster.cloud/createTask\", {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ clientKey: apiKey, task: { type: taskType, ...params } }),\n });\n const createData = await createResp.json() as { errorId: number; errorDescription?: string; taskId?: number };\n if (createData.errorId) throw new Error(`CapMonster create: ${createData.errorDescription}`);\n const taskId = createData.taskId;\n\n // Poll for result (max 120s, 2s intervals)\n for (let i = 0; i < 60; i++) {\n await new Promise(r => setTimeout(r, 2000));\n const resultResp = await fetch(\"https://api.capmonster.cloud/getTaskResult\", {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ clientKey: apiKey, taskId }),\n });\n const resultData = await resultResp.json() as {\n status: string;\n errorId: number;\n errorDescription?: string;\n solution?: { gRecaptchaResponse?: string; token?: string; cf_clearance?: string };\n };\n if (resultData.status === \"ready\") {\n return resultData.solution?.gRecaptchaResponse || resultData.solution?.token || resultData.solution?.cf_clearance || \"\";\n }\n if (resultData.errorId) throw new Error(`CapMonster result: ${resultData.errorDescription}`);\n }\n throw new Error(\"CapMonster: timeout waiting for solution (120s)\");\n}\n\nasync function detectAndSolveCaptcha(page: Page): Promise<CaptchaResult> {\n try {\n // Skip if no API key configured\n if (!process.env.CAPMONSTER_API_KEY) return { solved: false };\n\n const pageUrl = page.url();\n\n // Detect reCAPTCHA v2\n const recaptchaKey = await page.evaluate(() => {\n const el = document.querySelector('.g-recaptcha[data-sitekey], [data-sitekey]');\n if (el && el.classList.contains('g-recaptcha')) return el.getAttribute('data-sitekey');\n // Also check for reCAPTCHA iframe\n const iframe = document.querySelector('iframe[src*=\"recaptcha\"]');\n if (iframe) {\n const src = iframe.getAttribute('src') || '';\n const match = src.match(/[?&]k=([^&]+)/);\n return match ? match[1] : null;\n }\n return null;\n }).catch(() => null);\n\n if (recaptchaKey) {\n console.log(`[browser] Detected reCAPTCHA v2 on ${pageUrl}`);\n const token = await callCapMonster(\"RecaptchaV2TaskProxyless\", {\n websiteURL: pageUrl,\n websiteKey: recaptchaKey,\n });\n await page.evaluate((t: string) => {\n const textarea = document.getElementById('g-recaptcha-response') as HTMLTextAreaElement | null;\n if (textarea) { textarea.style.display = 'block'; textarea.value = t; }\n // Try triggering the callback\n const win = window as unknown as Record<string, unknown>;\n try {\n const cfg = win.___grecaptcha_cfg as Record<string, unknown> | undefined;\n if (cfg?.clients) {\n const clients = cfg.clients as Record<string, unknown>;\n const firstKey = Object.keys(clients)[0];\n const first = firstKey ? clients[firstKey] : null;\n if (first) {\n const findCallback = (obj: unknown, depth = 0): ((t: string) => void) | null => {\n if (depth > 4 || !obj || typeof obj !== 'object') return null;\n for (const val of Object.values(obj as Record<string, unknown>)) {\n if (typeof val === 'function') return val as (t: string) => void;\n const found = findCallback(val, depth + 1);\n if (found) return found;\n }\n return null;\n };\n const cb = findCallback(first);\n if (cb) cb(t);\n }\n }\n } catch { /* callback search failed, form submit will work */ }\n }, token);\n return { solved: true, type: \"reCAPTCHA v2\" };\n }\n\n // Detect hCaptcha\n const hcaptchaKey = await page.evaluate(() => {\n const el = document.querySelector('.h-captcha[data-sitekey]');\n return el?.getAttribute('data-sitekey') || null;\n }).catch(() => null);\n\n if (hcaptchaKey) {\n console.log(`[browser] Detected hCaptcha on ${pageUrl}`);\n const token = await callCapMonster(\"HCaptchaTaskProxyless\", {\n websiteURL: pageUrl,\n websiteKey: hcaptchaKey,\n });\n await page.evaluate((t: string) => {\n const textarea = document.querySelector('[name=\"h-captcha-response\"], [name=\"g-recaptcha-response\"]') as HTMLTextAreaElement | null;\n if (textarea) { textarea.value = t; }\n const win = window as unknown as Record<string, unknown>;\n if (typeof win.hcaptchaCallback === 'function') {\n (win.hcaptchaCallback as (t: string) => void)(t);\n }\n }, token);\n return { solved: true, type: \"hCaptcha\" };\n }\n\n // Detect Cloudflare Turnstile\n const turnstileKey = await page.evaluate(() => {\n const el = document.querySelector('.cf-turnstile[data-sitekey]');\n return el?.getAttribute('data-sitekey') || null;\n }).catch(() => null);\n\n if (turnstileKey) {\n console.log(`[browser] Detected Turnstile on ${pageUrl}`);\n const token = await callCapMonster(\"TurnstileTaskProxyless\", {\n websiteURL: pageUrl,\n websiteKey: turnstileKey,\n });\n await page.evaluate((t: string) => {\n const input = document.querySelector('[name=\"cf-turnstile-response\"]') as HTMLInputElement | null;\n if (input) input.value = t;\n const win = window as unknown as Record<string, unknown>;\n if (typeof win.turnstileCallback === 'function') {\n (win.turnstileCallback as (t: string) => void)(t);\n }\n }, token);\n return { solved: true, type: \"Turnstile\" };\n }\n\n return { solved: false };\n } catch (err) {\n console.log(`[browser] CAPTCHA detection/solving failed: ${err instanceof Error ? err.message : String(err)}`);\n return { solved: false };\n }\n}\n\n// ============================================================================\n// MAIN HANDLER\n// ============================================================================\n\nexport async function handleBrowser(\n _sb: SupabaseClient,\n args: Record<string, unknown>,\n _storeId?: string\n): Promise<{ success: boolean; data?: unknown; error?: string }> {\n const action = args.action as string;\n const url = args.url as string;\n const selector = args.selector as string | undefined;\n const value = args.value as string | undefined;\n const script = args.script as string | undefined;\n const waitFor = args.wait_for as string | undefined;\n\n if (!action) {\n return { success: false, error: \"action is required\" };\n }\n if (!url) {\n return { success: false, error: \"url is required\" };\n }\n\n // SSRF check\n const blocked = isBlockedUrl(url);\n if (blocked) {\n return { success: false, error: blocked };\n }\n\n try {\n switch (action) {\n case \"navigate\":\n return await actionNavigate(url, waitFor);\n case \"screenshot\":\n return await actionScreenshot(url, waitFor);\n case \"extract\":\n return await actionExtract(url, selector!, waitFor);\n case \"click\":\n return await actionClick(url, selector!, waitFor);\n case \"fill\":\n return await actionFill(url, selector!, value!, waitFor);\n case \"evaluate\":\n return await actionEvaluate(url, script!, waitFor);\n case \"pdf\":\n return await actionPdf(url, waitFor);\n default:\n return { success: false, error: `Unknown browser action: ${action}. Valid: navigate, screenshot, extract, click, fill, evaluate, pdf` };\n }\n } catch (err: unknown) {\n const message = err instanceof Error ? err.message : String(err);\n // Clean up common Playwright error noise\n const cleanMessage = message\n .replace(/=========================== logs ===========================[^]*?============================================================/gs, \"\")\n .trim();\n return { success: false, error: `Browser error: ${cleanMessage.substring(0, 500)}` };\n }\n}\n"],"mappings":"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","ignoreList":[]}
|
|
1
|
+
{"version":3,"file":"browser.js","names":["chromium","validateUrl","browserInstance","idleTimer","IDLE_TIMEOUT_MS","PAGE_TIMEOUT_MS","MAX_RESULT_BYTES","MAX_TEXT_CHARS","resetIdleTimer","clearTimeout","setTimeout","close","console","log","getBrowser","isConnected","launch","headless","executablePath","process","env","PLAYWRIGHT_CHROMIUM_EXECUTABLE_PATH","undefined","args","on","withPage","url","waitFor","fn","browser","context","newContext","userAgent","viewport","width","height","ignoreHTTPSErrors","setDefaultTimeout","page","newPage","goto","waitUntil","timeout","captchaResult","detectAndSolveCaptcha","solved","type","waitForTimeout","waitForSelector","catch","isBlockedUrl","parsed","URL","protocol","host","hostname","toLowerCase","endsWith","test","startsWith","includes","stripScriptsAndStyles","html","replace","trim","truncate","text","maxChars","length","substring","truncateResult","data","json","JSON","stringify","parse","actionNavigate","title","content","textContent","success","text_length","actionScreenshot","buffer","screenshot","fullPage","base64","toString","screenshot_base64","format","size_bytes","actionExtract","selector","error","elements","$$eval","els","map","el","tag","tagName","innerHTML","attributes","Object","fromEntries","Array","from","a","name","value","count","slice","actionClick","click","clicked","actionFill","fill","filled","actionEvaluate","script","result","evaluate","actionPdf","pdf","printBackground","margin","top","bottom","left","right","pdf_base64","callCapMonster","taskType","params","apiKey","CAPMONSTER_API_KEY","Error","createResp","fetch","method","headers","body","clientKey","task","createData","errorId","errorDescription","taskId","i","Promise","r","resultResp","resultData","status","solution","gRecaptchaResponse","token","cf_clearance","pageUrl","recaptchaKey","document","querySelector","classList","contains","getAttribute","iframe","src","match","websiteURL","websiteKey","t","textarea","getElementById","style","display","win","window","cfg","___grecaptcha_cfg","clients","firstKey","keys","first","findCallback","obj","depth","val","values","found","cb","hcaptchaKey","hcaptchaCallback","turnstileKey","input","turnstileCallback","err","message","String","handleBrowser","_sb","_storeId","action","wait_for","blocked","dnsBlocked","cleanMessage"],"sources":["../../../src/server/handlers/browser.ts"],"sourcesContent":["// server/handlers/browser.ts — Browser automation via Playwright\n// Supports: navigate, screenshot, extract, click, fill, evaluate, pdf\n\nimport type { SupabaseClient } from \"@supabase/supabase-js\";\nimport { chromium, type Browser, type Page } from \"playwright-core\";\nimport { validateUrl } from \"../lib/ssrf-guard.js\";\n\n// ============================================================================\n// BROWSER INSTANCE LIFECYCLE\n// ============================================================================\n\nlet browserInstance: Browser | null = null;\nlet idleTimer: ReturnType<typeof setTimeout> | null = null;\nconst IDLE_TIMEOUT_MS = 5 * 60 * 1000; // 5 minutes\nconst PAGE_TIMEOUT_MS = 60_000; // 60 seconds per page operation (increased for CAPTCHA solving)\nconst MAX_RESULT_BYTES = 500 * 1024; // 500KB safety cap — context_management handles limits\nconst MAX_TEXT_CHARS = 200_000; // 200K chars for text content\n\nfunction resetIdleTimer(): void {\n if (idleTimer) clearTimeout(idleTimer);\n idleTimer = setTimeout(async () => {\n if (browserInstance) {\n try {\n await browserInstance.close();\n } catch { /* browser may already be closed */ }\n browserInstance = null;\n console.log(\"[browser] Closed after idle timeout\");\n }\n }, IDLE_TIMEOUT_MS);\n}\n\nasync function getBrowser(): Promise<Browser> {\n if (browserInstance && browserInstance.isConnected()) {\n resetIdleTimer();\n return browserInstance;\n }\n\n browserInstance = await chromium.launch({\n headless: true,\n executablePath: process.env.PLAYWRIGHT_CHROMIUM_EXECUTABLE_PATH || undefined,\n args: [\n \"--no-sandbox\",\n \"--disable-setuid-sandbox\",\n \"--disable-dev-shm-usage\",\n \"--disable-gpu\",\n ],\n });\n\n browserInstance.on(\"disconnected\", () => {\n browserInstance = null;\n if (idleTimer) {\n clearTimeout(idleTimer);\n idleTimer = null;\n }\n });\n\n resetIdleTimer();\n console.log(\"[browser] Launched new instance\");\n return browserInstance;\n}\n\nasync function withPage<T>(url: string, waitFor: string | undefined, fn: (page: Page) => Promise<T>): Promise<T> {\n const browser = await getBrowser();\n const context = await browser.newContext({\n userAgent: \"WhaleBot/1.0 (https://whale.app)\",\n viewport: { width: 1280, height: 720 },\n ignoreHTTPSErrors: true,\n });\n context.setDefaultTimeout(PAGE_TIMEOUT_MS);\n\n const page = await context.newPage();\n try {\n await page.goto(url, { waitUntil: \"domcontentloaded\", timeout: PAGE_TIMEOUT_MS });\n\n // Auto-detect and solve CAPTCHAs after page load\n const captchaResult = await detectAndSolveCaptcha(page);\n if (captchaResult.solved) {\n console.log(`[browser] Solved ${captchaResult.type} CAPTCHA on ${url}`);\n // Wait for page to update after CAPTCHA solution\n await page.waitForTimeout(2000);\n }\n\n if (waitFor) {\n await page.waitForSelector(waitFor, { timeout: PAGE_TIMEOUT_MS });\n }\n\n return await fn(page);\n } finally {\n await page.close().catch(() => {});\n await context.close().catch(() => {});\n }\n}\n\n// ============================================================================\n// SSRF PROTECTION\n// ============================================================================\n\nfunction isBlockedUrl(url: string): string | null {\n let parsed: URL;\n try {\n parsed = new URL(url);\n } catch {\n return \"Invalid URL\";\n }\n\n if (parsed.protocol !== \"http:\" && parsed.protocol !== \"https:\") {\n return `Blocked protocol: ${parsed.protocol}`;\n }\n\n const host = parsed.hostname.toLowerCase();\n\n // Block localhost variants\n if (host === \"localhost\" || host === \"127.0.0.1\" || host === \"0.0.0.0\" ||\n host === \"::1\" || host === \"[::1]\") {\n return \"Blocked: localhost\";\n }\n\n // Block private/internal domains\n if (host.endsWith(\".internal\") || host.endsWith(\".local\") ||\n host === \"metadata.google.internal\") {\n return \"Blocked: internal/local domain\";\n }\n\n // Block private IP ranges\n if (/^10\\./.test(host) ||\n /^172\\.(1[6-9]|2\\d|3[01])\\./.test(host) ||\n /^192\\.168\\./.test(host) ||\n /^169\\.254\\./.test(host)) {\n return \"Blocked: private IP range\";\n }\n\n // Block IPv6 private\n if (host.startsWith(\"fd\") || host.startsWith(\"fc00:\") ||\n host.startsWith(\"fe80:\") || host.includes(\"::ffff:\")) {\n return \"Blocked: IPv6 private range\";\n }\n\n return null;\n}\n\n// ============================================================================\n// TEXT CLEANUP\n// ============================================================================\n\nfunction stripScriptsAndStyles(html: string): string {\n return html\n .replace(/<script[\\s\\S]*?<\\/script>/gi, \"\")\n .replace(/<style[\\s\\S]*?<\\/style>/gi, \"\")\n .replace(/<[^>]+>/g, \" \")\n .replace(/\\s+/g, \" \")\n .trim();\n}\n\nfunction truncate(text: string, maxChars: number): string {\n if (text.length <= maxChars) return text;\n return text.substring(0, maxChars) + \"\\n...[truncated]\";\n}\n\nfunction truncateResult(data: unknown): unknown {\n const json = JSON.stringify(data);\n if (json.length <= MAX_RESULT_BYTES) return data;\n // If too large, stringify and truncate\n return JSON.parse(truncate(json, MAX_RESULT_BYTES));\n}\n\n// ============================================================================\n// ACTION HANDLERS\n// ============================================================================\n\nasync function actionNavigate(url: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n return withPage(url, waitFor, async (page) => {\n const title = await page.title();\n const html = await page.content();\n const textContent = stripScriptsAndStyles(html);\n\n return {\n success: true,\n data: {\n url: page.url(),\n title,\n text: truncate(textContent, MAX_TEXT_CHARS),\n text_length: textContent.length,\n },\n };\n });\n}\n\nasync function actionScreenshot(url: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n return withPage(url, waitFor, async (page) => {\n const buffer = await page.screenshot({ type: \"png\", fullPage: false });\n const base64 = buffer.toString(\"base64\");\n const title = await page.title();\n\n return {\n success: true,\n data: {\n url: page.url(),\n title,\n screenshot_base64: base64,\n format: \"png\",\n size_bytes: buffer.length,\n },\n };\n });\n}\n\nasync function actionExtract(url: string, selector: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n if (!selector) {\n return { success: false, error: \"selector is required for extract action\" };\n }\n\n return withPage(url, waitFor, async (page) => {\n const elements = await page.$$eval(selector, (els) =>\n els.map((el) => ({\n tag: el.tagName.toLowerCase(),\n text: el.textContent?.trim() || \"\",\n html: el.innerHTML.substring(0, 2000),\n attributes: Object.fromEntries(\n Array.from(el.attributes).map((a) => [a.name, a.value])\n ),\n }))\n );\n\n return {\n success: true,\n data: {\n url: page.url(),\n selector,\n count: elements.length,\n elements: elements.slice(0, 50), // Cap at 50 elements\n },\n };\n });\n}\n\nasync function actionClick(url: string, selector: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n if (!selector) {\n return { success: false, error: \"selector is required for click action\" };\n }\n\n return withPage(url, waitFor, async (page) => {\n await page.click(selector, { timeout: PAGE_TIMEOUT_MS });\n\n // Wait a moment for any navigation or dynamic content\n await page.waitForTimeout(1000);\n\n const title = await page.title();\n const html = await page.content();\n const textContent = stripScriptsAndStyles(html);\n\n return {\n success: true,\n data: {\n url: page.url(),\n title,\n clicked: selector,\n text: truncate(textContent, MAX_TEXT_CHARS),\n },\n };\n });\n}\n\nasync function actionFill(url: string, selector: string, value: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n if (!selector) {\n return { success: false, error: \"selector is required for fill action\" };\n }\n if (value === undefined || value === null) {\n return { success: false, error: \"value is required for fill action\" };\n }\n\n return withPage(url, waitFor, async (page) => {\n await page.fill(selector, value, { timeout: PAGE_TIMEOUT_MS });\n\n return {\n success: true,\n data: {\n url: page.url(),\n filled: { selector, value },\n },\n };\n });\n}\n\nasync function actionEvaluate(url: string, script: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n if (!script) {\n return { success: false, error: \"script is required for evaluate action\" };\n }\n\n return withPage(url, waitFor, async (page) => {\n const result = await page.evaluate(script);\n\n return {\n success: true,\n data: {\n url: page.url(),\n result: truncateResult(result),\n },\n };\n });\n}\n\nasync function actionPdf(url: string, waitFor?: string): Promise<{ success: boolean; data?: unknown; error?: string }> {\n return withPage(url, waitFor, async (page) => {\n const buffer = await page.pdf({\n format: \"A4\",\n printBackground: true,\n margin: { top: \"1cm\", bottom: \"1cm\", left: \"1cm\", right: \"1cm\" },\n });\n\n const base64 = buffer.toString(\"base64\");\n const title = await page.title();\n\n return {\n success: true,\n data: {\n url: page.url(),\n title,\n pdf_base64: base64,\n format: \"A4\",\n size_bytes: buffer.length,\n },\n };\n });\n}\n\n// ============================================================================\n// CAPTCHA DETECTION & SOLVING (via CapMonster Cloud)\n// ============================================================================\n\ninterface CaptchaResult {\n solved: boolean;\n type?: string;\n}\n\nasync function callCapMonster(taskType: string, params: Record<string, unknown>): Promise<string> {\n const apiKey = process.env.CAPMONSTER_API_KEY;\n if (!apiKey) throw new Error(\"CAPMONSTER_API_KEY not configured\");\n\n const createResp = await fetch(\"https://api.capmonster.cloud/createTask\", {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ clientKey: apiKey, task: { type: taskType, ...params } }),\n });\n const createData = await createResp.json() as { errorId: number; errorDescription?: string; taskId?: number };\n if (createData.errorId) throw new Error(`CapMonster create: ${createData.errorDescription}`);\n const taskId = createData.taskId;\n\n // Poll for result (max 120s, 2s intervals)\n for (let i = 0; i < 60; i++) {\n await new Promise(r => setTimeout(r, 2000));\n const resultResp = await fetch(\"https://api.capmonster.cloud/getTaskResult\", {\n method: \"POST\",\n headers: { \"Content-Type\": \"application/json\" },\n body: JSON.stringify({ clientKey: apiKey, taskId }),\n });\n const resultData = await resultResp.json() as {\n status: string;\n errorId: number;\n errorDescription?: string;\n solution?: { gRecaptchaResponse?: string; token?: string; cf_clearance?: string };\n };\n if (resultData.status === \"ready\") {\n return resultData.solution?.gRecaptchaResponse || resultData.solution?.token || resultData.solution?.cf_clearance || \"\";\n }\n if (resultData.errorId) throw new Error(`CapMonster result: ${resultData.errorDescription}`);\n }\n throw new Error(\"CapMonster: timeout waiting for solution (120s)\");\n}\n\nasync function detectAndSolveCaptcha(page: Page): Promise<CaptchaResult> {\n try {\n // Skip if no API key configured\n if (!process.env.CAPMONSTER_API_KEY) return { solved: false };\n\n const pageUrl = page.url();\n\n // Detect reCAPTCHA v2\n const recaptchaKey = await page.evaluate(() => {\n const el = document.querySelector('.g-recaptcha[data-sitekey], [data-sitekey]');\n if (el && el.classList.contains('g-recaptcha')) return el.getAttribute('data-sitekey');\n // Also check for reCAPTCHA iframe\n const iframe = document.querySelector('iframe[src*=\"recaptcha\"]');\n if (iframe) {\n const src = iframe.getAttribute('src') || '';\n const match = src.match(/[?&]k=([^&]+)/);\n return match ? match[1] : null;\n }\n return null;\n }).catch(() => null);\n\n if (recaptchaKey) {\n console.log(`[browser] Detected reCAPTCHA v2 on ${pageUrl}`);\n const token = await callCapMonster(\"RecaptchaV2TaskProxyless\", {\n websiteURL: pageUrl,\n websiteKey: recaptchaKey,\n });\n await page.evaluate((t: string) => {\n const textarea = document.getElementById('g-recaptcha-response') as HTMLTextAreaElement | null;\n if (textarea) { textarea.style.display = 'block'; textarea.value = t; }\n // Try triggering the callback\n const win = window as unknown as Record<string, unknown>;\n try {\n const cfg = win.___grecaptcha_cfg as Record<string, unknown> | undefined;\n if (cfg?.clients) {\n const clients = cfg.clients as Record<string, unknown>;\n const firstKey = Object.keys(clients)[0];\n const first = firstKey ? clients[firstKey] : null;\n if (first) {\n const findCallback = (obj: unknown, depth = 0): ((t: string) => void) | null => {\n if (depth > 4 || !obj || typeof obj !== 'object') return null;\n for (const val of Object.values(obj as Record<string, unknown>)) {\n if (typeof val === 'function') return val as (t: string) => void;\n const found = findCallback(val, depth + 1);\n if (found) return found;\n }\n return null;\n };\n const cb = findCallback(first);\n if (cb) cb(t);\n }\n }\n } catch { /* callback search failed, form submit will work */ }\n }, token);\n return { solved: true, type: \"reCAPTCHA v2\" };\n }\n\n // Detect hCaptcha\n const hcaptchaKey = await page.evaluate(() => {\n const el = document.querySelector('.h-captcha[data-sitekey]');\n return el?.getAttribute('data-sitekey') || null;\n }).catch(() => null);\n\n if (hcaptchaKey) {\n console.log(`[browser] Detected hCaptcha on ${pageUrl}`);\n const token = await callCapMonster(\"HCaptchaTaskProxyless\", {\n websiteURL: pageUrl,\n websiteKey: hcaptchaKey,\n });\n await page.evaluate((t: string) => {\n const textarea = document.querySelector('[name=\"h-captcha-response\"], [name=\"g-recaptcha-response\"]') as HTMLTextAreaElement | null;\n if (textarea) { textarea.value = t; }\n const win = window as unknown as Record<string, unknown>;\n if (typeof win.hcaptchaCallback === 'function') {\n (win.hcaptchaCallback as (t: string) => void)(t);\n }\n }, token);\n return { solved: true, type: \"hCaptcha\" };\n }\n\n // Detect Cloudflare Turnstile\n const turnstileKey = await page.evaluate(() => {\n const el = document.querySelector('.cf-turnstile[data-sitekey]');\n return el?.getAttribute('data-sitekey') || null;\n }).catch(() => null);\n\n if (turnstileKey) {\n console.log(`[browser] Detected Turnstile on ${pageUrl}`);\n const token = await callCapMonster(\"TurnstileTaskProxyless\", {\n websiteURL: pageUrl,\n websiteKey: turnstileKey,\n });\n await page.evaluate((t: string) => {\n const input = document.querySelector('[name=\"cf-turnstile-response\"]') as HTMLInputElement | null;\n if (input) input.value = t;\n const win = window as unknown as Record<string, unknown>;\n if (typeof win.turnstileCallback === 'function') {\n (win.turnstileCallback as (t: string) => void)(t);\n }\n }, token);\n return { solved: true, type: \"Turnstile\" };\n }\n\n return { solved: false };\n } catch (err) {\n console.log(`[browser] CAPTCHA detection/solving failed: ${err instanceof Error ? err.message : String(err)}`);\n return { solved: false };\n }\n}\n\n// ============================================================================\n// MAIN HANDLER\n// ============================================================================\n\nexport async function handleBrowser(\n _sb: SupabaseClient,\n args: Record<string, unknown>,\n _storeId?: string\n): Promise<{ success: boolean; data?: unknown; error?: string }> {\n const action = args.action as string;\n const url = args.url as string;\n const selector = args.selector as string | undefined;\n const value = args.value as string | undefined;\n const script = args.script as string | undefined;\n const waitFor = args.wait_for as string | undefined;\n\n if (!action) {\n return { success: false, error: \"action is required\" };\n }\n if (!url) {\n return { success: false, error: \"url is required\" };\n }\n\n // SSRF check — fast synchronous pre-check, then DNS-based validation\n const blocked = isBlockedUrl(url);\n if (blocked) {\n return { success: false, error: blocked };\n }\n try {\n const dnsBlocked = await validateUrl(url);\n if (dnsBlocked) {\n return { success: false, error: dnsBlocked };\n }\n } catch {\n return { success: false, error: \"URL validation failed\" };\n }\n\n try {\n switch (action) {\n case \"navigate\":\n return await actionNavigate(url, waitFor);\n case \"screenshot\":\n return await actionScreenshot(url, waitFor);\n case \"extract\":\n return await actionExtract(url, selector!, waitFor);\n case \"click\":\n return await actionClick(url, selector!, waitFor);\n case \"fill\":\n return await actionFill(url, selector!, value!, waitFor);\n case \"evaluate\":\n return await actionEvaluate(url, script!, waitFor);\n case \"pdf\":\n return await actionPdf(url, waitFor);\n default:\n return { success: false, error: `Unknown browser action: ${action}. Valid: navigate, screenshot, extract, click, fill, evaluate, pdf` };\n }\n } catch (err: unknown) {\n const message = err instanceof Error ? err.message : String(err);\n // Clean up common Playwright error noise\n const cleanMessage = message\n .replace(/=========================== logs ===========================[^]*?============================================================/gs, \"\")\n .trim();\n return { success: false, error: `Browser error: ${cleanMessage.substring(0, 500)}` };\n }\n}\n"],"mappings":"AAAA;AACA;;AAGA,SAASA,QAAQ,QAAiC,iBAAiB;AACnE,SAASC,WAAW,QAAQ,sBAAsB;;AAElD;AACA;AACA;;AAEA,IAAIC,eAA+B,GAAG,IAAI;AAC1C,IAAIC,SAA+C,GAAG,IAAI;AAC1D,MAAMC,eAAe,GAAG,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC;AACvC,MAAMC,eAAe,GAAG,MAAM,CAAC,CAAC;AAChC,MAAMC,gBAAgB,GAAG,GAAG,GAAG,IAAI,CAAC,CAAC;AACrC,MAAMC,cAAc,GAAG,OAAO,CAAC,CAAC;;AAEhC,SAASC,cAAcA,CAAA,EAAS;EAC9B,IAAIL,SAAS,EAAEM,YAAY,CAACN,SAAS,CAAC;EACtCA,SAAS,GAAGO,UAAU,CAAC,YAAY;IACjC,IAAIR,eAAe,EAAE;MACnB,IAAI;QACF,MAAMA,eAAe,CAACS,KAAK,CAAC,CAAC;MAC/B,CAAC,CAAC,MAAM,CAAE;MACVT,eAAe,GAAG,IAAI;MACtBU,OAAO,CAACC,GAAG,CAAC,qCAAqC,CAAC;IACpD;EACF,CAAC,EAAET,eAAe,CAAC;AACrB;AAEA,eAAeU,UAAUA,CAAA,EAAqB;EAC5C,IAAIZ,eAAe,IAAIA,eAAe,CAACa,WAAW,CAAC,CAAC,EAAE;IACpDP,cAAc,CAAC,CAAC;IAChB,OAAON,eAAe;EACxB;EAEAA,eAAe,GAAG,MAAMF,QAAQ,CAACgB,MAAM,CAAC;IACtCC,QAAQ,EAAE,IAAI;IACdC,cAAc,EAAEC,OAAO,CAACC,GAAG,CAACC,mCAAmC,IAAIC,SAAS;IAC5EC,IAAI,EAAE,CACJ,cAAc,EACd,0BAA0B,EAC1B,yBAAyB,EACzB,eAAe;EAEnB,CAAC,CAAC;EAEFrB,eAAe,CAACsB,EAAE,CAAC,cAAc,EAAE,MAAM;IACvCtB,eAAe,GAAG,IAAI;IACtB,IAAIC,SAAS,EAAE;MACbM,YAAY,CAACN,SAAS,CAAC;MACvBA,SAAS,GAAG,IAAI;IAClB;EACF,CAAC,CAAC;EAEFK,cAAc,CAAC,CAAC;EAChBI,OAAO,CAACC,GAAG,CAAC,iCAAiC,CAAC;EAC9C,OAAOX,eAAe;AACxB;AAEA,eAAeuB,QAAQA,CAAIC,GAAW,EAAEC,OAA2B,EAAEC,EAA8B,EAAc;EAC/G,MAAMC,OAAO,GAAG,MAAMf,UAAU,CAAC,CAAC;EAClC,MAAMgB,OAAO,GAAG,MAAMD,OAAO,CAACE,UAAU,CAAC;IACvCC,SAAS,EAAE,kCAAkC;IAC7CC,QAAQ,EAAE;MAAEC,KAAK,EAAE,IAAI;MAAEC,MAAM,EAAE;IAAI,CAAC;IACtCC,iBAAiB,EAAE;EACrB,CAAC,CAAC;EACFN,OAAO,CAACO,iBAAiB,CAAChC,eAAe,CAAC;EAE1C,MAAMiC,IAAI,GAAG,MAAMR,OAAO,CAACS,OAAO,CAAC,CAAC;EACpC,IAAI;IACF,MAAMD,IAAI,CAACE,IAAI,CAACd,GAAG,EAAE;MAAEe,SAAS,EAAE,kBAAkB;MAAEC,OAAO,EAAErC;IAAgB,CAAC,CAAC;;IAEjF;IACA,MAAMsC,aAAa,GAAG,MAAMC,qBAAqB,CAACN,IAAI,CAAC;IACvD,IAAIK,aAAa,CAACE,MAAM,EAAE;MACxBjC,OAAO,CAACC,GAAG,CAAC,oBAAoB8B,aAAa,CAACG,IAAI,eAAepB,GAAG,EAAE,CAAC;MACvE;MACA,MAAMY,IAAI,CAACS,cAAc,CAAC,IAAI,CAAC;IACjC;IAEA,IAAIpB,OAAO,EAAE;MACX,MAAMW,IAAI,CAACU,eAAe,CAACrB,OAAO,EAAE;QAAEe,OAAO,EAAErC;MAAgB,CAAC,CAAC;IACnE;IAEA,OAAO,MAAMuB,EAAE,CAACU,IAAI,CAAC;EACvB,CAAC,SAAS;IACR,MAAMA,IAAI,CAAC3B,KAAK,CAAC,CAAC,CAACsC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC;IAClC,MAAMnB,OAAO,CAACnB,KAAK,CAAC,CAAC,CAACsC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC;EACvC;AACF;;AAEA;AACA;AACA;;AAEA,SAASC,YAAYA,CAACxB,GAAW,EAAiB;EAChD,IAAIyB,MAAW;EACf,IAAI;IACFA,MAAM,GAAG,IAAIC,GAAG,CAAC1B,GAAG,CAAC;EACvB,CAAC,CAAC,MAAM;IACN,OAAO,aAAa;EACtB;EAEA,IAAIyB,MAAM,CAACE,QAAQ,KAAK,OAAO,IAAIF,MAAM,CAACE,QAAQ,KAAK,QAAQ,EAAE;IAC/D,OAAO,qBAAqBF,MAAM,CAACE,QAAQ,EAAE;EAC/C;EAEA,MAAMC,IAAI,GAAGH,MAAM,CAACI,QAAQ,CAACC,WAAW,CAAC,CAAC;;EAE1C;EACA,IAAIF,IAAI,KAAK,WAAW,IAAIA,IAAI,KAAK,WAAW,IAAIA,IAAI,KAAK,SAAS,IAClEA,IAAI,KAAK,KAAK,IAAIA,IAAI,KAAK,OAAO,EAAE;IACtC,OAAO,oBAAoB;EAC7B;;EAEA;EACA,IAAIA,IAAI,CAACG,QAAQ,CAAC,WAAW,CAAC,IAAIH,IAAI,CAACG,QAAQ,CAAC,QAAQ,CAAC,IACrDH,IAAI,KAAK,0BAA0B,EAAE;IACvC,OAAO,gCAAgC;EACzC;;EAEA;EACA,IAAI,OAAO,CAACI,IAAI,CAACJ,IAAI,CAAC,IAClB,4BAA4B,CAACI,IAAI,CAACJ,IAAI,CAAC,IACvC,aAAa,CAACI,IAAI,CAACJ,IAAI,CAAC,IACxB,aAAa,CAACI,IAAI,CAACJ,IAAI,CAAC,EAAE;IAC5B,OAAO,2BAA2B;EACpC;;EAEA;EACA,IAAIA,IAAI,CAACK,UAAU,CAAC,IAAI,CAAC,IAAIL,IAAI,CAACK,UAAU,CAAC,OAAO,CAAC,IACjDL,IAAI,CAACK,UAAU,CAAC,OAAO,CAAC,IAAIL,IAAI,CAACM,QAAQ,CAAC,SAAS,CAAC,EAAE;IACxD,OAAO,6BAA6B;EACtC;EAEA,OAAO,IAAI;AACb;;AAEA;AACA;AACA;;AAEA,SAASC,qBAAqBA,CAACC,IAAY,EAAU;EACnD,OAAOA,IAAI,CACRC,OAAO,CAAC,6BAA6B,EAAE,EAAE,CAAC,CAC1CA,OAAO,CAAC,2BAA2B,EAAE,EAAE,CAAC,CACxCA,OAAO,CAAC,UAAU,EAAE,GAAG,CAAC,CACxBA,OAAO,CAAC,MAAM,EAAE,GAAG,CAAC,CACpBC,IAAI,CAAC,CAAC;AACX;AAEA,SAASC,QAAQA,CAACC,IAAY,EAAEC,QAAgB,EAAU;EACxD,IAAID,IAAI,CAACE,MAAM,IAAID,QAAQ,EAAE,OAAOD,IAAI;EACxC,OAAOA,IAAI,CAACG,SAAS,CAAC,CAAC,EAAEF,QAAQ,CAAC,GAAG,kBAAkB;AACzD;AAEA,SAASG,cAAcA,CAACC,IAAa,EAAW;EAC9C,MAAMC,IAAI,GAAGC,IAAI,CAACC,SAAS,CAACH,IAAI,CAAC;EACjC,IAAIC,IAAI,CAACJ,MAAM,IAAI9D,gBAAgB,EAAE,OAAOiE,IAAI;EAChD;EACA,OAAOE,IAAI,CAACE,KAAK,CAACV,QAAQ,CAACO,IAAI,EAAElE,gBAAgB,CAAC,CAAC;AACrD;;AAEA;AACA;AACA;;AAEA,eAAesE,cAAcA,CAAClD,GAAW,EAAEC,OAAgB,EAAiE;EAC1H,OAAOF,QAAQ,CAACC,GAAG,EAAEC,OAAO,EAAE,MAAOW,IAAI,IAAK;IAC5C,MAAMuC,KAAK,GAAG,MAAMvC,IAAI,CAACuC,KAAK,CAAC,CAAC;IAChC,MAAMf,IAAI,GAAG,MAAMxB,IAAI,CAACwC,OAAO,CAAC,CAAC;IACjC,MAAMC,WAAW,GAAGlB,qBAAqB,CAACC,IAAI,CAAC;IAE/C,OAAO;MACLkB,OAAO,EAAE,IAAI;MACbT,IAAI,EAAE;QACJ7C,GAAG,EAAEY,IAAI,CAACZ,GAAG,CAAC,CAAC;QACfmD,KAAK;QACLX,IAAI,EAAED,QAAQ,CAACc,WAAW,EAAExE,cAAc,CAAC;QAC3C0E,WAAW,EAAEF,WAAW,CAACX;MAC3B;IACF,CAAC;EACH,CAAC,CAAC;AACJ;AAEA,eAAec,gBAAgBA,CAACxD,GAAW,EAAEC,OAAgB,EAAiE;EAC5H,OAAOF,QAAQ,CAACC,GAAG,EAAEC,OAAO,EAAE,MAAOW,IAAI,IAAK;IAC5C,MAAM6C,MAAM,GAAG,MAAM7C,IAAI,CAAC8C,UAAU,CAAC;MAAEtC,IAAI,EAAE,KAAK;MAAEuC,QAAQ,EAAE;IAAM,CAAC,CAAC;IACtE,MAAMC,MAAM,GAAGH,MAAM,CAACI,QAAQ,CAAC,QAAQ,CAAC;IACxC,MAAMV,KAAK,GAAG,MAAMvC,IAAI,CAACuC,KAAK,CAAC,CAAC;IAEhC,OAAO;MACLG,OAAO,EAAE,IAAI;MACbT,IAAI,EAAE;QACJ7C,GAAG,EAAEY,IAAI,CAACZ,GAAG,CAAC,CAAC;QACfmD,KAAK;QACLW,iBAAiB,EAAEF,MAAM;QACzBG,MAAM,EAAE,KAAK;QACbC,UAAU,EAAEP,MAAM,CAACf;MACrB;IACF,CAAC;EACH,CAAC,CAAC;AACJ;AAEA,eAAeuB,aAAaA,CAACjE,GAAW,EAAEkE,QAAgB,EAAEjE,OAAgB,EAAiE;EAC3I,IAAI,CAACiE,QAAQ,EAAE;IACb,OAAO;MAAEZ,OAAO,EAAE,KAAK;MAAEa,KAAK,EAAE;IAA0C,CAAC;EAC7E;EAEA,OAAOpE,QAAQ,CAACC,GAAG,EAAEC,OAAO,EAAE,MAAOW,IAAI,IAAK;IAC5C,MAAMwD,QAAQ,GAAG,MAAMxD,IAAI,CAACyD,MAAM,CAACH,QAAQ,EAAGI,GAAG,IAC/CA,GAAG,CAACC,GAAG,CAAEC,EAAE,KAAM;MACfC,GAAG,EAAED,EAAE,CAACE,OAAO,CAAC5C,WAAW,CAAC,CAAC;MAC7BU,IAAI,EAAEgC,EAAE,CAACnB,WAAW,EAAEf,IAAI,CAAC,CAAC,IAAI,EAAE;MAClCF,IAAI,EAAEoC,EAAE,CAACG,SAAS,CAAChC,SAAS,CAAC,CAAC,EAAE,IAAI,CAAC;MACrCiC,UAAU,EAAEC,MAAM,CAACC,WAAW,CAC5BC,KAAK,CAACC,IAAI,CAACR,EAAE,CAACI,UAAU,CAAC,CAACL,GAAG,CAAEU,CAAC,IAAK,CAACA,CAAC,CAACC,IAAI,EAAED,CAAC,CAACE,KAAK,CAAC,CACxD;IACF,CAAC,CAAC,CACJ,CAAC;IAED,OAAO;MACL7B,OAAO,EAAE,IAAI;MACbT,IAAI,EAAE;QACJ7C,GAAG,EAAEY,IAAI,CAACZ,GAAG,CAAC,CAAC;QACfkE,QAAQ;QACRkB,KAAK,EAAEhB,QAAQ,CAAC1B,MAAM;QACtB0B,QAAQ,EAAEA,QAAQ,CAACiB,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAE;MACnC;IACF,CAAC;EACH,CAAC,CAAC;AACJ;AAEA,eAAeC,WAAWA,CAACtF,GAAW,EAAEkE,QAAgB,EAAEjE,OAAgB,EAAiE;EACzI,IAAI,CAACiE,QAAQ,EAAE;IACb,OAAO;MAAEZ,OAAO,EAAE,KAAK;MAAEa,KAAK,EAAE;IAAwC,CAAC;EAC3E;EAEA,OAAOpE,QAAQ,CAACC,GAAG,EAAEC,OAAO,EAAE,MAAOW,IAAI,IAAK;IAC5C,MAAMA,IAAI,CAAC2E,KAAK,CAACrB,QAAQ,EAAE;MAAElD,OAAO,EAAErC;IAAgB,CAAC,CAAC;;IAExD;IACA,MAAMiC,IAAI,CAACS,cAAc,CAAC,IAAI,CAAC;IAE/B,MAAM8B,KAAK,GAAG,MAAMvC,IAAI,CAACuC,KAAK,CAAC,CAAC;IAChC,MAAMf,IAAI,GAAG,MAAMxB,IAAI,CAACwC,OAAO,CAAC,CAAC;IACjC,MAAMC,WAAW,GAAGlB,qBAAqB,CAACC,IAAI,CAAC;IAE/C,OAAO;MACLkB,OAAO,EAAE,IAAI;MACbT,IAAI,EAAE;QACJ7C,GAAG,EAAEY,IAAI,CAACZ,GAAG,CAAC,CAAC;QACfmD,KAAK;QACLqC,OAAO,EAAEtB,QAAQ;QACjB1B,IAAI,EAAED,QAAQ,CAACc,WAAW,EAAExE,cAAc;MAC5C;IACF,CAAC;EACH,CAAC,CAAC;AACJ;AAEA,eAAe4G,UAAUA,CAACzF,GAAW,EAAEkE,QAAgB,EAAEiB,KAAa,EAAElF,OAAgB,EAAiE;EACvJ,IAAI,CAACiE,QAAQ,EAAE;IACb,OAAO;MAAEZ,OAAO,EAAE,KAAK;MAAEa,KAAK,EAAE;IAAuC,CAAC;EAC1E;EACA,IAAIgB,KAAK,KAAKvF,SAAS,IAAIuF,KAAK,KAAK,IAAI,EAAE;IACzC,OAAO;MAAE7B,OAAO,EAAE,KAAK;MAAEa,KAAK,EAAE;IAAoC,CAAC;EACvE;EAEA,OAAOpE,QAAQ,CAACC,GAAG,EAAEC,OAAO,EAAE,MAAOW,IAAI,IAAK;IAC5C,MAAMA,IAAI,CAAC8E,IAAI,CAACxB,QAAQ,EAAEiB,KAAK,EAAE;MAAEnE,OAAO,EAAErC;IAAgB,CAAC,CAAC;IAE9D,OAAO;MACL2E,OAAO,EAAE,IAAI;MACbT,IAAI,EAAE;QACJ7C,GAAG,EAAEY,IAAI,CAACZ,GAAG,CAAC,CAAC;QACf2F,MAAM,EAAE;UAAEzB,QAAQ;UAAEiB;QAAM;MAC5B;IACF,CAAC;EACH,CAAC,CAAC;AACJ;AAEA,eAAeS,cAAcA,CAAC5F,GAAW,EAAE6F,MAAc,EAAE5F,OAAgB,EAAiE;EAC1I,IAAI,CAAC4F,MAAM,EAAE;IACX,OAAO;MAAEvC,OAAO,EAAE,KAAK;MAAEa,KAAK,EAAE;IAAyC,CAAC;EAC5E;EAEA,OAAOpE,QAAQ,CAACC,GAAG,EAAEC,OAAO,EAAE,MAAOW,IAAI,IAAK;IAC5C,MAAMkF,MAAM,GAAG,MAAMlF,IAAI,CAACmF,QAAQ,CAACF,MAAM,CAAC;IAE1C,OAAO;MACLvC,OAAO,EAAE,IAAI;MACbT,IAAI,EAAE;QACJ7C,GAAG,EAAEY,IAAI,CAACZ,GAAG,CAAC,CAAC;QACf8F,MAAM,EAAElD,cAAc,CAACkD,MAAM;MAC/B;IACF,CAAC;EACH,CAAC,CAAC;AACJ;AAEA,eAAeE,SAASA,CAAChG,GAAW,EAAEC,OAAgB,EAAiE;EACrH,OAAOF,QAAQ,CAACC,GAAG,EAAEC,OAAO,EAAE,MAAOW,IAAI,IAAK;IAC5C,MAAM6C,MAAM,GAAG,MAAM7C,IAAI,CAACqF,GAAG,CAAC;MAC5BlC,MAAM,EAAE,IAAI;MACZmC,eAAe,EAAE,IAAI;MACrBC,MAAM,EAAE;QAAEC,GAAG,EAAE,KAAK;QAAEC,MAAM,EAAE,KAAK;QAAEC,IAAI,EAAE,KAAK;QAAEC,KAAK,EAAE;MAAM;IACjE,CAAC,CAAC;IAEF,MAAM3C,MAAM,GAAGH,MAAM,CAACI,QAAQ,CAAC,QAAQ,CAAC;IACxC,MAAMV,KAAK,GAAG,MAAMvC,IAAI,CAACuC,KAAK,CAAC,CAAC;IAEhC,OAAO;MACLG,OAAO,EAAE,IAAI;MACbT,IAAI,EAAE;QACJ7C,GAAG,EAAEY,IAAI,CAACZ,GAAG,CAAC,CAAC;QACfmD,KAAK;QACLqD,UAAU,EAAE5C,MAAM;QAClBG,MAAM,EAAE,IAAI;QACZC,UAAU,EAAEP,MAAM,CAACf;MACrB;IACF,CAAC;EACH,CAAC,CAAC;AACJ;;AAEA;AACA;AACA;;AAOA,eAAe+D,cAAcA,CAACC,QAAgB,EAAEC,MAA+B,EAAmB;EAChG,MAAMC,MAAM,GAAGnH,OAAO,CAACC,GAAG,CAACmH,kBAAkB;EAC7C,IAAI,CAACD,MAAM,EAAE,MAAM,IAAIE,KAAK,CAAC,mCAAmC,CAAC;EAEjE,MAAMC,UAAU,GAAG,MAAMC,KAAK,CAAC,yCAAyC,EAAE;IACxEC,MAAM,EAAE,MAAM;IACdC,OAAO,EAAE;MAAE,cAAc,EAAE;IAAmB,CAAC;IAC/CC,IAAI,EAAEpE,IAAI,CAACC,SAAS,CAAC;MAAEoE,SAAS,EAAER,MAAM;MAAES,IAAI,EAAE;QAAEjG,IAAI,EAAEsF,QAAQ;QAAE,GAAGC;MAAO;IAAE,CAAC;EACjF,CAAC,CAAC;EACF,MAAMW,UAAU,GAAG,MAAMP,UAAU,CAACjE,IAAI,CAAC,CAAoE;EAC7G,IAAIwE,UAAU,CAACC,OAAO,EAAE,MAAM,IAAIT,KAAK,CAAC,sBAAsBQ,UAAU,CAACE,gBAAgB,EAAE,CAAC;EAC5F,MAAMC,MAAM,GAAGH,UAAU,CAACG,MAAM;;EAEhC;EACA,KAAK,IAAIC,CAAC,GAAG,CAAC,EAAEA,CAAC,GAAG,EAAE,EAAEA,CAAC,EAAE,EAAE;IAC3B,MAAM,IAAIC,OAAO,CAACC,CAAC,IAAI5I,UAAU,CAAC4I,CAAC,EAAE,IAAI,CAAC,CAAC;IAC3C,MAAMC,UAAU,GAAG,MAAMb,KAAK,CAAC,4CAA4C,EAAE;MAC3EC,MAAM,EAAE,MAAM;MACdC,OAAO,EAAE;QAAE,cAAc,EAAE;MAAmB,CAAC;MAC/CC,IAAI,EAAEpE,IAAI,CAACC,SAAS,CAAC;QAAEoE,SAAS,EAAER,MAAM;QAAEa;MAAO,CAAC;IACpD,CAAC,CAAC;IACF,MAAMK,UAAU,GAAG,MAAMD,UAAU,CAAC/E,IAAI,CAAC,CAKxC;IACD,IAAIgF,UAAU,CAACC,MAAM,KAAK,OAAO,EAAE;MACjC,OAAOD,UAAU,CAACE,QAAQ,EAAEC,kBAAkB,IAAIH,UAAU,CAACE,QAAQ,EAAEE,KAAK,IAAIJ,UAAU,CAACE,QAAQ,EAAEG,YAAY,IAAI,EAAE;IACzH;IACA,IAAIL,UAAU,CAACP,OAAO,EAAE,MAAM,IAAIT,KAAK,CAAC,sBAAsBgB,UAAU,CAACN,gBAAgB,EAAE,CAAC;EAC9F;EACA,MAAM,IAAIV,KAAK,CAAC,iDAAiD,CAAC;AACpE;AAEA,eAAe5F,qBAAqBA,CAACN,IAAU,EAA0B;EACvE,IAAI;IACF;IACA,IAAI,CAACnB,OAAO,CAACC,GAAG,CAACmH,kBAAkB,EAAE,OAAO;MAAE1F,MAAM,EAAE;IAAM,CAAC;IAE7D,MAAMiH,OAAO,GAAGxH,IAAI,CAACZ,GAAG,CAAC,CAAC;;IAE1B;IACA,MAAMqI,YAAY,GAAG,MAAMzH,IAAI,CAACmF,QAAQ,CAAC,MAAM;MAC7C,MAAMvB,EAAE,GAAG8D,QAAQ,CAACC,aAAa,CAAC,4CAA4C,CAAC;MAC/E,IAAI/D,EAAE,IAAIA,EAAE,CAACgE,SAAS,CAACC,QAAQ,CAAC,aAAa,CAAC,EAAE,OAAOjE,EAAE,CAACkE,YAAY,CAAC,cAAc,CAAC;MACtF;MACA,MAAMC,MAAM,GAAGL,QAAQ,CAACC,aAAa,CAAC,0BAA0B,CAAC;MACjE,IAAII,MAAM,EAAE;QACV,MAAMC,GAAG,GAAGD,MAAM,CAACD,YAAY,CAAC,KAAK,CAAC,IAAI,EAAE;QAC5C,MAAMG,KAAK,GAAGD,GAAG,CAACC,KAAK,CAAC,eAAe,CAAC;QACxC,OAAOA,KAAK,GAAGA,KAAK,CAAC,CAAC,CAAC,GAAG,IAAI;MAChC;MACA,OAAO,IAAI;IACb,CAAC,CAAC,CAACtH,KAAK,CAAC,MAAM,IAAI,CAAC;IAEpB,IAAI8G,YAAY,EAAE;MAChBnJ,OAAO,CAACC,GAAG,CAAC,sCAAsCiJ,OAAO,EAAE,CAAC;MAC5D,MAAMF,KAAK,GAAG,MAAMzB,cAAc,CAAC,0BAA0B,EAAE;QAC7DqC,UAAU,EAAEV,OAAO;QACnBW,UAAU,EAAEV;MACd,CAAC,CAAC;MACF,MAAMzH,IAAI,CAACmF,QAAQ,CAAEiD,CAAS,IAAK;QACjC,MAAMC,QAAQ,GAAGX,QAAQ,CAACY,cAAc,CAAC,sBAAsB,CAA+B;QAC9F,IAAID,QAAQ,EAAE;UAAEA,QAAQ,CAACE,KAAK,CAACC,OAAO,GAAG,OAAO;UAAEH,QAAQ,CAAC9D,KAAK,GAAG6D,CAAC;QAAE;QACtE;QACA,MAAMK,GAAG,GAAGC,MAA4C;QACxD,IAAI;UACF,MAAMC,GAAG,GAAGF,GAAG,CAACG,iBAAwD;UACxE,IAAID,GAAG,EAAEE,OAAO,EAAE;YAChB,MAAMA,OAAO,GAAGF,GAAG,CAACE,OAAkC;YACtD,MAAMC,QAAQ,GAAG7E,MAAM,CAAC8E,IAAI,CAACF,OAAO,CAAC,CAAC,CAAC,CAAC;YACxC,MAAMG,KAAK,GAAGF,QAAQ,GAAGD,OAAO,CAACC,QAAQ,CAAC,GAAG,IAAI;YACjD,IAAIE,KAAK,EAAE;cACT,MAAMC,YAAY,GAAGA,CAACC,GAAY,EAAEC,KAAK,GAAG,CAAC,KAAmC;gBAC9E,IAAIA,KAAK,GAAG,CAAC,IAAI,CAACD,GAAG,IAAI,OAAOA,GAAG,KAAK,QAAQ,EAAE,OAAO,IAAI;gBAC7D,KAAK,MAAME,GAAG,IAAInF,MAAM,CAACoF,MAAM,CAACH,GAA8B,CAAC,EAAE;kBAC/D,IAAI,OAAOE,GAAG,KAAK,UAAU,EAAE,OAAOA,GAAG;kBACzC,MAAME,KAAK,GAAGL,YAAY,CAACG,GAAG,EAAED,KAAK,GAAG,CAAC,CAAC;kBAC1C,IAAIG,KAAK,EAAE,OAAOA,KAAK;gBACzB;gBACA,OAAO,IAAI;cACb,CAAC;cACD,MAAMC,EAAE,GAAGN,YAAY,CAACD,KAAK,CAAC;cAC9B,IAAIO,EAAE,EAAEA,EAAE,CAACnB,CAAC,CAAC;YACf;UACF;QACF,CAAC,CAAC,MAAM,CAAE;MACZ,CAAC,EAAEd,KAAK,CAAC;MACT,OAAO;QAAE/G,MAAM,EAAE,IAAI;QAAEC,IAAI,EAAE;MAAe,CAAC;IAC/C;;IAEA;IACA,MAAMgJ,WAAW,GAAG,MAAMxJ,IAAI,CAACmF,QAAQ,CAAC,MAAM;MAC5C,MAAMvB,EAAE,GAAG8D,QAAQ,CAACC,aAAa,CAAC,0BAA0B,CAAC;MAC7D,OAAO/D,EAAE,EAAEkE,YAAY,CAAC,cAAc,CAAC,IAAI,IAAI;IACjD,CAAC,CAAC,CAACnH,KAAK,CAAC,MAAM,IAAI,CAAC;IAEpB,IAAI6I,WAAW,EAAE;MACflL,OAAO,CAACC,GAAG,CAAC,kCAAkCiJ,OAAO,EAAE,CAAC;MACxD,MAAMF,KAAK,GAAG,MAAMzB,cAAc,CAAC,uBAAuB,EAAE;QAC1DqC,UAAU,EAAEV,OAAO;QACnBW,UAAU,EAAEqB;MACd,CAAC,CAAC;MACF,MAAMxJ,IAAI,CAACmF,QAAQ,CAAEiD,CAAS,IAAK;QACjC,MAAMC,QAAQ,GAAGX,QAAQ,CAACC,aAAa,CAAC,4DAA4D,CAA+B;QACnI,IAAIU,QAAQ,EAAE;UAAEA,QAAQ,CAAC9D,KAAK,GAAG6D,CAAC;QAAE;QACpC,MAAMK,GAAG,GAAGC,MAA4C;QACxD,IAAI,OAAOD,GAAG,CAACgB,gBAAgB,KAAK,UAAU,EAAE;UAC7ChB,GAAG,CAACgB,gBAAgB,CAAyBrB,CAAC,CAAC;QAClD;MACF,CAAC,EAAEd,KAAK,CAAC;MACT,OAAO;QAAE/G,MAAM,EAAE,IAAI;QAAEC,IAAI,EAAE;MAAW,CAAC;IAC3C;;IAEA;IACA,MAAMkJ,YAAY,GAAG,MAAM1J,IAAI,CAACmF,QAAQ,CAAC,MAAM;MAC7C,MAAMvB,EAAE,GAAG8D,QAAQ,CAACC,aAAa,CAAC,6BAA6B,CAAC;MAChE,OAAO/D,EAAE,EAAEkE,YAAY,CAAC,cAAc,CAAC,IAAI,IAAI;IACjD,CAAC,CAAC,CAACnH,KAAK,CAAC,MAAM,IAAI,CAAC;IAEpB,IAAI+I,YAAY,EAAE;MAChBpL,OAAO,CAACC,GAAG,CAAC,mCAAmCiJ,OAAO,EAAE,CAAC;MACzD,MAAMF,KAAK,GAAG,MAAMzB,cAAc,CAAC,wBAAwB,EAAE;QAC3DqC,UAAU,EAAEV,OAAO;QACnBW,UAAU,EAAEuB;MACd,CAAC,CAAC;MACF,MAAM1J,IAAI,CAACmF,QAAQ,CAAEiD,CAAS,IAAK;QACjC,MAAMuB,KAAK,GAAGjC,QAAQ,CAACC,aAAa,CAAC,gCAAgC,CAA4B;QACjG,IAAIgC,KAAK,EAAEA,KAAK,CAACpF,KAAK,GAAG6D,CAAC;QAC1B,MAAMK,GAAG,GAAGC,MAA4C;QACxD,IAAI,OAAOD,GAAG,CAACmB,iBAAiB,KAAK,UAAU,EAAE;UAC9CnB,GAAG,CAACmB,iBAAiB,CAAyBxB,CAAC,CAAC;QACnD;MACF,CAAC,EAAEd,KAAK,CAAC;MACT,OAAO;QAAE/G,MAAM,EAAE,IAAI;QAAEC,IAAI,EAAE;MAAY,CAAC;IAC5C;IAEA,OAAO;MAAED,MAAM,EAAE;IAAM,CAAC;EAC1B,CAAC,CAAC,OAAOsJ,GAAG,EAAE;IACZvL,OAAO,CAACC,GAAG,CAAC,+CAA+CsL,GAAG,YAAY3D,KAAK,GAAG2D,GAAG,CAACC,OAAO,GAAGC,MAAM,CAACF,GAAG,CAAC,EAAE,CAAC;IAC9G,OAAO;MAAEtJ,MAAM,EAAE;IAAM,CAAC;EAC1B;AACF;;AAEA;AACA;AACA;;AAEA,OAAO,eAAeyJ,aAAaA,CACjCC,GAAmB,EACnBhL,IAA6B,EAC7BiL,QAAiB,EAC8C;EAC/D,MAAMC,MAAM,GAAGlL,IAAI,CAACkL,MAAgB;EACpC,MAAM/K,GAAG,GAAGH,IAAI,CAACG,GAAa;EAC9B,MAAMkE,QAAQ,GAAGrE,IAAI,CAACqE,QAA8B;EACpD,MAAMiB,KAAK,GAAGtF,IAAI,CAACsF,KAA2B;EAC9C,MAAMU,MAAM,GAAGhG,IAAI,CAACgG,MAA4B;EAChD,MAAM5F,OAAO,GAAGJ,IAAI,CAACmL,QAA8B;EAEnD,IAAI,CAACD,MAAM,EAAE;IACX,OAAO;MAAEzH,OAAO,EAAE,KAAK;MAAEa,KAAK,EAAE;IAAqB,CAAC;EACxD;EACA,IAAI,CAACnE,GAAG,EAAE;IACR,OAAO;MAAEsD,OAAO,EAAE,KAAK;MAAEa,KAAK,EAAE;IAAkB,CAAC;EACrD;;EAEA;EACA,MAAM8G,OAAO,GAAGzJ,YAAY,CAACxB,GAAG,CAAC;EACjC,IAAIiL,OAAO,EAAE;IACX,OAAO;MAAE3H,OAAO,EAAE,KAAK;MAAEa,KAAK,EAAE8G;IAAQ,CAAC;EAC3C;EACA,IAAI;IACF,MAAMC,UAAU,GAAG,MAAM3M,WAAW,CAACyB,GAAG,CAAC;IACzC,IAAIkL,UAAU,EAAE;MACd,OAAO;QAAE5H,OAAO,EAAE,KAAK;QAAEa,KAAK,EAAE+G;MAAW,CAAC;IAC9C;EACF,CAAC,CAAC,MAAM;IACN,OAAO;MAAE5H,OAAO,EAAE,KAAK;MAAEa,KAAK,EAAE;IAAwB,CAAC;EAC3D;EAEA,IAAI;IACF,QAAQ4G,MAAM;MACZ,KAAK,UAAU;QACb,OAAO,MAAM7H,cAAc,CAAClD,GAAG,EAAEC,OAAO,CAAC;MAC3C,KAAK,YAAY;QACf,OAAO,MAAMuD,gBAAgB,CAACxD,GAAG,EAAEC,OAAO,CAAC;MAC7C,KAAK,SAAS;QACZ,OAAO,MAAMgE,aAAa,CAACjE,GAAG,EAAEkE,QAAQ,EAAGjE,OAAO,CAAC;MACrD,KAAK,OAAO;QACV,OAAO,MAAMqF,WAAW,CAACtF,GAAG,EAAEkE,QAAQ,EAAGjE,OAAO,CAAC;MACnD,KAAK,MAAM;QACT,OAAO,MAAMwF,UAAU,CAACzF,GAAG,EAAEkE,QAAQ,EAAGiB,KAAK,EAAGlF,OAAO,CAAC;MAC1D,KAAK,UAAU;QACb,OAAO,MAAM2F,cAAc,CAAC5F,GAAG,EAAE6F,MAAM,EAAG5F,OAAO,CAAC;MACpD,KAAK,KAAK;QACR,OAAO,MAAM+F,SAAS,CAAChG,GAAG,EAAEC,OAAO,CAAC;MACtC;QACE,OAAO;UAAEqD,OAAO,EAAE,KAAK;UAAEa,KAAK,EAAE,2BAA2B4G,MAAM;QAAqE,CAAC;IAC3I;EACF,CAAC,CAAC,OAAON,GAAY,EAAE;IACrB,MAAMC,OAAO,GAAGD,GAAG,YAAY3D,KAAK,GAAG2D,GAAG,CAACC,OAAO,GAAGC,MAAM,CAACF,GAAG,CAAC;IAChE;IACA,MAAMU,YAAY,GAAGT,OAAO,CACzBrI,OAAO,CAAC,iIAAiI,EAAE,EAAE,CAAC,CAC9IC,IAAI,CAAC,CAAC;IACT,OAAO;MAAEgB,OAAO,EAAE,KAAK;MAAEa,KAAK,EAAE,kBAAkBgH,YAAY,CAACxI,SAAS,CAAC,CAAC,EAAE,GAAG,CAAC;IAAG,CAAC;EACtF;AACF","ignoreList":[]}
|
|
@@ -133,6 +133,17 @@ export async function handleCampaigns(sb, args, storeId) {
|
|
|
133
133
|
if (args.scheduled_at !== undefined) updates.scheduled_at = args.scheduled_at;
|
|
134
134
|
if (args.channels !== undefined) updates.channels = args.channels;
|
|
135
135
|
if (args.channel_config !== undefined) updates.channel_config = args.channel_config;
|
|
136
|
+
if (args.deal_id !== undefined) updates.deal_id = args.deal_id;
|
|
137
|
+
if (args.template_id !== undefined) updates.template_id = args.template_id;
|
|
138
|
+
if (args.template_slug !== undefined) updates.template_slug = args.template_slug;
|
|
139
|
+
if (args.preview_text !== undefined) updates.preview_text = args.preview_text;
|
|
140
|
+
if (args.from_name !== undefined) updates.from_name = args.from_name;
|
|
141
|
+
if (args.from_email !== undefined) updates.from_email = args.from_email;
|
|
142
|
+
if (args.reply_to !== undefined) updates.reply_to = args.reply_to;
|
|
143
|
+
if (args.utm_source !== undefined) updates.utm_source = args.utm_source;
|
|
144
|
+
if (args.utm_medium !== undefined) updates.utm_medium = args.utm_medium;
|
|
145
|
+
if (args.utm_campaign !== undefined) updates.utm_campaign = args.utm_campaign;
|
|
146
|
+
if (args.budget !== undefined) updates.budget = args.budget;
|
|
136
147
|
const {
|
|
137
148
|
data,
|
|
138
149
|
error
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"campaigns.js","names":["sanitizeFilterValue","handleCampaigns","sb","args","storeId","sid","action","q","from","select","eq","order","ascending","status","campaign_type","query","sq","String","or","limit","data","error","success","message","count","length","campaignId","campaign_id","single","events","eventBreakdown","e","key","channel","event_type","event_breakdown","name","record","store_id","description","undefined","subject","preview_text","html_content","text_content","template_id","template_slug","from_name","from_email","reply_to","segment_id","channels","channel_config","deal_id","scheduled_at","utm_source","utm_medium","utm_campaign","budget","insert","updates","updated_at","Date","toISOString","update","messageBody","message_body","is_mms","media_url","scheduled_for"],"sources":["../../../src/server/handlers/campaigns.ts"],"sourcesContent":["// server/handlers/campaigns.ts — Marketing campaign management (email, SMS, multi-channel)\n\nimport type { SupabaseClient } from \"@supabase/supabase-js\";\nimport { sanitizeFilterValue } from \"../lib/utils.js\";\n\ntype Result = { success: boolean; data?: unknown; error?: string; [key: string]: unknown };\n\nexport async function handleCampaigns(\n sb: SupabaseClient,\n args: Record<string, unknown>,\n storeId?: string,\n): Promise<Result> {\n const sid = storeId as string;\n\n switch (args.action) {\n // ---- LIST: list campaigns ----\n case \"list\": {\n let q = sb.from(\"campaigns\")\n .select(\"id, name, description, campaign_type, channels, status, segment_id, total_recipients, total_sent, total_delivered, total_opened, total_clicked, open_rate, click_rate, revenue, scheduled_at, sent_at, created_at\")\n .eq(\"store_id\", sid)\n .order(\"created_at\", { ascending: false });\n if (args.status) q = q.eq(\"status\", args.status as string);\n if (args.campaign_type) q = q.eq(\"campaign_type\", args.campaign_type as string);\n if (args.query) {\n const sq = sanitizeFilterValue(String(args.query));\n q = q.or(`name.ilike.%${sq}%,description.ilike.%${sq}%`);\n }\n q = q.limit(args.limit as number || 50);\n const { data, error } = await q;\n return error ? { success: false, error: error.message } : { success: true, count: data?.length, data };\n }\n\n // ---- GET: get campaign detail with performance metrics ----\n case \"get\": {\n const campaignId = args.campaign_id as string;\n if (!campaignId) return { success: false, error: \"campaign_id is required\" };\n const { data, error } = await sb.from(\"campaigns\")\n .select(\"*\")\n .eq(\"id\", campaignId)\n .eq(\"store_id\", sid)\n .single();\n if (error) return { success: false, error: error.message };\n\n // Get event breakdown\n const { data: events } = await sb.from(\"marketing_campaign_events\")\n .select(\"event_type, channel\")\n .eq(\"campaign_id\", campaignId)\n .eq(\"store_id\", sid);\n const eventBreakdown: Record<string, number> = {};\n for (const e of events || []) {\n const key = `${e.channel}_${e.event_type}`;\n eventBreakdown[key] = (eventBreakdown[key] || 0) + 1;\n }\n\n return { success: true, data: { ...data, event_breakdown: eventBreakdown } };\n }\n\n // ---- CREATE: create a new campaign ----\n case \"create\": {\n const name = args.name as string;\n if (!name) return { success: false, error: \"name is required\" };\n const record: Record<string, unknown> = {\n store_id: sid,\n name,\n status: \"draft\",\n };\n if (args.description !== undefined) record.description = args.description;\n if (args.campaign_type !== undefined) record.campaign_type = args.campaign_type;\n if (args.subject !== undefined) record.subject = args.subject;\n if (args.preview_text !== undefined) record.preview_text = args.preview_text;\n if (args.html_content !== undefined) record.html_content = args.html_content;\n if (args.text_content !== undefined) record.text_content = args.text_content;\n if (args.template_id !== undefined) record.template_id = args.template_id;\n if (args.template_slug !== undefined) record.template_slug = args.template_slug;\n if (args.from_name !== undefined) record.from_name = args.from_name;\n if (args.from_email !== undefined) record.from_email = args.from_email;\n if (args.reply_to !== undefined) record.reply_to = args.reply_to;\n if (args.segment_id !== undefined) record.segment_id = args.segment_id;\n if (args.channels !== undefined) record.channels = args.channels;\n if (args.channel_config !== undefined) record.channel_config = args.channel_config;\n if (args.deal_id !== undefined) record.deal_id = args.deal_id;\n if (args.scheduled_at !== undefined) record.scheduled_at = args.scheduled_at;\n if (args.utm_source !== undefined) record.utm_source = args.utm_source;\n if (args.utm_medium !== undefined) record.utm_medium = args.utm_medium;\n if (args.utm_campaign !== undefined) record.utm_campaign = args.utm_campaign;\n if (args.budget !== undefined) record.budget = args.budget;\n\n const { data, error } = await sb.from(\"campaigns\")\n .insert(record)\n .select()\n .single();\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n\n // ---- UPDATE: modify a campaign ----\n case \"update\": {\n const campaignId = args.campaign_id as string;\n if (!campaignId) return { success: false, error: \"campaign_id is required\" };\n const updates: Record<string, unknown> = { updated_at: new Date().toISOString() };\n if (args.name !== undefined) updates.name = args.name;\n if (args.description !== undefined) updates.description = args.description;\n if (args.subject !== undefined) updates.subject = args.subject;\n if (args.html_content !== undefined) updates.html_content = args.html_content;\n if (args.text_content !== undefined) updates.text_content = args.text_content;\n if (args.status !== undefined) updates.status = args.status;\n if (args.segment_id !== undefined) updates.segment_id = args.segment_id;\n if (args.scheduled_at !== undefined) updates.scheduled_at = args.scheduled_at;\n if (args.channels !== undefined) updates.channels = args.channels;\n if (args.channel_config !== undefined) updates.channel_config = args.channel_config;\n\n const { data, error } = await sb.from(\"campaigns\")\n .update(updates)\n .eq(\"id\", campaignId)\n .eq(\"store_id\", sid)\n .select()\n .single();\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n\n // ---- LIST_SMS: list SMS campaigns ----\n case \"list_sms\": {\n let q = sb.from(\"sms_campaigns\")\n .select(\"*\")\n .eq(\"store_id\", sid)\n .order(\"created_at\", { ascending: false });\n if (args.status) q = q.eq(\"status\", args.status as string);\n q = q.limit(args.limit as number || 50);\n const { data, error } = await q;\n return error ? { success: false, error: error.message } : { success: true, count: data?.length, data };\n }\n\n // ---- CREATE_SMS: create an SMS campaign ----\n case \"create_sms\": {\n const name = args.name as string;\n const messageBody = args.message_body as string;\n if (!name || !messageBody) return { success: false, error: \"name and message_body are required\" };\n const record: Record<string, unknown> = {\n store_id: sid,\n name,\n message_body: messageBody,\n status: \"draft\",\n };\n if (args.description !== undefined) record.description = args.description;\n if (args.segment_id !== undefined) record.segment_id = args.segment_id;\n if (args.is_mms !== undefined) record.is_mms = args.is_mms;\n if (args.media_url !== undefined) record.media_url = args.media_url;\n if (args.scheduled_for !== undefined) record.scheduled_for = args.scheduled_for;\n\n const { data, error } = await sb.from(\"sms_campaigns\")\n .insert(record)\n .select()\n .single();\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n\n // ---- EVENTS: list campaign events/analytics ----\n case \"events\": {\n const campaignId = args.campaign_id as string;\n if (!campaignId) return { success: false, error: \"campaign_id is required\" };\n let q = sb.from(\"marketing_campaign_events\")\n .select(\"*\")\n .eq(\"campaign_id\", campaignId)\n .eq(\"store_id\", sid)\n .order(\"created_at\", { ascending: false });\n if (args.event_type) q = q.eq(\"event_type\", args.event_type as string);\n q = q.limit(args.limit as number || 100);\n const { data, error } = await q;\n return error ? { success: false, error: error.message } : { success: true, count: data?.length, data };\n }\n\n default:\n return { success: false, error: `Unknown campaigns action: ${args.action}. Valid: list, get, create, update, list_sms, create_sms, events` };\n }\n}\n"],"mappings":"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","ignoreList":[]}
|
|
1
|
+
{"version":3,"file":"campaigns.js","names":["sanitizeFilterValue","handleCampaigns","sb","args","storeId","sid","action","q","from","select","eq","order","ascending","status","campaign_type","query","sq","String","or","limit","data","error","success","message","count","length","campaignId","campaign_id","single","events","eventBreakdown","e","key","channel","event_type","event_breakdown","name","record","store_id","description","undefined","subject","preview_text","html_content","text_content","template_id","template_slug","from_name","from_email","reply_to","segment_id","channels","channel_config","deal_id","scheduled_at","utm_source","utm_medium","utm_campaign","budget","insert","updates","updated_at","Date","toISOString","update","messageBody","message_body","is_mms","media_url","scheduled_for"],"sources":["../../../src/server/handlers/campaigns.ts"],"sourcesContent":["// server/handlers/campaigns.ts — Marketing campaign management (email, SMS, multi-channel)\n\nimport type { SupabaseClient } from \"@supabase/supabase-js\";\nimport { sanitizeFilterValue } from \"../lib/utils.js\";\n\ntype Result = { success: boolean; data?: unknown; error?: string; [key: string]: unknown };\n\nexport async function handleCampaigns(\n sb: SupabaseClient,\n args: Record<string, unknown>,\n storeId?: string,\n): Promise<Result> {\n const sid = storeId as string;\n\n switch (args.action) {\n // ---- LIST: list campaigns ----\n case \"list\": {\n let q = sb.from(\"campaigns\")\n .select(\"id, name, description, campaign_type, channels, status, segment_id, total_recipients, total_sent, total_delivered, total_opened, total_clicked, open_rate, click_rate, revenue, scheduled_at, sent_at, created_at\")\n .eq(\"store_id\", sid)\n .order(\"created_at\", { ascending: false });\n if (args.status) q = q.eq(\"status\", args.status as string);\n if (args.campaign_type) q = q.eq(\"campaign_type\", args.campaign_type as string);\n if (args.query) {\n const sq = sanitizeFilterValue(String(args.query));\n q = q.or(`name.ilike.%${sq}%,description.ilike.%${sq}%`);\n }\n q = q.limit(args.limit as number || 50);\n const { data, error } = await q;\n return error ? { success: false, error: error.message } : { success: true, count: data?.length, data };\n }\n\n // ---- GET: get campaign detail with performance metrics ----\n case \"get\": {\n const campaignId = args.campaign_id as string;\n if (!campaignId) return { success: false, error: \"campaign_id is required\" };\n const { data, error } = await sb.from(\"campaigns\")\n .select(\"*\")\n .eq(\"id\", campaignId)\n .eq(\"store_id\", sid)\n .single();\n if (error) return { success: false, error: error.message };\n\n // Get event breakdown\n const { data: events } = await sb.from(\"marketing_campaign_events\")\n .select(\"event_type, channel\")\n .eq(\"campaign_id\", campaignId)\n .eq(\"store_id\", sid);\n const eventBreakdown: Record<string, number> = {};\n for (const e of events || []) {\n const key = `${e.channel}_${e.event_type}`;\n eventBreakdown[key] = (eventBreakdown[key] || 0) + 1;\n }\n\n return { success: true, data: { ...data, event_breakdown: eventBreakdown } };\n }\n\n // ---- CREATE: create a new campaign ----\n case \"create\": {\n const name = args.name as string;\n if (!name) return { success: false, error: \"name is required\" };\n const record: Record<string, unknown> = {\n store_id: sid,\n name,\n status: \"draft\",\n };\n if (args.description !== undefined) record.description = args.description;\n if (args.campaign_type !== undefined) record.campaign_type = args.campaign_type;\n if (args.subject !== undefined) record.subject = args.subject;\n if (args.preview_text !== undefined) record.preview_text = args.preview_text;\n if (args.html_content !== undefined) record.html_content = args.html_content;\n if (args.text_content !== undefined) record.text_content = args.text_content;\n if (args.template_id !== undefined) record.template_id = args.template_id;\n if (args.template_slug !== undefined) record.template_slug = args.template_slug;\n if (args.from_name !== undefined) record.from_name = args.from_name;\n if (args.from_email !== undefined) record.from_email = args.from_email;\n if (args.reply_to !== undefined) record.reply_to = args.reply_to;\n if (args.segment_id !== undefined) record.segment_id = args.segment_id;\n if (args.channels !== undefined) record.channels = args.channels;\n if (args.channel_config !== undefined) record.channel_config = args.channel_config;\n if (args.deal_id !== undefined) record.deal_id = args.deal_id;\n if (args.scheduled_at !== undefined) record.scheduled_at = args.scheduled_at;\n if (args.utm_source !== undefined) record.utm_source = args.utm_source;\n if (args.utm_medium !== undefined) record.utm_medium = args.utm_medium;\n if (args.utm_campaign !== undefined) record.utm_campaign = args.utm_campaign;\n if (args.budget !== undefined) record.budget = args.budget;\n\n const { data, error } = await sb.from(\"campaigns\")\n .insert(record)\n .select()\n .single();\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n\n // ---- UPDATE: modify a campaign ----\n case \"update\": {\n const campaignId = args.campaign_id as string;\n if (!campaignId) return { success: false, error: \"campaign_id is required\" };\n const updates: Record<string, unknown> = { updated_at: new Date().toISOString() };\n if (args.name !== undefined) updates.name = args.name;\n if (args.description !== undefined) updates.description = args.description;\n if (args.subject !== undefined) updates.subject = args.subject;\n if (args.html_content !== undefined) updates.html_content = args.html_content;\n if (args.text_content !== undefined) updates.text_content = args.text_content;\n if (args.status !== undefined) updates.status = args.status;\n if (args.segment_id !== undefined) updates.segment_id = args.segment_id;\n if (args.scheduled_at !== undefined) updates.scheduled_at = args.scheduled_at;\n if (args.channels !== undefined) updates.channels = args.channels;\n if (args.channel_config !== undefined) updates.channel_config = args.channel_config;\n if (args.deal_id !== undefined) updates.deal_id = args.deal_id;\n if (args.template_id !== undefined) updates.template_id = args.template_id;\n if (args.template_slug !== undefined) updates.template_slug = args.template_slug;\n if (args.preview_text !== undefined) updates.preview_text = args.preview_text;\n if (args.from_name !== undefined) updates.from_name = args.from_name;\n if (args.from_email !== undefined) updates.from_email = args.from_email;\n if (args.reply_to !== undefined) updates.reply_to = args.reply_to;\n if (args.utm_source !== undefined) updates.utm_source = args.utm_source;\n if (args.utm_medium !== undefined) updates.utm_medium = args.utm_medium;\n if (args.utm_campaign !== undefined) updates.utm_campaign = args.utm_campaign;\n if (args.budget !== undefined) updates.budget = args.budget;\n\n const { data, error } = await sb.from(\"campaigns\")\n .update(updates)\n .eq(\"id\", campaignId)\n .eq(\"store_id\", sid)\n .select()\n .single();\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n\n // ---- LIST_SMS: list SMS campaigns ----\n case \"list_sms\": {\n let q = sb.from(\"sms_campaigns\")\n .select(\"*\")\n .eq(\"store_id\", sid)\n .order(\"created_at\", { ascending: false });\n if (args.status) q = q.eq(\"status\", args.status as string);\n q = q.limit(args.limit as number || 50);\n const { data, error } = await q;\n return error ? { success: false, error: error.message } : { success: true, count: data?.length, data };\n }\n\n // ---- CREATE_SMS: create an SMS campaign ----\n case \"create_sms\": {\n const name = args.name as string;\n const messageBody = args.message_body as string;\n if (!name || !messageBody) return { success: false, error: \"name and message_body are required\" };\n const record: Record<string, unknown> = {\n store_id: sid,\n name,\n message_body: messageBody,\n status: \"draft\",\n };\n if (args.description !== undefined) record.description = args.description;\n if (args.segment_id !== undefined) record.segment_id = args.segment_id;\n if (args.is_mms !== undefined) record.is_mms = args.is_mms;\n if (args.media_url !== undefined) record.media_url = args.media_url;\n if (args.scheduled_for !== undefined) record.scheduled_for = args.scheduled_for;\n\n const { data, error } = await sb.from(\"sms_campaigns\")\n .insert(record)\n .select()\n .single();\n return error ? { success: false, error: error.message } : { success: true, data };\n }\n\n // ---- EVENTS: list campaign events/analytics ----\n case \"events\": {\n const campaignId = args.campaign_id as string;\n if (!campaignId) return { success: false, error: \"campaign_id is required\" };\n let q = sb.from(\"marketing_campaign_events\")\n .select(\"*\")\n .eq(\"campaign_id\", campaignId)\n .eq(\"store_id\", sid)\n .order(\"created_at\", { ascending: false });\n if (args.event_type) q = q.eq(\"event_type\", args.event_type as string);\n q = q.limit(args.limit as number || 100);\n const { data, error } = await q;\n return error ? { success: false, error: error.message } : { success: true, count: data?.length, data };\n }\n\n default:\n return { success: false, error: `Unknown campaigns action: ${args.action}. Valid: list, get, create, update, list_sms, create_sms, events` };\n }\n}\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -42,7 +42,7 @@ export async function handleProductActions(sb, args, sid) {
|
|
|
42
42
|
case "find":
|
|
43
43
|
{
|
|
44
44
|
const offset = args.offset || 0;
|
|
45
|
-
let q = sb.from("products").select("id, name, sku, status, type, stock_quantity, category:categories!primary_category_id(name)", {
|
|
45
|
+
let q = sb.from("products").select("id, name, sku, status, type, stock_quantity, custom_fields, category:categories!primary_category_id(name)", {
|
|
46
46
|
count: "exact"
|
|
47
47
|
}).eq("store_id", sid).order("created_at", {
|
|
48
48
|
ascending: false
|
|
@@ -96,11 +96,24 @@ export async function handleProductActions(sb, args, sid) {
|
|
|
96
96
|
const flattened = (data || []).map(row => {
|
|
97
97
|
const {
|
|
98
98
|
category,
|
|
99
|
+
custom_fields,
|
|
99
100
|
...rest
|
|
100
101
|
} = row;
|
|
102
|
+
// Strip null/empty values from custom_fields to keep find response lean
|
|
103
|
+
let fields;
|
|
104
|
+
if (custom_fields && typeof custom_fields === "object") {
|
|
105
|
+
const populated = {};
|
|
106
|
+
for (const [k, v] of Object.entries(custom_fields)) {
|
|
107
|
+
if (v !== null && v !== undefined && v !== "") populated[k] = v;
|
|
108
|
+
}
|
|
109
|
+
if (Object.keys(populated).length > 0) fields = populated;
|
|
110
|
+
}
|
|
101
111
|
return {
|
|
102
112
|
...rest,
|
|
103
|
-
category_name: category?.name || null
|
|
113
|
+
category_name: category?.name || null,
|
|
114
|
+
...(fields ? {
|
|
115
|
+
field_values: fields
|
|
116
|
+
} : {})
|
|
104
117
|
};
|
|
105
118
|
});
|
|
106
119
|
return {
|
|
@@ -236,7 +249,7 @@ export async function handleProductActions(sb, args, sid) {
|
|
|
236
249
|
fieldValues[key] = f.default ?? null;
|
|
237
250
|
}
|
|
238
251
|
}
|
|
239
|
-
const agentValues = args.custom_fields || {};
|
|
252
|
+
const agentValues = (args.custom_fields ?? args.field_values) || {};
|
|
240
253
|
for (const [k, v] of Object.entries(agentValues)) {
|
|
241
254
|
if (schemaKeys.has(k)) fieldValues[k] = v;
|
|
242
255
|
}
|
|
@@ -299,8 +312,9 @@ export async function handleProductActions(sb, args, sid) {
|
|
|
299
312
|
if (args.catalog_id !== undefined) updates.catalog_id = args.catalog_id;
|
|
300
313
|
if (args.pricing_schema_id !== undefined) updates.pricing_schema_id = args.pricing_schema_id;
|
|
301
314
|
if (args.pricing_data !== undefined) updates.pricing_data = args.pricing_data;
|
|
302
|
-
|
|
303
|
-
|
|
315
|
+
const customFieldsArg = args.custom_fields ?? args.field_values;
|
|
316
|
+
if (customFieldsArg !== undefined) {
|
|
317
|
+
const agentFV = customFieldsArg;
|
|
304
318
|
const {
|
|
305
319
|
data: pfs
|
|
306
320
|
} = await sb.from("product_field_schemas").select("field_schema_id").eq("product_id", pid).limit(1);
|