fengming 0.3.8 → 0.3.10
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/CHANGELOG.md +12 -0
- package/dist/abort-CCR8oZyg.js +277 -0
- package/dist/abort.runtime-eqx84NZa.js +2 -0
- package/dist/abort.runtime.js +1 -1
- package/dist/acp-spawn-BcDyvaXh.js +2 -0
- package/dist/acp-spawn-CCtSel-_.js +1286 -0
- package/dist/acp-stateful-target-driver-B3pCvuOV.js +89 -0
- package/dist/active-tool-schema-warnings-Cf-5q5WU.js +105 -0
- package/dist/active-tool-schema-warnings-Dzt3XE-y.js +2 -0
- package/dist/agent-1M2dVE2G.js +2 -0
- package/dist/agent-DLDIbUT4.js +1825 -0
- package/dist/agent-DZi6j3o6.js +3 -0
- package/dist/agent-command-ppfDBwET.js +1435 -0
- package/dist/agent-harness-runtime-C0bo62eY.d.ts +913 -0
- package/dist/agent-harness-runtime-RkTGiliR.js +207 -0
- package/dist/agent-runner-utils-B4YeVAO_.js +267 -0
- package/dist/agent-runner.runtime-T_3tlZN1.js +3784 -0
- package/dist/agent-runner.runtime.js +1 -1
- package/dist/agent-runtime-CloshyK-.d.ts +207 -0
- package/dist/agent-runtime-DZsmp1xr.js +199 -0
- package/dist/agent-tools-CET9usCz.js +2506 -0
- package/dist/agent-via-gateway-CZ0X0YkM.js +486 -0
- package/dist/agent-wait-dedupe-I81_F2tr.js +180 -0
- package/dist/agents/embedded-agent-runner/run/runtime-context-prompt.d.ts +1 -1
- package/dist/agents/embedded-agent-runner/tool-split.d.ts +1 -1
- package/dist/agents/model-catalog.runtime.d.ts +1 -1
- package/dist/api-BDB5xHYj.js +3 -0
- package/dist/api-BH7bfI5d.js +6 -0
- package/dist/api-BxJNXitd.js +2 -0
- package/dist/approval-client-helpers-CfQA9Jzh.d.ts +78 -0
- package/dist/approval-native-helpers-B2zyhxWc.d.ts +241 -0
- package/dist/approval-renderers-CMqSXyvm.d.ts +39 -0
- package/dist/assistant-Cu_Mzzgu.js +291 -0
- package/dist/attachment-normalize-Bb3v5iCC.js +213 -0
- package/dist/attempt-execution-BskbAq5I.js +584 -0
- package/dist/attempt-execution.runtime-BuT1wpNS.js +3 -0
- package/dist/attempt-execution.runtime.js +1 -1
- package/dist/attempt.prompt-helpers-ByKj-vQ7.js +543 -0
- package/dist/auto-reply/reply/commands-crestodian.d.ts +1 -1
- package/dist/binding-routing-EGRNvRC4.js +113 -0
- package/dist/binding-targets-B9vYS8n6.js +121 -0
- package/dist/bridge-server-DG9YtKQe.js +113 -0
- package/dist/browser-cli-B0Ou-Nbm.js +2 -0
- package/dist/browser-cli-GF25gL6M.js +230 -0
- package/dist/browser-cli-actions-input-CuER6RVe.js +522 -0
- package/dist/browser-cli-actions-observe-Dq-Zz_FA.js +81 -0
- package/dist/browser-cli-debug-7hvb49S1.js +137 -0
- package/dist/browser-cli-inspect-WU5KeWOK.js +117 -0
- package/dist/browser-cli-manage-B9D9BvSe.js +446 -0
- package/dist/browser-cli-resize-BUOusOPb.js +32 -0
- package/dist/browser-cli-shared-DmpSN-Qi.js +69 -0
- package/dist/browser-cli-state-D_YxTqRw.js +371 -0
- package/dist/browser-control-auth-Cqd1n9FA.js +2 -0
- package/dist/browser-profiles-9tQ05wYh.js +2 -0
- package/dist/browser-runtime-Bbzumoha.js +389 -0
- package/dist/build-CBXuT2K1.js +261 -0
- package/dist/build-info.json +3 -3
- package/dist/bundled/boot-md/handler.js +2 -2
- package/dist/bundled/session-memory/handler.js +1 -1
- package/dist/canvas-host/a2ui/.bundle.hash +1 -1
- package/dist/capability-cli-CiVGX7ut.js +1809 -0
- package/dist/channel-BIVwHq7P.js +2309 -0
- package/dist/channel-CiA2FVdL.d.ts +427 -0
- package/dist/channel-core-Cj9JTqeu.js +5 -0
- package/dist/channel-core-DjVpcH1C.d.ts +6 -0
- package/dist/channel-entry-contract-Bs_54m1D.d.ts +114 -0
- package/dist/channel-inbound-CdUdeQRa.js +121 -0
- package/dist/channel-inbound-iGiqx6CS.d.ts +97 -0
- package/dist/channel-message-C3QPaDUk.js +12 -0
- package/dist/channel-message-CXDjxb2U.d.ts +9 -0
- package/dist/channel-outbound-ofvvmMSK.d.ts +325 -0
- package/dist/channel-pairing-DKscOV1K.d.ts +58 -0
- package/dist/channel.runtime-VW6PW_ez.js +697 -0
- package/dist/channel.runtime.js +1 -1
- package/dist/chat-CFkQepUQ.js +3 -0
- package/dist/chat-Czh_tXM0.js +2940 -0
- package/dist/chrome-B2cq8YyH.js +1517 -0
- package/dist/cli/run-main.js +5 -5
- package/dist/cli-compaction-Dxv4nt7R.js +363 -0
- package/dist/cli-runner-CimXSTVs.js +2 -0
- package/dist/cli-runner-orCzE1Sr.js +597 -0
- package/dist/cli-runner.runtime-CsW0dXJW.js +3 -0
- package/dist/cli-runner.runtime-DgPrc1do.js +4 -0
- package/dist/cli-runner.runtime.js +1 -1
- package/dist/cli-startup-metadata.json +10 -12
- package/dist/command-registry-C5ooX6PF.js +4 -0
- package/dist/command-registry-DMB-HKIk.js +9 -0
- package/dist/command-registry-core-B2w_XWvn.js +114 -0
- package/dist/command-status.runtime-Bdy3Dkar.js +90 -0
- package/dist/command-status.runtime.js +1 -1
- package/dist/commands-compact.runtime-Bk2jTQo6.js +10 -0
- package/dist/commands-compact.runtime.js +1 -1
- package/dist/commands-handlers.runtime-Csw1og0U.js +6327 -0
- package/dist/commands-handlers.runtime.js +1 -1
- package/dist/commands-mRqmLIVz.d.ts +117 -0
- package/dist/commands-status-Dgrj_ubK.js +3 -0
- package/dist/commands-status-s32HJOpD.js +16 -0
- package/dist/commands-status.runtime-Dgrj_ubK.js +3 -0
- package/dist/commands-status.runtime.js +1 -1
- package/dist/commands-subagents-control.runtime-C_hnOO9l.js +2 -0
- package/dist/commands-subagents-control.runtime.js +1 -1
- package/dist/commands-system-prompt-CQV742Cc.js +2 -0
- package/dist/commands-system-prompt-DTfsKwK1.js +161 -0
- package/dist/commands-types-MXHhrssO.d.ts +132 -0
- package/dist/commands.runtime-akiVK67l.js +175 -0
- package/dist/commands.runtime.js +1 -1
- package/dist/commitments/runtime.js +1 -1
- package/dist/compact-U6ZhvPtD.js +1165 -0
- package/dist/compact.runtime-36E5vKsC.js +12 -0
- package/dist/compact.runtime.js +1 -1
- package/dist/completion-cli-Bf4mEw2W.js +393 -0
- package/dist/config-BJJhHN9E.js +374 -0
- package/dist/config-mutations-swLu-j_p.js +161 -0
- package/dist/config-schema-Tjner6bM.d.ts +20 -0
- package/dist/context-engine-host-compat-6fkn_daV.js +280 -0
- package/dist/context-engine-host-compat-DtAZC1bS.js +2 -0
- package/dist/context-engine-lifecycle-R__Idxi1.js +627 -0
- package/dist/control-auth-DnkI94_D.js +114 -0
- package/dist/control-service-BBsamNjq.js +40 -0
- package/dist/control-service-BGpHj7RL.js +3 -0
- package/dist/control-ui/assets/activity-D-mnRThU.js +124 -0
- package/dist/control-ui/assets/agents-U_KSP5I_.js +1030 -0
- package/dist/control-ui/assets/channels-ohK9_G1O.js +120 -0
- package/dist/control-ui/assets/cron-6ZCzfU29.js +1016 -0
- package/dist/control-ui/assets/debug-CSsDLg_s.js +97 -0
- package/dist/control-ui/assets/index-jUDczxhd.js +7214 -0
- package/dist/control-ui/assets/instances-782ZoDT4.js +57 -0
- package/dist/control-ui/assets/nodes-BMX16BKM.js +444 -0
- package/dist/control-ui/assets/sessions-jLGSApYa.js +425 -0
- package/dist/control-ui/assets/skills-DweBwUhs.js +362 -0
- package/dist/control-ui/assets/workboard-BsU-FXIo.js +402 -0
- package/dist/control-ui/index.html +1 -1
- package/dist/control-ui/sw.js +1 -1
- package/dist/conversation-runtime-DDekWU-U.js +31 -0
- package/dist/core-Bpk-qSJH.js +284 -0
- package/dist/core-Chrz4oRB.d.ts +223 -0
- package/dist/core-api-DSpUKNVW.js +2 -0
- package/dist/core-api-hLsW03Wo.js +5 -0
- package/dist/crestodian/crestodian.js +1 -1
- package/dist/crestodian/rescue-message.d.ts +1 -1
- package/dist/crestodian/rescue-message.js +1 -1
- package/dist/crestodian-C0x7JjF-.js +55 -0
- package/dist/delegate-BArFIZ4B.d.ts +30 -0
- package/dist/deliver-BnVp3VbL.d.ts +111 -0
- package/dist/delivery-queue-BFOASdf5.d.ts +161 -0
- package/dist/delivery-queue-runtime-BFfc8AEs.d.ts +9 -0
- package/dist/dialogue-No0NvYX7.js +37 -0
- package/dist/direct-dm-C-h88JJH.d.ts +79 -0
- package/dist/directive-handling.fast-lane-DTyDKhZb.js +70 -0
- package/dist/directive-handling.impl-DMReM6tu.js +2 -0
- package/dist/directive-handling.impl-DzPouhV-.js +823 -0
- package/dist/directive-handling.model-selection-BjwTBAZJ.js +122 -0
- package/dist/directive-handling.persist.runtime-D_O1okkR.js +274 -0
- package/dist/directive-handling.persist.runtime.js +1 -1
- package/dist/dispatch-Dtl-oRuN.js +2057 -0
- package/dist/dispatch-acp-transcript.runtime-M4y0Dq74.js +40 -0
- package/dist/dispatch-acp-transcript.runtime.js +1 -1
- package/dist/dispatch-acp.runtime-B8uCIKZS.js +18 -0
- package/dist/dispatch-acp.runtime.js +1 -1
- package/dist/dispatcher-DslvaRcj.js +106 -0
- package/dist/doctor-config-flow-LU2C94af.js +1819 -0
- package/dist/doctor-core-checks-BqcjExIZ.js +666 -0
- package/dist/doctor-core-checks-Diuk8l0N.js +2 -0
- package/dist/doctor-core-checks.runtime-DbS-kMZN.js +278 -0
- package/dist/doctor-core-checks.runtime.js +1 -1
- package/dist/doctor-health-BdqHwfYE.js +65 -0
- package/dist/doctor-health-contributions-DLLTVnll.js +874 -0
- package/dist/doctor-lint-PZqkVMWf.js +95 -0
- package/dist/doctor-mRfJMSb5.js +6 -0
- package/dist/doctor-state-integrity-Cdlm-peH.js +1257 -0
- package/dist/draft-stream-controls-B7uLonbw.d.ts +159 -0
- package/dist/embedded-agent-CEpOPW6X.js +4 -0
- package/dist/embedded-agent-CNINO_M-.d.ts +5 -0
- package/dist/embedded-agent-DxwzoZkp.js +4074 -0
- package/dist/embedded-agent.runtime-DDbhA85-.js +4 -0
- package/dist/embedded-agent.runtime.js +1 -1
- package/dist/embedded-backend-ChaKCepC.js +1581 -0
- package/dist/embedded-gateway-stub.runtime-e_ZxGhcW.js +12 -0
- package/dist/embedded-gateway-stub.runtime.js +1 -1
- package/dist/entry.d.ts +1 -1
- package/dist/extensionAPI.d.ts +2 -2
- package/dist/extensionAPI.js +1 -1
- package/dist/extensions/active-memory/index.d.ts +1 -1
- package/dist/extensions/active-memory/index.js +1 -1
- package/dist/extensions/admin-http-rpc/index.d.ts +1 -1
- package/dist/extensions/admin-http-rpc/index.js +1 -1
- package/dist/extensions/bonjour/index.d.ts +1 -1
- package/dist/extensions/browser/browser-bridge.js +1 -1
- package/dist/extensions/browser/browser-config.js +4 -4
- package/dist/extensions/browser/browser-control-auth.js +2 -2
- package/dist/extensions/browser/browser-doctor.js +2 -2
- package/dist/extensions/browser/browser-maintenance.js +1 -1
- package/dist/extensions/browser/browser-profiles.js +2 -2
- package/dist/extensions/browser/browser-runtime-api.js +12 -12
- package/dist/extensions/browser/cli-metadata.d.ts +1 -1
- package/dist/extensions/browser/cli-metadata.js +1 -1
- package/dist/extensions/browser/index.d.ts +1 -1
- package/dist/extensions/browser/index.js +1 -1
- package/dist/extensions/browser/plugin-registration.d.ts +1 -1
- package/dist/extensions/browser/plugin-registration.js +1 -1
- package/dist/extensions/browser/register.runtime.d.ts +2 -2
- package/dist/extensions/browser/register.runtime.js +4 -4
- package/dist/extensions/browser/runtime-api.d.ts +3 -3
- package/dist/extensions/browser/runtime-api.js +14 -14
- package/dist/extensions/browser/setup-api.d.ts +1 -1
- package/dist/extensions/canvas/cli-metadata.d.ts +1 -1
- package/dist/extensions/canvas/index.d.ts +1 -1
- package/dist/extensions/canvas/index.js +1 -1
- package/dist/extensions/canvas/setup-api.d.ts +1 -1
- package/dist/extensions/deepseek/api.d.ts +1 -1
- package/dist/extensions/deepseek/index.d.ts +1 -1
- package/dist/extensions/deepseek/provider-discovery.d.ts +1 -1
- package/dist/extensions/deepseek/stream.d.ts +1 -1
- package/dist/extensions/device-pair/api.d.ts +4 -4
- package/dist/extensions/device-pair/api.js +1 -1
- package/dist/extensions/device-pair/index.d.ts +1 -1
- package/dist/extensions/device-pair/notify.d.ts +1 -1
- package/dist/extensions/device-pair/pair-command-approve.js +1 -1
- package/dist/extensions/device-pair/qr-image.d.ts +1 -1
- package/dist/extensions/memory-core/api.d.ts +1 -1
- package/dist/extensions/memory-core/cli-metadata.d.ts +1 -1
- package/dist/extensions/memory-core/cli-metadata.js +1 -1
- package/dist/extensions/memory-core/index.d.ts +1 -1
- package/dist/extensions/memory-core/manager-runtime.d.ts +1 -1
- package/dist/extensions/memory-core/runtime-api.d.ts +3 -3
- package/dist/extensions/skill-workshop/api.d.ts +2 -2
- package/dist/extensions/skill-workshop/api.js +1 -1
- package/dist/extensions/skill-workshop/index.d.ts +1 -1
- package/dist/extensions/skill-workshop/index.js +2 -2
- package/dist/extensions/tavily/index.d.ts +1 -1
- package/dist/extensions/tavily/web-search-contract-api.d.ts +1 -1
- package/dist/extensions/tavily/web-search-provider.d.ts +1 -1
- package/dist/extensions/thread-ownership/api.d.ts +2 -2
- package/dist/extensions/thread-ownership/index.d.ts +1 -1
- package/dist/extensions/webhooks/api.d.ts +2 -2
- package/dist/extensions/webhooks/api.js +1 -1
- package/dist/extensions/webhooks/index.d.ts +1 -1
- package/dist/extensions/webhooks/index.js +1 -1
- package/dist/extensions/webhooks/runtime-api.d.ts +1 -1
- package/dist/extensions/workboard/api.d.ts +2 -2
- package/dist/extensions/workboard/index.d.ts +1 -1
- package/dist/extensions/workboard/index.js +1 -1
- package/dist/extensions/workboard/runtime-api.d.ts +1 -1
- package/dist/fengming-runtime-CtRd5677.d.ts +153 -0
- package/dist/fengming-tools-DBLsJfsf.js +12221 -0
- package/dist/gateway-cli-BgDV2HF9.js +443 -0
- package/dist/gateway-method-runtime-d9oN_XO9.js +21 -0
- package/dist/get-reply-CEMtvaTJ.js +5198 -0
- package/dist/get-reply-from-config.runtime-CE8zmX7o.js +2 -0
- package/dist/get-reply-from-config.runtime.js +1 -1
- package/dist/heartbeat-runner-H8SmaKmJ.js +5 -0
- package/dist/heartbeat-runner.runtime-6hUaxEbl.js +3 -0
- package/dist/heartbeat-runner.runtime.js +1 -1
- package/dist/hooks-DkEvkwzS.js +536 -0
- package/dist/host-compat-DeAq3dnI.d.ts +21 -0
- package/dist/http-registry-BDoApjTY.d.ts +23 -0
- package/dist/inbound-reply-dispatch-CIYP2OPo.d.ts +156 -0
- package/dist/inbound-reply-dispatch-geHu6oUK.js +147 -0
- package/dist/inbound-reply-dispatch-vwW5Hl-_.js +2 -0
- package/dist/index.js +1 -1
- package/dist/init-DpE_6dG4.js +59 -0
- package/dist/interactive-Cb_1f91G.d.ts +26 -0
- package/dist/isolated-agent-B_upYOOM.js +2 -0
- package/dist/isolated-agent-KH9uwWhw.js +1097 -0
- package/dist/kernel-BHnBXnm2.d.ts +241 -0
- package/dist/lifecycle-BmZwopzF.js +570 -0
- package/dist/list.probe-5kzWm9Jk.js +451 -0
- package/dist/list.probe-9zBcGGQ4.js +2 -0
- package/dist/list.status-command-DY2ifqp1.js +815 -0
- package/dist/llm-slug-generator-Bmx0I84M.js +78 -0
- package/dist/llm-slug-generator.js +1 -1
- package/dist/loader-BVz75gSb.d.ts +142 -0
- package/dist/local-dispatch.runtime-CX3IOY1E.js +10 -0
- package/dist/local-dispatch.runtime.js +1 -1
- package/dist/manager-BXGg8bfG.d.ts +409 -0
- package/dist/mcp-http-B1lnh67s.js +2 -0
- package/dist/mcp-http-CamghE-W.js +583 -0
- package/dist/media-runtime-DpykroJR.d.ts +261 -0
- package/dist/memory-core-host-engine-embeddings-N2dX5P40.d.ts +324 -0
- package/dist/memory-core-host-engine-storage-WQfkQMer.d.ts +54 -0
- package/dist/message-handler-Ca_pqGVS.js +1806 -0
- package/dist/model-catalog-BBMLIjhq.d.ts +88 -0
- package/dist/model-selection-Cq82FXLy.js +352 -0
- package/dist/models-cli-Dm_393dw.js +257 -0
- package/dist/monitor-d0eyE2k0.js +60 -0
- package/dist/monitor.account-vLQ3bKHu.js +5382 -0
- package/dist/nodes-Bunvrb33.js +1483 -0
- package/dist/nodes-edNlxb2I.js +3 -0
- package/dist/nodes-pending-DEIwVh9v.js +211 -0
- package/dist/openai-compat-errors-CvWEoG98.js +136 -0
- package/dist/openai-http-Bskdv4Tv.js +836 -0
- package/dist/openresponses-http-DxdgCxFU.js +1175 -0
- package/dist/operations-Z85LFqsT.js +805 -0
- package/dist/outbound.types-DVkbsxo8.d.ts +291 -0
- package/dist/plugin-enabled-fvhTpvYS.js +232 -0
- package/dist/plugin-entry-CunlVUw6.d.ts +47 -0
- package/dist/plugin-registration-9ovnK_Tk.js +97 -0
- package/dist/plugin-runtime-DH2ZM9P5.d.ts +117 -0
- package/dist/plugin-sdk/.boundary-entry-shims.stamp +1 -1
- package/dist/plugin-sdk/acp-runtime-backend.js +1 -1
- package/dist/plugin-sdk/acp-runtime.js +1 -1
- package/dist/plugin-sdk/agent-harness-runtime.js +5 -5
- package/dist/plugin-sdk/agent-harness-task-runtime.js +1 -1
- package/dist/plugin-sdk/agent-harness.js +6 -6
- package/dist/plugin-sdk/agent-runtime.js +2 -2
- package/dist/plugin-sdk/bundled-channel-config-schema-Dfn3b8sF.d.ts +3169 -0
- package/dist/plugin-sdk/bundled-channel-config-schema.d.ts +1 -1
- package/dist/plugin-sdk/channel-config-schema-legacy.d.ts +1 -1
- package/dist/plugin-sdk/channel-core.js +2 -2
- package/dist/plugin-sdk/channel-envelope.js +1 -1
- package/dist/plugin-sdk/channel-inbound-roots.js +1 -1
- package/dist/plugin-sdk/channel-inbound.js +2 -2
- package/dist/plugin-sdk/channel-location.js +1 -1
- package/dist/plugin-sdk/channel-message-runtime.js +3 -3
- package/dist/plugin-sdk/channel-message.js +2 -2
- package/dist/plugin-sdk/channel-runtime.js +0 -1
- package/dist/plugin-sdk/command-status-runtime.js +1 -1
- package/dist/plugin-sdk/compat.js +1 -1
- package/dist/plugin-sdk/conversation-binding-runtime.js +1 -1
- package/dist/plugin-sdk/conversation-runtime.js +3 -3
- package/dist/plugin-sdk/core.js +2 -2
- package/dist/plugin-sdk/discord.d.ts +1 -1
- package/dist/plugin-sdk/gateway-method-runtime.js +1 -1
- package/dist/plugin-sdk/health.js +1 -1
- package/dist/plugin-sdk/hook-runtime.js +0 -1
- package/dist/plugin-sdk/inbound-reply-dispatch.js +2 -2
- package/dist/plugin-sdk/index.js +1 -1
- package/dist/plugin-sdk/infra-runtime.js +3 -1
- package/dist/plugin-sdk/provider-auth-api-key.js +0 -1
- package/dist/plugin-sdk/provider-stream-family.js +0 -1
- package/dist/plugin-sdk/provider-usage.js +649 -1
- package/dist/plugin-sdk/reply-runtime.js +4 -4
- package/dist/plugin-sdk/video-generation.js +206 -1
- package/dist/plugin-service-BdZxoKBZ.js +1249 -0
- package/dist/plugin-service-quTl5hT0.d.ts +24 -0
- package/dist/plugins/build-smoke-entry.d.ts +2 -2
- package/dist/plugins/loader.d.ts +1 -1
- package/dist/plugins/provider-discovery.runtime.d.ts +1 -1
- package/dist/plugins/provider-runtime.runtime.d.ts +1 -1
- package/dist/plugins/runtime/index.d.ts +1 -1
- package/dist/plugins/runtime/index.js +4 -4
- package/dist/plugins/tools.d.ts +1 -1
- package/dist/prepare.runtime-DFvkUqBZ.js +798 -0
- package/dist/prepare.runtime.js +1 -1
- package/dist/preview-warnings-CGzc8ccG.js +618 -0
- package/dist/program-D19g2jaa.js +131 -0
- package/dist/provider-api-key-auth-B8GgTfo8.d.ts +27 -0
- package/dist/provider-auth-result-Diw-woMA.d.ts +21 -0
- package/dist/provider-catalog-shared-hMvzzDgL.d.ts +62 -0
- package/dist/provider-dispatcher-DCTc4lG_.js +22 -0
- package/dist/provider-dispatcher.runtime.js +1 -1
- package/dist/provider-model-shared-BUCh3uCL.d.ts +143 -0
- package/dist/provider-registry-BIokPlxa.d.ts +8 -0
- package/dist/provider-registry-CyUOXHG-.d.ts +8 -0
- package/dist/provider-registry-i--H79Ao.d.ts +29 -0
- package/dist/provider-self-hosted-setup-BF8UR8wg.d.ts +74 -0
- package/dist/provider-stream-NF0XJnar.d.ts +139 -0
- package/dist/provider-stream-shared-DLwDaYed.d.ts +132 -0
- package/dist/provider-web-search-contract-fields-D61Vl5Kl.d.ts +25 -0
- package/dist/pw-ai-DYR-D7xR.js +3064 -0
- package/dist/register.agent-DwW0mQPk.js +152 -0
- package/dist/register.crestodian-BhIukKDA.js +24 -0
- package/dist/register.maintenance-DWHlvztJ.js +85 -0
- package/dist/register.subclis-Bsvdh8RI.js +3 -0
- package/dist/register.subclis-DVk0HU4k.js +31 -0
- package/dist/register.subclis-core-BK7nVvl6.js +278 -0
- package/dist/registry-hscEPAcC.d.ts +8 -0
- package/dist/registry-types-Ce-n1tuw.d.ts +392 -0
- package/dist/repair-sequencing-0-qGNSUO.js +652 -0
- package/dist/reply-payload-mCw4ZND6.d.ts +200 -0
- package/dist/reply-turn-admission-DMWNadoS.js +2056 -0
- package/dist/reply.runtime-CE8zmX7o.js +2 -0
- package/dist/reply.runtime.js +1 -1
- package/dist/result-fallback-classifier-BZmv2ACy.js +98 -0
- package/dist/route-qQ-jYpFa.js +475 -0
- package/dist/routes-dTCmw98g.js +2 -0
- package/dist/routes-r8DRKa83.js +3701 -0
- package/dist/run-BgH7EPGH.js +1162 -0
- package/dist/run-command-DCsM-BVh.js +23 -0
- package/dist/run-command-I2ib4dwS.js +2 -0
- package/dist/run-context-CWaKUKKJ.js +66 -0
- package/dist/run-embedded.runtime-CBnDBWN0.js +4 -0
- package/dist/run-embedded.runtime.js +1 -1
- package/dist/run-execution-cli.runtime-DlYXI-lw.js +4 -0
- package/dist/run-execution-cli.runtime.js +1 -1
- package/dist/run-executor.runtime-DwepGrmB.js +330 -0
- package/dist/run-executor.runtime.js +1 -1
- package/dist/run-subagent-registry.runtime-CeVpoIhj.js +2 -0
- package/dist/run-subagent-registry.runtime.js +1 -1
- package/dist/runtime-D2ee-rNh.js +436 -0
- package/dist/runtime-api-B99ZlkNt.d.ts +5 -0
- package/dist/runtime-api-Cta2L_Yo.js +12 -0
- package/dist/runtime-channel-LKOkML3M.js +2 -0
- package/dist/runtime-channel-UtWvrTZ5.js +148 -0
- package/dist/runtime-embedded-agent.runtime-9O2Idzyb.js +2 -0
- package/dist/runtime-embedded-agent.runtime.js +1 -1
- package/dist/runtime-forwarders-DjI8RFL5.d.ts +39 -0
- package/dist/sdk-setup-tools-B-X04pa6.js +8 -0
- package/dist/selection-B02h8Old.js +3 -0
- package/dist/selection-DEPvzrW2.js +18365 -0
- package/dist/server-CMYi8gDo.js +24 -0
- package/dist/server-Cx07rsiY.js +72 -0
- package/dist/server-close.runtime.js +1 -1
- package/dist/server-context-BIemGRt4.js +2 -0
- package/dist/server-context-CB_an9iy.js +955 -0
- package/dist/server-cron-BeyuBUjb.js +3173 -0
- package/dist/server-cron-CV7KkVeB.js +2 -0
- package/dist/server-methods-C7EnpOhB.js +497 -0
- package/dist/server-node-events-DMMzZciN.js +597 -0
- package/dist/server-plugin-bootstrap-O6MzvzL8.js +71 -0
- package/dist/server-plugins-Dwnaz9kX.js +435 -0
- package/dist/server-reload-handlers-C-6TyPvI.js +719 -0
- package/dist/server-restart-sentinel-Dc35eYgk.js +700 -0
- package/dist/server-runtime-services-DCs-gqh_.js +3 -0
- package/dist/server-runtime-services-DF2fzzVd.js +147 -0
- package/dist/server-startup-plugins-t-YeYibm.js +127 -0
- package/dist/server-startup-post-attach-CiEki-DC.js +793 -0
- package/dist/server-ws-runtime-uaUpI-e8.js +374 -0
- package/dist/server.impl-MoHjSMr5.js +2622 -0
- package/dist/session-kill-http-CYCiQpt2.js +121 -0
- package/dist/session-reset-service-BYbADY57.js +651 -0
- package/dist/session-status.runtime-8NASbeO4.js +2 -0
- package/dist/session-status.runtime.js +1 -1
- package/dist/session-subagent-reactivation.runtime-DGeNY2Rb.js +2 -0
- package/dist/session-subagent-reactivation.runtime.js +1 -1
- package/dist/session-tab-registry-C2eElZrt.js +551 -0
- package/dist/sessions-DXAdVXIx.js +1917 -0
- package/dist/sessions-history-http-BwK7b8OH.js +432 -0
- package/dist/sessions-patch-BYC5gvY1.js +401 -0
- package/dist/sessions-resolve-C3ORcdmo.js +180 -0
- package/dist/sessions.runtime-BijldeSY.js +2 -0
- package/dist/sessions.runtime.js +1 -1
- package/dist/snapshot-urls-C5CfP3Co.js +317 -0
- package/dist/speech-core-wWkTZPpQ.d.ts +49 -0
- package/dist/standalone-CHrieUsw.js +42 -0
- package/dist/startup-context-CZfmG8-g.js +314 -0
- package/dist/status-subagents.runtime-D2XMebiS.js +32 -0
- package/dist/status-subagents.runtime.js +1 -1
- package/dist/status-text-JR7IPyzZ.js +301 -0
- package/dist/stream-BJgTkLEI.d.ts +5 -0
- package/dist/subagent-announce-B9cfs_KZ.js +353 -0
- package/dist/subagent-announce-delivery-CtmEvLTS.js +1369 -0
- package/dist/subagent-control-DVqLHi9O.js +492 -0
- package/dist/subagent-hooks-D251uSvy.js +230 -0
- package/dist/subagent-hooks-api-DvdMKxsC.js +23 -0
- package/dist/subagent-hooks-uLORYChc.js +2 -0
- package/dist/subagent-registry-BW2l_oYu.js +3 -0
- package/dist/subagent-registry-BxEMHuiN.js +2627 -0
- package/dist/subagent-registry.runtime.d.ts +1 -1
- package/dist/subagent-registry.runtime.js +1 -1
- package/dist/subagent-session-cleanup-BC5wV2qQ.js +390 -0
- package/dist/system-fvgHsr2x.js +111 -0
- package/dist/talk-DU0Sod_K.js +2454 -0
- package/dist/target-id-BR2xJIkd.js +107 -0
- package/dist/task-registry-control.runtime.js +1 -1
- package/dist/thread-bindings-8XNu5U2p.js +228 -0
- package/dist/tool-Boeg0N5g.js +143 -0
- package/dist/tool-dispatch-BhLLaL2g.js +155 -0
- package/dist/tool-resolution-CzsLs-87.js +153 -0
- package/dist/tool-split-Bedy42Ms.d.ts +19 -0
- package/dist/tools-B6egHpE3.d.ts +38 -0
- package/dist/tools-effective-NXscxK8n.js +442 -0
- package/dist/tools-effective-inventory-NsGMUVo-.js +379 -0
- package/dist/tools-invoke-_sSu96Kq.js +51 -0
- package/dist/tools-invoke-http-BDhlRl-G.js +68 -0
- package/dist/tools-invoke-shared-BH-T9Bcg.js +200 -0
- package/dist/tts-runtime-C1wu3o15.d.ts +230 -0
- package/dist/tui-C733Qov0.js +2 -0
- package/dist/tui-DqbscVN5.js +3 -0
- package/dist/tui-backend-_Pn3Byj-.js +257 -0
- package/dist/tui-cli-CAiC39zd.js +40 -0
- package/dist/tui-ink-run-ChXEGj1h.js +7414 -0
- package/dist/tui-ink-run-D4mSfLHj.js +2 -0
- package/dist/types-BQw1qXGl.d.ts +7034 -0
- package/dist/types-BzMoU6-C.d.ts +111 -0
- package/dist/types-DltHmoCX.d.ts +393 -0
- package/dist/types.public-C_bVIMBl.d.ts +70 -0
- package/dist/web-fetch/runtime.d.ts +1 -1
- package/dist/webhook-targets-DW2jhddP.d.ts +99 -0
- package/npm-shrinkwrap.json +12861 -11889
- package/package.json +2 -5
- package/skills/batch/SKILL.md +118 -0
- package/skills/code-review/SKILL.md +107 -0
- package/skills/debug/SKILL.md +83 -0
- package/skills/loop/SKILL.md +118 -0
- package/skills/run/SKILL.md +79 -0
- package/skills/run-skill-generator/SKILL.md +179 -0
- package/skills/verify/SKILL.md +103 -0
- package/dist/abort-DGskei2p.js +0 -277
- package/dist/abort.runtime-Buq9IZxn.js +0 -2
- package/dist/acp-spawn-DC6IyYaB.js +0 -1286
- package/dist/acp-spawn-Diqb3nel.js +0 -2
- package/dist/acp-stateful-target-driver-Clhe_L8v.js +0 -89
- package/dist/active-tool-schema-warnings-BRhKkyvt.js +0 -2
- package/dist/active-tool-schema-warnings-C6N0-ce6.js +0 -105
- package/dist/agent-C5lhsEZJ.js +0 -2
- package/dist/agent-WEb757bl.js +0 -1825
- package/dist/agent-command-iLD_nsVY.js +0 -1435
- package/dist/agent-core-BeDN8Ns5.d.ts +0 -13
- package/dist/agent-harness-runtime-C89_Q-bW.d.ts +0 -913
- package/dist/agent-harness-runtime-Dfn5rik2.js +0 -207
- package/dist/agent-runner-utils-DNiuuo43.js +0 -267
- package/dist/agent-runner.runtime-CFF_qJ5V.js +0 -3784
- package/dist/agent-runtime-BkMtWXxn.js +0 -199
- package/dist/agent-runtime-HufMO_YR.d.ts +0 -207
- package/dist/agent-tools-HmaDv4ot.js +0 -2506
- package/dist/agent-via-gateway-CZQG8RYL.js +0 -486
- package/dist/agent-wait-dedupe-C3xQk2Ww.js +0 -180
- package/dist/agent-z1cs3c7n.js +0 -3
- package/dist/api-B4IMKjSe.js +0 -3
- package/dist/api-BwSbBWI8.js +0 -3
- package/dist/api-CE9In9m4.js +0 -5
- package/dist/api-ClPvYNGa.js +0 -32
- package/dist/api-CntBCaZf.js +0 -3
- package/dist/api-DBZBwTsn.js +0 -6
- package/dist/api-OCPwGOvK.js +0 -2
- package/dist/api-yYhEo7gK.js +0 -4
- package/dist/approval-client-helpers-CfdQ3-vv.d.ts +0 -78
- package/dist/approval-native-helpers-DSHPksK4.d.ts +0 -241
- package/dist/approval-renderers-BfEfwk44.d.ts +0 -39
- package/dist/assistant-v5fdOYu7.js +0 -291
- package/dist/attachment-normalize-BHAbLiL2.js +0 -213
- package/dist/attempt-execution-5w9WYbaJ.js +0 -584
- package/dist/attempt-execution.runtime-DnhOWGzr.js +0 -3
- package/dist/attempt.prompt-helpers-C4M4erF7.js +0 -543
- package/dist/binding-routing-Dpes-QF1.js +0 -113
- package/dist/binding-targets-B6H5Pd-A.js +0 -121
- package/dist/bridge-server-BCpxCRm_.js +0 -113
- package/dist/browser-cli-DMhXHopl.js +0 -230
- package/dist/browser-cli-Dy_VugK0.js +0 -2
- package/dist/browser-cli-actions-input-WMP7_lm6.js +0 -522
- package/dist/browser-cli-actions-observe-DQWAWhwU.js +0 -81
- package/dist/browser-cli-debug-CaT2ZKAx.js +0 -137
- package/dist/browser-cli-inspect-DU-LUXq1.js +0 -117
- package/dist/browser-cli-manage-DAuogqIh.js +0 -446
- package/dist/browser-cli-resize-Cz5uO_aR.js +0 -32
- package/dist/browser-cli-shared-CjPZcG3j.js +0 -69
- package/dist/browser-cli-state-CC3l77-K.js +0 -371
- package/dist/browser-control-auth-ELccIUZy.js +0 -2
- package/dist/browser-profiles-Cuy4ia6_.js +0 -2
- package/dist/browser-runtime-De-iUfME.js +0 -389
- package/dist/build-DEF8Per9.js +0 -261
- package/dist/capability-cli-Bydel4E7.js +0 -1809
- package/dist/channel-6SGL4R5P.js +0 -2309
- package/dist/channel-D3Q3b8J-.d.ts +0 -427
- package/dist/channel-core-Bj71kAB5.d.ts +0 -6
- package/dist/channel-core-DMvyWnHg.js +0 -5
- package/dist/channel-entry-contract-zYxRmEdf.d.ts +0 -114
- package/dist/channel-inbound-DVJzBcJ8.d.ts +0 -97
- package/dist/channel-inbound-DlCa7eJe.js +0 -121
- package/dist/channel-message-CyPGMMFB.js +0 -12
- package/dist/channel-message-Czl4cdoA.d.ts +0 -9
- package/dist/channel-outbound-BNbhmruA.d.ts +0 -325
- package/dist/channel-pairing-BiS-tSvl.d.ts +0 -58
- package/dist/channel-runtime-D8hntg7H.js +0 -7
- package/dist/channel.runtime-DxErReJR.js +0 -697
- package/dist/chat-BLA8ORQI.js +0 -3
- package/dist/chat-DWRXkuvU.js +0 -2940
- package/dist/chrome-DPwFYi-g.js +0 -1517
- package/dist/cli-compaction-YZpssARf.js +0 -363
- package/dist/cli-runner-D2OAqxu3.js +0 -597
- package/dist/cli-runner-DO4SORQf.js +0 -2
- package/dist/cli-runner.runtime-CNSIpbeT.js +0 -4
- package/dist/cli-runner.runtime-DAOYvpVQ.js +0 -3
- package/dist/command-registry-DpD0fb8D.js +0 -4
- package/dist/command-registry-core-Cl3tLG8G.js +0 -114
- package/dist/command-registry-nbP7c8RT.js +0 -9
- package/dist/command-status.runtime-BBuXTkq0.js +0 -90
- package/dist/commands-CR8MVvlD.d.ts +0 -117
- package/dist/commands-compact.runtime-BoafIjjg.js +0 -10
- package/dist/commands-handlers.runtime-D9jViG_x.js +0 -6327
- package/dist/commands-status-BqqJ7PVq.js +0 -16
- package/dist/commands-status-CMd41Vxf.js +0 -3
- package/dist/commands-status.runtime-CMd41Vxf.js +0 -3
- package/dist/commands-subagents-control.runtime-CU4I3A_n.js +0 -2
- package/dist/commands-system-prompt-CTtu1D3-.js +0 -2
- package/dist/commands-system-prompt-xswhORdM.js +0 -161
- package/dist/commands-types-B67CsqXf.d.ts +0 -132
- package/dist/commands.runtime-BUFhkrjQ.js +0 -175
- package/dist/compact-Dz_WvRkQ.js +0 -1165
- package/dist/compact.runtime-n-AKErni.js +0 -12
- package/dist/completion-cli-DJYs_L4_.js +0 -393
- package/dist/config-CFMbHJb0.js +0 -374
- package/dist/config-mutations-DCAloTKR.js +0 -161
- package/dist/config-schema-Drw1zrnG.d.ts +0 -20
- package/dist/context-engine-host-compat-4mNm1HCE.js +0 -2
- package/dist/context-engine-host-compat-BzJ7fUIn.js +0 -280
- package/dist/context-engine-lifecycle-V4PNQp6k.js +0 -627
- package/dist/control-auth-DG_cw-aN.js +0 -114
- package/dist/control-service-CurYipgK.js +0 -3
- package/dist/control-service-VyncoV7j.js +0 -40
- package/dist/control-ui/assets/activity-D5Plhlo-.js +0 -124
- package/dist/control-ui/assets/agents-Chcdfe1E.js +0 -1030
- package/dist/control-ui/assets/channels-BEtB4H37.js +0 -120
- package/dist/control-ui/assets/cron-CZyPkxSU.js +0 -1016
- package/dist/control-ui/assets/debug-DvM8iG47.js +0 -97
- package/dist/control-ui/assets/index-Rmpgh0f1.js +0 -7214
- package/dist/control-ui/assets/instances-yTC_uu60.js +0 -57
- package/dist/control-ui/assets/nodes-vbAxVHIH.js +0 -444
- package/dist/control-ui/assets/sessions-DOviHme5.js +0 -425
- package/dist/control-ui/assets/skills-Bfp5HEGW.js +0 -362
- package/dist/control-ui/assets/workboard-5sU2kHsV.js +0 -402
- package/dist/conversation-runtime-DgaABwHh.js +0 -31
- package/dist/core-BeBXdneV.js +0 -284
- package/dist/core-Chqb7X6l.d.ts +0 -223
- package/dist/core-api-BlK0FgBM.js +0 -2
- package/dist/core-api-pAvYk716.js +0 -5
- package/dist/crestodian-FeGTBqO1.js +0 -55
- package/dist/delegate-BjIjSU_E.d.ts +0 -30
- package/dist/deliver-CvtWN4Ey.d.ts +0 -111
- package/dist/delivery-queue-CQ-cj3KG.d.ts +0 -161
- package/dist/delivery-queue-runtime-ut7MG04m.d.ts +0 -9
- package/dist/detect-BjXPyrwn.js +0 -115
- package/dist/detect-C1xeIemQ.d.ts +0 -16
- package/dist/dialogue-BowVYhEC.js +0 -37
- package/dist/direct-dm-DoZZHpA0.d.ts +0 -79
- package/dist/directive-handling.fast-lane-hnmQ_CvD.js +0 -70
- package/dist/directive-handling.impl-Cai-CFS1.js +0 -2
- package/dist/directive-handling.impl-DY84qIfU.js +0 -823
- package/dist/directive-handling.model-selection-DASssLFQ.js +0 -122
- package/dist/directive-handling.persist.runtime-BY7tJUs2.js +0 -274
- package/dist/dispatch-D5iG5A8j.js +0 -2057
- package/dist/dispatch-acp-transcript.runtime-D3r16hbD.js +0 -40
- package/dist/dispatch-acp.runtime-CAIau5qX.js +0 -18
- package/dist/dispatcher-7-d2gw3J.js +0 -106
- package/dist/doctor-DD5YEMmf.js +0 -6
- package/dist/doctor-config-flow-CdlLHJmX.js +0 -1819
- package/dist/doctor-core-checks-6MP99TQG.js +0 -666
- package/dist/doctor-core-checks-Dm_o576z.js +0 -2
- package/dist/doctor-core-checks.runtime-B2qbKATd.js +0 -278
- package/dist/doctor-health-BKrhOv1v.js +0 -65
- package/dist/doctor-health-contributions-bIBLmw69.js +0 -874
- package/dist/doctor-lint-aOLOWli4.js +0 -95
- package/dist/doctor-state-integrity-23NQNNuo.js +0 -1257
- package/dist/draft-stream-controls-Bk1GVJ1l.d.ts +0 -159
- package/dist/embedded-agent-BeK8FhZr.d.ts +0 -5
- package/dist/embedded-agent-CNp_y7jW.js +0 -4074
- package/dist/embedded-agent-NEmNlXDR.js +0 -4
- package/dist/embedded-agent.runtime-hEby8P2s.js +0 -4
- package/dist/embedded-backend-BdbgfpBP.js +0 -1581
- package/dist/embedded-gateway-stub.runtime-ySZUA3Gy.js +0 -12
- package/dist/extensions/alibaba/fengming.plugin.json +0 -47
- package/dist/extensions/alibaba/index.d.ts +0 -12
- package/dist/extensions/alibaba/index.js +0 -13
- package/dist/extensions/alibaba/package.json +0 -15
- package/dist/extensions/alibaba/video-generation-provider.d.ts +0 -6
- package/dist/extensions/alibaba/video-generation-provider.js +0 -2
- package/dist/extensions/baichuan/fengming.plugin.json +0 -69
- package/dist/extensions/baichuan/index.d.ts +0 -11
- package/dist/extensions/baichuan/index.js +0 -45
- package/dist/extensions/baichuan/models.d.ts +0 -7
- package/dist/extensions/baichuan/models.js +0 -2
- package/dist/extensions/baichuan/onboard.d.ts +0 -5
- package/dist/extensions/baichuan/onboard.js +0 -2
- package/dist/extensions/baichuan/package.json +0 -15
- package/dist/extensions/baichuan/provider-catalog.d.ts +0 -2
- package/dist/extensions/baichuan/provider-catalog.js +0 -2
- package/dist/extensions/baichuan/provider-discovery.d.ts +0 -2
- package/dist/extensions/baichuan/provider-discovery.js +0 -5
- package/dist/extensions/byteplus/api.d.ts +0 -3
- package/dist/extensions/byteplus/api.js +0 -3
- package/dist/extensions/byteplus/fengming.plugin.json +0 -196
- package/dist/extensions/byteplus/index.d.ts +0 -12
- package/dist/extensions/byteplus/index.js +0 -85
- package/dist/extensions/byteplus/models.d.ts +0 -2
- package/dist/extensions/byteplus/models.js +0 -2
- package/dist/extensions/byteplus/package.json +0 -15
- package/dist/extensions/byteplus/provider-catalog.d.ts +0 -2
- package/dist/extensions/byteplus/provider-catalog.js +0 -2
- package/dist/extensions/byteplus/provider-discovery.d.ts +0 -5
- package/dist/extensions/byteplus/provider-discovery.js +0 -23
- package/dist/extensions/byteplus/video-generation-provider.d.ts +0 -6
- package/dist/extensions/byteplus/video-generation-provider.js +0 -2
- package/dist/extensions/longcat/fengming.plugin.json +0 -84
- package/dist/extensions/longcat/index.d.ts +0 -11
- package/dist/extensions/longcat/index.js +0 -45
- package/dist/extensions/longcat/models.d.ts +0 -7
- package/dist/extensions/longcat/models.js +0 -2
- package/dist/extensions/longcat/onboard.d.ts +0 -5
- package/dist/extensions/longcat/onboard.js +0 -2
- package/dist/extensions/longcat/package.json +0 -15
- package/dist/extensions/longcat/provider-catalog.d.ts +0 -2
- package/dist/extensions/longcat/provider-catalog.js +0 -2
- package/dist/extensions/longcat/provider-discovery.d.ts +0 -2
- package/dist/extensions/longcat/provider-discovery.js +0 -5
- package/dist/extensions/minimax/api.d.ts +0 -5
- package/dist/extensions/minimax/api.js +0 -6
- package/dist/extensions/minimax/fengming.plugin.json +0 -206
- package/dist/extensions/minimax/image-generation-provider.d.ts +0 -6
- package/dist/extensions/minimax/image-generation-provider.js +0 -2
- package/dist/extensions/minimax/index.d.ts +0 -12
- package/dist/extensions/minimax/index.js +0 -29
- package/dist/extensions/minimax/media-understanding-provider.d.ts +0 -6
- package/dist/extensions/minimax/media-understanding-provider.js +0 -2
- package/dist/extensions/minimax/model-definitions.d.ts +0 -2
- package/dist/extensions/minimax/model-definitions.js +0 -2
- package/dist/extensions/minimax/music-generation-provider.d.ts +0 -6
- package/dist/extensions/minimax/music-generation-provider.js +0 -2
- package/dist/extensions/minimax/oauth.d.ts +0 -2
- package/dist/extensions/minimax/oauth.js +0 -2
- package/dist/extensions/minimax/oauth.runtime.d.ts +0 -2
- package/dist/extensions/minimax/oauth.runtime.js +0 -2
- package/dist/extensions/minimax/onboard.d.ts +0 -2
- package/dist/extensions/minimax/onboard.js +0 -2
- package/dist/extensions/minimax/package.json +0 -15
- package/dist/extensions/minimax/provider-catalog.d.ts +0 -2
- package/dist/extensions/minimax/provider-catalog.js +0 -2
- package/dist/extensions/minimax/provider-contract-api.d.ts +0 -6
- package/dist/extensions/minimax/provider-contract-api.js +0 -77
- package/dist/extensions/minimax/provider-discovery.d.ts +0 -5
- package/dist/extensions/minimax/provider-discovery.js +0 -23
- package/dist/extensions/minimax/provider-models.d.ts +0 -2
- package/dist/extensions/minimax/provider-models.js +0 -2
- package/dist/extensions/minimax/provider-registration.d.ts +0 -7
- package/dist/extensions/minimax/provider-registration.js +0 -2
- package/dist/extensions/minimax/speech-provider.d.ts +0 -5
- package/dist/extensions/minimax/speech-provider.js +0 -2
- package/dist/extensions/minimax/tts.d.ts +0 -20
- package/dist/extensions/minimax/tts.js +0 -2
- package/dist/extensions/minimax/video-generation-provider.d.ts +0 -7
- package/dist/extensions/minimax/video-generation-provider.js +0 -2
- package/dist/extensions/minimax/web-search-contract-api.d.ts +0 -5
- package/dist/extensions/minimax/web-search-contract-api.js +0 -31
- package/dist/extensions/minimax/web-search-provider.d.ts +0 -5
- package/dist/extensions/minimax/web-search-provider.js +0 -2
- package/dist/extensions/moonshot/api.d.ts +0 -3
- package/dist/extensions/moonshot/api.js +0 -4
- package/dist/extensions/moonshot/fengming.plugin.json +0 -250
- package/dist/extensions/moonshot/index.d.ts +0 -11
- package/dist/extensions/moonshot/index.js +0 -70
- package/dist/extensions/moonshot/media-understanding-provider.d.ts +0 -6
- package/dist/extensions/moonshot/media-understanding-provider.js +0 -2
- package/dist/extensions/moonshot/onboard.d.ts +0 -2
- package/dist/extensions/moonshot/onboard.js +0 -2
- package/dist/extensions/moonshot/package.json +0 -15
- package/dist/extensions/moonshot/provider-catalog.d.ts +0 -2
- package/dist/extensions/moonshot/provider-catalog.js +0 -2
- package/dist/extensions/moonshot/provider-contract-api.d.ts +0 -5
- package/dist/extensions/moonshot/provider-contract-api.js +0 -27
- package/dist/extensions/moonshot/provider-discovery.d.ts +0 -5
- package/dist/extensions/moonshot/provider-discovery.js +0 -15
- package/dist/extensions/moonshot/web-search-contract-api.d.ts +0 -5
- package/dist/extensions/moonshot/web-search-contract-api.js +0 -29
- package/dist/extensions/moonshot/web-search-provider.d.ts +0 -5
- package/dist/extensions/moonshot/web-search-provider.js +0 -2
- package/dist/extensions/qianfan/api.d.ts +0 -3
- package/dist/extensions/qianfan/api.js +0 -3
- package/dist/extensions/qianfan/fengming.plugin.json +0 -89
- package/dist/extensions/qianfan/index.d.ts +0 -11
- package/dist/extensions/qianfan/index.js +0 -26
- package/dist/extensions/qianfan/onboard.d.ts +0 -2
- package/dist/extensions/qianfan/onboard.js +0 -2
- package/dist/extensions/qianfan/package.json +0 -15
- package/dist/extensions/qianfan/provider-catalog.d.ts +0 -2
- package/dist/extensions/qianfan/provider-catalog.js +0 -2
- package/dist/extensions/qwen/api.d.ts +0 -4
- package/dist/extensions/qwen/api.js +0 -5
- package/dist/extensions/qwen/fengming.plugin.json +0 -389
- package/dist/extensions/qwen/index.d.ts +0 -11
- package/dist/extensions/qwen/index.js +0 -202
- package/dist/extensions/qwen/media-understanding-provider.d.ts +0 -6
- package/dist/extensions/qwen/media-understanding-provider.js +0 -2
- package/dist/extensions/qwen/model-definitions.d.ts +0 -2
- package/dist/extensions/qwen/model-definitions.js +0 -2
- package/dist/extensions/qwen/models.d.ts +0 -2
- package/dist/extensions/qwen/models.js +0 -2
- package/dist/extensions/qwen/onboard.d.ts +0 -21
- package/dist/extensions/qwen/onboard.js +0 -2
- package/dist/extensions/qwen/package.json +0 -15
- package/dist/extensions/qwen/provider-catalog.d.ts +0 -2
- package/dist/extensions/qwen/provider-catalog.js +0 -2
- package/dist/extensions/qwen/stream.d.ts +0 -2
- package/dist/extensions/qwen/stream.js +0 -2
- package/dist/extensions/qwen/video-generation-provider.d.ts +0 -6
- package/dist/extensions/qwen/video-generation-provider.js +0 -2
- package/dist/extensions/sensenova/fengming.plugin.json +0 -69
- package/dist/extensions/sensenova/index.d.ts +0 -11
- package/dist/extensions/sensenova/index.js +0 -45
- package/dist/extensions/sensenova/models.d.ts +0 -7
- package/dist/extensions/sensenova/models.js +0 -2
- package/dist/extensions/sensenova/onboard.d.ts +0 -5
- package/dist/extensions/sensenova/onboard.js +0 -2
- package/dist/extensions/sensenova/package.json +0 -15
- package/dist/extensions/sensenova/provider-catalog.d.ts +0 -2
- package/dist/extensions/sensenova/provider-catalog.js +0 -2
- package/dist/extensions/sensenova/provider-discovery.d.ts +0 -2
- package/dist/extensions/sensenova/provider-discovery.js +0 -5
- package/dist/extensions/stepfun/fengming.plugin.json +0 -162
- package/dist/extensions/stepfun/index.d.ts +0 -12
- package/dist/extensions/stepfun/index.js +0 -165
- package/dist/extensions/stepfun/onboard.d.ts +0 -7
- package/dist/extensions/stepfun/onboard.js +0 -2
- package/dist/extensions/stepfun/package.json +0 -15
- package/dist/extensions/stepfun/provider-catalog.d.ts +0 -14
- package/dist/extensions/stepfun/provider-catalog.js +0 -2
- package/dist/extensions/tencent/api.d.ts +0 -3
- package/dist/extensions/tencent/api.js +0 -4
- package/dist/extensions/tencent/fengming.plugin.json +0 -105
- package/dist/extensions/tencent/index.d.ts +0 -12
- package/dist/extensions/tencent/index.js +0 -62
- package/dist/extensions/tencent/models.d.ts +0 -2
- package/dist/extensions/tencent/models.js +0 -2
- package/dist/extensions/tencent/onboard.d.ts +0 -6
- package/dist/extensions/tencent/onboard.js +0 -2
- package/dist/extensions/tencent/package.json +0 -15
- package/dist/extensions/tencent/provider-catalog.d.ts +0 -2
- package/dist/extensions/tencent/provider-catalog.js +0 -2
- package/dist/extensions/tencent/provider-discovery.d.ts +0 -5
- package/dist/extensions/tencent/provider-discovery.js +0 -14
- package/dist/extensions/tiangong/fengming.plugin.json +0 -69
- package/dist/extensions/tiangong/index.d.ts +0 -11
- package/dist/extensions/tiangong/index.js +0 -45
- package/dist/extensions/tiangong/models.d.ts +0 -7
- package/dist/extensions/tiangong/models.js +0 -2
- package/dist/extensions/tiangong/onboard.d.ts +0 -5
- package/dist/extensions/tiangong/onboard.js +0 -2
- package/dist/extensions/tiangong/package.json +0 -15
- package/dist/extensions/tiangong/provider-catalog.d.ts +0 -2
- package/dist/extensions/tiangong/provider-catalog.js +0 -2
- package/dist/extensions/tiangong/provider-discovery.d.ts +0 -2
- package/dist/extensions/tiangong/provider-discovery.js +0 -5
- package/dist/extensions/volcengine/api.d.ts +0 -12
- package/dist/extensions/volcengine/api.js +0 -4
- package/dist/extensions/volcengine/fengming.plugin.json +0 -263
- package/dist/extensions/volcengine/index.d.ts +0 -12
- package/dist/extensions/volcengine/index.js +0 -88
- package/dist/extensions/volcengine/models.d.ts +0 -2
- package/dist/extensions/volcengine/models.js +0 -2
- package/dist/extensions/volcengine/package.json +0 -15
- package/dist/extensions/volcengine/provider-catalog.d.ts +0 -2
- package/dist/extensions/volcengine/provider-catalog.js +0 -2
- package/dist/extensions/volcengine/provider-discovery.d.ts +0 -5
- package/dist/extensions/volcengine/provider-discovery.js +0 -23
- package/dist/extensions/volcengine/speech-provider.d.ts +0 -5
- package/dist/extensions/volcengine/speech-provider.js +0 -2
- package/dist/extensions/volcengine/tts.d.ts +0 -22
- package/dist/extensions/volcengine/tts.js +0 -2
- package/dist/extensions/weixin/fengming.plugin.json +0 -22
- package/dist/extensions/weixin/index.d.ts +0 -26
- package/dist/extensions/weixin/index.js +0 -862
- package/dist/extensions/weixin/package.json +0 -45
- package/dist/extensions/xiaomi/api.d.ts +0 -3
- package/dist/extensions/xiaomi/api.js +0 -3
- package/dist/extensions/xiaomi/fengming.plugin.json +0 -260
- package/dist/extensions/xiaomi/index.d.ts +0 -12
- package/dist/extensions/xiaomi/index.js +0 -284
- package/dist/extensions/xiaomi/onboard.d.ts +0 -2
- package/dist/extensions/xiaomi/onboard.js +0 -2
- package/dist/extensions/xiaomi/package.json +0 -15
- package/dist/extensions/xiaomi/provider-catalog.d.ts +0 -2
- package/dist/extensions/xiaomi/provider-catalog.js +0 -2
- package/dist/extensions/xiaomi/speech-provider.d.ts +0 -5
- package/dist/extensions/xiaomi/speech-provider.js +0 -2
- package/dist/extensions/xiaomi/stream.d.ts +0 -5
- package/dist/extensions/xiaomi/stream.js +0 -2
- package/dist/extensions/xiaomi/thinking.d.ts +0 -11
- package/dist/extensions/xiaomi/thinking.js +0 -2
- package/dist/extensions/xingchen/fengming.plugin.json +0 -69
- package/dist/extensions/xingchen/index.d.ts +0 -11
- package/dist/extensions/xingchen/index.js +0 -45
- package/dist/extensions/xingchen/models.d.ts +0 -7
- package/dist/extensions/xingchen/models.js +0 -2
- package/dist/extensions/xingchen/onboard.d.ts +0 -5
- package/dist/extensions/xingchen/onboard.js +0 -2
- package/dist/extensions/xingchen/package.json +0 -15
- package/dist/extensions/xingchen/provider-catalog.d.ts +0 -2
- package/dist/extensions/xingchen/provider-catalog.js +0 -2
- package/dist/extensions/xingchen/provider-discovery.d.ts +0 -2
- package/dist/extensions/xingchen/provider-discovery.js +0 -5
- package/dist/extensions/yi/fengming.plugin.json +0 -84
- package/dist/extensions/yi/index.d.ts +0 -11
- package/dist/extensions/yi/index.js +0 -45
- package/dist/extensions/yi/models.d.ts +0 -7
- package/dist/extensions/yi/models.js +0 -2
- package/dist/extensions/yi/onboard.d.ts +0 -5
- package/dist/extensions/yi/onboard.js +0 -2
- package/dist/extensions/yi/package.json +0 -15
- package/dist/extensions/yi/provider-catalog.d.ts +0 -2
- package/dist/extensions/yi/provider-catalog.js +0 -2
- package/dist/extensions/yi/provider-discovery.d.ts +0 -2
- package/dist/extensions/yi/provider-discovery.js +0 -5
- package/dist/extensions/zai/api.d.ts +0 -4
- package/dist/extensions/zai/api.js +0 -4
- package/dist/extensions/zai/detect.d.ts +0 -2
- package/dist/extensions/zai/detect.js +0 -2
- package/dist/extensions/zai/fengming.plugin.json +0 -377
- package/dist/extensions/zai/index.d.ts +0 -12
- package/dist/extensions/zai/index.js +0 -297
- package/dist/extensions/zai/media-understanding-provider.d.ts +0 -5
- package/dist/extensions/zai/media-understanding-provider.js +0 -2
- package/dist/extensions/zai/model-definitions.d.ts +0 -2
- package/dist/extensions/zai/model-definitions.js +0 -2
- package/dist/extensions/zai/onboard.d.ts +0 -2
- package/dist/extensions/zai/onboard.js +0 -2
- package/dist/extensions/zai/package.json +0 -15
- package/dist/extensions/zai/runtime-api.d.ts +0 -2
- package/dist/extensions/zai/runtime-api.js +0 -2
- package/dist/extensions/zhinao/fengming.plugin.json +0 -69
- package/dist/extensions/zhinao/index.d.ts +0 -11
- package/dist/extensions/zhinao/index.js +0 -45
- package/dist/extensions/zhinao/models.d.ts +0 -7
- package/dist/extensions/zhinao/models.js +0 -2
- package/dist/extensions/zhinao/onboard.d.ts +0 -5
- package/dist/extensions/zhinao/onboard.js +0 -2
- package/dist/extensions/zhinao/package.json +0 -15
- package/dist/extensions/zhinao/provider-catalog.d.ts +0 -2
- package/dist/extensions/zhinao/provider-catalog.js +0 -2
- package/dist/extensions/zhinao/provider-discovery.d.ts +0 -2
- package/dist/extensions/zhinao/provider-discovery.js +0 -5
- package/dist/fengming-runtime-0jdu_329.d.ts +0 -153
- package/dist/fengming-tools-gQkwsWYz.js +0 -12221
- package/dist/fengming.plugin-C-Kdi1_5.js +0 -130
- package/dist/fengming.plugin-CRPqMj85.js +0 -166
- package/dist/gateway-cli-BV1V43-D.js +0 -443
- package/dist/gateway-method-runtime-J2OPP_oH.js +0 -21
- package/dist/get-reply-BE8ZGJos.js +0 -5198
- package/dist/get-reply-from-config.runtime-C5wfxVI_.js +0 -2
- package/dist/heartbeat-runner-_0HlObMb.js +0 -5
- package/dist/heartbeat-runner.runtime-DvYz_4Z3.js +0 -3
- package/dist/hook-runtime-BH9moP5T.js +0 -4
- package/dist/hooks-icCwsmrQ.js +0 -536
- package/dist/host-compat-dfJvEfe7.d.ts +0 -21
- package/dist/http-registry-Buj7R-F_.d.ts +0 -23
- package/dist/image-generation-provider-hrRXkkGc.js +0 -152
- package/dist/inbound-reply-dispatch-5AYt56Yt.js +0 -147
- package/dist/inbound-reply-dispatch-B5weFW8i.js +0 -2
- package/dist/inbound-reply-dispatch-cJh4H31y.d.ts +0 -156
- package/dist/infra-runtime-3_0R8nmO.js +0 -32
- package/dist/init-BnfkYG_k.js +0 -59
- package/dist/interactive-V8NfYsTW.d.ts +0 -26
- package/dist/isolated-agent-CgH7dfOj.js +0 -1097
- package/dist/isolated-agent-dBWkiw0a.js +0 -2
- package/dist/kernel-Ds2aqAJF.d.ts +0 -241
- package/dist/kimi-web-search-provider-QJT3Ftj3.js +0 -80
- package/dist/kimi-web-search-provider.runtime-Dj3SS4T5.js +0 -307
- package/dist/kimi-web-search-provider.runtime.js +0 -1
- package/dist/lib-Dg4yjNFQ.js +0 -871
- package/dist/lifecycle-B9k7QGsS.js +0 -570
- package/dist/list.probe-CbVHFNwf.js +0 -2
- package/dist/list.probe-CxiEBmyW.js +0 -451
- package/dist/list.status-command-DE-edGgB.js +0 -815
- package/dist/llm-slug-generator-DJgq9eFd.js +0 -78
- package/dist/loader-5AqYM9PC.d.ts +0 -142
- package/dist/local-dispatch.runtime-D3F4v51B.js +0 -10
- package/dist/manager-BWf1ks-Z.d.ts +0 -409
- package/dist/mcp-http-DU7Nsg4P.js +0 -583
- package/dist/mcp-http-iZCW6Cet.js +0 -2
- package/dist/media-runtime-DZ5RpQN7.d.ts +0 -261
- package/dist/media-understanding-DEdEyoQB.d.ts +0 -46
- package/dist/media-understanding-provider-4JHrQOUE.js +0 -70
- package/dist/media-understanding-provider-BV7O82XV.js +0 -29
- package/dist/media-understanding-provider-BlPRhYkx.js +0 -69
- package/dist/media-understanding-provider-BuX8eQLj.js +0 -13
- package/dist/memory-core-host-engine-embeddings-BDu5fx8E.d.ts +0 -324
- package/dist/memory-core-host-engine-storage-CdCuH-E2.d.ts +0 -54
- package/dist/message-handler-L6QLWNVP.js +0 -1806
- package/dist/minimax-web-search-provider-_gxeEOy8.js +0 -58
- package/dist/minimax-web-search-provider.runtime-BF4mGi6U.js +0 -148
- package/dist/minimax-web-search-provider.runtime.js +0 -1
- package/dist/model-catalog-DCnRkX8f.d.ts +0 -88
- package/dist/model-definitions-B2gY43hI.d.ts +0 -34
- package/dist/model-definitions-BLOyeH5h.js +0 -73
- package/dist/model-definitions-CoByf5mT.js +0 -243
- package/dist/model-definitions-WP3OmzbS.d.ts +0 -57
- package/dist/model-selection-DhTE6GZD.js +0 -352
- package/dist/models--iAR9QkZ.js +0 -175
- package/dist/models-8ImVEkvh.js +0 -36
- package/dist/models-BIDM8htk.js +0 -48
- package/dist/models-BRgRfrcS.js +0 -36
- package/dist/models-Bib5-APc.js +0 -67
- package/dist/models-Bl67zOoe.js +0 -36
- package/dist/models-BqDDYFE3.d.ts +0 -65
- package/dist/models-BtRQoRIu.js +0 -36
- package/dist/models-BvXmOXik.js +0 -48
- package/dist/models-C-sJciOD.d.ts +0 -9
- package/dist/models-COnXPdlL.js +0 -24
- package/dist/models-CXTmk-Da.d.ts +0 -8
- package/dist/models-Cz0C_8re.js +0 -36
- package/dist/models-DbwEIt-m.d.ts +0 -15
- package/dist/models-DgXkSADi.js +0 -30
- package/dist/models-cli-Bv3y3JgQ.js +0 -257
- package/dist/monitor-BiVOsbbN.js +0 -1024
- package/dist/monitor-BumfRp1t.js +0 -60
- package/dist/monitor.account-Cd6EwtuZ.js +0 -5382
- package/dist/music-generation-provider-ZdDMiC-c.js +0 -308
- package/dist/nodes-C0f8XgD5.js +0 -1483
- package/dist/nodes-Dk4vOgg9.js +0 -3
- package/dist/nodes-pending-Cjg09MXz.js +0 -211
- package/dist/oauth-BIO69Qw0.d.ts +0 -25
- package/dist/oauth-CnO10TN2.js +0 -207
- package/dist/onboard-B3BYT5k7.js +0 -34
- package/dist/onboard-BDMNV6RE.js +0 -23
- package/dist/onboard-B_WNNy5F.d.ts +0 -6
- package/dist/onboard-BbyMaErU.js +0 -69
- package/dist/onboard-BuYPNE6j2.js +0 -23
- package/dist/onboard-C394zMnM.d.ts +0 -11
- package/dist/onboard-CHn4oVbY.js +0 -24
- package/dist/onboard-CPpVbb0O.js +0 -73
- package/dist/onboard-CWDx7Crt.js +0 -23
- package/dist/onboard-CbzkwBzu.d.ts +0 -12
- package/dist/onboard-D099qUd0.js +0 -23
- package/dist/onboard-D7dbzfHc.js +0 -23
- package/dist/onboard-DB-x0nHF.js +0 -30
- package/dist/onboard-DFVrRnxJ.js +0 -23
- package/dist/onboard-DFiqoOc2.d.ts +0 -7
- package/dist/onboard-DJaMK3rr.d.ts +0 -6
- package/dist/onboard-DMdK8D_h.js +0 -67
- package/dist/onboard-J-KL-I6m.js +0 -48
- package/dist/onboard-MIBU-Rmv.js +0 -39
- package/dist/onboard-vmGylfFe.js +0 -23
- package/dist/openai-compat-errors-Dcr5Y8bF.js +0 -136
- package/dist/openai-http-CcqspzU6.js +0 -836
- package/dist/openresponses-http-BnyYYvUF.js +0 -1175
- package/dist/operations-H2Oq0KYz.js +0 -805
- package/dist/outbound.types-BhRehecY.d.ts +0 -291
- package/dist/plugin-enabled-CEIKWKrq.js +0 -232
- package/dist/plugin-entry-CTVRRaaA.d.ts +0 -47
- package/dist/plugin-registration-BTyO5Fwt.js +0 -97
- package/dist/plugin-runtime-_XF2N_UQ.d.ts +0 -117
- package/dist/plugin-sdk/bundled-channel-config-schema-BsOWCrJT.d.ts +0 -3169
- package/dist/plugin-service-B91jVlmZ.d.ts +0 -24
- package/dist/plugin-service-CtGwVz8V.js +0 -1249
- package/dist/prepare.runtime-9dlboph7.js +0 -798
- package/dist/preview-warnings-DJx4KJpC.js +0 -618
- package/dist/program-CWC-NBBB.js +0 -131
- package/dist/provider-api-key-auth-BmNcYRMl.d.ts +0 -27
- package/dist/provider-auth-api-key-CCaFiqY3.js +0 -5
- package/dist/provider-auth-result-D_E9dcVc.d.ts +0 -21
- package/dist/provider-catalog-5KZLmrDO.js +0 -11
- package/dist/provider-catalog-7P6AvDzS.js +0 -11
- package/dist/provider-catalog-B2gyTjTU.js +0 -88
- package/dist/provider-catalog-B3YBhe77.js +0 -17
- package/dist/provider-catalog-B7XEeuUm.js +0 -11
- package/dist/provider-catalog-BFGPRd9v.js +0 -17
- package/dist/provider-catalog-BLvkIMSk.d.ts +0 -6
- package/dist/provider-catalog-BPBL9mJf.d.ts +0 -5
- package/dist/provider-catalog-BRkZ6-HD.d.ts +0 -5
- package/dist/provider-catalog-Bfl_AoTZ.js +0 -142
- package/dist/provider-catalog-BpiHWHu1.js +0 -11
- package/dist/provider-catalog-C1qDLekT.d.ts +0 -5
- package/dist/provider-catalog-CKWNCfry.js +0 -11
- package/dist/provider-catalog-CUHB2pSt.d.ts +0 -7
- package/dist/provider-catalog-CWqN2j6J.d.ts +0 -5
- package/dist/provider-catalog-CZ8oYbx3.js +0 -11
- package/dist/provider-catalog-CcQ5-4ZW.d.ts +0 -6
- package/dist/provider-catalog-Cd16uZ0U.js +0 -20
- package/dist/provider-catalog-CpF2D0VK.js +0 -61
- package/dist/provider-catalog-CvXq36zW.d.ts +0 -5
- package/dist/provider-catalog-D2pgEME3.js +0 -48
- package/dist/provider-catalog-DPzcupEl.d.ts +0 -5
- package/dist/provider-catalog-DaeI606G.d.ts +0 -9
- package/dist/provider-catalog-DrOCtTb-.js +0 -11
- package/dist/provider-catalog-DwZ1J2Al.d.ts +0 -6
- package/dist/provider-catalog-Dy7IcHmS.js +0 -107
- package/dist/provider-catalog-TsZS52nq.d.ts +0 -10
- package/dist/provider-catalog-YqIFRCND.d.ts +0 -5
- package/dist/provider-catalog-Ywb5jRwG.d.ts +0 -5
- package/dist/provider-catalog-evknl1oN.js +0 -11
- package/dist/provider-catalog-l0hFpFO2.d.ts +0 -17
- package/dist/provider-catalog-shared-DsRBv0Tp.d.ts +0 -62
- package/dist/provider-dispatcher-BMy9mBJ1.js +0 -22
- package/dist/provider-model-shared-CPAfQBNs.d.ts +0 -143
- package/dist/provider-models-Diu65OcG.d.ts +0 -18
- package/dist/provider-models-LE7PlLYY.js +0 -22
- package/dist/provider-onboard-CpvXEmvz.d.ts +0 -91
- package/dist/provider-registration-DF-LkmNE.js +0 -235
- package/dist/provider-registry-D9cTPW1F.d.ts +0 -8
- package/dist/provider-registry-DI7gMKUP.d.ts +0 -8
- package/dist/provider-registry-DZtgZDkl.d.ts +0 -29
- package/dist/provider-self-hosted-setup-CoHvoyKm.d.ts +0 -74
- package/dist/provider-stream-BpXJr5Ap.d.ts +0 -139
- package/dist/provider-stream-family-Bj5aBD8w.js +0 -2
- package/dist/provider-stream-shared-BaUkhUHj.d.ts +0 -132
- package/dist/provider-usage-DFUhW2do.js +0 -651
- package/dist/provider-web-search-contract-fields-CkXzSsWu.d.ts +0 -25
- package/dist/pw-ai-9Q_dIq4B.js +0 -3064
- package/dist/register.agent-CbfrlzXB.js +0 -152
- package/dist/register.crestodian-CEg0rPfK.js +0 -24
- package/dist/register.maintenance-k9N8I4Wg.js +0 -85
- package/dist/register.subclis-CrXOeaS3.js +0 -3
- package/dist/register.subclis-DfKlni8N.js +0 -31
- package/dist/register.subclis-core-Bg4wbDsO.js +0 -278
- package/dist/registry-Bh3-P2HL.d.ts +0 -8
- package/dist/registry-types-BmEUS4d3.d.ts +0 -392
- package/dist/repair-sequencing-E4yViXG9.js +0 -652
- package/dist/reply-payload-S2mrc_Mh.d.ts +0 -200
- package/dist/reply-turn-admission-BBoPjmGB.js +0 -2056
- package/dist/reply.runtime-C5wfxVI_.js +0 -2
- package/dist/result-fallback-classifier-CX4iLD1G.js +0 -98
- package/dist/route-CifxcQZ1.js +0 -475
- package/dist/routes-B3XAOeWo.js +0 -2
- package/dist/routes-H185h3U-.js +0 -3701
- package/dist/run-CTJFbwbB.js +0 -1162
- package/dist/run-command-B7B53tYk.js +0 -23
- package/dist/run-command-BFuxRDxS.js +0 -2
- package/dist/run-context-C7im9ICg.js +0 -66
- package/dist/run-embedded.runtime-TljBTbzh.js +0 -4
- package/dist/run-execution-cli.runtime-Bt5zwx1W.js +0 -4
- package/dist/run-executor.runtime-hmbWX2Ct.js +0 -330
- package/dist/run-subagent-registry.runtime-B70X80nS.js +0 -2
- package/dist/runtime-DoKE0o7v.js +0 -436
- package/dist/runtime-api-Ca4Llbgf.js +0 -12
- package/dist/runtime-api-pa8xcEmg.d.ts +0 -5
- package/dist/runtime-channel-CFQ59svm.js +0 -148
- package/dist/runtime-channel-DRwCWGUx.js +0 -2
- package/dist/runtime-embedded-agent.runtime-DwmqKUVp.js +0 -2
- package/dist/runtime-forwarders-BMThPHg_.d.ts +0 -39
- package/dist/sdk-setup-tools-Cg_Tabrf.js +0 -8
- package/dist/selection-COhr7g82.js +0 -18365
- package/dist/selection-_G44EVqd.js +0 -3
- package/dist/send-media-BNc67G7I.js +0 -2072
- package/dist/server-5rR0RCpI.js +0 -24
- package/dist/server-context-BhiPROPA.js +0 -955
- package/dist/server-context-OShBAJZQ.js +0 -2
- package/dist/server-cron-Bkzb9edh.js +0 -3173
- package/dist/server-cron-DdR-ugiU.js +0 -2
- package/dist/server-lwtC1vaS.js +0 -72
- package/dist/server-methods-BY_ZqDFJ.js +0 -497
- package/dist/server-node-events-CLvE94AS.js +0 -597
- package/dist/server-plugin-bootstrap-cKOAH5GL.js +0 -71
- package/dist/server-plugins-CPpUykw5.js +0 -435
- package/dist/server-reload-handlers-uzt4VDZ-.js +0 -719
- package/dist/server-restart-sentinel-CpvV0t4O.js +0 -700
- package/dist/server-runtime-services-BhOHoerM.js +0 -147
- package/dist/server-runtime-services-D3Ig68nC.js +0 -3
- package/dist/server-startup-plugins-DslzKVHK.js +0 -127
- package/dist/server-startup-post-attach-DPFBTQez.js +0 -793
- package/dist/server-ws-runtime-D0zoWoiz.js +0 -374
- package/dist/server.impl-CzqLQ3qt.js +0 -2622
- package/dist/session-kill-http-D8JhwZVS.js +0 -121
- package/dist/session-reset-service-uoi7E4Xp.js +0 -651
- package/dist/session-status.runtime-CZK5IU8w.js +0 -2
- package/dist/session-subagent-reactivation.runtime-BSO00-FY.js +0 -2
- package/dist/session-tab-registry-DM9U7e3o.js +0 -551
- package/dist/sessions-B-SkIoaa.js +0 -1917
- package/dist/sessions-history-http-DCiOG4FK.js +0 -432
- package/dist/sessions-patch-DlAAvQvB.js +0 -401
- package/dist/sessions-resolve-DfMXookg.js +0 -180
- package/dist/sessions.runtime-0V2YxKxB.js +0 -2
- package/dist/snapshot-urls-Ble1-NEW.js +0 -317
- package/dist/speech-core-Bk60ZS_y.d.ts +0 -49
- package/dist/speech-provider-DQO9eZd0.js +0 -233
- package/dist/speech-provider-DnBCla4V.js +0 -171
- package/dist/speech-provider-DyYHFxT5.js +0 -227
- package/dist/standalone-9EWfcxeO.js +0 -42
- package/dist/startup-context-nti4X0_w.js +0 -314
- package/dist/status-subagents.runtime-CPZb1EF1.js +0 -32
- package/dist/status-text-C1Hf37lF.js +0 -301
- package/dist/stream-9VBt1MDs.js +0 -26
- package/dist/stream-B_3P7v7P.js +0 -86
- package/dist/stream-CXsue2-v.d.ts +0 -9
- package/dist/stream-oNBFxfKt.d.ts +0 -5
- package/dist/subagent-announce-CPjQQLy8.js +0 -353
- package/dist/subagent-announce-delivery-B6iBOicL.js +0 -1369
- package/dist/subagent-control-DP72sk-l.js +0 -492
- package/dist/subagent-hooks-B1oUIYH3.js +0 -2
- package/dist/subagent-hooks-BkGj4_xI.js +0 -230
- package/dist/subagent-hooks-api-D2mulK3S.js +0 -23
- package/dist/subagent-registry-CEKAUB5h.js +0 -3
- package/dist/subagent-registry-OUVucPAn.js +0 -2627
- package/dist/subagent-session-cleanup-Bx8d3kw0.js +0 -390
- package/dist/system-CelaP2zI.js +0 -111
- package/dist/talk-DGOI3Lu3.js +0 -2454
- package/dist/target-id-BXRG7x9x.js +0 -107
- package/dist/thinking-B8V29FhB.js +0 -35
- package/dist/thread-bindings-DpVdEPZ0.js +0 -228
- package/dist/tool-DHzDpxE4.js +0 -143
- package/dist/tool-dispatch-ClP3Rc7g.js +0 -155
- package/dist/tool-resolution-CZcLod1d.js +0 -153
- package/dist/tool-split-BhiQ8676.d.ts +0 -19
- package/dist/tools-ZvSvbsCW.d.ts +0 -38
- package/dist/tools-effective-C2mHZT-A.js +0 -442
- package/dist/tools-effective-inventory-ctnM7hc6.js +0 -379
- package/dist/tools-invoke-Ci6Rux2s.js +0 -51
- package/dist/tools-invoke-http-CJflXcJk.js +0 -68
- package/dist/tools-invoke-shared-BLu_mJEX.js +0 -200
- package/dist/tts-B2rPJPij.js +0 -83
- package/dist/tts-Gp9FI3_n.js +0 -163
- package/dist/tts-runtime-DNi1HXPF.d.ts +0 -230
- package/dist/tui-BUhfQ9vD.js +0 -3
- package/dist/tui-BhH5mvLf.js +0 -2
- package/dist/tui-backend-C_4ajTHI.js +0 -257
- package/dist/tui-cli-BhWJ-QoB.js +0 -40
- package/dist/tui-ink-run-BTWbUQGb.js +0 -7414
- package/dist/tui-ink-run-DfTdivkh.js +0 -2
- package/dist/types-B4fW3r5y.d.ts +0 -111
- package/dist/types-DI62NfFe.d.ts +0 -7034
- package/dist/types-sAih_uQb.d.ts +0 -393
- package/dist/types.public-B3MKhuo2.d.ts +0 -70
- package/dist/video-generation-B9c6a5cw.js +0 -207
- package/dist/video-generation-BgJp7UIA.d.ts +0 -224
- package/dist/video-generation-provider-BjiVjf40.js +0 -325
- package/dist/video-generation-provider-CsnQJg_h.js +0 -297
- package/dist/video-generation-provider-DtU-ZPqP.js +0 -64
- package/dist/video-generation-provider-wZ0bzv0e.js +0 -77
- package/dist/webhook-targets-Cy8e7y3g.d.ts +0 -99
- package/skills/canvas/SKILL.md +0 -78
- package/skills/clawhub/SKILL.md +0 -77
- package/skills/coding-agent/SKILL.md +0 -143
- package/skills/diagram-maker/SKILL.md +0 -53
- package/skills/diagram-maker/references/excalidraw-patterns.md +0 -85
- package/skills/diagram-maker/references/svg-template.md +0 -112
- package/skills/gemini/SKILL.md +0 -47
- package/skills/gh-issues/SKILL.md +0 -213
- package/skills/gifgrep/SKILL.md +0 -85
- package/skills/github/SKILL.md +0 -84
- package/skills/healthcheck/SKILL.md +0 -105
- package/skills/mcporter/SKILL.md +0 -61
- package/skills/meme-maker/SKILL.md +0 -42
- package/skills/meme-maker/references/templates.json +0 -358
- package/skills/meme-maker/scripts/meme.mjs +0 -398
- package/skills/model-usage/SKILL.md +0 -71
- package/skills/model-usage/references/codexbar-cli.md +0 -33
- package/skills/model-usage/scripts/model_usage.py +0 -319
- package/skills/model-usage/scripts/test_model_usage.py +0 -40
- package/skills/nano-pdf/SKILL.md +0 -38
- package/skills/node-connect/SKILL.md +0 -143
- package/skills/node-inspect-debugger/SKILL.md +0 -85
- package/skills/openai-whisper/SKILL.md +0 -38
- package/skills/openai-whisper-api/SKILL.md +0 -71
- package/skills/openai-whisper-api/scripts/transcribe.sh +0 -154
- package/skills/oracle/SKILL.md +0 -126
- package/skills/pyproject.toml +0 -10
- package/skills/python-debugpy/SKILL.md +0 -73
- package/skills/sag/SKILL.md +0 -87
- package/skills/session-logs/SKILL.md +0 -151
- package/skills/sherpa-onnx-tts/SKILL.md +0 -109
- package/skills/sherpa-onnx-tts/bin/sherpa-onnx-tts +0 -178
- package/skills/skill-creator/SKILL.md +0 -78
- package/skills/skill-creator/license.txt +0 -202
- package/skills/skill-creator/scripts/init_skill.py +0 -378
- package/skills/skill-creator/scripts/package_skill.py +0 -139
- package/skills/skill-creator/scripts/quick_validate.py +0 -169
- package/skills/skill-creator/scripts/test_package_skill.py +0 -161
- package/skills/skill-creator/scripts/test_quick_validate.py +0 -116
- package/skills/spike/SKILL.md +0 -51
- package/skills/summarize/SKILL.md +0 -87
- package/skills/taskflow/SKILL.md +0 -149
- package/skills/taskflow/examples/inbox-triage.lobster +0 -33
- package/skills/taskflow/examples/pr-intake.lobster +0 -32
- package/skills/taskflow-inbox-triage/SKILL.md +0 -119
- package/skills/video-frames/SKILL.md +0 -46
- package/skills/video-frames/scripts/frame.sh +0 -81
- package/skills/voice-call/SKILL.md +0 -45
- package/skills/weather/SKILL.md +0 -64
- /package/dist/{acp-runtime-backend-DbchQ02o.js → acp-runtime-backend-DZ1Lnt7f.js} +0 -0
- /package/dist/{delegate-k1aptKei.js → delegate-CwhxUdeb.js} +0 -0
- /package/dist/{dispatch-acp-CD4YxPpf.js → dispatch-acp-BP4I5ZQf.js} +0 -0
- /package/dist/{exec-approvals-ByWUCFQM.js → exec-approvals-ByWUCFQM2.js} +0 -0
- /package/dist/{heartbeat-runner-CM0UZxa_.js → heartbeat-runner-CL3alQ8-.js} +0 -0
- /package/dist/{index-B0VJdRJQ.d.ts → index-B0VJdRJQ2.d.ts} +0 -0
- /package/dist/{library-CQ71yATP.js → library-CiTr_aqC.js} +0 -0
- /package/dist/{run-session-state-DbDeH-q6.js → run-session-state-BOMUtBKZ.js} +0 -0
- /package/dist/{session-subagent-reactivation-Bj91A2ms.js → session-subagent-reactivation-CH0C2I6Y.js} +0 -0
- /package/dist/{types-C4HgagiY2.d.ts → types-C4HgagiY.d.ts} +0 -0
|
@@ -0,0 +1,3064 @@
|
|
|
1
|
+
import { a as normalizeLowercaseStringOrEmpty, c as normalizeOptionalString, p as readStringValue } from "./string-coerce-DKw2K5wM.js";
|
|
2
|
+
import { C as resolveNonNegativeIntegerOption, S as resolveIntegerOption, p as parseFiniteNumber } from "./number-coercion-D1aDmIZp.js";
|
|
3
|
+
import { i as formatErrorMessage } from "./errors-C_Wa6a5T.js";
|
|
4
|
+
import { v as uniqueValues } from "./string-normalization-B8G0vlWE.js";
|
|
5
|
+
import { m as resolveStrictExistingPathsWithinRoot, r as writeExternalFileWithinRoot } from "./fs-safe-CR_om3kX.js";
|
|
6
|
+
import { r as sanitizeUntrustedFileName } from "./sibling-temp-DOXxHtE7.js";
|
|
7
|
+
import { n as resolvePreferredFengMingTmpDir } from "./tmp-fengming-dir-0p6iJKE-.js";
|
|
8
|
+
import { t as SsrFBlockedError } from "./ssrf-DXorK84V.js";
|
|
9
|
+
import "./number-runtime-Cu8gqEYx.js";
|
|
10
|
+
import "./string-coerce-runtime-BurriTLB.js";
|
|
11
|
+
import { a as resolveActInteractionTimeoutMs, n as ACT_MAX_VIEWPORT_DIMENSION, o as resolveActWaitTimeoutMs, r as ACT_MAX_WAIT_TIME_MS, t as ACT_MAX_CLICK_DELAY_MS } from "./act-policy-gF8d1KDG.js";
|
|
12
|
+
import "./sdk-security-runtime-CM9KnGnN.js";
|
|
13
|
+
import { S as BrowserTabNotFoundError, Y as withNoProxyForCdpUrl, c as isWebSocketUrl, d as withCdpSocket, i as fetchJson, l as normalizeCdpHttpBaseForJsonEndpoints, n as assertCdpEndpointAllowed, o as getHeadersWithAuth, t as appendCdpPath } from "./cdp.helpers-DtlBcug2.js";
|
|
14
|
+
import { a as DEFAULT_DOWNLOAD_DIR, o as DEFAULT_TRACE_DIR, s as DEFAULT_UPLOAD_DIR } from "./config-BJJhHN9E.js";
|
|
15
|
+
import "./tmp-fengming-dir-Dp-T2Uh8.js";
|
|
16
|
+
import "./errors-bTbwOddL.js";
|
|
17
|
+
import { C as assertBrowserNavigationRedirectChainAllowed, E as withBrowserNavigationPolicy, S as assertBrowserNavigationAllowed, f as formatAriaSnapshot, l as AX_REF_PATTERN, p as normalizeCdpWsUrl, t as getChromeWebSocketUrl, w as assertBrowserNavigationResultAllowed, x as InvalidBrowserNavigationUrlError, y as ensureOutputDirectory } from "./chrome-B2cq8YyH.js";
|
|
18
|
+
import { a as parseRoleRef, i as getRoleSnapshotStats, l as matchBrowserUrlPattern, n as buildRoleSnapshotFromAiSnapshot, r as buildRoleSnapshotFromAriaSnapshot, t as appendSnapshotUrls } from "./snapshot-urls-C5CfP3Co.js";
|
|
19
|
+
import { n as markPwAiLoaded } from "./pw-ai-state-DL7FXLI8.js";
|
|
20
|
+
import { t as isSelectableCdpBrowserTarget } from "./cdp-target-filter-CaIsrb0G.js";
|
|
21
|
+
import { createRequire } from "node:module";
|
|
22
|
+
import path from "node:path";
|
|
23
|
+
import crypto from "node:crypto";
|
|
24
|
+
//#region extensions/browser/src/browser/output-atomic.ts
|
|
25
|
+
async function writeViaSiblingTempPath(params) {
|
|
26
|
+
await ensureOutputDirectory(params.rootDir);
|
|
27
|
+
await writeExternalFileWithinRoot({
|
|
28
|
+
rootDir: params.rootDir,
|
|
29
|
+
path: params.targetPath,
|
|
30
|
+
write: params.writeTemp
|
|
31
|
+
});
|
|
32
|
+
}
|
|
33
|
+
const playwrightCore = createRequire(import.meta.url)("playwright-core");
|
|
34
|
+
//#endregion
|
|
35
|
+
//#region extensions/browser/src/browser/pw-session.page-cdp.ts
|
|
36
|
+
const BROWSER_REF_MARKER_ATTRIBUTE = "data-fengming-browser-ref";
|
|
37
|
+
async function withPlaywrightPageCdpSession(page, fn) {
|
|
38
|
+
const session = await page.context().newCDPSession(page);
|
|
39
|
+
try {
|
|
40
|
+
return await fn(session);
|
|
41
|
+
} finally {
|
|
42
|
+
await session.detach().catch(() => {});
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
async function withPageScopedCdpClient(opts) {
|
|
46
|
+
return await withPlaywrightPageCdpSession(opts.page, async (session) => {
|
|
47
|
+
return await opts.fn((method, params) => session.send(method, params));
|
|
48
|
+
});
|
|
49
|
+
}
|
|
50
|
+
async function markBackendDomRefsOnPage(opts) {
|
|
51
|
+
await opts.page.locator(`[${BROWSER_REF_MARKER_ATTRIBUTE}]`).evaluateAll((elements, attr) => {
|
|
52
|
+
for (const element of elements) if (element instanceof Element) element.removeAttribute(attr);
|
|
53
|
+
}, BROWSER_REF_MARKER_ATTRIBUTE).catch(() => {});
|
|
54
|
+
const refs = opts.refs.filter((entry) => /^ax\d+$/.test(entry.ref) && Number.isFinite(entry.backendDOMNodeId) && Math.floor(entry.backendDOMNodeId) > 0);
|
|
55
|
+
const marked = /* @__PURE__ */ new Set();
|
|
56
|
+
if (!refs.length) return marked;
|
|
57
|
+
return await withPlaywrightPageCdpSession(opts.page, async (session) => {
|
|
58
|
+
const send = async (method, params) => await session.send(method, params);
|
|
59
|
+
await send("DOM.enable").catch(() => {});
|
|
60
|
+
const backendNodeIds = uniqueValues(refs.map((entry) => Math.floor(entry.backendDOMNodeId)));
|
|
61
|
+
const pushed = await send("DOM.pushNodesByBackendIdsToFrontend", { backendNodeIds }).catch(() => ({}));
|
|
62
|
+
const nodeIds = Array.isArray(pushed.nodeIds) ? pushed.nodeIds : [];
|
|
63
|
+
const nodeIdByBackendId = /* @__PURE__ */ new Map();
|
|
64
|
+
for (let index = 0; index < backendNodeIds.length; index += 1) {
|
|
65
|
+
const backendNodeId = backendNodeIds[index];
|
|
66
|
+
const nodeId = nodeIds[index];
|
|
67
|
+
if (backendNodeId && typeof nodeId === "number" && nodeId > 0) nodeIdByBackendId.set(backendNodeId, nodeId);
|
|
68
|
+
}
|
|
69
|
+
for (const entry of refs) {
|
|
70
|
+
const nodeId = nodeIdByBackendId.get(Math.floor(entry.backendDOMNodeId));
|
|
71
|
+
if (!nodeId) continue;
|
|
72
|
+
try {
|
|
73
|
+
await send("DOM.setAttributeValue", {
|
|
74
|
+
nodeId,
|
|
75
|
+
name: BROWSER_REF_MARKER_ATTRIBUTE,
|
|
76
|
+
value: entry.ref
|
|
77
|
+
});
|
|
78
|
+
marked.add(entry.ref);
|
|
79
|
+
} catch {}
|
|
80
|
+
}
|
|
81
|
+
return marked;
|
|
82
|
+
});
|
|
83
|
+
}
|
|
84
|
+
//#endregion
|
|
85
|
+
//#region extensions/browser/src/browser/pw-session.ts
|
|
86
|
+
const { chromium } = playwrightCore;
|
|
87
|
+
var BrowserObservedDialogBlockedError = class extends Error {
|
|
88
|
+
constructor(browserState) {
|
|
89
|
+
super("Browser action blocked by a modal dialog.");
|
|
90
|
+
this.name = "BrowserObservedDialogBlockedError";
|
|
91
|
+
this.browserState = browserState;
|
|
92
|
+
}
|
|
93
|
+
};
|
|
94
|
+
function isBrowserObservedDialogBlockedError(err) {
|
|
95
|
+
return err instanceof BrowserObservedDialogBlockedError;
|
|
96
|
+
}
|
|
97
|
+
const pageStates = /* @__PURE__ */ new WeakMap();
|
|
98
|
+
const contextStates = /* @__PURE__ */ new WeakMap();
|
|
99
|
+
const observedContexts = /* @__PURE__ */ new WeakSet();
|
|
100
|
+
const observedPages = /* @__PURE__ */ new WeakSet();
|
|
101
|
+
const roleRefsByTarget = /* @__PURE__ */ new Map();
|
|
102
|
+
const MAX_ROLE_REFS_CACHE = 50;
|
|
103
|
+
const MAX_CONSOLE_MESSAGES = 500;
|
|
104
|
+
const MAX_PAGE_ERRORS = 200;
|
|
105
|
+
const MAX_NETWORK_REQUESTS = 500;
|
|
106
|
+
const MAX_RECENT_DIALOGS = 20;
|
|
107
|
+
const OBSERVED_DIALOG_TIMEOUT_MS = 12e4;
|
|
108
|
+
const cachedByCdpUrl = /* @__PURE__ */ new Map();
|
|
109
|
+
const connectingByCdpUrl = /* @__PURE__ */ new Map();
|
|
110
|
+
const blockedTargetsByCdpUrl = /* @__PURE__ */ new Set();
|
|
111
|
+
const blockedPageRefsByCdpUrl = /* @__PURE__ */ new Map();
|
|
112
|
+
function resolveObservedDialogTimeoutMs(timeoutMs) {
|
|
113
|
+
const parsed = parseFiniteNumber(timeoutMs);
|
|
114
|
+
return Math.max(1, Math.floor(parsed ?? OBSERVED_DIALOG_TIMEOUT_MS));
|
|
115
|
+
}
|
|
116
|
+
function normalizeCdpUrl(raw) {
|
|
117
|
+
return raw.replace(/\/$/, "");
|
|
118
|
+
}
|
|
119
|
+
function buildManagedDownloadPath(fileName) {
|
|
120
|
+
const id = crypto.randomUUID();
|
|
121
|
+
const safeName = sanitizeUntrustedFileName(fileName, "download.bin");
|
|
122
|
+
return path.join(DEFAULT_DOWNLOAD_DIR, `${id}-${safeName}`);
|
|
123
|
+
}
|
|
124
|
+
function hasCachedPlaywrightBrowserConnection(cdpUrl) {
|
|
125
|
+
return cachedByCdpUrl.has(normalizeCdpUrl(cdpUrl));
|
|
126
|
+
}
|
|
127
|
+
function isRecoverablePlaywrightDisconnectError(err) {
|
|
128
|
+
const message = formatErrorMessage(err).toLowerCase();
|
|
129
|
+
return message.includes("target page, context or browser has been closed") || message.includes("browser has been closed") || message.includes("browser disconnected") || message.includes("target closed") || message.includes("connection closed") || message.includes("websocket closed") || message.includes("cdp socket closed");
|
|
130
|
+
}
|
|
131
|
+
function isRecoverableStalePageSelectionError(err, reusedCachedBrowser) {
|
|
132
|
+
if (!reusedCachedBrowser) return false;
|
|
133
|
+
if (err instanceof Error && err.message.includes("No pages available in the connected browser.")) return true;
|
|
134
|
+
if (err instanceof BrowserTabNotFoundError) return true;
|
|
135
|
+
return (err instanceof Error ? err.message : formatErrorMessage(err)).toLowerCase().includes("tab not found");
|
|
136
|
+
}
|
|
137
|
+
function findNetworkRequestById(state, id) {
|
|
138
|
+
for (let i = state.requests.length - 1; i >= 0; i -= 1) {
|
|
139
|
+
const candidate = state.requests[i];
|
|
140
|
+
if (candidate && candidate.id === id) return candidate;
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
function appendRecentDialog(state, record) {
|
|
144
|
+
state.recentDialogs.push(record);
|
|
145
|
+
while (state.recentDialogs.length > MAX_RECENT_DIALOGS) state.recentDialogs.shift();
|
|
146
|
+
}
|
|
147
|
+
function serializeDialogRecord(dialog) {
|
|
148
|
+
return {
|
|
149
|
+
id: dialog.id,
|
|
150
|
+
type: dialog.type,
|
|
151
|
+
message: dialog.message,
|
|
152
|
+
...dialog.defaultValue !== void 0 ? { defaultValue: dialog.defaultValue } : {},
|
|
153
|
+
openedAt: dialog.openedAt,
|
|
154
|
+
...dialog.closedAt !== void 0 ? { closedAt: dialog.closedAt } : {},
|
|
155
|
+
...dialog.closedBy !== void 0 ? { closedBy: dialog.closedBy } : {}
|
|
156
|
+
};
|
|
157
|
+
}
|
|
158
|
+
function serializePendingDialog(dialog) {
|
|
159
|
+
return serializeDialogRecord(dialog);
|
|
160
|
+
}
|
|
161
|
+
function serializeObservedBrowserState(state) {
|
|
162
|
+
return { dialogs: {
|
|
163
|
+
pending: state.pendingDialogs.map(serializePendingDialog),
|
|
164
|
+
recent: state.recentDialogs.map(serializeDialogRecord)
|
|
165
|
+
} };
|
|
166
|
+
}
|
|
167
|
+
function clearArmedDialogResponse(state) {
|
|
168
|
+
if (state.armedDialogResponse?.timer) clearTimeout(state.armedDialogResponse.timer);
|
|
169
|
+
state.armedDialogResponse = void 0;
|
|
170
|
+
}
|
|
171
|
+
function abortActionsBlockedByDialog(state) {
|
|
172
|
+
if (state.dialogAbortControllers.size === 0) return;
|
|
173
|
+
const err = new BrowserObservedDialogBlockedError(serializeObservedBrowserState(state));
|
|
174
|
+
for (const controller of state.dialogAbortControllers) if (!controller.signal.aborted) controller.abort(err);
|
|
175
|
+
state.dialogAbortControllers.clear();
|
|
176
|
+
}
|
|
177
|
+
function isNoDialogShowingError(err) {
|
|
178
|
+
return (err instanceof Error ? err.message : String(err)).toLowerCase().includes("no dialog is showing");
|
|
179
|
+
}
|
|
180
|
+
async function settleObservedDialog(params) {
|
|
181
|
+
const { state, pending } = params;
|
|
182
|
+
state.pendingDialogs = state.pendingDialogs.filter((dialog) => dialog.id !== pending.id);
|
|
183
|
+
let closedBy = params.closedBy;
|
|
184
|
+
try {
|
|
185
|
+
if (params.accept) await pending.dialog.accept(params.promptText);
|
|
186
|
+
else await pending.dialog.dismiss();
|
|
187
|
+
} catch (err) {
|
|
188
|
+
if (!isNoDialogShowingError(err)) {
|
|
189
|
+
if (params.closedBy === "agent") state.pendingDialogs.push(pending);
|
|
190
|
+
throw err;
|
|
191
|
+
}
|
|
192
|
+
closedBy = "remote";
|
|
193
|
+
}
|
|
194
|
+
const record = {
|
|
195
|
+
id: pending.id,
|
|
196
|
+
type: pending.type,
|
|
197
|
+
message: pending.message,
|
|
198
|
+
...pending.defaultValue !== void 0 ? { defaultValue: pending.defaultValue } : {},
|
|
199
|
+
openedAt: pending.openedAt,
|
|
200
|
+
closedAt: (/* @__PURE__ */ new Date()).toISOString(),
|
|
201
|
+
closedBy
|
|
202
|
+
};
|
|
203
|
+
appendRecentDialog(state, record);
|
|
204
|
+
return record;
|
|
205
|
+
}
|
|
206
|
+
function observeDialog(pageState, dialog) {
|
|
207
|
+
pageState.nextObservedDialogId += 1;
|
|
208
|
+
const type = dialog.type();
|
|
209
|
+
const defaultValue = dialog.defaultValue();
|
|
210
|
+
const pending = {
|
|
211
|
+
id: `d${pageState.nextObservedDialogId}`,
|
|
212
|
+
type,
|
|
213
|
+
message: dialog.message(),
|
|
214
|
+
openedAt: (/* @__PURE__ */ new Date()).toISOString(),
|
|
215
|
+
dialog,
|
|
216
|
+
...type === "prompt" ? { defaultValue } : {}
|
|
217
|
+
};
|
|
218
|
+
pageState.pendingDialogs.push(pending);
|
|
219
|
+
const armed = pageState.armedDialogResponse;
|
|
220
|
+
if (armed && armed.expiresAt >= Date.now()) {
|
|
221
|
+
clearArmedDialogResponse(pageState);
|
|
222
|
+
settleObservedDialog({
|
|
223
|
+
state: pageState,
|
|
224
|
+
pending,
|
|
225
|
+
accept: armed.accept,
|
|
226
|
+
...armed.promptText !== void 0 ? { promptText: armed.promptText } : {},
|
|
227
|
+
closedBy: "armed"
|
|
228
|
+
}).catch(() => {});
|
|
229
|
+
return;
|
|
230
|
+
}
|
|
231
|
+
if (armed) clearArmedDialogResponse(pageState);
|
|
232
|
+
abortActionsBlockedByDialog(pageState);
|
|
233
|
+
}
|
|
234
|
+
function targetKey(cdpUrl, targetId) {
|
|
235
|
+
return `${normalizeCdpUrl(cdpUrl)}::${targetId}`;
|
|
236
|
+
}
|
|
237
|
+
function roleRefsKey(cdpUrl, targetId) {
|
|
238
|
+
return targetKey(cdpUrl, targetId);
|
|
239
|
+
}
|
|
240
|
+
function isBlockedTarget(cdpUrl, targetId) {
|
|
241
|
+
const normalizedTargetId = normalizeOptionalString(targetId) ?? "";
|
|
242
|
+
if (!normalizedTargetId) return false;
|
|
243
|
+
return blockedTargetsByCdpUrl.has(targetKey(cdpUrl, normalizedTargetId));
|
|
244
|
+
}
|
|
245
|
+
function markTargetBlocked(cdpUrl, targetId) {
|
|
246
|
+
const normalizedTargetId = normalizeOptionalString(targetId) ?? "";
|
|
247
|
+
if (!normalizedTargetId) return;
|
|
248
|
+
blockedTargetsByCdpUrl.add(targetKey(cdpUrl, normalizedTargetId));
|
|
249
|
+
}
|
|
250
|
+
function clearBlockedTarget(cdpUrl, targetId) {
|
|
251
|
+
const normalizedTargetId = normalizeOptionalString(targetId) ?? "";
|
|
252
|
+
if (!normalizedTargetId) return;
|
|
253
|
+
blockedTargetsByCdpUrl.delete(targetKey(cdpUrl, normalizedTargetId));
|
|
254
|
+
}
|
|
255
|
+
function clearBlockedTargetsForCdpUrl(cdpUrl) {
|
|
256
|
+
if (!cdpUrl) {
|
|
257
|
+
blockedTargetsByCdpUrl.clear();
|
|
258
|
+
return;
|
|
259
|
+
}
|
|
260
|
+
const prefix = `${normalizeCdpUrl(cdpUrl)}::`;
|
|
261
|
+
for (const key of blockedTargetsByCdpUrl) if (key.startsWith(prefix)) blockedTargetsByCdpUrl.delete(key);
|
|
262
|
+
}
|
|
263
|
+
function blockedPageRefsForCdpUrl(cdpUrl) {
|
|
264
|
+
const normalized = normalizeCdpUrl(cdpUrl);
|
|
265
|
+
const existing = blockedPageRefsByCdpUrl.get(normalized);
|
|
266
|
+
if (existing) return existing;
|
|
267
|
+
const created = /* @__PURE__ */ new WeakSet();
|
|
268
|
+
blockedPageRefsByCdpUrl.set(normalized, created);
|
|
269
|
+
return created;
|
|
270
|
+
}
|
|
271
|
+
function isBlockedPageRef(cdpUrl, page) {
|
|
272
|
+
return blockedPageRefsByCdpUrl.get(normalizeCdpUrl(cdpUrl))?.has(page) ?? false;
|
|
273
|
+
}
|
|
274
|
+
function markPageRefBlocked(cdpUrl, page) {
|
|
275
|
+
blockedPageRefsForCdpUrl(cdpUrl).add(page);
|
|
276
|
+
}
|
|
277
|
+
function clearBlockedPageRefsForCdpUrl(cdpUrl) {
|
|
278
|
+
if (!cdpUrl) {
|
|
279
|
+
blockedPageRefsByCdpUrl.clear();
|
|
280
|
+
return;
|
|
281
|
+
}
|
|
282
|
+
blockedPageRefsByCdpUrl.delete(normalizeCdpUrl(cdpUrl));
|
|
283
|
+
}
|
|
284
|
+
function clearBlockedPageRef(cdpUrl, page) {
|
|
285
|
+
blockedPageRefsByCdpUrl.get(normalizeCdpUrl(cdpUrl))?.delete(page);
|
|
286
|
+
}
|
|
287
|
+
function takeCachedPlaywrightBrowserConnection(cdpUrl) {
|
|
288
|
+
const normalized = normalizeCdpUrl(cdpUrl);
|
|
289
|
+
const cur = cachedByCdpUrl.get(normalized);
|
|
290
|
+
cachedByCdpUrl.delete(normalized);
|
|
291
|
+
connectingByCdpUrl.delete(normalized);
|
|
292
|
+
if (!cur) return null;
|
|
293
|
+
if (cur.onDisconnected && typeof cur.browser.off === "function") cur.browser.off("disconnected", cur.onDisconnected);
|
|
294
|
+
return cur;
|
|
295
|
+
}
|
|
296
|
+
function evictStalePlaywrightBrowserConnection(cdpUrl) {
|
|
297
|
+
takeCachedPlaywrightBrowserConnection(cdpUrl)?.browser.close().catch(() => {});
|
|
298
|
+
}
|
|
299
|
+
function hasBlockedTargetsForCdpUrl(cdpUrl) {
|
|
300
|
+
const prefix = `${normalizeCdpUrl(cdpUrl)}::`;
|
|
301
|
+
for (const key of blockedTargetsByCdpUrl) if (key.startsWith(prefix)) return true;
|
|
302
|
+
return false;
|
|
303
|
+
}
|
|
304
|
+
var BlockedBrowserTargetError = class extends Error {
|
|
305
|
+
constructor() {
|
|
306
|
+
super("Browser target is unavailable after SSRF policy blocked its navigation.");
|
|
307
|
+
this.name = "BlockedBrowserTargetError";
|
|
308
|
+
}
|
|
309
|
+
};
|
|
310
|
+
function rememberRoleRefsForTarget(opts) {
|
|
311
|
+
const targetId = normalizeOptionalString(opts.targetId) ?? "";
|
|
312
|
+
if (!targetId) return;
|
|
313
|
+
roleRefsByTarget.set(roleRefsKey(opts.cdpUrl, targetId), {
|
|
314
|
+
refs: opts.refs,
|
|
315
|
+
...opts.frameSelector ? { frameSelector: opts.frameSelector } : {},
|
|
316
|
+
...opts.mode ? { mode: opts.mode } : {}
|
|
317
|
+
});
|
|
318
|
+
while (roleRefsByTarget.size > MAX_ROLE_REFS_CACHE) {
|
|
319
|
+
const first = roleRefsByTarget.keys().next();
|
|
320
|
+
if (first.done) break;
|
|
321
|
+
roleRefsByTarget.delete(first.value);
|
|
322
|
+
}
|
|
323
|
+
}
|
|
324
|
+
function storeRoleRefsForTarget(opts) {
|
|
325
|
+
const state = ensurePageState(opts.page);
|
|
326
|
+
state.roleRefs = opts.refs;
|
|
327
|
+
state.roleRefsFrameSelector = opts.frameSelector;
|
|
328
|
+
state.roleRefsMode = opts.mode;
|
|
329
|
+
const targetId = normalizeOptionalString(opts.targetId);
|
|
330
|
+
if (!targetId) return;
|
|
331
|
+
rememberRoleRefsForTarget({
|
|
332
|
+
cdpUrl: opts.cdpUrl,
|
|
333
|
+
targetId,
|
|
334
|
+
refs: opts.refs,
|
|
335
|
+
frameSelector: opts.frameSelector,
|
|
336
|
+
mode: opts.mode
|
|
337
|
+
});
|
|
338
|
+
}
|
|
339
|
+
function restoreRoleRefsForTarget(opts) {
|
|
340
|
+
const targetId = normalizeOptionalString(opts.targetId) ?? "";
|
|
341
|
+
if (!targetId) return;
|
|
342
|
+
const cached = roleRefsByTarget.get(roleRefsKey(opts.cdpUrl, targetId));
|
|
343
|
+
if (!cached) return;
|
|
344
|
+
const state = ensurePageState(opts.page);
|
|
345
|
+
if (state.roleRefs) return;
|
|
346
|
+
state.roleRefs = cached.refs;
|
|
347
|
+
state.roleRefsFrameSelector = cached.frameSelector;
|
|
348
|
+
state.roleRefsMode = cached.mode;
|
|
349
|
+
}
|
|
350
|
+
function ensurePageState(page) {
|
|
351
|
+
const existing = pageStates.get(page);
|
|
352
|
+
if (existing) return existing;
|
|
353
|
+
const state = {
|
|
354
|
+
console: [],
|
|
355
|
+
errors: [],
|
|
356
|
+
requests: [],
|
|
357
|
+
requestIds: /* @__PURE__ */ new WeakMap(),
|
|
358
|
+
nextRequestId: 0,
|
|
359
|
+
armIdUpload: 0,
|
|
360
|
+
armIdDownload: 0,
|
|
361
|
+
downloadWaiterDepth: 0,
|
|
362
|
+
nextObservedDialogId: 0,
|
|
363
|
+
pendingDialogs: [],
|
|
364
|
+
recentDialogs: [],
|
|
365
|
+
dialogAbortControllers: /* @__PURE__ */ new Set()
|
|
366
|
+
};
|
|
367
|
+
pageStates.set(page, state);
|
|
368
|
+
if (!observedPages.has(page)) {
|
|
369
|
+
observedPages.add(page);
|
|
370
|
+
page.on("console", (msg) => {
|
|
371
|
+
const entry = {
|
|
372
|
+
type: msg.type(),
|
|
373
|
+
text: msg.text(),
|
|
374
|
+
timestamp: (/* @__PURE__ */ new Date()).toISOString(),
|
|
375
|
+
location: msg.location()
|
|
376
|
+
};
|
|
377
|
+
state.console.push(entry);
|
|
378
|
+
if (state.console.length > MAX_CONSOLE_MESSAGES) state.console.shift();
|
|
379
|
+
});
|
|
380
|
+
page.on("pageerror", (err) => {
|
|
381
|
+
state.errors.push({
|
|
382
|
+
message: err.message || String(err),
|
|
383
|
+
name: err.name || void 0,
|
|
384
|
+
stack: err.stack || void 0,
|
|
385
|
+
timestamp: (/* @__PURE__ */ new Date()).toISOString()
|
|
386
|
+
});
|
|
387
|
+
if (state.errors.length > MAX_PAGE_ERRORS) state.errors.shift();
|
|
388
|
+
});
|
|
389
|
+
page.on("request", (req) => {
|
|
390
|
+
state.nextRequestId += 1;
|
|
391
|
+
const id = `r${state.nextRequestId}`;
|
|
392
|
+
state.requestIds.set(req, id);
|
|
393
|
+
state.requests.push({
|
|
394
|
+
id,
|
|
395
|
+
timestamp: (/* @__PURE__ */ new Date()).toISOString(),
|
|
396
|
+
method: req.method(),
|
|
397
|
+
url: req.url(),
|
|
398
|
+
resourceType: req.resourceType()
|
|
399
|
+
});
|
|
400
|
+
if (state.requests.length > MAX_NETWORK_REQUESTS) state.requests.shift();
|
|
401
|
+
});
|
|
402
|
+
page.on("response", (resp) => {
|
|
403
|
+
const req = resp.request();
|
|
404
|
+
const id = state.requestIds.get(req);
|
|
405
|
+
if (!id) return;
|
|
406
|
+
const rec = findNetworkRequestById(state, id);
|
|
407
|
+
if (!rec) return;
|
|
408
|
+
rec.status = resp.status();
|
|
409
|
+
rec.ok = resp.ok();
|
|
410
|
+
});
|
|
411
|
+
page.on("requestfailed", (req) => {
|
|
412
|
+
const id = state.requestIds.get(req);
|
|
413
|
+
if (!id) return;
|
|
414
|
+
const rec = findNetworkRequestById(state, id);
|
|
415
|
+
if (!rec) return;
|
|
416
|
+
rec.failureText = req.failure()?.errorText;
|
|
417
|
+
rec.ok = false;
|
|
418
|
+
});
|
|
419
|
+
page.on("dialog", (dialog) => {
|
|
420
|
+
observeDialog(state, dialog);
|
|
421
|
+
});
|
|
422
|
+
page.on("download", (download) => {
|
|
423
|
+
if (state.downloadWaiterDepth > 0) return;
|
|
424
|
+
const managedPath = buildManagedDownloadPath(sanitizeUntrustedFileName(download.suggestedFilename?.() || "download.bin", "download.bin"));
|
|
425
|
+
const managedSave = (async () => {
|
|
426
|
+
await writeViaSiblingTempPath({
|
|
427
|
+
rootDir: DEFAULT_DOWNLOAD_DIR,
|
|
428
|
+
targetPath: managedPath,
|
|
429
|
+
writeTemp: async (tempPath) => {
|
|
430
|
+
await download.saveAs?.(tempPath);
|
|
431
|
+
}
|
|
432
|
+
});
|
|
433
|
+
return managedPath;
|
|
434
|
+
})();
|
|
435
|
+
managedSave.catch(() => {});
|
|
436
|
+
download.path = async () => await managedSave;
|
|
437
|
+
});
|
|
438
|
+
page.on("close", () => {
|
|
439
|
+
clearArmedDialogResponse(state);
|
|
440
|
+
for (const controller of state.dialogAbortControllers) if (!controller.signal.aborted) controller.abort(/* @__PURE__ */ new Error("页面在浏览器操作完成前已关闭"));
|
|
441
|
+
state.dialogAbortControllers.clear();
|
|
442
|
+
state.pendingDialogs = [];
|
|
443
|
+
pageStates.delete(page);
|
|
444
|
+
observedPages.delete(page);
|
|
445
|
+
});
|
|
446
|
+
}
|
|
447
|
+
return state;
|
|
448
|
+
}
|
|
449
|
+
function getObservedBrowserStateForPage(page) {
|
|
450
|
+
return serializeObservedBrowserState(ensurePageState(page));
|
|
451
|
+
}
|
|
452
|
+
async function getObservedBrowserStateViaPlaywright(opts) {
|
|
453
|
+
return getObservedBrowserStateForPage(await getPageForTargetId(opts));
|
|
454
|
+
}
|
|
455
|
+
function resolvePendingDialogForResponse(params) {
|
|
456
|
+
const dialogId = normalizeOptionalString(params.dialogId);
|
|
457
|
+
if (dialogId) {
|
|
458
|
+
const found = params.state.pendingDialogs.find((dialog) => dialog.id === dialogId);
|
|
459
|
+
if (found) return found;
|
|
460
|
+
throw new Error(`Dialog "${dialogId}" is not pending.`);
|
|
461
|
+
}
|
|
462
|
+
if (params.state.pendingDialogs.length === 1) return params.state.pendingDialogs[0];
|
|
463
|
+
if (params.state.pendingDialogs.length > 1) throw new Error("存在多个待处理对话框,请传入 dialogId");
|
|
464
|
+
throw new Error("没有待处理的对话框");
|
|
465
|
+
}
|
|
466
|
+
async function respondToObservedDialogOnPage(opts) {
|
|
467
|
+
const state = ensurePageState(opts.page);
|
|
468
|
+
return await settleObservedDialog({
|
|
469
|
+
state,
|
|
470
|
+
pending: resolvePendingDialogForResponse({
|
|
471
|
+
state,
|
|
472
|
+
...opts.dialogId !== void 0 ? { dialogId: opts.dialogId } : {}
|
|
473
|
+
}),
|
|
474
|
+
accept: opts.accept,
|
|
475
|
+
...opts.promptText !== void 0 ? { promptText: opts.promptText } : {},
|
|
476
|
+
closedBy: opts.closedBy ?? "agent"
|
|
477
|
+
});
|
|
478
|
+
}
|
|
479
|
+
async function respondToObservedDialogViaPlaywright(opts) {
|
|
480
|
+
return await respondToObservedDialogOnPage({
|
|
481
|
+
page: await getPageForTargetId(opts),
|
|
482
|
+
accept: opts.accept,
|
|
483
|
+
...opts.dialogId !== void 0 ? { dialogId: opts.dialogId } : {},
|
|
484
|
+
...opts.promptText !== void 0 ? { promptText: opts.promptText } : {}
|
|
485
|
+
});
|
|
486
|
+
}
|
|
487
|
+
function markObservedDialogsHandledRemotelyForPage(page) {
|
|
488
|
+
const state = ensurePageState(page);
|
|
489
|
+
const pending = state.pendingDialogs.splice(0);
|
|
490
|
+
const closedAt = (/* @__PURE__ */ new Date()).toISOString();
|
|
491
|
+
for (const dialog of pending) appendRecentDialog(state, {
|
|
492
|
+
id: dialog.id,
|
|
493
|
+
type: dialog.type,
|
|
494
|
+
message: dialog.message,
|
|
495
|
+
...dialog.defaultValue !== void 0 ? { defaultValue: dialog.defaultValue } : {},
|
|
496
|
+
openedAt: dialog.openedAt,
|
|
497
|
+
closedAt,
|
|
498
|
+
closedBy: "remote"
|
|
499
|
+
});
|
|
500
|
+
return serializeObservedBrowserState(state);
|
|
501
|
+
}
|
|
502
|
+
function armObservedDialogResponseOnPage(opts) {
|
|
503
|
+
const state = ensurePageState(opts.page);
|
|
504
|
+
clearArmedDialogResponse(state);
|
|
505
|
+
const timeoutMs = resolveObservedDialogTimeoutMs(opts.timeoutMs);
|
|
506
|
+
const response = {
|
|
507
|
+
accept: opts.accept,
|
|
508
|
+
expiresAt: Date.now() + timeoutMs,
|
|
509
|
+
...opts.promptText !== void 0 ? { promptText: opts.promptText } : {}
|
|
510
|
+
};
|
|
511
|
+
response.timer = setTimeout(() => {
|
|
512
|
+
if (state.armedDialogResponse === response) state.armedDialogResponse = void 0;
|
|
513
|
+
}, timeoutMs);
|
|
514
|
+
state.armedDialogResponse = response;
|
|
515
|
+
}
|
|
516
|
+
function createObservedDialogAbortSignalForPage(opts) {
|
|
517
|
+
const state = ensurePageState(opts.page);
|
|
518
|
+
const controller = new AbortController();
|
|
519
|
+
const abortForCurrentDialog = () => {
|
|
520
|
+
if (!controller.signal.aborted) controller.abort(new BrowserObservedDialogBlockedError(serializeObservedBrowserState(state)));
|
|
521
|
+
};
|
|
522
|
+
const abortForParent = () => {
|
|
523
|
+
if (!controller.signal.aborted) controller.abort(opts.parentSignal?.reason ?? /* @__PURE__ */ new Error("已中止"));
|
|
524
|
+
};
|
|
525
|
+
if (state.pendingDialogs.length > 0) abortForCurrentDialog();
|
|
526
|
+
else state.dialogAbortControllers.add(controller);
|
|
527
|
+
if (opts.parentSignal) if (opts.parentSignal.aborted) abortForParent();
|
|
528
|
+
else opts.parentSignal.addEventListener("abort", abortForParent, { once: true });
|
|
529
|
+
return {
|
|
530
|
+
signal: controller.signal,
|
|
531
|
+
cleanup: () => {
|
|
532
|
+
state.dialogAbortControllers.delete(controller);
|
|
533
|
+
opts.parentSignal?.removeEventListener("abort", abortForParent);
|
|
534
|
+
}
|
|
535
|
+
};
|
|
536
|
+
}
|
|
537
|
+
function observeContext(context) {
|
|
538
|
+
if (observedContexts.has(context)) return;
|
|
539
|
+
observedContexts.add(context);
|
|
540
|
+
ensureContextState(context);
|
|
541
|
+
for (const page of context.pages()) ensurePageState(page);
|
|
542
|
+
context.on("page", (page) => ensurePageState(page));
|
|
543
|
+
}
|
|
544
|
+
function ensureContextState(context) {
|
|
545
|
+
const existing = contextStates.get(context);
|
|
546
|
+
if (existing) return existing;
|
|
547
|
+
const state = { traceActive: false };
|
|
548
|
+
contextStates.set(context, state);
|
|
549
|
+
return state;
|
|
550
|
+
}
|
|
551
|
+
function observeBrowser(browser) {
|
|
552
|
+
for (const context of browser.contexts()) observeContext(context);
|
|
553
|
+
}
|
|
554
|
+
async function connectBrowser(cdpUrl, ssrfPolicy) {
|
|
555
|
+
const normalized = normalizeCdpUrl(cdpUrl);
|
|
556
|
+
const cached = cachedByCdpUrl.get(normalized);
|
|
557
|
+
if (cached) return cached;
|
|
558
|
+
await assertCdpEndpointAllowed(normalized, ssrfPolicy);
|
|
559
|
+
const connecting = connectingByCdpUrl.get(normalized);
|
|
560
|
+
if (connecting) return await connecting;
|
|
561
|
+
const connectWithRetry = async () => {
|
|
562
|
+
let lastErr;
|
|
563
|
+
for (let attempt = 0; attempt < 3; attempt += 1) try {
|
|
564
|
+
const timeout = 5e3 + attempt * 2e3;
|
|
565
|
+
const endpoint = await getChromeWebSocketUrl(normalized, timeout, ssrfPolicy).catch(() => null) ?? normalized;
|
|
566
|
+
const connectEndpoint = async (target) => {
|
|
567
|
+
const headers = getHeadersWithAuth(target);
|
|
568
|
+
return await withNoProxyForCdpUrl(target, () => chromium.connectOverCDP(target, {
|
|
569
|
+
timeout,
|
|
570
|
+
headers
|
|
571
|
+
}));
|
|
572
|
+
};
|
|
573
|
+
let browser;
|
|
574
|
+
try {
|
|
575
|
+
browser = await connectEndpoint(endpoint);
|
|
576
|
+
} catch (err) {
|
|
577
|
+
if (!isWebSocketUrl(normalized) || endpoint === normalized) throw err;
|
|
578
|
+
browser = await connectEndpoint(normalized);
|
|
579
|
+
}
|
|
580
|
+
const onDisconnected = () => {
|
|
581
|
+
if (cachedByCdpUrl.get(normalized)?.browser === browser) cachedByCdpUrl.delete(normalized);
|
|
582
|
+
};
|
|
583
|
+
const connected = {
|
|
584
|
+
browser,
|
|
585
|
+
cdpUrl: normalized,
|
|
586
|
+
onDisconnected
|
|
587
|
+
};
|
|
588
|
+
cachedByCdpUrl.set(normalized, connected);
|
|
589
|
+
browser.on("disconnected", onDisconnected);
|
|
590
|
+
observeBrowser(browser);
|
|
591
|
+
return connected;
|
|
592
|
+
} catch (err) {
|
|
593
|
+
lastErr = err;
|
|
594
|
+
if (formatErrorMessage(err).includes("rate limit")) break;
|
|
595
|
+
const delay = 250 + attempt * 250;
|
|
596
|
+
await new Promise((r) => setTimeout(r, delay));
|
|
597
|
+
}
|
|
598
|
+
if (lastErr instanceof Error) throw lastErr;
|
|
599
|
+
const message = lastErr ? formatErrorMessage(lastErr) : "CDP connect failed";
|
|
600
|
+
throw new Error(message);
|
|
601
|
+
};
|
|
602
|
+
const pending = connectWithRetry().finally(() => {
|
|
603
|
+
connectingByCdpUrl.delete(normalized);
|
|
604
|
+
});
|
|
605
|
+
connectingByCdpUrl.set(normalized, pending);
|
|
606
|
+
return await pending;
|
|
607
|
+
}
|
|
608
|
+
async function getAllPages(browser) {
|
|
609
|
+
return browser.contexts().flatMap((c) => c.pages());
|
|
610
|
+
}
|
|
611
|
+
async function partitionAccessiblePages(opts) {
|
|
612
|
+
const accessible = [];
|
|
613
|
+
let blockedCount = 0;
|
|
614
|
+
for (const page of opts.pages) {
|
|
615
|
+
if (isBlockedPageRef(opts.cdpUrl, page)) {
|
|
616
|
+
blockedCount += 1;
|
|
617
|
+
continue;
|
|
618
|
+
}
|
|
619
|
+
const targetId = await pageTargetId(page).catch(() => null);
|
|
620
|
+
if (!targetId) {
|
|
621
|
+
if (hasBlockedTargetsForCdpUrl(opts.cdpUrl)) {
|
|
622
|
+
blockedCount += 1;
|
|
623
|
+
continue;
|
|
624
|
+
}
|
|
625
|
+
accessible.push(page);
|
|
626
|
+
continue;
|
|
627
|
+
}
|
|
628
|
+
if (isBlockedTarget(opts.cdpUrl, targetId)) {
|
|
629
|
+
blockedCount += 1;
|
|
630
|
+
continue;
|
|
631
|
+
}
|
|
632
|
+
accessible.push(page);
|
|
633
|
+
}
|
|
634
|
+
return {
|
|
635
|
+
accessible,
|
|
636
|
+
blockedCount
|
|
637
|
+
};
|
|
638
|
+
}
|
|
639
|
+
async function pageTargetId(page) {
|
|
640
|
+
const session = await page.context().newCDPSession(page);
|
|
641
|
+
try {
|
|
642
|
+
return (normalizeOptionalString((await session.send("Target.getTargetInfo"))?.targetInfo?.targetId) ?? "") || null;
|
|
643
|
+
} finally {
|
|
644
|
+
await session.detach().catch(() => {});
|
|
645
|
+
}
|
|
646
|
+
}
|
|
647
|
+
function matchPageByTargetList(pages, targets, targetId) {
|
|
648
|
+
const target = targets.find((entry) => entry.id === targetId);
|
|
649
|
+
if (!target) return null;
|
|
650
|
+
const urlMatch = pages.filter((page) => page.url() === target.url);
|
|
651
|
+
if (urlMatch.length === 1) return urlMatch[0] ?? null;
|
|
652
|
+
if (urlMatch.length > 1) {
|
|
653
|
+
const sameUrlTargets = targets.filter((entry) => entry.url === target.url);
|
|
654
|
+
if (sameUrlTargets.length === urlMatch.length) {
|
|
655
|
+
const idx = sameUrlTargets.findIndex((entry) => entry.id === targetId);
|
|
656
|
+
if (idx >= 0 && idx < urlMatch.length) return urlMatch[idx] ?? null;
|
|
657
|
+
}
|
|
658
|
+
}
|
|
659
|
+
return null;
|
|
660
|
+
}
|
|
661
|
+
async function findPageByTargetIdViaTargetList(pages, targetId, cdpUrl, ssrfPolicy) {
|
|
662
|
+
const cdpHttpBase = normalizeCdpHttpBaseForJsonEndpoints(cdpUrl);
|
|
663
|
+
await assertCdpEndpointAllowed(cdpUrl, ssrfPolicy);
|
|
664
|
+
return matchPageByTargetList(pages, await fetchJson(appendCdpPath(cdpHttpBase, "/json/list"), 2e3), targetId);
|
|
665
|
+
}
|
|
666
|
+
async function findPageByTargetId(browser, targetId, cdpUrl, ssrfPolicy) {
|
|
667
|
+
const pages = await getAllPages(browser);
|
|
668
|
+
let resolvedViaCdp = false;
|
|
669
|
+
for (const page of pages) {
|
|
670
|
+
let tid = null;
|
|
671
|
+
try {
|
|
672
|
+
tid = await pageTargetId(page);
|
|
673
|
+
resolvedViaCdp = true;
|
|
674
|
+
} catch {
|
|
675
|
+
tid = null;
|
|
676
|
+
}
|
|
677
|
+
if (tid && tid === targetId) return page;
|
|
678
|
+
}
|
|
679
|
+
if (cdpUrl) try {
|
|
680
|
+
return await findPageByTargetIdViaTargetList(pages, targetId, cdpUrl, ssrfPolicy);
|
|
681
|
+
} catch {}
|
|
682
|
+
if (!resolvedViaCdp && pages.length === 1) return pages[0] ?? null;
|
|
683
|
+
return null;
|
|
684
|
+
}
|
|
685
|
+
async function resolvePageByTargetIdOrThrow(opts) {
|
|
686
|
+
if (isBlockedTarget(opts.cdpUrl, opts.targetId)) throw new BlockedBrowserTargetError();
|
|
687
|
+
const { browser } = await connectBrowser(opts.cdpUrl, opts.ssrfPolicy);
|
|
688
|
+
const page = await findPageByTargetId(browser, opts.targetId, opts.cdpUrl, opts.ssrfPolicy);
|
|
689
|
+
if (!page) throw new BrowserTabNotFoundError();
|
|
690
|
+
return page;
|
|
691
|
+
}
|
|
692
|
+
async function getPageForTargetIdOnce(opts) {
|
|
693
|
+
if (opts.targetId && isBlockedTarget(opts.cdpUrl, opts.targetId)) throw new BlockedBrowserTargetError();
|
|
694
|
+
const { browser } = await connectBrowser(opts.cdpUrl, opts.ssrfPolicy);
|
|
695
|
+
const pages = await getAllPages(browser);
|
|
696
|
+
if (!pages.length) throw new Error("连接到的浏览器中没有可用页面");
|
|
697
|
+
const { accessible, blockedCount } = await partitionAccessiblePages({
|
|
698
|
+
cdpUrl: opts.cdpUrl,
|
|
699
|
+
pages
|
|
700
|
+
});
|
|
701
|
+
if (!accessible.length) {
|
|
702
|
+
if (blockedCount > 0) throw new BlockedBrowserTargetError();
|
|
703
|
+
throw new Error("连接到的浏览器中没有可用页面");
|
|
704
|
+
}
|
|
705
|
+
const first = accessible[0];
|
|
706
|
+
if (!opts.targetId) return first;
|
|
707
|
+
const found = await findPageByTargetId(browser, opts.targetId, opts.cdpUrl, opts.ssrfPolicy);
|
|
708
|
+
if (found) {
|
|
709
|
+
if (isBlockedPageRef(opts.cdpUrl, found)) throw new BlockedBrowserTargetError();
|
|
710
|
+
const foundTargetId = await pageTargetId(found).catch(() => null);
|
|
711
|
+
if (foundTargetId && isBlockedTarget(opts.cdpUrl, foundTargetId)) throw new BlockedBrowserTargetError();
|
|
712
|
+
return found;
|
|
713
|
+
}
|
|
714
|
+
if (pages.length === 1) return first;
|
|
715
|
+
throw new BrowserTabNotFoundError();
|
|
716
|
+
}
|
|
717
|
+
async function getPageForTargetId(opts) {
|
|
718
|
+
const reusedCachedBrowser = hasCachedPlaywrightBrowserConnection(opts.cdpUrl);
|
|
719
|
+
try {
|
|
720
|
+
return await getPageForTargetIdOnce(opts);
|
|
721
|
+
} catch (err) {
|
|
722
|
+
if (!isRecoverableStalePageSelectionError(err, reusedCachedBrowser)) throw err;
|
|
723
|
+
await closePlaywrightBrowserConnection({ cdpUrl: opts.cdpUrl });
|
|
724
|
+
return await getPageForTargetIdOnce(opts);
|
|
725
|
+
}
|
|
726
|
+
}
|
|
727
|
+
function isTopLevelNavigationRequest(page, request) {
|
|
728
|
+
let sameMainFrame = false;
|
|
729
|
+
try {
|
|
730
|
+
sameMainFrame = request.frame() === page.mainFrame();
|
|
731
|
+
} catch {
|
|
732
|
+
sameMainFrame = true;
|
|
733
|
+
}
|
|
734
|
+
if (!sameMainFrame) return false;
|
|
735
|
+
try {
|
|
736
|
+
if (request.isNavigationRequest()) return true;
|
|
737
|
+
} catch {}
|
|
738
|
+
try {
|
|
739
|
+
return request.resourceType() === "document";
|
|
740
|
+
} catch {
|
|
741
|
+
return false;
|
|
742
|
+
}
|
|
743
|
+
}
|
|
744
|
+
function isSubframeDocumentNavigationRequest(page, request) {
|
|
745
|
+
let sameMainFrame = false;
|
|
746
|
+
try {
|
|
747
|
+
sameMainFrame = request.frame() === page.mainFrame();
|
|
748
|
+
} catch {
|
|
749
|
+
return true;
|
|
750
|
+
}
|
|
751
|
+
if (sameMainFrame) return false;
|
|
752
|
+
try {
|
|
753
|
+
if (request.isNavigationRequest()) return true;
|
|
754
|
+
} catch {}
|
|
755
|
+
try {
|
|
756
|
+
return request.resourceType() === "document";
|
|
757
|
+
} catch {
|
|
758
|
+
return false;
|
|
759
|
+
}
|
|
760
|
+
}
|
|
761
|
+
function isPolicyDenyNavigationError(err) {
|
|
762
|
+
return err instanceof SsrFBlockedError || err instanceof InvalidBrowserNavigationUrlError;
|
|
763
|
+
}
|
|
764
|
+
async function quarantineBlockedTarget(opts) {
|
|
765
|
+
markPageRefBlocked(opts.cdpUrl, opts.page);
|
|
766
|
+
const resolvedTargetId = await pageTargetId(opts.page).catch(() => null);
|
|
767
|
+
const fallbackTargetId = normalizeOptionalString(opts.targetId) ?? "";
|
|
768
|
+
const targetIdToBlock = resolvedTargetId || fallbackTargetId;
|
|
769
|
+
if (targetIdToBlock) markTargetBlocked(opts.cdpUrl, targetIdToBlock);
|
|
770
|
+
}
|
|
771
|
+
async function closeBlockedNavigationTarget(opts) {
|
|
772
|
+
await quarantineBlockedTarget(opts);
|
|
773
|
+
await opts.page.close().catch(() => {});
|
|
774
|
+
}
|
|
775
|
+
async function assertPageNavigationCompletedSafely(opts) {
|
|
776
|
+
const navigationPolicy = withBrowserNavigationPolicy(opts.ssrfPolicy, { browserProxyMode: opts.browserProxyMode });
|
|
777
|
+
try {
|
|
778
|
+
await assertBrowserNavigationRedirectChainAllowed({
|
|
779
|
+
request: opts.response?.request(),
|
|
780
|
+
...navigationPolicy
|
|
781
|
+
});
|
|
782
|
+
await assertBrowserNavigationResultAllowed({
|
|
783
|
+
url: opts.page.url(),
|
|
784
|
+
...navigationPolicy
|
|
785
|
+
});
|
|
786
|
+
} catch (err) {
|
|
787
|
+
if (isPolicyDenyNavigationError(err)) await quarantineBlockedTarget({
|
|
788
|
+
cdpUrl: opts.cdpUrl,
|
|
789
|
+
page: opts.page,
|
|
790
|
+
targetId: opts.targetId
|
|
791
|
+
});
|
|
792
|
+
throw err;
|
|
793
|
+
}
|
|
794
|
+
}
|
|
795
|
+
async function continueRouteSafely(route) {
|
|
796
|
+
try {
|
|
797
|
+
await route.continue();
|
|
798
|
+
} catch (err) {
|
|
799
|
+
if ((err instanceof Error ? err.message : "").includes("Route is already handled")) return;
|
|
800
|
+
throw err;
|
|
801
|
+
}
|
|
802
|
+
}
|
|
803
|
+
async function gotoPageWithNavigationGuard(opts) {
|
|
804
|
+
const navigationPolicy = withBrowserNavigationPolicy(opts.ssrfPolicy, { browserProxyMode: opts.browserProxyMode });
|
|
805
|
+
let blockedError = null;
|
|
806
|
+
const handler = async (route, request) => {
|
|
807
|
+
if (blockedError) {
|
|
808
|
+
await route.abort().catch(() => {});
|
|
809
|
+
return;
|
|
810
|
+
}
|
|
811
|
+
const isTopLevel = isTopLevelNavigationRequest(opts.page, request);
|
|
812
|
+
const isSubframeDocument = !isTopLevel && isSubframeDocumentNavigationRequest(opts.page, request);
|
|
813
|
+
if (!isTopLevel && !isSubframeDocument) {
|
|
814
|
+
await continueRouteSafely(route);
|
|
815
|
+
return;
|
|
816
|
+
}
|
|
817
|
+
try {
|
|
818
|
+
await assertBrowserNavigationAllowed({
|
|
819
|
+
url: request.url(),
|
|
820
|
+
...navigationPolicy
|
|
821
|
+
});
|
|
822
|
+
} catch (err) {
|
|
823
|
+
if (isPolicyDenyNavigationError(err)) {
|
|
824
|
+
if (isTopLevel) blockedError = err;
|
|
825
|
+
await route.abort().catch(() => {});
|
|
826
|
+
return;
|
|
827
|
+
}
|
|
828
|
+
throw err;
|
|
829
|
+
}
|
|
830
|
+
await continueRouteSafely(route);
|
|
831
|
+
};
|
|
832
|
+
await opts.page.route("**", handler);
|
|
833
|
+
try {
|
|
834
|
+
const response = await opts.page.goto(opts.url, { timeout: opts.timeoutMs });
|
|
835
|
+
if (blockedError) throw blockedError;
|
|
836
|
+
return response;
|
|
837
|
+
} catch (err) {
|
|
838
|
+
if (blockedError) throw blockedError;
|
|
839
|
+
throw err;
|
|
840
|
+
} finally {
|
|
841
|
+
await opts.page.unroute("**", handler).catch(() => {});
|
|
842
|
+
if (blockedError) await closeBlockedNavigationTarget({
|
|
843
|
+
cdpUrl: opts.cdpUrl,
|
|
844
|
+
page: opts.page,
|
|
845
|
+
targetId: opts.targetId
|
|
846
|
+
});
|
|
847
|
+
}
|
|
848
|
+
}
|
|
849
|
+
function refLocator(page, ref) {
|
|
850
|
+
const normalized = ref.startsWith("@") ? ref.slice(1) : ref.startsWith("ref=") ? ref.slice(4) : ref;
|
|
851
|
+
if (/^e\d+$/.test(normalized)) {
|
|
852
|
+
const state = pageStates.get(page);
|
|
853
|
+
if (state?.roleRefsMode === "aria") return (state.roleRefsFrameSelector ? page.frameLocator(state.roleRefsFrameSelector) : page).locator(`aria-ref=${normalized}`);
|
|
854
|
+
const info = state?.roleRefs?.[normalized];
|
|
855
|
+
if (!info) throw new Error(`Unknown ref "${normalized}". Run a new snapshot and use a ref from that snapshot.`);
|
|
856
|
+
const locAny = state?.roleRefsFrameSelector ? page.frameLocator(state.roleRefsFrameSelector) : page;
|
|
857
|
+
const locator = info.name ? locAny.getByRole(info.role, {
|
|
858
|
+
name: info.name,
|
|
859
|
+
exact: true
|
|
860
|
+
}) : locAny.getByRole(info.role);
|
|
861
|
+
return info.nth !== void 0 ? locator.nth(info.nth) : locator;
|
|
862
|
+
}
|
|
863
|
+
if (AX_REF_PATTERN.test(normalized)) {
|
|
864
|
+
const state = pageStates.get(page);
|
|
865
|
+
const info = state?.roleRefs?.[normalized];
|
|
866
|
+
if (!info) throw new Error(`Unknown ref "${normalized}". Run a new snapshot and use a ref from that snapshot.`);
|
|
867
|
+
const scope = state.roleRefsFrameSelector ? page.frameLocator(state.roleRefsFrameSelector) : page;
|
|
868
|
+
if (info.domMarker) return scope.locator(`[${BROWSER_REF_MARKER_ATTRIBUTE}="${normalized}"]`);
|
|
869
|
+
const locAny = scope;
|
|
870
|
+
const locator = info.name ? locAny.getByRole(info.role, {
|
|
871
|
+
name: info.name,
|
|
872
|
+
exact: true
|
|
873
|
+
}) : locAny.getByRole(info.role);
|
|
874
|
+
return info.nth !== void 0 ? locator.nth(info.nth) : locator;
|
|
875
|
+
}
|
|
876
|
+
return page.locator(`aria-ref=${normalized}`);
|
|
877
|
+
}
|
|
878
|
+
async function closePlaywrightBrowserConnection(opts) {
|
|
879
|
+
const normalized = opts?.cdpUrl ? normalizeCdpUrl(opts.cdpUrl) : null;
|
|
880
|
+
if (normalized) {
|
|
881
|
+
clearBlockedTargetsForCdpUrl(normalized);
|
|
882
|
+
clearBlockedPageRefsForCdpUrl(normalized);
|
|
883
|
+
const cur = takeCachedPlaywrightBrowserConnection(normalized);
|
|
884
|
+
if (!cur) return;
|
|
885
|
+
await cur.browser.close().catch(() => {});
|
|
886
|
+
return;
|
|
887
|
+
}
|
|
888
|
+
const connections = Array.from(cachedByCdpUrl.values());
|
|
889
|
+
clearBlockedTargetsForCdpUrl();
|
|
890
|
+
clearBlockedPageRefsForCdpUrl();
|
|
891
|
+
cachedByCdpUrl.clear();
|
|
892
|
+
connectingByCdpUrl.clear();
|
|
893
|
+
for (const cur of connections) {
|
|
894
|
+
if (cur.onDisconnected && typeof cur.browser.off === "function") cur.browser.off("disconnected", cur.onDisconnected);
|
|
895
|
+
await cur.browser.close().catch(() => {});
|
|
896
|
+
}
|
|
897
|
+
}
|
|
898
|
+
function cdpSocketNeedsAttach(wsUrl) {
|
|
899
|
+
try {
|
|
900
|
+
const pathname = new URL(wsUrl).pathname;
|
|
901
|
+
return pathname === "/cdp" || pathname.endsWith("/cdp") || pathname.includes("/devtools/browser/");
|
|
902
|
+
} catch {
|
|
903
|
+
return false;
|
|
904
|
+
}
|
|
905
|
+
}
|
|
906
|
+
async function tryTerminateExecutionViaCdp(opts) {
|
|
907
|
+
await assertCdpEndpointAllowed(opts.cdpUrl, opts.ssrfPolicy);
|
|
908
|
+
const cdpHttpBase = normalizeCdpHttpBaseForJsonEndpoints(opts.cdpUrl);
|
|
909
|
+
const pages = await fetchJson(appendCdpPath(cdpHttpBase, "/json/list"), 2e3).catch(() => null);
|
|
910
|
+
if (!pages || pages.length === 0) return;
|
|
911
|
+
const targetId = normalizeOptionalString(opts.targetId) ?? "";
|
|
912
|
+
const wsUrlRaw = normalizeOptionalString(pages.find((p) => normalizeOptionalString(p.id) === targetId)?.webSocketDebuggerUrl) ?? "";
|
|
913
|
+
if (!wsUrlRaw) return;
|
|
914
|
+
const wsUrl = normalizeCdpWsUrl(wsUrlRaw, cdpHttpBase);
|
|
915
|
+
const needsAttach = cdpSocketNeedsAttach(wsUrl);
|
|
916
|
+
const runWithTimeout = async (work, ms) => {
|
|
917
|
+
let timer;
|
|
918
|
+
const timeoutPromise = new Promise((_, reject) => {
|
|
919
|
+
timer = setTimeout(() => reject(/* @__PURE__ */ new Error("CDP 命令超时")), ms);
|
|
920
|
+
});
|
|
921
|
+
try {
|
|
922
|
+
return await Promise.race([work, timeoutPromise]);
|
|
923
|
+
} finally {
|
|
924
|
+
if (timer) clearTimeout(timer);
|
|
925
|
+
}
|
|
926
|
+
};
|
|
927
|
+
await withCdpSocket(wsUrl, async (send) => {
|
|
928
|
+
let sessionId;
|
|
929
|
+
try {
|
|
930
|
+
if (needsAttach) {
|
|
931
|
+
const attachedSessionId = normalizeOptionalString((await runWithTimeout(send("Target.attachToTarget", {
|
|
932
|
+
targetId: opts.targetId,
|
|
933
|
+
flatten: true
|
|
934
|
+
}), 1500))?.sessionId);
|
|
935
|
+
if (attachedSessionId) sessionId = attachedSessionId;
|
|
936
|
+
}
|
|
937
|
+
await runWithTimeout(send("Runtime.terminateExecution", void 0, sessionId), 1500);
|
|
938
|
+
if (sessionId) send("Target.detachFromTarget", { sessionId }).catch(() => {});
|
|
939
|
+
} catch {}
|
|
940
|
+
}, { handshakeTimeoutMs: 2e3 }).catch(() => {});
|
|
941
|
+
}
|
|
942
|
+
/**
|
|
943
|
+
* Best-effort cancellation for stuck page operations.
|
|
944
|
+
*
|
|
945
|
+
* Playwright serializes CDP commands per page; a long-running or stuck operation (notably evaluate)
|
|
946
|
+
* can block all subsequent commands. We cannot safely "cancel" an individual command, and we do
|
|
947
|
+
* not want to close the actual Chromium tab. Instead, we disconnect Playwright's CDP connection
|
|
948
|
+
* so in-flight commands fail fast and the next request reconnects transparently.
|
|
949
|
+
*
|
|
950
|
+
* IMPORTANT: We CANNOT call Connection.close() because Playwright shares a single Connection
|
|
951
|
+
* across all objects (BrowserType, Browser, etc.). Closing it corrupts the entire Playwright
|
|
952
|
+
* instance, preventing reconnection.
|
|
953
|
+
*
|
|
954
|
+
* Instead we:
|
|
955
|
+
* 1. Null out `cached` so the next call triggers a fresh connectOverCDP
|
|
956
|
+
* 2. Fire-and-forget browser.close() — it may hang but won't block us
|
|
957
|
+
* 3. The next connectBrowser() creates a completely new CDP WebSocket connection
|
|
958
|
+
*
|
|
959
|
+
* The old browser.close() eventually resolves when the in-browser evaluate timeout fires,
|
|
960
|
+
* or the old connection gets GC'd. Either way, it doesn't affect the fresh connection.
|
|
961
|
+
*/
|
|
962
|
+
async function forceDisconnectPlaywrightForTarget(opts) {
|
|
963
|
+
const normalized = normalizeCdpUrl(opts.cdpUrl);
|
|
964
|
+
const cur = takeCachedPlaywrightBrowserConnection(normalized);
|
|
965
|
+
if (!cur) return;
|
|
966
|
+
const targetId = normalizeOptionalString(opts.targetId) ?? "";
|
|
967
|
+
if (targetId) await tryTerminateExecutionViaCdp({
|
|
968
|
+
cdpUrl: normalized,
|
|
969
|
+
targetId,
|
|
970
|
+
ssrfPolicy: opts.ssrfPolicy
|
|
971
|
+
}).catch(() => {});
|
|
972
|
+
cur.browser.close().catch(() => {});
|
|
973
|
+
}
|
|
974
|
+
async function withPlaywrightSafeReadReconnect(cdpUrl, run) {
|
|
975
|
+
try {
|
|
976
|
+
return await run();
|
|
977
|
+
} catch (err) {
|
|
978
|
+
if (!isRecoverablePlaywrightDisconnectError(err)) throw err;
|
|
979
|
+
evictStalePlaywrightBrowserConnection(cdpUrl);
|
|
980
|
+
return await run();
|
|
981
|
+
}
|
|
982
|
+
}
|
|
983
|
+
/**
|
|
984
|
+
* List all pages/tabs from the persistent Playwright connection.
|
|
985
|
+
* Used for remote profiles where HTTP-based /json/list is ephemeral.
|
|
986
|
+
*/
|
|
987
|
+
async function listPagesViaPlaywright(opts) {
|
|
988
|
+
return await withPlaywrightSafeReadReconnect(opts.cdpUrl, async () => {
|
|
989
|
+
const { browser } = await connectBrowser(opts.cdpUrl, opts.ssrfPolicy);
|
|
990
|
+
const pages = await getAllPages(browser);
|
|
991
|
+
const results = [];
|
|
992
|
+
for (const page of pages) {
|
|
993
|
+
if (isBlockedPageRef(opts.cdpUrl, page)) continue;
|
|
994
|
+
let tid;
|
|
995
|
+
try {
|
|
996
|
+
tid = await pageTargetId(page);
|
|
997
|
+
} catch (err) {
|
|
998
|
+
if (isRecoverablePlaywrightDisconnectError(err)) throw err;
|
|
999
|
+
tid = null;
|
|
1000
|
+
}
|
|
1001
|
+
if (tid && !isBlockedTarget(opts.cdpUrl, tid)) {
|
|
1002
|
+
let title = "";
|
|
1003
|
+
try {
|
|
1004
|
+
title = await page.title();
|
|
1005
|
+
} catch (err) {
|
|
1006
|
+
if (isRecoverablePlaywrightDisconnectError(err)) throw err;
|
|
1007
|
+
}
|
|
1008
|
+
let url = "";
|
|
1009
|
+
try {
|
|
1010
|
+
url = page.url();
|
|
1011
|
+
} catch (err) {
|
|
1012
|
+
if (isRecoverablePlaywrightDisconnectError(err)) throw err;
|
|
1013
|
+
}
|
|
1014
|
+
if (!isSelectableCdpBrowserTarget({ url })) continue;
|
|
1015
|
+
results.push({
|
|
1016
|
+
targetId: tid,
|
|
1017
|
+
title,
|
|
1018
|
+
url,
|
|
1019
|
+
type: "page"
|
|
1020
|
+
});
|
|
1021
|
+
}
|
|
1022
|
+
}
|
|
1023
|
+
return results;
|
|
1024
|
+
});
|
|
1025
|
+
}
|
|
1026
|
+
/**
|
|
1027
|
+
* Create a new page/tab using the persistent Playwright connection.
|
|
1028
|
+
* Used for remote profiles where HTTP-based /json/new is ephemeral.
|
|
1029
|
+
* Returns the new page's targetId and metadata.
|
|
1030
|
+
*/
|
|
1031
|
+
async function createPageViaPlaywright(opts) {
|
|
1032
|
+
const { browser } = await connectBrowser(opts.cdpUrl, opts.ssrfPolicy);
|
|
1033
|
+
const context = browser.contexts()[0] ?? await browser.newContext();
|
|
1034
|
+
ensureContextState(context);
|
|
1035
|
+
const page = await context.newPage();
|
|
1036
|
+
ensurePageState(page);
|
|
1037
|
+
clearBlockedPageRef(opts.cdpUrl, page);
|
|
1038
|
+
const createdTargetId = await pageTargetId(page).catch(() => null);
|
|
1039
|
+
clearBlockedTarget(opts.cdpUrl, createdTargetId ?? void 0);
|
|
1040
|
+
const targetUrl = opts.url.trim() || "about:blank";
|
|
1041
|
+
if (targetUrl !== "about:blank") {
|
|
1042
|
+
await assertBrowserNavigationAllowed({
|
|
1043
|
+
url: targetUrl,
|
|
1044
|
+
...withBrowserNavigationPolicy(opts.ssrfPolicy, { browserProxyMode: opts.browserProxyMode })
|
|
1045
|
+
});
|
|
1046
|
+
let response = null;
|
|
1047
|
+
try {
|
|
1048
|
+
response = await gotoPageWithNavigationGuard({
|
|
1049
|
+
cdpUrl: opts.cdpUrl,
|
|
1050
|
+
page,
|
|
1051
|
+
url: targetUrl,
|
|
1052
|
+
timeoutMs: 3e4,
|
|
1053
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
1054
|
+
browserProxyMode: opts.browserProxyMode,
|
|
1055
|
+
targetId: createdTargetId ?? void 0
|
|
1056
|
+
});
|
|
1057
|
+
} catch (err) {
|
|
1058
|
+
if (isPolicyDenyNavigationError(err) || err instanceof BlockedBrowserTargetError) throw err;
|
|
1059
|
+
}
|
|
1060
|
+
try {
|
|
1061
|
+
await assertPageNavigationCompletedSafely({
|
|
1062
|
+
cdpUrl: opts.cdpUrl,
|
|
1063
|
+
page,
|
|
1064
|
+
response,
|
|
1065
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
1066
|
+
browserProxyMode: opts.browserProxyMode,
|
|
1067
|
+
targetId: createdTargetId ?? void 0
|
|
1068
|
+
});
|
|
1069
|
+
} catch (err) {
|
|
1070
|
+
if (isPolicyDenyNavigationError(err)) await closeBlockedNavigationTarget({
|
|
1071
|
+
cdpUrl: opts.cdpUrl,
|
|
1072
|
+
page,
|
|
1073
|
+
targetId: createdTargetId ?? void 0
|
|
1074
|
+
});
|
|
1075
|
+
throw err;
|
|
1076
|
+
}
|
|
1077
|
+
}
|
|
1078
|
+
const tid = createdTargetId || await pageTargetId(page).catch(() => null);
|
|
1079
|
+
if (!tid) throw new Error("Failed to get targetId for new page");
|
|
1080
|
+
return {
|
|
1081
|
+
targetId: tid,
|
|
1082
|
+
title: await page.title().catch(() => ""),
|
|
1083
|
+
url: page.url(),
|
|
1084
|
+
type: "page"
|
|
1085
|
+
};
|
|
1086
|
+
}
|
|
1087
|
+
/**
|
|
1088
|
+
* Close a page/tab by targetId using the persistent Playwright connection.
|
|
1089
|
+
* Used for remote profiles where HTTP-based /json/close is ephemeral.
|
|
1090
|
+
*/
|
|
1091
|
+
async function closePageByTargetIdViaPlaywright(opts) {
|
|
1092
|
+
await (await resolvePageByTargetIdOrThrow(opts)).close();
|
|
1093
|
+
}
|
|
1094
|
+
/**
|
|
1095
|
+
* Focus a page/tab by targetId using the persistent Playwright connection.
|
|
1096
|
+
* Used for remote profiles where HTTP-based /json/activate can be ephemeral.
|
|
1097
|
+
*/
|
|
1098
|
+
async function focusPageByTargetIdViaPlaywright(opts) {
|
|
1099
|
+
const page = await resolvePageByTargetIdOrThrow(opts);
|
|
1100
|
+
try {
|
|
1101
|
+
await page.bringToFront();
|
|
1102
|
+
} catch (err) {
|
|
1103
|
+
try {
|
|
1104
|
+
await withPageScopedCdpClient({
|
|
1105
|
+
cdpUrl: opts.cdpUrl,
|
|
1106
|
+
page,
|
|
1107
|
+
targetId: opts.targetId,
|
|
1108
|
+
fn: async (send) => {
|
|
1109
|
+
await send("Page.bringToFront");
|
|
1110
|
+
}
|
|
1111
|
+
});
|
|
1112
|
+
return;
|
|
1113
|
+
} catch {
|
|
1114
|
+
throw err;
|
|
1115
|
+
}
|
|
1116
|
+
}
|
|
1117
|
+
}
|
|
1118
|
+
//#endregion
|
|
1119
|
+
//#region extensions/browser/src/browser/pw-tools-core.activity.ts
|
|
1120
|
+
async function getPageErrorsViaPlaywright(opts) {
|
|
1121
|
+
const state = ensurePageState(await getPageForTargetId(opts));
|
|
1122
|
+
const errors = [...state.errors];
|
|
1123
|
+
if (opts.clear) state.errors = [];
|
|
1124
|
+
return { errors };
|
|
1125
|
+
}
|
|
1126
|
+
async function getNetworkRequestsViaPlaywright(opts) {
|
|
1127
|
+
const state = ensurePageState(await getPageForTargetId(opts));
|
|
1128
|
+
const raw = [...state.requests];
|
|
1129
|
+
const filter = typeof opts.filter === "string" ? opts.filter.trim() : "";
|
|
1130
|
+
const requests = filter ? raw.filter((r) => r.url.includes(filter)) : raw;
|
|
1131
|
+
if (opts.clear) {
|
|
1132
|
+
state.requests = [];
|
|
1133
|
+
state.requestIds = /* @__PURE__ */ new WeakMap();
|
|
1134
|
+
}
|
|
1135
|
+
return { requests };
|
|
1136
|
+
}
|
|
1137
|
+
function consolePriority(level) {
|
|
1138
|
+
switch (level) {
|
|
1139
|
+
case "error": return 3;
|
|
1140
|
+
case "warning": return 2;
|
|
1141
|
+
case "info":
|
|
1142
|
+
case "log": return 1;
|
|
1143
|
+
case "debug": return 0;
|
|
1144
|
+
default: return 1;
|
|
1145
|
+
}
|
|
1146
|
+
}
|
|
1147
|
+
async function getConsoleMessagesViaPlaywright(opts) {
|
|
1148
|
+
const state = ensurePageState(await getPageForTargetId(opts));
|
|
1149
|
+
if (!opts.level) return [...state.console];
|
|
1150
|
+
const min = consolePriority(opts.level);
|
|
1151
|
+
return state.console.filter((msg) => consolePriority(msg.type) >= min);
|
|
1152
|
+
}
|
|
1153
|
+
//#endregion
|
|
1154
|
+
//#region extensions/browser/src/browser/output-files.ts
|
|
1155
|
+
async function writeExternalFileWithinOutputRoot(params) {
|
|
1156
|
+
const outputPath = params.path.trim();
|
|
1157
|
+
if (!outputPath) throw new Error("需要 output 路径");
|
|
1158
|
+
const rootDir = params.rootDir ? path.resolve(params.rootDir) : path.dirname(path.resolve(outputPath));
|
|
1159
|
+
await ensureOutputDirectory(rootDir);
|
|
1160
|
+
return (await writeExternalFileWithinRoot({
|
|
1161
|
+
rootDir,
|
|
1162
|
+
path: outputPath,
|
|
1163
|
+
write: params.write
|
|
1164
|
+
}).catch((err) => {
|
|
1165
|
+
if (err instanceof Error && /file not found/i.test(err.message)) throw new Error("写入文件时输出目录发生了变化");
|
|
1166
|
+
throw err;
|
|
1167
|
+
})).path;
|
|
1168
|
+
}
|
|
1169
|
+
//#endregion
|
|
1170
|
+
//#region extensions/browser/src/browser/pw-tools-core.shared.ts
|
|
1171
|
+
let nextUploadArmId = 0;
|
|
1172
|
+
let nextDownloadArmId = 0;
|
|
1173
|
+
function bumpUploadArmId() {
|
|
1174
|
+
nextUploadArmId += 1;
|
|
1175
|
+
return nextUploadArmId;
|
|
1176
|
+
}
|
|
1177
|
+
function bumpDownloadArmId() {
|
|
1178
|
+
nextDownloadArmId += 1;
|
|
1179
|
+
return nextDownloadArmId;
|
|
1180
|
+
}
|
|
1181
|
+
function requireRef(value) {
|
|
1182
|
+
const raw = normalizeOptionalString(value) ?? "";
|
|
1183
|
+
const ref = (raw ? parseRoleRef(raw) : null) ?? (raw.startsWith("@") ? raw.slice(1) : raw);
|
|
1184
|
+
if (!ref) throw new Error("ref 是必需的");
|
|
1185
|
+
return ref;
|
|
1186
|
+
}
|
|
1187
|
+
function requireRefOrSelector(ref, selector) {
|
|
1188
|
+
const trimmedRef = normalizeOptionalString(ref) ?? "";
|
|
1189
|
+
const trimmedSelector = normalizeOptionalString(selector) ?? "";
|
|
1190
|
+
if (!trimmedRef && !trimmedSelector) throw new Error("ref 或 selector 是必需的");
|
|
1191
|
+
return {
|
|
1192
|
+
ref: trimmedRef || void 0,
|
|
1193
|
+
selector: trimmedSelector || void 0
|
|
1194
|
+
};
|
|
1195
|
+
}
|
|
1196
|
+
function normalizeTimeoutMs(timeoutMs, fallback) {
|
|
1197
|
+
const parsed = parseFiniteNumber(timeoutMs);
|
|
1198
|
+
return Math.max(500, Math.min(12e4, Math.floor(parsed ?? fallback)));
|
|
1199
|
+
}
|
|
1200
|
+
function toAIFriendlyError(error, selector) {
|
|
1201
|
+
const message = formatErrorMessage(error);
|
|
1202
|
+
if (message.includes("strict mode violation")) {
|
|
1203
|
+
const countMatch = message.match(/resolved to (\d+) elements/);
|
|
1204
|
+
const count = countMatch ? countMatch[1] : "multiple";
|
|
1205
|
+
return /* @__PURE__ */ new Error(`Selector "${selector}" matched ${count} elements. Run a new snapshot to get updated refs, or use a different ref.`);
|
|
1206
|
+
}
|
|
1207
|
+
if ((message.includes("Timeout") || message.includes("waiting for")) && (message.includes("to be visible") || message.includes("not visible") || message.includes("waiting for locator("))) return /* @__PURE__ */ new Error(`Element "${selector}" not found or not visible. Run a new snapshot to see current page elements.`);
|
|
1208
|
+
if (message.includes("intercepts pointer events") || message.includes("not visible") || message.includes("not receive pointer events")) return /* @__PURE__ */ new Error(`Element "${selector}" is not interactable (hidden or covered). Try scrolling it into view, closing overlays, or re-snapshotting.`);
|
|
1209
|
+
return error instanceof Error ? error : new Error(message);
|
|
1210
|
+
}
|
|
1211
|
+
//#endregion
|
|
1212
|
+
//#region extensions/browser/src/browser/pw-tools-core.downloads.ts
|
|
1213
|
+
function buildTempDownloadPath(fileName) {
|
|
1214
|
+
const id = crypto.randomUUID();
|
|
1215
|
+
const safeName = sanitizeUntrustedFileName(fileName, "download.bin");
|
|
1216
|
+
return path.join(resolvePreferredFengMingTmpDir(), "downloads", `${id}-${safeName}`);
|
|
1217
|
+
}
|
|
1218
|
+
function createPageDownloadWaiter(page, timeoutMs) {
|
|
1219
|
+
const state = ensurePageState(page);
|
|
1220
|
+
state.downloadWaiterDepth += 1;
|
|
1221
|
+
let done = false;
|
|
1222
|
+
let timer;
|
|
1223
|
+
let handler;
|
|
1224
|
+
let depthReleased = false;
|
|
1225
|
+
const cleanup = () => {
|
|
1226
|
+
if (!depthReleased) {
|
|
1227
|
+
depthReleased = true;
|
|
1228
|
+
state.downloadWaiterDepth = Math.max(0, state.downloadWaiterDepth - 1);
|
|
1229
|
+
}
|
|
1230
|
+
if (timer) clearTimeout(timer);
|
|
1231
|
+
timer = void 0;
|
|
1232
|
+
if (handler) {
|
|
1233
|
+
page.off("download", handler);
|
|
1234
|
+
handler = void 0;
|
|
1235
|
+
}
|
|
1236
|
+
};
|
|
1237
|
+
return {
|
|
1238
|
+
promise: new Promise((resolve, reject) => {
|
|
1239
|
+
handler = (download) => {
|
|
1240
|
+
if (done) return;
|
|
1241
|
+
done = true;
|
|
1242
|
+
cleanup();
|
|
1243
|
+
resolve(download);
|
|
1244
|
+
};
|
|
1245
|
+
page.on("download", handler);
|
|
1246
|
+
timer = setTimeout(() => {
|
|
1247
|
+
if (done) return;
|
|
1248
|
+
done = true;
|
|
1249
|
+
cleanup();
|
|
1250
|
+
reject(/* @__PURE__ */ new Error("等待下载超时"));
|
|
1251
|
+
}, timeoutMs);
|
|
1252
|
+
}),
|
|
1253
|
+
cancel: () => {
|
|
1254
|
+
if (done) return;
|
|
1255
|
+
done = true;
|
|
1256
|
+
cleanup();
|
|
1257
|
+
}
|
|
1258
|
+
};
|
|
1259
|
+
}
|
|
1260
|
+
async function saveDownloadPayload(download, outPath, rootDir) {
|
|
1261
|
+
const suggested = download.suggestedFilename?.() || "download.bin";
|
|
1262
|
+
const requestedPath = outPath?.trim();
|
|
1263
|
+
const finalPath = await writeExternalFileWithinOutputRoot({
|
|
1264
|
+
rootDir,
|
|
1265
|
+
path: path.resolve(requestedPath || buildTempDownloadPath(suggested)),
|
|
1266
|
+
write: async (tempPath) => {
|
|
1267
|
+
await download.saveAs?.(tempPath);
|
|
1268
|
+
}
|
|
1269
|
+
});
|
|
1270
|
+
return {
|
|
1271
|
+
url: download.url?.() || "",
|
|
1272
|
+
suggestedFilename: suggested,
|
|
1273
|
+
path: finalPath
|
|
1274
|
+
};
|
|
1275
|
+
}
|
|
1276
|
+
async function awaitDownloadPayload(params) {
|
|
1277
|
+
try {
|
|
1278
|
+
const download = await params.waiter.promise;
|
|
1279
|
+
if (params.state.armIdDownload !== params.armId) throw new Error("下载被另一个等待者替代");
|
|
1280
|
+
return await saveDownloadPayload(download, params.outPath ?? "", params.rootDir);
|
|
1281
|
+
} catch (err) {
|
|
1282
|
+
params.waiter.cancel();
|
|
1283
|
+
throw err;
|
|
1284
|
+
}
|
|
1285
|
+
}
|
|
1286
|
+
async function armFileUploadViaPlaywright(opts) {
|
|
1287
|
+
const page = await getPageForTargetId(opts);
|
|
1288
|
+
const state = ensurePageState(page);
|
|
1289
|
+
const timeout = normalizeTimeoutMs(opts.timeoutMs, 12e4);
|
|
1290
|
+
state.armIdUpload = bumpUploadArmId();
|
|
1291
|
+
const armId = state.armIdUpload;
|
|
1292
|
+
page.waitForEvent("filechooser", { timeout }).then(async (fileChooser) => {
|
|
1293
|
+
if (state.armIdUpload !== armId) return;
|
|
1294
|
+
if (!opts.paths?.length) {
|
|
1295
|
+
try {
|
|
1296
|
+
await page.keyboard.press("Escape");
|
|
1297
|
+
} catch {}
|
|
1298
|
+
return;
|
|
1299
|
+
}
|
|
1300
|
+
const uploadPathsResult = await resolveStrictExistingPathsWithinRoot({
|
|
1301
|
+
rootDir: DEFAULT_UPLOAD_DIR,
|
|
1302
|
+
requestedPaths: opts.paths,
|
|
1303
|
+
scopeLabel: `uploads directory (${DEFAULT_UPLOAD_DIR})`
|
|
1304
|
+
});
|
|
1305
|
+
if (!uploadPathsResult.ok) {
|
|
1306
|
+
try {
|
|
1307
|
+
await page.keyboard.press("Escape");
|
|
1308
|
+
} catch {}
|
|
1309
|
+
return;
|
|
1310
|
+
}
|
|
1311
|
+
await fileChooser.setFiles(uploadPathsResult.paths);
|
|
1312
|
+
try {
|
|
1313
|
+
const input = typeof fileChooser.element === "function" ? await Promise.resolve(fileChooser.element()) : null;
|
|
1314
|
+
if (input) await input.evaluate((el) => {
|
|
1315
|
+
el.dispatchEvent(new Event("input", { bubbles: true }));
|
|
1316
|
+
el.dispatchEvent(new Event("change", { bubbles: true }));
|
|
1317
|
+
});
|
|
1318
|
+
} catch {}
|
|
1319
|
+
}).catch(() => {});
|
|
1320
|
+
}
|
|
1321
|
+
async function armDialogViaPlaywright(opts) {
|
|
1322
|
+
const page = await getPageForTargetId(opts);
|
|
1323
|
+
const timeout = normalizeTimeoutMs(opts.timeoutMs, 12e4);
|
|
1324
|
+
try {
|
|
1325
|
+
await respondToObservedDialogOnPage({
|
|
1326
|
+
page,
|
|
1327
|
+
accept: opts.accept,
|
|
1328
|
+
closedBy: "agent",
|
|
1329
|
+
...opts.dialogId !== void 0 ? { dialogId: opts.dialogId } : {},
|
|
1330
|
+
...opts.promptText !== void 0 ? { promptText: opts.promptText } : {}
|
|
1331
|
+
});
|
|
1332
|
+
return;
|
|
1333
|
+
} catch (err) {
|
|
1334
|
+
if (opts.dialogId || err instanceof Error && !err.message.includes("No dialog is pending")) throw err;
|
|
1335
|
+
}
|
|
1336
|
+
armObservedDialogResponseOnPage({
|
|
1337
|
+
page,
|
|
1338
|
+
accept: opts.accept,
|
|
1339
|
+
timeoutMs: timeout,
|
|
1340
|
+
...opts.promptText !== void 0 ? { promptText: opts.promptText } : {}
|
|
1341
|
+
});
|
|
1342
|
+
}
|
|
1343
|
+
async function waitForDownloadViaPlaywright(opts) {
|
|
1344
|
+
const page = await getPageForTargetId(opts);
|
|
1345
|
+
const state = ensurePageState(page);
|
|
1346
|
+
const timeout = normalizeTimeoutMs(opts.timeoutMs, 12e4);
|
|
1347
|
+
state.armIdDownload = bumpDownloadArmId();
|
|
1348
|
+
const armId = state.armIdDownload;
|
|
1349
|
+
return await awaitDownloadPayload({
|
|
1350
|
+
waiter: createPageDownloadWaiter(page, timeout),
|
|
1351
|
+
state,
|
|
1352
|
+
armId,
|
|
1353
|
+
outPath: opts.path,
|
|
1354
|
+
rootDir: opts.rootDir
|
|
1355
|
+
});
|
|
1356
|
+
}
|
|
1357
|
+
async function downloadViaPlaywright(opts) {
|
|
1358
|
+
const page = await getPageForTargetId(opts);
|
|
1359
|
+
const state = ensurePageState(page);
|
|
1360
|
+
restoreRoleRefsForTarget({
|
|
1361
|
+
cdpUrl: opts.cdpUrl,
|
|
1362
|
+
targetId: opts.targetId,
|
|
1363
|
+
page
|
|
1364
|
+
});
|
|
1365
|
+
const timeout = normalizeTimeoutMs(opts.timeoutMs, 12e4);
|
|
1366
|
+
const ref = requireRef(opts.ref);
|
|
1367
|
+
const outPath = opts.path?.trim() ?? "";
|
|
1368
|
+
if (!outPath) throw new Error("路径是必需的");
|
|
1369
|
+
state.armIdDownload = bumpDownloadArmId();
|
|
1370
|
+
const armId = state.armIdDownload;
|
|
1371
|
+
const waiter = createPageDownloadWaiter(page, timeout);
|
|
1372
|
+
try {
|
|
1373
|
+
const locator = refLocator(page, ref);
|
|
1374
|
+
try {
|
|
1375
|
+
await locator.click({ timeout });
|
|
1376
|
+
} catch (err) {
|
|
1377
|
+
throw toAIFriendlyError(err, ref);
|
|
1378
|
+
}
|
|
1379
|
+
return await awaitDownloadPayload({
|
|
1380
|
+
waiter,
|
|
1381
|
+
state,
|
|
1382
|
+
armId,
|
|
1383
|
+
outPath,
|
|
1384
|
+
rootDir: opts.rootDir
|
|
1385
|
+
});
|
|
1386
|
+
} catch (err) {
|
|
1387
|
+
waiter.cancel();
|
|
1388
|
+
throw err;
|
|
1389
|
+
}
|
|
1390
|
+
}
|
|
1391
|
+
//#endregion
|
|
1392
|
+
//#region extensions/browser/src/browser/pw-tools-core.snapshot.ts
|
|
1393
|
+
function resolveBoundedTimeoutMs(timeoutMs, fallbackMs, minMs, maxMs) {
|
|
1394
|
+
const parsed = parseFiniteNumber(timeoutMs);
|
|
1395
|
+
return Math.max(minMs, Math.min(maxMs, Math.floor(parsed ?? fallbackMs)));
|
|
1396
|
+
}
|
|
1397
|
+
function resolveSnapshotTimeoutMs(timeoutMs) {
|
|
1398
|
+
return resolveBoundedTimeoutMs(timeoutMs, 5e3, 500, 6e4);
|
|
1399
|
+
}
|
|
1400
|
+
function resolveNavigationTimeoutMs(timeoutMs) {
|
|
1401
|
+
return resolveBoundedTimeoutMs(timeoutMs, 2e4, 1e3, 12e4);
|
|
1402
|
+
}
|
|
1403
|
+
function resolveViewportDimension(value, label) {
|
|
1404
|
+
const dimension = resolveIntegerOption(value, 1, { min: 1 });
|
|
1405
|
+
if (dimension > 8192) throw new Error(`viewport ${label} exceeds maximum of ${ACT_MAX_VIEWPORT_DIMENSION}`);
|
|
1406
|
+
return dimension;
|
|
1407
|
+
}
|
|
1408
|
+
async function collectSnapshotUrls(page) {
|
|
1409
|
+
const urls = await page.evaluate(() => {
|
|
1410
|
+
const seen = /* @__PURE__ */ new Set();
|
|
1411
|
+
const out = [];
|
|
1412
|
+
for (const anchor of Array.from(document.querySelectorAll("a[href]"))) {
|
|
1413
|
+
const href = anchor instanceof HTMLAnchorElement ? anchor.href : "";
|
|
1414
|
+
if (!href || seen.has(href)) continue;
|
|
1415
|
+
const text = (anchor.textContent || anchor.getAttribute("aria-label") || "").replace(/\s+/g, " ").trim().slice(0, 120) || href;
|
|
1416
|
+
seen.add(href);
|
|
1417
|
+
out.push({
|
|
1418
|
+
text,
|
|
1419
|
+
url: href
|
|
1420
|
+
});
|
|
1421
|
+
if (out.length >= 100) break;
|
|
1422
|
+
}
|
|
1423
|
+
return out;
|
|
1424
|
+
}).catch(() => []);
|
|
1425
|
+
return Array.isArray(urls) ? urls : [];
|
|
1426
|
+
}
|
|
1427
|
+
function buildStoredAriaRefs(nodes, markedRefs) {
|
|
1428
|
+
const refs = {};
|
|
1429
|
+
const counts = /* @__PURE__ */ new Map();
|
|
1430
|
+
const refsByKey = /* @__PURE__ */ new Map();
|
|
1431
|
+
for (const node of nodes) {
|
|
1432
|
+
const role = normalizeLowercaseStringOrEmpty(node.role) || "unknown";
|
|
1433
|
+
const name = node.name.trim() || void 0;
|
|
1434
|
+
const key = `${role}:${name ?? ""}`;
|
|
1435
|
+
const nth = counts.get(key) ?? 0;
|
|
1436
|
+
counts.set(key, nth + 1);
|
|
1437
|
+
refsByKey.set(key, [...refsByKey.get(key) ?? [], node.ref]);
|
|
1438
|
+
refs[node.ref] = {
|
|
1439
|
+
role,
|
|
1440
|
+
...name ? { name } : {},
|
|
1441
|
+
...nth > 0 ? { nth } : {},
|
|
1442
|
+
...markedRefs.has(node.ref) ? { domMarker: true } : {}
|
|
1443
|
+
};
|
|
1444
|
+
}
|
|
1445
|
+
for (const refsForKey of refsByKey.values()) {
|
|
1446
|
+
if (refsForKey.length > 1) continue;
|
|
1447
|
+
const ref = refsForKey[0];
|
|
1448
|
+
if (ref) delete refs[ref]?.nth;
|
|
1449
|
+
}
|
|
1450
|
+
return refs;
|
|
1451
|
+
}
|
|
1452
|
+
async function storeAriaSnapshotRefsViaPlaywright(opts) {
|
|
1453
|
+
const page = opts.page ?? await getPageForTargetId({
|
|
1454
|
+
cdpUrl: opts.cdpUrl,
|
|
1455
|
+
targetId: opts.targetId
|
|
1456
|
+
});
|
|
1457
|
+
ensurePageState(page);
|
|
1458
|
+
const markedRefs = await markBackendDomRefsOnPage({
|
|
1459
|
+
page,
|
|
1460
|
+
refs: opts.nodes.flatMap((node) => typeof node.backendDOMNodeId === "number" ? [{
|
|
1461
|
+
ref: node.ref,
|
|
1462
|
+
backendDOMNodeId: node.backendDOMNodeId
|
|
1463
|
+
}] : [])
|
|
1464
|
+
});
|
|
1465
|
+
storeRoleRefsForTarget({
|
|
1466
|
+
page,
|
|
1467
|
+
cdpUrl: opts.cdpUrl,
|
|
1468
|
+
targetId: opts.targetId,
|
|
1469
|
+
refs: buildStoredAriaRefs(opts.nodes, markedRefs),
|
|
1470
|
+
mode: "role"
|
|
1471
|
+
});
|
|
1472
|
+
}
|
|
1473
|
+
async function prepareSnapshotPageViaPlaywright(opts) {
|
|
1474
|
+
const page = await getPageForTargetId({
|
|
1475
|
+
cdpUrl: opts.cdpUrl,
|
|
1476
|
+
targetId: opts.targetId
|
|
1477
|
+
});
|
|
1478
|
+
ensurePageState(page);
|
|
1479
|
+
if (opts.ssrfPolicy) await assertPageNavigationCompletedSafely({
|
|
1480
|
+
cdpUrl: opts.cdpUrl,
|
|
1481
|
+
page,
|
|
1482
|
+
response: null,
|
|
1483
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
1484
|
+
targetId: opts.targetId
|
|
1485
|
+
});
|
|
1486
|
+
return page;
|
|
1487
|
+
}
|
|
1488
|
+
async function snapshotAriaViaPlaywright(opts) {
|
|
1489
|
+
const limit = resolveIntegerOption(opts.limit, 500, {
|
|
1490
|
+
min: 1,
|
|
1491
|
+
max: 2e3
|
|
1492
|
+
});
|
|
1493
|
+
const page = await prepareSnapshotPageViaPlaywright({
|
|
1494
|
+
cdpUrl: opts.cdpUrl,
|
|
1495
|
+
targetId: opts.targetId,
|
|
1496
|
+
ssrfPolicy: opts.ssrfPolicy
|
|
1497
|
+
});
|
|
1498
|
+
const ariaTimeoutMs = typeof opts.timeoutMs === "number" && Number.isFinite(opts.timeoutMs) && opts.timeoutMs > 0 ? Math.max(500, Math.min(6e4, Math.floor(opts.timeoutMs))) : void 0;
|
|
1499
|
+
const collectAxTree = withPageScopedCdpClient({
|
|
1500
|
+
cdpUrl: opts.cdpUrl,
|
|
1501
|
+
page,
|
|
1502
|
+
targetId: opts.targetId,
|
|
1503
|
+
fn: async (send) => {
|
|
1504
|
+
await send("Accessibility.enable").catch(() => {});
|
|
1505
|
+
return await send("Accessibility.getFullAXTree");
|
|
1506
|
+
}
|
|
1507
|
+
});
|
|
1508
|
+
const res = await (ariaTimeoutMs === void 0 ? collectAxTree : (() => {
|
|
1509
|
+
let timer;
|
|
1510
|
+
const timeout = new Promise((_, reject) => {
|
|
1511
|
+
timer = setTimeout(() => {
|
|
1512
|
+
reject(/* @__PURE__ */ new Error(`Aria snapshot via Playwright timed out after ${ariaTimeoutMs}ms.`));
|
|
1513
|
+
}, ariaTimeoutMs);
|
|
1514
|
+
timer.unref?.();
|
|
1515
|
+
});
|
|
1516
|
+
return Promise.race([collectAxTree, timeout]).finally(() => {
|
|
1517
|
+
if (timer) clearTimeout(timer);
|
|
1518
|
+
});
|
|
1519
|
+
})());
|
|
1520
|
+
const formatted = formatAriaSnapshot(Array.isArray(res?.nodes) ? res.nodes : [], limit);
|
|
1521
|
+
await storeAriaSnapshotRefsViaPlaywright({
|
|
1522
|
+
cdpUrl: opts.cdpUrl,
|
|
1523
|
+
targetId: opts.targetId,
|
|
1524
|
+
nodes: formatted,
|
|
1525
|
+
page
|
|
1526
|
+
});
|
|
1527
|
+
return { nodes: formatted };
|
|
1528
|
+
}
|
|
1529
|
+
async function snapshotAiViaPlaywright(opts) {
|
|
1530
|
+
const page = await prepareSnapshotPageViaPlaywright({
|
|
1531
|
+
cdpUrl: opts.cdpUrl,
|
|
1532
|
+
targetId: opts.targetId,
|
|
1533
|
+
ssrfPolicy: opts.ssrfPolicy
|
|
1534
|
+
});
|
|
1535
|
+
let snapshot = await page.ariaSnapshot({
|
|
1536
|
+
mode: "ai",
|
|
1537
|
+
timeout: resolveSnapshotTimeoutMs(opts.timeoutMs)
|
|
1538
|
+
});
|
|
1539
|
+
if (opts.urls) snapshot = appendSnapshotUrls(snapshot, await collectSnapshotUrls(page));
|
|
1540
|
+
const maxChars = opts.maxChars;
|
|
1541
|
+
const limit = typeof maxChars === "number" && Number.isFinite(maxChars) && maxChars > 0 ? Math.floor(maxChars) : void 0;
|
|
1542
|
+
let truncated = false;
|
|
1543
|
+
if (limit && snapshot.length > limit) {
|
|
1544
|
+
snapshot = `${snapshot.slice(0, limit)}\n\n[...TRUNCATED - page too large]`;
|
|
1545
|
+
truncated = true;
|
|
1546
|
+
}
|
|
1547
|
+
const built = buildRoleSnapshotFromAiSnapshot(snapshot);
|
|
1548
|
+
storeRoleRefsForTarget({
|
|
1549
|
+
page,
|
|
1550
|
+
cdpUrl: opts.cdpUrl,
|
|
1551
|
+
targetId: opts.targetId,
|
|
1552
|
+
refs: built.refs,
|
|
1553
|
+
mode: "aria"
|
|
1554
|
+
});
|
|
1555
|
+
return truncated ? {
|
|
1556
|
+
snapshot,
|
|
1557
|
+
truncated,
|
|
1558
|
+
refs: built.refs
|
|
1559
|
+
} : {
|
|
1560
|
+
snapshot,
|
|
1561
|
+
refs: built.refs
|
|
1562
|
+
};
|
|
1563
|
+
}
|
|
1564
|
+
async function finalizeRoleSnapshotViaPlaywright(params) {
|
|
1565
|
+
const snapshot = params.urls ? appendSnapshotUrls(params.built.snapshot, await collectSnapshotUrls(params.page)) : params.built.snapshot;
|
|
1566
|
+
storeRoleRefsForTarget({
|
|
1567
|
+
page: params.page,
|
|
1568
|
+
cdpUrl: params.cdpUrl,
|
|
1569
|
+
targetId: params.targetId,
|
|
1570
|
+
refs: params.built.refs,
|
|
1571
|
+
...params.frameSelector ? { frameSelector: params.frameSelector } : {},
|
|
1572
|
+
mode: params.mode
|
|
1573
|
+
});
|
|
1574
|
+
return {
|
|
1575
|
+
snapshot,
|
|
1576
|
+
refs: params.built.refs,
|
|
1577
|
+
stats: getRoleSnapshotStats(snapshot, params.built.refs)
|
|
1578
|
+
};
|
|
1579
|
+
}
|
|
1580
|
+
async function snapshotRoleViaPlaywright(opts) {
|
|
1581
|
+
const page = await prepareSnapshotPageViaPlaywright({
|
|
1582
|
+
cdpUrl: opts.cdpUrl,
|
|
1583
|
+
targetId: opts.targetId,
|
|
1584
|
+
ssrfPolicy: opts.ssrfPolicy
|
|
1585
|
+
});
|
|
1586
|
+
const ariaSnapshotTimeout = resolveSnapshotTimeoutMs(opts.timeoutMs);
|
|
1587
|
+
if (opts.refsMode === "aria") {
|
|
1588
|
+
if (normalizeOptionalString(opts.selector) || normalizeOptionalString(opts.frameSelector)) throw new Error("refs=aria does not support selector/frame snapshots yet.");
|
|
1589
|
+
const built = buildRoleSnapshotFromAiSnapshot(await page.ariaSnapshot({
|
|
1590
|
+
mode: "ai",
|
|
1591
|
+
timeout: ariaSnapshotTimeout
|
|
1592
|
+
}), opts.options);
|
|
1593
|
+
return await finalizeRoleSnapshotViaPlaywright({
|
|
1594
|
+
page,
|
|
1595
|
+
cdpUrl: opts.cdpUrl,
|
|
1596
|
+
targetId: opts.targetId,
|
|
1597
|
+
built,
|
|
1598
|
+
mode: "aria",
|
|
1599
|
+
urls: opts.urls
|
|
1600
|
+
});
|
|
1601
|
+
}
|
|
1602
|
+
const frameSelector = normalizeOptionalString(opts.frameSelector) ?? "";
|
|
1603
|
+
const selector = normalizeOptionalString(opts.selector) ?? "";
|
|
1604
|
+
const built = buildRoleSnapshotFromAriaSnapshot(await (frameSelector ? selector ? page.frameLocator(frameSelector).locator(selector) : page.frameLocator(frameSelector).locator(":root") : selector ? page.locator(selector) : page.locator(":root")).ariaSnapshot({ timeout: ariaSnapshotTimeout }) ?? "", opts.options);
|
|
1605
|
+
return await finalizeRoleSnapshotViaPlaywright({
|
|
1606
|
+
page,
|
|
1607
|
+
cdpUrl: opts.cdpUrl,
|
|
1608
|
+
targetId: opts.targetId,
|
|
1609
|
+
frameSelector: frameSelector || void 0,
|
|
1610
|
+
built,
|
|
1611
|
+
mode: "role",
|
|
1612
|
+
urls: opts.urls
|
|
1613
|
+
});
|
|
1614
|
+
}
|
|
1615
|
+
async function navigateViaPlaywright(opts) {
|
|
1616
|
+
const isRetryableNavigateError = (err) => {
|
|
1617
|
+
const msg = typeof err === "string" ? err.toLowerCase() : err instanceof Error ? err.message.toLowerCase() : "";
|
|
1618
|
+
return msg.includes("frame has been detached") || msg.includes("target page, context or browser has been closed");
|
|
1619
|
+
};
|
|
1620
|
+
const url = normalizeOptionalString(opts.url) ?? "";
|
|
1621
|
+
if (!url) throw new Error("URL 是必需的");
|
|
1622
|
+
await assertBrowserNavigationAllowed({
|
|
1623
|
+
url,
|
|
1624
|
+
...withBrowserNavigationPolicy(opts.ssrfPolicy, { browserProxyMode: opts.browserProxyMode })
|
|
1625
|
+
});
|
|
1626
|
+
const timeout = resolveNavigationTimeoutMs(opts.timeoutMs);
|
|
1627
|
+
let page = await getPageForTargetId(opts);
|
|
1628
|
+
ensurePageState(page);
|
|
1629
|
+
const navigate = async () => await gotoPageWithNavigationGuard({
|
|
1630
|
+
cdpUrl: opts.cdpUrl,
|
|
1631
|
+
page,
|
|
1632
|
+
url,
|
|
1633
|
+
timeoutMs: timeout,
|
|
1634
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
1635
|
+
browserProxyMode: opts.browserProxyMode,
|
|
1636
|
+
targetId: opts.targetId
|
|
1637
|
+
});
|
|
1638
|
+
let response;
|
|
1639
|
+
try {
|
|
1640
|
+
response = await navigate();
|
|
1641
|
+
} catch (err) {
|
|
1642
|
+
if (!isRetryableNavigateError(err)) throw err;
|
|
1643
|
+
await forceDisconnectPlaywrightForTarget({
|
|
1644
|
+
cdpUrl: opts.cdpUrl,
|
|
1645
|
+
targetId: opts.targetId,
|
|
1646
|
+
reason: "retry navigate after detached frame"
|
|
1647
|
+
}).catch(() => {});
|
|
1648
|
+
page = await getPageForTargetId(opts);
|
|
1649
|
+
ensurePageState(page);
|
|
1650
|
+
response = await navigate();
|
|
1651
|
+
}
|
|
1652
|
+
try {
|
|
1653
|
+
await assertPageNavigationCompletedSafely({
|
|
1654
|
+
cdpUrl: opts.cdpUrl,
|
|
1655
|
+
page,
|
|
1656
|
+
response,
|
|
1657
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
1658
|
+
browserProxyMode: opts.browserProxyMode,
|
|
1659
|
+
targetId: opts.targetId
|
|
1660
|
+
});
|
|
1661
|
+
} catch (err) {
|
|
1662
|
+
if (isPolicyDenyNavigationError(err)) await closeBlockedNavigationTarget({
|
|
1663
|
+
cdpUrl: opts.cdpUrl,
|
|
1664
|
+
page,
|
|
1665
|
+
targetId: opts.targetId
|
|
1666
|
+
});
|
|
1667
|
+
throw err;
|
|
1668
|
+
}
|
|
1669
|
+
return { url: page.url() };
|
|
1670
|
+
}
|
|
1671
|
+
async function resizeViewportViaPlaywright(opts) {
|
|
1672
|
+
const page = await getPageForTargetId(opts);
|
|
1673
|
+
ensurePageState(page);
|
|
1674
|
+
await page.setViewportSize({
|
|
1675
|
+
width: resolveViewportDimension(opts.width, "width"),
|
|
1676
|
+
height: resolveViewportDimension(opts.height, "height")
|
|
1677
|
+
});
|
|
1678
|
+
}
|
|
1679
|
+
async function closePageViaPlaywright(opts) {
|
|
1680
|
+
const page = await getPageForTargetId(opts);
|
|
1681
|
+
ensurePageState(page);
|
|
1682
|
+
await page.close();
|
|
1683
|
+
}
|
|
1684
|
+
async function pdfViaPlaywright(opts) {
|
|
1685
|
+
const page = await getPageForTargetId(opts);
|
|
1686
|
+
ensurePageState(page);
|
|
1687
|
+
return { buffer: await page.pdf({ printBackground: true }) };
|
|
1688
|
+
}
|
|
1689
|
+
//#endregion
|
|
1690
|
+
//#region extensions/browser/src/browser/pw-tools-core.interactions.ts
|
|
1691
|
+
const INTERACTION_NAVIGATION_GRACE_MS = 250;
|
|
1692
|
+
const pendingInteractionNavigationGuardCleanup = /* @__PURE__ */ new WeakMap();
|
|
1693
|
+
function resolveBoundedDelayMs(value, label, maxMs) {
|
|
1694
|
+
const normalized = Math.floor(value ?? 0);
|
|
1695
|
+
if (!Number.isFinite(normalized) || normalized < 0) throw new Error(`${label} must be >= 0`);
|
|
1696
|
+
if (normalized > maxMs) throw new Error(`${label} exceeds maximum of ${maxMs}ms`);
|
|
1697
|
+
return normalized;
|
|
1698
|
+
}
|
|
1699
|
+
async function getRestoredPageForTarget(opts) {
|
|
1700
|
+
const page = await getPageForTargetId(opts);
|
|
1701
|
+
ensurePageState(page);
|
|
1702
|
+
restoreRoleRefsForTarget({
|
|
1703
|
+
cdpUrl: opts.cdpUrl,
|
|
1704
|
+
targetId: opts.targetId,
|
|
1705
|
+
page
|
|
1706
|
+
});
|
|
1707
|
+
return page;
|
|
1708
|
+
}
|
|
1709
|
+
function toFriendlyInteractionError(err, label) {
|
|
1710
|
+
return isBrowserObservedDialogBlockedError(err) ? err : toAIFriendlyError(err, label);
|
|
1711
|
+
}
|
|
1712
|
+
function reconcileRemoteDialogAfterActionSettled(page, signal) {
|
|
1713
|
+
if (isBrowserObservedDialogBlockedError(signal?.reason)) markObservedDialogsHandledRemotelyForPage(page);
|
|
1714
|
+
}
|
|
1715
|
+
const resolveInteractionTimeoutMs = resolveActInteractionTimeoutMs;
|
|
1716
|
+
function didCrossDocumentUrlChange(page, previousUrl) {
|
|
1717
|
+
const currentUrl = page.url();
|
|
1718
|
+
if (currentUrl === previousUrl) return false;
|
|
1719
|
+
try {
|
|
1720
|
+
const prev = new URL(previousUrl);
|
|
1721
|
+
const curr = new URL(currentUrl);
|
|
1722
|
+
if (prev.origin === curr.origin && prev.pathname === curr.pathname && prev.search === curr.search) return false;
|
|
1723
|
+
} catch {}
|
|
1724
|
+
return true;
|
|
1725
|
+
}
|
|
1726
|
+
function isHashOnlyNavigation(currentUrl, previousUrl) {
|
|
1727
|
+
if (currentUrl === previousUrl) return false;
|
|
1728
|
+
try {
|
|
1729
|
+
const prev = new URL(previousUrl);
|
|
1730
|
+
const curr = new URL(currentUrl);
|
|
1731
|
+
return prev.origin === curr.origin && prev.pathname === curr.pathname && prev.search === curr.search;
|
|
1732
|
+
} catch {
|
|
1733
|
+
return false;
|
|
1734
|
+
}
|
|
1735
|
+
}
|
|
1736
|
+
function isMainFrameNavigation(page, frame) {
|
|
1737
|
+
if (typeof page.mainFrame !== "function") return true;
|
|
1738
|
+
return frame === page.mainFrame();
|
|
1739
|
+
}
|
|
1740
|
+
async function assertSubframeNavigationAllowed(frameUrl, ssrfPolicy) {
|
|
1741
|
+
if (!ssrfPolicy || !frameUrl.startsWith("http://") && !frameUrl.startsWith("https://")) return;
|
|
1742
|
+
await assertBrowserNavigationResultAllowed({
|
|
1743
|
+
url: frameUrl,
|
|
1744
|
+
...withBrowserNavigationPolicy(ssrfPolicy)
|
|
1745
|
+
});
|
|
1746
|
+
}
|
|
1747
|
+
function snapshotNetworkFrameUrl(frame) {
|
|
1748
|
+
try {
|
|
1749
|
+
const frameUrl = frame.url();
|
|
1750
|
+
return frameUrl.startsWith("http://") || frameUrl.startsWith("https://") ? frameUrl : null;
|
|
1751
|
+
} catch {
|
|
1752
|
+
return null;
|
|
1753
|
+
}
|
|
1754
|
+
}
|
|
1755
|
+
async function assertObservedDelayedNavigations(opts) {
|
|
1756
|
+
let subframeError;
|
|
1757
|
+
try {
|
|
1758
|
+
for (const frameUrl of opts.observed.subframes) await assertSubframeNavigationAllowed(frameUrl, opts.ssrfPolicy);
|
|
1759
|
+
} catch (err) {
|
|
1760
|
+
subframeError = err;
|
|
1761
|
+
}
|
|
1762
|
+
if (opts.observed.mainFrameNavigated) await assertPageNavigationCompletedSafely({
|
|
1763
|
+
cdpUrl: opts.cdpUrl,
|
|
1764
|
+
page: opts.page,
|
|
1765
|
+
response: null,
|
|
1766
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
1767
|
+
targetId: opts.targetId
|
|
1768
|
+
});
|
|
1769
|
+
if (subframeError) throw subframeError;
|
|
1770
|
+
}
|
|
1771
|
+
function observeDelayedInteractionNavigation(page, previousUrl) {
|
|
1772
|
+
if (didCrossDocumentUrlChange(page, previousUrl)) return Promise.resolve({
|
|
1773
|
+
mainFrameNavigated: true,
|
|
1774
|
+
subframes: []
|
|
1775
|
+
});
|
|
1776
|
+
if (typeof page.on !== "function" || typeof page.off !== "function") return Promise.resolve({
|
|
1777
|
+
mainFrameNavigated: false,
|
|
1778
|
+
subframes: []
|
|
1779
|
+
});
|
|
1780
|
+
return new Promise((resolve) => {
|
|
1781
|
+
const subframes = [];
|
|
1782
|
+
const onFrameNavigated = (frame) => {
|
|
1783
|
+
if (!isMainFrameNavigation(page, frame)) {
|
|
1784
|
+
const frameUrl = snapshotNetworkFrameUrl(frame);
|
|
1785
|
+
if (frameUrl) subframes.push(frameUrl);
|
|
1786
|
+
return;
|
|
1787
|
+
}
|
|
1788
|
+
if (isHashOnlyNavigation(page.url(), previousUrl)) return;
|
|
1789
|
+
cleanup();
|
|
1790
|
+
resolve({
|
|
1791
|
+
mainFrameNavigated: true,
|
|
1792
|
+
subframes
|
|
1793
|
+
});
|
|
1794
|
+
};
|
|
1795
|
+
const timeout = setTimeout(() => {
|
|
1796
|
+
cleanup();
|
|
1797
|
+
resolve({
|
|
1798
|
+
mainFrameNavigated: didCrossDocumentUrlChange(page, previousUrl),
|
|
1799
|
+
subframes
|
|
1800
|
+
});
|
|
1801
|
+
}, INTERACTION_NAVIGATION_GRACE_MS);
|
|
1802
|
+
const cleanup = () => {
|
|
1803
|
+
clearTimeout(timeout);
|
|
1804
|
+
page.off("framenavigated", onFrameNavigated);
|
|
1805
|
+
};
|
|
1806
|
+
page.on("framenavigated", onFrameNavigated);
|
|
1807
|
+
});
|
|
1808
|
+
}
|
|
1809
|
+
function scheduleDelayedInteractionNavigationGuard(opts) {
|
|
1810
|
+
if (!opts.ssrfPolicy) return Promise.resolve();
|
|
1811
|
+
const page = opts.page;
|
|
1812
|
+
if (didCrossDocumentUrlChange(page, opts.previousUrl)) return assertPageNavigationCompletedSafely({
|
|
1813
|
+
cdpUrl: opts.cdpUrl,
|
|
1814
|
+
page: opts.page,
|
|
1815
|
+
response: null,
|
|
1816
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
1817
|
+
targetId: opts.targetId
|
|
1818
|
+
});
|
|
1819
|
+
if (typeof page.on !== "function" || typeof page.off !== "function") return Promise.resolve();
|
|
1820
|
+
pendingInteractionNavigationGuardCleanup.get(opts.page)?.();
|
|
1821
|
+
return new Promise((resolve, reject) => {
|
|
1822
|
+
const settle = (err) => {
|
|
1823
|
+
cleanup();
|
|
1824
|
+
if (err) {
|
|
1825
|
+
reject(err);
|
|
1826
|
+
return;
|
|
1827
|
+
}
|
|
1828
|
+
resolve();
|
|
1829
|
+
};
|
|
1830
|
+
const subframes = [];
|
|
1831
|
+
const onFrameNavigated = (frame) => {
|
|
1832
|
+
if (!isMainFrameNavigation(page, frame)) {
|
|
1833
|
+
const frameUrl = snapshotNetworkFrameUrl(frame);
|
|
1834
|
+
if (frameUrl) subframes.push(frameUrl);
|
|
1835
|
+
return;
|
|
1836
|
+
}
|
|
1837
|
+
if (isHashOnlyNavigation(page.url(), opts.previousUrl)) return;
|
|
1838
|
+
cleanup();
|
|
1839
|
+
assertObservedDelayedNavigations({
|
|
1840
|
+
cdpUrl: opts.cdpUrl,
|
|
1841
|
+
page: opts.page,
|
|
1842
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
1843
|
+
targetId: opts.targetId,
|
|
1844
|
+
observed: {
|
|
1845
|
+
mainFrameNavigated: true,
|
|
1846
|
+
subframes
|
|
1847
|
+
}
|
|
1848
|
+
}).then(() => settle(), settle);
|
|
1849
|
+
};
|
|
1850
|
+
const timeout = setTimeout(() => {
|
|
1851
|
+
cleanup();
|
|
1852
|
+
assertObservedDelayedNavigations({
|
|
1853
|
+
cdpUrl: opts.cdpUrl,
|
|
1854
|
+
page: opts.page,
|
|
1855
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
1856
|
+
targetId: opts.targetId,
|
|
1857
|
+
observed: {
|
|
1858
|
+
mainFrameNavigated: didCrossDocumentUrlChange(page, opts.previousUrl),
|
|
1859
|
+
subframes
|
|
1860
|
+
}
|
|
1861
|
+
}).then(() => settle(), settle);
|
|
1862
|
+
}, INTERACTION_NAVIGATION_GRACE_MS);
|
|
1863
|
+
const cleanup = () => {
|
|
1864
|
+
clearTimeout(timeout);
|
|
1865
|
+
page.off("framenavigated", onFrameNavigated);
|
|
1866
|
+
if (pendingInteractionNavigationGuardCleanup.get(opts.page) === settle) pendingInteractionNavigationGuardCleanup.delete(opts.page);
|
|
1867
|
+
};
|
|
1868
|
+
pendingInteractionNavigationGuardCleanup.set(opts.page, settle);
|
|
1869
|
+
page.on("framenavigated", onFrameNavigated);
|
|
1870
|
+
});
|
|
1871
|
+
}
|
|
1872
|
+
async function assertInteractionNavigationCompletedSafely(opts) {
|
|
1873
|
+
if (!opts.ssrfPolicy) return await opts.action();
|
|
1874
|
+
const navPage = opts.page;
|
|
1875
|
+
let navigatedDuringAction = false;
|
|
1876
|
+
const subframeNavigationsDuringAction = [];
|
|
1877
|
+
const onFrameNavigated = (frame) => {
|
|
1878
|
+
if (!isMainFrameNavigation(navPage, frame)) {
|
|
1879
|
+
const frameUrl = snapshotNetworkFrameUrl(frame);
|
|
1880
|
+
if (frameUrl) subframeNavigationsDuringAction.push(frameUrl);
|
|
1881
|
+
return;
|
|
1882
|
+
}
|
|
1883
|
+
if (!isHashOnlyNavigation(opts.page.url(), opts.previousUrl)) navigatedDuringAction = true;
|
|
1884
|
+
};
|
|
1885
|
+
if (typeof navPage.on === "function") navPage.on("framenavigated", onFrameNavigated);
|
|
1886
|
+
let result;
|
|
1887
|
+
let actionError = null;
|
|
1888
|
+
try {
|
|
1889
|
+
result = await opts.action();
|
|
1890
|
+
} catch (err) {
|
|
1891
|
+
actionError = err;
|
|
1892
|
+
} finally {
|
|
1893
|
+
if (typeof navPage.off === "function") navPage.off("framenavigated", onFrameNavigated);
|
|
1894
|
+
}
|
|
1895
|
+
const navigationObserved = navigatedDuringAction || didCrossDocumentUrlChange(opts.page, opts.previousUrl);
|
|
1896
|
+
let subframeError;
|
|
1897
|
+
try {
|
|
1898
|
+
for (const frameUrl of subframeNavigationsDuringAction) await assertSubframeNavigationAllowed(frameUrl, opts.ssrfPolicy);
|
|
1899
|
+
} catch (err) {
|
|
1900
|
+
subframeError = err;
|
|
1901
|
+
}
|
|
1902
|
+
if (navigationObserved) await assertPageNavigationCompletedSafely({
|
|
1903
|
+
cdpUrl: opts.cdpUrl,
|
|
1904
|
+
page: opts.page,
|
|
1905
|
+
response: null,
|
|
1906
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
1907
|
+
targetId: opts.targetId
|
|
1908
|
+
});
|
|
1909
|
+
else if (actionError) {
|
|
1910
|
+
const observed = await observeDelayedInteractionNavigation(opts.page, opts.previousUrl);
|
|
1911
|
+
if (observed.mainFrameNavigated || observed.subframes.length > 0) await assertObservedDelayedNavigations({
|
|
1912
|
+
cdpUrl: opts.cdpUrl,
|
|
1913
|
+
page: opts.page,
|
|
1914
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
1915
|
+
targetId: opts.targetId,
|
|
1916
|
+
observed
|
|
1917
|
+
});
|
|
1918
|
+
} else await scheduleDelayedInteractionNavigationGuard({
|
|
1919
|
+
cdpUrl: opts.cdpUrl,
|
|
1920
|
+
page: opts.page,
|
|
1921
|
+
previousUrl: opts.previousUrl,
|
|
1922
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
1923
|
+
targetId: opts.targetId
|
|
1924
|
+
});
|
|
1925
|
+
if (subframeError) throw subframeError;
|
|
1926
|
+
if (actionError) throw actionError;
|
|
1927
|
+
return result;
|
|
1928
|
+
}
|
|
1929
|
+
async function awaitActionWithAbort(actionPromise, abortPromise, onActionResolvedAfterAbort) {
|
|
1930
|
+
if (!abortPromise) return await actionPromise;
|
|
1931
|
+
try {
|
|
1932
|
+
return await Promise.race([actionPromise, abortPromise]);
|
|
1933
|
+
} catch (err) {
|
|
1934
|
+
actionPromise.then(() => onActionResolvedAfterAbort?.(), () => {});
|
|
1935
|
+
throw err;
|
|
1936
|
+
}
|
|
1937
|
+
}
|
|
1938
|
+
function createAbortPromise(signal) {
|
|
1939
|
+
return createAbortPromiseWithListener(signal);
|
|
1940
|
+
}
|
|
1941
|
+
function createAbortPromiseWithListener(signal, onAbort) {
|
|
1942
|
+
if (!signal) return { cleanup: () => {} };
|
|
1943
|
+
let abortListener;
|
|
1944
|
+
const abortPromise = signal.aborted ? (() => {
|
|
1945
|
+
onAbort?.(signal.reason);
|
|
1946
|
+
return Promise.reject(signal.reason ?? /* @__PURE__ */ new Error("已中止"));
|
|
1947
|
+
})() : new Promise((_, reject) => {
|
|
1948
|
+
abortListener = () => {
|
|
1949
|
+
onAbort?.(signal.reason);
|
|
1950
|
+
reject(signal.reason ?? /* @__PURE__ */ new Error("已中止"));
|
|
1951
|
+
};
|
|
1952
|
+
signal.addEventListener("abort", abortListener, { once: true });
|
|
1953
|
+
});
|
|
1954
|
+
abortPromise.catch(() => {});
|
|
1955
|
+
return {
|
|
1956
|
+
abortPromise,
|
|
1957
|
+
cleanup: () => {
|
|
1958
|
+
if (abortListener) signal.removeEventListener("abort", abortListener);
|
|
1959
|
+
}
|
|
1960
|
+
};
|
|
1961
|
+
}
|
|
1962
|
+
async function highlightViaPlaywright(opts) {
|
|
1963
|
+
const page = await getRestoredPageForTarget(opts);
|
|
1964
|
+
const ref = requireRef(opts.ref);
|
|
1965
|
+
try {
|
|
1966
|
+
await refLocator(page, ref).highlight();
|
|
1967
|
+
} catch (err) {
|
|
1968
|
+
throw toFriendlyInteractionError(err, ref);
|
|
1969
|
+
}
|
|
1970
|
+
}
|
|
1971
|
+
async function clickViaPlaywright(opts) {
|
|
1972
|
+
const resolved = requireRefOrSelector(opts.ref, opts.selector);
|
|
1973
|
+
const page = await getRestoredPageForTarget(opts);
|
|
1974
|
+
const label = resolved.ref ?? resolved.selector;
|
|
1975
|
+
const locator = resolved.ref ? refLocator(page, requireRef(resolved.ref)) : page.locator(resolved.selector);
|
|
1976
|
+
const timeout = resolveInteractionTimeoutMs(opts.timeoutMs);
|
|
1977
|
+
const previousUrl = page.url();
|
|
1978
|
+
const signal = opts.signal;
|
|
1979
|
+
let abortListener;
|
|
1980
|
+
let abortReject;
|
|
1981
|
+
let abortPromise;
|
|
1982
|
+
if (signal) {
|
|
1983
|
+
abortPromise = new Promise((_, reject) => {
|
|
1984
|
+
abortReject = reject;
|
|
1985
|
+
});
|
|
1986
|
+
abortPromise.catch(() => {});
|
|
1987
|
+
const disconnect = () => {
|
|
1988
|
+
if (isBrowserObservedDialogBlockedError(signal.reason)) return;
|
|
1989
|
+
forceDisconnectPlaywrightForTarget({
|
|
1990
|
+
cdpUrl: opts.cdpUrl,
|
|
1991
|
+
targetId: opts.targetId,
|
|
1992
|
+
reason: "click aborted"
|
|
1993
|
+
}).catch(() => {});
|
|
1994
|
+
};
|
|
1995
|
+
if (signal.aborted) {
|
|
1996
|
+
disconnect();
|
|
1997
|
+
throw signal.reason ?? /* @__PURE__ */ new Error("已中止");
|
|
1998
|
+
}
|
|
1999
|
+
abortListener = () => {
|
|
2000
|
+
disconnect();
|
|
2001
|
+
abortReject?.(signal.reason ?? /* @__PURE__ */ new Error("已中止"));
|
|
2002
|
+
};
|
|
2003
|
+
signal.addEventListener("abort", abortListener, { once: true });
|
|
2004
|
+
if (signal.aborted) {
|
|
2005
|
+
abortListener();
|
|
2006
|
+
throw signal.reason ?? /* @__PURE__ */ new Error("已中止");
|
|
2007
|
+
}
|
|
2008
|
+
}
|
|
2009
|
+
const reconcileRemoteDialog = () => reconcileRemoteDialogAfterActionSettled(page, signal);
|
|
2010
|
+
try {
|
|
2011
|
+
await assertInteractionNavigationCompletedSafely({
|
|
2012
|
+
action: async () => {
|
|
2013
|
+
const delayMs = resolveBoundedDelayMs(opts.delayMs, "click delayMs", ACT_MAX_CLICK_DELAY_MS);
|
|
2014
|
+
if (delayMs > 0) {
|
|
2015
|
+
await awaitActionWithAbort(locator.hover({ timeout }), abortPromise, reconcileRemoteDialog);
|
|
2016
|
+
await new Promise((resolve) => setTimeout(resolve, delayMs));
|
|
2017
|
+
}
|
|
2018
|
+
if (opts.doubleClick) {
|
|
2019
|
+
await awaitActionWithAbort(locator.dblclick({
|
|
2020
|
+
timeout,
|
|
2021
|
+
button: opts.button,
|
|
2022
|
+
modifiers: opts.modifiers
|
|
2023
|
+
}), abortPromise, reconcileRemoteDialog);
|
|
2024
|
+
return;
|
|
2025
|
+
}
|
|
2026
|
+
await awaitActionWithAbort(locator.click({
|
|
2027
|
+
timeout,
|
|
2028
|
+
button: opts.button,
|
|
2029
|
+
modifiers: opts.modifiers
|
|
2030
|
+
}), abortPromise, reconcileRemoteDialog);
|
|
2031
|
+
},
|
|
2032
|
+
cdpUrl: opts.cdpUrl,
|
|
2033
|
+
page,
|
|
2034
|
+
previousUrl,
|
|
2035
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
2036
|
+
targetId: opts.targetId
|
|
2037
|
+
});
|
|
2038
|
+
} catch (err) {
|
|
2039
|
+
throw toFriendlyInteractionError(err, label);
|
|
2040
|
+
} finally {
|
|
2041
|
+
if (signal && abortListener) signal.removeEventListener("abort", abortListener);
|
|
2042
|
+
}
|
|
2043
|
+
}
|
|
2044
|
+
async function clickCoordsViaPlaywright(opts) {
|
|
2045
|
+
const page = await getRestoredPageForTarget(opts);
|
|
2046
|
+
const previousUrl = page.url();
|
|
2047
|
+
const { abortPromise, cleanup } = createAbortPromise(opts.signal);
|
|
2048
|
+
const reconcileRemoteDialog = () => reconcileRemoteDialogAfterActionSettled(page, opts.signal);
|
|
2049
|
+
await assertInteractionNavigationCompletedSafely({
|
|
2050
|
+
action: async () => {
|
|
2051
|
+
await awaitActionWithAbort(page.mouse.click(opts.x, opts.y, {
|
|
2052
|
+
button: opts.button,
|
|
2053
|
+
clickCount: opts.doubleClick ? 2 : 1,
|
|
2054
|
+
delay: resolveBoundedDelayMs(opts.delayMs, "clickCoords delayMs", ACT_MAX_CLICK_DELAY_MS)
|
|
2055
|
+
}), abortPromise, reconcileRemoteDialog);
|
|
2056
|
+
},
|
|
2057
|
+
cdpUrl: opts.cdpUrl,
|
|
2058
|
+
page,
|
|
2059
|
+
previousUrl,
|
|
2060
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
2061
|
+
targetId: opts.targetId
|
|
2062
|
+
}).finally(cleanup);
|
|
2063
|
+
}
|
|
2064
|
+
async function hoverViaPlaywright(opts) {
|
|
2065
|
+
const resolved = requireRefOrSelector(opts.ref, opts.selector);
|
|
2066
|
+
const page = await getRestoredPageForTarget(opts);
|
|
2067
|
+
const label = resolved.ref ?? resolved.selector;
|
|
2068
|
+
const locator = resolved.ref ? refLocator(page, requireRef(resolved.ref)) : page.locator(resolved.selector);
|
|
2069
|
+
const { abortPromise, cleanup } = createAbortPromise(opts.signal);
|
|
2070
|
+
const reconcileRemoteDialog = () => reconcileRemoteDialogAfterActionSettled(page, opts.signal);
|
|
2071
|
+
try {
|
|
2072
|
+
await awaitActionWithAbort(locator.hover({ timeout: resolveInteractionTimeoutMs(opts.timeoutMs) }), abortPromise, reconcileRemoteDialog);
|
|
2073
|
+
} catch (err) {
|
|
2074
|
+
throw toFriendlyInteractionError(err, label);
|
|
2075
|
+
} finally {
|
|
2076
|
+
cleanup();
|
|
2077
|
+
}
|
|
2078
|
+
}
|
|
2079
|
+
async function dragViaPlaywright(opts) {
|
|
2080
|
+
const resolvedStart = requireRefOrSelector(opts.startRef, opts.startSelector);
|
|
2081
|
+
const resolvedEnd = requireRefOrSelector(opts.endRef, opts.endSelector);
|
|
2082
|
+
const page = await getRestoredPageForTarget(opts);
|
|
2083
|
+
const startLocator = resolvedStart.ref ? refLocator(page, requireRef(resolvedStart.ref)) : page.locator(resolvedStart.selector);
|
|
2084
|
+
const endLocator = resolvedEnd.ref ? refLocator(page, requireRef(resolvedEnd.ref)) : page.locator(resolvedEnd.selector);
|
|
2085
|
+
const startLabel = resolvedStart.ref ?? resolvedStart.selector;
|
|
2086
|
+
const endLabel = resolvedEnd.ref ?? resolvedEnd.selector;
|
|
2087
|
+
const { abortPromise, cleanup } = createAbortPromise(opts.signal);
|
|
2088
|
+
const reconcileRemoteDialog = () => reconcileRemoteDialogAfterActionSettled(page, opts.signal);
|
|
2089
|
+
try {
|
|
2090
|
+
await awaitActionWithAbort(startLocator.dragTo(endLocator, { timeout: resolveInteractionTimeoutMs(opts.timeoutMs) }), abortPromise, reconcileRemoteDialog);
|
|
2091
|
+
} catch (err) {
|
|
2092
|
+
throw toFriendlyInteractionError(err, `${startLabel} -> ${endLabel}`);
|
|
2093
|
+
} finally {
|
|
2094
|
+
cleanup();
|
|
2095
|
+
}
|
|
2096
|
+
}
|
|
2097
|
+
async function selectOptionViaPlaywright(opts) {
|
|
2098
|
+
const resolved = requireRefOrSelector(opts.ref, opts.selector);
|
|
2099
|
+
if (!opts.values?.length) throw new Error("值列表是必需的");
|
|
2100
|
+
const page = await getRestoredPageForTarget(opts);
|
|
2101
|
+
const label = resolved.ref ?? resolved.selector;
|
|
2102
|
+
const locator = resolved.ref ? refLocator(page, requireRef(resolved.ref)) : page.locator(resolved.selector);
|
|
2103
|
+
const previousUrl = page.url();
|
|
2104
|
+
const { abortPromise, cleanup } = createAbortPromise(opts.signal);
|
|
2105
|
+
const reconcileRemoteDialog = () => reconcileRemoteDialogAfterActionSettled(page, opts.signal);
|
|
2106
|
+
try {
|
|
2107
|
+
await assertInteractionNavigationCompletedSafely({
|
|
2108
|
+
action: async () => {
|
|
2109
|
+
await awaitActionWithAbort(locator.selectOption(opts.values, { timeout: resolveInteractionTimeoutMs(opts.timeoutMs) }), abortPromise, reconcileRemoteDialog);
|
|
2110
|
+
},
|
|
2111
|
+
cdpUrl: opts.cdpUrl,
|
|
2112
|
+
page,
|
|
2113
|
+
previousUrl,
|
|
2114
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
2115
|
+
targetId: opts.targetId
|
|
2116
|
+
});
|
|
2117
|
+
} catch (err) {
|
|
2118
|
+
throw toFriendlyInteractionError(err, label);
|
|
2119
|
+
} finally {
|
|
2120
|
+
cleanup();
|
|
2121
|
+
}
|
|
2122
|
+
}
|
|
2123
|
+
async function pressKeyViaPlaywright(opts) {
|
|
2124
|
+
const key = normalizeOptionalString(opts.key) ?? "";
|
|
2125
|
+
if (!key) throw new Error("key 是必需的");
|
|
2126
|
+
const page = await getPageForTargetId(opts);
|
|
2127
|
+
ensurePageState(page);
|
|
2128
|
+
const previousUrl = page.url();
|
|
2129
|
+
const { abortPromise, cleanup } = createAbortPromise(opts.signal);
|
|
2130
|
+
const reconcileRemoteDialog = () => reconcileRemoteDialogAfterActionSettled(page, opts.signal);
|
|
2131
|
+
try {
|
|
2132
|
+
await assertInteractionNavigationCompletedSafely({
|
|
2133
|
+
action: async () => {
|
|
2134
|
+
await awaitActionWithAbort(page.keyboard.press(key, { delay: resolveNonNegativeIntegerOption(opts.delayMs, 0) }), abortPromise, reconcileRemoteDialog);
|
|
2135
|
+
},
|
|
2136
|
+
cdpUrl: opts.cdpUrl,
|
|
2137
|
+
page,
|
|
2138
|
+
previousUrl,
|
|
2139
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
2140
|
+
targetId: opts.targetId
|
|
2141
|
+
});
|
|
2142
|
+
} finally {
|
|
2143
|
+
cleanup();
|
|
2144
|
+
}
|
|
2145
|
+
}
|
|
2146
|
+
async function typeViaPlaywright(opts) {
|
|
2147
|
+
const resolved = requireRefOrSelector(opts.ref, opts.selector);
|
|
2148
|
+
const text = opts.text ?? "";
|
|
2149
|
+
const page = await getRestoredPageForTarget(opts);
|
|
2150
|
+
const label = resolved.ref ?? resolved.selector;
|
|
2151
|
+
const locator = resolved.ref ? refLocator(page, requireRef(resolved.ref)) : page.locator(resolved.selector);
|
|
2152
|
+
const timeout = resolveInteractionTimeoutMs(opts.timeoutMs);
|
|
2153
|
+
const { abortPromise, cleanup } = createAbortPromise(opts.signal);
|
|
2154
|
+
const reconcileRemoteDialog = () => reconcileRemoteDialogAfterActionSettled(page, opts.signal);
|
|
2155
|
+
try {
|
|
2156
|
+
const previousUrl = page.url();
|
|
2157
|
+
if (opts.slowly) await assertInteractionNavigationCompletedSafely({
|
|
2158
|
+
action: async () => {
|
|
2159
|
+
await awaitActionWithAbort(locator.click({ timeout }), abortPromise, reconcileRemoteDialog);
|
|
2160
|
+
await awaitActionWithAbort(locator.type(text, {
|
|
2161
|
+
timeout,
|
|
2162
|
+
delay: 75
|
|
2163
|
+
}), abortPromise, reconcileRemoteDialog);
|
|
2164
|
+
if (opts.submit) await awaitActionWithAbort(locator.press("Enter", { timeout }), abortPromise, reconcileRemoteDialog);
|
|
2165
|
+
},
|
|
2166
|
+
cdpUrl: opts.cdpUrl,
|
|
2167
|
+
page,
|
|
2168
|
+
previousUrl,
|
|
2169
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
2170
|
+
targetId: opts.targetId
|
|
2171
|
+
});
|
|
2172
|
+
else await assertInteractionNavigationCompletedSafely({
|
|
2173
|
+
action: async () => {
|
|
2174
|
+
await awaitActionWithAbort(locator.fill(text, { timeout }), abortPromise, reconcileRemoteDialog);
|
|
2175
|
+
if (opts.submit) await awaitActionWithAbort(locator.press("Enter", { timeout }), abortPromise, reconcileRemoteDialog);
|
|
2176
|
+
},
|
|
2177
|
+
cdpUrl: opts.cdpUrl,
|
|
2178
|
+
page,
|
|
2179
|
+
previousUrl,
|
|
2180
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
2181
|
+
targetId: opts.targetId
|
|
2182
|
+
});
|
|
2183
|
+
} catch (err) {
|
|
2184
|
+
throw toFriendlyInteractionError(err, label);
|
|
2185
|
+
} finally {
|
|
2186
|
+
cleanup();
|
|
2187
|
+
}
|
|
2188
|
+
}
|
|
2189
|
+
async function fillFormViaPlaywright(opts) {
|
|
2190
|
+
const page = await getRestoredPageForTarget(opts);
|
|
2191
|
+
const timeout = resolveInteractionTimeoutMs(opts.timeoutMs);
|
|
2192
|
+
const { abortPromise, cleanup } = createAbortPromise(opts.signal);
|
|
2193
|
+
const reconcileRemoteDialog = () => reconcileRemoteDialogAfterActionSettled(page, opts.signal);
|
|
2194
|
+
try {
|
|
2195
|
+
for (const field of opts.fields) {
|
|
2196
|
+
const ref = field.ref.trim();
|
|
2197
|
+
const type = (field.type || "text").trim() || "text";
|
|
2198
|
+
const rawValue = field.value;
|
|
2199
|
+
const value = typeof rawValue === "string" ? rawValue : typeof rawValue === "number" || typeof rawValue === "boolean" ? String(rawValue) : "";
|
|
2200
|
+
if (!ref) continue;
|
|
2201
|
+
const locator = refLocator(page, ref);
|
|
2202
|
+
if (type === "checkbox" || type === "radio") {
|
|
2203
|
+
const checked = rawValue === true || rawValue === 1 || rawValue === "1" || rawValue === "true";
|
|
2204
|
+
try {
|
|
2205
|
+
const previousUrl = page.url();
|
|
2206
|
+
await assertInteractionNavigationCompletedSafely({
|
|
2207
|
+
action: async () => {
|
|
2208
|
+
await awaitActionWithAbort(locator.setChecked(checked, { timeout }), abortPromise, reconcileRemoteDialog);
|
|
2209
|
+
},
|
|
2210
|
+
cdpUrl: opts.cdpUrl,
|
|
2211
|
+
page,
|
|
2212
|
+
previousUrl,
|
|
2213
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
2214
|
+
targetId: opts.targetId
|
|
2215
|
+
});
|
|
2216
|
+
} catch (err) {
|
|
2217
|
+
throw toFriendlyInteractionError(err, ref);
|
|
2218
|
+
}
|
|
2219
|
+
continue;
|
|
2220
|
+
}
|
|
2221
|
+
try {
|
|
2222
|
+
const previousUrl = page.url();
|
|
2223
|
+
await assertInteractionNavigationCompletedSafely({
|
|
2224
|
+
action: async () => {
|
|
2225
|
+
await awaitActionWithAbort(locator.fill(value, { timeout }), abortPromise, reconcileRemoteDialog);
|
|
2226
|
+
},
|
|
2227
|
+
cdpUrl: opts.cdpUrl,
|
|
2228
|
+
page,
|
|
2229
|
+
previousUrl,
|
|
2230
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
2231
|
+
targetId: opts.targetId
|
|
2232
|
+
});
|
|
2233
|
+
} catch (err) {
|
|
2234
|
+
throw toFriendlyInteractionError(err, ref);
|
|
2235
|
+
}
|
|
2236
|
+
}
|
|
2237
|
+
} finally {
|
|
2238
|
+
cleanup();
|
|
2239
|
+
}
|
|
2240
|
+
}
|
|
2241
|
+
async function evaluateViaPlaywright(opts) {
|
|
2242
|
+
const fnText = normalizeOptionalString(opts.fn) ?? "";
|
|
2243
|
+
if (!fnText) throw new Error("需要 function");
|
|
2244
|
+
const page = await getRestoredPageForTarget(opts);
|
|
2245
|
+
const outerTimeout = normalizeTimeoutMs(opts.timeoutMs, 2e4);
|
|
2246
|
+
let evaluateTimeout = Math.max(1e3, Math.min(12e4, outerTimeout - 500));
|
|
2247
|
+
evaluateTimeout = Math.min(evaluateTimeout, outerTimeout);
|
|
2248
|
+
const signal = opts.signal;
|
|
2249
|
+
const { abortPromise, cleanup } = createAbortPromiseWithListener(signal, (reason) => {
|
|
2250
|
+
if (isBrowserObservedDialogBlockedError(reason)) return;
|
|
2251
|
+
forceDisconnectPlaywrightForTarget({
|
|
2252
|
+
cdpUrl: opts.cdpUrl,
|
|
2253
|
+
targetId: opts.targetId,
|
|
2254
|
+
reason: "evaluate aborted"
|
|
2255
|
+
}).catch(() => {});
|
|
2256
|
+
});
|
|
2257
|
+
if (signal?.aborted) throw signal.reason ?? /* @__PURE__ */ new Error("已中止");
|
|
2258
|
+
try {
|
|
2259
|
+
const previousUrl = page.url();
|
|
2260
|
+
if (opts.ssrfPolicy) await assertPageNavigationCompletedSafely({
|
|
2261
|
+
cdpUrl: opts.cdpUrl,
|
|
2262
|
+
page,
|
|
2263
|
+
response: null,
|
|
2264
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
2265
|
+
targetId: opts.targetId
|
|
2266
|
+
});
|
|
2267
|
+
if (opts.ref) {
|
|
2268
|
+
const locator = refLocator(page, opts.ref);
|
|
2269
|
+
const elementEvaluator = new Function("el", "args", `
|
|
2270
|
+
"use strict";
|
|
2271
|
+
var fnBody = args.fnBody, timeoutMs = args.timeoutMs;
|
|
2272
|
+
try {
|
|
2273
|
+
var candidate = eval("(" + fnBody + ")");
|
|
2274
|
+
var result = typeof candidate === "function" ? candidate(el) : candidate;
|
|
2275
|
+
if (result && typeof result.then === "function") {
|
|
2276
|
+
return Promise.race([
|
|
2277
|
+
result,
|
|
2278
|
+
new Promise(function(_, reject) {
|
|
2279
|
+
setTimeout(function() { reject(new Error("evaluate timed out after " + timeoutMs + "ms")); }, timeoutMs);
|
|
2280
|
+
})
|
|
2281
|
+
]);
|
|
2282
|
+
}
|
|
2283
|
+
return result;
|
|
2284
|
+
} catch (err) {
|
|
2285
|
+
throw new Error("Invalid evaluate function: " + (err && err.message ? err.message : String(err)));
|
|
2286
|
+
}
|
|
2287
|
+
`);
|
|
2288
|
+
const evalPromise = locator.evaluate(elementEvaluator, {
|
|
2289
|
+
fnBody: fnText,
|
|
2290
|
+
timeoutMs: evaluateTimeout
|
|
2291
|
+
});
|
|
2292
|
+
const reconcileRemoteDialog = () => reconcileRemoteDialogAfterActionSettled(page, signal);
|
|
2293
|
+
return await assertInteractionNavigationCompletedSafely({
|
|
2294
|
+
action: () => awaitActionWithAbort(evalPromise, abortPromise, reconcileRemoteDialog),
|
|
2295
|
+
cdpUrl: opts.cdpUrl,
|
|
2296
|
+
page,
|
|
2297
|
+
previousUrl,
|
|
2298
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
2299
|
+
targetId: opts.targetId
|
|
2300
|
+
});
|
|
2301
|
+
}
|
|
2302
|
+
const browserEvaluator = new Function("args", `
|
|
2303
|
+
"use strict";
|
|
2304
|
+
var fnBody = args.fnBody, timeoutMs = args.timeoutMs;
|
|
2305
|
+
try {
|
|
2306
|
+
var candidate = eval("(" + fnBody + ")");
|
|
2307
|
+
var result = typeof candidate === "function" ? candidate() : candidate;
|
|
2308
|
+
if (result && typeof result.then === "function") {
|
|
2309
|
+
return Promise.race([
|
|
2310
|
+
result,
|
|
2311
|
+
new Promise(function(_, reject) {
|
|
2312
|
+
setTimeout(function() { reject(new Error("evaluate timed out after " + timeoutMs + "ms")); }, timeoutMs);
|
|
2313
|
+
})
|
|
2314
|
+
]);
|
|
2315
|
+
}
|
|
2316
|
+
return result;
|
|
2317
|
+
} catch (err) {
|
|
2318
|
+
throw new Error("Invalid evaluate function: " + (err && err.message ? err.message : String(err)));
|
|
2319
|
+
}
|
|
2320
|
+
`);
|
|
2321
|
+
const evalPromise = page.evaluate(browserEvaluator, {
|
|
2322
|
+
fnBody: fnText,
|
|
2323
|
+
timeoutMs: evaluateTimeout
|
|
2324
|
+
});
|
|
2325
|
+
const reconcileRemoteDialog = () => reconcileRemoteDialogAfterActionSettled(page, signal);
|
|
2326
|
+
return await assertInteractionNavigationCompletedSafely({
|
|
2327
|
+
action: () => awaitActionWithAbort(evalPromise, abortPromise, reconcileRemoteDialog),
|
|
2328
|
+
cdpUrl: opts.cdpUrl,
|
|
2329
|
+
page,
|
|
2330
|
+
previousUrl,
|
|
2331
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
2332
|
+
targetId: opts.targetId
|
|
2333
|
+
});
|
|
2334
|
+
} finally {
|
|
2335
|
+
cleanup();
|
|
2336
|
+
}
|
|
2337
|
+
}
|
|
2338
|
+
async function scrollIntoViewViaPlaywright(opts) {
|
|
2339
|
+
const resolved = requireRefOrSelector(opts.ref, opts.selector);
|
|
2340
|
+
const page = await getRestoredPageForTarget(opts);
|
|
2341
|
+
const timeout = normalizeTimeoutMs(opts.timeoutMs, 2e4);
|
|
2342
|
+
const label = resolved.ref ?? resolved.selector;
|
|
2343
|
+
const locator = resolved.ref ? refLocator(page, requireRef(resolved.ref)) : page.locator(resolved.selector);
|
|
2344
|
+
const { abortPromise, cleanup } = createAbortPromise(opts.signal);
|
|
2345
|
+
const reconcileRemoteDialog = () => reconcileRemoteDialogAfterActionSettled(page, opts.signal);
|
|
2346
|
+
try {
|
|
2347
|
+
await awaitActionWithAbort(locator.scrollIntoViewIfNeeded({ timeout }), abortPromise, reconcileRemoteDialog);
|
|
2348
|
+
} catch (err) {
|
|
2349
|
+
throw toFriendlyInteractionError(err, label);
|
|
2350
|
+
} finally {
|
|
2351
|
+
cleanup();
|
|
2352
|
+
}
|
|
2353
|
+
}
|
|
2354
|
+
async function waitForViaPlaywright(opts) {
|
|
2355
|
+
const page = await getPageForTargetId(opts);
|
|
2356
|
+
ensurePageState(page);
|
|
2357
|
+
const timeout = resolveActWaitTimeoutMs(opts.timeoutMs);
|
|
2358
|
+
const { abortPromise, cleanup } = createAbortPromise(opts.signal);
|
|
2359
|
+
const reconcileRemoteDialog = () => reconcileRemoteDialogAfterActionSettled(page, opts.signal);
|
|
2360
|
+
const waitForStep = async (stepPromise) => {
|
|
2361
|
+
await awaitActionWithAbort(stepPromise, abortPromise, reconcileRemoteDialog);
|
|
2362
|
+
};
|
|
2363
|
+
try {
|
|
2364
|
+
if (typeof opts.timeMs === "number" && Number.isFinite(opts.timeMs)) await waitForStep(page.waitForTimeout(resolveBoundedDelayMs(opts.timeMs, "wait timeMs", ACT_MAX_WAIT_TIME_MS)));
|
|
2365
|
+
if (opts.text) await waitForStep(page.getByText(opts.text).first().waitFor({
|
|
2366
|
+
state: "visible",
|
|
2367
|
+
timeout
|
|
2368
|
+
}));
|
|
2369
|
+
if (opts.textGone) await waitForStep(page.getByText(opts.textGone).first().waitFor({
|
|
2370
|
+
state: "hidden",
|
|
2371
|
+
timeout
|
|
2372
|
+
}));
|
|
2373
|
+
if (opts.selector) {
|
|
2374
|
+
const selector = normalizeOptionalString(opts.selector) ?? "";
|
|
2375
|
+
if (selector) await waitForStep(page.locator(selector).first().waitFor({
|
|
2376
|
+
state: "visible",
|
|
2377
|
+
timeout
|
|
2378
|
+
}));
|
|
2379
|
+
}
|
|
2380
|
+
if (opts.url) {
|
|
2381
|
+
const url = normalizeOptionalString(opts.url) ?? "";
|
|
2382
|
+
if (url) await waitForStep(page.waitForURL(url, { timeout }));
|
|
2383
|
+
}
|
|
2384
|
+
if (opts.loadState) await waitForStep(page.waitForLoadState(opts.loadState, { timeout }));
|
|
2385
|
+
if (opts.fn) {
|
|
2386
|
+
const fn = normalizeOptionalString(opts.fn) ?? "";
|
|
2387
|
+
if (fn) await waitForStep(page.waitForFunction(fn, { timeout }));
|
|
2388
|
+
}
|
|
2389
|
+
} finally {
|
|
2390
|
+
cleanup();
|
|
2391
|
+
}
|
|
2392
|
+
}
|
|
2393
|
+
async function takeScreenshotViaPlaywright(opts) {
|
|
2394
|
+
const page = await getPageForTargetId(opts);
|
|
2395
|
+
ensurePageState(page);
|
|
2396
|
+
restoreRoleRefsForTarget({
|
|
2397
|
+
cdpUrl: opts.cdpUrl,
|
|
2398
|
+
targetId: opts.targetId,
|
|
2399
|
+
page
|
|
2400
|
+
});
|
|
2401
|
+
const type = opts.type ?? "png";
|
|
2402
|
+
if (opts.ref) {
|
|
2403
|
+
if (opts.fullPage) throw new Error("fullPage 不支持元素截图");
|
|
2404
|
+
return { buffer: await refLocator(page, opts.ref).screenshot({
|
|
2405
|
+
type,
|
|
2406
|
+
timeout: opts.timeoutMs
|
|
2407
|
+
}) };
|
|
2408
|
+
}
|
|
2409
|
+
if (opts.element) {
|
|
2410
|
+
if (opts.fullPage) throw new Error("fullPage 不支持元素截图");
|
|
2411
|
+
return { buffer: await page.locator(opts.element).first().screenshot({
|
|
2412
|
+
type,
|
|
2413
|
+
timeout: opts.timeoutMs
|
|
2414
|
+
}) };
|
|
2415
|
+
}
|
|
2416
|
+
return { buffer: await page.screenshot({
|
|
2417
|
+
type,
|
|
2418
|
+
fullPage: Boolean(opts.fullPage),
|
|
2419
|
+
timeout: opts.timeoutMs
|
|
2420
|
+
}) };
|
|
2421
|
+
}
|
|
2422
|
+
async function screenshotWithLabelsViaPlaywright(opts) {
|
|
2423
|
+
const page = await getPageForTargetId(opts);
|
|
2424
|
+
ensurePageState(page);
|
|
2425
|
+
restoreRoleRefsForTarget({
|
|
2426
|
+
cdpUrl: opts.cdpUrl,
|
|
2427
|
+
targetId: opts.targetId,
|
|
2428
|
+
page
|
|
2429
|
+
});
|
|
2430
|
+
const type = opts.type ?? "png";
|
|
2431
|
+
const maxLabels = typeof opts.maxLabels === "number" && Number.isFinite(opts.maxLabels) ? Math.max(1, Math.floor(opts.maxLabels)) : 150;
|
|
2432
|
+
const viewport = await page.evaluate(() => ({
|
|
2433
|
+
scrollX: window.scrollX || 0,
|
|
2434
|
+
scrollY: window.scrollY || 0,
|
|
2435
|
+
width: window.innerWidth || 0,
|
|
2436
|
+
height: window.innerHeight || 0
|
|
2437
|
+
}));
|
|
2438
|
+
const refs = Object.keys(opts.refs ?? {});
|
|
2439
|
+
const boxes = [];
|
|
2440
|
+
let skipped = 0;
|
|
2441
|
+
for (const ref of refs) {
|
|
2442
|
+
if (boxes.length >= maxLabels) {
|
|
2443
|
+
skipped += 1;
|
|
2444
|
+
continue;
|
|
2445
|
+
}
|
|
2446
|
+
try {
|
|
2447
|
+
const box = await refLocator(page, ref).boundingBox();
|
|
2448
|
+
if (!box) {
|
|
2449
|
+
skipped += 1;
|
|
2450
|
+
continue;
|
|
2451
|
+
}
|
|
2452
|
+
const x0 = box.x;
|
|
2453
|
+
const y0 = box.y;
|
|
2454
|
+
const x1 = box.x + box.width;
|
|
2455
|
+
const y1 = box.y + box.height;
|
|
2456
|
+
const vx0 = viewport.scrollX;
|
|
2457
|
+
const vy0 = viewport.scrollY;
|
|
2458
|
+
const vx1 = viewport.scrollX + viewport.width;
|
|
2459
|
+
const vy1 = viewport.scrollY + viewport.height;
|
|
2460
|
+
if (x1 < vx0 || x0 > vx1 || y1 < vy0 || y0 > vy1) {
|
|
2461
|
+
skipped += 1;
|
|
2462
|
+
continue;
|
|
2463
|
+
}
|
|
2464
|
+
boxes.push({
|
|
2465
|
+
ref,
|
|
2466
|
+
x: x0 - viewport.scrollX,
|
|
2467
|
+
y: y0 - viewport.scrollY,
|
|
2468
|
+
w: Math.max(1, box.width),
|
|
2469
|
+
h: Math.max(1, box.height)
|
|
2470
|
+
});
|
|
2471
|
+
} catch {
|
|
2472
|
+
skipped += 1;
|
|
2473
|
+
}
|
|
2474
|
+
}
|
|
2475
|
+
try {
|
|
2476
|
+
if (boxes.length > 0) await page.evaluate((labels) => {
|
|
2477
|
+
document.querySelectorAll("[data-fengming-labels]").forEach((el) => el.remove());
|
|
2478
|
+
const root = document.createElement("div");
|
|
2479
|
+
root.setAttribute("data-fengming-labels", "1");
|
|
2480
|
+
root.style.position = "fixed";
|
|
2481
|
+
root.style.left = "0";
|
|
2482
|
+
root.style.top = "0";
|
|
2483
|
+
root.style.zIndex = "2147483647";
|
|
2484
|
+
root.style.pointerEvents = "none";
|
|
2485
|
+
root.style.fontFamily = "\"SF Mono\",\"SFMono-Regular\",Menlo,Monaco,Consolas,\"Liberation Mono\",\"Courier New\",monospace";
|
|
2486
|
+
const clamp = (value, min, max) => Math.min(max, Math.max(min, value));
|
|
2487
|
+
for (const label of labels) {
|
|
2488
|
+
const box = document.createElement("div");
|
|
2489
|
+
box.setAttribute("data-fengming-labels", "1");
|
|
2490
|
+
box.style.position = "absolute";
|
|
2491
|
+
box.style.left = `${label.x}px`;
|
|
2492
|
+
box.style.top = `${label.y}px`;
|
|
2493
|
+
box.style.width = `${label.w}px`;
|
|
2494
|
+
box.style.height = `${label.h}px`;
|
|
2495
|
+
box.style.border = "2px solid #ffb020";
|
|
2496
|
+
box.style.boxSizing = "border-box";
|
|
2497
|
+
const tag = document.createElement("div");
|
|
2498
|
+
tag.setAttribute("data-fengming-labels", "1");
|
|
2499
|
+
tag.textContent = label.ref;
|
|
2500
|
+
tag.style.position = "absolute";
|
|
2501
|
+
tag.style.left = `${label.x}px`;
|
|
2502
|
+
tag.style.top = `${clamp(label.y - 18, 0, 2e4)}px`;
|
|
2503
|
+
tag.style.background = "#ffb020";
|
|
2504
|
+
tag.style.color = "#1a1a1a";
|
|
2505
|
+
tag.style.fontSize = "12px";
|
|
2506
|
+
tag.style.lineHeight = "14px";
|
|
2507
|
+
tag.style.padding = "1px 4px";
|
|
2508
|
+
tag.style.borderRadius = "3px";
|
|
2509
|
+
tag.style.boxShadow = "0 1px 2px rgba(0,0,0,0.35)";
|
|
2510
|
+
tag.style.whiteSpace = "nowrap";
|
|
2511
|
+
root.appendChild(box);
|
|
2512
|
+
root.appendChild(tag);
|
|
2513
|
+
}
|
|
2514
|
+
document.documentElement.appendChild(root);
|
|
2515
|
+
}, boxes);
|
|
2516
|
+
return {
|
|
2517
|
+
buffer: await page.screenshot({
|
|
2518
|
+
type,
|
|
2519
|
+
timeout: opts.timeoutMs
|
|
2520
|
+
}),
|
|
2521
|
+
labels: boxes.length,
|
|
2522
|
+
skipped
|
|
2523
|
+
};
|
|
2524
|
+
} finally {
|
|
2525
|
+
await page.evaluate(() => {
|
|
2526
|
+
document.querySelectorAll("[data-fengming-labels]").forEach((el) => el.remove());
|
|
2527
|
+
}).catch(() => {});
|
|
2528
|
+
}
|
|
2529
|
+
}
|
|
2530
|
+
async function setInputFilesViaPlaywright(opts) {
|
|
2531
|
+
const page = await getPageForTargetId(opts);
|
|
2532
|
+
ensurePageState(page);
|
|
2533
|
+
restoreRoleRefsForTarget({
|
|
2534
|
+
cdpUrl: opts.cdpUrl,
|
|
2535
|
+
targetId: opts.targetId,
|
|
2536
|
+
page
|
|
2537
|
+
});
|
|
2538
|
+
if (!opts.paths.length) throw new Error("路径是必需的");
|
|
2539
|
+
const inputRef = normalizeOptionalString(opts.inputRef) ?? "";
|
|
2540
|
+
const element = normalizeOptionalString(opts.element) ?? "";
|
|
2541
|
+
if (inputRef && element) throw new Error("inputRef 和 element 互斥");
|
|
2542
|
+
if (!inputRef && !element) throw new Error("需要 inputRef 或 element");
|
|
2543
|
+
const locator = inputRef ? refLocator(page, inputRef) : page.locator(element).first();
|
|
2544
|
+
const uploadPathsResult = await resolveStrictExistingPathsWithinRoot({
|
|
2545
|
+
rootDir: DEFAULT_UPLOAD_DIR,
|
|
2546
|
+
requestedPaths: opts.paths,
|
|
2547
|
+
scopeLabel: `uploads directory (${DEFAULT_UPLOAD_DIR})`
|
|
2548
|
+
});
|
|
2549
|
+
if (!uploadPathsResult.ok) throw new Error(uploadPathsResult.error);
|
|
2550
|
+
const resolvedPaths = uploadPathsResult.paths;
|
|
2551
|
+
try {
|
|
2552
|
+
await locator.setInputFiles(resolvedPaths);
|
|
2553
|
+
} catch (err) {
|
|
2554
|
+
throw toFriendlyInteractionError(err, inputRef || element);
|
|
2555
|
+
}
|
|
2556
|
+
try {
|
|
2557
|
+
const handle = await locator.elementHandle();
|
|
2558
|
+
if (handle) await handle.evaluate((el) => {
|
|
2559
|
+
el.dispatchEvent(new Event("input", { bubbles: true }));
|
|
2560
|
+
el.dispatchEvent(new Event("change", { bubbles: true }));
|
|
2561
|
+
});
|
|
2562
|
+
} catch {}
|
|
2563
|
+
}
|
|
2564
|
+
async function executeSingleAction(action, cdpUrl, targetId, evaluateEnabled, ssrfPolicy, depth = 0, signal) {
|
|
2565
|
+
if (depth > 5) throw new Error(`Batch nesting depth exceeds maximum of 5`);
|
|
2566
|
+
const effectiveTargetId = action.targetId ?? targetId;
|
|
2567
|
+
switch (action.kind) {
|
|
2568
|
+
case "click":
|
|
2569
|
+
await clickViaPlaywright({
|
|
2570
|
+
cdpUrl,
|
|
2571
|
+
targetId: effectiveTargetId,
|
|
2572
|
+
ref: action.ref,
|
|
2573
|
+
selector: action.selector,
|
|
2574
|
+
doubleClick: action.doubleClick,
|
|
2575
|
+
button: action.button,
|
|
2576
|
+
modifiers: action.modifiers,
|
|
2577
|
+
delayMs: action.delayMs,
|
|
2578
|
+
timeoutMs: action.timeoutMs,
|
|
2579
|
+
ssrfPolicy,
|
|
2580
|
+
signal
|
|
2581
|
+
});
|
|
2582
|
+
break;
|
|
2583
|
+
case "clickCoords":
|
|
2584
|
+
await clickCoordsViaPlaywright({
|
|
2585
|
+
cdpUrl,
|
|
2586
|
+
targetId: effectiveTargetId,
|
|
2587
|
+
x: action.x,
|
|
2588
|
+
y: action.y,
|
|
2589
|
+
doubleClick: action.doubleClick,
|
|
2590
|
+
button: action.button,
|
|
2591
|
+
delayMs: action.delayMs,
|
|
2592
|
+
timeoutMs: action.timeoutMs,
|
|
2593
|
+
ssrfPolicy,
|
|
2594
|
+
signal
|
|
2595
|
+
});
|
|
2596
|
+
break;
|
|
2597
|
+
case "type":
|
|
2598
|
+
await typeViaPlaywright({
|
|
2599
|
+
cdpUrl,
|
|
2600
|
+
targetId: effectiveTargetId,
|
|
2601
|
+
ref: action.ref,
|
|
2602
|
+
selector: action.selector,
|
|
2603
|
+
text: action.text,
|
|
2604
|
+
submit: action.submit,
|
|
2605
|
+
slowly: action.slowly,
|
|
2606
|
+
timeoutMs: action.timeoutMs,
|
|
2607
|
+
ssrfPolicy,
|
|
2608
|
+
signal
|
|
2609
|
+
});
|
|
2610
|
+
break;
|
|
2611
|
+
case "press":
|
|
2612
|
+
await pressKeyViaPlaywright({
|
|
2613
|
+
cdpUrl,
|
|
2614
|
+
targetId: effectiveTargetId,
|
|
2615
|
+
key: action.key,
|
|
2616
|
+
delayMs: action.delayMs,
|
|
2617
|
+
ssrfPolicy,
|
|
2618
|
+
signal
|
|
2619
|
+
});
|
|
2620
|
+
break;
|
|
2621
|
+
case "hover":
|
|
2622
|
+
await hoverViaPlaywright({
|
|
2623
|
+
cdpUrl,
|
|
2624
|
+
targetId: effectiveTargetId,
|
|
2625
|
+
ref: action.ref,
|
|
2626
|
+
selector: action.selector,
|
|
2627
|
+
timeoutMs: action.timeoutMs,
|
|
2628
|
+
signal
|
|
2629
|
+
});
|
|
2630
|
+
break;
|
|
2631
|
+
case "scrollIntoView":
|
|
2632
|
+
await scrollIntoViewViaPlaywright({
|
|
2633
|
+
cdpUrl,
|
|
2634
|
+
targetId: effectiveTargetId,
|
|
2635
|
+
ref: action.ref,
|
|
2636
|
+
selector: action.selector,
|
|
2637
|
+
timeoutMs: action.timeoutMs,
|
|
2638
|
+
signal
|
|
2639
|
+
});
|
|
2640
|
+
break;
|
|
2641
|
+
case "drag":
|
|
2642
|
+
await dragViaPlaywright({
|
|
2643
|
+
cdpUrl,
|
|
2644
|
+
targetId: effectiveTargetId,
|
|
2645
|
+
startRef: action.startRef,
|
|
2646
|
+
startSelector: action.startSelector,
|
|
2647
|
+
endRef: action.endRef,
|
|
2648
|
+
endSelector: action.endSelector,
|
|
2649
|
+
timeoutMs: action.timeoutMs,
|
|
2650
|
+
signal
|
|
2651
|
+
});
|
|
2652
|
+
break;
|
|
2653
|
+
case "select":
|
|
2654
|
+
await selectOptionViaPlaywright({
|
|
2655
|
+
cdpUrl,
|
|
2656
|
+
targetId: effectiveTargetId,
|
|
2657
|
+
ref: action.ref,
|
|
2658
|
+
selector: action.selector,
|
|
2659
|
+
values: action.values,
|
|
2660
|
+
timeoutMs: action.timeoutMs,
|
|
2661
|
+
ssrfPolicy,
|
|
2662
|
+
signal
|
|
2663
|
+
});
|
|
2664
|
+
break;
|
|
2665
|
+
case "fill":
|
|
2666
|
+
await fillFormViaPlaywright({
|
|
2667
|
+
cdpUrl,
|
|
2668
|
+
targetId: effectiveTargetId,
|
|
2669
|
+
fields: action.fields,
|
|
2670
|
+
timeoutMs: action.timeoutMs,
|
|
2671
|
+
ssrfPolicy,
|
|
2672
|
+
signal
|
|
2673
|
+
});
|
|
2674
|
+
break;
|
|
2675
|
+
case "resize":
|
|
2676
|
+
await resizeViewportViaPlaywright({
|
|
2677
|
+
cdpUrl,
|
|
2678
|
+
targetId: effectiveTargetId,
|
|
2679
|
+
width: action.width,
|
|
2680
|
+
height: action.height
|
|
2681
|
+
});
|
|
2682
|
+
break;
|
|
2683
|
+
case "wait":
|
|
2684
|
+
if (action.fn && !evaluateEnabled) throw new Error("wait --fn 已被配置禁用");
|
|
2685
|
+
await waitForViaPlaywright({
|
|
2686
|
+
cdpUrl,
|
|
2687
|
+
targetId: effectiveTargetId,
|
|
2688
|
+
timeMs: action.timeMs,
|
|
2689
|
+
text: action.text,
|
|
2690
|
+
textGone: action.textGone,
|
|
2691
|
+
selector: action.selector,
|
|
2692
|
+
url: action.url,
|
|
2693
|
+
loadState: action.loadState,
|
|
2694
|
+
fn: action.fn,
|
|
2695
|
+
timeoutMs: action.timeoutMs,
|
|
2696
|
+
signal
|
|
2697
|
+
});
|
|
2698
|
+
break;
|
|
2699
|
+
case "evaluate":
|
|
2700
|
+
if (!evaluateEnabled) throw new Error("act:evaluate is disabled by config (browser.evaluateEnabled=false)");
|
|
2701
|
+
return await evaluateViaPlaywright({
|
|
2702
|
+
cdpUrl,
|
|
2703
|
+
targetId: effectiveTargetId,
|
|
2704
|
+
ssrfPolicy,
|
|
2705
|
+
fn: action.fn,
|
|
2706
|
+
ref: action.ref,
|
|
2707
|
+
timeoutMs: action.timeoutMs,
|
|
2708
|
+
signal
|
|
2709
|
+
});
|
|
2710
|
+
case "close":
|
|
2711
|
+
await closePageViaPlaywright({
|
|
2712
|
+
cdpUrl,
|
|
2713
|
+
targetId: effectiveTargetId
|
|
2714
|
+
});
|
|
2715
|
+
break;
|
|
2716
|
+
case "batch":
|
|
2717
|
+
await batchViaPlaywright({
|
|
2718
|
+
cdpUrl,
|
|
2719
|
+
targetId: effectiveTargetId,
|
|
2720
|
+
ssrfPolicy,
|
|
2721
|
+
actions: action.actions,
|
|
2722
|
+
stopOnError: action.stopOnError,
|
|
2723
|
+
evaluateEnabled,
|
|
2724
|
+
depth: depth + 1,
|
|
2725
|
+
signal
|
|
2726
|
+
});
|
|
2727
|
+
break;
|
|
2728
|
+
default: throw new Error(`Unsupported batch action kind: ${action.kind}`);
|
|
2729
|
+
}
|
|
2730
|
+
}
|
|
2731
|
+
async function executeActViaPlaywright(opts) {
|
|
2732
|
+
const dialogAbort = createObservedDialogAbortSignalForPage({
|
|
2733
|
+
page: await getPageForTargetId({
|
|
2734
|
+
cdpUrl: opts.cdpUrl,
|
|
2735
|
+
targetId: opts.targetId,
|
|
2736
|
+
ssrfPolicy: opts.ssrfPolicy
|
|
2737
|
+
}),
|
|
2738
|
+
parentSignal: opts.signal
|
|
2739
|
+
});
|
|
2740
|
+
try {
|
|
2741
|
+
if (opts.action.kind === "batch") return { results: (await batchViaPlaywright({
|
|
2742
|
+
cdpUrl: opts.cdpUrl,
|
|
2743
|
+
targetId: opts.targetId,
|
|
2744
|
+
ssrfPolicy: opts.ssrfPolicy,
|
|
2745
|
+
actions: opts.action.actions,
|
|
2746
|
+
stopOnError: opts.action.stopOnError,
|
|
2747
|
+
evaluateEnabled: opts.evaluateEnabled,
|
|
2748
|
+
signal: dialogAbort.signal
|
|
2749
|
+
})).results };
|
|
2750
|
+
const result = await executeSingleAction(opts.action, opts.cdpUrl, opts.targetId, opts.evaluateEnabled, opts.ssrfPolicy, 0, dialogAbort.signal);
|
|
2751
|
+
if (opts.action.kind === "evaluate") return { result };
|
|
2752
|
+
return {};
|
|
2753
|
+
} catch (err) {
|
|
2754
|
+
if (isBrowserObservedDialogBlockedError(err)) return {
|
|
2755
|
+
blockedByDialog: true,
|
|
2756
|
+
browserState: err.browserState
|
|
2757
|
+
};
|
|
2758
|
+
throw err;
|
|
2759
|
+
} finally {
|
|
2760
|
+
dialogAbort.cleanup();
|
|
2761
|
+
}
|
|
2762
|
+
}
|
|
2763
|
+
async function batchViaPlaywright(opts) {
|
|
2764
|
+
const depth = opts.depth ?? 0;
|
|
2765
|
+
if (depth > 5) throw new Error(`Batch nesting depth exceeds maximum of 5`);
|
|
2766
|
+
if (opts.actions.length > 100) throw new Error(`Batch exceeds maximum of 100 actions`);
|
|
2767
|
+
const results = [];
|
|
2768
|
+
for (const action of opts.actions) {
|
|
2769
|
+
if (opts.signal?.aborted) throw opts.signal.reason ?? /* @__PURE__ */ new Error("已中止");
|
|
2770
|
+
try {
|
|
2771
|
+
await executeSingleAction(action, opts.cdpUrl, opts.targetId, opts.evaluateEnabled, opts.ssrfPolicy, depth, opts.signal);
|
|
2772
|
+
results.push({ ok: true });
|
|
2773
|
+
} catch (err) {
|
|
2774
|
+
if (isBrowserObservedDialogBlockedError(err)) throw err;
|
|
2775
|
+
const message = formatErrorMessage(err);
|
|
2776
|
+
results.push({
|
|
2777
|
+
ok: false,
|
|
2778
|
+
error: message
|
|
2779
|
+
});
|
|
2780
|
+
if (opts.stopOnError !== false) break;
|
|
2781
|
+
}
|
|
2782
|
+
}
|
|
2783
|
+
return { results };
|
|
2784
|
+
}
|
|
2785
|
+
//#endregion
|
|
2786
|
+
//#region extensions/browser/src/browser/pw-tools-core.responses.ts
|
|
2787
|
+
async function responseBodyViaPlaywright(opts) {
|
|
2788
|
+
const pattern = normalizeOptionalString(opts.url) ?? "";
|
|
2789
|
+
if (!pattern) throw new Error("URL 是必需的");
|
|
2790
|
+
const maxChars = typeof opts.maxChars === "number" && Number.isFinite(opts.maxChars) ? Math.max(1, Math.min(5e6, Math.floor(opts.maxChars))) : 2e5;
|
|
2791
|
+
const timeout = normalizeTimeoutMs(opts.timeoutMs, 2e4);
|
|
2792
|
+
const page = await getPageForTargetId(opts);
|
|
2793
|
+
ensurePageState(page);
|
|
2794
|
+
const resp = await new Promise((resolve, reject) => {
|
|
2795
|
+
let done = false;
|
|
2796
|
+
let timer;
|
|
2797
|
+
let handler;
|
|
2798
|
+
const cleanup = () => {
|
|
2799
|
+
if (timer) clearTimeout(timer);
|
|
2800
|
+
timer = void 0;
|
|
2801
|
+
if (handler) page.off("response", handler);
|
|
2802
|
+
};
|
|
2803
|
+
handler = (resp) => {
|
|
2804
|
+
if (done) return;
|
|
2805
|
+
if (!matchBrowserUrlPattern(pattern, resp.url?.() || "")) return;
|
|
2806
|
+
done = true;
|
|
2807
|
+
cleanup();
|
|
2808
|
+
resolve(resp);
|
|
2809
|
+
};
|
|
2810
|
+
page.on("response", handler);
|
|
2811
|
+
timer = setTimeout(() => {
|
|
2812
|
+
if (done) return;
|
|
2813
|
+
done = true;
|
|
2814
|
+
cleanup();
|
|
2815
|
+
reject(/* @__PURE__ */ new Error(`Response not found for url pattern "${pattern}". Run 'fengming browser requests' to inspect recent network activity.`));
|
|
2816
|
+
}, timeout);
|
|
2817
|
+
});
|
|
2818
|
+
const url = resp.url?.() || "";
|
|
2819
|
+
const status = resp.status?.();
|
|
2820
|
+
const headers = resp.headers?.();
|
|
2821
|
+
let bodyText = "";
|
|
2822
|
+
try {
|
|
2823
|
+
if (typeof resp.text === "function") bodyText = await resp.text();
|
|
2824
|
+
else if (typeof resp.body === "function") {
|
|
2825
|
+
const buf = await resp.body();
|
|
2826
|
+
bodyText = new TextDecoder("utf-8").decode(buf);
|
|
2827
|
+
}
|
|
2828
|
+
} catch (err) {
|
|
2829
|
+
throw new Error(`Failed to read response body for "${url}": ${String(err)}`, { cause: err });
|
|
2830
|
+
}
|
|
2831
|
+
return {
|
|
2832
|
+
url,
|
|
2833
|
+
status,
|
|
2834
|
+
headers,
|
|
2835
|
+
body: bodyText.length > maxChars ? bodyText.slice(0, maxChars) : bodyText,
|
|
2836
|
+
truncated: bodyText.length > maxChars ? true : void 0
|
|
2837
|
+
};
|
|
2838
|
+
}
|
|
2839
|
+
//#endregion
|
|
2840
|
+
//#region extensions/browser/src/browser/pw-tools-core.state.ts
|
|
2841
|
+
const { devices: playwrightDevices } = playwrightCore;
|
|
2842
|
+
async function setOfflineViaPlaywright(opts) {
|
|
2843
|
+
const page = await getPageForTargetId(opts);
|
|
2844
|
+
ensurePageState(page);
|
|
2845
|
+
await page.context().setOffline(opts.offline);
|
|
2846
|
+
}
|
|
2847
|
+
async function setExtraHTTPHeadersViaPlaywright(opts) {
|
|
2848
|
+
const page = await getPageForTargetId(opts);
|
|
2849
|
+
ensurePageState(page);
|
|
2850
|
+
await page.context().setExtraHTTPHeaders(opts.headers);
|
|
2851
|
+
}
|
|
2852
|
+
async function setHttpCredentialsViaPlaywright(opts) {
|
|
2853
|
+
const page = await getPageForTargetId(opts);
|
|
2854
|
+
ensurePageState(page);
|
|
2855
|
+
if (opts.clear) {
|
|
2856
|
+
await page.context().setHTTPCredentials(null);
|
|
2857
|
+
return;
|
|
2858
|
+
}
|
|
2859
|
+
const username = opts.username ?? "";
|
|
2860
|
+
const password = opts.password ?? "";
|
|
2861
|
+
if (!username) throw new Error("用户名是必需的(或设置 clear=true)");
|
|
2862
|
+
await page.context().setHTTPCredentials({
|
|
2863
|
+
username,
|
|
2864
|
+
password
|
|
2865
|
+
});
|
|
2866
|
+
}
|
|
2867
|
+
async function setGeolocationViaPlaywright(opts) {
|
|
2868
|
+
const page = await getPageForTargetId(opts);
|
|
2869
|
+
ensurePageState(page);
|
|
2870
|
+
const context = page.context();
|
|
2871
|
+
if (opts.clear) {
|
|
2872
|
+
await context.setGeolocation(null);
|
|
2873
|
+
await context.clearPermissions().catch(() => {});
|
|
2874
|
+
return;
|
|
2875
|
+
}
|
|
2876
|
+
if (typeof opts.latitude !== "number" || typeof opts.longitude !== "number") throw new Error("latitude and longitude are required (or set clear=true)");
|
|
2877
|
+
await context.setGeolocation({
|
|
2878
|
+
latitude: opts.latitude,
|
|
2879
|
+
longitude: opts.longitude,
|
|
2880
|
+
accuracy: typeof opts.accuracy === "number" ? opts.accuracy : void 0
|
|
2881
|
+
});
|
|
2882
|
+
const origin = normalizeOptionalString(opts.origin) || (() => {
|
|
2883
|
+
try {
|
|
2884
|
+
return new URL(page.url()).origin;
|
|
2885
|
+
} catch {
|
|
2886
|
+
return "";
|
|
2887
|
+
}
|
|
2888
|
+
})();
|
|
2889
|
+
if (origin) await context.grantPermissions(["geolocation"], { origin }).catch(() => {});
|
|
2890
|
+
}
|
|
2891
|
+
async function emulateMediaViaPlaywright(opts) {
|
|
2892
|
+
const page = await getPageForTargetId(opts);
|
|
2893
|
+
ensurePageState(page);
|
|
2894
|
+
await page.emulateMedia({ colorScheme: opts.colorScheme });
|
|
2895
|
+
}
|
|
2896
|
+
async function setLocaleViaPlaywright(opts) {
|
|
2897
|
+
const page = await getPageForTargetId(opts);
|
|
2898
|
+
ensurePageState(page);
|
|
2899
|
+
const locale = normalizeOptionalString(opts.locale) ?? "";
|
|
2900
|
+
if (!locale) throw new Error("locale 是必需的");
|
|
2901
|
+
await withPageScopedCdpClient({
|
|
2902
|
+
cdpUrl: opts.cdpUrl,
|
|
2903
|
+
page,
|
|
2904
|
+
targetId: opts.targetId,
|
|
2905
|
+
fn: async (send) => {
|
|
2906
|
+
try {
|
|
2907
|
+
await send("Emulation.setLocaleOverride", { locale });
|
|
2908
|
+
} catch (err) {
|
|
2909
|
+
if (String(err).includes("Another locale override is already in effect")) return;
|
|
2910
|
+
throw err;
|
|
2911
|
+
}
|
|
2912
|
+
}
|
|
2913
|
+
});
|
|
2914
|
+
}
|
|
2915
|
+
async function setTimezoneViaPlaywright(opts) {
|
|
2916
|
+
const page = await getPageForTargetId(opts);
|
|
2917
|
+
ensurePageState(page);
|
|
2918
|
+
const timezoneId = normalizeOptionalString(opts.timezoneId) ?? "";
|
|
2919
|
+
if (!timezoneId) throw new Error("timezoneId 是必需的");
|
|
2920
|
+
await withPageScopedCdpClient({
|
|
2921
|
+
cdpUrl: opts.cdpUrl,
|
|
2922
|
+
page,
|
|
2923
|
+
targetId: opts.targetId,
|
|
2924
|
+
fn: async (send) => {
|
|
2925
|
+
try {
|
|
2926
|
+
await send("Emulation.setTimezoneOverride", { timezoneId });
|
|
2927
|
+
} catch (err) {
|
|
2928
|
+
const msg = String(err);
|
|
2929
|
+
if (msg.includes("Timezone override is already in effect")) return;
|
|
2930
|
+
if (msg.includes("Invalid timezone")) throw new Error(`Invalid timezone ID: ${timezoneId}`, { cause: err });
|
|
2931
|
+
throw err;
|
|
2932
|
+
}
|
|
2933
|
+
}
|
|
2934
|
+
});
|
|
2935
|
+
}
|
|
2936
|
+
async function setDeviceViaPlaywright(opts) {
|
|
2937
|
+
const page = await getPageForTargetId(opts);
|
|
2938
|
+
ensurePageState(page);
|
|
2939
|
+
const name = normalizeOptionalString(opts.name) ?? "";
|
|
2940
|
+
if (!name) throw new Error("设备名称是必需的");
|
|
2941
|
+
const descriptor = playwrightDevices[name];
|
|
2942
|
+
if (!descriptor) throw new Error(`Unknown device "${name}".`);
|
|
2943
|
+
if (descriptor.viewport) await page.setViewportSize({
|
|
2944
|
+
width: descriptor.viewport.width,
|
|
2945
|
+
height: descriptor.viewport.height
|
|
2946
|
+
});
|
|
2947
|
+
await withPageScopedCdpClient({
|
|
2948
|
+
cdpUrl: opts.cdpUrl,
|
|
2949
|
+
page,
|
|
2950
|
+
targetId: opts.targetId,
|
|
2951
|
+
fn: async (send) => {
|
|
2952
|
+
if (descriptor.userAgent || descriptor.locale) await send("Emulation.setUserAgentOverride", {
|
|
2953
|
+
userAgent: descriptor.userAgent ?? "",
|
|
2954
|
+
acceptLanguage: descriptor.locale ?? void 0
|
|
2955
|
+
});
|
|
2956
|
+
if (descriptor.viewport) await send("Emulation.setDeviceMetricsOverride", {
|
|
2957
|
+
mobile: Boolean(descriptor.isMobile),
|
|
2958
|
+
width: descriptor.viewport.width,
|
|
2959
|
+
height: descriptor.viewport.height,
|
|
2960
|
+
deviceScaleFactor: descriptor.deviceScaleFactor ?? 1,
|
|
2961
|
+
screenWidth: descriptor.viewport.width,
|
|
2962
|
+
screenHeight: descriptor.viewport.height
|
|
2963
|
+
});
|
|
2964
|
+
if (descriptor.hasTouch) await send("Emulation.setTouchEmulationEnabled", { enabled: true });
|
|
2965
|
+
}
|
|
2966
|
+
});
|
|
2967
|
+
}
|
|
2968
|
+
//#endregion
|
|
2969
|
+
//#region extensions/browser/src/browser/pw-tools-core.storage.ts
|
|
2970
|
+
async function cookiesGetViaPlaywright(opts) {
|
|
2971
|
+
const page = await getPageForTargetId(opts);
|
|
2972
|
+
ensurePageState(page);
|
|
2973
|
+
return { cookies: await page.context().cookies() };
|
|
2974
|
+
}
|
|
2975
|
+
async function cookiesSetViaPlaywright(opts) {
|
|
2976
|
+
const page = await getPageForTargetId(opts);
|
|
2977
|
+
ensurePageState(page);
|
|
2978
|
+
const cookie = opts.cookie;
|
|
2979
|
+
if (!cookie.name || cookie.value === void 0) throw new Error("cookie 的 name 和 value 是必需的");
|
|
2980
|
+
const hasUrl = typeof cookie.url === "string" && cookie.url.trim();
|
|
2981
|
+
const hasDomainPath = typeof cookie.domain === "string" && cookie.domain.trim() && typeof cookie.path === "string" && cookie.path.trim();
|
|
2982
|
+
if (!hasUrl && !hasDomainPath) throw new Error("cookie 需要 url 或 domain+path");
|
|
2983
|
+
await page.context().addCookies([cookie]);
|
|
2984
|
+
}
|
|
2985
|
+
async function cookiesClearViaPlaywright(opts) {
|
|
2986
|
+
const page = await getPageForTargetId(opts);
|
|
2987
|
+
ensurePageState(page);
|
|
2988
|
+
await page.context().clearCookies();
|
|
2989
|
+
}
|
|
2990
|
+
async function storageGetViaPlaywright(opts) {
|
|
2991
|
+
const page = await getPageForTargetId(opts);
|
|
2992
|
+
ensurePageState(page);
|
|
2993
|
+
const kind = opts.kind;
|
|
2994
|
+
const key = readStringValue(opts.key);
|
|
2995
|
+
return { values: await page.evaluate(({ kind: kind2, key: key2 }) => {
|
|
2996
|
+
const store = kind2 === "session" ? window.sessionStorage : window.localStorage;
|
|
2997
|
+
if (key2) {
|
|
2998
|
+
const value = store.getItem(key2);
|
|
2999
|
+
return value === null ? {} : { [key2]: value };
|
|
3000
|
+
}
|
|
3001
|
+
const out = {};
|
|
3002
|
+
for (let i = 0; i < store.length; i += 1) {
|
|
3003
|
+
const k = store.key(i);
|
|
3004
|
+
if (!k) continue;
|
|
3005
|
+
const v = store.getItem(k);
|
|
3006
|
+
if (v !== null) out[k] = v;
|
|
3007
|
+
}
|
|
3008
|
+
return out;
|
|
3009
|
+
}, {
|
|
3010
|
+
kind,
|
|
3011
|
+
key
|
|
3012
|
+
}) ?? {} };
|
|
3013
|
+
}
|
|
3014
|
+
async function storageSetViaPlaywright(opts) {
|
|
3015
|
+
const page = await getPageForTargetId(opts);
|
|
3016
|
+
ensurePageState(page);
|
|
3017
|
+
const key = opts.key;
|
|
3018
|
+
if (!key) throw new Error("key 是必需的");
|
|
3019
|
+
await page.evaluate(({ kind, key: k, value }) => {
|
|
3020
|
+
(kind === "session" ? window.sessionStorage : window.localStorage).setItem(k, value);
|
|
3021
|
+
}, {
|
|
3022
|
+
kind: opts.kind,
|
|
3023
|
+
key,
|
|
3024
|
+
value: opts.value
|
|
3025
|
+
});
|
|
3026
|
+
}
|
|
3027
|
+
async function storageClearViaPlaywright(opts) {
|
|
3028
|
+
const page = await getPageForTargetId(opts);
|
|
3029
|
+
ensurePageState(page);
|
|
3030
|
+
await page.evaluate(({ kind }) => {
|
|
3031
|
+
(kind === "session" ? window.sessionStorage : window.localStorage).clear();
|
|
3032
|
+
}, { kind: opts.kind });
|
|
3033
|
+
}
|
|
3034
|
+
//#endregion
|
|
3035
|
+
//#region extensions/browser/src/browser/pw-tools-core.trace.ts
|
|
3036
|
+
async function traceStartViaPlaywright(opts) {
|
|
3037
|
+
const context = (await getPageForTargetId(opts)).context();
|
|
3038
|
+
const ctxState = ensureContextState(context);
|
|
3039
|
+
if (ctxState.traceActive) throw new Error("Trace 已在运行中,请先停止当前 trace 再启动新的");
|
|
3040
|
+
await context.tracing.start({
|
|
3041
|
+
screenshots: opts.screenshots ?? true,
|
|
3042
|
+
snapshots: opts.snapshots ?? true,
|
|
3043
|
+
sources: opts.sources ?? false
|
|
3044
|
+
});
|
|
3045
|
+
ctxState.traceActive = true;
|
|
3046
|
+
}
|
|
3047
|
+
async function traceStopViaPlaywright(opts) {
|
|
3048
|
+
const context = (await getPageForTargetId(opts)).context();
|
|
3049
|
+
const ctxState = ensureContextState(context);
|
|
3050
|
+
if (!ctxState.traceActive) throw new Error("没有活跃的 trace,请先启动 trace 再停止");
|
|
3051
|
+
await writeViaSiblingTempPath({
|
|
3052
|
+
rootDir: DEFAULT_TRACE_DIR,
|
|
3053
|
+
targetPath: opts.path,
|
|
3054
|
+
writeTemp: async (tempPath) => {
|
|
3055
|
+
await context.tracing.stop({ path: tempPath });
|
|
3056
|
+
}
|
|
3057
|
+
});
|
|
3058
|
+
ctxState.traceActive = false;
|
|
3059
|
+
}
|
|
3060
|
+
//#endregion
|
|
3061
|
+
//#region extensions/browser/src/browser/pw-ai.ts
|
|
3062
|
+
markPwAiLoaded();
|
|
3063
|
+
//#endregion
|
|
3064
|
+
export { armDialogViaPlaywright, armFileUploadViaPlaywright, batchViaPlaywright, clickViaPlaywright, closePageByTargetIdViaPlaywright, closePageViaPlaywright, closePlaywrightBrowserConnection, cookiesClearViaPlaywright, cookiesGetViaPlaywright, cookiesSetViaPlaywright, createObservedDialogAbortSignalForPage, createPageViaPlaywright, downloadViaPlaywright, dragViaPlaywright, emulateMediaViaPlaywright, ensurePageState, evaluateViaPlaywright, executeActViaPlaywright, fillFormViaPlaywright, focusPageByTargetIdViaPlaywright, forceDisconnectPlaywrightForTarget, getConsoleMessagesViaPlaywright, getNetworkRequestsViaPlaywright, getObservedBrowserStateForPage, getObservedBrowserStateViaPlaywright, getPageErrorsViaPlaywright, getPageForTargetId, highlightViaPlaywright, hoverViaPlaywright, isBrowserObservedDialogBlockedError, listPagesViaPlaywright, markObservedDialogsHandledRemotelyForPage, navigateViaPlaywright, pdfViaPlaywright, pressKeyViaPlaywright, refLocator, resizeViewportViaPlaywright, respondToObservedDialogOnPage, respondToObservedDialogViaPlaywright, responseBodyViaPlaywright, screenshotWithLabelsViaPlaywright, scrollIntoViewViaPlaywright, selectOptionViaPlaywright, setDeviceViaPlaywright, setExtraHTTPHeadersViaPlaywright, setGeolocationViaPlaywright, setHttpCredentialsViaPlaywright, setInputFilesViaPlaywright, setLocaleViaPlaywright, setOfflineViaPlaywright, setTimezoneViaPlaywright, snapshotAiViaPlaywright, snapshotAriaViaPlaywright, snapshotRoleViaPlaywright, storageClearViaPlaywright, storageGetViaPlaywright, storageSetViaPlaywright, storeAriaSnapshotRefsViaPlaywright, takeScreenshotViaPlaywright, traceStartViaPlaywright, traceStopViaPlaywright, typeViaPlaywright, waitForDownloadViaPlaywright, waitForViaPlaywright };
|