@openhands/agent-canvas 1.0.0-alpha.6 → 1.0.0-alpha.8
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/README.md +32 -7
- package/bin/agent-canvas.mjs +35 -2
- package/build/assets/{QueryClientProvider-DITRCGAK.js → QueryClientProvider-B7kl84Kj.js} +1 -1
- package/build/assets/{Trans-D43bd3yR.js → Trans-1j65oy9O.js} +1 -1
- package/build/assets/{acp-providers-SCPK1BIU.js → acp-providers-DauuOsW9.js} +1 -1
- package/build/assets/{acp-route-guard-IWlFmS6x.js → acp-route-guard-CQTmeJwM.js} +1 -1
- package/build/assets/{active-backend-context-CkP3ZEJs.js → active-backend-context-TVbjnvmP.js} +1 -1
- package/build/assets/add-backend-modal-KMmPQNZU.js +1 -0
- package/build/assets/{agent-server-client-options-8OJSXbm8.js → agent-server-client-options-DT2GP6VJ.js} +1 -1
- package/build/assets/{agent-server-compatibility-DvKtnXHw.js → agent-server-compatibility-2aOx5iWd.js} +1 -1
- package/build/assets/{agent-server-conversation-service.api-BdEre_71.js → agent-server-conversation-service.api-DSl9G5UR.js} +3 -3
- package/build/assets/{agent-settings-DdisD2Xx.js → agent-settings-B247S9G3.js} +2 -2
- package/build/assets/{alert-banner-CvTYN73l.js → alert-banner-BWoqueRw.js} +1 -1
- package/build/assets/{analytics-consent-form-modal-BKgT9i2w.js → analytics-consent-form-modal-C7sXfxRh.js} +1 -1
- package/build/assets/{app-settings-DcYXtxGP.js → app-settings-BVeSaty9.js} +1 -1
- package/build/assets/automation-detail-g5-RZ0da.js +1 -0
- package/build/assets/automations-list-DHoq_0MM.js +1 -0
- package/build/assets/{backend-form-modal-KudhWUX8.js → backend-form-modal-K6IMCr3p.js} +1 -1
- package/build/assets/{backend-synced-settings-badge-BFy2HylT.js → backend-synced-settings-badge-nAfiUWvM.js} +1 -1
- package/build/assets/{base-modal-B4HvlFHE.js → base-modal-CQRvRHu1.js} +1 -1
- package/build/assets/{brand-button-8fVVei4i.js → brand-button-C2nEKopC.js} +1 -1
- package/build/assets/{browser-vYpdU3CR.js → browser-DKG63inJ.js} +1 -1
- package/build/assets/{browser-tab-DTM6RyoV.js → browser-tab-B_BuTvrO.js} +1 -1
- package/build/assets/{checkmark-BcvXE9bf.js → checkmark-BJJrZUF8.js} +1 -1
- package/build/assets/{chevron-left-small-BqSkXTeq.js → chevron-left-small-CSh-sE9L.js} +1 -1
- package/build/assets/{circle-plus-check-toggle-DRvuu-RD.js → circle-plus-check-toggle-qs8Va1cC.js} +1 -1
- package/build/assets/{clock-DfoVUZVq.js → clock-ZR4Kn-_Y.js} +1 -1
- package/build/assets/{close-SnIy2eLD.js → close-BdmyeRqS.js} +1 -1
- package/build/assets/{combobox-caret-BMsz5mQX.js → combobox-caret-B53O9Hsq.js} +1 -1
- package/build/assets/{condenser-settings-DduLQcpV.js → condenser-settings-A35V3yng.js} +1 -1
- package/build/assets/{confirmation-modal-B-DOYMUH.js → confirmation-modal-C9-La0h3.js} +1 -1
- package/build/assets/{context-menu-list-item-DzjPB8aC.js → context-menu-list-item-Buu9nc0q.js} +1 -1
- package/build/assets/conversation-BD5WemJI.js +19 -0
- package/build/assets/conversation-C47K62n8.js +1 -0
- package/build/assets/conversation-panel-Dn-S56Gk.js +1 -0
- package/build/assets/{conversation-service.api-YTGTw0pz.js → conversation-service.api-C8pYCyV6.js} +1 -1
- package/build/assets/{conversation-tab-empty-state-BtFDbyTe.js → conversation-tab-empty-state-D8dNvo-V.js} +1 -1
- package/build/assets/conversation-websocket-context-Ywrxd_9p.js +3 -0
- package/build/assets/{copy-BxgbrjDT.js → copy-C7Ti2d8C.js} +1 -1
- package/build/assets/{custom-toast-handlers-BYxhSr3t.js → custom-toast-handlers-BOc3qeQ7.js} +1 -1
- package/build/assets/declaration-D378OjpZ.js +1 -0
- package/build/assets/{device-verify-CTbXX9CQ.js → device-verify-CMusn8nX.js} +1 -1
- package/build/assets/edit-automation-modal-Dnjxbjn7.js +1 -0
- package/build/assets/{ellipsis-button-BoU2-xlG.js → ellipsis-button-ugUATsNo.js} +1 -1
- package/build/assets/{entry.client-DU7-q4ZU.js → entry.client-D9uR9Blz.js} +2 -2
- package/build/assets/{enum-filter-dropdown-BJt-NplD.js → enum-filter-dropdown-1vpOGySB.js} +1 -1
- package/build/assets/{environment-switch-overlay-DQ1n6Iu6.js → environment-switch-overlay-CTCTQikP.js} +1 -1
- package/build/assets/{extensions-hub-BW1FAKFJ.js → extensions-hub-BSUseHVF.js} +1 -1
- package/build/assets/{extensions-navigation-CbPMhSML.js → extensions-navigation-CT1kc1u_.js} +1 -1
- package/build/assets/{files-tab-CbJ4s7Ik.js → files-tab-B3A1NDlZ.js} +1 -1
- package/build/assets/{folder-CerIk8uG.js → folder-0WSMImNX.js} +1 -1
- package/build/assets/git-control-bar-branch-button-CcIpmyfM.js +27 -0
- package/build/assets/{git-provider-icon-D8RE4unY.js → git-provider-icon-DYE9n7fs.js} +1 -1
- package/build/assets/{home-D9fJfhQA.js → home-dIzxi5Dd.js} +1 -1
- package/build/assets/{i18n-DkYgs32x.js → i18n-DjAGhTis.js} +1 -1
- package/build/assets/install-server-modal-z5VaHeXd.js +1 -0
- package/build/assets/{launch-DKCU9uJH.js → launch-hZ0ifhcV.js} +1 -1
- package/build/assets/{lesson-plan-CmkRbe6Z.js → lesson-plan-DRYG5SLI.js} +1 -1
- package/build/assets/{link-external-CvxB0BmI.js → link-external-Df8J52xI.js} +1 -1
- package/build/assets/{llm-client-BpIfxETv.js → llm-client-ChQzg4wX.js} +1 -1
- package/build/assets/llm-settings-2036m7Wt.js +1 -0
- package/build/assets/{llm-settings-BOJC4vD-.js → llm-settings-CcHqGOYL.js} +1 -1
- package/build/assets/{loading-spinner-91b5FiMQ.js → loading-spinner-C04FGh14.js} +1 -1
- package/build/assets/{manage-backends-modal-DqpzcxdI.js → manage-backends-modal-rYeyGx7j.js} +1 -1
- package/build/assets/{manage-workspaces-modal-eG6XgAvw.js → manage-workspaces-modal-C5EuW8m1.js} +1 -1
- package/build/assets/manifest-97e839da.js +1 -0
- package/build/assets/{markdown-renderer-wZnLDbA1.js → markdown-renderer-CEX4Becj.js} +1 -1
- package/build/assets/mcp-C06YssEI.js +9 -0
- package/build/assets/messages-T2ewVkbp.js +36 -0
- package/build/assets/{modal-backdrop-B04pVYAD.js → modal-backdrop-DTYGVmOR.js} +1 -1
- package/build/assets/{modal-body-CgUoFQA1.js → modal-body-YElmM1dV.js} +1 -1
- package/build/assets/{modal-close-button-SM_WXzDY.js → modal-close-button-C_GpQt9F.js} +1 -1
- package/build/assets/{model-selector-7id-Uirf.js → model-selector-DeMmw-Xa.js} +1 -1
- package/build/assets/{navigation-context-BFjstyH6.js → navigation-context-DeIPtGPp.js} +1 -1
- package/build/assets/{navigation-link-DFQ7YcWq.js → navigation-link-C9JD4PYD.js} +1 -1
- package/build/assets/{openhands-logo-DkDp75rC.js → openhands-logo-CI5Fhn1W.js} +1 -1
- package/build/assets/{option-service.api-DN0ZcGjw.js → option-service.api-DsI1UW7N.js} +1 -1
- package/build/assets/{organization-service.api-Ct2dZF8M.js → organization-service.api-COwMPFg5.js} +1 -1
- package/build/assets/{path-utils-D1ZtqFC7.js → path-utils-CqJboYxo.js} +1 -1
- package/build/assets/{plan-components-gOm-daR3.js → plan-components-DEjMuDDG.js} +1 -1
- package/build/assets/{planner-tab-yubfN-6U.js → planner-tab-BrntFmb1.js} +1 -1
- package/build/assets/{profiles-client-D4twHRVf.js → profiles-client-BGkKEV9j.js} +1 -1
- package/build/assets/{providers-C2T07PM3.js → providers-DXvCAN_u.js} +1 -1
- package/build/assets/{proxy-BMZyC45G.js → proxy-CurRmrqf.js} +1 -1
- package/build/assets/{query-client-config-CiK0GJJO.js → query-client-config-Ba7qAAoO.js} +1 -1
- package/build/assets/recommended-automations-launcher-BI9NhG8Y.js +52 -0
- package/build/assets/root-BS1Td78t.js +2 -0
- package/build/assets/root-DHeCXo9N.css +1 -0
- package/build/assets/root-layout-BLjAEgle.js +2 -0
- package/build/assets/{sdk-section-page-03k88tIR.js → sdk-section-page-CJW0G04-.js} +1 -1
- package/build/assets/{sdk-settings-schema-BY8dOy3a.js → sdk-settings-schema-QBYH-ONX.js} +1 -1
- package/build/assets/{search-BCAF9EDS.js → search-Cq_cFrDt.js} +1 -1
- package/build/assets/{secrets-service-Z3qtRb_G.js → secrets-service-Bwd5DeUs.js} +1 -1
- package/build/assets/{secrets-settings-BnlByuMZ.js → secrets-settings-MLXqOtX2.js} +1 -1
- package/build/assets/{server-client-CG1zHqph.js → server-client-C3mC8Hl3.js} +1 -1
- package/build/assets/{settings-DyzGLF_d.js → settings-D7E2U5tK.js} +1 -1
- package/build/assets/{settings-client-CkXDJwIY.js → settings-client-CwjfwoiB.js} +1 -1
- package/build/assets/{settings-dropdown-input-CAQWQgx-.js → settings-dropdown-input-VwAXNrOb.js} +1 -1
- package/build/assets/{settings-gear-D4ZkEDGb.js → settings-gear-BJwWR1ej.js} +1 -1
- package/build/assets/{settings-index-KtTw49xL.js → settings-index-J-3BNR0W.js} +1 -1
- package/build/assets/{settings-input-BWCZt9g2.js → settings-input-DBywAnA7.js} +1 -1
- package/build/assets/{settings-list-classes-xMleGkTC.js → settings-list-classes-BOS092DR.js} +1 -1
- package/build/assets/{settings-modal-Cv2YWSUY.js → settings-modal-B8vgWDTe.js} +1 -1
- package/build/assets/{settings-section-header-context-1wfkgjZZ.js → settings-section-header-context-32x6WTyL.js} +1 -1
- package/build/assets/settings-service.api-FvJGK45W.js +1 -0
- package/build/assets/{settings-switch-CGap2LtG.js → settings-switch-DTKmHC8F.js} +1 -1
- package/build/assets/{settings-utils-BBozxqqi.js → settings-utils-BsvSU3OM.js} +1 -1
- package/build/assets/{shared-conversation-BfZNCsvo.js → shared-conversation-a0QV8o99.js} +1 -1
- package/build/assets/{sidebar-mobile-menu-toggle-DXplko7u.js → sidebar-mobile-menu-toggle-DTUNI1WQ.js} +1 -1
- package/build/assets/{sidebar-nav-link-B4h8naZ7.js → sidebar-nav-link-CnWoZcwc.js} +1 -1
- package/build/assets/{skill-card-pill-row-D0oTWx-a.js → skill-card-pill-row-tZ599jli.js} +1 -1
- package/build/assets/{skills-BN8atjgW.js → skills-ZyAO5dyK.js} +1 -1
- package/build/assets/{skills-plugins-BTnp7QcQ.js → skills-plugins-BSRz041I.js} +1 -1
- package/build/assets/{skills-settings-CbOQvzkR.js → skills-settings-DOnMn9q1.js} +2 -2
- package/build/assets/{status-DDL-ipIP.js → status-CsatcFbK.js} +1 -1
- package/build/assets/{styled-tooltip-Awq4HMw3.js → styled-tooltip-CS3mB_1X.js} +1 -1
- package/build/assets/{switch-skeleton-Bv21RGWd.js → switch-skeleton-C-CfhYYV.js} +1 -1
- package/build/assets/{task-list-tab-B45ktzHM.js → task-list-tab-Day9nhRT.js} +1 -1
- package/build/assets/{terminal-DGuR4559.js → terminal-LNa-iU5c.js} +1 -1
- package/build/assets/{terminal-D5pzR9Ru.js → terminal-ro4SNjUU.js} +1 -1
- package/build/assets/{toggle-switch-gj6T-wsU.js → toggle-switch-k-IZCDbt.js} +1 -1
- package/build/assets/{typography-BbaUAC4V.js → typography-vVUMoNUg.js} +1 -1
- package/build/assets/{u-check-circle-DHGiAi-w.js → u-check-circle-DplbarS5.js} +1 -1
- package/build/assets/{u-check-circle-half-BPcWtWwv.js → u-check-circle-half-yDuiSZHC.js} +1 -1
- package/build/assets/{u-circuit-B_nK9hOu.js → u-circuit-C9tYkpeK.js} +1 -1
- package/build/assets/{u-edit-BPFJBd34.js → u-edit-KAUlufD8.js} +1 -1
- package/build/assets/{use-active-conversation-Bu5J9iLy.js → use-active-conversation-D15D9GgR.js} +1 -1
- package/build/assets/{use-agent-settings-schema-BbtOsR7P.js → use-agent-settings-schema-Bvp5UzV8.js} +1 -1
- package/build/assets/{use-agent-state-DN9Nc5pP.js → use-agent-state-DE5dlEXJ.js} +1 -1
- package/build/assets/{use-cloud-current-user-id-B_rMUiu8.js → use-cloud-current-user-id-DWVar4st.js} +1 -1
- package/build/assets/{use-config-Bcz2JL2t.js → use-config-BSu_53GL.js} +1 -1
- package/build/assets/{use-conversation-id-BOaaZahn.js → use-conversation-id-DajhCn2A.js} +1 -1
- package/build/assets/{use-create-conversation-BWFA_FId.js → use-create-conversation-DW7AGgLA.js} +1 -1
- package/build/assets/{use-handle-plan-click-CgrCGmT1.js → use-handle-plan-click-DpgEQDAV.js} +1 -1
- package/build/assets/use-is-authed-hXC8vxgT.js +1 -0
- package/build/assets/{use-is-creating-conversation-DhoM7UAB.js → use-is-creating-conversation-DhDeeWfA.js} +1 -1
- package/build/assets/{use-launch-skill-in-chat-DOyQsXFO.js → use-launch-skill-in-chat-DVGPFrbI.js} +1 -1
- package/build/assets/{use-llm-profiles-CAIzHJDX.js → use-llm-profiles-D3-KXwQ0.js} +1 -1
- package/build/assets/use-runtime-is-ready-XFbT16BD.js +1 -0
- package/build/assets/{use-save-settings-5m3w89Ph.js → use-save-settings-CEEKSTWG.js} +1 -1
- package/build/assets/{use-settings-DzG0C3vO.js → use-settings-DQ7Oo1Hj.js} +1 -1
- package/build/assets/{use-settings-nav-items-BIsKeX52.js → use-settings-nav-items-YmrXrjn9.js} +2 -2
- package/build/assets/{use-skills-Cn-78xP1.js → use-skills-Xe0vjPMt.js} +1 -1
- package/build/assets/{use-unified-vscode-url-C5iI-Z5A.js → use-unified-vscode-url-BOsIOd-b.js} +1 -1
- package/build/assets/use-user-conversation-Mc0mQgkl.js +1 -0
- package/build/assets/{useMutation-CRJwk4cR.js → useMutation-B4OUESdw.js} +1 -1
- package/build/assets/{useTranslation-01pF7z10.js → useTranslation-CpIcQBq6.js} +1 -1
- package/build/assets/{utils-Czcl6buL.js → utils-D-HX7JCe.js} +1 -1
- package/build/assets/{vendor~conversation-panel~conversation-CbjvWBSu.js → vendor~conversation-panel~conversation-BlCIz9XQ.js} +1 -1
- package/build/assets/{vendor~home~conversation-panel~conversation~shared-conversation~planner-tab~files-tab-CofhIDpd.js → vendor~home~conversation-panel~conversation~shared-conversation~planner-tab~files-tab-Ds9quNZ9.js} +1 -1
- package/build/assets/vendor~home~mcp~automations-list-C5PoHCy6.js +1 -0
- package/build/assets/{vendor~home~mcp~llm-settings~agent-settings~condenser-settings~verification-settings~app-se~ocm3mykx-BQPOygpY.js → vendor~home~mcp~llm-settings~agent-settings~condenser-settings~verification-settings~app-se~ocm3mykx-CGlZoBKa.js} +1 -1
- package/build/assets/{vendor~home~mcp~llm-settings~agent-settings~condenser-settings~verification-settings~app-se~ocm3mykx-CyYIBiBk.js → vendor~home~mcp~llm-settings~agent-settings~condenser-settings~verification-settings~app-se~ocm3mykx-DE11mPxp.js} +1 -1
- package/build/assets/{vendor~root-layout~home~conversation-panel~conversation~extensions-hub~skills-settings~skil~iguv7bgw-CuGq_cxH.js → vendor~root-layout~home~conversation-panel~conversation~extensions-hub~skills-settings~skil~iguv7bgw-8b8V5bfO.js} +1 -1
- package/build/assets/{vendor~root-layout~home~conversation-panel~conversation~extensions-hub~skills-settings~skil~iguv7bgw-CFpDeb9o.js → vendor~root-layout~home~conversation-panel~conversation~extensions-hub~skills-settings~skil~iguv7bgw-Dy7L6fMG.js} +1 -1
- package/build/assets/{vendor~root-layout~home~conversation-panel~conversation~extensions-hub~skills-settings~skil~k776hupu-C1p8-pMr.js → vendor~root-layout~home~conversation-panel~conversation~extensions-hub~skills-settings~skil~k776hupu-D40EXhZx.js} +1 -1
- package/build/assets/vendor~root-layout~home~conversation-panel~conversation~launch~extensions-hub~skills-settin~dp08i1qy-CHrEOFl6.js +48 -0
- package/build/assets/{vendor~root~root-layout~home~conversation-panel~conversation~launch~extensions-hub~skills-s~kyz9p27j-DlKA6SoO.js → vendor~root~root-layout~home~conversation-panel~conversation~launch~extensions-hub~skills-s~kyz9p27j-CyUbhpbm.js} +1 -1
- package/build/assets/{verification-settings-DbziMp4K.js → verification-settings-BtlTiHP8.js} +1 -1
- package/build/assets/{vscode-tab-BVhQR2rt.js → vscode-tab-C0ShhiSU.js} +1 -1
- package/build/assets/{waiting-for-runtime-message-JotSOBdC.js → waiting-for-runtime-message-DWPl_Yby.js} +1 -1
- package/build/assets/{x-mark-CZ57VvRX.js → x-mark-CWI0f9yI.js} +1 -1
- package/build/favicon.svg +1 -0
- package/build/index.html +4 -4
- package/build/locales/ar/openhands.json +8 -0
- package/build/locales/ca/openhands.json +8 -0
- package/build/locales/de/openhands.json +8 -0
- package/build/locales/en/openhands.json +8 -0
- package/build/locales/es/openhands.json +8 -0
- package/build/locales/fr/openhands.json +8 -0
- package/build/locales/it/openhands.json +8 -0
- package/build/locales/ja/openhands.json +8 -0
- package/build/locales/ko-KR/openhands.json +8 -0
- package/build/locales/no/openhands.json +8 -0
- package/build/locales/pt/openhands.json +8 -0
- package/build/locales/tr/openhands.json +8 -0
- package/build/locales/uk/openhands.json +8 -0
- package/build/locales/zh-CN/openhands.json +8 -0
- package/build/locales/zh-TW/openhands.json +8 -0
- package/config/defaults.json +1 -1
- package/dist/api/agent-server-config.cjs +1 -1
- package/dist/api/agent-server-config.cjs.map +1 -1
- package/dist/api/agent-server-config.d.ts +1 -1
- package/dist/api/agent-server-config.js +1 -1
- package/dist/api/agent-server-config.js.map +1 -1
- package/dist/api/conversation-service/agent-server-conversation-service.api.cjs +1 -1
- package/dist/api/conversation-service/agent-server-conversation-service.api.cjs.map +1 -1
- package/dist/api/conversation-service/agent-server-conversation-service.api.d.ts +12 -0
- package/dist/api/conversation-service/agent-server-conversation-service.api.js +4 -0
- package/dist/api/conversation-service/agent-server-conversation-service.api.js.map +1 -1
- package/dist/api/mcp-service/mcp-service.api.cjs +2 -0
- package/dist/api/mcp-service/mcp-service.api.cjs.map +1 -0
- package/dist/api/mcp-service/mcp-service.api.d.ts +6 -0
- package/dist/api/mcp-service/mcp-service.api.js +36 -0
- package/dist/api/mcp-service/mcp-service.api.js.map +1 -0
- package/dist/api/settings-service/settings-service.api.cjs +1 -1
- package/dist/api/settings-service/settings-service.api.cjs.map +1 -1
- package/dist/api/settings-service/settings-service.api.d.ts +1 -0
- package/dist/api/settings-service/settings-service.api.js +59 -41
- package/dist/api/settings-service/settings-service.api.js.map +1 -1
- package/dist/components/features/automations/detail/activity-log-item.d.ts +1 -1
- package/dist/components/features/automations/recommended-automations-launcher.d.ts +1 -1
- package/dist/components/features/backends/backend-selector.cjs +1 -1
- package/dist/components/features/backends/backend-selector.cjs.map +1 -1
- package/dist/components/features/backends/backend-selector.js +95 -95
- package/dist/components/features/backends/backend-selector.js.map +1 -1
- package/dist/components/features/chat/change-agent-button.cjs +1 -1
- package/dist/components/features/chat/change-agent-button.cjs.map +1 -1
- package/dist/components/features/chat/change-agent-button.js +65 -59
- package/dist/components/features/chat/change-agent-button.js.map +1 -1
- package/dist/components/features/chat/chat-interface.cjs +2 -2
- package/dist/components/features/chat/chat-interface.cjs.map +1 -1
- package/dist/components/features/chat/chat-interface.js +15 -14
- package/dist/components/features/chat/chat-interface.js.map +1 -1
- package/dist/components/features/chat/components/chat-input-actions.cjs +1 -1
- package/dist/components/features/chat/components/chat-input-actions.cjs.map +1 -1
- package/dist/components/features/chat/components/chat-input-actions.js +115 -137
- package/dist/components/features/chat/components/chat-input-actions.js.map +1 -1
- package/dist/components/features/chat/components/chat-input-model.cjs +1 -1
- package/dist/components/features/chat/components/chat-input-model.cjs.map +1 -1
- package/dist/components/features/chat/components/chat-input-model.d.ts +10 -0
- package/dist/components/features/chat/components/chat-input-model.js +95 -60
- package/dist/components/features/chat/components/chat-input-model.js.map +1 -1
- package/dist/components/features/chat/components/slash-command-menu.cjs +1 -1
- package/dist/components/features/chat/components/slash-command-menu.cjs.map +1 -1
- package/dist/components/features/chat/components/slash-command-menu.js +1 -1
- package/dist/components/features/chat/components/slash-command-menu.js.map +1 -1
- package/dist/components/features/chat/git-control-bar.cjs +1 -1
- package/dist/components/features/chat/git-control-bar.cjs.map +1 -1
- package/dist/components/features/chat/git-control-bar.js +60 -59
- package/dist/components/features/chat/git-control-bar.js.map +1 -1
- package/dist/components/features/conversation/conversation-name-with-status.cjs +1 -1
- package/dist/components/features/conversation/conversation-name-with-status.cjs.map +1 -1
- package/dist/components/features/conversation/conversation-name-with-status.js +2 -2
- package/dist/components/features/conversation/conversation-name-with-status.js.map +1 -1
- package/dist/components/features/conversation/conversation-name.cjs +1 -1
- package/dist/components/features/conversation/conversation-name.cjs.map +1 -1
- package/dist/components/features/conversation/conversation-name.js +3 -3
- package/dist/components/features/conversation/conversation-name.js.map +1 -1
- package/dist/components/features/conversation/conversation-tabs/conversation-tabs-context-menu.cjs +1 -1
- package/dist/components/features/conversation/conversation-tabs/conversation-tabs-context-menu.cjs.map +1 -1
- package/dist/components/features/conversation/conversation-tabs/conversation-tabs-context-menu.js +1 -1
- package/dist/components/features/conversation/conversation-tabs/conversation-tabs-context-menu.js.map +1 -1
- package/dist/components/features/conversation/conversation-tabs/conversation-tabs.cjs +1 -1
- package/dist/components/features/conversation/conversation-tabs/conversation-tabs.cjs.map +1 -1
- package/dist/components/features/conversation/conversation-tabs/conversation-tabs.js +16 -16
- package/dist/components/features/conversation/conversation-tabs/conversation-tabs.js.map +1 -1
- package/dist/components/features/mcp-logo-badge.cjs +1 -1
- package/dist/components/features/mcp-logo-badge.cjs.map +1 -1
- package/dist/components/features/mcp-logo-badge.d.ts +2 -2
- package/dist/components/features/mcp-logo-badge.js +1 -1
- package/dist/components/features/mcp-logo-badge.js.map +1 -1
- package/dist/components/features/mcp-page/custom-server-editor.cjs +1 -1
- package/dist/components/features/mcp-page/custom-server-editor.cjs.map +1 -1
- package/dist/components/features/mcp-page/custom-server-editor.js +64 -41
- package/dist/components/features/mcp-page/custom-server-editor.js.map +1 -1
- package/dist/components/features/mcp-page/install-server-modal.cjs +1 -1
- package/dist/components/features/mcp-page/install-server-modal.cjs.map +1 -1
- package/dist/components/features/mcp-page/install-server-modal.d.ts +1 -1
- package/dist/components/features/mcp-page/install-server-modal.js +126 -102
- package/dist/components/features/mcp-page/install-server-modal.js.map +1 -1
- package/dist/components/features/mcp-page/installed-server-card.cjs +1 -1
- package/dist/components/features/mcp-page/installed-server-card.cjs.map +1 -1
- package/dist/components/features/mcp-page/installed-server-card.js +1 -1
- package/dist/components/features/mcp-page/installed-server-card.js.map +1 -1
- package/dist/components/features/mcp-page/marketplace-card.cjs +1 -1
- package/dist/components/features/mcp-page/marketplace-card.cjs.map +1 -1
- package/dist/components/features/mcp-page/marketplace-card.d.ts +1 -1
- package/dist/components/features/mcp-page/marketplace-card.js +27 -25
- package/dist/components/features/mcp-page/marketplace-card.js.map +1 -1
- package/dist/components/features/mcp-page/marketplace-section.cjs +1 -1
- package/dist/components/features/mcp-page/marketplace-section.cjs.map +1 -1
- package/dist/components/features/mcp-page/marketplace-section.d.ts +1 -1
- package/dist/components/features/mcp-page/marketplace-section.js +1 -1
- package/dist/components/features/mcp-page/marketplace-section.js.map +1 -1
- package/dist/components/features/mcp-page/mcp-logo-stack-badge.d.ts +2 -2
- package/dist/components/features/settings/mcp-settings/mcp-server-form.cjs +7 -7
- package/dist/components/features/settings/mcp-settings/mcp-server-form.cjs.map +1 -1
- package/dist/components/features/settings/mcp-settings/mcp-server-form.d.ts +8 -12
- package/dist/components/features/settings/mcp-settings/mcp-server-form.js +114 -87
- package/dist/components/features/settings/mcp-settings/mcp-server-form.js.map +1 -1
- package/dist/components/features/sidebar/sidebar-rail-body.cjs +1 -1
- package/dist/components/features/sidebar/sidebar-rail-body.cjs.map +1 -1
- package/dist/components/features/sidebar/sidebar-rail-body.d.ts +1 -2
- package/dist/components/features/sidebar/sidebar-rail-body.js +104 -104
- package/dist/components/features/sidebar/sidebar-rail-body.js.map +1 -1
- package/dist/components/features/sidebar/sidebar.cjs +1 -1
- package/dist/components/features/sidebar/sidebar.cjs.map +1 -1
- package/dist/components/features/sidebar/sidebar.js +82 -83
- package/dist/components/features/sidebar/sidebar.js.map +1 -1
- package/dist/context/scroll-context.cjs +1 -1
- package/dist/context/scroll-context.cjs.map +1 -1
- package/dist/context/scroll-context.d.ts +1 -0
- package/dist/context/scroll-context.js +4 -1
- package/dist/context/scroll-context.js.map +1 -1
- package/dist/contexts/conversation-websocket-context.cjs +3 -3
- package/dist/contexts/conversation-websocket-context.cjs.map +1 -1
- package/dist/contexts/conversation-websocket-context.js +36 -36
- package/dist/contexts/conversation-websocket-context.js.map +1 -1
- package/dist/favicon.svg +1 -0
- package/dist/hooks/mutation/use-switch-acp-model.cjs +2 -0
- package/dist/hooks/mutation/use-switch-acp-model.cjs.map +1 -0
- package/dist/hooks/mutation/use-switch-acp-model.d.ts +23 -0
- package/dist/hooks/mutation/use-switch-acp-model.js +26 -0
- package/dist/hooks/mutation/use-switch-acp-model.js.map +1 -0
- package/dist/hooks/mutation/use-test-mcp-server.cjs +2 -0
- package/dist/hooks/mutation/use-test-mcp-server.cjs.map +1 -0
- package/dist/hooks/mutation/use-test-mcp-server.d.ts +2 -0
- package/dist/hooks/mutation/use-test-mcp-server.js +10 -0
- package/dist/hooks/mutation/use-test-mcp-server.js.map +1 -0
- package/dist/hooks/query/use-automation-detail.d.ts +3 -2
- package/dist/hooks/query/use-local-git-info.cjs +3 -1
- package/dist/hooks/query/use-local-git-info.cjs.map +1 -1
- package/dist/hooks/query/use-local-git-info.d.ts +2 -2
- package/dist/hooks/query/use-local-git-info.js +27 -24
- package/dist/hooks/query/use-local-git-info.js.map +1 -1
- package/dist/hooks/use-acp-model-context.cjs.map +1 -1
- package/dist/hooks/use-acp-model-context.d.ts +3 -4
- package/dist/hooks/use-acp-model-context.js.map +1 -1
- package/dist/hooks/use-chat-input-model-state.cjs +2 -0
- package/dist/hooks/use-chat-input-model-state.cjs.map +1 -0
- package/dist/hooks/use-chat-input-model-state.d.ts +12 -0
- package/dist/hooks/use-chat-input-model-state.js +29 -0
- package/dist/hooks/use-chat-input-model-state.js.map +1 -0
- package/dist/i18n/declaration.cjs +1 -1
- package/dist/i18n/declaration.cjs.map +1 -1
- package/dist/i18n/declaration.d.ts +8 -0
- package/dist/i18n/declaration.js +1 -1
- package/dist/i18n/declaration.js.map +1 -1
- package/dist/i18n/translation.cjs +2 -2
- package/dist/i18n/translation.cjs.map +1 -1
- package/dist/i18n/translation.js +136 -0
- package/dist/i18n/translation.js.map +1 -1
- package/dist/locales/ar/openhands.json +8 -0
- package/dist/locales/ca/openhands.json +8 -0
- package/dist/locales/de/openhands.json +8 -0
- package/dist/locales/en/openhands.json +8 -0
- package/dist/locales/es/openhands.json +8 -0
- package/dist/locales/fr/openhands.json +8 -0
- package/dist/locales/it/openhands.json +8 -0
- package/dist/locales/ja/openhands.json +8 -0
- package/dist/locales/ko-KR/openhands.json +8 -0
- package/dist/locales/no/openhands.json +8 -0
- package/dist/locales/pt/openhands.json +8 -0
- package/dist/locales/tr/openhands.json +8 -0
- package/dist/locales/uk/openhands.json +8 -0
- package/dist/locales/zh-CN/openhands.json +8 -0
- package/dist/locales/zh-TW/openhands.json +8 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/airtable.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/airtable.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/airtable.js +37 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/airtable.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/apify.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/apify.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/apify.js +36 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/apify.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/atlassian.cjs +1 -1
- package/dist/node_modules/@openhands/extensions/integrations/catalog/atlassian.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/atlassian.js +15 -5
- package/dist/node_modules/@openhands/extensions/integrations/catalog/atlassian.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/brave-search.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/brave-search.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/brave-search.js +36 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/brave-search.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/browser-mcp.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/browser-mcp.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/browser-mcp.js +31 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/browser-mcp.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/clickhouse.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/clickhouse.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/clickhouse.js +52 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/clickhouse.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/cloudflare-bindings.cjs +1 -1
- package/dist/node_modules/@openhands/extensions/integrations/catalog/cloudflare-bindings.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/cloudflare-bindings.js +15 -5
- package/dist/node_modules/@openhands/extensions/integrations/catalog/cloudflare-bindings.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/cloudflare-browser-rendering.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/cloudflare-browser-rendering.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/cloudflare-browser-rendering.js +15 -5
- package/dist/node_modules/@openhands/extensions/integrations/catalog/cloudflare-browser-rendering.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/cloudflare-builds.cjs +1 -1
- package/dist/node_modules/@openhands/extensions/integrations/catalog/cloudflare-builds.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/cloudflare-builds.js +15 -5
- package/dist/node_modules/@openhands/extensions/integrations/catalog/cloudflare-builds.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/cloudflare-docs.cjs +1 -1
- package/dist/node_modules/@openhands/extensions/integrations/catalog/cloudflare-docs.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/cloudflare-docs.js +15 -5
- package/dist/node_modules/@openhands/extensions/integrations/catalog/cloudflare-docs.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/cloudflare-observability.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/cloudflare-observability.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/cloudflare-observability.js +15 -5
- package/dist/node_modules/@openhands/extensions/integrations/catalog/cloudflare-observability.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/deepwiki.cjs +1 -1
- package/dist/node_modules/@openhands/extensions/integrations/catalog/deepwiki.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/deepwiki.js +15 -5
- package/dist/node_modules/@openhands/extensions/integrations/catalog/deepwiki.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/elevenlabs.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/elevenlabs.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/elevenlabs.js +36 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/elevenlabs.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/everything.cjs +1 -1
- package/dist/node_modules/@openhands/extensions/integrations/catalog/everything.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/everything.js +13 -6
- package/dist/node_modules/@openhands/extensions/integrations/catalog/everything.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/exa.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/exa.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/exa.js +36 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/exa.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/fetch.cjs +1 -1
- package/dist/node_modules/@openhands/extensions/integrations/catalog/fetch.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/fetch.js +13 -6
- package/dist/node_modules/@openhands/extensions/integrations/catalog/fetch.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/figma.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/figma.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/figma.js +40 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/figma.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/filesystem.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/filesystem.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/filesystem.js +39 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/filesystem.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/firecrawl.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/firecrawl.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/firecrawl.js +36 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/firecrawl.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/git.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/git.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/git.js +40 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/git.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/github.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/github.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/github.js +47 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/github.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/huggingface.cjs +1 -1
- package/dist/node_modules/@openhands/extensions/integrations/catalog/huggingface.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/huggingface.js +15 -5
- package/dist/node_modules/@openhands/extensions/integrations/catalog/huggingface.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/kagi.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/kagi.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/kagi.js +36 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/kagi.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/linear.cjs +1 -1
- package/dist/node_modules/@openhands/extensions/integrations/catalog/linear.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/linear.js +15 -5
- package/dist/node_modules/@openhands/extensions/integrations/catalog/linear.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/memory.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/memory.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/memory.js +13 -6
- package/dist/node_modules/@openhands/extensions/integrations/catalog/memory.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/mongodb.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/mongodb.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/mongodb.js +36 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/mongodb.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/neon.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/neon.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/neon.js +40 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/neon.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/notion.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/notion.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/notion.js +39 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/notion.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/obsidian.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/obsidian.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/obsidian.js +38 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/obsidian.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/paypal.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/paypal.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/paypal.js +15 -5
- package/dist/node_modules/@openhands/extensions/integrations/catalog/paypal.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/playwright.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/playwright.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/playwright.js +13 -6
- package/dist/node_modules/@openhands/extensions/integrations/catalog/playwright.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/redis.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/redis.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/redis.js +43 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/redis.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/resend.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/resend.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/resend.js +41 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/resend.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/sentry.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/sentry.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/sentry.js +15 -5
- package/dist/node_modules/@openhands/extensions/integrations/catalog/sentry.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/sequential-thinking.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/sequential-thinking.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/sequential-thinking.js +13 -6
- package/dist/node_modules/@openhands/extensions/integrations/catalog/sequential-thinking.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/slack.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/slack.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/slack.js +45 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/slack.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/stripe.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/stripe.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/stripe.js +15 -5
- package/dist/node_modules/@openhands/extensions/integrations/catalog/stripe.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/supabase.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/supabase.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/supabase.js +37 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/supabase.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/tavily.cjs +1 -1
- package/dist/node_modules/@openhands/extensions/integrations/catalog/tavily.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/tavily.js +39 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/tavily.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/time.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/catalog/time.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/catalog/time.js +13 -6
- package/dist/node_modules/@openhands/extensions/integrations/catalog/time.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/index.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/index.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/index.js +175 -0
- package/dist/node_modules/@openhands/extensions/integrations/index.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/logos.cjs +1 -1
- package/dist/node_modules/@openhands/extensions/integrations/logos.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/{mcps → integrations}/logos.js +2 -2
- package/dist/node_modules/@openhands/extensions/integrations/logos.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/oauth-provider-catalog.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/oauth-provider-catalog.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/oauth-provider-catalog.js +548 -0
- package/dist/node_modules/@openhands/extensions/integrations/oauth-provider-catalog.js.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/oauth-provider-registration-defaults.cjs +2 -0
- package/dist/node_modules/@openhands/extensions/integrations/oauth-provider-registration-defaults.cjs.map +1 -0
- package/dist/node_modules/@openhands/extensions/integrations/oauth-provider-registration-defaults.js +482 -0
- package/dist/node_modules/@openhands/extensions/integrations/oauth-provider-registration-defaults.js.map +1 -0
- package/dist/node_modules/@openhands/typescript-client/dist/client/conversation-client.cjs +1 -1
- package/dist/node_modules/@openhands/typescript-client/dist/client/conversation-client.cjs.map +1 -1
- package/dist/node_modules/@openhands/typescript-client/dist/client/conversation-client.js +3 -0
- package/dist/node_modules/@openhands/typescript-client/dist/client/conversation-client.js.map +1 -1
- package/dist/node_modules/@openhands/typescript-client/dist/client/mcp-client.cjs +2 -0
- package/dist/node_modules/@openhands/typescript-client/dist/client/mcp-client.cjs.map +1 -0
- package/dist/node_modules/@openhands/typescript-client/dist/client/mcp-client.js +22 -0
- package/dist/node_modules/@openhands/typescript-client/dist/client/mcp-client.js.map +1 -0
- package/dist/node_modules/@openhands/typescript-client/dist/index.cjs +1 -1
- package/dist/node_modules/@openhands/typescript-client/dist/index.js +1 -0
- package/dist/node_modules/@openhands/typescript-client/dist/models/acp-providers.cjs +1 -1
- package/dist/node_modules/@openhands/typescript-client/dist/models/acp-providers.cjs.map +1 -1
- package/dist/node_modules/@openhands/typescript-client/dist/models/acp-providers.js +3 -0
- package/dist/node_modules/@openhands/typescript-client/dist/models/acp-providers.js.map +1 -1
- package/dist/package.cjs +1 -1
- package/dist/package.cjs.map +1 -1
- package/dist/package.js +6 -4
- package/dist/package.js.map +1 -1
- package/dist/routes/mcp.cjs +1 -1
- package/dist/routes/mcp.cjs.map +1 -1
- package/dist/routes/mcp.js +1 -1
- package/dist/routes/mcp.js.map +1 -1
- package/dist/stores/error-message-store.cjs +1 -1
- package/dist/stores/error-message-store.cjs.map +1 -1
- package/dist/stores/error-message-store.d.ts +10 -1
- package/dist/stores/error-message-store.js +16 -3
- package/dist/stores/error-message-store.js.map +1 -1
- package/dist/utils/mcp-marketplace-utils.cjs +1 -1
- package/dist/utils/mcp-marketplace-utils.cjs.map +1 -1
- package/dist/utils/mcp-marketplace-utils.d.ts +21 -1
- package/dist/utils/mcp-marketplace-utils.js +23 -13
- package/dist/utils/mcp-marketplace-utils.js.map +1 -1
- package/dist/utils/settings-utils.cjs.map +1 -1
- package/dist/utils/settings-utils.js.map +1 -1
- package/package.json +6 -4
- package/scripts/check-sdk-version-sync.mjs +6 -6
- package/scripts/dev-safe.mjs +25 -7
- package/scripts/dev-static.mjs +6 -0
- package/scripts/dev-with-automation.mjs +12 -1
- package/scripts/static-server.mjs +85 -4
- package/tools/canvas_ui_tool.py +129 -0
- package/build/assets/add-backend-modal-CqjNjGqY.js +0 -1
- package/build/assets/automation-detail-CQrtk33s.js +0 -1
- package/build/assets/automations-list-COmogz0S.js +0 -1
- package/build/assets/conversation-CeGMBOyB.js +0 -1
- package/build/assets/conversation-D8scXOe7.js +0 -17
- package/build/assets/conversation-panel-DMz46ji-.js +0 -1
- package/build/assets/conversation-websocket-context-B0Gd3yiT.js +0 -3
- package/build/assets/declaration-C9nuq2Dj.js +0 -1
- package/build/assets/edit-automation-modal-DnTHJrf1.js +0 -1
- package/build/assets/git-control-bar-branch-button-DhpPgadK.js +0 -27
- package/build/assets/install-server-modal-VB5hOBpW.js +0 -1
- package/build/assets/llm-settings-CIdxmimN.js +0 -1
- package/build/assets/manifest-6400820c.js +0 -1
- package/build/assets/mcp-BdfyCW1l.js +0 -9
- package/build/assets/messages-BfaEAG2q.js +0 -36
- package/build/assets/recommended-automations-launcher-Cx7svuGE.js +0 -52
- package/build/assets/root-6AdVEJBT.js +0 -2
- package/build/assets/root-DEotKI6b.css +0 -1
- package/build/assets/root-layout-DvYGxAnr.js +0 -2
- package/build/assets/settings-service.api-Z6x0l0GU.js +0 -1
- package/build/assets/use-is-authed-BFoh8Ogh.js +0 -1
- package/build/assets/use-runtime-is-ready-BQWLEyqa.js +0 -1
- package/build/assets/use-user-conversation-BCYpbPT1.js +0 -1
- package/build/assets/vendor~home~mcp~automations-list-DRfWZRnF.js +0 -1
- package/build/assets/vendor~root-layout~home~conversation-panel~conversation~launch~extensions-hub~skills-settin~dp08i1qy-BJm2mGIp.js +0 -48
- package/dist/node_modules/@openhands/extensions/mcps/catalog/airtable.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/airtable.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/airtable.js +0 -30
- package/dist/node_modules/@openhands/extensions/mcps/catalog/airtable.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/apify.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/apify.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/apify.js +0 -29
- package/dist/node_modules/@openhands/extensions/mcps/catalog/apify.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/atlassian.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/atlassian.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/brave-search.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/brave-search.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/brave-search.js +0 -29
- package/dist/node_modules/@openhands/extensions/mcps/catalog/brave-search.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/browser-mcp.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/browser-mcp.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/browser-mcp.js +0 -24
- package/dist/node_modules/@openhands/extensions/mcps/catalog/browser-mcp.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/clickhouse.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/clickhouse.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/clickhouse.js +0 -45
- package/dist/node_modules/@openhands/extensions/mcps/catalog/clickhouse.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/cloudflare-bindings.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/cloudflare-bindings.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/cloudflare-browser-rendering.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/cloudflare-browser-rendering.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/cloudflare-browser-rendering.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/cloudflare-builds.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/cloudflare-builds.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/cloudflare-docs.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/cloudflare-docs.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/cloudflare-observability.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/cloudflare-observability.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/cloudflare-observability.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/deepwiki.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/deepwiki.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/elevenlabs.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/elevenlabs.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/elevenlabs.js +0 -29
- package/dist/node_modules/@openhands/extensions/mcps/catalog/elevenlabs.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/everything.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/everything.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/exa.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/exa.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/exa.js +0 -29
- package/dist/node_modules/@openhands/extensions/mcps/catalog/exa.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/fetch.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/fetch.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/figma.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/figma.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/figma.js +0 -33
- package/dist/node_modules/@openhands/extensions/mcps/catalog/figma.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/filesystem.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/filesystem.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/filesystem.js +0 -32
- package/dist/node_modules/@openhands/extensions/mcps/catalog/filesystem.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/firecrawl.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/firecrawl.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/firecrawl.js +0 -29
- package/dist/node_modules/@openhands/extensions/mcps/catalog/firecrawl.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/git.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/git.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/git.js +0 -33
- package/dist/node_modules/@openhands/extensions/mcps/catalog/git.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/github.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/github.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/github.js +0 -40
- package/dist/node_modules/@openhands/extensions/mcps/catalog/github.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/huggingface.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/huggingface.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/kagi.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/kagi.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/kagi.js +0 -29
- package/dist/node_modules/@openhands/extensions/mcps/catalog/kagi.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/linear.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/linear.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/memory.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/memory.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/memory.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/mongodb.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/mongodb.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/mongodb.js +0 -29
- package/dist/node_modules/@openhands/extensions/mcps/catalog/mongodb.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/neon.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/neon.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/neon.js +0 -33
- package/dist/node_modules/@openhands/extensions/mcps/catalog/neon.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/notion.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/notion.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/notion.js +0 -32
- package/dist/node_modules/@openhands/extensions/mcps/catalog/notion.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/obsidian.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/obsidian.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/obsidian.js +0 -31
- package/dist/node_modules/@openhands/extensions/mcps/catalog/obsidian.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/paypal.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/paypal.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/paypal.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/playwright.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/playwright.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/playwright.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/redis.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/redis.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/redis.js +0 -36
- package/dist/node_modules/@openhands/extensions/mcps/catalog/redis.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/resend.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/resend.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/resend.js +0 -34
- package/dist/node_modules/@openhands/extensions/mcps/catalog/resend.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/sentry.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/sentry.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/sentry.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/sequential-thinking.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/sequential-thinking.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/sequential-thinking.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/slack.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/slack.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/slack.js +0 -38
- package/dist/node_modules/@openhands/extensions/mcps/catalog/slack.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/stripe.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/stripe.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/stripe.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/supabase.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/supabase.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/supabase.js +0 -30
- package/dist/node_modules/@openhands/extensions/mcps/catalog/supabase.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/tavily.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/tavily.js +0 -32
- package/dist/node_modules/@openhands/extensions/mcps/catalog/tavily.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/time.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/catalog/time.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/catalog/time.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/index.cjs +0 -2
- package/dist/node_modules/@openhands/extensions/mcps/index.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/index.js +0 -87
- package/dist/node_modules/@openhands/extensions/mcps/index.js.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/logos.cjs.map +0 -1
- package/dist/node_modules/@openhands/extensions/mcps/logos.js.map +0 -1
|
@@ -513,6 +513,7 @@
|
|
|
513
513
|
"CHAT_INTERFACE$DISCONNECTED": "Disconnesso",
|
|
514
514
|
"CHAT_INTERFACE$COMMANDS": "Comandi",
|
|
515
515
|
"MODEL$AVAILABLE_PROFILES": "Profili disponibili ({{count}})",
|
|
516
|
+
"MODEL$AVAILABLE_MODELS": "Modelli disponibili",
|
|
516
517
|
"MODEL$NO_SAVED_PROFILES": "Nessun profilo salvato",
|
|
517
518
|
"MODEL$NO_PROFILES_HINT": "Usa la pagina delle impostazioni LLM per creare un profilo, poi esegui /model <name> per cambiare.",
|
|
518
519
|
"MODEL$SWITCHED_TO_PROFILE": "Passato al profilo <cmd>{{name}}</cmd>",
|
|
@@ -806,6 +807,12 @@
|
|
|
806
807
|
"MCP$VIEW_DOCS": "Visualizza la documentazione →",
|
|
807
808
|
"MCP$INSTALL_SUCCESS": "Server MCP salvato.",
|
|
808
809
|
"MCP$REMOVE_SUCCESS": "Server MCP rimosso.",
|
|
810
|
+
"MCP$TEST_BUTTON": "Prova connessione",
|
|
811
|
+
"MCP$VERIFYING": "Verifica in corso…",
|
|
812
|
+
"MCP$TEST_SUCCESS": "Connesso — {{count}} strumento/i disponibile/i",
|
|
813
|
+
"MCP$TEST_ERROR_TIMEOUT": "Connessione scaduta. Controlla l'URL e riprova.",
|
|
814
|
+
"MCP$TEST_ERROR_CONNECTION": "Impossibile raggiungere il server. Controlla l'URL e il tipo di server.",
|
|
815
|
+
"MCP$TEST_ERROR_UNKNOWN": "Connessione non riuscita: {{-error}}",
|
|
809
816
|
"MCP$EDIT_SERVER_ARIA": "Modifica {{name}}",
|
|
810
817
|
"MCP$DELETE_SERVER_ARIA": "Rimuovi {{name}}",
|
|
811
818
|
"MCP$ERROR_FIELD_REQUIRED": "Questo campo è obbligatorio.",
|
|
@@ -1364,6 +1371,7 @@
|
|
|
1364
1371
|
"COMMON$LET_S_WORK_ON_A_PLAN": "Lavoriamo su un piano",
|
|
1365
1372
|
"COMMON$CODE_AGENT_DESCRIPTION": "Scrivi, modifica e esegui il debug con assistenza AI in tempo reale.",
|
|
1366
1373
|
"COMMON$PLAN_AGENT_DESCRIPTION": "Definisci gli obiettivi, struttura le attività e pianifica i prossimi passi.",
|
|
1374
|
+
"CHANGE_AGENT$SWITCH_AFTER_CONVERSATION": "La modalità agente può essere cambiata dopo l'inizio di una conversazione",
|
|
1367
1375
|
"PLANNING_AGENTT$PLANNING_AGENT_INITIALIZED": "Agente di pianificazione inizializzato",
|
|
1368
1376
|
"OBSERVATION_MESSAGE$SKILL_READY": "Abilità pronta",
|
|
1369
1377
|
"ACCOUNT_SETTINGS$SETTINGS": "Impostazioni",
|
|
@@ -513,6 +513,7 @@
|
|
|
513
513
|
"CHAT_INTERFACE$DISCONNECTED": "切断されました",
|
|
514
514
|
"CHAT_INTERFACE$COMMANDS": "コマンド",
|
|
515
515
|
"MODEL$AVAILABLE_PROFILES": "利用可能なプロファイル ({{count}})",
|
|
516
|
+
"MODEL$AVAILABLE_MODELS": "利用可能なモデル",
|
|
516
517
|
"MODEL$NO_SAVED_PROFILES": "保存済みプロファイルはありません",
|
|
517
518
|
"MODEL$NO_PROFILES_HINT": "LLM 設定ページでプロファイルを作成し、/model <name> を実行して切り替えてください。",
|
|
518
519
|
"MODEL$SWITCHED_TO_PROFILE": "プロファイル <cmd>{{name}}</cmd> に切り替えました",
|
|
@@ -806,6 +807,12 @@
|
|
|
806
807
|
"MCP$VIEW_DOCS": "ドキュメントを表示 →",
|
|
807
808
|
"MCP$INSTALL_SUCCESS": "MCP サーバーを保存しました。",
|
|
808
809
|
"MCP$REMOVE_SUCCESS": "MCP サーバーを削除しました。",
|
|
810
|
+
"MCP$TEST_BUTTON": "接続をテスト",
|
|
811
|
+
"MCP$VERIFYING": "検証中…",
|
|
812
|
+
"MCP$TEST_SUCCESS": "接続しました — {{count}}個のツールが利用可能です",
|
|
813
|
+
"MCP$TEST_ERROR_TIMEOUT": "接続がタイムアウトしました。URLを確認して、もう一度お試しください。",
|
|
814
|
+
"MCP$TEST_ERROR_CONNECTION": "サーバーに接続できませんでした。URLとサーバーの種類を確認してください。",
|
|
815
|
+
"MCP$TEST_ERROR_UNKNOWN": "接続に失敗しました: {{-error}}",
|
|
809
816
|
"MCP$EDIT_SERVER_ARIA": "{{name}} を編集",
|
|
810
817
|
"MCP$DELETE_SERVER_ARIA": "{{name}} を削除",
|
|
811
818
|
"MCP$ERROR_FIELD_REQUIRED": "このフィールドは必須です。",
|
|
@@ -1364,6 +1371,7 @@
|
|
|
1364
1371
|
"COMMON$LET_S_WORK_ON_A_PLAN": "プランに取り組みましょう",
|
|
1365
1372
|
"COMMON$CODE_AGENT_DESCRIPTION": "AIの支援をリアルタイムで受けながら、コードの作成、編集、デバッグを行いましょう。",
|
|
1366
1373
|
"COMMON$PLAN_AGENT_DESCRIPTION": "目標を明確にし、タスクを構造化し、次のステップを計画しましょう。",
|
|
1374
|
+
"CHANGE_AGENT$SWITCH_AFTER_CONVERSATION": "会話が始まると、エージェントモードを切り替えられます",
|
|
1367
1375
|
"PLANNING_AGENTT$PLANNING_AGENT_INITIALIZED": "プランニングエージェントが初期化されました",
|
|
1368
1376
|
"OBSERVATION_MESSAGE$SKILL_READY": "スキル準備完了",
|
|
1369
1377
|
"ACCOUNT_SETTINGS$SETTINGS": "設定",
|
|
@@ -513,6 +513,7 @@
|
|
|
513
513
|
"CHAT_INTERFACE$DISCONNECTED": "연결 끊김",
|
|
514
514
|
"CHAT_INTERFACE$COMMANDS": "명령어",
|
|
515
515
|
"MODEL$AVAILABLE_PROFILES": "사용 가능한 프로필 ({{count}})",
|
|
516
|
+
"MODEL$AVAILABLE_MODELS": "사용 가능한 모델",
|
|
516
517
|
"MODEL$NO_SAVED_PROFILES": "저장된 프로필이 없습니다",
|
|
517
518
|
"MODEL$NO_PROFILES_HINT": "LLM 설정 페이지에서 프로필을 만든 다음 /model <name>을 실행해 전환하세요.",
|
|
518
519
|
"MODEL$SWITCHED_TO_PROFILE": "<cmd>{{name}}</cmd> 프로필로 전환했습니다",
|
|
@@ -806,6 +807,12 @@
|
|
|
806
807
|
"MCP$VIEW_DOCS": "문서 보기 →",
|
|
807
808
|
"MCP$INSTALL_SUCCESS": "MCP 서버가 저장되었습니다.",
|
|
808
809
|
"MCP$REMOVE_SUCCESS": "MCP 서버가 제거되었습니다.",
|
|
810
|
+
"MCP$TEST_BUTTON": "연결 테스트",
|
|
811
|
+
"MCP$VERIFYING": "확인 중…",
|
|
812
|
+
"MCP$TEST_SUCCESS": "연결됨 — {{count}}개의 도구를 사용할 수 있습니다",
|
|
813
|
+
"MCP$TEST_ERROR_TIMEOUT": "연결 시간이 초과되었습니다. URL을 확인하고 다시 시도하세요.",
|
|
814
|
+
"MCP$TEST_ERROR_CONNECTION": "서버에 연결할 수 없습니다. URL과 서버 유형을 확인하세요.",
|
|
815
|
+
"MCP$TEST_ERROR_UNKNOWN": "연결에 실패했습니다: {{-error}}",
|
|
809
816
|
"MCP$EDIT_SERVER_ARIA": "{{name}} 편집",
|
|
810
817
|
"MCP$DELETE_SERVER_ARIA": "{{name}} 제거",
|
|
811
818
|
"MCP$ERROR_FIELD_REQUIRED": "이 필드는 필수입니다.",
|
|
@@ -1364,6 +1371,7 @@
|
|
|
1364
1371
|
"COMMON$LET_S_WORK_ON_A_PLAN": "계획을 세워봅시다",
|
|
1365
1372
|
"COMMON$CODE_AGENT_DESCRIPTION": "AI의 지원을 받아 실시간으로 작성, 편집 및 디버깅하세요.",
|
|
1366
1373
|
"COMMON$PLAN_AGENT_DESCRIPTION": "목표를 개요하고, 작업을 구조화하며, 다음 단계를 구상하세요.",
|
|
1374
|
+
"CHANGE_AGENT$SWITCH_AFTER_CONVERSATION": "대화가 시작된 후에 에이전트 모드를 전환할 수 있습니다",
|
|
1367
1375
|
"PLANNING_AGENTT$PLANNING_AGENT_INITIALIZED": "계획 에이전트가 초기화되었습니다",
|
|
1368
1376
|
"OBSERVATION_MESSAGE$SKILL_READY": "스킬 준비 완료",
|
|
1369
1377
|
"ACCOUNT_SETTINGS$SETTINGS": "설정",
|
|
@@ -513,6 +513,7 @@
|
|
|
513
513
|
"CHAT_INTERFACE$DISCONNECTED": "Frakoblet",
|
|
514
514
|
"CHAT_INTERFACE$COMMANDS": "Kommandoer",
|
|
515
515
|
"MODEL$AVAILABLE_PROFILES": "Tilgjengelige profiler ({{count}})",
|
|
516
|
+
"MODEL$AVAILABLE_MODELS": "Tilgjengelige modeller",
|
|
516
517
|
"MODEL$NO_SAVED_PROFILES": "Ingen lagrede profiler",
|
|
517
518
|
"MODEL$NO_PROFILES_HINT": "Bruk LLM-innstillingssiden til å opprette en profil, og kjør deretter /model <name> for å bytte.",
|
|
518
519
|
"MODEL$SWITCHED_TO_PROFILE": "Byttet til profilen <cmd>{{name}}</cmd>",
|
|
@@ -806,6 +807,12 @@
|
|
|
806
807
|
"MCP$VIEW_DOCS": "Vis dokumentasjon →",
|
|
807
808
|
"MCP$INSTALL_SUCCESS": "MCP-server lagret.",
|
|
808
809
|
"MCP$REMOVE_SUCCESS": "MCP-server fjernet.",
|
|
810
|
+
"MCP$TEST_BUTTON": "Test tilkobling",
|
|
811
|
+
"MCP$VERIFYING": "Verifiserer…",
|
|
812
|
+
"MCP$TEST_SUCCESS": "Tilkoblet — {{count}} verktøy tilgjengelig",
|
|
813
|
+
"MCP$TEST_ERROR_TIMEOUT": "Tilkoblingen ble tidsavbrutt. Sjekk URL-en og prøv igjen.",
|
|
814
|
+
"MCP$TEST_ERROR_CONNECTION": "Kunne ikke nå serveren. Sjekk URL-en og servertypen.",
|
|
815
|
+
"MCP$TEST_ERROR_UNKNOWN": "Tilkobling mislyktes: {{-error}}",
|
|
809
816
|
"MCP$EDIT_SERVER_ARIA": "Rediger {{name}}",
|
|
810
817
|
"MCP$DELETE_SERVER_ARIA": "Fjern {{name}}",
|
|
811
818
|
"MCP$ERROR_FIELD_REQUIRED": "Dette feltet er påkrevd.",
|
|
@@ -1364,6 +1371,7 @@
|
|
|
1364
1371
|
"COMMON$LET_S_WORK_ON_A_PLAN": "La oss lage en plan",
|
|
1365
1372
|
"COMMON$CODE_AGENT_DESCRIPTION": "Skriv, rediger og feilsøk med AI-assistanse i sanntid.",
|
|
1366
1373
|
"COMMON$PLAN_AGENT_DESCRIPTION": "Skisser mål, strukturer oppgaver og planlegg dine neste steg.",
|
|
1374
|
+
"CHANGE_AGENT$SWITCH_AFTER_CONVERSATION": "Agentmodus kan byttes etter at en samtale har startet",
|
|
1367
1375
|
"PLANNING_AGENTT$PLANNING_AGENT_INITIALIZED": "Planleggingsagent er initialisert",
|
|
1368
1376
|
"OBSERVATION_MESSAGE$SKILL_READY": "Ferdighet klar",
|
|
1369
1377
|
"ACCOUNT_SETTINGS$SETTINGS": "Innstillinger",
|
|
@@ -513,6 +513,7 @@
|
|
|
513
513
|
"CHAT_INTERFACE$DISCONNECTED": "Desconectado",
|
|
514
514
|
"CHAT_INTERFACE$COMMANDS": "Comandos",
|
|
515
515
|
"MODEL$AVAILABLE_PROFILES": "Perfis disponíveis ({{count}})",
|
|
516
|
+
"MODEL$AVAILABLE_MODELS": "Modelos disponíveis",
|
|
516
517
|
"MODEL$NO_SAVED_PROFILES": "Nenhum perfil salvo",
|
|
517
518
|
"MODEL$NO_PROFILES_HINT": "Use a página de configurações de LLM para criar um perfil e execute /model <name> para trocar.",
|
|
518
519
|
"MODEL$SWITCHED_TO_PROFILE": "Alternado para o perfil <cmd>{{name}}</cmd>",
|
|
@@ -806,6 +807,12 @@
|
|
|
806
807
|
"MCP$VIEW_DOCS": "Ver documentação →",
|
|
807
808
|
"MCP$INSTALL_SUCCESS": "Servidor MCP salvo.",
|
|
808
809
|
"MCP$REMOVE_SUCCESS": "Servidor MCP removido.",
|
|
810
|
+
"MCP$TEST_BUTTON": "Testar conexão",
|
|
811
|
+
"MCP$VERIFYING": "Verificando…",
|
|
812
|
+
"MCP$TEST_SUCCESS": "Conectado — {{count}} ferramenta(s) disponível(is)",
|
|
813
|
+
"MCP$TEST_ERROR_TIMEOUT": "Tempo limite de conexão esgotado. Verifique a URL e tente novamente.",
|
|
814
|
+
"MCP$TEST_ERROR_CONNECTION": "Não foi possível acessar o servidor. Verifique a URL e o tipo de servidor.",
|
|
815
|
+
"MCP$TEST_ERROR_UNKNOWN": "Falha na conexão: {{-error}}",
|
|
809
816
|
"MCP$EDIT_SERVER_ARIA": "Editar {{name}}",
|
|
810
817
|
"MCP$DELETE_SERVER_ARIA": "Remover {{name}}",
|
|
811
818
|
"MCP$ERROR_FIELD_REQUIRED": "Este campo é obrigatório.",
|
|
@@ -1364,6 +1371,7 @@
|
|
|
1364
1371
|
"COMMON$LET_S_WORK_ON_A_PLAN": "Vamos trabalhar em um plano",
|
|
1365
1372
|
"COMMON$CODE_AGENT_DESCRIPTION": "Escreva, edite e depure com assistência de IA em tempo real.",
|
|
1366
1373
|
"COMMON$PLAN_AGENT_DESCRIPTION": "Esboce objetivos, estruture tarefas e trace seus próximos passos.",
|
|
1374
|
+
"CHANGE_AGENT$SWITCH_AFTER_CONVERSATION": "O modo do agente pode ser alterado após o início de uma conversa",
|
|
1367
1375
|
"PLANNING_AGENTT$PLANNING_AGENT_INITIALIZED": "Agente de planejamento inicializado",
|
|
1368
1376
|
"OBSERVATION_MESSAGE$SKILL_READY": "Habilidade pronta",
|
|
1369
1377
|
"ACCOUNT_SETTINGS$SETTINGS": "Configurações",
|
|
@@ -513,6 +513,7 @@
|
|
|
513
513
|
"CHAT_INTERFACE$DISCONNECTED": "Bağlantı kesildi",
|
|
514
514
|
"CHAT_INTERFACE$COMMANDS": "Komutlar",
|
|
515
515
|
"MODEL$AVAILABLE_PROFILES": "Kullanılabilir profiller ({{count}})",
|
|
516
|
+
"MODEL$AVAILABLE_MODELS": "Kullanılabilir modeller",
|
|
516
517
|
"MODEL$NO_SAVED_PROFILES": "Kaydedilmiş profil yok",
|
|
517
518
|
"MODEL$NO_PROFILES_HINT": "Bir profil oluşturmak için LLM ayarları sayfasını kullanın, ardından geçiş yapmak için /model <name> çalıştırın.",
|
|
518
519
|
"MODEL$SWITCHED_TO_PROFILE": "<cmd>{{name}}</cmd> profiline geçildi",
|
|
@@ -806,6 +807,12 @@
|
|
|
806
807
|
"MCP$VIEW_DOCS": "Belgeleri görüntüle →",
|
|
807
808
|
"MCP$INSTALL_SUCCESS": "MCP sunucusu kaydedildi.",
|
|
808
809
|
"MCP$REMOVE_SUCCESS": "MCP sunucusu kaldırıldı.",
|
|
810
|
+
"MCP$TEST_BUTTON": "Bağlantıyı test et",
|
|
811
|
+
"MCP$VERIFYING": "Doğrulanıyor…",
|
|
812
|
+
"MCP$TEST_SUCCESS": "Bağlandı — {{count}} araç kullanılabilir",
|
|
813
|
+
"MCP$TEST_ERROR_TIMEOUT": "Bağlantı zaman aşımına uğradı. URL'yi kontrol edip tekrar deneyin.",
|
|
814
|
+
"MCP$TEST_ERROR_CONNECTION": "Sunucuya ulaşılamadı. URL'yi ve sunucu türünü kontrol edin.",
|
|
815
|
+
"MCP$TEST_ERROR_UNKNOWN": "Bağlantı başarısız: {{-error}}",
|
|
809
816
|
"MCP$EDIT_SERVER_ARIA": "{{name}} öğesini düzenle",
|
|
810
817
|
"MCP$DELETE_SERVER_ARIA": "{{name}} öğesini kaldır",
|
|
811
818
|
"MCP$ERROR_FIELD_REQUIRED": "Bu alan zorunludur.",
|
|
@@ -1364,6 +1371,7 @@
|
|
|
1364
1371
|
"COMMON$LET_S_WORK_ON_A_PLAN": "Bir plan üzerinde çalışalım",
|
|
1365
1372
|
"COMMON$CODE_AGENT_DESCRIPTION": "AI desteğiyle gerçek zamanlı olarak yazın, düzenleyin ve hata ayıklayın.",
|
|
1366
1373
|
"COMMON$PLAN_AGENT_DESCRIPTION": "Hedefleri belirtin, görevleri yapılandırın ve sonraki adımlarınızı belirleyin.",
|
|
1374
|
+
"CHANGE_AGENT$SWITCH_AFTER_CONVERSATION": "Aracı modu, bir konuşma başladıktan sonra değiştirilebilir",
|
|
1367
1375
|
"PLANNING_AGENTT$PLANNING_AGENT_INITIALIZED": "Planlama ajanı başlatıldı",
|
|
1368
1376
|
"OBSERVATION_MESSAGE$SKILL_READY": "Yetenek hazır",
|
|
1369
1377
|
"ACCOUNT_SETTINGS$SETTINGS": "Ayarlar",
|
|
@@ -513,6 +513,7 @@
|
|
|
513
513
|
"CHAT_INTERFACE$DISCONNECTED": "Від'єднано",
|
|
514
514
|
"CHAT_INTERFACE$COMMANDS": "Команди",
|
|
515
515
|
"MODEL$AVAILABLE_PROFILES": "Доступні профілі ({{count}})",
|
|
516
|
+
"MODEL$AVAILABLE_MODELS": "Доступні моделі",
|
|
516
517
|
"MODEL$NO_SAVED_PROFILES": "Немає збережених профілів",
|
|
517
518
|
"MODEL$NO_PROFILES_HINT": "Скористайтеся сторінкою налаштувань LLM, щоб створити профіль, а потім виконайте /model <name> для перемикання.",
|
|
518
519
|
"MODEL$SWITCHED_TO_PROFILE": "Перемкнено на профіль <cmd>{{name}}</cmd>",
|
|
@@ -806,6 +807,12 @@
|
|
|
806
807
|
"MCP$VIEW_DOCS": "Переглянути документацію →",
|
|
807
808
|
"MCP$INSTALL_SUCCESS": "Сервер MCP збережено.",
|
|
808
809
|
"MCP$REMOVE_SUCCESS": "Сервер MCP видалено.",
|
|
810
|
+
"MCP$TEST_BUTTON": "Перевірити з'єднання",
|
|
811
|
+
"MCP$VERIFYING": "Перевірка…",
|
|
812
|
+
"MCP$TEST_SUCCESS": "З'єднано — доступно інструментів: {{count}}",
|
|
813
|
+
"MCP$TEST_ERROR_TIMEOUT": "Час очікування з'єднання вичерпано. Перевірте URL-адресу та спробуйте ще раз.",
|
|
814
|
+
"MCP$TEST_ERROR_CONNECTION": "Не вдалося підключитися до сервера. Перевірте URL-адресу та тип сервера.",
|
|
815
|
+
"MCP$TEST_ERROR_UNKNOWN": "Помилка з'єднання: {{-error}}",
|
|
809
816
|
"MCP$EDIT_SERVER_ARIA": "Редагувати {{name}}",
|
|
810
817
|
"MCP$DELETE_SERVER_ARIA": "Видалити {{name}}",
|
|
811
818
|
"MCP$ERROR_FIELD_REQUIRED": "Це поле обов'язкове.",
|
|
@@ -1364,6 +1371,7 @@
|
|
|
1364
1371
|
"COMMON$LET_S_WORK_ON_A_PLAN": "Давайте розробимо план",
|
|
1365
1372
|
"COMMON$CODE_AGENT_DESCRIPTION": "Пишіть, редагуйте та налагоджуйте з підтримкою ШІ у реальному часі.",
|
|
1366
1373
|
"COMMON$PLAN_AGENT_DESCRIPTION": "Окресліть цілі, структуруйте завдання та сплануйте наступні кроки.",
|
|
1374
|
+
"CHANGE_AGENT$SWITCH_AFTER_CONVERSATION": "Режим агента можна змінити після початку розмови",
|
|
1367
1375
|
"PLANNING_AGENTT$PLANNING_AGENT_INITIALIZED": "Агент планування ініціалізовано",
|
|
1368
1376
|
"OBSERVATION_MESSAGE$SKILL_READY": "Навичка готова",
|
|
1369
1377
|
"ACCOUNT_SETTINGS$SETTINGS": "Налаштування",
|
|
@@ -513,6 +513,7 @@
|
|
|
513
513
|
"CHAT_INTERFACE$DISCONNECTED": "已断开连接",
|
|
514
514
|
"CHAT_INTERFACE$COMMANDS": "命令",
|
|
515
515
|
"MODEL$AVAILABLE_PROFILES": "可用配置文件 ({{count}})",
|
|
516
|
+
"MODEL$AVAILABLE_MODELS": "可用模型",
|
|
516
517
|
"MODEL$NO_SAVED_PROFILES": "没有已保存的配置文件",
|
|
517
518
|
"MODEL$NO_PROFILES_HINT": "使用 LLM 设置页面创建配置文件,然后运行 /model <name> 进行切换。",
|
|
518
519
|
"MODEL$SWITCHED_TO_PROFILE": "已切换到配置文件 <cmd>{{name}}</cmd>",
|
|
@@ -806,6 +807,12 @@
|
|
|
806
807
|
"MCP$VIEW_DOCS": "查看文档 →",
|
|
807
808
|
"MCP$INSTALL_SUCCESS": "MCP 服务器已保存。",
|
|
808
809
|
"MCP$REMOVE_SUCCESS": "MCP 服务器已移除。",
|
|
810
|
+
"MCP$TEST_BUTTON": "测试连接",
|
|
811
|
+
"MCP$VERIFYING": "正在验证…",
|
|
812
|
+
"MCP$TEST_SUCCESS": "已连接 — 有 {{count}} 个工具可用",
|
|
813
|
+
"MCP$TEST_ERROR_TIMEOUT": "连接超时。请检查 URL 后重试。",
|
|
814
|
+
"MCP$TEST_ERROR_CONNECTION": "无法连接到服务器。请检查 URL 和服务器类型。",
|
|
815
|
+
"MCP$TEST_ERROR_UNKNOWN": "连接失败:{{-error}}",
|
|
809
816
|
"MCP$EDIT_SERVER_ARIA": "编辑 {{name}}",
|
|
810
817
|
"MCP$DELETE_SERVER_ARIA": "移除 {{name}}",
|
|
811
818
|
"MCP$ERROR_FIELD_REQUIRED": "此字段为必填项。",
|
|
@@ -1364,6 +1371,7 @@
|
|
|
1364
1371
|
"COMMON$LET_S_WORK_ON_A_PLAN": "让我们制定一个计划吧",
|
|
1365
1372
|
"COMMON$CODE_AGENT_DESCRIPTION": "实时在 AI 协助下编写、编辑和调试。",
|
|
1366
1373
|
"COMMON$PLAN_AGENT_DESCRIPTION": "概述目标、结构化任务,并规划下一步。",
|
|
1374
|
+
"CHANGE_AGENT$SWITCH_AFTER_CONVERSATION": "对话开始后即可切换智能体模式",
|
|
1367
1375
|
"PLANNING_AGENTT$PLANNING_AGENT_INITIALIZED": "规划代理已初始化",
|
|
1368
1376
|
"OBSERVATION_MESSAGE$SKILL_READY": "技能已就绪",
|
|
1369
1377
|
"ACCOUNT_SETTINGS$SETTINGS": "设置",
|
|
@@ -513,6 +513,7 @@
|
|
|
513
513
|
"CHAT_INTERFACE$DISCONNECTED": "已斷開連接",
|
|
514
514
|
"CHAT_INTERFACE$COMMANDS": "命令",
|
|
515
515
|
"MODEL$AVAILABLE_PROFILES": "可用設定檔 ({{count}})",
|
|
516
|
+
"MODEL$AVAILABLE_MODELS": "可用模型",
|
|
516
517
|
"MODEL$NO_SAVED_PROFILES": "沒有已儲存的設定檔",
|
|
517
518
|
"MODEL$NO_PROFILES_HINT": "使用 LLM 設定頁面建立設定檔,然後執行 /model <name> 進行切換。",
|
|
518
519
|
"MODEL$SWITCHED_TO_PROFILE": "已切換到設定檔 <cmd>{{name}}</cmd>",
|
|
@@ -806,6 +807,12 @@
|
|
|
806
807
|
"MCP$VIEW_DOCS": "檢視文件 →",
|
|
807
808
|
"MCP$INSTALL_SUCCESS": "MCP 伺服器已儲存。",
|
|
808
809
|
"MCP$REMOVE_SUCCESS": "MCP 伺服器已移除。",
|
|
810
|
+
"MCP$TEST_BUTTON": "測試連線",
|
|
811
|
+
"MCP$VERIFYING": "正在驗證…",
|
|
812
|
+
"MCP$TEST_SUCCESS": "已連線 — 有 {{count}} 個工具可用",
|
|
813
|
+
"MCP$TEST_ERROR_TIMEOUT": "連線逾時。請檢查 URL 後重試。",
|
|
814
|
+
"MCP$TEST_ERROR_CONNECTION": "無法連線到伺服器。請檢查 URL 和伺服器類型。",
|
|
815
|
+
"MCP$TEST_ERROR_UNKNOWN": "連線失敗:{{-error}}",
|
|
809
816
|
"MCP$EDIT_SERVER_ARIA": "編輯 {{name}}",
|
|
810
817
|
"MCP$DELETE_SERVER_ARIA": "移除 {{name}}",
|
|
811
818
|
"MCP$ERROR_FIELD_REQUIRED": "此欄位為必填。",
|
|
@@ -1364,6 +1371,7 @@
|
|
|
1364
1371
|
"COMMON$LET_S_WORK_ON_A_PLAN": "讓我們來制定計劃吧",
|
|
1365
1372
|
"COMMON$CODE_AGENT_DESCRIPTION": "即時在 AI 協助下編寫、編輯和除錯。",
|
|
1366
1373
|
"COMMON$PLAN_AGENT_DESCRIPTION": "概述目標、結構化任務,並規劃下一步。",
|
|
1374
|
+
"CHANGE_AGENT$SWITCH_AFTER_CONVERSATION": "對話開始後即可切換代理模式",
|
|
1367
1375
|
"PLANNING_AGENTT$PLANNING_AGENT_INITIALIZED": "規劃代理已初始化",
|
|
1368
1376
|
"OBSERVATION_MESSAGE$SKILL_READY": "技能已就緒",
|
|
1369
1377
|
"ACCOUNT_SETTINGS$SETTINGS": "設定",
|
package/config/defaults.json
CHANGED
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
"_comment": "Single source of truth for version pins, ports, paths, and defaults shared across the npm and Docker install paths. Read by scripts/dev-safe.mjs, scripts/dev-with-automation.mjs, docker/entrypoint.sh (via generated defaults.env), and .github/workflows/docker.yml.",
|
|
3
3
|
|
|
4
4
|
"versions": {
|
|
5
|
-
"agentServer": "1.
|
|
5
|
+
"agentServer": "1.24.0",
|
|
6
6
|
"automation": "1.0.0a5",
|
|
7
7
|
"automationSdk": "1.22.1"
|
|
8
8
|
},
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
require(`../_virtual/_rolldown/runtime.cjs`);var e=`openhands-agent-server-config`,t=`workspace/project`;function n(){if(typeof window>`u`)return{};try{let t=window.localStorage.getItem(e);return t?JSON.parse(t)??{}:{}}catch{return{}}}function r(e){return e?.trim()||null}function i(e){if(!e)return null;let t=e.trim().replace(/\/$/,``);return t?/^https?:\/\//i.test(t)?t:typeof window<`u`?`${window.location.protocol}//${t}`:`http://${t}`:null}function a(){return i(n().baseUrl)||i(void 0)}function o(){return r(n().sessionApiKey)||r(void 0)}function s(e){if(typeof window>`u`)return!1;try{let t=new URL(e),n=new Set([`127.0.0.1`,`localhost`,`0.0.0.0`]),r=window.location.hostname;return n.has(t.hostname)&&!n.has(r)}catch{return!1}}function c(e){return e?s(e)?window.location.origin:e:null}function l(){return c(a())||(typeof window<`u`?window.location.origin:`http://127.0.0.1:8000`)}function u(){return o()}function d(){return(void 0)?.trim()||n().workingDir?.trim()||t}function f(e){return`${d().replace(/\/+$/,``)}/${e.replace(/-/g,``)}`}function p(){let e=u();return e?{"X-Session-API-Key":e}:{}}function m(){return!
|
|
1
|
+
require(`../_virtual/_rolldown/runtime.cjs`);var e=`openhands-agent-server-config`,t=`workspace/project`;function n(){if(typeof window>`u`)return{};try{let t=window.localStorage.getItem(e);return t?JSON.parse(t)??{}:{}}catch{return{}}}function r(e){return e?.trim()||null}function i(e){if(!e)return null;let t=e.trim().replace(/\/$/,``);return t?/^https?:\/\//i.test(t)?t:typeof window<`u`?`${window.location.protocol}//${t}`:`http://${t}`:null}function a(){return i(n().baseUrl)||i(void 0)}function o(){return r(n().sessionApiKey)||r(void 0)}function s(e){if(typeof window>`u`)return!1;try{let t=new URL(e),n=new Set([`127.0.0.1`,`localhost`,`0.0.0.0`]),r=window.location.hostname;return n.has(t.hostname)&&!n.has(r)}catch{return!1}}function c(e){return e?s(e)?window.location.origin:e:null}function l(){return c(a())||(typeof window<`u`?window.location.origin:`http://127.0.0.1:8000`)}function u(){return o()}function d(){return(void 0)?.trim()||n().workingDir?.trim()||t}function f(e){return`${d().replace(/\/+$/,``)}/${e.replace(/-/g,``)}`}function p(){let e=u();return e?{"X-Session-API-Key":e}:{}}function m(){return!0}exports.DEFAULT_WORKING_DIR=t,exports.buildConversationWorkingDir=f,exports.getAgentServerBaseUrl=l,exports.getAgentServerHeaders=p,exports.getAgentServerSessionApiKey=u,exports.getAgentServerWorkingDir=d,exports.shouldLoadPublicSkills=m;
|
|
2
2
|
//# sourceMappingURL=agent-server-config.cjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"agent-server-config.cjs","names":[],"sources":["../../src/api/agent-server-config.ts"],"sourcesContent":["export const AGENT_SERVER_CONFIG_STORAGE_KEY = \"openhands-agent-server-config\";\nexport const DEFAULT_WORKING_DIR = \"workspace/project\";\n\ninterface StoredAgentServerConfig {\n baseUrl?: string | null;\n sessionApiKey?: string | null;\n workingDir?: string | null;\n}\n\nexport interface AgentServerFormDefaults {\n baseUrl: string;\n sessionApiKey: string;\n}\n\nfunction readStoredConfig(): StoredAgentServerConfig {\n if (typeof window === \"undefined\") return {};\n\n try {\n const raw = window.localStorage.getItem(AGENT_SERVER_CONFIG_STORAGE_KEY);\n if (!raw) return {};\n const parsed = JSON.parse(raw) as StoredAgentServerConfig;\n return parsed ?? {};\n } catch {\n return {};\n }\n}\n\nfunction writeStoredConfig(config: StoredAgentServerConfig): void {\n if (typeof window === \"undefined\") return;\n\n const nextConfig = Object.fromEntries(\n Object.entries(config).flatMap(([key, value]) => {\n if (typeof value !== \"string\") return [];\n\n const trimmed = value.trim();\n if (!trimmed) return [];\n\n return [[key, trimmed]];\n }),\n ) as StoredAgentServerConfig;\n\n if (Object.keys(nextConfig).length === 0) {\n window.localStorage.removeItem(AGENT_SERVER_CONFIG_STORAGE_KEY);\n return;\n }\n\n window.localStorage.setItem(\n AGENT_SERVER_CONFIG_STORAGE_KEY,\n JSON.stringify(nextConfig),\n );\n}\n\nfunction trimToNull(value?: string | null): string | null {\n return value?.trim() || null;\n}\n\nfunction normalizeBaseUrl(value?: string | null): string | null {\n if (!value) return null;\n\n const trimmed = value.trim().replace(/\\/$/, \"\");\n if (!trimmed) return null;\n\n if (/^https?:\\/\\//i.test(trimmed)) {\n return trimmed;\n }\n\n if (typeof window !== \"undefined\") {\n return `${window.location.protocol}//${trimmed}`;\n }\n\n return `http://${trimmed}`;\n}\n\nfunction getConfiguredBaseUrl(): string | null {\n const storedUrl = normalizeBaseUrl(readStoredConfig().baseUrl);\n if (storedUrl) return storedUrl;\n\n return normalizeBaseUrl(import.meta.env.VITE_BACKEND_BASE_URL);\n}\n\nfunction getConfiguredSessionApiKey(): string | null {\n const storedKey = trimToNull(readStoredConfig().sessionApiKey);\n if (storedKey) return storedKey;\n\n return trimToNull(import.meta.env.VITE_SESSION_API_KEY);\n}\n\nfunction shouldUseProxyOrigin(baseUrl: string): boolean {\n if (typeof window === \"undefined\") {\n return false;\n }\n\n try {\n const configuredUrl = new URL(baseUrl);\n const localHosts = new Set([\"127.0.0.1\", \"localhost\", \"0.0.0.0\"]);\n const browserHostname = window.location.hostname;\n\n return (\n localHosts.has(configuredUrl.hostname) && !localHosts.has(browserHostname)\n );\n } catch {\n return false;\n }\n}\n\nfunction resolveAgentServerBaseUrl(baseUrl: string | null): string | null {\n if (!baseUrl) {\n return null;\n }\n\n if (shouldUseProxyOrigin(baseUrl)) {\n return window.location.origin;\n }\n\n return baseUrl;\n}\n\nexport function getAgentServerFormDefaults(): AgentServerFormDefaults {\n return {\n baseUrl: getConfiguredBaseUrl() ?? \"\",\n sessionApiKey: getConfiguredSessionApiKey() ?? \"\",\n };\n}\n\nexport function saveAgentServerConfig(config: AgentServerFormDefaults): void {\n const currentConfig = readStoredConfig();\n\n writeStoredConfig({\n ...currentConfig,\n baseUrl: normalizeBaseUrl(config.baseUrl),\n sessionApiKey: trimToNull(config.sessionApiKey),\n });\n}\n\nexport function getAgentServerBaseUrl(): string {\n const configuredUrl = resolveAgentServerBaseUrl(getConfiguredBaseUrl());\n if (configuredUrl) return configuredUrl;\n\n if (typeof window !== \"undefined\") {\n return window.location.origin;\n }\n\n return \"http://127.0.0.1:8000\";\n}\n\nexport function getAgentServerSessionApiKey(): string | null {\n return getConfiguredSessionApiKey();\n}\n\nexport function getAgentServerWorkingDir(): string {\n const envDir = import.meta.env.VITE_WORKING_DIR?.trim();\n if (envDir) return envDir;\n\n const storedDir = readStoredConfig().workingDir?.trim();\n if (storedDir) return storedDir;\n\n return DEFAULT_WORKING_DIR;\n}\n\nexport function buildConversationWorkingDir(conversationId: string): string {\n const base = getAgentServerWorkingDir().replace(/\\/+$/, \"\");\n const hex = conversationId.replace(/-/g, \"\");\n return `${base}/${hex}`;\n}\n\nexport function getConfiguredWorkerUrls(): string[] {\n const raw = import.meta.env.VITE_WORKER_URLS?.trim();\n if (!raw) return [];\n\n return raw\n .split(\",\")\n .map((url: string) => normalizeBaseUrl(url))\n .filter((url: string | null): url is string => Boolean(url));\n}\n\nexport function getAgentServerHeaders(): Record<string, string> {\n const sessionApiKey = getAgentServerSessionApiKey();\n return sessionApiKey ? { \"X-Session-API-Key\": sessionApiKey } : {};\n}\n\n/**\n * Returns whether public skills from the OpenHands extensions marketplace\n * (https://github.com/OpenHands/extensions) should be loaded.\n *\n * Defaults to
|
|
1
|
+
{"version":3,"file":"agent-server-config.cjs","names":[],"sources":["../../src/api/agent-server-config.ts"],"sourcesContent":["export const AGENT_SERVER_CONFIG_STORAGE_KEY = \"openhands-agent-server-config\";\nexport const DEFAULT_WORKING_DIR = \"workspace/project\";\n\ninterface StoredAgentServerConfig {\n baseUrl?: string | null;\n sessionApiKey?: string | null;\n workingDir?: string | null;\n}\n\nexport interface AgentServerFormDefaults {\n baseUrl: string;\n sessionApiKey: string;\n}\n\nfunction readStoredConfig(): StoredAgentServerConfig {\n if (typeof window === \"undefined\") return {};\n\n try {\n const raw = window.localStorage.getItem(AGENT_SERVER_CONFIG_STORAGE_KEY);\n if (!raw) return {};\n const parsed = JSON.parse(raw) as StoredAgentServerConfig;\n return parsed ?? {};\n } catch {\n return {};\n }\n}\n\nfunction writeStoredConfig(config: StoredAgentServerConfig): void {\n if (typeof window === \"undefined\") return;\n\n const nextConfig = Object.fromEntries(\n Object.entries(config).flatMap(([key, value]) => {\n if (typeof value !== \"string\") return [];\n\n const trimmed = value.trim();\n if (!trimmed) return [];\n\n return [[key, trimmed]];\n }),\n ) as StoredAgentServerConfig;\n\n if (Object.keys(nextConfig).length === 0) {\n window.localStorage.removeItem(AGENT_SERVER_CONFIG_STORAGE_KEY);\n return;\n }\n\n window.localStorage.setItem(\n AGENT_SERVER_CONFIG_STORAGE_KEY,\n JSON.stringify(nextConfig),\n );\n}\n\nfunction trimToNull(value?: string | null): string | null {\n return value?.trim() || null;\n}\n\nfunction normalizeBaseUrl(value?: string | null): string | null {\n if (!value) return null;\n\n const trimmed = value.trim().replace(/\\/$/, \"\");\n if (!trimmed) return null;\n\n if (/^https?:\\/\\//i.test(trimmed)) {\n return trimmed;\n }\n\n if (typeof window !== \"undefined\") {\n return `${window.location.protocol}//${trimmed}`;\n }\n\n return `http://${trimmed}`;\n}\n\nfunction getConfiguredBaseUrl(): string | null {\n const storedUrl = normalizeBaseUrl(readStoredConfig().baseUrl);\n if (storedUrl) return storedUrl;\n\n return normalizeBaseUrl(import.meta.env.VITE_BACKEND_BASE_URL);\n}\n\nfunction getConfiguredSessionApiKey(): string | null {\n const storedKey = trimToNull(readStoredConfig().sessionApiKey);\n if (storedKey) return storedKey;\n\n return trimToNull(import.meta.env.VITE_SESSION_API_KEY);\n}\n\nfunction shouldUseProxyOrigin(baseUrl: string): boolean {\n if (typeof window === \"undefined\") {\n return false;\n }\n\n try {\n const configuredUrl = new URL(baseUrl);\n const localHosts = new Set([\"127.0.0.1\", \"localhost\", \"0.0.0.0\"]);\n const browserHostname = window.location.hostname;\n\n return (\n localHosts.has(configuredUrl.hostname) && !localHosts.has(browserHostname)\n );\n } catch {\n return false;\n }\n}\n\nfunction resolveAgentServerBaseUrl(baseUrl: string | null): string | null {\n if (!baseUrl) {\n return null;\n }\n\n if (shouldUseProxyOrigin(baseUrl)) {\n return window.location.origin;\n }\n\n return baseUrl;\n}\n\nexport function getAgentServerFormDefaults(): AgentServerFormDefaults {\n return {\n baseUrl: getConfiguredBaseUrl() ?? \"\",\n sessionApiKey: getConfiguredSessionApiKey() ?? \"\",\n };\n}\n\nexport function saveAgentServerConfig(config: AgentServerFormDefaults): void {\n const currentConfig = readStoredConfig();\n\n writeStoredConfig({\n ...currentConfig,\n baseUrl: normalizeBaseUrl(config.baseUrl),\n sessionApiKey: trimToNull(config.sessionApiKey),\n });\n}\n\nexport function getAgentServerBaseUrl(): string {\n const configuredUrl = resolveAgentServerBaseUrl(getConfiguredBaseUrl());\n if (configuredUrl) return configuredUrl;\n\n if (typeof window !== \"undefined\") {\n return window.location.origin;\n }\n\n return \"http://127.0.0.1:8000\";\n}\n\nexport function getAgentServerSessionApiKey(): string | null {\n return getConfiguredSessionApiKey();\n}\n\nexport function getAgentServerWorkingDir(): string {\n const envDir = import.meta.env.VITE_WORKING_DIR?.trim();\n if (envDir) return envDir;\n\n const storedDir = readStoredConfig().workingDir?.trim();\n if (storedDir) return storedDir;\n\n return DEFAULT_WORKING_DIR;\n}\n\nexport function buildConversationWorkingDir(conversationId: string): string {\n const base = getAgentServerWorkingDir().replace(/\\/+$/, \"\");\n const hex = conversationId.replace(/-/g, \"\");\n return `${base}/${hex}`;\n}\n\nexport function getConfiguredWorkerUrls(): string[] {\n const raw = import.meta.env.VITE_WORKER_URLS?.trim();\n if (!raw) return [];\n\n return raw\n .split(\",\")\n .map((url: string) => normalizeBaseUrl(url))\n .filter((url: string | null): url is string => Boolean(url));\n}\n\nexport function getAgentServerHeaders(): Record<string, string> {\n const sessionApiKey = getAgentServerSessionApiKey();\n return sessionApiKey ? { \"X-Session-API-Key\": sessionApiKey } : {};\n}\n\n/**\n * Returns whether public skills from the OpenHands extensions marketplace\n * (https://github.com/OpenHands/extensions) should be loaded.\n *\n * Defaults to true. Set VITE_LOAD_PUBLIC_SKILLS=false to disable.\n */\nexport function shouldLoadPublicSkills(): boolean {\n return import.meta.env.VITE_LOAD_PUBLIC_SKILLS !== \"false\";\n}\n"],"mappings":"6CAAA,IAAa,EAAkC,gCAClC,EAAsB,oBAanC,SAAS,GAA4C,CACnD,GAAI,OAAO,OAAW,IAAa,MAAO,EAAE,CAE5C,GAAI,CACF,IAAM,EAAM,OAAO,aAAa,QAAQ,EAAgC,CAGxE,OAFK,EACU,KAAK,MAAM,EACnB,EAAU,EAAE,CAFF,EAAE,MAGb,CACN,MAAO,EAAE,EA6Bb,SAAS,EAAW,EAAsC,CACxD,OAAO,GAAO,MAAM,EAAI,KAG1B,SAAS,EAAiB,EAAsC,CAC9D,GAAI,CAAC,EAAO,OAAO,KAEnB,IAAM,EAAU,EAAM,MAAM,CAAC,QAAQ,MAAO,GAAG,CAW/C,OAVK,EAED,gBAAgB,KAAK,EAAQ,CACxB,EAGL,OAAO,OAAW,IACb,GAAG,OAAO,SAAS,SAAS,IAAI,IAGlC,UAAU,IAVI,KAavB,SAAS,GAAsC,CAI7C,OAHkB,EAAiB,GAAkB,CAAC,QAClD,EAEG,EAAA,IAAA,GAAuD,CAGhE,SAAS,GAA4C,CAInD,OAHkB,EAAW,GAAkB,CAAC,cAC5C,EAEG,EAAA,IAAA,GAAgD,CAGzD,SAAS,EAAqB,EAA0B,CACtD,GAAI,OAAO,OAAW,IACpB,MAAO,GAGT,GAAI,CACF,IAAM,EAAgB,IAAI,IAAI,EAAQ,CAChC,EAAa,IAAI,IAAI,CAAC,YAAa,YAAa,UAAU,CAAC,CAC3D,EAAkB,OAAO,SAAS,SAExC,OACE,EAAW,IAAI,EAAc,SAAS,EAAI,CAAC,EAAW,IAAI,EAAgB,MAEtE,CACN,MAAO,IAIX,SAAS,EAA0B,EAAuC,CASxE,OARK,EAID,EAAqB,EAAQ,CACxB,OAAO,SAAS,OAGlB,EAPE,KA2BX,SAAgB,GAAgC,CAQ9C,OAPsB,EAA0B,GAAsB,CAClE,GAEA,OAAO,OAAW,IACb,OAAO,SAAS,OAGlB,yBAGT,SAAgB,GAA6C,CAC3D,OAAO,GAA4B,CAGrC,SAAgB,GAAmC,CAOjD,OANM,IAAA,KAA2C,MAAM,EAGrC,GAAkB,CAAC,YAAY,MAAM,EAGhD,EAGT,SAAgB,EAA4B,EAAgC,CAG1E,MAAO,GAFM,GAA0B,CAAC,QAAQ,OAAQ,GAE9C,CAAK,GADH,EAAe,QAAQ,KAAM,GACvB,GAapB,SAAgB,GAAgD,CAC9D,IAAM,EAAgB,GAA6B,CACnD,OAAO,EAAgB,CAAE,oBAAqB,EAAe,CAAG,EAAE,CASpE,SAAgB,GAAkC,CAChD,MAAO"}
|
|
@@ -16,6 +16,6 @@ export declare function getAgentServerHeaders(): Record<string, string>;
|
|
|
16
16
|
* Returns whether public skills from the OpenHands extensions marketplace
|
|
17
17
|
* (https://github.com/OpenHands/extensions) should be loaded.
|
|
18
18
|
*
|
|
19
|
-
* Defaults to
|
|
19
|
+
* Defaults to true. Set VITE_LOAD_PUBLIC_SKILLS=false to disable.
|
|
20
20
|
*/
|
|
21
21
|
export declare function shouldLoadPublicSkills(): boolean;
|
|
@@ -56,7 +56,7 @@ function p() {
|
|
|
56
56
|
return e ? { "X-Session-API-Key": e } : {};
|
|
57
57
|
}
|
|
58
58
|
function m() {
|
|
59
|
-
return !
|
|
59
|
+
return !0;
|
|
60
60
|
}
|
|
61
61
|
//#endregion
|
|
62
62
|
export { t as DEFAULT_WORKING_DIR, f as buildConversationWorkingDir, l as getAgentServerBaseUrl, p as getAgentServerHeaders, u as getAgentServerSessionApiKey, d as getAgentServerWorkingDir, m as shouldLoadPublicSkills };
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"agent-server-config.js","names":[],"sources":["../../src/api/agent-server-config.ts"],"sourcesContent":["export const AGENT_SERVER_CONFIG_STORAGE_KEY = \"openhands-agent-server-config\";\nexport const DEFAULT_WORKING_DIR = \"workspace/project\";\n\ninterface StoredAgentServerConfig {\n baseUrl?: string | null;\n sessionApiKey?: string | null;\n workingDir?: string | null;\n}\n\nexport interface AgentServerFormDefaults {\n baseUrl: string;\n sessionApiKey: string;\n}\n\nfunction readStoredConfig(): StoredAgentServerConfig {\n if (typeof window === \"undefined\") return {};\n\n try {\n const raw = window.localStorage.getItem(AGENT_SERVER_CONFIG_STORAGE_KEY);\n if (!raw) return {};\n const parsed = JSON.parse(raw) as StoredAgentServerConfig;\n return parsed ?? {};\n } catch {\n return {};\n }\n}\n\nfunction writeStoredConfig(config: StoredAgentServerConfig): void {\n if (typeof window === \"undefined\") return;\n\n const nextConfig = Object.fromEntries(\n Object.entries(config).flatMap(([key, value]) => {\n if (typeof value !== \"string\") return [];\n\n const trimmed = value.trim();\n if (!trimmed) return [];\n\n return [[key, trimmed]];\n }),\n ) as StoredAgentServerConfig;\n\n if (Object.keys(nextConfig).length === 0) {\n window.localStorage.removeItem(AGENT_SERVER_CONFIG_STORAGE_KEY);\n return;\n }\n\n window.localStorage.setItem(\n AGENT_SERVER_CONFIG_STORAGE_KEY,\n JSON.stringify(nextConfig),\n );\n}\n\nfunction trimToNull(value?: string | null): string | null {\n return value?.trim() || null;\n}\n\nfunction normalizeBaseUrl(value?: string | null): string | null {\n if (!value) return null;\n\n const trimmed = value.trim().replace(/\\/$/, \"\");\n if (!trimmed) return null;\n\n if (/^https?:\\/\\//i.test(trimmed)) {\n return trimmed;\n }\n\n if (typeof window !== \"undefined\") {\n return `${window.location.protocol}//${trimmed}`;\n }\n\n return `http://${trimmed}`;\n}\n\nfunction getConfiguredBaseUrl(): string | null {\n const storedUrl = normalizeBaseUrl(readStoredConfig().baseUrl);\n if (storedUrl) return storedUrl;\n\n return normalizeBaseUrl(import.meta.env.VITE_BACKEND_BASE_URL);\n}\n\nfunction getConfiguredSessionApiKey(): string | null {\n const storedKey = trimToNull(readStoredConfig().sessionApiKey);\n if (storedKey) return storedKey;\n\n return trimToNull(import.meta.env.VITE_SESSION_API_KEY);\n}\n\nfunction shouldUseProxyOrigin(baseUrl: string): boolean {\n if (typeof window === \"undefined\") {\n return false;\n }\n\n try {\n const configuredUrl = new URL(baseUrl);\n const localHosts = new Set([\"127.0.0.1\", \"localhost\", \"0.0.0.0\"]);\n const browserHostname = window.location.hostname;\n\n return (\n localHosts.has(configuredUrl.hostname) && !localHosts.has(browserHostname)\n );\n } catch {\n return false;\n }\n}\n\nfunction resolveAgentServerBaseUrl(baseUrl: string | null): string | null {\n if (!baseUrl) {\n return null;\n }\n\n if (shouldUseProxyOrigin(baseUrl)) {\n return window.location.origin;\n }\n\n return baseUrl;\n}\n\nexport function getAgentServerFormDefaults(): AgentServerFormDefaults {\n return {\n baseUrl: getConfiguredBaseUrl() ?? \"\",\n sessionApiKey: getConfiguredSessionApiKey() ?? \"\",\n };\n}\n\nexport function saveAgentServerConfig(config: AgentServerFormDefaults): void {\n const currentConfig = readStoredConfig();\n\n writeStoredConfig({\n ...currentConfig,\n baseUrl: normalizeBaseUrl(config.baseUrl),\n sessionApiKey: trimToNull(config.sessionApiKey),\n });\n}\n\nexport function getAgentServerBaseUrl(): string {\n const configuredUrl = resolveAgentServerBaseUrl(getConfiguredBaseUrl());\n if (configuredUrl) return configuredUrl;\n\n if (typeof window !== \"undefined\") {\n return window.location.origin;\n }\n\n return \"http://127.0.0.1:8000\";\n}\n\nexport function getAgentServerSessionApiKey(): string | null {\n return getConfiguredSessionApiKey();\n}\n\nexport function getAgentServerWorkingDir(): string {\n const envDir = import.meta.env.VITE_WORKING_DIR?.trim();\n if (envDir) return envDir;\n\n const storedDir = readStoredConfig().workingDir?.trim();\n if (storedDir) return storedDir;\n\n return DEFAULT_WORKING_DIR;\n}\n\nexport function buildConversationWorkingDir(conversationId: string): string {\n const base = getAgentServerWorkingDir().replace(/\\/+$/, \"\");\n const hex = conversationId.replace(/-/g, \"\");\n return `${base}/${hex}`;\n}\n\nexport function getConfiguredWorkerUrls(): string[] {\n const raw = import.meta.env.VITE_WORKER_URLS?.trim();\n if (!raw) return [];\n\n return raw\n .split(\",\")\n .map((url: string) => normalizeBaseUrl(url))\n .filter((url: string | null): url is string => Boolean(url));\n}\n\nexport function getAgentServerHeaders(): Record<string, string> {\n const sessionApiKey = getAgentServerSessionApiKey();\n return sessionApiKey ? { \"X-Session-API-Key\": sessionApiKey } : {};\n}\n\n/**\n * Returns whether public skills from the OpenHands extensions marketplace\n * (https://github.com/OpenHands/extensions) should be loaded.\n *\n * Defaults to
|
|
1
|
+
{"version":3,"file":"agent-server-config.js","names":[],"sources":["../../src/api/agent-server-config.ts"],"sourcesContent":["export const AGENT_SERVER_CONFIG_STORAGE_KEY = \"openhands-agent-server-config\";\nexport const DEFAULT_WORKING_DIR = \"workspace/project\";\n\ninterface StoredAgentServerConfig {\n baseUrl?: string | null;\n sessionApiKey?: string | null;\n workingDir?: string | null;\n}\n\nexport interface AgentServerFormDefaults {\n baseUrl: string;\n sessionApiKey: string;\n}\n\nfunction readStoredConfig(): StoredAgentServerConfig {\n if (typeof window === \"undefined\") return {};\n\n try {\n const raw = window.localStorage.getItem(AGENT_SERVER_CONFIG_STORAGE_KEY);\n if (!raw) return {};\n const parsed = JSON.parse(raw) as StoredAgentServerConfig;\n return parsed ?? {};\n } catch {\n return {};\n }\n}\n\nfunction writeStoredConfig(config: StoredAgentServerConfig): void {\n if (typeof window === \"undefined\") return;\n\n const nextConfig = Object.fromEntries(\n Object.entries(config).flatMap(([key, value]) => {\n if (typeof value !== \"string\") return [];\n\n const trimmed = value.trim();\n if (!trimmed) return [];\n\n return [[key, trimmed]];\n }),\n ) as StoredAgentServerConfig;\n\n if (Object.keys(nextConfig).length === 0) {\n window.localStorage.removeItem(AGENT_SERVER_CONFIG_STORAGE_KEY);\n return;\n }\n\n window.localStorage.setItem(\n AGENT_SERVER_CONFIG_STORAGE_KEY,\n JSON.stringify(nextConfig),\n );\n}\n\nfunction trimToNull(value?: string | null): string | null {\n return value?.trim() || null;\n}\n\nfunction normalizeBaseUrl(value?: string | null): string | null {\n if (!value) return null;\n\n const trimmed = value.trim().replace(/\\/$/, \"\");\n if (!trimmed) return null;\n\n if (/^https?:\\/\\//i.test(trimmed)) {\n return trimmed;\n }\n\n if (typeof window !== \"undefined\") {\n return `${window.location.protocol}//${trimmed}`;\n }\n\n return `http://${trimmed}`;\n}\n\nfunction getConfiguredBaseUrl(): string | null {\n const storedUrl = normalizeBaseUrl(readStoredConfig().baseUrl);\n if (storedUrl) return storedUrl;\n\n return normalizeBaseUrl(import.meta.env.VITE_BACKEND_BASE_URL);\n}\n\nfunction getConfiguredSessionApiKey(): string | null {\n const storedKey = trimToNull(readStoredConfig().sessionApiKey);\n if (storedKey) return storedKey;\n\n return trimToNull(import.meta.env.VITE_SESSION_API_KEY);\n}\n\nfunction shouldUseProxyOrigin(baseUrl: string): boolean {\n if (typeof window === \"undefined\") {\n return false;\n }\n\n try {\n const configuredUrl = new URL(baseUrl);\n const localHosts = new Set([\"127.0.0.1\", \"localhost\", \"0.0.0.0\"]);\n const browserHostname = window.location.hostname;\n\n return (\n localHosts.has(configuredUrl.hostname) && !localHosts.has(browserHostname)\n );\n } catch {\n return false;\n }\n}\n\nfunction resolveAgentServerBaseUrl(baseUrl: string | null): string | null {\n if (!baseUrl) {\n return null;\n }\n\n if (shouldUseProxyOrigin(baseUrl)) {\n return window.location.origin;\n }\n\n return baseUrl;\n}\n\nexport function getAgentServerFormDefaults(): AgentServerFormDefaults {\n return {\n baseUrl: getConfiguredBaseUrl() ?? \"\",\n sessionApiKey: getConfiguredSessionApiKey() ?? \"\",\n };\n}\n\nexport function saveAgentServerConfig(config: AgentServerFormDefaults): void {\n const currentConfig = readStoredConfig();\n\n writeStoredConfig({\n ...currentConfig,\n baseUrl: normalizeBaseUrl(config.baseUrl),\n sessionApiKey: trimToNull(config.sessionApiKey),\n });\n}\n\nexport function getAgentServerBaseUrl(): string {\n const configuredUrl = resolveAgentServerBaseUrl(getConfiguredBaseUrl());\n if (configuredUrl) return configuredUrl;\n\n if (typeof window !== \"undefined\") {\n return window.location.origin;\n }\n\n return \"http://127.0.0.1:8000\";\n}\n\nexport function getAgentServerSessionApiKey(): string | null {\n return getConfiguredSessionApiKey();\n}\n\nexport function getAgentServerWorkingDir(): string {\n const envDir = import.meta.env.VITE_WORKING_DIR?.trim();\n if (envDir) return envDir;\n\n const storedDir = readStoredConfig().workingDir?.trim();\n if (storedDir) return storedDir;\n\n return DEFAULT_WORKING_DIR;\n}\n\nexport function buildConversationWorkingDir(conversationId: string): string {\n const base = getAgentServerWorkingDir().replace(/\\/+$/, \"\");\n const hex = conversationId.replace(/-/g, \"\");\n return `${base}/${hex}`;\n}\n\nexport function getConfiguredWorkerUrls(): string[] {\n const raw = import.meta.env.VITE_WORKER_URLS?.trim();\n if (!raw) return [];\n\n return raw\n .split(\",\")\n .map((url: string) => normalizeBaseUrl(url))\n .filter((url: string | null): url is string => Boolean(url));\n}\n\nexport function getAgentServerHeaders(): Record<string, string> {\n const sessionApiKey = getAgentServerSessionApiKey();\n return sessionApiKey ? { \"X-Session-API-Key\": sessionApiKey } : {};\n}\n\n/**\n * Returns whether public skills from the OpenHands extensions marketplace\n * (https://github.com/OpenHands/extensions) should be loaded.\n *\n * Defaults to true. Set VITE_LOAD_PUBLIC_SKILLS=false to disable.\n */\nexport function shouldLoadPublicSkills(): boolean {\n return import.meta.env.VITE_LOAD_PUBLIC_SKILLS !== \"false\";\n}\n"],"mappings":";AAAA,IAAa,IAAkC,iCAClC,IAAsB;AAanC,SAAS,IAA4C;AACnD,KAAI,OAAO,SAAW,IAAa,QAAO,EAAE;AAE5C,KAAI;EACF,IAAM,IAAM,OAAO,aAAa,QAAQ,EAAgC;AAGxE,SAFK,IACU,KAAK,MAAM,EACnB,IAAU,EAAE,GAFF,EAAE;SAGb;AACN,SAAO,EAAE;;;AA6Bb,SAAS,EAAW,GAAsC;AACxD,QAAO,GAAO,MAAM,IAAI;;AAG1B,SAAS,EAAiB,GAAsC;AAC9D,KAAI,CAAC,EAAO,QAAO;CAEnB,IAAM,IAAU,EAAM,MAAM,CAAC,QAAQ,OAAO,GAAG;AAW/C,QAVK,IAED,gBAAgB,KAAK,EAAQ,GACxB,IAGL,OAAO,SAAW,MACb,GAAG,OAAO,SAAS,SAAS,IAAI,MAGlC,UAAU,MAVI;;AAavB,SAAS,IAAsC;AAI7C,QAHkB,EAAiB,GAAkB,CAAC,QAClD,IAEG,EAAA,KAAA,EAAuD;;AAGhE,SAAS,IAA4C;AAInD,QAHkB,EAAW,GAAkB,CAAC,cAC5C,IAEG,EAAA,KAAA,EAAgD;;AAGzD,SAAS,EAAqB,GAA0B;AACtD,KAAI,OAAO,SAAW,IACpB,QAAO;AAGT,KAAI;EACF,IAAM,IAAgB,IAAI,IAAI,EAAQ,EAChC,IAAa,IAAI,IAAI;GAAC;GAAa;GAAa;GAAU,CAAC,EAC3D,IAAkB,OAAO,SAAS;AAExC,SACE,EAAW,IAAI,EAAc,SAAS,IAAI,CAAC,EAAW,IAAI,EAAgB;SAEtE;AACN,SAAO;;;AAIX,SAAS,EAA0B,GAAuC;AASxE,QARK,IAID,EAAqB,EAAQ,GACxB,OAAO,SAAS,SAGlB,IAPE;;AA2BX,SAAgB,IAAgC;AAQ9C,QAPsB,EAA0B,GAAsB,CAClE,KAEA,OAAO,SAAW,MACb,OAAO,SAAS,SAGlB;;AAGT,SAAgB,IAA6C;AAC3D,QAAO,GAA4B;;AAGrC,SAAgB,IAAmC;AAOjD,SANM,KAAA,IAA2C,MAAM,IAGrC,GAAkB,CAAC,YAAY,MAAM,IAGhD;;AAGT,SAAgB,EAA4B,GAAgC;AAG1E,QAAO,GAFM,GAA0B,CAAC,QAAQ,QAAQ,GAE9C,CAAK,GADH,EAAe,QAAQ,MAAM,GACvB;;AAapB,SAAgB,IAAgD;CAC9D,IAAM,IAAgB,GAA6B;AACnD,QAAO,IAAgB,EAAE,qBAAqB,GAAe,GAAG,EAAE;;AASpE,SAAgB,IAAkC;AAChD,QAAO"}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
require(`../../_virtual/_rolldown/runtime.cjs`);const e=require(`../agent-server-config.cjs`),t=require(`../backend-registry/active-store.cjs`),n=require(`../../node_modules/@openhands/typescript-client/dist/client/conversation-client.cjs`),r=require(`../../node_modules/@openhands/typescript-client/dist/client/file-client.cjs`),i=require(`../../node_modules/@openhands/typescript-client/dist/client/profiles-client.cjs`),a=require(`../../node_modules/@openhands/typescript-client/dist/client/vscode-client.cjs`),o=require(`../../node_modules/uuid/dist/v4.cjs`),s=require(`../../utils/websocket-url.cjs`),c=require(`../cloud/proxy.cjs`),l=require(`../agent-server-client-options.cjs`),u=require(`../../node_modules/@openhands/typescript-client/dist/models/conversation.cjs`);require(`../../node_modules/@openhands/typescript-client/dist/index.cjs`);const d=require(`../conversation-metadata-store.cjs`),f=require(`../cloud/conversation-service.api.cjs`),p=require(`../settings-service/settings-service.api.cjs`),m=require(`../agent-server-adapter.cjs`);var h=`1970-01-01T00:00:00.000Z`,g=`Unable to load conversations because the selected agent server returned data this UI does not understand. Check the backend URL/session key and update the agent server if needed.`,_=`Unable to switch LLM profiles because the selected agent server returned profile data this UI does not understand. Check the backend URL/session key and update the agent server if needed.`;function v(){return Error(g)}function y(e){return typeof e==`object`&&!!e&&!Array.isArray(e)}function b(e){return y(e)&&typeof e.model==`string`}function x(e){return typeof e==`number`?e:null}function S(e){return typeof e==`number`?e:0}function C(e){return typeof e==`string`?e:null}function w(e,t,n){let r=e[t]??e[n];return typeof r==`string`&&r.trim()?r:h}function T(e){return y(e)?{prompt_tokens:S(e.prompt_tokens),completion_tokens:S(e.completion_tokens),cache_read_tokens:S(e.cache_read_tokens),cache_write_tokens:S(e.cache_write_tokens),context_window:S(e.context_window),per_turn_token:S(e.per_turn_token)}:null}function E(e){return y(e)?{accumulated_cost:x(e.accumulated_cost),max_budget_per_task:x(e.max_budget_per_task),accumulated_token_usage:T(e.accumulated_token_usage)}:null}function D(e){if(!y(e))return null;let t=y(e.llm)?{model:C(e.llm.model)}:null;return{kind:C(e.kind),acp_model:C(e.acp_model),llm:t}}function O(e){return y(e)?{working_dir:C(e.working_dir)}:null}function k(e){if(!y(e))return null;let t={};for(let[n,r]of Object.entries(e))typeof r==`string`&&(t[n]=r);return t}function A(e){if(!e.startsWith(`/`))return null;let t=[];for(let n of e.split(`/`))if(n&&n!==`.`)if(n===`..`){if(!t.length)return null;t.pop()}else t.push(n);return`/${t.join(`/`)}`}function j(e,t){let n=A(e),r=A(t);if(!n||!r||n!==r&&!n.startsWith(`${r}/`))throw Error(`Conversation file path must stay inside the workspace`);return n}function M(e){if(!y(e)||typeof e.id!=`string`||!e.id.trim())throw v();return{id:e.id.trim(),title:C(e.title),created_at:w(e,`created_at`,`createdAt`),updated_at:w(e,`updated_at`,`updatedAt`),execution_status:C(e.execution_status),sandbox_status:C(e.sandbox_status),metrics:E(e.metrics),agent:D(e.agent),workspace:O(e.workspace),tags:k(e.tags),current_model_id:C(e.current_model_id),current_model_name:C(e.current_model_name)}}function N(e){if(!Array.isArray(e))throw v();return e.map(M)}function P(e){if(Array.isArray(e))return{items:N(e),next_page_id:null};if(!y(e))throw v();return{items:N(e.items),next_page_id:typeof e.next_page_id==`string`?e.next_page_id:null}}var F=new Set([`idle`,`running`,`paused`,`waiting_for_confirmation`,`finished`,`error`,`stuck`]);function I(e){let t=e??`idle`;return F.has(t)?t:`idle`}function L(e,t){if(!e)throw Error(`Conversation ${t} was not found`);return e}var R=class{static async sendMessage(e,r,i){let a=t.getActiveBackend().backend,o=i?.conversationUrl??null,u=i?.sessionApiKey??null;if(a.kind===`cloud`){if(!o||!u){let[t]=await f.batchGetCloudConversations([e]);o=t?.conversation_url?.trim()??null,u=t?.session_api_key?.trim()??null}if(!o||!u)throw Error(`Conversation sandbox is still starting. Wait for it to finish, then try again.`);return await c.callCloudProxy({backend:a,method:`POST`,hostOverride:s.buildHttpBaseUrl(o),path:`/api/conversations/${e}/events`,body:{...r,run:!0},authMode:`session-api-key`,sessionApiKey:u}),r}return await new n.ConversationClient(l.getAgentServerClientOptions({conversationUrl:o,sessionApiKey:u})).sendEvent(e,r,{run:!0}),r}static async createConversation(r,i,a,s,c,u,h,g){if(t.getActiveBackend().backend.kind===`cloud`)return f.createCloudAppConversation({initial_message:r?{role:`user`,content:[{type:`text`,text:r}]}:null,title:i??null,selected_repository:s?.selected_repository??null,selected_branch:s?.selected_branch??null,git_provider:s?.git_provider??null,plugins:a??null,parent_conversation_id:u??null,agent_type:h,sandbox_id:g??null});let _=await p.default.getSettings(),v=o.default(),y=await m.buildStartConversationRequestWithEncryptedSettings({settings:_,query:r,conversationInstructions:i,plugins:a,conversationId:v,workingDir:c??e.buildConversationWorkingDir(v)}),b=await new n.ConversationClient(l.getAgentServerClientOptions()).createConversation(y);return(s?.selected_repository||c)&&d.setStoredConversationMetadata(b.id,{selected_repository:s?.selected_repository??null,selected_branch:s?.selected_branch??null,git_provider:s?.git_provider??null,selected_workspace:c??null}),{id:b.id,created_by_user_id:null,status:`READY`,detail:null,app_conversation_id:b.id,agent_server_url:t.getEffectiveLocalBackend().host,request:{initial_message:y.initial_message,plugins:a??null},created_at:b.created_at,updated_at:b.updated_at}}static async getStartTask(e){return t.getActiveBackend().backend.kind===`cloud`?f.getCloudAppConversationStartTask(e):null}static async getVSCodeUrl(e,t,n){let r=await this.resolveConversationWorkingDir(e);return{vscode_url:await new a.VSCodeClient(l.getAgentServerClientOptions({conversationUrl:t,sessionApiKey:n})).getUrl({baseUrl:typeof window<`u`?window.location.origin:void 0,workspaceDir:r})}}static async resolveConversationWorkingDir(t){let[n]=await this.batchGetAppConversations([t]);return n?.workspace?.working_dir??e.getAgentServerWorkingDir()}static async batchGetAppConversations(e){return e.length===0?[]:t.getActiveBackend().backend.kind===`cloud`?f.batchGetCloudConversations(e):N(await new n.ConversationClient(l.getAgentServerClientOptions()).getConversations(e)).map(e=>m.toAppConversation(e))}static async updateConversationPublicFlag(e,n){if(t.getActiveBackend().backend.kind!==`cloud`)throw Error(`Public sharing requires a cloud backend.`);return f.updateCloudConversationPublicFlag(e,n)}static async updateConversationRepository(e,t,n,r){t?d.setStoredConversationMetadata(e,{...d.getStoredConversationMetadata(e)??{},selected_repository:t,selected_branch:n??null,git_provider:r??null}):d.removeStoredConversationMetadata(e);let[i]=await this.batchGetAppConversations([e]);return L(i,e)}static async readConversationFile(e,n){if(t.getActiveBackend().backend.kind===`cloud`)return f.readCloudConversationFile(e,j(n??`/workspace/project/.agents_tmp/PLAN.md`,`/workspace/project`));let i=await this.resolveConversationWorkingDir(e),a=j(n??`${i}/.agents_tmp/PLAN.md`,i);return new r.FileClient(l.getAgentServerClientOptions()).downloadTextFile(a)}static async downloadConversation(e){return t.getActiveBackend().backend.kind===`cloud`?f.downloadCloudConversation(e):new r.FileClient(l.getAgentServerClientOptions()).downloadTrajectory(e)}static async getHooks(e){return m.emptyHooksResponse()}static async getRuntimeConversation(e,r,i){let a=t.getActiveBackend().backend,o=a.kind===`cloud`&&r?await c.callCloudProxy({backend:a,method:`GET`,hostOverride:s.buildHttpBaseUrl(r),path:`/api/conversations/${e}`,authMode:`session-api-key`,sessionApiKey:i}):await new n.ConversationClient(l.getAgentServerClientOptions({conversationUrl:r,sessionApiKey:i})).getConversation(e),u=M(o),d=y(o)?o.stats:null;return{id:u.id,title:u.title?.trim()?u.title:m.getDefaultConversationTitle(u.id),metrics:E(u.metrics),created_at:u.created_at,updated_at:u.updated_at,status:I(u.execution_status),stats:y(d)?d:{usage_to_metrics:{}}}}static async searchConversations(e=20,r){return t.getActiveBackend().backend.kind===`cloud`?f.searchCloudConversations(e,r):m.toConversationPage(P(await new n.ConversationClient(l.getAgentServerClientOptions()).searchConversations({limit:e,page_id:r,sort_order:u.ConversationSortOrder.UPDATED_AT_DESC})))}static async deleteConversation(e){t.getActiveBackend().backend.kind===`cloud`?await f.deleteCloudConversation(e):await new n.ConversationClient(l.getAgentServerClientOptions()).deleteConversation(e),d.removeStoredConversationMetadata(e)}static async updateConversationTitle(e,t){await new n.ConversationClient(l.getAgentServerClientOptions()).updateConversation(e,{title:t});let[r]=await this.batchGetAppConversations([e]);return L(r,e)}static async switchProfile(e,r){if(t.getActiveBackend().backend.kind===`cloud`)throw Error(`LLM profile switching is only supported for local agent-server backends.`);let a=new i.ProfilesClient(l.getAgentServerClientOptions());if(!e){await a.activateProfile(r);return}let o=await a.getProfile(r,{exposeSecrets:`encrypted`});if(!b(o.config))throw Error(_);await new n.ConversationClient(l.getAgentServerClientOptions()).switchLLM(e,o.config)}};exports.default=R;
|
|
1
|
+
require(`../../_virtual/_rolldown/runtime.cjs`);const e=require(`../agent-server-config.cjs`),t=require(`../backend-registry/active-store.cjs`),n=require(`../../node_modules/@openhands/typescript-client/dist/client/conversation-client.cjs`),r=require(`../../node_modules/@openhands/typescript-client/dist/client/file-client.cjs`),i=require(`../../node_modules/@openhands/typescript-client/dist/client/profiles-client.cjs`),a=require(`../../node_modules/@openhands/typescript-client/dist/client/vscode-client.cjs`),o=require(`../../node_modules/uuid/dist/v4.cjs`),s=require(`../../utils/websocket-url.cjs`),c=require(`../cloud/proxy.cjs`),l=require(`../agent-server-client-options.cjs`),u=require(`../../node_modules/@openhands/typescript-client/dist/models/conversation.cjs`);require(`../../node_modules/@openhands/typescript-client/dist/index.cjs`);const d=require(`../conversation-metadata-store.cjs`),f=require(`../cloud/conversation-service.api.cjs`),p=require(`../settings-service/settings-service.api.cjs`),m=require(`../agent-server-adapter.cjs`);var h=`1970-01-01T00:00:00.000Z`,g=`Unable to load conversations because the selected agent server returned data this UI does not understand. Check the backend URL/session key and update the agent server if needed.`,_=`Unable to switch LLM profiles because the selected agent server returned profile data this UI does not understand. Check the backend URL/session key and update the agent server if needed.`;function v(){return Error(g)}function y(e){return typeof e==`object`&&!!e&&!Array.isArray(e)}function b(e){return y(e)&&typeof e.model==`string`}function x(e){return typeof e==`number`?e:null}function S(e){return typeof e==`number`?e:0}function C(e){return typeof e==`string`?e:null}function w(e,t,n){let r=e[t]??e[n];return typeof r==`string`&&r.trim()?r:h}function T(e){return y(e)?{prompt_tokens:S(e.prompt_tokens),completion_tokens:S(e.completion_tokens),cache_read_tokens:S(e.cache_read_tokens),cache_write_tokens:S(e.cache_write_tokens),context_window:S(e.context_window),per_turn_token:S(e.per_turn_token)}:null}function E(e){return y(e)?{accumulated_cost:x(e.accumulated_cost),max_budget_per_task:x(e.max_budget_per_task),accumulated_token_usage:T(e.accumulated_token_usage)}:null}function D(e){if(!y(e))return null;let t=y(e.llm)?{model:C(e.llm.model)}:null;return{kind:C(e.kind),acp_model:C(e.acp_model),llm:t}}function O(e){return y(e)?{working_dir:C(e.working_dir)}:null}function k(e){if(!y(e))return null;let t={};for(let[n,r]of Object.entries(e))typeof r==`string`&&(t[n]=r);return t}function A(e){if(!e.startsWith(`/`))return null;let t=[];for(let n of e.split(`/`))if(n&&n!==`.`)if(n===`..`){if(!t.length)return null;t.pop()}else t.push(n);return`/${t.join(`/`)}`}function j(e,t){let n=A(e),r=A(t);if(!n||!r||n!==r&&!n.startsWith(`${r}/`))throw Error(`Conversation file path must stay inside the workspace`);return n}function M(e){if(!y(e)||typeof e.id!=`string`||!e.id.trim())throw v();return{id:e.id.trim(),title:C(e.title),created_at:w(e,`created_at`,`createdAt`),updated_at:w(e,`updated_at`,`updatedAt`),execution_status:C(e.execution_status),sandbox_status:C(e.sandbox_status),metrics:E(e.metrics),agent:D(e.agent),workspace:O(e.workspace),tags:k(e.tags),current_model_id:C(e.current_model_id),current_model_name:C(e.current_model_name)}}function N(e){if(!Array.isArray(e))throw v();return e.map(M)}function P(e){if(Array.isArray(e))return{items:N(e),next_page_id:null};if(!y(e))throw v();return{items:N(e.items),next_page_id:typeof e.next_page_id==`string`?e.next_page_id:null}}var F=new Set([`idle`,`running`,`paused`,`waiting_for_confirmation`,`finished`,`error`,`stuck`]);function I(e){let t=e??`idle`;return F.has(t)?t:`idle`}function L(e,t){if(!e)throw Error(`Conversation ${t} was not found`);return e}var R=class{static async sendMessage(e,r,i){let a=t.getActiveBackend().backend,o=i?.conversationUrl??null,u=i?.sessionApiKey??null;if(a.kind===`cloud`){if(!o||!u){let[t]=await f.batchGetCloudConversations([e]);o=t?.conversation_url?.trim()??null,u=t?.session_api_key?.trim()??null}if(!o||!u)throw Error(`Conversation sandbox is still starting. Wait for it to finish, then try again.`);return await c.callCloudProxy({backend:a,method:`POST`,hostOverride:s.buildHttpBaseUrl(o),path:`/api/conversations/${e}/events`,body:{...r,run:!0},authMode:`session-api-key`,sessionApiKey:u}),r}return await new n.ConversationClient(l.getAgentServerClientOptions({conversationUrl:o,sessionApiKey:u})).sendEvent(e,r,{run:!0}),r}static async createConversation(r,i,a,s,c,u,h,g){if(t.getActiveBackend().backend.kind===`cloud`)return f.createCloudAppConversation({initial_message:r?{role:`user`,content:[{type:`text`,text:r}]}:null,title:i??null,selected_repository:s?.selected_repository??null,selected_branch:s?.selected_branch??null,git_provider:s?.git_provider??null,plugins:a??null,parent_conversation_id:u??null,agent_type:h,sandbox_id:g??null});let _=await p.default.getSettings(),v=o.default(),y=await m.buildStartConversationRequestWithEncryptedSettings({settings:_,query:r,conversationInstructions:i,plugins:a,conversationId:v,workingDir:c??e.buildConversationWorkingDir(v)}),b=await new n.ConversationClient(l.getAgentServerClientOptions()).createConversation(y);return(s?.selected_repository||c)&&d.setStoredConversationMetadata(b.id,{selected_repository:s?.selected_repository??null,selected_branch:s?.selected_branch??null,git_provider:s?.git_provider??null,selected_workspace:c??null}),{id:b.id,created_by_user_id:null,status:`READY`,detail:null,app_conversation_id:b.id,agent_server_url:t.getEffectiveLocalBackend().host,request:{initial_message:y.initial_message,plugins:a??null},created_at:b.created_at,updated_at:b.updated_at}}static async getStartTask(e){return t.getActiveBackend().backend.kind===`cloud`?f.getCloudAppConversationStartTask(e):null}static async getVSCodeUrl(e,t,n){let r=await this.resolveConversationWorkingDir(e);return{vscode_url:await new a.VSCodeClient(l.getAgentServerClientOptions({conversationUrl:t,sessionApiKey:n})).getUrl({baseUrl:typeof window<`u`?window.location.origin:void 0,workspaceDir:r})}}static async resolveConversationWorkingDir(t){let[n]=await this.batchGetAppConversations([t]);return n?.workspace?.working_dir??e.getAgentServerWorkingDir()}static async batchGetAppConversations(e){return e.length===0?[]:t.getActiveBackend().backend.kind===`cloud`?f.batchGetCloudConversations(e):N(await new n.ConversationClient(l.getAgentServerClientOptions()).getConversations(e)).map(e=>m.toAppConversation(e))}static async updateConversationPublicFlag(e,n){if(t.getActiveBackend().backend.kind!==`cloud`)throw Error(`Public sharing requires a cloud backend.`);return f.updateCloudConversationPublicFlag(e,n)}static async updateConversationRepository(e,t,n,r){t?d.setStoredConversationMetadata(e,{...d.getStoredConversationMetadata(e)??{},selected_repository:t,selected_branch:n??null,git_provider:r??null}):d.removeStoredConversationMetadata(e);let[i]=await this.batchGetAppConversations([e]);return L(i,e)}static async readConversationFile(e,n){if(t.getActiveBackend().backend.kind===`cloud`)return f.readCloudConversationFile(e,j(n??`/workspace/project/.agents_tmp/PLAN.md`,`/workspace/project`));let i=await this.resolveConversationWorkingDir(e),a=j(n??`${i}/.agents_tmp/PLAN.md`,i);return new r.FileClient(l.getAgentServerClientOptions()).downloadTextFile(a)}static async downloadConversation(e){return t.getActiveBackend().backend.kind===`cloud`?f.downloadCloudConversation(e):new r.FileClient(l.getAgentServerClientOptions()).downloadTrajectory(e)}static async getHooks(e){return m.emptyHooksResponse()}static async getRuntimeConversation(e,r,i){let a=t.getActiveBackend().backend,o=a.kind===`cloud`&&r?await c.callCloudProxy({backend:a,method:`GET`,hostOverride:s.buildHttpBaseUrl(r),path:`/api/conversations/${e}`,authMode:`session-api-key`,sessionApiKey:i}):await new n.ConversationClient(l.getAgentServerClientOptions({conversationUrl:r,sessionApiKey:i})).getConversation(e),u=M(o),d=y(o)?o.stats:null;return{id:u.id,title:u.title?.trim()?u.title:m.getDefaultConversationTitle(u.id),metrics:E(u.metrics),created_at:u.created_at,updated_at:u.updated_at,status:I(u.execution_status),stats:y(d)?d:{usage_to_metrics:{}}}}static async searchConversations(e=20,r){return t.getActiveBackend().backend.kind===`cloud`?f.searchCloudConversations(e,r):m.toConversationPage(P(await new n.ConversationClient(l.getAgentServerClientOptions()).searchConversations({limit:e,page_id:r,sort_order:u.ConversationSortOrder.UPDATED_AT_DESC})))}static async deleteConversation(e){t.getActiveBackend().backend.kind===`cloud`?await f.deleteCloudConversation(e):await new n.ConversationClient(l.getAgentServerClientOptions()).deleteConversation(e),d.removeStoredConversationMetadata(e)}static async updateConversationTitle(e,t){await new n.ConversationClient(l.getAgentServerClientOptions()).updateConversation(e,{title:t});let[r]=await this.batchGetAppConversations([e]);return L(r,e)}static async switchProfile(e,r){if(t.getActiveBackend().backend.kind===`cloud`)throw Error(`LLM profile switching is only supported for local agent-server backends.`);let a=new i.ProfilesClient(l.getAgentServerClientOptions());if(!e){await a.activateProfile(r);return}let o=await a.getProfile(r,{exposeSecrets:`encrypted`});if(!b(o.config))throw Error(_);await new n.ConversationClient(l.getAgentServerClientOptions()).switchLLM(e,o.config)}static async switchAcpModel(e,r){if(t.getActiveBackend().backend.kind===`cloud`)throw Error(`ACP model switching is only supported for local agent-server backends.`);await new n.ConversationClient(l.getAgentServerClientOptions()).switchAcpModel(e,r)}};exports.default=R;
|
|
2
2
|
//# sourceMappingURL=agent-server-conversation-service.api.cjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"agent-server-conversation-service.api.cjs","names":[],"sources":["../../../src/api/conversation-service/agent-server-conversation-service.api.ts"],"sourcesContent":["import {\n ConversationSortOrder,\n type LLMConfig,\n} from \"@openhands/typescript-client\";\nimport {\n ConversationClient,\n FileClient,\n ProfilesClient,\n VSCodeClient,\n} from \"@openhands/typescript-client/clients\";\nimport { v4 as uuidv4 } from \"uuid\";\nimport { Provider } from \"#/types/settings\";\nimport type { ConversationRuntimeContext } from \"#/api/conversation-file-upload.api\";\nimport { buildHttpBaseUrl } from \"#/utils/websocket-url\";\nimport {\n buildConversationWorkingDir,\n getAgentServerWorkingDir,\n} from \"../agent-server-config\";\nimport {\n getActiveBackend,\n getEffectiveLocalBackend,\n} from \"../backend-registry/active-store\";\nimport { callCloudProxy } from \"../cloud/proxy\";\nimport {\n batchGetCloudConversations,\n createCloudAppConversation,\n deleteCloudConversation,\n downloadCloudConversation,\n getCloudAppConversationStartTask,\n readCloudConversationFile,\n searchCloudConversations,\n updateCloudConversationPublicFlag,\n} from \"../cloud/conversation-service.api\";\nimport {\n DirectConversationInfo,\n buildStartConversationRequestWithEncryptedSettings,\n emptyHooksResponse,\n getDefaultConversationTitle,\n toAppConversation,\n toConversationPage,\n} from \"../agent-server-adapter\";\nimport { GetVSCodeUrlResponse } from \"../open-hands.types\";\nimport { getAgentServerClientOptions } from \"../agent-server-client-options\";\nimport SettingsService from \"../settings-service/settings-service.api\";\nimport {\n ConversationMetadata,\n getStoredConversationMetadata,\n removeStoredConversationMetadata,\n setStoredConversationMetadata,\n} from \"../conversation-metadata-store\";\nimport type {\n GetHooksResponse,\n PluginSpec,\n AppConversation,\n AppConversationPage,\n AppConversationStartRequest,\n AppConversationStartTask,\n MetricsSnapshot,\n RuntimeConversationInfo,\n SendMessageRequest,\n SendMessageResponse,\n} from \"./agent-server-conversation-service.types\";\n\nconst DEFAULT_CONVERSATION_TIMESTAMP = \"1970-01-01T00:00:00.000Z\";\nconst INVALID_CONVERSATION_RESPONSE_MESSAGE =\n \"Unable to load conversations because the selected agent server returned \" +\n \"data this UI does not understand. Check the backend URL/session key and \" +\n \"update the agent server if needed.\";\nconst INVALID_PROFILE_CONFIG_MESSAGE =\n \"Unable to switch LLM profiles because the selected agent server returned \" +\n \"profile data this UI does not understand. Check the backend URL/session \" +\n \"key and update the agent server if needed.\";\n\nfunction invalidConversationResponse(): Error {\n return new Error(INVALID_CONVERSATION_RESPONSE_MESSAGE);\n}\n\nfunction isRecord(value: unknown): value is Record<string, unknown> {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nfunction isLLMConfig(value: unknown): value is LLMConfig {\n return isRecord(value) && typeof value.model === \"string\";\n}\n\nfunction numberOrNull(value: unknown): number | null {\n return typeof value === \"number\" ? value : null;\n}\n\nfunction numberOrZero(value: unknown): number {\n return typeof value === \"number\" ? value : 0;\n}\n\nfunction stringOrNull(value: unknown): string | null {\n return typeof value === \"string\" ? value : null;\n}\n\nfunction readTimestamp(\n item: Record<string, unknown>,\n snakeKey: \"created_at\" | \"updated_at\",\n camelKey: \"createdAt\" | \"updatedAt\",\n): string {\n const value = item[snakeKey] ?? item[camelKey];\n return typeof value === \"string\" && value.trim()\n ? value\n : DEFAULT_CONVERSATION_TIMESTAMP;\n}\n\nfunction normalizeTokenUsage(\n value: unknown,\n): NonNullable<MetricsSnapshot[\"accumulated_token_usage\"]> | null {\n if (!isRecord(value)) return null;\n\n return {\n prompt_tokens: numberOrZero(value.prompt_tokens),\n completion_tokens: numberOrZero(value.completion_tokens),\n cache_read_tokens: numberOrZero(value.cache_read_tokens),\n cache_write_tokens: numberOrZero(value.cache_write_tokens),\n context_window: numberOrZero(value.context_window),\n per_turn_token: numberOrZero(value.per_turn_token),\n };\n}\n\nfunction normalizeMetrics(value: unknown): MetricsSnapshot | null {\n if (!isRecord(value)) return null;\n\n return {\n accumulated_cost: numberOrNull(value.accumulated_cost),\n max_budget_per_task: numberOrNull(value.max_budget_per_task),\n accumulated_token_usage: normalizeTokenUsage(value.accumulated_token_usage),\n };\n}\n\nfunction normalizeAgent(value: unknown): DirectConversationInfo[\"agent\"] {\n if (!isRecord(value)) return null;\n const llm = isRecord(value.llm)\n ? { model: stringOrNull(value.llm.model) }\n : null;\n // ``kind`` is the SDK's pydantic discriminator (``\"Agent\"`` vs ``\"ACPAgent\"``);\n // ``toAppConversation`` reads it to derive ``agent_kind``. ``acp_model`` is\n // the Canvas-configured model on the ACPAgent — preserved so the conversation\n // adapter and the conversation chip can fall back to it when the SDK runtime\n // model fields aren't populated. Preserving these here makes the wire path\n // agree with the unit-test path that builds ``DirectConversationInfo``\n // directly (e.g. ``__tests__/api/agent-server-adapter.test.ts``).\n return {\n kind: stringOrNull(value.kind),\n acp_model: stringOrNull(value.acp_model),\n llm,\n };\n}\n\nfunction normalizeWorkspace(\n value: unknown,\n): DirectConversationInfo[\"workspace\"] {\n if (!isRecord(value)) return null;\n return { working_dir: stringOrNull(value.working_dir) };\n}\n\n/**\n * Accept the agent-server's ``tags: Record[str, str]`` payload defensively:\n * the wire shape is guaranteed by the server-side validator (keys\n * ``^[a-z0-9]+$``, string values), but a non-conforming response (older\n * server, raw API write, future schema drift) must never crash the parser\n * — Canvas only consumes ``acpserver`` and falls back to a generic chip\n * for anything it doesn't recognize. Drop entries whose value isn't a\n * plain string; return ``null`` when the wire field is absent or not an\n * object so consumers can use ``info.tags?.[KEY] ?? null`` uniformly.\n */\nfunction normalizeTags(value: unknown): Record<string, string> | null {\n if (!isRecord(value)) return null;\n const tags: Record<string, string> = {};\n for (const [key, entry] of Object.entries(value)) {\n if (typeof entry === \"string\") {\n tags[key] = entry;\n }\n }\n return tags;\n}\n\nfunction normalizeAbsolutePath(path: string): string | null {\n if (!path.startsWith(\"/\")) return null;\n\n const segments: string[] = [];\n for (const segment of path.split(\"/\")) {\n if (segment && segment !== \".\") {\n if (segment === \"..\") {\n if (!segments.length) return null;\n segments.pop();\n } else {\n segments.push(segment);\n }\n }\n }\n\n return `/${segments.join(\"/\")}`;\n}\n\nfunction requirePathInsideDirectory(path: string, directory: string): string {\n const normalizedPath = normalizeAbsolutePath(path);\n const normalizedDirectory = normalizeAbsolutePath(directory);\n\n if (\n !normalizedPath ||\n !normalizedDirectory ||\n (normalizedPath !== normalizedDirectory &&\n !normalizedPath.startsWith(`${normalizedDirectory}/`))\n ) {\n throw new Error(\"Conversation file path must stay inside the workspace\");\n }\n\n return normalizedPath;\n}\n\nfunction requireDirectConversationInfo(item: unknown): DirectConversationInfo {\n if (!isRecord(item) || typeof item.id !== \"string\" || !item.id.trim()) {\n throw invalidConversationResponse();\n }\n\n return {\n id: item.id.trim(),\n title: stringOrNull(item.title),\n created_at: readTimestamp(item, \"created_at\", \"createdAt\"),\n updated_at: readTimestamp(item, \"updated_at\", \"updatedAt\"),\n execution_status: stringOrNull(item.execution_status),\n sandbox_status: stringOrNull(item.sandbox_status),\n metrics: normalizeMetrics(item.metrics),\n agent: normalizeAgent(item.agent),\n workspace: normalizeWorkspace(item.workspace),\n tags: normalizeTags(item.tags),\n // SDK-runtime ACP model fields (populated when the agent-server supports\n // ``ConversationInfo.current_model_*``). Consumed by the conversation\n // adapter to drive the per-card chip's model text. Older agent-servers\n // omit these — adapter handles ``undefined`` / ``null`` gracefully.\n current_model_id: stringOrNull(item.current_model_id),\n current_model_name: stringOrNull(item.current_model_name),\n };\n}\n\nfunction requireDirectConversationItems(\n items: unknown,\n): DirectConversationInfo[] {\n if (!Array.isArray(items)) {\n throw invalidConversationResponse();\n }\n return items.map(requireDirectConversationInfo);\n}\n\nfunction requireConversationSearchPage(page: unknown): {\n items: DirectConversationInfo[];\n next_page_id: string | null;\n} {\n if (Array.isArray(page)) {\n return {\n items: requireDirectConversationItems(page),\n next_page_id: null,\n };\n }\n\n if (!isRecord(page)) {\n throw invalidConversationResponse();\n }\n\n return {\n items: requireDirectConversationItems(page.items),\n next_page_id:\n typeof page.next_page_id === \"string\" ? page.next_page_id : null,\n };\n}\n\nconst RUNTIME_STATUSES = new Set<string>([\n \"idle\",\n \"running\",\n \"paused\",\n \"waiting_for_confirmation\",\n \"finished\",\n \"error\",\n \"stuck\",\n]);\n\nfunction toRuntimeStatus(\n status: DirectConversationInfo[\"execution_status\"],\n): RuntimeConversationInfo[\"status\"] {\n const nextStatus = status ?? \"idle\";\n return (\n RUNTIME_STATUSES.has(nextStatus) ? nextStatus : \"idle\"\n ) as RuntimeConversationInfo[\"status\"];\n}\n\nfunction requireAppConversation(\n conversation: AppConversation | null | undefined,\n conversationId: string,\n): AppConversation {\n if (!conversation) {\n throw new Error(`Conversation ${conversationId} was not found`);\n }\n return conversation;\n}\n\nclass AgentServerConversationService {\n static async sendMessage(\n conversationId: string,\n message: SendMessageRequest,\n runtime?: ConversationRuntimeContext | null,\n ): Promise<SendMessageResponse> {\n const active = getActiveBackend().backend;\n let conversationUrl = runtime?.conversationUrl ?? null;\n let sessionApiKey = runtime?.sessionApiKey ?? null;\n\n if (active.kind === \"cloud\") {\n if (!conversationUrl || !sessionApiKey) {\n const [conversation] = await batchGetCloudConversations([\n conversationId,\n ]);\n conversationUrl = conversation?.conversation_url?.trim() ?? null;\n sessionApiKey = conversation?.session_api_key?.trim() ?? null;\n }\n\n if (!conversationUrl || !sessionApiKey) {\n throw new Error(\n \"Conversation sandbox is still starting. Wait for it to finish, then try again.\",\n );\n }\n\n await callCloudProxy({\n backend: active,\n method: \"POST\",\n hostOverride: buildHttpBaseUrl(conversationUrl),\n path: `/api/conversations/${conversationId}/events`,\n body: { ...message, run: true },\n authMode: \"session-api-key\",\n sessionApiKey,\n });\n\n return message;\n }\n\n await new ConversationClient(\n getAgentServerClientOptions({ conversationUrl, sessionApiKey }),\n ).sendEvent(conversationId, message, {\n run: true,\n });\n\n return message;\n }\n\n static async createConversation(\n initialUserMsg?: string,\n conversationInstructions?: string,\n plugins?: PluginSpec[],\n metadata?: ConversationMetadata | null,\n workingDirOverride?: string,\n parentConversationId?: string,\n agentType?: \"default\" | \"plan\",\n sandboxId?: string,\n ): Promise<AppConversationStartTask> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n // Cloud path mirrors OpenHands' frontend: build a flat\n // AppConversationStartRequest, POST /api/v1/app-conversations\n // (returns a WORKING task), and let the conversation route's\n // useTaskPolling drive it to READY. NO encrypted-settings\n // round-trip — the cloud backend holds secrets server-side.\n const request: AppConversationStartRequest = {\n initial_message: initialUserMsg\n ? {\n role: \"user\",\n content: [{ type: \"text\", text: initialUserMsg }],\n }\n : null,\n title: conversationInstructions ?? null,\n selected_repository: metadata?.selected_repository ?? null,\n selected_branch: metadata?.selected_branch ?? null,\n git_provider: metadata?.git_provider ?? null,\n plugins: plugins ?? null,\n parent_conversation_id: parentConversationId ?? null,\n agent_type: agentType,\n sandbox_id: sandboxId ?? null,\n };\n return createCloudAppConversation(request);\n }\n\n const settings = await SettingsService.getSettings();\n const conversationId = uuidv4();\n const workingDir =\n workingDirOverride ?? buildConversationWorkingDir(conversationId);\n\n // Use encrypted settings to avoid exposing secrets in the browser\n const payload = await buildStartConversationRequestWithEncryptedSettings({\n settings,\n query: initialUserMsg,\n conversationInstructions,\n plugins,\n conversationId,\n workingDir,\n });\n\n const data = await new ConversationClient(\n getAgentServerClientOptions(),\n ).createConversation<DirectConversationInfo>(payload);\n\n if (metadata?.selected_repository || workingDirOverride) {\n // The agent-server runtime has no concept of selected repo/branch/\n // workspace, so persist the home-page selection client-side.\n // `toAppConversation` reads the repo/branch fields back to hydrate\n // the chat-page badges; `useHasAttachedSource` reads\n // `selected_workspace` to default the Files tab to Diff mode when\n // the user explicitly attached a local workspace.\n setStoredConversationMetadata(data.id, {\n selected_repository: metadata?.selected_repository ?? null,\n selected_branch: metadata?.selected_branch ?? null,\n git_provider: metadata?.git_provider ?? null,\n selected_workspace: workingDirOverride ?? null,\n });\n }\n\n return {\n id: data.id,\n created_by_user_id: null,\n status: \"READY\",\n detail: null,\n app_conversation_id: data.id,\n agent_server_url: getEffectiveLocalBackend().host,\n request: {\n initial_message: payload.initial_message as\n | AppConversationStartRequest[\"initial_message\"]\n | undefined,\n plugins: plugins ?? null,\n },\n created_at: data.created_at,\n updated_at: data.updated_at,\n };\n }\n\n static async getStartTask(\n taskId: string,\n ): Promise<AppConversationStartTask | null> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n return getCloudAppConversationStartTask(taskId);\n }\n // Local agent-server creates conversations synchronously — every\n // local \"task\" is already READY when createConversation returns, so\n // there's nothing to poll for.\n return null;\n }\n\n static async getVSCodeUrl(\n conversationId: string,\n conversationUrl: string | null | undefined,\n sessionApiKey?: string | null,\n ): Promise<GetVSCodeUrlResponse> {\n // Local-only path. Cloud conversations read the VSCode URL straight\n // from the cloud-computed `sandbox.exposed_urls` (see\n // `useUnifiedVSCodeUrl` + `useCloudSandbox`); the runtime's own\n // `/api/vscode/url` only knows its internal `localhost:8001`, which\n // the user's browser can't reach.\n const workspaceDir =\n await this.resolveConversationWorkingDir(conversationId);\n // Local mode: the typescript-client targets the local agent-server\n // directly via the conversationUrl override.\n const vscodeUrl = await new VSCodeClient(\n getAgentServerClientOptions({\n conversationUrl,\n sessionApiKey,\n }),\n ).getUrl({\n baseUrl:\n typeof window !== \"undefined\" ? window.location.origin : undefined,\n workspaceDir,\n });\n\n return { vscode_url: vscodeUrl };\n }\n\n static async resolveConversationWorkingDir(\n conversationId: string,\n ): Promise<string> {\n const [conversation] = await this.batchGetAppConversations([\n conversationId,\n ]);\n return conversation?.workspace?.working_dir ?? getAgentServerWorkingDir();\n }\n\n static async batchGetAppConversations(\n ids: string[],\n ): Promise<(AppConversation | null)[]> {\n if (ids.length === 0) return [];\n\n if (getActiveBackend().backend.kind === \"cloud\") {\n return batchGetCloudConversations(ids);\n }\n\n const data = await new ConversationClient(\n getAgentServerClientOptions(),\n ).getConversations<DirectConversationInfo>(ids);\n\n return requireDirectConversationItems(data).map((item) =>\n toAppConversation(item),\n );\n }\n\n static async updateConversationPublicFlag(\n conversationId: string,\n isPublic: boolean,\n ): Promise<AppConversation> {\n if (getActiveBackend().backend.kind !== \"cloud\") {\n throw new Error(\"Public sharing requires a cloud backend.\");\n }\n return updateCloudConversationPublicFlag(conversationId, isPublic);\n }\n\n static async updateConversationRepository(\n conversationId: string,\n repository: string | null,\n branch?: string | null,\n gitProvider?: string | null,\n ): Promise<AppConversation> {\n if (repository) {\n const existing = getStoredConversationMetadata(conversationId);\n setStoredConversationMetadata(conversationId, {\n ...(existing ?? {}),\n selected_repository: repository,\n selected_branch: branch ?? null,\n git_provider: (gitProvider as Provider | null | undefined) ?? null,\n });\n } else {\n removeStoredConversationMetadata(conversationId);\n }\n const [conversation] = await this.batchGetAppConversations([\n conversationId,\n ]);\n return requireAppConversation(conversation, conversationId);\n }\n\n static async readConversationFile(\n conversationId: string,\n filePath?: string,\n ): Promise<string> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n // Cloud exposes a per-conversation file endpoint; the sandbox\n // working dir is fixed (`/workspace/project`), so PLAN.md lives at\n // a known absolute path. Mirrors OpenHands' readConversationFile.\n const path = requirePathInsideDirectory(\n filePath ?? \"/workspace/project/.agents_tmp/PLAN.md\",\n \"/workspace/project\",\n );\n return readCloudConversationFile(conversationId, path);\n }\n\n const workingDir = await this.resolveConversationWorkingDir(conversationId);\n const path = requirePathInsideDirectory(\n filePath ?? `${workingDir}/.agents_tmp/PLAN.md`,\n workingDir,\n );\n return new FileClient(getAgentServerClientOptions()).downloadTextFile(path);\n }\n\n static async downloadConversation(conversationId: string): Promise<Blob> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n return downloadCloudConversation(conversationId);\n }\n\n return new FileClient(getAgentServerClientOptions()).downloadTrajectory(\n conversationId,\n );\n }\n\n static async getHooks(conversationId: string): Promise<GetHooksResponse> {\n if (!conversationId) {\n return emptyHooksResponse();\n }\n return emptyHooksResponse();\n }\n\n static async getRuntimeConversation(\n conversationId: string,\n conversationUrl: string | null | undefined,\n sessionApiKey?: string | null,\n ): Promise<RuntimeConversationInfo> {\n const active = getActiveBackend().backend;\n\n type RawRuntime = DirectConversationInfo & {\n stats?: RuntimeConversationInfo[\"stats\"];\n };\n\n // Cloud mode: route through the cloud-proxy to the runtime sandbox at\n // the conversation's runtime URL — same pattern as getVSCodeUrl. Local\n // mode forwards conversationUrl so the host explicitly resolves to the\n // conversation's runtime instead of falling back to the active backend.\n const response =\n active.kind === \"cloud\" && conversationUrl\n ? await callCloudProxy<RawRuntime>({\n backend: active,\n method: \"GET\",\n hostOverride: buildHttpBaseUrl(conversationUrl),\n path: `/api/conversations/${conversationId}`,\n authMode: \"session-api-key\",\n sessionApiKey,\n })\n : await new ConversationClient(\n getAgentServerClientOptions({\n conversationUrl,\n sessionApiKey,\n }),\n ).getConversation<RawRuntime>(conversationId);\n const data = requireDirectConversationInfo(response);\n const stats = isRecord(response) ? response.stats : null;\n\n return {\n id: data.id,\n title: data.title?.trim()\n ? data.title\n : getDefaultConversationTitle(data.id),\n metrics: normalizeMetrics(data.metrics),\n created_at: data.created_at,\n updated_at: data.updated_at,\n status: toRuntimeStatus(data.execution_status),\n stats: isRecord(stats) ? stats : { usage_to_metrics: {} },\n };\n }\n\n static async searchConversations(\n limit: number = 20,\n pageId?: string,\n ): Promise<AppConversationPage> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n return searchCloudConversations(limit, pageId);\n }\n\n const data = await new ConversationClient(\n getAgentServerClientOptions(),\n ).searchConversations({\n limit,\n page_id: pageId,\n sort_order: ConversationSortOrder.UPDATED_AT_DESC,\n });\n\n return toConversationPage(requireConversationSearchPage(data));\n }\n\n static async deleteConversation(conversationId: string): Promise<void> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n await deleteCloudConversation(conversationId);\n } else {\n await new ConversationClient(\n getAgentServerClientOptions(),\n ).deleteConversation(conversationId);\n }\n removeStoredConversationMetadata(conversationId);\n }\n\n static async updateConversationTitle(\n conversationId: string,\n title: string,\n ): Promise<AppConversation> {\n await new ConversationClient(\n getAgentServerClientOptions(),\n ).updateConversation(conversationId, {\n title,\n });\n const [conversation] = await this.batchGetAppConversations([\n conversationId,\n ]);\n return requireAppConversation(conversation, conversationId);\n }\n\n /**\n * Switches the LLM profile for the running conversation when one is open\n * (POST /switch_llm — per-conversation swap, doesn't change the user's\n * default profile). When called without a conversationId (home page),\n * falls back to POST /activate so the next conversation created picks up\n * the chosen profile.\n *\n * The /switch_llm body needs the LLM config, which we fetch with encrypted\n * secrets — same flow as conversation-start.\n */\n static async switchProfile(\n conversationId: string | null,\n profileName: string,\n ): Promise<void> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n throw new Error(\n \"LLM profile switching is only supported for local agent-server backends.\",\n );\n }\n\n const profilesClient = new ProfilesClient(getAgentServerClientOptions());\n\n if (!conversationId) {\n await profilesClient.activateProfile(profileName);\n return;\n }\n\n const profile = await profilesClient.getProfile(profileName, {\n exposeSecrets: \"encrypted\",\n });\n if (!isLLMConfig(profile.config)) {\n throw new Error(INVALID_PROFILE_CONFIG_MESSAGE);\n }\n\n await new ConversationClient(getAgentServerClientOptions()).switchLLM(\n conversationId,\n profile.config,\n );\n }\n}\n\nexport default AgentServerConversationService;\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"agent-server-conversation-service.api.cjs","names":[],"sources":["../../../src/api/conversation-service/agent-server-conversation-service.api.ts"],"sourcesContent":["import {\n ConversationSortOrder,\n type LLMConfig,\n} from \"@openhands/typescript-client\";\nimport {\n ConversationClient,\n FileClient,\n ProfilesClient,\n VSCodeClient,\n} from \"@openhands/typescript-client/clients\";\nimport { v4 as uuidv4 } from \"uuid\";\nimport { Provider } from \"#/types/settings\";\nimport type { ConversationRuntimeContext } from \"#/api/conversation-file-upload.api\";\nimport { buildHttpBaseUrl } from \"#/utils/websocket-url\";\nimport {\n buildConversationWorkingDir,\n getAgentServerWorkingDir,\n} from \"../agent-server-config\";\nimport {\n getActiveBackend,\n getEffectiveLocalBackend,\n} from \"../backend-registry/active-store\";\nimport { callCloudProxy } from \"../cloud/proxy\";\nimport {\n batchGetCloudConversations,\n createCloudAppConversation,\n deleteCloudConversation,\n downloadCloudConversation,\n getCloudAppConversationStartTask,\n readCloudConversationFile,\n searchCloudConversations,\n updateCloudConversationPublicFlag,\n} from \"../cloud/conversation-service.api\";\nimport {\n DirectConversationInfo,\n buildStartConversationRequestWithEncryptedSettings,\n emptyHooksResponse,\n getDefaultConversationTitle,\n toAppConversation,\n toConversationPage,\n} from \"../agent-server-adapter\";\nimport { GetVSCodeUrlResponse } from \"../open-hands.types\";\nimport { getAgentServerClientOptions } from \"../agent-server-client-options\";\nimport SettingsService from \"../settings-service/settings-service.api\";\nimport {\n ConversationMetadata,\n getStoredConversationMetadata,\n removeStoredConversationMetadata,\n setStoredConversationMetadata,\n} from \"../conversation-metadata-store\";\nimport type {\n GetHooksResponse,\n PluginSpec,\n AppConversation,\n AppConversationPage,\n AppConversationStartRequest,\n AppConversationStartTask,\n MetricsSnapshot,\n RuntimeConversationInfo,\n SendMessageRequest,\n SendMessageResponse,\n} from \"./agent-server-conversation-service.types\";\n\nconst DEFAULT_CONVERSATION_TIMESTAMP = \"1970-01-01T00:00:00.000Z\";\nconst INVALID_CONVERSATION_RESPONSE_MESSAGE =\n \"Unable to load conversations because the selected agent server returned \" +\n \"data this UI does not understand. Check the backend URL/session key and \" +\n \"update the agent server if needed.\";\nconst INVALID_PROFILE_CONFIG_MESSAGE =\n \"Unable to switch LLM profiles because the selected agent server returned \" +\n \"profile data this UI does not understand. Check the backend URL/session \" +\n \"key and update the agent server if needed.\";\n\nfunction invalidConversationResponse(): Error {\n return new Error(INVALID_CONVERSATION_RESPONSE_MESSAGE);\n}\n\nfunction isRecord(value: unknown): value is Record<string, unknown> {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nfunction isLLMConfig(value: unknown): value is LLMConfig {\n return isRecord(value) && typeof value.model === \"string\";\n}\n\nfunction numberOrNull(value: unknown): number | null {\n return typeof value === \"number\" ? value : null;\n}\n\nfunction numberOrZero(value: unknown): number {\n return typeof value === \"number\" ? value : 0;\n}\n\nfunction stringOrNull(value: unknown): string | null {\n return typeof value === \"string\" ? value : null;\n}\n\nfunction readTimestamp(\n item: Record<string, unknown>,\n snakeKey: \"created_at\" | \"updated_at\",\n camelKey: \"createdAt\" | \"updatedAt\",\n): string {\n const value = item[snakeKey] ?? item[camelKey];\n return typeof value === \"string\" && value.trim()\n ? value\n : DEFAULT_CONVERSATION_TIMESTAMP;\n}\n\nfunction normalizeTokenUsage(\n value: unknown,\n): NonNullable<MetricsSnapshot[\"accumulated_token_usage\"]> | null {\n if (!isRecord(value)) return null;\n\n return {\n prompt_tokens: numberOrZero(value.prompt_tokens),\n completion_tokens: numberOrZero(value.completion_tokens),\n cache_read_tokens: numberOrZero(value.cache_read_tokens),\n cache_write_tokens: numberOrZero(value.cache_write_tokens),\n context_window: numberOrZero(value.context_window),\n per_turn_token: numberOrZero(value.per_turn_token),\n };\n}\n\nfunction normalizeMetrics(value: unknown): MetricsSnapshot | null {\n if (!isRecord(value)) return null;\n\n return {\n accumulated_cost: numberOrNull(value.accumulated_cost),\n max_budget_per_task: numberOrNull(value.max_budget_per_task),\n accumulated_token_usage: normalizeTokenUsage(value.accumulated_token_usage),\n };\n}\n\nfunction normalizeAgent(value: unknown): DirectConversationInfo[\"agent\"] {\n if (!isRecord(value)) return null;\n const llm = isRecord(value.llm)\n ? { model: stringOrNull(value.llm.model) }\n : null;\n // ``kind`` is the SDK's pydantic discriminator (``\"Agent\"`` vs ``\"ACPAgent\"``);\n // ``toAppConversation`` reads it to derive ``agent_kind``. ``acp_model`` is\n // the Canvas-configured model on the ACPAgent — preserved so the conversation\n // adapter and the conversation chip can fall back to it when the SDK runtime\n // model fields aren't populated. Preserving these here makes the wire path\n // agree with the unit-test path that builds ``DirectConversationInfo``\n // directly (e.g. ``__tests__/api/agent-server-adapter.test.ts``).\n return {\n kind: stringOrNull(value.kind),\n acp_model: stringOrNull(value.acp_model),\n llm,\n };\n}\n\nfunction normalizeWorkspace(\n value: unknown,\n): DirectConversationInfo[\"workspace\"] {\n if (!isRecord(value)) return null;\n return { working_dir: stringOrNull(value.working_dir) };\n}\n\n/**\n * Accept the agent-server's ``tags: Record[str, str]`` payload defensively:\n * the wire shape is guaranteed by the server-side validator (keys\n * ``^[a-z0-9]+$``, string values), but a non-conforming response (older\n * server, raw API write, future schema drift) must never crash the parser\n * — Canvas only consumes ``acpserver`` and falls back to a generic chip\n * for anything it doesn't recognize. Drop entries whose value isn't a\n * plain string; return ``null`` when the wire field is absent or not an\n * object so consumers can use ``info.tags?.[KEY] ?? null`` uniformly.\n */\nfunction normalizeTags(value: unknown): Record<string, string> | null {\n if (!isRecord(value)) return null;\n const tags: Record<string, string> = {};\n for (const [key, entry] of Object.entries(value)) {\n if (typeof entry === \"string\") {\n tags[key] = entry;\n }\n }\n return tags;\n}\n\nfunction normalizeAbsolutePath(path: string): string | null {\n if (!path.startsWith(\"/\")) return null;\n\n const segments: string[] = [];\n for (const segment of path.split(\"/\")) {\n if (segment && segment !== \".\") {\n if (segment === \"..\") {\n if (!segments.length) return null;\n segments.pop();\n } else {\n segments.push(segment);\n }\n }\n }\n\n return `/${segments.join(\"/\")}`;\n}\n\nfunction requirePathInsideDirectory(path: string, directory: string): string {\n const normalizedPath = normalizeAbsolutePath(path);\n const normalizedDirectory = normalizeAbsolutePath(directory);\n\n if (\n !normalizedPath ||\n !normalizedDirectory ||\n (normalizedPath !== normalizedDirectory &&\n !normalizedPath.startsWith(`${normalizedDirectory}/`))\n ) {\n throw new Error(\"Conversation file path must stay inside the workspace\");\n }\n\n return normalizedPath;\n}\n\nfunction requireDirectConversationInfo(item: unknown): DirectConversationInfo {\n if (!isRecord(item) || typeof item.id !== \"string\" || !item.id.trim()) {\n throw invalidConversationResponse();\n }\n\n return {\n id: item.id.trim(),\n title: stringOrNull(item.title),\n created_at: readTimestamp(item, \"created_at\", \"createdAt\"),\n updated_at: readTimestamp(item, \"updated_at\", \"updatedAt\"),\n execution_status: stringOrNull(item.execution_status),\n sandbox_status: stringOrNull(item.sandbox_status),\n metrics: normalizeMetrics(item.metrics),\n agent: normalizeAgent(item.agent),\n workspace: normalizeWorkspace(item.workspace),\n tags: normalizeTags(item.tags),\n // SDK-runtime ACP model fields (populated when the agent-server supports\n // ``ConversationInfo.current_model_*``). Consumed by the conversation\n // adapter to drive the per-card chip's model text. Older agent-servers\n // omit these — adapter handles ``undefined`` / ``null`` gracefully.\n current_model_id: stringOrNull(item.current_model_id),\n current_model_name: stringOrNull(item.current_model_name),\n };\n}\n\nfunction requireDirectConversationItems(\n items: unknown,\n): DirectConversationInfo[] {\n if (!Array.isArray(items)) {\n throw invalidConversationResponse();\n }\n return items.map(requireDirectConversationInfo);\n}\n\nfunction requireConversationSearchPage(page: unknown): {\n items: DirectConversationInfo[];\n next_page_id: string | null;\n} {\n if (Array.isArray(page)) {\n return {\n items: requireDirectConversationItems(page),\n next_page_id: null,\n };\n }\n\n if (!isRecord(page)) {\n throw invalidConversationResponse();\n }\n\n return {\n items: requireDirectConversationItems(page.items),\n next_page_id:\n typeof page.next_page_id === \"string\" ? page.next_page_id : null,\n };\n}\n\nconst RUNTIME_STATUSES = new Set<string>([\n \"idle\",\n \"running\",\n \"paused\",\n \"waiting_for_confirmation\",\n \"finished\",\n \"error\",\n \"stuck\",\n]);\n\nfunction toRuntimeStatus(\n status: DirectConversationInfo[\"execution_status\"],\n): RuntimeConversationInfo[\"status\"] {\n const nextStatus = status ?? \"idle\";\n return (\n RUNTIME_STATUSES.has(nextStatus) ? nextStatus : \"idle\"\n ) as RuntimeConversationInfo[\"status\"];\n}\n\nfunction requireAppConversation(\n conversation: AppConversation | null | undefined,\n conversationId: string,\n): AppConversation {\n if (!conversation) {\n throw new Error(`Conversation ${conversationId} was not found`);\n }\n return conversation;\n}\n\nclass AgentServerConversationService {\n static async sendMessage(\n conversationId: string,\n message: SendMessageRequest,\n runtime?: ConversationRuntimeContext | null,\n ): Promise<SendMessageResponse> {\n const active = getActiveBackend().backend;\n let conversationUrl = runtime?.conversationUrl ?? null;\n let sessionApiKey = runtime?.sessionApiKey ?? null;\n\n if (active.kind === \"cloud\") {\n if (!conversationUrl || !sessionApiKey) {\n const [conversation] = await batchGetCloudConversations([\n conversationId,\n ]);\n conversationUrl = conversation?.conversation_url?.trim() ?? null;\n sessionApiKey = conversation?.session_api_key?.trim() ?? null;\n }\n\n if (!conversationUrl || !sessionApiKey) {\n throw new Error(\n \"Conversation sandbox is still starting. Wait for it to finish, then try again.\",\n );\n }\n\n await callCloudProxy({\n backend: active,\n method: \"POST\",\n hostOverride: buildHttpBaseUrl(conversationUrl),\n path: `/api/conversations/${conversationId}/events`,\n body: { ...message, run: true },\n authMode: \"session-api-key\",\n sessionApiKey,\n });\n\n return message;\n }\n\n await new ConversationClient(\n getAgentServerClientOptions({ conversationUrl, sessionApiKey }),\n ).sendEvent(conversationId, message, {\n run: true,\n });\n\n return message;\n }\n\n static async createConversation(\n initialUserMsg?: string,\n conversationInstructions?: string,\n plugins?: PluginSpec[],\n metadata?: ConversationMetadata | null,\n workingDirOverride?: string,\n parentConversationId?: string,\n agentType?: \"default\" | \"plan\",\n sandboxId?: string,\n ): Promise<AppConversationStartTask> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n // Cloud path mirrors OpenHands' frontend: build a flat\n // AppConversationStartRequest, POST /api/v1/app-conversations\n // (returns a WORKING task), and let the conversation route's\n // useTaskPolling drive it to READY. NO encrypted-settings\n // round-trip — the cloud backend holds secrets server-side.\n const request: AppConversationStartRequest = {\n initial_message: initialUserMsg\n ? {\n role: \"user\",\n content: [{ type: \"text\", text: initialUserMsg }],\n }\n : null,\n title: conversationInstructions ?? null,\n selected_repository: metadata?.selected_repository ?? null,\n selected_branch: metadata?.selected_branch ?? null,\n git_provider: metadata?.git_provider ?? null,\n plugins: plugins ?? null,\n parent_conversation_id: parentConversationId ?? null,\n agent_type: agentType,\n sandbox_id: sandboxId ?? null,\n };\n return createCloudAppConversation(request);\n }\n\n const settings = await SettingsService.getSettings();\n const conversationId = uuidv4();\n const workingDir =\n workingDirOverride ?? buildConversationWorkingDir(conversationId);\n\n // Use encrypted settings to avoid exposing secrets in the browser\n const payload = await buildStartConversationRequestWithEncryptedSettings({\n settings,\n query: initialUserMsg,\n conversationInstructions,\n plugins,\n conversationId,\n workingDir,\n });\n\n const data = await new ConversationClient(\n getAgentServerClientOptions(),\n ).createConversation<DirectConversationInfo>(payload);\n\n if (metadata?.selected_repository || workingDirOverride) {\n // The agent-server runtime has no concept of selected repo/branch/\n // workspace, so persist the home-page selection client-side.\n // `toAppConversation` reads the repo/branch fields back to hydrate\n // the chat-page badges; `useHasAttachedSource` reads\n // `selected_workspace` to default the Files tab to Diff mode when\n // the user explicitly attached a local workspace.\n setStoredConversationMetadata(data.id, {\n selected_repository: metadata?.selected_repository ?? null,\n selected_branch: metadata?.selected_branch ?? null,\n git_provider: metadata?.git_provider ?? null,\n selected_workspace: workingDirOverride ?? null,\n });\n }\n\n return {\n id: data.id,\n created_by_user_id: null,\n status: \"READY\",\n detail: null,\n app_conversation_id: data.id,\n agent_server_url: getEffectiveLocalBackend().host,\n request: {\n initial_message: payload.initial_message as\n | AppConversationStartRequest[\"initial_message\"]\n | undefined,\n plugins: plugins ?? null,\n },\n created_at: data.created_at,\n updated_at: data.updated_at,\n };\n }\n\n static async getStartTask(\n taskId: string,\n ): Promise<AppConversationStartTask | null> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n return getCloudAppConversationStartTask(taskId);\n }\n // Local agent-server creates conversations synchronously — every\n // local \"task\" is already READY when createConversation returns, so\n // there's nothing to poll for.\n return null;\n }\n\n static async getVSCodeUrl(\n conversationId: string,\n conversationUrl: string | null | undefined,\n sessionApiKey?: string | null,\n ): Promise<GetVSCodeUrlResponse> {\n // Local-only path. Cloud conversations read the VSCode URL straight\n // from the cloud-computed `sandbox.exposed_urls` (see\n // `useUnifiedVSCodeUrl` + `useCloudSandbox`); the runtime's own\n // `/api/vscode/url` only knows its internal `localhost:8001`, which\n // the user's browser can't reach.\n const workspaceDir =\n await this.resolveConversationWorkingDir(conversationId);\n // Local mode: the typescript-client targets the local agent-server\n // directly via the conversationUrl override.\n const vscodeUrl = await new VSCodeClient(\n getAgentServerClientOptions({\n conversationUrl,\n sessionApiKey,\n }),\n ).getUrl({\n baseUrl:\n typeof window !== \"undefined\" ? window.location.origin : undefined,\n workspaceDir,\n });\n\n return { vscode_url: vscodeUrl };\n }\n\n static async resolveConversationWorkingDir(\n conversationId: string,\n ): Promise<string> {\n const [conversation] = await this.batchGetAppConversations([\n conversationId,\n ]);\n return conversation?.workspace?.working_dir ?? getAgentServerWorkingDir();\n }\n\n static async batchGetAppConversations(\n ids: string[],\n ): Promise<(AppConversation | null)[]> {\n if (ids.length === 0) return [];\n\n if (getActiveBackend().backend.kind === \"cloud\") {\n return batchGetCloudConversations(ids);\n }\n\n const data = await new ConversationClient(\n getAgentServerClientOptions(),\n ).getConversations<DirectConversationInfo>(ids);\n\n return requireDirectConversationItems(data).map((item) =>\n toAppConversation(item),\n );\n }\n\n static async updateConversationPublicFlag(\n conversationId: string,\n isPublic: boolean,\n ): Promise<AppConversation> {\n if (getActiveBackend().backend.kind !== \"cloud\") {\n throw new Error(\"Public sharing requires a cloud backend.\");\n }\n return updateCloudConversationPublicFlag(conversationId, isPublic);\n }\n\n static async updateConversationRepository(\n conversationId: string,\n repository: string | null,\n branch?: string | null,\n gitProvider?: string | null,\n ): Promise<AppConversation> {\n if (repository) {\n const existing = getStoredConversationMetadata(conversationId);\n setStoredConversationMetadata(conversationId, {\n ...(existing ?? {}),\n selected_repository: repository,\n selected_branch: branch ?? null,\n git_provider: (gitProvider as Provider | null | undefined) ?? null,\n });\n } else {\n removeStoredConversationMetadata(conversationId);\n }\n const [conversation] = await this.batchGetAppConversations([\n conversationId,\n ]);\n return requireAppConversation(conversation, conversationId);\n }\n\n static async readConversationFile(\n conversationId: string,\n filePath?: string,\n ): Promise<string> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n // Cloud exposes a per-conversation file endpoint; the sandbox\n // working dir is fixed (`/workspace/project`), so PLAN.md lives at\n // a known absolute path. Mirrors OpenHands' readConversationFile.\n const path = requirePathInsideDirectory(\n filePath ?? \"/workspace/project/.agents_tmp/PLAN.md\",\n \"/workspace/project\",\n );\n return readCloudConversationFile(conversationId, path);\n }\n\n const workingDir = await this.resolveConversationWorkingDir(conversationId);\n const path = requirePathInsideDirectory(\n filePath ?? `${workingDir}/.agents_tmp/PLAN.md`,\n workingDir,\n );\n return new FileClient(getAgentServerClientOptions()).downloadTextFile(path);\n }\n\n static async downloadConversation(conversationId: string): Promise<Blob> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n return downloadCloudConversation(conversationId);\n }\n\n return new FileClient(getAgentServerClientOptions()).downloadTrajectory(\n conversationId,\n );\n }\n\n static async getHooks(conversationId: string): Promise<GetHooksResponse> {\n if (!conversationId) {\n return emptyHooksResponse();\n }\n return emptyHooksResponse();\n }\n\n static async getRuntimeConversation(\n conversationId: string,\n conversationUrl: string | null | undefined,\n sessionApiKey?: string | null,\n ): Promise<RuntimeConversationInfo> {\n const active = getActiveBackend().backend;\n\n type RawRuntime = DirectConversationInfo & {\n stats?: RuntimeConversationInfo[\"stats\"];\n };\n\n // Cloud mode: route through the cloud-proxy to the runtime sandbox at\n // the conversation's runtime URL — same pattern as getVSCodeUrl. Local\n // mode forwards conversationUrl so the host explicitly resolves to the\n // conversation's runtime instead of falling back to the active backend.\n const response =\n active.kind === \"cloud\" && conversationUrl\n ? await callCloudProxy<RawRuntime>({\n backend: active,\n method: \"GET\",\n hostOverride: buildHttpBaseUrl(conversationUrl),\n path: `/api/conversations/${conversationId}`,\n authMode: \"session-api-key\",\n sessionApiKey,\n })\n : await new ConversationClient(\n getAgentServerClientOptions({\n conversationUrl,\n sessionApiKey,\n }),\n ).getConversation<RawRuntime>(conversationId);\n const data = requireDirectConversationInfo(response);\n const stats = isRecord(response) ? response.stats : null;\n\n return {\n id: data.id,\n title: data.title?.trim()\n ? data.title\n : getDefaultConversationTitle(data.id),\n metrics: normalizeMetrics(data.metrics),\n created_at: data.created_at,\n updated_at: data.updated_at,\n status: toRuntimeStatus(data.execution_status),\n stats: isRecord(stats) ? stats : { usage_to_metrics: {} },\n };\n }\n\n static async searchConversations(\n limit: number = 20,\n pageId?: string,\n ): Promise<AppConversationPage> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n return searchCloudConversations(limit, pageId);\n }\n\n const data = await new ConversationClient(\n getAgentServerClientOptions(),\n ).searchConversations({\n limit,\n page_id: pageId,\n sort_order: ConversationSortOrder.UPDATED_AT_DESC,\n });\n\n return toConversationPage(requireConversationSearchPage(data));\n }\n\n static async deleteConversation(conversationId: string): Promise<void> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n await deleteCloudConversation(conversationId);\n } else {\n await new ConversationClient(\n getAgentServerClientOptions(),\n ).deleteConversation(conversationId);\n }\n removeStoredConversationMetadata(conversationId);\n }\n\n static async updateConversationTitle(\n conversationId: string,\n title: string,\n ): Promise<AppConversation> {\n await new ConversationClient(\n getAgentServerClientOptions(),\n ).updateConversation(conversationId, {\n title,\n });\n const [conversation] = await this.batchGetAppConversations([\n conversationId,\n ]);\n return requireAppConversation(conversation, conversationId);\n }\n\n /**\n * Switches the LLM profile for the running conversation when one is open\n * (POST /switch_llm — per-conversation swap, doesn't change the user's\n * default profile). When called without a conversationId (home page),\n * falls back to POST /activate so the next conversation created picks up\n * the chosen profile.\n *\n * The /switch_llm body needs the LLM config, which we fetch with encrypted\n * secrets — same flow as conversation-start.\n */\n static async switchProfile(\n conversationId: string | null,\n profileName: string,\n ): Promise<void> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n throw new Error(\n \"LLM profile switching is only supported for local agent-server backends.\",\n );\n }\n\n const profilesClient = new ProfilesClient(getAgentServerClientOptions());\n\n if (!conversationId) {\n await profilesClient.activateProfile(profileName);\n return;\n }\n\n const profile = await profilesClient.getProfile(profileName, {\n exposeSecrets: \"encrypted\",\n });\n if (!isLLMConfig(profile.config)) {\n throw new Error(INVALID_PROFILE_CONFIG_MESSAGE);\n }\n\n await new ConversationClient(getAgentServerClientOptions()).switchLLM(\n conversationId,\n profile.config,\n );\n }\n\n /**\n * Switches the model of a running ACP conversation in place (POST\n * /switch_acp_model — the ACP analog of {@link switchProfile}'s /switch_llm).\n * The agent-server calls the ACP wrapper's ``session/set_model`` on the live\n * session, preserving context. Mirrors {@link switchProfile}'s\n * local-backend-only guard and per-conversation ConversationClient call.\n *\n * Only valid once an ACP session exists (after the first message); the\n * agent-server returns 409 before then — the home/no-session default is\n * persisted via Settings instead (see ``use-switch-acp-model``).\n */\n static async switchAcpModel(\n conversationId: string,\n model: string,\n ): Promise<void> {\n if (getActiveBackend().backend.kind === \"cloud\") {\n throw new Error(\n \"ACP model switching is only supported for local agent-server backends.\",\n );\n }\n\n await new ConversationClient(getAgentServerClientOptions()).switchAcpModel(\n conversationId,\n model,\n );\n }\n}\n\nexport default AgentServerConversationService;\n"],"mappings":"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"}
|