tower-studio 0.1.2 → 0.1.3
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/bin/tower.mjs +5 -15
- package/next.config.ts +1 -8
- package/package.json +10 -11
- package/prisma/dev.db +0 -0
- package/public/vs/_commonjsHelpers-CT9FvmAN.js +1 -0
- package/public/vs/abap-D-t0cyap.js +1 -0
- package/public/vs/apex-CcIm7xu6.js +1 -0
- package/public/vs/assets/css.worker-HnVq6Ewq.js +93 -0
- package/public/vs/assets/editor.worker-Be8ye1pW.js +26 -0
- package/public/vs/assets/html.worker-B51mlPHg.js +470 -0
- package/public/vs/assets/json.worker-DKiEKt88.js +58 -0
- package/public/vs/assets/ts.worker-CMbG-7ft.js +67731 -0
- package/public/vs/azcli-BA0tQDCg.js +1 -0
- package/public/vs/basic-languages/monaco.contribution.js +1 -0
- package/public/vs/bat-C397hTD6.js +1 -0
- package/public/vs/bicep-DF5aW17k.js +2 -0
- package/public/vs/cameligo-plsz8qhj.js +1 -0
- package/public/vs/clojure-Y2auQMzK.js +1 -0
- package/public/vs/coffee-Bu45yuWE.js +1 -0
- package/public/vs/cpp-CkKPQIni.js +1 -0
- package/public/vs/csharp-CX28MZyh.js +1 -0
- package/public/vs/csp-D8uWnyxW.js +1 -0
- package/public/vs/css-CaeNmE3S.js +3 -0
- package/public/vs/cssMode-CjiAH6dQ.js +1 -0
- package/public/vs/cypher-DVThT8BS.js +1 -0
- package/public/vs/dart-CmGfCvrO.js +1 -0
- package/public/vs/dockerfile-CZqqYdch.js +1 -0
- package/public/vs/ecl-30fUercY.js +1 -0
- package/public/vs/editor/editor.main.css +1 -0
- package/public/vs/editor/editor.main.js +5 -0
- package/public/vs/editor.api-CalNCsUg.js +903 -0
- package/public/vs/elixir-xjPaIfzF.js +1 -0
- package/public/vs/flow9-DqtmStfK.js +1 -0
- package/public/vs/freemarker2-Cz_sV6Md.js +3 -0
- package/public/vs/fsharp-BOMdg4U1.js +1 -0
- package/public/vs/go-D_hbi-Jt.js +1 -0
- package/public/vs/graphql-CKUU4kLG.js +1 -0
- package/public/vs/handlebars-OwglfO-1.js +1 -0
- package/public/vs/hcl-DTaboeZW.js +1 -0
- package/public/vs/html-Pa1xEWsY.js +1 -0
- package/public/vs/htmlMode-Bz67EXwp.js +1 -0
- package/public/vs/ini-CsNwO04R.js +1 -0
- package/public/vs/java-CI4ZMsH9.js +1 -0
- package/public/vs/javascript-PczUCGdz.js +1 -0
- package/public/vs/jsonMode-DULH5oaX.js +7 -0
- package/public/vs/julia-BwzEvaQw.js +1 -0
- package/public/vs/kotlin-IUYPiTV8.js +1 -0
- package/public/vs/language/css/monaco.contribution.js +1 -0
- package/public/vs/language/html/monaco.contribution.js +1 -0
- package/public/vs/language/json/monaco.contribution.js +1 -0
- package/public/vs/language/typescript/monaco.contribution.js +1 -0
- package/public/vs/less-C0eDYdqa.js +2 -0
- package/public/vs/lexon-iON-Kj97.js +1 -0
- package/public/vs/liquid-DqKjdPGy.js +1 -0
- package/public/vs/loader.js +1368 -0
- package/public/vs/lspLanguageFeatures-kM9O9rjY.js +4 -0
- package/public/vs/lua-DtygF91M.js +1 -0
- package/public/vs/m3-CsR4AuFi.js +1 -0
- package/public/vs/markdown-C_rD0bIw.js +1 -0
- package/public/vs/mdx-DEWtB1K5.js +1 -0
- package/public/vs/mips-CiYP61RB.js +1 -0
- package/public/vs/monaco.contribution-D2OdxNBt.js +1 -0
- package/public/vs/monaco.contribution-DO3azKX8.js +1 -0
- package/public/vs/monaco.contribution-EcChJV6a.js +1 -0
- package/public/vs/monaco.contribution-qLAYrEOP.js +1 -0
- package/public/vs/msdax-C38-sJlp.js +1 -0
- package/public/vs/mysql-CdtbpvbG.js +1 -0
- package/public/vs/nls.messages-loader.js +1 -0
- package/public/vs/nls.messages.cs.js.js +17 -0
- package/public/vs/nls.messages.de.js.js +17 -0
- package/public/vs/nls.messages.es.js.js +17 -0
- package/public/vs/nls.messages.fr.js.js +15 -0
- package/public/vs/nls.messages.it.js.js +15 -0
- package/public/vs/nls.messages.ja.js.js +17 -0
- package/public/vs/nls.messages.js.js +10 -0
- package/public/vs/nls.messages.ko.js.js +25 -0
- package/public/vs/nls.messages.pl.js.js +17 -0
- package/public/vs/nls.messages.pt-br.js.js +6 -0
- package/public/vs/nls.messages.ru.js.js +17 -0
- package/public/vs/nls.messages.tr.js.js +15 -0
- package/public/vs/nls.messages.zh-cn.js.js +17 -0
- package/public/vs/nls.messages.zh-tw.js.js +15 -0
- package/public/vs/objective-c-CntZFaHX.js +1 -0
- package/public/vs/pascal-r6kuqfl_.js +1 -0
- package/public/vs/pascaligo-BiXoTmXh.js +1 -0
- package/public/vs/perl-DABw_TcH.js +1 -0
- package/public/vs/pgsql-me_jFXeX.js +1 -0
- package/public/vs/php-D_kh-9LK.js +1 -0
- package/public/vs/pla-VfZjczW0.js +1 -0
- package/public/vs/postiats-BBSzz8Pk.js +1 -0
- package/public/vs/powerquery-Dt-g_2cc.js +1 -0
- package/public/vs/powershell-B-7ap1zc.js +1 -0
- package/public/vs/protobuf-BmtuEB1A.js +2 -0
- package/public/vs/pug-BRpRNeEb.js +1 -0
- package/public/vs/python-Cr0UkIbn.js +1 -0
- package/public/vs/qsharp-BzsFaUU9.js +1 -0
- package/public/vs/r-f8dDdrp4.js +1 -0
- package/public/vs/razor-BYAHOTkz.js +1 -0
- package/public/vs/redis-fvZQY4PI.js +1 -0
- package/public/vs/redshift-45Et0LQi.js +1 -0
- package/public/vs/restructuredtext-C7UUFKFD.js +1 -0
- package/public/vs/ruby-CZO8zYTz.js +1 -0
- package/public/vs/rust-Bfetafyc.js +1 -0
- package/public/vs/sb-3GYllVck.js +1 -0
- package/public/vs/scala-foMgrKo1.js +1 -0
- package/public/vs/scheme-CHdMtr7p.js +1 -0
- package/public/vs/scss-C1cmLt9V.js +3 -0
- package/public/vs/shell-ClXCKCEW.js +1 -0
- package/public/vs/solidity-MZ6ExpPy.js +1 -0
- package/public/vs/sophia-DWkuSsPQ.js +1 -0
- package/public/vs/sparql-AUGFYSyk.js +1 -0
- package/public/vs/sql-32GpJSV2.js +1 -0
- package/public/vs/st-CuDFIVZ_.js +1 -0
- package/public/vs/swift-n-2HociN.js +3 -0
- package/public/vs/systemverilog-Ch4vA8Yt.js +1 -0
- package/public/vs/tcl-D74tq1nH.js +1 -0
- package/public/vs/tsMode-CZz1Umrk.js +11 -0
- package/public/vs/twig-C6taOxMV.js +1 -0
- package/public/vs/typescript-DfOrAzoV.js +1 -0
- package/public/vs/typespec-D-PIh9Xw.js +1 -0
- package/public/vs/vb-Dyb2648j.js +1 -0
- package/public/vs/wgsl-BhLXMOR0.js +298 -0
- package/public/vs/workers-DcJshg-q.js +1 -0
- package/public/vs/xml-CdsdnY8S.js +1 -0
- package/public/vs/yaml-DYGvmE88.js +1 -0
- package/src/actions/__tests__/agent-actions-username.test.ts +30 -0
- package/src/actions/__tests__/asset-actions.test.ts +251 -0
- package/src/actions/__tests__/assistant-actions.test.ts +20 -0
- package/src/actions/__tests__/cli-profile-actions.test.ts +243 -0
- package/src/actions/__tests__/label-actions.test.ts +187 -0
- package/src/actions/__tests__/note-actions.test.ts +237 -0
- package/src/actions/__tests__/onboarding-actions.test.ts +265 -0
- package/src/actions/__tests__/project-actions.test.ts +179 -0
- package/src/actions/__tests__/prompt-actions.test.ts +213 -0
- package/src/actions/__tests__/report-actions.test.ts +246 -0
- package/src/actions/__tests__/search-code-actions.test.ts +308 -0
- package/src/actions/__tests__/task-actions-overview.test.ts +58 -0
- package/src/actions/__tests__/task-actions-pin.test.ts +79 -0
- package/src/actions/__tests__/workspace-actions.test.ts +256 -0
- package/src/components/layout/__tests__/top-bar-username.test.tsx +24 -0
- package/src/components/onboarding/__tests__/onboarding-wizard.test.tsx +185 -0
- package/src/hooks/__tests__/sse-event-reducer.test.ts +263 -0
- package/src/hooks/__tests__/use-assistant-chat.test.ts +34 -0
- package/src/hooks/__tests__/use-image-upload.test.ts +443 -0
- package/src/lib/__tests__/assistant-message-converter.test.ts +162 -0
- package/src/lib/__tests__/assistant-sessions.test.ts +253 -0
- package/src/lib/__tests__/build-multimodal-prompt.test.ts +173 -0
- package/src/lib/__tests__/config-reader.test.ts +75 -0
- package/src/lib/__tests__/diff-parser.test.ts +212 -0
- package/src/lib/__tests__/execution-summary.test.ts +237 -0
- package/src/lib/__tests__/file-serve.test.ts +178 -0
- package/src/lib/__tests__/file-utils.test.ts +177 -0
- package/src/lib/__tests__/internal-api-guard.test.ts +151 -0
- package/src/lib/__tests__/logger.test.ts +181 -0
- package/src/lib/__tests__/platform.test.ts +566 -0
- package/src/lib/__tests__/reveal-route-security.test.ts +65 -0
- package/src/lib/__tests__/schemas.test.ts +377 -0
- package/src/lib/__tests__/terminal-link-provider.test.ts +160 -0
- package/src/lib/__tests__/upload-route-security.test.ts +120 -0
- package/src/lib/ai/__tests__/capability-resolver.test.ts +71 -0
- package/src/lib/ai/__tests__/claude-cli-adapter.test.ts +103 -0
- package/src/lib/ai/__tests__/provider-registry.test.ts +74 -0
- package/src/lib/pty/__tests__/ws-server-assistant.test.ts +7 -0
- package/.next/standalone/tower/.claude/rules/process-lifecycle.md +0 -31
- package/.next/standalone/tower/.claude/rules/security.md +0 -29
- package/.next/standalone/tower/.claude/rules/ui.md +0 -105
- package/.next/standalone/tower/AGENTS.md +0 -325
- package/.next/standalone/tower/CHANGELOG.md +0 -63
- package/.next/standalone/tower/CLAUDE.md +0 -33
- package/.next/standalone/tower/LICENSE +0 -21
- package/.next/standalone/tower/README.md +0 -214
- package/.next/standalone/tower/README.zh.md +0 -212
- package/.next/standalone/tower/bin/tower.mjs +0 -180
- package/.next/standalone/tower/docs/README.md +0 -51
- package/.next/standalone/tower/docs/ai/README.md +0 -91
- package/.next/standalone/tower/docs/ai/cli-abstraction-design.md +0 -398
- package/.next/standalone/tower/docs/ai/cli-abstraction-plan.md +0 -1345
- package/.next/standalone/tower/docs/assets-notes/README.md +0 -41
- package/.next/standalone/tower/docs/assistant/README.md +0 -48
- package/.next/standalone/tower/docs/board/README.md +0 -36
- package/.next/standalone/tower/docs/diagrams/tower-ai-architecture-en.html +0 -259
- package/.next/standalone/tower/docs/diagrams/tower-ai-architecture.html +0 -259
- package/.next/standalone/tower/docs/diagrams/tower-data-model-en.html +0 -344
- package/.next/standalone/tower/docs/diagrams/tower-data-model.html +0 -344
- package/.next/standalone/tower/docs/diagrams/tower-module-map-en.html +0 -176
- package/.next/standalone/tower/docs/diagrams/tower-module-map.html +0 -176
- package/.next/standalone/tower/docs/diagrams/tower-system-architecture-en.html +0 -336
- package/.next/standalone/tower/docs/diagrams/tower-system-architecture.html +0 -336
- package/.next/standalone/tower/docs/diagrams/tower-task-lifecycle-en.html +0 -251
- package/.next/standalone/tower/docs/diagrams/tower-task-lifecycle.html +0 -251
- package/.next/standalone/tower/docs/en/guide/architecture.md +0 -79
- package/.next/standalone/tower/docs/en/guide/diagrams.md +0 -24
- package/.next/standalone/tower/docs/en/guide/getting-started.md +0 -80
- package/.next/standalone/tower/docs/en/guide/introduction.md +0 -84
- package/.next/standalone/tower/docs/en/index.md +0 -36
- package/.next/standalone/tower/docs/en/modules/ai.md +0 -110
- package/.next/standalone/tower/docs/en/modules/assets-notes.md +0 -56
- package/.next/standalone/tower/docs/en/modules/assistant.md +0 -63
- package/.next/standalone/tower/docs/en/modules/board.md +0 -44
- package/.next/standalone/tower/docs/en/modules/git.md +0 -55
- package/.next/standalone/tower/docs/en/modules/i18n.md +0 -35
- package/.next/standalone/tower/docs/en/modules/mcp.md +0 -62
- package/.next/standalone/tower/docs/en/modules/missions.md +0 -47
- package/.next/standalone/tower/docs/en/modules/project.md +0 -57
- package/.next/standalone/tower/docs/en/modules/search.md +0 -49
- package/.next/standalone/tower/docs/en/modules/settings.md +0 -63
- package/.next/standalone/tower/docs/en/modules/task.md +0 -82
- package/.next/standalone/tower/docs/en/modules/terminal.md +0 -73
- package/.next/standalone/tower/docs/en/modules/workspace.md +0 -61
- package/.next/standalone/tower/docs/git/README.md +0 -42
- package/.next/standalone/tower/docs/guide/architecture.md +0 -29
- package/.next/standalone/tower/docs/guide/diagrams.md +0 -24
- package/.next/standalone/tower/docs/guide/getting-started.md +0 -80
- package/.next/standalone/tower/docs/guide/introduction.md +0 -84
- package/.next/standalone/tower/docs/i18n/README.md +0 -23
- package/.next/standalone/tower/docs/index.md +0 -36
- package/.next/standalone/tower/docs/mcp/README.md +0 -50
- package/.next/standalone/tower/docs/missions/README.md +0 -40
- package/.next/standalone/tower/docs/modules/ai.md +0 -104
- package/.next/standalone/tower/docs/modules/assets-notes.md +0 -68
- package/.next/standalone/tower/docs/modules/assistant.md +0 -74
- package/.next/standalone/tower/docs/modules/board.md +0 -56
- package/.next/standalone/tower/docs/modules/git.md +0 -61
- package/.next/standalone/tower/docs/modules/i18n.md +0 -38
- package/.next/standalone/tower/docs/modules/mcp.md +0 -68
- package/.next/standalone/tower/docs/modules/missions.md +0 -54
- package/.next/standalone/tower/docs/modules/project.md +0 -65
- package/.next/standalone/tower/docs/modules/search.md +0 -57
- package/.next/standalone/tower/docs/modules/settings.md +0 -72
- package/.next/standalone/tower/docs/modules/task.md +0 -82
- package/.next/standalone/tower/docs/modules/terminal.md +0 -70
- package/.next/standalone/tower/docs/modules/workspace.md +0 -68
- package/.next/standalone/tower/docs/package.json +0 -11
- package/.next/standalone/tower/docs/pnpm-lock.yaml +0 -1620
- package/.next/standalone/tower/docs/postcss.config.mjs +0 -2
- package/.next/standalone/tower/docs/project/README.md +0 -45
- package/.next/standalone/tower/docs/public/banner.png +0 -0
- package/.next/standalone/tower/docs/public/diagrams/tower-ai-architecture-en.html +0 -259
- package/.next/standalone/tower/docs/public/diagrams/tower-ai-architecture.html +0 -259
- package/.next/standalone/tower/docs/public/diagrams/tower-data-model-en.html +0 -344
- package/.next/standalone/tower/docs/public/diagrams/tower-data-model.html +0 -344
- package/.next/standalone/tower/docs/public/diagrams/tower-module-map-en.html +0 -176
- package/.next/standalone/tower/docs/public/diagrams/tower-module-map.html +0 -176
- package/.next/standalone/tower/docs/public/diagrams/tower-system-architecture-en.html +0 -336
- package/.next/standalone/tower/docs/public/diagrams/tower-system-architecture.html +0 -336
- package/.next/standalone/tower/docs/public/diagrams/tower-task-lifecycle-en.html +0 -251
- package/.next/standalone/tower/docs/public/diagrams/tower-task-lifecycle.html +0 -251
- package/.next/standalone/tower/docs/search/README.md +0 -37
- package/.next/standalone/tower/docs/settings/README.md +0 -43
- package/.next/standalone/tower/docs/task/README.md +0 -67
- package/.next/standalone/tower/docs/terminal/README.md +0 -58
- package/.next/standalone/tower/docs/workspace/README.md +0 -50
- package/.next/standalone/tower/docs/zh/guide/architecture.md +0 -29
- package/.next/standalone/tower/docs/zh/guide/diagrams.md +0 -24
- package/.next/standalone/tower/docs/zh/guide/getting-started.md +0 -80
- package/.next/standalone/tower/docs/zh/guide/introduction.md +0 -66
- package/.next/standalone/tower/docs/zh/index.md +0 -23
- package/.next/standalone/tower/docs/zh/modules/ai.md +0 -104
- package/.next/standalone/tower/docs/zh/modules/assets-notes.md +0 -68
- package/.next/standalone/tower/docs/zh/modules/assistant.md +0 -74
- package/.next/standalone/tower/docs/zh/modules/board.md +0 -56
- package/.next/standalone/tower/docs/zh/modules/git.md +0 -61
- package/.next/standalone/tower/docs/zh/modules/i18n.md +0 -38
- package/.next/standalone/tower/docs/zh/modules/mcp.md +0 -68
- package/.next/standalone/tower/docs/zh/modules/missions.md +0 -54
- package/.next/standalone/tower/docs/zh/modules/project.md +0 -65
- package/.next/standalone/tower/docs/zh/modules/search.md +0 -57
- package/.next/standalone/tower/docs/zh/modules/settings.md +0 -72
- package/.next/standalone/tower/docs/zh/modules/task.md +0 -82
- package/.next/standalone/tower/docs/zh/modules/terminal.md +0 -70
- package/.next/standalone/tower/docs/zh/modules/workspace.md +0 -68
- package/.next/standalone/tower/eslint.config.mjs +0 -18
- package/.next/standalone/tower/next.config.ts +0 -15
- package/.next/standalone/tower/package.json +0 -117
- package/.next/standalone/tower/playwright.config.ts +0 -28
- package/.next/standalone/tower/pnpm-lock.yaml +0 -10524
- package/.next/standalone/tower/pnpm-workspace.yaml +0 -6
- package/.next/standalone/tower/prisma/init-fts.ts +0 -23
- package/.next/standalone/tower/prisma/schema.prisma +0 -290
- package/.next/standalone/tower/prompts/backend-developer.md +0 -20
- package/.next/standalone/tower/prompts/code-reviewer.md +0 -19
- package/.next/standalone/tower/prompts/frontend-developer.md +0 -20
- package/.next/standalone/tower/prompts/product-manager.md +0 -16
- package/.next/standalone/tower/public/banner.jpg +0 -0
- package/.next/standalone/tower/public/logo.png +0 -0
- package/.next/standalone/tower/scripts/copy-monaco.js +0 -27
- package/.next/standalone/tower/scripts/init-db.ts +0 -74
- package/.next/standalone/tower/scripts/migrate-data.ts +0 -102
- package/.next/standalone/tower/server.js +0 -38
- package/.next/standalone/tower/src/app/favicon.ico +0 -0
- package/.next/standalone/tower/src/mcp/db.ts +0 -11
- package/.next/standalone/tower/src/mcp/index.ts +0 -15
- package/.next/standalone/tower/src/mcp/server.ts +0 -52
- package/.next/standalone/tower/src/mcp/tools/knowledge-tools.ts +0 -100
- package/.next/standalone/tower/src/mcp/tools/label-tools.ts +0 -70
- package/.next/standalone/tower/src/mcp/tools/note-asset-tools.ts +0 -271
- package/.next/standalone/tower/src/mcp/tools/project-tools.ts +0 -79
- package/.next/standalone/tower/src/mcp/tools/report-tools.ts +0 -214
- package/.next/standalone/tower/src/mcp/tools/search-tools.ts +0 -32
- package/.next/standalone/tower/src/mcp/tools/task-tools.ts +0 -250
- package/.next/standalone/tower/src/mcp/tools/terminal-tools.ts +0 -154
- package/.next/standalone/tower/src/mcp/tools/workspace-tools.ts +0 -73
- package/.next/standalone/tower/tests/e2e/chat-flow.spec.ts +0 -249
- package/.next/standalone/tower/tests/e2e/notes-assets.spec.ts +0 -287
- package/.next/standalone/tower/tests/e2e/search.spec.ts +0 -186
- package/.next/standalone/tower/tests/e2e/settings-flow.spec.ts +0 -250
- package/.next/standalone/tower/tests/e2e/settings.spec.ts +0 -261
- package/.next/standalone/tower/tests/e2e/smoke.spec.ts +0 -317
- package/.next/standalone/tower/tests/e2e/task-flow.spec.ts +0 -287
- package/.next/standalone/tower/tests/e2e/workbench.spec.ts +0 -307
- package/.next/standalone/tower/tests/setup.ts +0 -15
- package/.next/standalone/tower/tests/unit/actions/agent-actions.test.ts +0 -157
- package/.next/standalone/tower/tests/unit/actions/config-actions.test.ts +0 -170
- package/.next/standalone/tower/tests/unit/actions/file-actions.test.ts +0 -283
- package/.next/standalone/tower/tests/unit/actions/git-actions.test.ts +0 -34
- package/.next/standalone/tower/tests/unit/actions/preview-actions.test.ts +0 -177
- package/.next/standalone/tower/tests/unit/actions/search-actions.test.ts +0 -358
- package/.next/standalone/tower/tests/unit/actions/task-actions.test.ts +0 -173
- package/.next/standalone/tower/tests/unit/api/diff-route.test.ts +0 -9
- package/.next/standalone/tower/tests/unit/api/file-serving.test.ts +0 -88
- package/.next/standalone/tower/tests/unit/api/merge-route.test.ts +0 -9
- package/.next/standalone/tower/tests/unit/api/stream-persist-result.test.ts +0 -7
- package/.next/standalone/tower/tests/unit/api/stream-send-back.test.ts +0 -7
- package/.next/standalone/tower/tests/unit/components/asset-item.test.tsx +0 -105
- package/.next/standalone/tower/tests/unit/components/asset-list.test.tsx +0 -51
- package/.next/standalone/tower/tests/unit/components/assets/asset-upload.test.tsx +0 -70
- package/.next/standalone/tower/tests/unit/components/board-stats.test.tsx +0 -32
- package/.next/standalone/tower/tests/unit/components/category-filter.test.tsx +0 -58
- package/.next/standalone/tower/tests/unit/components/cli-adapter-tester.test.tsx +0 -160
- package/.next/standalone/tower/tests/unit/components/create-task-dialog.test.tsx +0 -162
- package/.next/standalone/tower/tests/unit/components/file-tree.test.tsx +0 -172
- package/.next/standalone/tower/tests/unit/components/note-card.test.tsx +0 -71
- package/.next/standalone/tower/tests/unit/components/note-editor.test.tsx +0 -39
- package/.next/standalone/tower/tests/unit/components/prompts-config.test.tsx +0 -200
- package/.next/standalone/tower/tests/unit/components/search-dialog.test.tsx +0 -223
- package/.next/standalone/tower/tests/unit/components/system-config.test.tsx +0 -32
- package/.next/standalone/tower/tests/unit/hooks/hook-registration.test.ts +0 -89
- package/.next/standalone/tower/tests/unit/hooks/stop-hook.test.ts +0 -108
- package/.next/standalone/tower/tests/unit/lib/asset-actions.test.ts +0 -257
- package/.next/standalone/tower/tests/unit/lib/file-utils.test.ts +0 -102
- package/.next/standalone/tower/tests/unit/lib/fs-security.test.ts +0 -34
- package/.next/standalone/tower/tests/unit/lib/fts.test.ts +0 -214
- package/.next/standalone/tower/tests/unit/lib/git-url.test.ts +0 -258
- package/.next/standalone/tower/tests/unit/lib/instrumentation.test.ts +0 -86
- package/.next/standalone/tower/tests/unit/lib/local-path-to-api-url.test.ts +0 -40
- package/.next/standalone/tower/tests/unit/lib/mime-magic.test.ts +0 -95
- package/.next/standalone/tower/tests/unit/lib/note-actions.test.ts +0 -296
- package/.next/standalone/tower/tests/unit/lib/preview-process-manager.test.ts +0 -107
- package/.next/standalone/tower/tests/unit/lib/process-manager.test.ts +0 -10
- package/.next/standalone/tower/tests/unit/lib/pty-session.test.ts +0 -171
- package/.next/standalone/tower/tests/unit/lib/search.test.ts +0 -289
- package/.next/standalone/tower/tests/unit/lib/session-store.test.ts +0 -111
- package/.next/standalone/tower/tests/unit/lib/utils.test.ts +0 -33
- package/.next/standalone/tower/tests/unit/lib/worktree.test.ts +0 -190
- package/.next/standalone/tower/tests/unit/mcp/identify-project.test.ts +0 -217
- package/.next/standalone/tower/tests/unit/mcp/manage-assets.test.ts +0 -199
- package/.next/standalone/tower/tests/unit/mcp/manage-notes.test.ts +0 -304
- package/.next/standalone/tower/tests/unit/mcp/search-tools.test.ts +0 -186
- package/.next/standalone/tower/tests/unit/missions-merge.test.ts +0 -43
- package/.next/standalone/tower/tower-studio-0.1.1.tgz +0 -0
- package/.next/standalone/tower/tsconfig.json +0 -34
- package/.next/standalone/tower/vitest.config.ts +0 -17
- package/.next/static/a6HR9cTK7s4aKfE5dWzjz/_buildManifest.js +0 -11
- package/.next/static/a6HR9cTK7s4aKfE5dWzjz/_clientMiddlewareManifest.js +0 -1
- package/.next/static/a6HR9cTK7s4aKfE5dWzjz/_ssgManifest.js +0 -1
- package/.next/static/chunks/00z-g3x93ngvn.js +0 -1
- package/.next/static/chunks/024pdsc27tjdd.js +0 -5
- package/.next/static/chunks/02dd83gbiv778.js +0 -1
- package/.next/static/chunks/03e.4ymu.j5wl.js +0 -1
- package/.next/static/chunks/03~yq9q893hmn.js +0 -1
- package/.next/static/chunks/05-b9qqm3av9~.js +0 -1
- package/.next/static/chunks/05~v02mkan5z..js +0 -1
- package/.next/static/chunks/0abtpeymj-58i.js +0 -1
- package/.next/static/chunks/0b6us7uq72u-d.js +0 -1
- package/.next/static/chunks/0drgc-oztq6o-.css +0 -1
- package/.next/static/chunks/0eaa2lmymh2fx.js +0 -1
- package/.next/static/chunks/0gb82g.6g90mn.js +0 -1
- package/.next/static/chunks/0gnm22yv~f54b.js +0 -1
- package/.next/static/chunks/0j9qriqni_r1..js +0 -2
- package/.next/static/chunks/0k.u8sxy~e469.js +0 -4
- package/.next/static/chunks/0lnhjf2a~jaco.js +0 -83
- package/.next/static/chunks/0lvd52mjiit6s.js +0 -1
- package/.next/static/chunks/0mq0uqbbbb1~2.js +0 -1
- package/.next/static/chunks/0neevhl_o1ozu.css +0 -2
- package/.next/static/chunks/0omj~p3uxkic-.js +0 -1
- package/.next/static/chunks/0t-gr6j-c65qb.js +0 -1
- package/.next/static/chunks/0t16ai99uv4j3.js +0 -1
- package/.next/static/chunks/0tcl81ybuob5i.js +0 -1
- package/.next/static/chunks/0uqimvsni_op~.js +0 -1
- package/.next/static/chunks/0wt3kws~_yr8z.js +0 -1
- package/.next/static/chunks/0xzdu87n_for1.js +0 -5
- package/.next/static/chunks/0y0tdl.rl6v1u.js +0 -1
- package/.next/static/chunks/0z2bzovqhl2f5.js +0 -1
- package/.next/static/chunks/0z4y0x1ifgy.e.js +0 -1
- package/.next/static/chunks/0z7bwntvfhxzi.js +0 -12
- package/.next/static/chunks/102ijqpvi0z-d.js +0 -1
- package/.next/static/chunks/10n23t.1hpb-1.js +0 -1
- package/.next/static/chunks/11_e3-j5gzbj4.js +0 -1
- package/.next/static/chunks/14.ims4y7osot.js +0 -1
- package/.next/static/chunks/14xzmrt5ly6gq.js +0 -31
- package/.next/static/chunks/151wr~6x8aclx.js +0 -1
- package/.next/static/chunks/169po6_~f3-d5.js +0 -1
- package/.next/static/chunks/16w-ap~msrwpj.js +0 -1
- package/.next/static/chunks/176n7f13ve~a9.js +0 -1
- package/.next/static/chunks/17oc2l.ekcs8b.css +0 -1
- package/.next/static/chunks/turbopack-0wjmrsi.z32s0.js +0 -1
- package/.next/static/media/4fa387ec64143e14-s.0q3udbd2bu5yp.woff2 +0 -0
- package/.next/static/media/7178b3e590c64307-s.11.cyxs5p-0z~.woff2 +0 -0
- package/.next/static/media/797e433ab948586e-s.p.0.q-h669a_dqa.woff2 +0 -0
- package/.next/static/media/8a480f0b521d4e75-s.06d3mdzz5bre_.woff2 +0 -0
- package/.next/static/media/apple-icon.16aocl-s-v2qz.png +0 -0
- package/.next/static/media/bbc41e54d2fcbd21-s.0gw~uztddq1df.woff2 +0 -0
- package/.next/static/media/caa3a2e1cccd8315-s.p.16t1db8_9y2o~.woff2 +0 -0
- package/.next/static/media/favicon.0y2d6j9cou~8p.ico +0 -0
- package/.next/static/media/icon0.0a6mkq6meyird.svg +0 -1
- package/.next/static/media/icon1.04ux133882seb.png +0 -0
- /package/{.next/standalone/tower/components.json → components.json} +0 -0
- /package/{.next/standalone/tower/postcss.config.mjs → postcss.config.mjs} +0 -0
- /package/{.next/standalone/tower/prisma → prisma}/prisma/dev.db +0 -0
- /package/{.next/standalone/tower/prisma → prisma}/seed.ts +0 -0
- /package/{.next/standalone/tower/docs/public → public}/banner.jpg +0 -0
- /package/{.next/standalone/tower/docs/public → public}/logo.png +0 -0
- /package/{.next/standalone/tower/public → public}/web-app-manifest-192x192.png +0 -0
- /package/{.next/standalone/tower/public → public}/web-app-manifest-512x512.png +0 -0
- /package/{.next/standalone/tower/scripts → scripts}/init-tower.ts +0 -0
- /package/{.next/standalone/tower/scripts → scripts}/post-tool-hook.js +0 -0
- /package/{.next/standalone/tower/scripts → scripts}/session-start-hook.js +0 -0
- /package/{.next/standalone/tower/scripts → scripts}/stop-hook.js +0 -0
- /package/{.next/standalone/tower/skills → skills}/tower/SKILL.md +0 -0
- /package/{.next/standalone/tower/src → src}/actions/agent-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/agent-config-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/ai-config-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/asset-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/assistant-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/cli-profile-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/config-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/file-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/git-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/label-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/note-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/onboarding-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/preview-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/project-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/prompt-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/report-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/search-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/search-code-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/task-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/actions/workspace-actions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/adapters/test/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/browse-fs/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/files/assets/[projectId]/[filename]/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/git/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/assets/[projectId]/[filename]/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/assets/reveal/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/assistant/chat/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/assistant/images/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/assistant/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/assistant/sessions/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/cache/[...segments]/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/hooks/install/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/hooks/session/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/hooks/stop/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/hooks/upload/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/notifications/pending/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/terminal/[taskId]/buffer/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/terminal/[taskId]/input/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/internal/terminal/[taskId]/start/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/tasks/[taskId]/diff/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/api/tasks/[taskId]/merge/route.ts +0 -0
- /package/{.next/standalone/tower/src → src}/app/apple-icon.png +0 -0
- /package/{.next/standalone/tower/docs/public → src/app}/favicon.ico +0 -0
- /package/{.next/standalone/tower/src → src}/app/globals.css +0 -0
- /package/{.next/standalone/tower/src → src}/app/icon0.svg +0 -0
- /package/{.next/standalone/tower/src → src}/app/icon1.png +0 -0
- /package/{.next/standalone/tower/src → src}/app/layout.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/manifest.json +0 -0
- /package/{.next/standalone/tower/src → src}/app/missions/missions-client.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/missions/page.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/onboarding/page.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/page.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/settings/page.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/workspaces/[workspaceId]/archive/archive-page-client.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/workspaces/[workspaceId]/archive/page.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/workspaces/[workspaceId]/assets/assets-page-client.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/workspaces/[workspaceId]/assets/page.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/workspaces/[workspaceId]/board-page-client.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/workspaces/[workspaceId]/notes/notes-page-client.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/workspaces/[workspaceId]/notes/page.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/workspaces/[workspaceId]/page.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/workspaces/[workspaceId]/projects/[projectId]/page.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/workspaces/[workspaceId]/tasks/[taskId]/page.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/workspaces/[workspaceId]/tasks/[taskId]/task-page-client.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/app/workspaces/page.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/assets/asset-item.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/assets/asset-list.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/assets/asset-upload.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/assets/image-lightbox.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/assets/text-preview-dialog.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/assistant/assistant-chat-bubble.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/assistant/assistant-chat.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/assistant/assistant-panel.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/assistant/assistant-provider.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/assistant/image-preview-modal.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/assistant/image-thumbnail-strip.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/board/board-column.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/board/board-filters.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/board/board-stats.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/board/column-tasks-dialog.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/board/create-task-dialog.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/board/kanban-board.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/board/project-tabs.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/board/task-card-context-menu.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/board/task-card.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/layout/app-sidebar.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/layout/folder-browser-dialog.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/layout/layout-client.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/layout/search-dialog.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/layout/sub-page-nav.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/layout/top-bar.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/missions/grid-layout-presets.ts +0 -0
- /package/{.next/standalone/tower/src → src}/components/missions/grid-preset-picker.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/missions/merge-missions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/components/missions/mission-card.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/missions/task-picker-dialog.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/notes/category-filter.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/notes/note-card.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/notes/note-editor.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/notes/note-list.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/notifications/notification-permission-banner.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/notifications/use-notification-listener.ts +0 -0
- /package/{.next/standalone/tower/src → src}/components/onboarding/guided-tour.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/onboarding/onboarding-wizard.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/onboarding/wizard-step-cli.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/onboarding/wizard-step-username.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/project/create-project-dialog.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/project/import-project-dialog.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/providers/theme-provider.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/repository/create-branch-dialog.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/repository/git-changes-panel.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/repository/git-log-panel.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/repository/git-stash-panel.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/repository/repo-sidebar.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/settings/cli-adapter-tester.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/settings/settings-page.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/code-editor.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/code-search.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/diff-editor.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/editor-git-panel.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/editor-tabs.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/execution-timeline.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/file-tree-context-menu.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/file-tree-node.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/file-tree.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/preview-panel.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/task-detail-panel.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/task-diff-view.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/task-file-changes.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/task-merge-confirm-dialog.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/task-metadata.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/task-notes-panel.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/task-overview-drawer.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/task-terminal.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/terminal-portal.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/task/types.ts +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/avatar.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/badge.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/button.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/card.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/command.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/dialog.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/dropdown-menu.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/empty-state.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/error-boundary.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/input-group.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/input.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/label.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/popover.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/scroll-area.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/segmented-control.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/select.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/separator.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/sheet.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/sonner.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/switch.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/tabs.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/textarea.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/toast.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/components/ui/tooltip.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/hooks/sse-event-reducer.ts +0 -0
- /package/{.next/standalone/tower/src → src}/hooks/use-assistant-chat.ts +0 -0
- /package/{.next/standalone/tower/src → src}/hooks/use-image-upload.ts +0 -0
- /package/{.next/standalone/tower/src → src}/instrumentation.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/ai/adapters/cli/claude-cli-adapter.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/ai/capability-resolver.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/ai/provider-registry.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/ai/providers/claude.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/ai/providers/index.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/ai/types.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/assistant-constants.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/assistant-message-converter.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/assistant-sessions.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/build-multimodal-prompt.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/claude-session.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/cli-test.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/config-defaults.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/config-reader.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/constants.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/db.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/diff-parser.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/execution-summary.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/file-serve-client.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/file-serve.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/file-utils.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/fs-security.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/fts.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/git-api.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/git-url.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/i18n/en.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/i18n/types.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/i18n/zh.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/i18n.tsx +0 -0
- /package/{.next/standalone/tower/src → src}/lib/init-tower.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/instrumentation-tasks.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/internal-api-guard.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/logger.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/mime-magic.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/platform.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/preview-process.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/pty/pty-session.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/pty/session-store.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/pty/ws-server.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/schemas.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/search.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/terminal-link-provider.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/tower-dir.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/utils.ts +0 -0
- /package/{.next/standalone/tower/src → src}/lib/worktree.ts +0 -0
- /package/{.next/standalone/tower/src → src}/stores/board-store.ts +0 -0
- /package/{.next/standalone/tower/src → src}/stores/task-execution-store.ts +0 -0
- /package/{.next/standalone/tower/src → src}/types/index.ts +0 -0
|
@@ -1,304 +0,0 @@
|
|
|
1
|
-
// @vitest-environment node
|
|
2
|
-
import { describe, it, expect, beforeAll, afterAll, afterEach } from "vitest";
|
|
3
|
-
import { PrismaClient } from "@prisma/client";
|
|
4
|
-
|
|
5
|
-
const testDb = new PrismaClient({
|
|
6
|
-
datasources: {
|
|
7
|
-
db: { url: process.env.DATABASE_URL ?? "file:./prisma/dev.db" },
|
|
8
|
-
},
|
|
9
|
-
});
|
|
10
|
-
|
|
11
|
-
let testWorkspaceId: string;
|
|
12
|
-
let testProjectId: string;
|
|
13
|
-
|
|
14
|
-
// Dynamically import handler after module exists
|
|
15
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
16
|
-
let manageNotesHandler: (args: any) => Promise<unknown>;
|
|
17
|
-
|
|
18
|
-
beforeAll(async () => {
|
|
19
|
-
await testDb.$connect();
|
|
20
|
-
|
|
21
|
-
// Ensure FTS5 table exists
|
|
22
|
-
await testDb.$executeRawUnsafe(`
|
|
23
|
-
CREATE VIRTUAL TABLE IF NOT EXISTS notes_fts
|
|
24
|
-
USING fts5(
|
|
25
|
-
note_id UNINDEXED,
|
|
26
|
-
title,
|
|
27
|
-
content,
|
|
28
|
-
tokenize='trigram case_sensitive 0'
|
|
29
|
-
)
|
|
30
|
-
`);
|
|
31
|
-
|
|
32
|
-
// Create test workspace and project
|
|
33
|
-
const workspace = await testDb.workspace.create({
|
|
34
|
-
data: { name: "Manage Notes MCP Test Workspace" },
|
|
35
|
-
});
|
|
36
|
-
testWorkspaceId = workspace.id;
|
|
37
|
-
|
|
38
|
-
const project = await testDb.project.create({
|
|
39
|
-
data: {
|
|
40
|
-
name: "Manage Notes MCP Test Project",
|
|
41
|
-
workspaceId: testWorkspaceId,
|
|
42
|
-
},
|
|
43
|
-
});
|
|
44
|
-
testProjectId = project.id;
|
|
45
|
-
|
|
46
|
-
// Import handler
|
|
47
|
-
const mod = await import("@/mcp/tools/note-asset-tools");
|
|
48
|
-
manageNotesHandler = mod.noteAssetTools.manage_notes.handler;
|
|
49
|
-
});
|
|
50
|
-
|
|
51
|
-
afterAll(async () => {
|
|
52
|
-
// Clean up test workspace (cascades to projects and notes)
|
|
53
|
-
await testDb.workspace.delete({ where: { id: testWorkspaceId } });
|
|
54
|
-
await testDb.$disconnect();
|
|
55
|
-
});
|
|
56
|
-
|
|
57
|
-
afterEach(async () => {
|
|
58
|
-
// Remove all notes for this project and clean FTS
|
|
59
|
-
const notes = await testDb.projectNote.findMany({
|
|
60
|
-
where: { projectId: testProjectId },
|
|
61
|
-
});
|
|
62
|
-
for (const note of notes) {
|
|
63
|
-
await testDb.$executeRawUnsafe("DELETE FROM notes_fts WHERE note_id = ?", note.id);
|
|
64
|
-
}
|
|
65
|
-
await testDb.projectNote.deleteMany({ where: { projectId: testProjectId } });
|
|
66
|
-
});
|
|
67
|
-
|
|
68
|
-
describe("manage_notes action=create", () => {
|
|
69
|
-
it("creates a note with default category 备忘", async () => {
|
|
70
|
-
const result = await manageNotesHandler({
|
|
71
|
-
action: "create",
|
|
72
|
-
projectId: testProjectId,
|
|
73
|
-
title: "My Test Note",
|
|
74
|
-
content: "Some content",
|
|
75
|
-
}) as { id: string; title: string; category: string; projectId: string };
|
|
76
|
-
|
|
77
|
-
expect(result.title).toBe("My Test Note");
|
|
78
|
-
expect(result.category).toBe("备忘");
|
|
79
|
-
expect(result.projectId).toBe(testProjectId);
|
|
80
|
-
expect(result.id).toBeDefined();
|
|
81
|
-
});
|
|
82
|
-
|
|
83
|
-
it("creates a note with custom category", async () => {
|
|
84
|
-
const result = await manageNotesHandler({
|
|
85
|
-
action: "create",
|
|
86
|
-
projectId: testProjectId,
|
|
87
|
-
title: "Env Note",
|
|
88
|
-
content: "env config",
|
|
89
|
-
category: "环境",
|
|
90
|
-
}) as { category: string };
|
|
91
|
-
|
|
92
|
-
expect(result.category).toBe("环境");
|
|
93
|
-
});
|
|
94
|
-
|
|
95
|
-
it("syncs note to FTS after create (action=search finds it)", async () => {
|
|
96
|
-
await manageNotesHandler({
|
|
97
|
-
action: "create",
|
|
98
|
-
projectId: testProjectId,
|
|
99
|
-
title: "Searchable API Note",
|
|
100
|
-
content: "This contains API config data",
|
|
101
|
-
});
|
|
102
|
-
|
|
103
|
-
const results = await manageNotesHandler({
|
|
104
|
-
action: "search",
|
|
105
|
-
projectId: testProjectId,
|
|
106
|
-
query: "API",
|
|
107
|
-
}) as Array<{ title: string }>;
|
|
108
|
-
|
|
109
|
-
expect(results.length).toBeGreaterThan(0);
|
|
110
|
-
expect(results[0].title).toBe("Searchable API Note");
|
|
111
|
-
});
|
|
112
|
-
|
|
113
|
-
it("throws when projectId is missing", async () => {
|
|
114
|
-
await expect(
|
|
115
|
-
manageNotesHandler({ action: "create", title: "No Project" })
|
|
116
|
-
).rejects.toThrow("projectId and title required");
|
|
117
|
-
});
|
|
118
|
-
|
|
119
|
-
it("throws when title is missing", async () => {
|
|
120
|
-
await expect(
|
|
121
|
-
manageNotesHandler({ action: "create", projectId: testProjectId })
|
|
122
|
-
).rejects.toThrow("projectId and title required");
|
|
123
|
-
});
|
|
124
|
-
});
|
|
125
|
-
|
|
126
|
-
describe("manage_notes action=update", () => {
|
|
127
|
-
it("updates note title", async () => {
|
|
128
|
-
const created = await manageNotesHandler({
|
|
129
|
-
action: "create",
|
|
130
|
-
projectId: testProjectId,
|
|
131
|
-
title: "Original Title",
|
|
132
|
-
content: "original content",
|
|
133
|
-
}) as { id: string };
|
|
134
|
-
|
|
135
|
-
const updated = await manageNotesHandler({
|
|
136
|
-
action: "update",
|
|
137
|
-
noteId: created.id,
|
|
138
|
-
title: "Updated Title",
|
|
139
|
-
}) as { title: string };
|
|
140
|
-
|
|
141
|
-
expect(updated.title).toBe("Updated Title");
|
|
142
|
-
});
|
|
143
|
-
|
|
144
|
-
it("syncs FTS after update", async () => {
|
|
145
|
-
const created = await manageNotesHandler({
|
|
146
|
-
action: "create",
|
|
147
|
-
projectId: testProjectId,
|
|
148
|
-
title: "Before Update",
|
|
149
|
-
content: "old content xyz",
|
|
150
|
-
}) as { id: string };
|
|
151
|
-
|
|
152
|
-
await manageNotesHandler({
|
|
153
|
-
action: "update",
|
|
154
|
-
noteId: created.id,
|
|
155
|
-
title: "After Update",
|
|
156
|
-
content: "new content foobar",
|
|
157
|
-
});
|
|
158
|
-
|
|
159
|
-
const results = await manageNotesHandler({
|
|
160
|
-
action: "search",
|
|
161
|
-
projectId: testProjectId,
|
|
162
|
-
query: "foobar",
|
|
163
|
-
}) as Array<{ note_id: string }>;
|
|
164
|
-
|
|
165
|
-
expect(results.length).toBeGreaterThan(0);
|
|
166
|
-
expect(results[0].note_id).toBe(created.id);
|
|
167
|
-
});
|
|
168
|
-
});
|
|
169
|
-
|
|
170
|
-
describe("manage_notes action=delete", () => {
|
|
171
|
-
it("deletes the note", async () => {
|
|
172
|
-
const created = await manageNotesHandler({
|
|
173
|
-
action: "create",
|
|
174
|
-
projectId: testProjectId,
|
|
175
|
-
title: "To Delete",
|
|
176
|
-
content: "delete me",
|
|
177
|
-
}) as { id: string };
|
|
178
|
-
|
|
179
|
-
const result = await manageNotesHandler({
|
|
180
|
-
action: "delete",
|
|
181
|
-
noteId: created.id,
|
|
182
|
-
}) as { deleted: boolean; noteId: string };
|
|
183
|
-
|
|
184
|
-
expect(result.deleted).toBe(true);
|
|
185
|
-
expect(result.noteId).toBe(created.id);
|
|
186
|
-
|
|
187
|
-
const found = await testDb.projectNote.findUnique({ where: { id: created.id } });
|
|
188
|
-
expect(found).toBeNull();
|
|
189
|
-
});
|
|
190
|
-
|
|
191
|
-
it("removes FTS entry after delete", async () => {
|
|
192
|
-
const created = await manageNotesHandler({
|
|
193
|
-
action: "create",
|
|
194
|
-
projectId: testProjectId,
|
|
195
|
-
title: "FTS Delete Test Note",
|
|
196
|
-
content: "unique-fts-delete-content",
|
|
197
|
-
}) as { id: string };
|
|
198
|
-
|
|
199
|
-
await manageNotesHandler({ action: "delete", noteId: created.id });
|
|
200
|
-
|
|
201
|
-
const rows = await testDb.$queryRawUnsafe<{ note_id: string }[]>(
|
|
202
|
-
"SELECT note_id FROM notes_fts WHERE note_id = ?",
|
|
203
|
-
created.id
|
|
204
|
-
);
|
|
205
|
-
expect(rows.length).toBe(0);
|
|
206
|
-
});
|
|
207
|
-
|
|
208
|
-
it("throws when noteId is missing", async () => {
|
|
209
|
-
await expect(
|
|
210
|
-
manageNotesHandler({ action: "delete" })
|
|
211
|
-
).rejects.toThrow("noteId required");
|
|
212
|
-
});
|
|
213
|
-
});
|
|
214
|
-
|
|
215
|
-
describe("manage_notes action=get", () => {
|
|
216
|
-
it("returns a note by ID", async () => {
|
|
217
|
-
const created = await manageNotesHandler({
|
|
218
|
-
action: "create",
|
|
219
|
-
projectId: testProjectId,
|
|
220
|
-
title: "Get Me",
|
|
221
|
-
content: "get content",
|
|
222
|
-
}) as { id: string };
|
|
223
|
-
|
|
224
|
-
const result = await manageNotesHandler({
|
|
225
|
-
action: "get",
|
|
226
|
-
noteId: created.id,
|
|
227
|
-
}) as { id: string; title: string };
|
|
228
|
-
|
|
229
|
-
expect(result.id).toBe(created.id);
|
|
230
|
-
expect(result.title).toBe("Get Me");
|
|
231
|
-
});
|
|
232
|
-
});
|
|
233
|
-
|
|
234
|
-
describe("manage_notes action=list", () => {
|
|
235
|
-
it("returns all notes for a project ordered by updatedAt desc", async () => {
|
|
236
|
-
await manageNotesHandler({
|
|
237
|
-
action: "create",
|
|
238
|
-
projectId: testProjectId,
|
|
239
|
-
title: "Note A",
|
|
240
|
-
content: "a",
|
|
241
|
-
});
|
|
242
|
-
await manageNotesHandler({
|
|
243
|
-
action: "create",
|
|
244
|
-
projectId: testProjectId,
|
|
245
|
-
title: "Note B",
|
|
246
|
-
content: "b",
|
|
247
|
-
});
|
|
248
|
-
|
|
249
|
-
const results = await manageNotesHandler({
|
|
250
|
-
action: "list",
|
|
251
|
-
projectId: testProjectId,
|
|
252
|
-
}) as Array<{ title: string }>;
|
|
253
|
-
|
|
254
|
-
expect(results.length).toBe(2);
|
|
255
|
-
// Most recently created should be first
|
|
256
|
-
expect(results[0].title).toBe("Note B");
|
|
257
|
-
});
|
|
258
|
-
|
|
259
|
-
it("filters by category when provided", async () => {
|
|
260
|
-
await manageNotesHandler({
|
|
261
|
-
action: "create",
|
|
262
|
-
projectId: testProjectId,
|
|
263
|
-
title: "Env Note",
|
|
264
|
-
content: "env",
|
|
265
|
-
category: "环境",
|
|
266
|
-
});
|
|
267
|
-
await manageNotesHandler({
|
|
268
|
-
action: "create",
|
|
269
|
-
projectId: testProjectId,
|
|
270
|
-
title: "Normal Note",
|
|
271
|
-
content: "normal",
|
|
272
|
-
});
|
|
273
|
-
|
|
274
|
-
const results = await manageNotesHandler({
|
|
275
|
-
action: "list",
|
|
276
|
-
projectId: testProjectId,
|
|
277
|
-
category: "环境",
|
|
278
|
-
}) as Array<{ title: string; category: string }>;
|
|
279
|
-
|
|
280
|
-
expect(results.length).toBe(1);
|
|
281
|
-
expect(results[0].title).toBe("Env Note");
|
|
282
|
-
expect(results[0].category).toBe("环境");
|
|
283
|
-
});
|
|
284
|
-
});
|
|
285
|
-
|
|
286
|
-
describe("manage_notes action=search", () => {
|
|
287
|
-
it("returns FTS results matching query", async () => {
|
|
288
|
-
await manageNotesHandler({
|
|
289
|
-
action: "create",
|
|
290
|
-
projectId: testProjectId,
|
|
291
|
-
title: "Database Schema",
|
|
292
|
-
content: "Describes the database schema and tables",
|
|
293
|
-
});
|
|
294
|
-
|
|
295
|
-
const results = await manageNotesHandler({
|
|
296
|
-
action: "search",
|
|
297
|
-
projectId: testProjectId,
|
|
298
|
-
query: "schema",
|
|
299
|
-
}) as Array<{ title: string }>;
|
|
300
|
-
|
|
301
|
-
expect(results.length).toBeGreaterThan(0);
|
|
302
|
-
expect(results[0].title).toBe("Database Schema");
|
|
303
|
-
});
|
|
304
|
-
});
|
|
@@ -1,186 +0,0 @@
|
|
|
1
|
-
// @vitest-environment node
|
|
2
|
-
import { describe, it, expect, beforeAll, afterAll, afterEach } from "vitest";
|
|
3
|
-
import { PrismaClient } from "@prisma/client";
|
|
4
|
-
import { syncNoteToFts } from "@/lib/fts";
|
|
5
|
-
import { z } from "zod";
|
|
6
|
-
|
|
7
|
-
const testDb = new PrismaClient({
|
|
8
|
-
datasources: {
|
|
9
|
-
db: { url: process.env.DATABASE_URL ?? "file:./prisma/dev.db" },
|
|
10
|
-
},
|
|
11
|
-
});
|
|
12
|
-
|
|
13
|
-
let testWorkspaceId: string;
|
|
14
|
-
let testProjectId: string;
|
|
15
|
-
|
|
16
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
17
|
-
let searchHandler: (args: any) => Promise<unknown>;
|
|
18
|
-
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
19
|
-
let searchSchema: z.ZodObject<any>;
|
|
20
|
-
|
|
21
|
-
beforeAll(async () => {
|
|
22
|
-
await testDb.$connect();
|
|
23
|
-
|
|
24
|
-
// Ensure FTS5 table exists
|
|
25
|
-
await testDb.$executeRawUnsafe(`
|
|
26
|
-
CREATE VIRTUAL TABLE IF NOT EXISTS notes_fts
|
|
27
|
-
USING fts5(
|
|
28
|
-
note_id UNINDEXED,
|
|
29
|
-
title,
|
|
30
|
-
content,
|
|
31
|
-
tokenize='trigram case_sensitive 0'
|
|
32
|
-
)
|
|
33
|
-
`);
|
|
34
|
-
|
|
35
|
-
// Create test workspace and project
|
|
36
|
-
const workspace = await testDb.workspace.create({
|
|
37
|
-
data: { name: "Search Tools MCP Test WS" },
|
|
38
|
-
});
|
|
39
|
-
testWorkspaceId = workspace.id;
|
|
40
|
-
|
|
41
|
-
const project = await testDb.project.create({
|
|
42
|
-
data: {
|
|
43
|
-
name: "Search Tools MCP Test Proj",
|
|
44
|
-
workspaceId: testWorkspaceId,
|
|
45
|
-
},
|
|
46
|
-
});
|
|
47
|
-
testProjectId = project.id;
|
|
48
|
-
|
|
49
|
-
// Dynamically import search-tools
|
|
50
|
-
const mod = await import("@/mcp/tools/search-tools");
|
|
51
|
-
searchHandler = mod.searchTools.search.handler;
|
|
52
|
-
searchSchema = mod.searchTools.search.schema;
|
|
53
|
-
});
|
|
54
|
-
|
|
55
|
-
afterAll(async () => {
|
|
56
|
-
await testDb.workspace.delete({ where: { id: testWorkspaceId } });
|
|
57
|
-
await testDb.$disconnect();
|
|
58
|
-
});
|
|
59
|
-
|
|
60
|
-
afterEach(async () => {
|
|
61
|
-
// Clean up notes + FTS
|
|
62
|
-
const notes = await testDb.projectNote.findMany({
|
|
63
|
-
where: { projectId: testProjectId },
|
|
64
|
-
});
|
|
65
|
-
for (const note of notes) {
|
|
66
|
-
await testDb.$executeRawUnsafe("DELETE FROM notes_fts WHERE note_id = ?", note.id);
|
|
67
|
-
}
|
|
68
|
-
await testDb.projectNote.deleteMany({ where: { projectId: testProjectId } });
|
|
69
|
-
|
|
70
|
-
// Clean up assets
|
|
71
|
-
await testDb.projectAsset.deleteMany({ where: { projectId: testProjectId } });
|
|
72
|
-
});
|
|
73
|
-
|
|
74
|
-
describe("search tool schema validation", () => {
|
|
75
|
-
it("accepts category 'note'", () => {
|
|
76
|
-
expect(() => searchSchema.parse({ query: "test", category: "note" })).not.toThrow();
|
|
77
|
-
});
|
|
78
|
-
|
|
79
|
-
it("accepts category 'asset'", () => {
|
|
80
|
-
expect(() => searchSchema.parse({ query: "test", category: "asset" })).not.toThrow();
|
|
81
|
-
});
|
|
82
|
-
|
|
83
|
-
it("accepts category 'all'", () => {
|
|
84
|
-
expect(() => searchSchema.parse({ query: "test", category: "all" })).not.toThrow();
|
|
85
|
-
});
|
|
86
|
-
|
|
87
|
-
it("accepts existing category 'task'", () => {
|
|
88
|
-
expect(() => searchSchema.parse({ query: "test", category: "task" })).not.toThrow();
|
|
89
|
-
});
|
|
90
|
-
|
|
91
|
-
it("accepts existing category 'project'", () => {
|
|
92
|
-
expect(() => searchSchema.parse({ query: "test", category: "project" })).not.toThrow();
|
|
93
|
-
});
|
|
94
|
-
|
|
95
|
-
it("accepts existing category 'repository'", () => {
|
|
96
|
-
expect(() => searchSchema.parse({ query: "test", category: "repository" })).not.toThrow();
|
|
97
|
-
});
|
|
98
|
-
|
|
99
|
-
it("rejects invalid category", () => {
|
|
100
|
-
expect(() => searchSchema.parse({ query: "test", category: "invalid" })).toThrow();
|
|
101
|
-
});
|
|
102
|
-
});
|
|
103
|
-
|
|
104
|
-
describe("search tool handler - note category", () => {
|
|
105
|
-
it("returns results matching note title", async () => {
|
|
106
|
-
const note = await testDb.projectNote.create({
|
|
107
|
-
data: {
|
|
108
|
-
title: "MCP API Documentation",
|
|
109
|
-
content: "This covers MCP API details",
|
|
110
|
-
projectId: testProjectId,
|
|
111
|
-
},
|
|
112
|
-
});
|
|
113
|
-
await syncNoteToFts(testDb, { id: note.id, title: note.title, content: note.content });
|
|
114
|
-
|
|
115
|
-
const results = await searchHandler({ query: "MCP API", category: "note" }) as Array<{ id: string; type: string }>;
|
|
116
|
-
expect(results.length).toBeGreaterThan(0);
|
|
117
|
-
const found = results.find((r) => r.id === note.id);
|
|
118
|
-
expect(found).toBeDefined();
|
|
119
|
-
expect(found?.type).toBe("note");
|
|
120
|
-
});
|
|
121
|
-
|
|
122
|
-
it("returns empty array for query with no matches", async () => {
|
|
123
|
-
const results = await searchHandler({ query: "zzz_no_match_12345", category: "note" }) as unknown[];
|
|
124
|
-
expect(Array.isArray(results)).toBe(true);
|
|
125
|
-
expect(results).toEqual([]);
|
|
126
|
-
});
|
|
127
|
-
});
|
|
128
|
-
|
|
129
|
-
describe("search tool handler - asset category", () => {
|
|
130
|
-
it("returns results matching asset filename", async () => {
|
|
131
|
-
await testDb.projectAsset.create({
|
|
132
|
-
data: {
|
|
133
|
-
filename: "mcp-asset-readme.md",
|
|
134
|
-
path: "/data/assets/mcp-asset-readme.md",
|
|
135
|
-
size: 1024,
|
|
136
|
-
mimeType: "text/markdown",
|
|
137
|
-
description: "asset for MCP test",
|
|
138
|
-
projectId: testProjectId,
|
|
139
|
-
},
|
|
140
|
-
});
|
|
141
|
-
|
|
142
|
-
const results = await searchHandler({ query: "mcp-asset-readme", category: "asset" }) as Array<{ title: string; type: string }>;
|
|
143
|
-
expect(results.length).toBeGreaterThan(0);
|
|
144
|
-
const found = results.find((r) => r.title === "mcp-asset-readme.md");
|
|
145
|
-
expect(found).toBeDefined();
|
|
146
|
-
expect(found?.type).toBe("asset");
|
|
147
|
-
});
|
|
148
|
-
|
|
149
|
-
it("returns empty array for asset query with no matches", async () => {
|
|
150
|
-
const results = await searchHandler({ query: "zzz_no_match_asset_12345", category: "asset" }) as unknown[];
|
|
151
|
-
expect(Array.isArray(results)).toBe(true);
|
|
152
|
-
expect(results).toEqual([]);
|
|
153
|
-
});
|
|
154
|
-
});
|
|
155
|
-
|
|
156
|
-
describe("search tool handler - all category", () => {
|
|
157
|
-
it("returns results from multiple types", async () => {
|
|
158
|
-
// Create note with common search term
|
|
159
|
-
const note = await testDb.projectNote.create({
|
|
160
|
-
data: {
|
|
161
|
-
title: "mcpall-search-note",
|
|
162
|
-
content: "mcpall content here",
|
|
163
|
-
projectId: testProjectId,
|
|
164
|
-
},
|
|
165
|
-
});
|
|
166
|
-
await syncNoteToFts(testDb, { id: note.id, title: note.title, content: note.content });
|
|
167
|
-
|
|
168
|
-
// Create asset with same term
|
|
169
|
-
await testDb.projectAsset.create({
|
|
170
|
-
data: {
|
|
171
|
-
filename: "mcpall-file.txt",
|
|
172
|
-
path: "/data/assets/mcpall-file.txt",
|
|
173
|
-
size: 256,
|
|
174
|
-
mimeType: "text/plain",
|
|
175
|
-
description: "",
|
|
176
|
-
projectId: testProjectId,
|
|
177
|
-
},
|
|
178
|
-
});
|
|
179
|
-
|
|
180
|
-
const results = await searchHandler({ query: "mcpall", category: "all" }) as Array<{ type: string }>;
|
|
181
|
-
expect(results.length).toBeGreaterThan(0);
|
|
182
|
-
|
|
183
|
-
const types = new Set(results.map((r) => r.type));
|
|
184
|
-
expect(types.size).toBeGreaterThan(1);
|
|
185
|
-
});
|
|
186
|
-
});
|
|
@@ -1,43 +0,0 @@
|
|
|
1
|
-
// @vitest-environment node
|
|
2
|
-
import { describe, it, expect } from "vitest";
|
|
3
|
-
import { mergeMissions } from "@/components/missions/merge-missions";
|
|
4
|
-
|
|
5
|
-
describe("mergeMissions", () => {
|
|
6
|
-
it("retains existing order when no changes", () => {
|
|
7
|
-
const prev = [{ executionId: "A" }, { executionId: "B" }, { executionId: "C" }];
|
|
8
|
-
const fresh = [{ executionId: "A" }, { executionId: "B" }, { executionId: "C" }];
|
|
9
|
-
const { merged, goneIds } = mergeMissions({ prev, fresh, removingIds: new Set() });
|
|
10
|
-
expect(merged.map((e) => e.executionId)).toEqual(["A", "B", "C"]);
|
|
11
|
-
expect(goneIds).toEqual([]);
|
|
12
|
-
});
|
|
13
|
-
|
|
14
|
-
it("appends new executions at end", () => {
|
|
15
|
-
const prev = [{ executionId: "A" }, { executionId: "B" }];
|
|
16
|
-
const fresh = [{ executionId: "A" }, { executionId: "B" }, { executionId: "C" }];
|
|
17
|
-
const { merged } = mergeMissions({ prev, fresh, removingIds: new Set() });
|
|
18
|
-
expect(merged.map((e) => e.executionId)).toEqual(["A", "B", "C"]);
|
|
19
|
-
});
|
|
20
|
-
|
|
21
|
-
it("detects gone executions", () => {
|
|
22
|
-
const prev = [{ executionId: "A" }, { executionId: "B" }, { executionId: "C" }];
|
|
23
|
-
const fresh = [{ executionId: "A" }, { executionId: "C" }];
|
|
24
|
-
const { goneIds } = mergeMissions({ prev, fresh, removingIds: new Set() });
|
|
25
|
-
expect(goneIds).toEqual(["B"]);
|
|
26
|
-
});
|
|
27
|
-
|
|
28
|
-
it("preserves removing cards (not counted as gone)", () => {
|
|
29
|
-
const prev = [{ executionId: "A" }, { executionId: "B" }];
|
|
30
|
-
const fresh = [{ executionId: "A" }];
|
|
31
|
-
const { merged, goneIds } = mergeMissions({ prev, fresh, removingIds: new Set(["B"]) });
|
|
32
|
-
expect(merged.map((e) => e.executionId)).toEqual(["A", "B"]);
|
|
33
|
-
expect(goneIds).toEqual([]);
|
|
34
|
-
});
|
|
35
|
-
|
|
36
|
-
it("handles empty prev", () => {
|
|
37
|
-
const prev: { executionId: string }[] = [];
|
|
38
|
-
const fresh = [{ executionId: "A" }, { executionId: "B" }];
|
|
39
|
-
const { merged, goneIds } = mergeMissions({ prev, fresh, removingIds: new Set() });
|
|
40
|
-
expect(merged.map((e) => e.executionId)).toEqual(["A", "B"]);
|
|
41
|
-
expect(goneIds).toEqual([]);
|
|
42
|
-
});
|
|
43
|
-
});
|
|
Binary file
|
|
@@ -1,34 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"compilerOptions": {
|
|
3
|
-
"target": "ES2017",
|
|
4
|
-
"lib": ["dom", "dom.iterable", "esnext"],
|
|
5
|
-
"allowJs": true,
|
|
6
|
-
"skipLibCheck": true,
|
|
7
|
-
"strict": true,
|
|
8
|
-
"noEmit": true,
|
|
9
|
-
"esModuleInterop": true,
|
|
10
|
-
"module": "esnext",
|
|
11
|
-
"moduleResolution": "bundler",
|
|
12
|
-
"resolveJsonModule": true,
|
|
13
|
-
"isolatedModules": true,
|
|
14
|
-
"jsx": "react-jsx",
|
|
15
|
-
"incremental": true,
|
|
16
|
-
"plugins": [
|
|
17
|
-
{
|
|
18
|
-
"name": "next"
|
|
19
|
-
}
|
|
20
|
-
],
|
|
21
|
-
"paths": {
|
|
22
|
-
"@/*": ["./src/*"]
|
|
23
|
-
}
|
|
24
|
-
},
|
|
25
|
-
"include": [
|
|
26
|
-
"next-env.d.ts",
|
|
27
|
-
"**/*.ts",
|
|
28
|
-
"**/*.tsx",
|
|
29
|
-
".next/types/**/*.ts",
|
|
30
|
-
".next/dev/types/**/*.ts",
|
|
31
|
-
"**/*.mts"
|
|
32
|
-
],
|
|
33
|
-
"exclude": ["node_modules"]
|
|
34
|
-
}
|
|
@@ -1,17 +0,0 @@
|
|
|
1
|
-
import { defineConfig } from "vitest/config";
|
|
2
|
-
import react from "@vitejs/plugin-react";
|
|
3
|
-
import path from "path";
|
|
4
|
-
|
|
5
|
-
export default defineConfig({
|
|
6
|
-
plugins: [react()],
|
|
7
|
-
test: {
|
|
8
|
-
environment: "jsdom",
|
|
9
|
-
setupFiles: ["./tests/setup.ts"],
|
|
10
|
-
include: ["tests/**/*.test.{ts,tsx}", "src/**/__tests__/**/*.test.{ts,tsx}"],
|
|
11
|
-
},
|
|
12
|
-
resolve: {
|
|
13
|
-
alias: {
|
|
14
|
-
"@": path.resolve(__dirname, "./src"),
|
|
15
|
-
},
|
|
16
|
-
},
|
|
17
|
-
});
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
self.__MIDDLEWARE_MATCHERS = [];self.__MIDDLEWARE_MATCHERS_CB && self.__MIDDLEWARE_MATCHERS_CB()
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
self.__SSG_MANIFEST=new Set([]);self.__SSG_MANIFEST_CB&&self.__SSG_MANIFEST_CB()
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
(globalThis.TURBOPACK||(globalThis.TURBOPACK=[])).push(["object"==typeof document?document.currentScript:void 0,522182,e=>{"use strict";let t=(0,e.i(258114).default)("user",[["path",{d:"M19 21v-2a4 4 0 0 0-4-4H9a4 4 0 0 0-4 4v2",key:"975kel"}],["circle",{cx:"12",cy:"7",r:"4",key:"17ys0d"}]]);e.s(["User",0,t],522182)},597736,e=>{"use strict";var t=e.i(794097);e.s(["ChevronRight",()=>t.default])},986633,e=>{"use strict";let t=(0,e.i(258114).default)("square",[["rect",{width:"18",height:"18",x:"3",y:"3",rx:"2",key:"afitv7"}]]);e.s(["Square",0,t],986633)},86630,622622,e=>{"use strict";var t=e.i(258114);let s=(0,t.default)("clipboard-list",[["rect",{width:"8",height:"4",x:"8",y:"2",rx:"1",ry:"1",key:"tgr4d6"}],["path",{d:"M16 4h2a2 2 0 0 1 2 2v14a2 2 0 0 1-2 2H6a2 2 0 0 1-2-2V6a2 2 0 0 1 2-2h2",key:"116196"}],["path",{d:"M12 11h4",key:"1jrz19"}],["path",{d:"M12 16h4",key:"n85exb"}],["path",{d:"M8 11h.01",key:"1dfujw"}],["path",{d:"M8 16h.01",key:"18s6g9"}]]);e.s(["ClipboardList",0,s],86630);let a=(0,t.default)("copy",[["rect",{width:"14",height:"14",x:"8",y:"8",rx:"2",ry:"2",key:"17jyea"}],["path",{d:"M4 16c-1.1 0-2-.9-2-2V4c0-1.1.9-2 2-2h10c1.1 0 2 .9 2 2",key:"zix9uf"}]]);e.s(["Copy",0,a],622622)},543228,e=>{"use strict";var t=e.i(430989),s=e.i(641338),a=e.i(38431),r=e.i(86630),l=e.i(258114);let i=(0,l.default)("folder-plus",[["path",{d:"M12 10v6",key:"1bos4e"}],["path",{d:"M9 13h6",key:"1uhe8q"}],["path",{d:"M20 20a2 2 0 0 0 2-2V8a2 2 0 0 0-2-2h-7.9a2 2 0 0 1-1.69-.9L9.6 3.9A2 2 0 0 0 7.93 3H4a2 2 0 0 0-2 2v13a2 2 0 0 0 2 2Z",key:"1kt360"}]]);var n=e.i(924609),o=e.i(811163);let d=(0,l.default)("send-horizontal",[["path",{d:"M3.714 3.048a.498.498 0 0 0-.683.627l2.843 7.627a2 2 0 0 1 0 1.396l-2.842 7.627a.498.498 0 0 0 .682.627l18-8.5a.5.5 0 0 0 0-.904z",key:"117uat"}],["path",{d:"M6 12h16",key:"s4cdu5"}]]);var c=e.i(986633);let u=(0,l.default)("trending-up",[["path",{d:"M16 7h6v6",key:"box55l"}],["path",{d:"m22 7-8.5 8.5-5-5L2 17",key:"1t1m79"}]]);var m=e.i(24887),x=e.i(779574),h=e.i(744838),p=e.i(349369),f=e.i(599104),g=e.i(201853),b=e.i(680538),v=e.i(698300),j=e.i(597736),y=e.i(622622);let N=(0,l.default)("image-off",[["line",{x1:"2",x2:"22",y1:"2",y2:"22",key:"a6p6uj"}],["path",{d:"M10.41 10.41a2 2 0 1 1-2.83-2.83",key:"1bzlo9"}],["line",{x1:"13.5",x2:"6",y1:"13.5",y2:"21",key:"1q0aeu"}],["line",{x1:"18",x2:"21",y1:"12",y2:"15",key:"5mozeu"}],["path",{d:"M3.59 3.59A1.99 1.99 0 0 0 3 5v14a2 2 0 0 0 2 2h14c.55 0 1.052-.22 1.41-.59",key:"mmje98"}],["path",{d:"M21 15V5a2 2 0 0 0-2-2H9",key:"43el77"}]]);var k=e.i(522182);function w({text:e,className:a}){let[r,l]=(0,s.useState)(!1),{t:i}=(0,p.useI18n)(),n=async t=>{t.stopPropagation();try{if(navigator.clipboard)await navigator.clipboard.writeText(e);else{let t=document.createElement("textarea");t.value=e,document.body.appendChild(t),t.select(),document.execCommand("copy"),document.body.removeChild(t)}l(!0),setTimeout(()=>l(!1),1500)}catch{}};return(0,t.jsx)(h.Button,{variant:"ghost",size:"icon-xs",onClick:n,className:`text-muted-foreground ${a??""}`,"aria-label":i("assistant.copy"),children:r?(0,t.jsx)(v.Check,{className:"h-3 w-3 text-emerald-400"}):(0,t.jsx)(y.Copy,{className:"h-3 w-3"})})}function C({filename:e,onPreview:a}){let{t:r}=(0,p.useI18n)(),[l,i]=(0,s.useState)(!1),n=`/api/internal/cache/${e}`;return l?(0,t.jsx)("div",{className:"size-16 rounded-md bg-muted flex items-center justify-center",title:r("assistant.brokenImage"),children:(0,t.jsx)(N,{className:"size-4 text-muted-foreground"})}):(0,t.jsx)("button",{type:"button",className:"relative size-16 rounded-md overflow-hidden bg-muted shrink-0",onClick:()=>a?.(n),children:(0,t.jsx)("img",{src:n,alt:"",loading:"lazy",className:"size-16 rounded-md object-cover cursor-pointer hover:opacity-80 transition-opacity",onError:()=>i(!0)})})}function z({content:e,imageFilenames:s,onImagePreview:a}){return(0,t.jsxs)("div",{className:"flex justify-end gap-2 animate-in fade-in slide-in-from-bottom-2 duration-200","aria-label":"You",children:[(0,t.jsxs)("div",{className:"bg-primary text-primary-foreground max-w-[80%] rounded-2xl rounded-br-sm px-3 py-2 text-sm whitespace-pre-wrap break-words",children:[s&&s.length>0&&(0,t.jsx)("div",{className:"flex flex-wrap gap-1.5 mb-2",style:{maxWidth:"calc(4 * 64px + 3 * 6px)"},children:s.map(e=>(0,t.jsx)(C,{filename:e,onPreview:a},e))}),e]}),(0,t.jsx)("div",{className:"flex size-7 shrink-0 items-center justify-center rounded-full bg-muted mt-1",children:(0,t.jsx)(k.User,{className:"size-3.5 text-muted-foreground"})})]})}function U({content:e}){return(0,t.jsxs)("div",{className:"group/bubble flex justify-start gap-2 animate-in fade-in slide-in-from-bottom-2 duration-200","aria-label":"Assistant",children:[(0,t.jsx)("div",{className:"flex size-7 shrink-0 items-center justify-center rounded-full bg-primary/10 mt-1",children:(0,t.jsx)(a.Bot,{className:"size-3.5 text-primary"})}),(0,t.jsxs)("div",{className:"max-w-[85%]",children:[(0,t.jsx)("div",{className:"bg-muted text-foreground rounded-2xl rounded-bl-sm px-3 py-2",children:(0,t.jsx)("div",{className:"prose prose-sm dark:prose-invert max-w-none",children:(0,t.jsx)(g.default,{remarkPlugins:[b.default],components:{code:({inline:e,className:s,children:a,...r})=>e?(0,t.jsx)("code",{className:"bg-muted/80 px-1 rounded text-[13px] font-mono",...r,children:a}):(0,t.jsx)("code",{className:"bg-muted rounded-md p-3 font-mono text-[13px] overflow-x-auto block",...r,children:a})},children:e})})}),(0,t.jsx)("div",{className:"flex items-center gap-0.5 mt-0.5 opacity-0 group-hover/bubble:opacity-100 transition-opacity",children:(0,t.jsx)(w,{text:e})})]})]})}function M(){let{t:e}=(0,p.useI18n)();return(0,t.jsxs)("div",{className:"flex justify-start gap-2 animate-in fade-in slide-in-from-bottom-2 duration-200",role:"status","aria-live":"polite","aria-label":e("assistant.thinking"),children:[(0,t.jsx)("div",{className:"flex size-7 shrink-0 items-center justify-center rounded-full bg-primary/10 mt-1",children:(0,t.jsx)(a.Bot,{className:"size-3.5 text-primary"})}),(0,t.jsx)("div",{className:"bg-muted/50 text-muted-foreground max-w-[200px] rounded-2xl rounded-bl-sm px-3 py-2",children:(0,t.jsxs)("div",{className:"flex items-center gap-1",children:[(0,t.jsx)("span",{className:"size-1.5 rounded-full bg-primary/60 animate-pulse",style:{animationDelay:"0ms"}}),(0,t.jsx)("span",{className:"size-1.5 rounded-full bg-primary/60 animate-pulse",style:{animationDelay:"150ms"}}),(0,t.jsx)("span",{className:"size-1.5 rounded-full bg-primary/60 animate-pulse",style:{animationDelay:"300ms"}})]})})]})}function R({content:e,toolName:a}){let[r,l]=(0,s.useState)(!1),{t:i}=(0,p.useI18n)(),n=a??i("assistant.toolLabel");return(0,t.jsxs)("div",{className:"group/tool flex justify-start gap-2 animate-in fade-in slide-in-from-bottom-2 duration-200",children:[(0,t.jsx)("div",{className:"size-7 shrink-0"}),(0,t.jsxs)("div",{className:"bg-muted/60 border border-border max-w-[90%] rounded-lg px-3 py-1",children:[(0,t.jsxs)("div",{className:"flex items-center gap-1.5 w-full text-left py-0.5 cursor-pointer",onClick:()=>l(e=>!e),role:"button","aria-expanded":r,"aria-label":i("assistant.expandTool"),children:[(0,t.jsx)(j.ChevronRight,{className:`h-3.5 w-3.5 text-muted-foreground shrink-0 transition-transform duration-150 ${r?"rotate-90":""}`}),(0,t.jsx)("span",{className:"text-xs font-semibold text-foreground truncate",children:n}),(0,t.jsx)("span",{className:"flex-1"}),(0,t.jsx)(w,{text:e,className:"opacity-0 group-hover/tool:opacity-100"}),(0,t.jsx)("span",{className:"bg-muted text-muted-foreground text-[10px] px-1.5 rounded shrink-0",children:i("assistant.toolLabel")})]}),r&&(0,t.jsx)("div",{className:"mt-1 border-t border-border pt-1",children:(0,t.jsx)("pre",{className:"font-mono text-[12px] max-h-[200px] overflow-y-auto whitespace-pre-wrap break-all text-muted-foreground",children:e})})]})]})}function L({message:e,onImagePreview:s}){switch(e.role){case"user":return(0,t.jsx)(z,{content:e.content,imageFilenames:e.imageFilenames,onImagePreview:s});case"assistant":return(0,t.jsx)(U,{content:e.content});case"thinking":return(0,t.jsx)(M,{});case"tool":return(0,t.jsx)(R,{content:e.content,toolName:e.toolName});default:return null}}var I=e.i(993453),S=e.i(621994);function P({pendingImages:e,onRemove:s,onPreview:a}){let{t:r}=(0,p.useI18n)();return 0===e.length?null:(0,t.jsx)("div",{className:"flex flex-row gap-2 px-4 pt-2 pb-0",children:e.map(e=>(0,t.jsxs)("div",{className:`relative h-12 w-12 shrink-0 rounded-md overflow-hidden ring-1 ${"error"===e.status?"ring-destructive":"ring-border"}`,children:[(0,t.jsx)("button",{type:"button",className:`h-full w-full p-0 border-0 bg-transparent ${"uploading"===e.status?"cursor-wait":"cursor-pointer"}`,disabled:"uploading"===e.status,"aria-label":r("assistant.previewImage"),onClick:()=>a(e),children:(0,t.jsx)("img",{src:e.blobUrl,className:"h-full w-full object-cover",alt:""})}),"uploading"===e.status&&(0,t.jsxs)("div",{className:"absolute inset-0 flex flex-col justify-end bg-black/30",children:[(0,t.jsx)("div",{role:"progressbar","aria-valuenow":e.progress,"aria-valuemin":0,"aria-valuemax":100,className:"h-1 bg-primary transition-all duration-150",style:{width:`${e.progress}%`}}),(0,t.jsxs)("span",{className:"absolute inset-0 flex items-center justify-center text-[10px] font-semibold text-white",children:[e.progress,"%"]})]}),"error"===e.status&&(0,t.jsx)("div",{className:"absolute inset-0 flex items-center justify-center bg-black/30",children:(0,t.jsx)(I.AlertCircle,{className:"h-4 w-4 text-white"})}),(0,t.jsx)("button",{className:"absolute top-1 right-1 h-4 w-4 rounded-full bg-black/60 text-white flex items-center justify-center hover:bg-black/80 transition-colors","aria-label":r("assistant.removeImage"),title:"error"===e.status?r("assistant.uploadFailedRemoveHint"):void 0,onClick:t=>{t.stopPropagation(),s(e.id)},children:(0,t.jsx)(S.X,{className:"h-2.5 w-2.5"})})]},e.id))})}var E=e.i(331548);function T({imageUrl:e,open:a,onOpenChange:r}){let{t:l}=(0,p.useI18n)(),[i,n]=(0,s.useState)(!1);return(0,s.useEffect)(()=>{n(!1)},[a]),(0,t.jsx)(E.Dialog,{open:a,onOpenChange:r,children:(0,t.jsxs)(E.DialogContent,{showCloseButton:!1,className:"max-w-[90vw] max-h-[90vh] p-0 overflow-hidden bg-black/90 ring-0",children:[(0,t.jsx)(E.DialogClose,{render:(0,t.jsx)(h.Button,{variant:"ghost",className:"absolute top-2 right-2 z-10 h-8 w-8 p-0 text-white hover:bg-white/20 hover:text-white"}),"aria-label":l("assistant.closePreview"),children:(0,t.jsx)(S.X,{className:"h-4 w-4"})}),e&&(0,t.jsx)("img",{src:e,alt:"",onClick:()=>n(e=>!e),className:i?"w-auto h-auto max-w-none cursor-zoom-out":"max-w-full max-h-[85vh] object-contain cursor-zoom-in mx-auto"})]})})}e.s(["AssistantChat",0,function(){let[e,l]=(0,s.useState)(""),g=(0,s.useRef)(null),b=(0,s.useRef)(null),{t:v}=(0,p.useI18n)(),{chatMessages:j,isChatThinking:y,isLoadingHistory:N,sendChatMessage:k,cancelChat:w}=(0,f.useAssistant)(),{pendingImages:C,addImages:z,removeImage:U,clearAll:M,hasUploading:R}=function(){let[e,t]=(0,s.useState)([]),a=(0,s.useRef)(new Map),r=(0,s.useRef)([]);(0,s.useEffect)(()=>{r.current=e},[e]),(0,s.useEffect)(()=>()=>{r.current.forEach(e=>URL.revokeObjectURL(e.blobUrl)),a.current.forEach(e=>e.abort()),a.current.clear()},[]);let l=(0,s.useCallback)(e=>{let s=e.map(e=>({id:crypto.randomUUID(),file:e,blobUrl:URL.createObjectURL(e),status:"uploading",progress:0}));for(let e of(r.current=[...r.current,...s],t(e=>[...e,...s]),s)){let s=new XMLHttpRequest;a.current.set(e.id,s),s.upload.onprogress=s=>{if(s.lengthComputable){let a=Math.round(s.loaded/s.total*100);t(t=>t.map(t=>t.id===e.id?{...t,progress:a}:t))}},s.onload=()=>{if(a.current.delete(e.id),200===s.status)try{let a=JSON.parse(s.responseText),r="string"==typeof a.filename&&a.filename?a.filename:null;if(!r)throw Error("Invalid upload response");t(t=>t.map(t=>t.id===e.id?{...t,status:"done",progress:100,filename:r}:t))}catch{t(t=>t.map(t=>t.id===e.id?{...t,status:"error"}:t))}else t(t=>t.map(t=>t.id===e.id?{...t,status:"error"}:t))},s.onerror=()=>{a.current.delete(e.id),t(t=>t.map(t=>t.id===e.id?{...t,status:"error"}:t))};let r=new FormData;r.append("file",e.file),s.open("POST","/api/internal/assistant/images"),s.send(r)}},[]),i=(0,s.useCallback)(e=>{let s=a.current.get(e);s&&(s.abort(),a.current.delete(e)),t(t=>{let s=t.find(t=>t.id===e);return s&&URL.revokeObjectURL(s.blobUrl),t.filter(t=>t.id!==e)})},[]),n=(0,s.useCallback)(()=>{t(e=>(e.forEach(e=>{let t=a.current.get(e.id);t&&(t.abort(),a.current.delete(e.id)),URL.revokeObjectURL(e.blobUrl)}),a.current.clear(),[]))},[]),o=e.some(e=>"uploading"===e.status);return{pendingImages:e,addImages:l,removeImage:i,clearAll:n,hasUploading:o}}(),[I,S]=(0,s.useState)(null),[E,D]=(0,s.useState)(null);(0,s.useEffect)(()=>{g.current?.focus()},[]);let A=(0,s.useRef)(!1),B=j[j.length-1]?.content.length??0;(0,s.useEffect)(()=>{let e=A.current?"smooth":"instant";b.current?.scrollIntoView({behavior:e}),A.current=!0},[j.length,B]);let O=(0,s.useCallback)(e=>{let t=[];for(let s of Array.from(e.clipboardData.items))if(s.type.startsWith("image/")){let e=s.getAsFile();e&&t.push(e)}t.length>0&&z(t)},[z]),$=()=>{let t=e.trim(),s=C.filter(e=>"done"===e.status).map(e=>e.filename);!t&&0===s.length||y||R||(k(t,{imageFilenames:s}),l(""),M(),g.current?.focus())},q=()=>{let e=w();e&&l(e),g.current?.focus()},F=C.some(e=>"done"===e.status),H=!e.trim()&&!F||y||R;return(0,t.jsxs)("div",{className:"flex flex-col h-full",children:[(0,t.jsx)(m.ScrollArea,{className:"flex-1 overflow-hidden",children:(0,t.jsxs)("div",{className:`flex flex-col gap-3 p-4 min-h-full ${N||0===j.length?"justify-center":""}`,role:"log","aria-live":"polite",children:[N?(0,t.jsxs)("div",{className:"flex flex-col items-center justify-center gap-2",children:[(0,t.jsx)(n.Loader2,{className:"size-5 animate-spin text-muted-foreground"}),(0,t.jsx)("span",{className:"text-xs text-muted-foreground",children:v("assistant.loading")})]}):0===j.length?(0,t.jsxs)("div",{className:"flex flex-col items-center gap-4 px-6",children:[(0,t.jsx)(a.Bot,{className:"size-8 text-muted-foreground/40"}),(0,t.jsx)("h3",{className:"text-sm font-semibold text-foreground",children:v("assistant.emptyTitle")}),(0,t.jsx)("div",{className:"grid grid-cols-1 gap-2 w-full max-w-[280px]",children:[{icon:i,label:v("assistant.suggestion.createProject")},{icon:r.ClipboardList,label:v("assistant.suggestion.createTask")},{icon:o.Search,label:v("assistant.suggestion.checkProgress")},{icon:u,label:v("assistant.suggestion.dailySummary")}].map(({icon:e,label:s})=>(0,t.jsxs)(h.Button,{variant:"outline",className:"flex items-center gap-2 rounded-lg px-3 py-2 text-xs text-muted-foreground transition-colors hover:bg-accent hover:text-foreground text-left justify-start h-auto",onClick:()=>{l(s),g.current?.focus()},children:[(0,t.jsx)(e,{className:"size-3.5 shrink-0"}),s]},s))})]}):j.map(e=>(0,t.jsx)(L,{message:e,onImagePreview:e=>D(e)},e.id)),(0,t.jsx)("div",{ref:b})]})}),(0,t.jsxs)("div",{className:"border-t border-border bg-sidebar p-4",children:[(0,t.jsx)(P,{pendingImages:C,onRemove:U,onPreview:e=>S(e)}),(0,t.jsxs)("div",{className:`flex items-end gap-2 ${C.length>0?"mt-2":""}`,children:[(0,t.jsx)(x.Textarea,{ref:g,value:e,onChange:e=>l(e.target.value),onKeyDown:e=>{!e.nativeEvent.isComposing&&("Enter"!==e.key||e.shiftKey||(e.preventDefault(),$()),"Escape"===e.key&&y&&(e.preventDefault(),q()))},onPaste:O,placeholder:v("assistant.inputPlaceholder"),className:"flex-1 min-h-[72px] max-h-[120px] resize-none text-sm",rows:3}),y?(0,t.jsx)(h.Button,{variant:"outline",size:"icon",className:"h-8 w-8 shrink-0 border-destructive/50 text-destructive transition-colors hover:bg-destructive/20 hover:text-destructive hover:border-destructive",onClick:q,"aria-label":v("assistant.cancelLabel"),children:(0,t.jsx)(c.Square,{className:"h-3 w-3 fill-current"})}):(0,t.jsx)(h.Button,{variant:"ghost",size:"icon",className:"h-8 w-8 shrink-0 text-muted-foreground transition-colors hover:bg-primary/10 hover:text-primary",onClick:$,disabled:H,"aria-label":v("assistant.sendLabel"),children:(0,t.jsx)(d,{className:"h-4 w-4"})})]})]}),(0,t.jsx)(T,{imageUrl:I?.blobUrl??null,open:null!==I,onOpenChange:e=>{e||S(null)}}),(0,t.jsx)(T,{imageUrl:E,open:null!==E,onOpenChange:e=>{e||D(null)}})]})}],543228)},143824,e=>{e.n(e.i(543228))}]);
|