@glwhappen/web-code 1.32.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +718 -0
- package/README.de.md +250 -0
- package/README.ja.md +242 -0
- package/README.ko.md +242 -0
- package/README.md +252 -0
- package/README.ru.md +250 -0
- package/README.tr.md +252 -0
- package/README.zh-CN.md +242 -0
- package/dist/api-docs.html +879 -0
- package/dist/assets/KaTeX_AMS-Regular-BQhdFMY1.woff2 +0 -0
- package/dist/assets/KaTeX_AMS-Regular-DMm9YOAa.woff +0 -0
- package/dist/assets/KaTeX_AMS-Regular-DRggAlZN.ttf +0 -0
- package/dist/assets/KaTeX_Caligraphic-Bold-ATXxdsX0.ttf +0 -0
- package/dist/assets/KaTeX_Caligraphic-Bold-BEiXGLvX.woff +0 -0
- package/dist/assets/KaTeX_Caligraphic-Bold-Dq_IR9rO.woff2 +0 -0
- package/dist/assets/KaTeX_Caligraphic-Regular-CTRA-rTL.woff +0 -0
- package/dist/assets/KaTeX_Caligraphic-Regular-Di6jR-x-.woff2 +0 -0
- package/dist/assets/KaTeX_Caligraphic-Regular-wX97UBjC.ttf +0 -0
- package/dist/assets/KaTeX_Fraktur-Bold-BdnERNNW.ttf +0 -0
- package/dist/assets/KaTeX_Fraktur-Bold-BsDP51OF.woff +0 -0
- package/dist/assets/KaTeX_Fraktur-Bold-CL6g_b3V.woff2 +0 -0
- package/dist/assets/KaTeX_Fraktur-Regular-CB_wures.ttf +0 -0
- package/dist/assets/KaTeX_Fraktur-Regular-CTYiF6lA.woff2 +0 -0
- package/dist/assets/KaTeX_Fraktur-Regular-Dxdc4cR9.woff +0 -0
- package/dist/assets/KaTeX_Main-Bold-Cx986IdX.woff2 +0 -0
- package/dist/assets/KaTeX_Main-Bold-Jm3AIy58.woff +0 -0
- package/dist/assets/KaTeX_Main-Bold-waoOVXN0.ttf +0 -0
- package/dist/assets/KaTeX_Main-BoldItalic-DxDJ3AOS.woff2 +0 -0
- package/dist/assets/KaTeX_Main-BoldItalic-DzxPMmG6.ttf +0 -0
- package/dist/assets/KaTeX_Main-BoldItalic-SpSLRI95.woff +0 -0
- package/dist/assets/KaTeX_Main-Italic-3WenGoN9.ttf +0 -0
- package/dist/assets/KaTeX_Main-Italic-BMLOBm91.woff +0 -0
- package/dist/assets/KaTeX_Main-Italic-NWA7e6Wa.woff2 +0 -0
- package/dist/assets/KaTeX_Main-Regular-B22Nviop.woff2 +0 -0
- package/dist/assets/KaTeX_Main-Regular-Dr94JaBh.woff +0 -0
- package/dist/assets/KaTeX_Main-Regular-ypZvNtVU.ttf +0 -0
- package/dist/assets/KaTeX_Math-BoldItalic-B3XSjfu4.ttf +0 -0
- package/dist/assets/KaTeX_Math-BoldItalic-CZnvNsCZ.woff2 +0 -0
- package/dist/assets/KaTeX_Math-BoldItalic-iY-2wyZ7.woff +0 -0
- package/dist/assets/KaTeX_Math-Italic-DA0__PXp.woff +0 -0
- package/dist/assets/KaTeX_Math-Italic-flOr_0UB.ttf +0 -0
- package/dist/assets/KaTeX_Math-Italic-t53AETM-.woff2 +0 -0
- package/dist/assets/KaTeX_SansSerif-Bold-CFMepnvq.ttf +0 -0
- package/dist/assets/KaTeX_SansSerif-Bold-D1sUS0GD.woff2 +0 -0
- package/dist/assets/KaTeX_SansSerif-Bold-DbIhKOiC.woff +0 -0
- package/dist/assets/KaTeX_SansSerif-Italic-C3H0VqGB.woff2 +0 -0
- package/dist/assets/KaTeX_SansSerif-Italic-DN2j7dab.woff +0 -0
- package/dist/assets/KaTeX_SansSerif-Italic-YYjJ1zSn.ttf +0 -0
- package/dist/assets/KaTeX_SansSerif-Regular-BNo7hRIc.ttf +0 -0
- package/dist/assets/KaTeX_SansSerif-Regular-CS6fqUqJ.woff +0 -0
- package/dist/assets/KaTeX_SansSerif-Regular-DDBCnlJ7.woff2 +0 -0
- package/dist/assets/KaTeX_Script-Regular-C5JkGWo-.ttf +0 -0
- package/dist/assets/KaTeX_Script-Regular-D3wIWfF6.woff2 +0 -0
- package/dist/assets/KaTeX_Script-Regular-D5yQViql.woff +0 -0
- package/dist/assets/KaTeX_Size1-Regular-C195tn64.woff +0 -0
- package/dist/assets/KaTeX_Size1-Regular-Dbsnue_I.ttf +0 -0
- package/dist/assets/KaTeX_Size1-Regular-mCD8mA8B.woff2 +0 -0
- package/dist/assets/KaTeX_Size2-Regular-B7gKUWhC.ttf +0 -0
- package/dist/assets/KaTeX_Size2-Regular-Dy4dx90m.woff2 +0 -0
- package/dist/assets/KaTeX_Size2-Regular-oD1tc_U0.woff +0 -0
- package/dist/assets/KaTeX_Size3-Regular-CTq5MqoE.woff +0 -0
- package/dist/assets/KaTeX_Size3-Regular-DgpXs0kz.ttf +0 -0
- package/dist/assets/KaTeX_Size4-Regular-BF-4gkZK.woff +0 -0
- package/dist/assets/KaTeX_Size4-Regular-DWFBv043.ttf +0 -0
- package/dist/assets/KaTeX_Size4-Regular-Dl5lxZxV.woff2 +0 -0
- package/dist/assets/KaTeX_Typewriter-Regular-C0xS9mPB.woff +0 -0
- package/dist/assets/KaTeX_Typewriter-Regular-CO6r4hn1.woff2 +0 -0
- package/dist/assets/KaTeX_Typewriter-Regular-D3Ib7_Hf.ttf +0 -0
- package/dist/assets/index-Ct6oPUQk.css +32 -0
- package/dist/assets/index-u6XmIqLb.js +1346 -0
- package/dist/assets/vendor-codemirror-OwyKSvPE.js +41 -0
- package/dist/assets/vendor-react-BGZc9oRE.js +59 -0
- package/dist/assets/vendor-xterm-CJZjLICi.js +66 -0
- package/dist/clear-cache.html +85 -0
- package/dist/convert-icons.md +53 -0
- package/dist/favicon.png +0 -0
- package/dist/favicon.svg +9 -0
- package/dist/generate-icons.js +49 -0
- package/dist/icons/claude-ai-icon.svg +1 -0
- package/dist/icons/codex-white.svg +3 -0
- package/dist/icons/codex.svg +3 -0
- package/dist/icons/cursor-white.svg +12 -0
- package/dist/icons/cursor.svg +1 -0
- package/dist/icons/gemini-ai-icon.svg +1 -0
- package/dist/icons/icon-128x128.png +0 -0
- package/dist/icons/icon-128x128.svg +12 -0
- package/dist/icons/icon-144x144.png +0 -0
- package/dist/icons/icon-144x144.svg +12 -0
- package/dist/icons/icon-152x152.png +0 -0
- package/dist/icons/icon-152x152.svg +12 -0
- package/dist/icons/icon-192x192.png +0 -0
- package/dist/icons/icon-192x192.svg +12 -0
- package/dist/icons/icon-384x384.png +0 -0
- package/dist/icons/icon-384x384.svg +12 -0
- package/dist/icons/icon-512x512.png +0 -0
- package/dist/icons/icon-512x512.svg +12 -0
- package/dist/icons/icon-72x72.png +0 -0
- package/dist/icons/icon-72x72.svg +12 -0
- package/dist/icons/icon-96x96.png +0 -0
- package/dist/icons/icon-96x96.svg +12 -0
- package/dist/icons/icon-template.svg +12 -0
- package/dist/index.html +52 -0
- package/dist/logo-128.png +0 -0
- package/dist/logo-256.png +0 -0
- package/dist/logo-32.png +0 -0
- package/dist/logo-512.png +0 -0
- package/dist/logo-64.png +0 -0
- package/dist/logo.svg +17 -0
- package/dist/manifest.json +61 -0
- package/dist/screenshots/cli-selection.png +0 -0
- package/dist/screenshots/desktop-main.png +0 -0
- package/dist/screenshots/mobile-chat.png +0 -0
- package/dist/screenshots/tools-modal.png +0 -0
- package/dist/sw.js +124 -0
- package/dist-server/server/claude-sdk.js +738 -0
- package/dist-server/server/claude-sdk.js.map +1 -0
- package/dist-server/server/cli.js +641 -0
- package/dist-server/server/cli.js.map +1 -0
- package/dist-server/server/constants/config.js +6 -0
- package/dist-server/server/constants/config.js.map +1 -0
- package/dist-server/server/cursor-cli.js +271 -0
- package/dist-server/server/cursor-cli.js.map +1 -0
- package/dist-server/server/gemini-cli.js +539 -0
- package/dist-server/server/gemini-cli.js.map +1 -0
- package/dist-server/server/gemini-response-handler.js +72 -0
- package/dist-server/server/gemini-response-handler.js.map +1 -0
- package/dist-server/server/index.js +1340 -0
- package/dist-server/server/index.js.map +1 -0
- package/dist-server/server/load-env.js +32 -0
- package/dist-server/server/load-env.js.map +1 -0
- package/dist-server/server/middleware/auth.js +117 -0
- package/dist-server/server/middleware/auth.js.map +1 -0
- package/dist-server/server/modules/database/connection.js +125 -0
- package/dist-server/server/modules/database/connection.js.map +1 -0
- package/dist-server/server/modules/database/index.js +13 -0
- package/dist-server/server/modules/database/index.js.map +1 -0
- package/dist-server/server/modules/database/init-db.js +18 -0
- package/dist-server/server/modules/database/init-db.js.map +1 -0
- package/dist-server/server/modules/database/migrations.js +419 -0
- package/dist-server/server/modules/database/migrations.js.map +1 -0
- package/dist-server/server/modules/database/repositories/api-keys.js +72 -0
- package/dist-server/server/modules/database/repositories/api-keys.js.map +1 -0
- package/dist-server/server/modules/database/repositories/app-config.js +47 -0
- package/dist-server/server/modules/database/repositories/app-config.js.map +1 -0
- package/dist-server/server/modules/database/repositories/credentials.js +68 -0
- package/dist-server/server/modules/database/repositories/credentials.js.map +1 -0
- package/dist-server/server/modules/database/repositories/github-tokens.js +54 -0
- package/dist-server/server/modules/database/repositories/github-tokens.js.map +1 -0
- package/dist-server/server/modules/database/repositories/notification-preferences.js +72 -0
- package/dist-server/server/modules/database/repositories/notification-preferences.js.map +1 -0
- package/dist-server/server/modules/database/repositories/projects.db.integration.test.js +67 -0
- package/dist-server/server/modules/database/repositories/projects.db.integration.test.js.map +1 -0
- package/dist-server/server/modules/database/repositories/projects.db.js +185 -0
- package/dist-server/server/modules/database/repositories/projects.db.js.map +1 -0
- package/dist-server/server/modules/database/repositories/push-subscriptions.js +49 -0
- package/dist-server/server/modules/database/repositories/push-subscriptions.js.map +1 -0
- package/dist-server/server/modules/database/repositories/scan-state.db.js +31 -0
- package/dist-server/server/modules/database/repositories/scan-state.db.js.map +1 -0
- package/dist-server/server/modules/database/repositories/sessions.db.integration.test.js +64 -0
- package/dist-server/server/modules/database/repositories/sessions.db.integration.test.js.map +1 -0
- package/dist-server/server/modules/database/repositories/sessions.db.js +150 -0
- package/dist-server/server/modules/database/repositories/sessions.db.js.map +1 -0
- package/dist-server/server/modules/database/repositories/users.js +116 -0
- package/dist-server/server/modules/database/repositories/users.js.map +1 -0
- package/dist-server/server/modules/database/repositories/vapid-keys.js +38 -0
- package/dist-server/server/modules/database/repositories/vapid-keys.js.map +1 -0
- package/dist-server/server/modules/database/schema.js +150 -0
- package/dist-server/server/modules/database/schema.js.map +1 -0
- package/dist-server/server/modules/projects/index.js +4 -0
- package/dist-server/server/modules/projects/index.js.map +1 -0
- package/dist-server/server/modules/projects/projects.routes.js +225 -0
- package/dist-server/server/modules/projects/projects.routes.js.map +1 -0
- package/dist-server/server/modules/projects/services/project-clone.service.js +220 -0
- package/dist-server/server/modules/projects/services/project-clone.service.js.map +1 -0
- package/dist-server/server/modules/projects/services/project-delete.service.js +83 -0
- package/dist-server/server/modules/projects/services/project-delete.service.js.map +1 -0
- package/dist-server/server/modules/projects/services/project-management.service.js +99 -0
- package/dist-server/server/modules/projects/services/project-management.service.js.map +1 -0
- package/dist-server/server/modules/projects/services/project-star.service.js +60 -0
- package/dist-server/server/modules/projects/services/project-star.service.js.map +1 -0
- package/dist-server/server/modules/projects/services/projects-has-taskmaster.service.js +171 -0
- package/dist-server/server/modules/projects/services/projects-has-taskmaster.service.js.map +1 -0
- package/dist-server/server/modules/projects/services/projects-with-sessions-fetch.service.js +213 -0
- package/dist-server/server/modules/projects/services/projects-with-sessions-fetch.service.js.map +1 -0
- package/dist-server/server/modules/projects/tests/project-clone.service.test.js +129 -0
- package/dist-server/server/modules/projects/tests/project-clone.service.test.js.map +1 -0
- package/dist-server/server/modules/projects/tests/project-management.service.test.js +89 -0
- package/dist-server/server/modules/projects/tests/project-management.service.test.js.map +1 -0
- package/dist-server/server/modules/projects/tests/project-star.service.test.js +99 -0
- package/dist-server/server/modules/projects/tests/project-star.service.test.js.map +1 -0
- package/dist-server/server/modules/projects/tests/projects-has-taskmaster.service.test.js +88 -0
- package/dist-server/server/modules/projects/tests/projects-has-taskmaster.service.test.js.map +1 -0
- package/dist-server/server/modules/providers/index.js +5 -0
- package/dist-server/server/modules/providers/index.js.map +1 -0
- package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js +104 -0
- package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/claude/claude-mcp.provider.js +103 -0
- package/dist-server/server/modules/providers/list/claude/claude-mcp.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/claude/claude-session-synchronizer.provider.js +116 -0
- package/dist-server/server/modules/providers/list/claude/claude-session-synchronizer.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/claude/claude-sessions.provider.js +546 -0
- package/dist-server/server/modules/providers/list/claude/claude-sessions.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/claude/claude-skills.provider.js +198 -0
- package/dist-server/server/modules/providers/list/claude/claude-skills.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/claude/claude.provider.js +17 -0
- package/dist-server/server/modules/providers/list/claude/claude.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js +84 -0
- package/dist-server/server/modules/providers/list/codex/codex-auth.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/codex/codex-mcp.provider.js +107 -0
- package/dist-server/server/modules/providers/list/codex/codex-mcp.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/codex/codex-session-synchronizer.provider.js +123 -0
- package/dist-server/server/modules/providers/list/codex/codex-session-synchronizer.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/codex/codex-sessions.provider.js +513 -0
- package/dist-server/server/modules/providers/list/codex/codex-sessions.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/codex/codex-skills.provider.js +82 -0
- package/dist-server/server/modules/providers/list/codex/codex-skills.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/codex/codex.provider.js +17 -0
- package/dist-server/server/modules/providers/list/codex/codex.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js +118 -0
- package/dist-server/server/modules/providers/list/cursor/cursor-auth.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/cursor/cursor-mcp.provider.js +80 -0
- package/dist-server/server/modules/providers/list/cursor/cursor-mcp.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/cursor/cursor-session-synchronizer.provider.js +105 -0
- package/dist-server/server/modules/providers/list/cursor/cursor-session-synchronizer.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/cursor/cursor-sessions.provider.js +545 -0
- package/dist-server/server/modules/providers/list/cursor/cursor-sessions.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/cursor/cursor-skills.provider.js +28 -0
- package/dist-server/server/modules/providers/list/cursor/cursor-skills.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/cursor/cursor.provider.js +17 -0
- package/dist-server/server/modules/providers/list/cursor/cursor.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js +254 -0
- package/dist-server/server/modules/providers/list/gemini/gemini-auth.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/gemini/gemini-mcp.provider.js +82 -0
- package/dist-server/server/modules/providers/list/gemini/gemini-mcp.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/gemini/gemini-session-synchronizer.provider.js +312 -0
- package/dist-server/server/modules/providers/list/gemini/gemini-session-synchronizer.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/gemini/gemini-sessions.provider.js +484 -0
- package/dist-server/server/modules/providers/list/gemini/gemini-sessions.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/gemini/gemini-skills.provider.js +33 -0
- package/dist-server/server/modules/providers/list/gemini/gemini-skills.provider.js.map +1 -0
- package/dist-server/server/modules/providers/list/gemini/gemini.provider.js +17 -0
- package/dist-server/server/modules/providers/list/gemini/gemini.provider.js.map +1 -0
- package/dist-server/server/modules/providers/provider.registry.js +31 -0
- package/dist-server/server/modules/providers/provider.registry.js.map +1 -0
- package/dist-server/server/modules/providers/provider.routes.js +377 -0
- package/dist-server/server/modules/providers/provider.routes.js.map +1 -0
- package/dist-server/server/modules/providers/services/mcp.service.js +69 -0
- package/dist-server/server/modules/providers/services/mcp.service.js.map +1 -0
- package/dist-server/server/modules/providers/services/provider-auth.service.js +25 -0
- package/dist-server/server/modules/providers/services/provider-auth.service.js.map +1 -0
- package/dist-server/server/modules/providers/services/session-conversations-search.service.js +984 -0
- package/dist-server/server/modules/providers/services/session-conversations-search.service.js.map +1 -0
- package/dist-server/server/modules/providers/services/session-synchronizer.service.js +56 -0
- package/dist-server/server/modules/providers/services/session-synchronizer.service.js.map +1 -0
- package/dist-server/server/modules/providers/services/sessions-watcher.service.js +269 -0
- package/dist-server/server/modules/providers/services/sessions-watcher.service.js.map +1 -0
- package/dist-server/server/modules/providers/services/sessions.service.js +179 -0
- package/dist-server/server/modules/providers/services/sessions.service.js.map +1 -0
- package/dist-server/server/modules/providers/services/skills.service.js +11 -0
- package/dist-server/server/modules/providers/services/skills.service.js.map +1 -0
- package/dist-server/server/modules/providers/shared/base/abstract.provider.js +14 -0
- package/dist-server/server/modules/providers/shared/base/abstract.provider.js.map +1 -0
- package/dist-server/server/modules/providers/shared/mcp/mcp.provider.js +102 -0
- package/dist-server/server/modules/providers/shared/mcp/mcp.provider.js.map +1 -0
- package/dist-server/server/modules/providers/shared/skills/skills.provider.js +45 -0
- package/dist-server/server/modules/providers/shared/skills/skills.provider.js.map +1 -0
- package/dist-server/server/modules/providers/tests/mcp.test.js +250 -0
- package/dist-server/server/modules/providers/tests/mcp.test.js.map +1 -0
- package/dist-server/server/modules/providers/tests/skills.test.js +226 -0
- package/dist-server/server/modules/providers/tests/skills.test.js.map +1 -0
- package/dist-server/server/modules/websocket/index.js +3 -0
- package/dist-server/server/modules/websocket/index.js.map +1 -0
- package/dist-server/server/modules/websocket/services/chat-websocket.service.js +192 -0
- package/dist-server/server/modules/websocket/services/chat-websocket.service.js.map +1 -0
- package/dist-server/server/modules/websocket/services/plugin-websocket-proxy.service.js +52 -0
- package/dist-server/server/modules/websocket/services/plugin-websocket-proxy.service.js.map +1 -0
- package/dist-server/server/modules/websocket/services/shell-websocket.service.js +360 -0
- package/dist-server/server/modules/websocket/services/shell-websocket.service.js.map +1 -0
- package/dist-server/server/modules/websocket/services/websocket-auth.service.js +32 -0
- package/dist-server/server/modules/websocket/services/websocket-auth.service.js.map +1 -0
- package/dist-server/server/modules/websocket/services/websocket-server.service.js +36 -0
- package/dist-server/server/modules/websocket/services/websocket-server.service.js.map +1 -0
- package/dist-server/server/modules/websocket/services/websocket-state.service.js +14 -0
- package/dist-server/server/modules/websocket/services/websocket-state.service.js.map +1 -0
- package/dist-server/server/modules/websocket/services/websocket-writer.service.js +32 -0
- package/dist-server/server/modules/websocket/services/websocket-writer.service.js.map +1 -0
- package/dist-server/server/openai-codex.js +418 -0
- package/dist-server/server/openai-codex.js.map +1 -0
- package/dist-server/server/routes/admin.js +109 -0
- package/dist-server/server/routes/admin.js.map +1 -0
- package/dist-server/server/routes/agent.js +1145 -0
- package/dist-server/server/routes/agent.js.map +1 -0
- package/dist-server/server/routes/auth.js +123 -0
- package/dist-server/server/routes/auth.js.map +1 -0
- package/dist-server/server/routes/commands.js +487 -0
- package/dist-server/server/routes/commands.js.map +1 -0
- package/dist-server/server/routes/cursor.js +49 -0
- package/dist-server/server/routes/cursor.js.map +1 -0
- package/dist-server/server/routes/gemini.js +25 -0
- package/dist-server/server/routes/gemini.js.map +1 -0
- package/dist-server/server/routes/git.js +1263 -0
- package/dist-server/server/routes/git.js.map +1 -0
- package/dist-server/server/routes/mcp-utils.js +29 -0
- package/dist-server/server/routes/mcp-utils.js.map +1 -0
- package/dist-server/server/routes/plugins.js +266 -0
- package/dist-server/server/routes/plugins.js.map +1 -0
- package/dist-server/server/routes/settings.js +259 -0
- package/dist-server/server/routes/settings.js.map +1 -0
- package/dist-server/server/routes/taskmaster.js +1360 -0
- package/dist-server/server/routes/taskmaster.js.map +1 -0
- package/dist-server/server/routes/user.js +115 -0
- package/dist-server/server/routes/user.js.map +1 -0
- package/dist-server/server/services/notification-orchestrator.js +177 -0
- package/dist-server/server/services/notification-orchestrator.js.map +1 -0
- package/dist-server/server/services/vapid-keys.js +27 -0
- package/dist-server/server/services/vapid-keys.js.map +1 -0
- package/dist-server/server/sessionManager.js +215 -0
- package/dist-server/server/sessionManager.js.map +1 -0
- package/dist-server/server/shared/claude-cli-path.js +103 -0
- package/dist-server/server/shared/claude-cli-path.js.map +1 -0
- package/dist-server/server/shared/claude-cli-path.test.js +45 -0
- package/dist-server/server/shared/claude-cli-path.test.js.map +1 -0
- package/dist-server/server/shared/default-user.js +29 -0
- package/dist-server/server/shared/default-user.js.map +1 -0
- package/dist-server/server/shared/frontmatter.js +16 -0
- package/dist-server/server/shared/frontmatter.js.map +1 -0
- package/dist-server/server/shared/interfaces.js +2 -0
- package/dist-server/server/shared/interfaces.js.map +1 -0
- package/dist-server/server/shared/types.js +2 -0
- package/dist-server/server/shared/types.js.map +1 -0
- package/dist-server/server/shared/utils.js +633 -0
- package/dist-server/server/shared/utils.js.map +1 -0
- package/dist-server/server/utils/colors.js +20 -0
- package/dist-server/server/utils/colors.js.map +1 -0
- package/dist-server/server/utils/commandParser.js +255 -0
- package/dist-server/server/utils/commandParser.js.map +1 -0
- package/dist-server/server/utils/gitConfig.js +36 -0
- package/dist-server/server/utils/gitConfig.js.map +1 -0
- package/dist-server/server/utils/mcp-detector.js +134 -0
- package/dist-server/server/utils/mcp-detector.js.map +1 -0
- package/dist-server/server/utils/plugin-loader.js +413 -0
- package/dist-server/server/utils/plugin-loader.js.map +1 -0
- package/dist-server/server/utils/plugin-process-manager.js +163 -0
- package/dist-server/server/utils/plugin-process-manager.js.map +1 -0
- package/dist-server/server/utils/runtime-paths.js +30 -0
- package/dist-server/server/utils/runtime-paths.js.map +1 -0
- package/dist-server/server/utils/taskmaster-websocket.js +124 -0
- package/dist-server/server/utils/taskmaster-websocket.js.map +1 -0
- package/dist-server/server/utils/url-detection.js +58 -0
- package/dist-server/server/utils/url-detection.js.map +1 -0
- package/dist-server/shared/modelConstants.js +99 -0
- package/dist-server/shared/modelConstants.js.map +1 -0
- package/dist-server/shared/networkHosts.js +20 -0
- package/dist-server/shared/networkHosts.js.map +1 -0
- package/package.json +169 -0
- package/scripts/fix-node-pty.js +67 -0
- package/server/claude-sdk.js +864 -0
- package/server/cli.js +688 -0
- package/server/constants/config.js +5 -0
- package/server/cursor-cli.js +334 -0
- package/server/gemini-cli.js +622 -0
- package/server/gemini-response-handler.js +79 -0
- package/server/index.js +1505 -0
- package/server/load-env.js +34 -0
- package/server/middleware/auth.js +142 -0
- package/server/modules/database/connection.ts +143 -0
- package/server/modules/database/index.ts +12 -0
- package/server/modules/database/init-db.ts +17 -0
- package/server/modules/database/migrations.ts +496 -0
- package/server/modules/database/repositories/api-keys.ts +119 -0
- package/server/modules/database/repositories/app-config.ts +53 -0
- package/server/modules/database/repositories/credentials.ts +106 -0
- package/server/modules/database/repositories/github-tokens.ts +100 -0
- package/server/modules/database/repositories/notification-preferences.ts +103 -0
- package/server/modules/database/repositories/projects.db.integration.test.ts +78 -0
- package/server/modules/database/repositories/projects.db.ts +210 -0
- package/server/modules/database/repositories/push-subscriptions.ts +80 -0
- package/server/modules/database/repositories/scan-state.db.ts +42 -0
- package/server/modules/database/repositories/sessions.db.integration.test.ts +78 -0
- package/server/modules/database/repositories/sessions.db.ts +230 -0
- package/server/modules/database/repositories/users.ts +186 -0
- package/server/modules/database/repositories/vapid-keys.ts +57 -0
- package/server/modules/database/schema.ts +159 -0
- package/server/modules/projects/index.ts +6 -0
- package/server/modules/projects/projects.routes.ts +292 -0
- package/server/modules/projects/services/project-clone.service.ts +327 -0
- package/server/modules/projects/services/project-delete.service.ts +95 -0
- package/server/modules/projects/services/project-management.service.ts +158 -0
- package/server/modules/projects/services/project-star.service.ts +78 -0
- package/server/modules/projects/services/projects-has-taskmaster.service.ts +257 -0
- package/server/modules/projects/services/projects-with-sessions-fetch.service.ts +355 -0
- package/server/modules/projects/tests/project-clone.service.test.ts +186 -0
- package/server/modules/projects/tests/project-management.service.test.ts +122 -0
- package/server/modules/projects/tests/project-star.service.test.ts +128 -0
- package/server/modules/projects/tests/projects-has-taskmaster.service.test.ts +107 -0
- package/server/modules/providers/README.md +346 -0
- package/server/modules/providers/index.ts +5 -0
- package/server/modules/providers/list/claude/claude-auth.provider.ts +124 -0
- package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -0
- package/server/modules/providers/list/claude/claude-session-synchronizer.provider.ts +179 -0
- package/server/modules/providers/list/claude/claude-sessions.provider.ts +642 -0
- package/server/modules/providers/list/claude/claude-skills.provider.ts +257 -0
- package/server/modules/providers/list/claude/claude.provider.ts +24 -0
- package/server/modules/providers/list/codex/codex-auth.provider.ts +100 -0
- package/server/modules/providers/list/codex/codex-mcp.provider.ts +135 -0
- package/server/modules/providers/list/codex/codex-session-synchronizer.provider.ts +182 -0
- package/server/modules/providers/list/codex/codex-sessions.provider.ts +589 -0
- package/server/modules/providers/list/codex/codex-skills.provider.ts +100 -0
- package/server/modules/providers/list/codex/codex.provider.ts +24 -0
- package/server/modules/providers/list/cursor/cursor-auth.provider.ts +143 -0
- package/server/modules/providers/list/cursor/cursor-mcp.provider.ts +108 -0
- package/server/modules/providers/list/cursor/cursor-session-synchronizer.provider.ts +155 -0
- package/server/modules/providers/list/cursor/cursor-sessions.provider.ts +624 -0
- package/server/modules/providers/list/cursor/cursor-skills.provider.ts +31 -0
- package/server/modules/providers/list/cursor/cursor.provider.ts +24 -0
- package/server/modules/providers/list/gemini/gemini-auth.provider.ts +307 -0
- package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -0
- package/server/modules/providers/list/gemini/gemini-session-synchronizer.provider.ts +407 -0
- package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +552 -0
- package/server/modules/providers/list/gemini/gemini-skills.provider.ts +36 -0
- package/server/modules/providers/list/gemini/gemini.provider.ts +24 -0
- package/server/modules/providers/provider.registry.ts +36 -0
- package/server/modules/providers/provider.routes.ts +488 -0
- package/server/modules/providers/services/mcp.service.ts +94 -0
- package/server/modules/providers/services/provider-auth.service.ts +26 -0
- package/server/modules/providers/services/session-conversations-search.service.ts +1319 -0
- package/server/modules/providers/services/session-synchronizer.service.ts +75 -0
- package/server/modules/providers/services/sessions-watcher.service.ts +318 -0
- package/server/modules/providers/services/sessions.service.ts +240 -0
- package/server/modules/providers/services/skills.service.ts +15 -0
- package/server/modules/providers/shared/base/abstract.provider.ts +29 -0
- package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -0
- package/server/modules/providers/shared/skills/skills.provider.ts +64 -0
- package/server/modules/providers/tests/mcp.test.ts +293 -0
- package/server/modules/providers/tests/skills.test.ts +446 -0
- package/server/modules/websocket/README.md +267 -0
- package/server/modules/websocket/index.ts +2 -0
- package/server/modules/websocket/services/chat-websocket.service.ts +275 -0
- package/server/modules/websocket/services/plugin-websocket-proxy.service.ts +65 -0
- package/server/modules/websocket/services/shell-websocket.service.ts +489 -0
- package/server/modules/websocket/services/websocket-auth.service.ts +54 -0
- package/server/modules/websocket/services/websocket-server.service.ts +58 -0
- package/server/modules/websocket/services/websocket-state.service.ts +16 -0
- package/server/modules/websocket/services/websocket-writer.service.ts +38 -0
- package/server/openai-codex.js +474 -0
- package/server/routes/admin.js +128 -0
- package/server/routes/agent.js +1246 -0
- package/server/routes/auth.js +144 -0
- package/server/routes/commands.js +556 -0
- package/server/routes/cursor.js +52 -0
- package/server/routes/gemini.js +30 -0
- package/server/routes/git.js +1493 -0
- package/server/routes/mcp-utils.js +31 -0
- package/server/routes/plugins.js +307 -0
- package/server/routes/settings.js +286 -0
- package/server/routes/taskmaster.js +1468 -0
- package/server/routes/user.js +123 -0
- package/server/services/notification-orchestrator.js +228 -0
- package/server/services/vapid-keys.js +36 -0
- package/server/sessionManager.js +248 -0
- package/server/shared/claude-cli-path.test.ts +61 -0
- package/server/shared/claude-cli-path.ts +139 -0
- package/server/shared/default-user.ts +30 -0
- package/server/shared/frontmatter.ts +18 -0
- package/server/shared/interfaces.ts +111 -0
- package/server/shared/types.ts +406 -0
- package/server/shared/utils.ts +763 -0
- package/server/tsconfig.json +36 -0
- package/server/utils/colors.js +21 -0
- package/server/utils/commandParser.js +305 -0
- package/server/utils/gitConfig.js +34 -0
- package/server/utils/mcp-detector.js +147 -0
- package/server/utils/plugin-loader.js +457 -0
- package/server/utils/plugin-process-manager.js +184 -0
- package/server/utils/runtime-paths.js +37 -0
- package/server/utils/taskmaster-websocket.js +135 -0
- package/server/utils/url-detection.js +71 -0
- package/shared/modelConstants.js +107 -0
- package/shared/networkHosts.js +22 -0
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
import assert from 'node:assert/strict';
|
|
2
|
+
import test from 'node:test';
|
|
3
|
+
import { createProject } from '../../../modules/projects/services/project-management.service.js';
|
|
4
|
+
import { AppError } from '../../../shared/utils.js';
|
|
5
|
+
const TEST_USER_ID = 1;
|
|
6
|
+
const projectRow = {
|
|
7
|
+
project_id: 'project-1',
|
|
8
|
+
project_path: '/workspace/my-project',
|
|
9
|
+
custom_project_name: 'my-project',
|
|
10
|
+
isStarred: 0,
|
|
11
|
+
isArchived: 0,
|
|
12
|
+
};
|
|
13
|
+
test('createProject throws when project path is missing', async () => {
|
|
14
|
+
await assert.rejects(async () => createProject({ userId: TEST_USER_ID, projectPath: '' }), (error) => {
|
|
15
|
+
assert.ok(error instanceof AppError);
|
|
16
|
+
assert.equal(error.code, 'PROJECT_PATH_REQUIRED');
|
|
17
|
+
assert.equal(error.statusCode, 400);
|
|
18
|
+
return true;
|
|
19
|
+
});
|
|
20
|
+
});
|
|
21
|
+
test('createProject throws when path validation fails', async () => {
|
|
22
|
+
await assert.rejects(async () => createProject({ userId: TEST_USER_ID, projectPath: '/invalid/path' }, {
|
|
23
|
+
validatePath: async () => ({ valid: false, error: 'blocked path' }),
|
|
24
|
+
ensureWorkspaceDirectory: async () => undefined,
|
|
25
|
+
persistProjectPath: () => ({ outcome: 'created', project: projectRow }),
|
|
26
|
+
getProjectByPath: () => projectRow,
|
|
27
|
+
}), (error) => {
|
|
28
|
+
assert.ok(error instanceof AppError);
|
|
29
|
+
assert.equal(error.code, 'INVALID_PROJECT_PATH');
|
|
30
|
+
assert.equal(error.statusCode, 400);
|
|
31
|
+
assert.equal(error.details, 'blocked path');
|
|
32
|
+
return true;
|
|
33
|
+
});
|
|
34
|
+
});
|
|
35
|
+
test('createProject throws conflict when active project path already exists', async () => {
|
|
36
|
+
await assert.rejects(async () => createProject({ userId: TEST_USER_ID, projectPath: '/workspace/my-project' }, {
|
|
37
|
+
validatePath: async () => ({ valid: true, resolvedPath: '/workspace/my-project' }),
|
|
38
|
+
ensureWorkspaceDirectory: async () => undefined,
|
|
39
|
+
persistProjectPath: () => ({ outcome: 'active_conflict', project: projectRow }),
|
|
40
|
+
getProjectByPath: () => projectRow,
|
|
41
|
+
}), (error) => {
|
|
42
|
+
assert.ok(error instanceof AppError);
|
|
43
|
+
assert.equal(error.code, 'PROJECT_ALREADY_EXISTS');
|
|
44
|
+
assert.equal(error.statusCode, 409);
|
|
45
|
+
assert.equal(error.details, 'Project path already exists: /workspace/my-project');
|
|
46
|
+
return true;
|
|
47
|
+
});
|
|
48
|
+
});
|
|
49
|
+
test('createProject falls back to directory name when custom name is not provided', async () => {
|
|
50
|
+
let capturedUserId = 0;
|
|
51
|
+
let capturedCustomName = null;
|
|
52
|
+
const result = await createProject({ userId: TEST_USER_ID, projectPath: '/workspace/my-project', customName: '' }, {
|
|
53
|
+
validatePath: async () => ({ valid: true, resolvedPath: '/workspace/my-project' }),
|
|
54
|
+
ensureWorkspaceDirectory: async () => undefined,
|
|
55
|
+
persistProjectPath: (userId, _projectPath, customName) => {
|
|
56
|
+
capturedUserId = userId;
|
|
57
|
+
capturedCustomName = customName;
|
|
58
|
+
return {
|
|
59
|
+
outcome: 'created',
|
|
60
|
+
project: {
|
|
61
|
+
...projectRow,
|
|
62
|
+
custom_project_name: customName,
|
|
63
|
+
},
|
|
64
|
+
};
|
|
65
|
+
},
|
|
66
|
+
getProjectByPath: () => projectRow,
|
|
67
|
+
});
|
|
68
|
+
assert.equal(capturedUserId, TEST_USER_ID);
|
|
69
|
+
assert.equal(capturedCustomName, 'my-project');
|
|
70
|
+
assert.equal(result.outcome, 'created');
|
|
71
|
+
assert.equal(result.project.displayName, 'my-project');
|
|
72
|
+
});
|
|
73
|
+
test('createProject returns archived reuse outcome when archived row is reused', async () => {
|
|
74
|
+
const result = await createProject({ userId: TEST_USER_ID, projectPath: '/workspace/my-project' }, {
|
|
75
|
+
validatePath: async () => ({ valid: true, resolvedPath: '/workspace/my-project' }),
|
|
76
|
+
ensureWorkspaceDirectory: async () => undefined,
|
|
77
|
+
persistProjectPath: () => ({
|
|
78
|
+
outcome: 'reactivated_archived',
|
|
79
|
+
project: {
|
|
80
|
+
...projectRow,
|
|
81
|
+
isArchived: 1,
|
|
82
|
+
},
|
|
83
|
+
}),
|
|
84
|
+
getProjectByPath: () => projectRow,
|
|
85
|
+
});
|
|
86
|
+
assert.equal(result.outcome, 'reactivated_archived');
|
|
87
|
+
assert.equal(result.project.isArchived, true);
|
|
88
|
+
});
|
|
89
|
+
//# sourceMappingURL=project-management.service.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"project-management.service.test.js","sourceRoot":"","sources":["../../../../../server/modules/projects/tests/project-management.service.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
import assert from 'node:assert/strict';
|
|
2
|
+
import test from 'node:test';
|
|
3
|
+
import { projectsDb } from '../../../modules/database/index.js';
|
|
4
|
+
import { applyLegacyStarredProjectIds, toggleProjectStar } from '../../../modules/projects/services/project-star.service.js';
|
|
5
|
+
import { AppError } from '../../../shared/utils.js';
|
|
6
|
+
const TEST_USER_ID = 1;
|
|
7
|
+
test('toggleProjectStar throws when projectId is missing', () => {
|
|
8
|
+
assert.throws(() => toggleProjectStar(TEST_USER_ID, ' '), (error) => error instanceof AppError
|
|
9
|
+
&& error.code === 'PROJECT_ID_REQUIRED'
|
|
10
|
+
&& error.statusCode === 400);
|
|
11
|
+
});
|
|
12
|
+
test('toggleProjectStar throws when project does not exist', () => {
|
|
13
|
+
const originalGetProjectById = projectsDb.getProjectById;
|
|
14
|
+
try {
|
|
15
|
+
projectsDb.getProjectById = () => null;
|
|
16
|
+
assert.throws(() => toggleProjectStar(TEST_USER_ID, 'project-1'), (error) => error instanceof AppError
|
|
17
|
+
&& error.code === 'PROJECT_NOT_FOUND'
|
|
18
|
+
&& error.statusCode === 404);
|
|
19
|
+
}
|
|
20
|
+
finally {
|
|
21
|
+
projectsDb.getProjectById = originalGetProjectById;
|
|
22
|
+
}
|
|
23
|
+
});
|
|
24
|
+
test('toggleProjectStar flips star state and persists it', () => {
|
|
25
|
+
const originalGetProjectById = projectsDb.getProjectById;
|
|
26
|
+
const originalUpdateProjectIsStarredById = projectsDb.updateProjectIsStarredById;
|
|
27
|
+
let capturedUserId = 0;
|
|
28
|
+
let capturedProjectId = '';
|
|
29
|
+
let capturedState = false;
|
|
30
|
+
try {
|
|
31
|
+
projectsDb.getProjectById = () => ({
|
|
32
|
+
project_id: 'project-1',
|
|
33
|
+
project_path: '/workspace/project-1',
|
|
34
|
+
custom_project_name: 'project-1',
|
|
35
|
+
isStarred: 0,
|
|
36
|
+
isArchived: 0,
|
|
37
|
+
});
|
|
38
|
+
projectsDb.updateProjectIsStarredById = (userId, projectId, isStarred) => {
|
|
39
|
+
capturedUserId = userId;
|
|
40
|
+
capturedProjectId = projectId;
|
|
41
|
+
capturedState = isStarred;
|
|
42
|
+
};
|
|
43
|
+
const result = toggleProjectStar(TEST_USER_ID, 'project-1');
|
|
44
|
+
assert.equal(result.isStarred, true);
|
|
45
|
+
assert.equal(capturedUserId, TEST_USER_ID);
|
|
46
|
+
assert.equal(capturedProjectId, 'project-1');
|
|
47
|
+
assert.equal(capturedState, true);
|
|
48
|
+
}
|
|
49
|
+
finally {
|
|
50
|
+
projectsDb.getProjectById = originalGetProjectById;
|
|
51
|
+
projectsDb.updateProjectIsStarredById = originalUpdateProjectIsStarredById;
|
|
52
|
+
}
|
|
53
|
+
});
|
|
54
|
+
test('applyLegacyStarredProjectIds stars only valid, unstarred projects', () => {
|
|
55
|
+
const originalGetProjectById = projectsDb.getProjectById;
|
|
56
|
+
const originalUpdateProjectIsStarredById = projectsDb.updateProjectIsStarredById;
|
|
57
|
+
const updatedProjectIds = [];
|
|
58
|
+
try {
|
|
59
|
+
projectsDb.getProjectById = (_userId, projectId) => {
|
|
60
|
+
if (projectId === 'project-a') {
|
|
61
|
+
return {
|
|
62
|
+
project_id: 'project-a',
|
|
63
|
+
project_path: '/workspace/project-a',
|
|
64
|
+
custom_project_name: 'A',
|
|
65
|
+
isStarred: 0,
|
|
66
|
+
isArchived: 0,
|
|
67
|
+
};
|
|
68
|
+
}
|
|
69
|
+
if (projectId === 'project-b') {
|
|
70
|
+
return {
|
|
71
|
+
project_id: 'project-b',
|
|
72
|
+
project_path: '/workspace/project-b',
|
|
73
|
+
custom_project_name: 'B',
|
|
74
|
+
isStarred: 1,
|
|
75
|
+
isArchived: 0,
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
return null;
|
|
79
|
+
};
|
|
80
|
+
projectsDb.updateProjectIsStarredById = (_userId, projectId) => {
|
|
81
|
+
updatedProjectIds.push(projectId);
|
|
82
|
+
};
|
|
83
|
+
const result = applyLegacyStarredProjectIds(TEST_USER_ID, [
|
|
84
|
+
'project-a',
|
|
85
|
+
'project-b',
|
|
86
|
+
'missing-project',
|
|
87
|
+
'project-a',
|
|
88
|
+
'',
|
|
89
|
+
' ',
|
|
90
|
+
]);
|
|
91
|
+
assert.equal(result.updated, 1);
|
|
92
|
+
assert.deepEqual(updatedProjectIds, ['project-a']);
|
|
93
|
+
}
|
|
94
|
+
finally {
|
|
95
|
+
projectsDb.getProjectById = originalGetProjectById;
|
|
96
|
+
projectsDb.updateProjectIsStarredById = originalUpdateProjectIsStarredById;
|
|
97
|
+
}
|
|
98
|
+
});
|
|
99
|
+
//# sourceMappingURL=project-star.service.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"project-star.service.test.js","sourceRoot":"","sources":["../../../../../server/modules/projects/tests/project-star.service.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
import assert from 'node:assert/strict';
|
|
2
|
+
import test from 'node:test';
|
|
3
|
+
import { getProjectTaskMaster, getProjectTaskMasterById, } from '../../../modules/projects/services/projects-has-taskmaster.service.js';
|
|
4
|
+
import { AppError } from '../../../shared/utils.js';
|
|
5
|
+
const TEST_USER_ID = 1;
|
|
6
|
+
test('getProjectTaskMasterById returns null when project path is missing', async () => {
|
|
7
|
+
const result = await getProjectTaskMasterById(TEST_USER_ID, 'project-1', {
|
|
8
|
+
resolveProjectPathById: () => null,
|
|
9
|
+
detectTaskMasterFolder: async () => {
|
|
10
|
+
throw new Error('detectTaskMasterFolder should not be called when path is missing');
|
|
11
|
+
},
|
|
12
|
+
});
|
|
13
|
+
assert.equal(result, null);
|
|
14
|
+
});
|
|
15
|
+
test('getProjectTaskMasterById returns configured status when taskmaster exists with essential files', async () => {
|
|
16
|
+
const result = await getProjectTaskMasterById(TEST_USER_ID, 'project-1', {
|
|
17
|
+
resolveProjectPathById: () => '/workspace/project-1',
|
|
18
|
+
detectTaskMasterFolder: async () => ({
|
|
19
|
+
hasTaskmaster: true,
|
|
20
|
+
hasEssentialFiles: true,
|
|
21
|
+
metadata: {
|
|
22
|
+
taskCount: 3,
|
|
23
|
+
subtaskCount: 0,
|
|
24
|
+
completed: 1,
|
|
25
|
+
pending: 2,
|
|
26
|
+
inProgress: 0,
|
|
27
|
+
review: 0,
|
|
28
|
+
completionPercentage: 33,
|
|
29
|
+
lastModified: '2026-01-01T00:00:00.000Z',
|
|
30
|
+
},
|
|
31
|
+
}),
|
|
32
|
+
});
|
|
33
|
+
assert.ok(result);
|
|
34
|
+
assert.equal(result.projectId, 'project-1');
|
|
35
|
+
assert.equal(result.projectPath, '/workspace/project-1');
|
|
36
|
+
assert.equal(result.taskmaster.hasTaskmaster, true);
|
|
37
|
+
assert.equal(result.taskmaster.hasEssentialFiles, true);
|
|
38
|
+
assert.equal(result.taskmaster.status, 'configured');
|
|
39
|
+
assert.deepEqual(result.taskmaster.metadata, {
|
|
40
|
+
taskCount: 3,
|
|
41
|
+
subtaskCount: 0,
|
|
42
|
+
completed: 1,
|
|
43
|
+
pending: 2,
|
|
44
|
+
inProgress: 0,
|
|
45
|
+
review: 0,
|
|
46
|
+
completionPercentage: 33,
|
|
47
|
+
lastModified: '2026-01-01T00:00:00.000Z',
|
|
48
|
+
});
|
|
49
|
+
});
|
|
50
|
+
test('getProjectTaskMasterById returns not-configured status when taskmaster is missing', async () => {
|
|
51
|
+
const result = await getProjectTaskMasterById(TEST_USER_ID, 'project-1', {
|
|
52
|
+
resolveProjectPathById: () => '/workspace/project-1',
|
|
53
|
+
detectTaskMasterFolder: async () => ({
|
|
54
|
+
hasTaskmaster: false,
|
|
55
|
+
}),
|
|
56
|
+
});
|
|
57
|
+
assert.ok(result);
|
|
58
|
+
assert.equal(result.taskmaster.hasTaskmaster, false);
|
|
59
|
+
assert.equal(result.taskmaster.hasEssentialFiles, false);
|
|
60
|
+
assert.equal(result.taskmaster.status, 'not-configured');
|
|
61
|
+
assert.equal(result.taskmaster.metadata, null);
|
|
62
|
+
});
|
|
63
|
+
test('getProjectTaskMaster throws when project id is missing', async () => {
|
|
64
|
+
await assert.rejects(async () => getProjectTaskMaster(TEST_USER_ID, '', async () => ({
|
|
65
|
+
projectId: 'project-1',
|
|
66
|
+
projectPath: '/workspace/project-1',
|
|
67
|
+
taskmaster: {
|
|
68
|
+
hasTaskmaster: true,
|
|
69
|
+
hasEssentialFiles: true,
|
|
70
|
+
metadata: null,
|
|
71
|
+
status: 'configured',
|
|
72
|
+
},
|
|
73
|
+
})), (error) => {
|
|
74
|
+
assert.ok(error instanceof AppError);
|
|
75
|
+
assert.equal(error.code, 'PROJECT_ID_REQUIRED');
|
|
76
|
+
assert.equal(error.statusCode, 400);
|
|
77
|
+
return true;
|
|
78
|
+
});
|
|
79
|
+
});
|
|
80
|
+
test('getProjectTaskMaster throws when project does not exist', async () => {
|
|
81
|
+
await assert.rejects(async () => getProjectTaskMaster(TEST_USER_ID, 'project-that-does-not-exist', async () => null), (error) => {
|
|
82
|
+
assert.ok(error instanceof AppError);
|
|
83
|
+
assert.equal(error.code, 'PROJECT_NOT_FOUND');
|
|
84
|
+
assert.equal(error.statusCode, 404);
|
|
85
|
+
return true;
|
|
86
|
+
});
|
|
87
|
+
});
|
|
88
|
+
//# sourceMappingURL=projects-has-taskmaster.service.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"projects-has-taskmaster.service.test.js","sourceRoot":"","sources":["../../../../../server/modules/projects/tests/projects-has-taskmaster.service.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
export { sessionSynchronizerService } from './services/session-synchronizer.service.js';
|
|
2
|
+
export { providerSkillsService } from './services/skills.service.js';
|
|
3
|
+
export { initializeSessionsWatcher } from './services/sessions-watcher.service.js';
|
|
4
|
+
export { closeSessionsWatcher } from './services/sessions-watcher.service.js';
|
|
5
|
+
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../server/modules/providers/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,0BAA0B,EAAE,MAAM,4CAA4C,CAAC;AACxF,OAAO,EAAE,qBAAqB,EAAE,MAAM,8BAA8B,CAAC;AAErE,OAAO,EAAE,yBAAyB,EAAE,MAAM,wCAAwC,CAAC;AACnF,OAAO,EAAE,oBAAoB,EAAE,MAAM,wCAAwC,CAAC"}
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
import { readFile } from 'node:fs/promises';
|
|
2
|
+
import os from 'node:os';
|
|
3
|
+
import path from 'node:path';
|
|
4
|
+
import spawn from 'cross-spawn';
|
|
5
|
+
import { resolveClaudeCodeExecutablePath } from '../../../../shared/claude-cli-path.js';
|
|
6
|
+
import { readObjectRecord, readOptionalString } from '../../../../shared/utils.js';
|
|
7
|
+
export class ClaudeProviderAuth {
|
|
8
|
+
/**
|
|
9
|
+
* Checks whether the Claude Code CLI is available on this host.
|
|
10
|
+
*/
|
|
11
|
+
checkInstalled() {
|
|
12
|
+
const cliPath = resolveClaudeCodeExecutablePath(process.env.CLAUDE_CLI_PATH);
|
|
13
|
+
try {
|
|
14
|
+
spawn.sync(cliPath, ['--version'], { stdio: 'ignore', timeout: 5000 });
|
|
15
|
+
return true;
|
|
16
|
+
}
|
|
17
|
+
catch {
|
|
18
|
+
return false;
|
|
19
|
+
}
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Returns Claude installation and credential status using Claude Code's auth priority.
|
|
23
|
+
*/
|
|
24
|
+
async getStatus() {
|
|
25
|
+
const installed = this.checkInstalled();
|
|
26
|
+
if (!installed) {
|
|
27
|
+
return {
|
|
28
|
+
installed,
|
|
29
|
+
provider: 'claude',
|
|
30
|
+
authenticated: false,
|
|
31
|
+
email: null,
|
|
32
|
+
method: null,
|
|
33
|
+
error: 'Claude Code CLI is not installed',
|
|
34
|
+
};
|
|
35
|
+
}
|
|
36
|
+
const credentials = await this.checkCredentials();
|
|
37
|
+
return {
|
|
38
|
+
installed,
|
|
39
|
+
provider: 'claude',
|
|
40
|
+
authenticated: credentials.authenticated,
|
|
41
|
+
email: credentials.authenticated ? credentials.email || 'Authenticated' : credentials.email,
|
|
42
|
+
method: credentials.method,
|
|
43
|
+
error: credentials.authenticated ? undefined : credentials.error || 'Not authenticated',
|
|
44
|
+
};
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Reads Claude settings env values that the CLI can use even when the server process env is empty.
|
|
48
|
+
*/
|
|
49
|
+
async loadSettingsEnv() {
|
|
50
|
+
try {
|
|
51
|
+
const settingsPath = path.join(os.homedir(), '.claude', 'settings.json');
|
|
52
|
+
const content = await readFile(settingsPath, 'utf8');
|
|
53
|
+
const settings = readObjectRecord(JSON.parse(content));
|
|
54
|
+
return readObjectRecord(settings?.env) ?? {};
|
|
55
|
+
}
|
|
56
|
+
catch {
|
|
57
|
+
return {};
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Checks Claude credentials in the same priority order used by Claude Code.
|
|
62
|
+
*/
|
|
63
|
+
async checkCredentials() {
|
|
64
|
+
if (process.env.ANTHROPIC_API_KEY?.trim()) {
|
|
65
|
+
return { authenticated: true, email: 'API Key Auth', method: 'api_key' };
|
|
66
|
+
}
|
|
67
|
+
const settingsEnv = await this.loadSettingsEnv();
|
|
68
|
+
if (readOptionalString(settingsEnv.ANTHROPIC_API_KEY)) {
|
|
69
|
+
return { authenticated: true, email: 'API Key Auth', method: 'api_key' };
|
|
70
|
+
}
|
|
71
|
+
if (readOptionalString(settingsEnv.ANTHROPIC_AUTH_TOKEN)) {
|
|
72
|
+
return { authenticated: true, email: 'Configured via settings.json', method: 'api_key' };
|
|
73
|
+
}
|
|
74
|
+
try {
|
|
75
|
+
const credPath = path.join(os.homedir(), '.claude', '.credentials.json');
|
|
76
|
+
const content = await readFile(credPath, 'utf8');
|
|
77
|
+
const creds = readObjectRecord(JSON.parse(content)) ?? {};
|
|
78
|
+
const oauth = readObjectRecord(creds.claudeAiOauth);
|
|
79
|
+
const accessToken = readOptionalString(oauth?.accessToken);
|
|
80
|
+
if (accessToken) {
|
|
81
|
+
const expiresAt = typeof oauth?.expiresAt === 'number' ? oauth.expiresAt : undefined;
|
|
82
|
+
const email = readOptionalString(creds.email) ?? readOptionalString(creds.user) ?? null;
|
|
83
|
+
if (!expiresAt || Date.now() < expiresAt) {
|
|
84
|
+
return {
|
|
85
|
+
authenticated: true,
|
|
86
|
+
email,
|
|
87
|
+
method: 'credentials_file',
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
return {
|
|
91
|
+
authenticated: false,
|
|
92
|
+
email,
|
|
93
|
+
method: 'credentials_file',
|
|
94
|
+
error: 'OAuth token has expired. Please re-authenticate with claude login',
|
|
95
|
+
};
|
|
96
|
+
}
|
|
97
|
+
return { authenticated: false, email: null, method: null };
|
|
98
|
+
}
|
|
99
|
+
catch {
|
|
100
|
+
return { authenticated: false, email: null, method: null };
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
//# sourceMappingURL=claude-auth.provider.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"claude-auth.provider.js","sourceRoot":"","sources":["../../../../../../server/modules/providers/list/claude/claude-auth.provider.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
import os from 'node:os';
|
|
2
|
+
import path from 'node:path';
|
|
3
|
+
import { McpProvider } from '../../../../modules/providers/shared/mcp/mcp.provider.js';
|
|
4
|
+
import { AppError, readJsonConfig, readObjectRecord, readOptionalString, readStringArray, readStringRecord, writeJsonConfig, } from '../../../../shared/utils.js';
|
|
5
|
+
export class ClaudeMcpProvider extends McpProvider {
|
|
6
|
+
constructor() {
|
|
7
|
+
super('claude', ['user', 'local', 'project'], ['stdio', 'http', 'sse']);
|
|
8
|
+
}
|
|
9
|
+
async readScopedServers(scope, workspacePath) {
|
|
10
|
+
if (scope === 'project') {
|
|
11
|
+
const filePath = path.join(workspacePath, '.mcp.json');
|
|
12
|
+
const config = await readJsonConfig(filePath);
|
|
13
|
+
return readObjectRecord(config.mcpServers) ?? {};
|
|
14
|
+
}
|
|
15
|
+
const filePath = path.join(os.homedir(), '.claude.json');
|
|
16
|
+
const config = await readJsonConfig(filePath);
|
|
17
|
+
if (scope === 'user') {
|
|
18
|
+
return readObjectRecord(config.mcpServers) ?? {};
|
|
19
|
+
}
|
|
20
|
+
const projects = readObjectRecord(config.projects) ?? {};
|
|
21
|
+
const projectConfig = readObjectRecord(projects[workspacePath]) ?? {};
|
|
22
|
+
return readObjectRecord(projectConfig.mcpServers) ?? {};
|
|
23
|
+
}
|
|
24
|
+
async writeScopedServers(scope, workspacePath, servers) {
|
|
25
|
+
if (scope === 'project') {
|
|
26
|
+
const filePath = path.join(workspacePath, '.mcp.json');
|
|
27
|
+
const config = await readJsonConfig(filePath);
|
|
28
|
+
config.mcpServers = servers;
|
|
29
|
+
await writeJsonConfig(filePath, config);
|
|
30
|
+
return;
|
|
31
|
+
}
|
|
32
|
+
const filePath = path.join(os.homedir(), '.claude.json');
|
|
33
|
+
const config = await readJsonConfig(filePath);
|
|
34
|
+
if (scope === 'user') {
|
|
35
|
+
config.mcpServers = servers;
|
|
36
|
+
await writeJsonConfig(filePath, config);
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
39
|
+
const projects = readObjectRecord(config.projects) ?? {};
|
|
40
|
+
const projectConfig = readObjectRecord(projects[workspacePath]) ?? {};
|
|
41
|
+
projectConfig.mcpServers = servers;
|
|
42
|
+
projects[workspacePath] = projectConfig;
|
|
43
|
+
config.projects = projects;
|
|
44
|
+
await writeJsonConfig(filePath, config);
|
|
45
|
+
}
|
|
46
|
+
buildServerConfig(input) {
|
|
47
|
+
if (input.transport === 'stdio') {
|
|
48
|
+
if (!input.command?.trim()) {
|
|
49
|
+
throw new AppError('command is required for stdio MCP servers.', {
|
|
50
|
+
code: 'MCP_COMMAND_REQUIRED',
|
|
51
|
+
statusCode: 400,
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
return {
|
|
55
|
+
type: 'stdio',
|
|
56
|
+
command: input.command,
|
|
57
|
+
args: input.args ?? [],
|
|
58
|
+
env: input.env ?? {},
|
|
59
|
+
};
|
|
60
|
+
}
|
|
61
|
+
if (!input.url?.trim()) {
|
|
62
|
+
throw new AppError('url is required for http/sse MCP servers.', {
|
|
63
|
+
code: 'MCP_URL_REQUIRED',
|
|
64
|
+
statusCode: 400,
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
return {
|
|
68
|
+
type: input.transport,
|
|
69
|
+
url: input.url,
|
|
70
|
+
headers: input.headers ?? {},
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
normalizeServerConfig(scope, name, rawConfig) {
|
|
74
|
+
if (!rawConfig || typeof rawConfig !== 'object') {
|
|
75
|
+
return null;
|
|
76
|
+
}
|
|
77
|
+
const config = rawConfig;
|
|
78
|
+
if (typeof config.command === 'string') {
|
|
79
|
+
return {
|
|
80
|
+
provider: 'claude',
|
|
81
|
+
name,
|
|
82
|
+
scope,
|
|
83
|
+
transport: 'stdio',
|
|
84
|
+
command: config.command,
|
|
85
|
+
args: readStringArray(config.args),
|
|
86
|
+
env: readStringRecord(config.env),
|
|
87
|
+
};
|
|
88
|
+
}
|
|
89
|
+
if (typeof config.url === 'string') {
|
|
90
|
+
const transport = readOptionalString(config.type) === 'sse' ? 'sse' : 'http';
|
|
91
|
+
return {
|
|
92
|
+
provider: 'claude',
|
|
93
|
+
name,
|
|
94
|
+
scope,
|
|
95
|
+
transport,
|
|
96
|
+
url: config.url,
|
|
97
|
+
headers: readStringRecord(config.headers),
|
|
98
|
+
};
|
|
99
|
+
}
|
|
100
|
+
return null;
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
//# sourceMappingURL=claude-mcp.provider.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"claude-mcp.provider.js","sourceRoot":"","sources":["../../../../../../server/modules/providers/list/claude/claude-mcp.provider.ts"],"names":[],"mappings":"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"}
|
package/dist-server/server/modules/providers/list/claude/claude-session-synchronizer.provider.js
ADDED
|
@@ -0,0 +1,116 @@
|
|
|
1
|
+
import os from 'node:os';
|
|
2
|
+
import path from 'node:path';
|
|
3
|
+
import { readFile } from 'node:fs/promises';
|
|
4
|
+
import { sessionsDb } from '../../../../modules/database/index.js';
|
|
5
|
+
import { buildLookupMap, extractFirstValidJsonlData, findFilesRecursivelyCreatedAfter, normalizeSessionName, readFileTimestamps, } from '../../../../shared/utils.js';
|
|
6
|
+
/**
|
|
7
|
+
* Session indexer for Claude transcript artifacts.
|
|
8
|
+
*/
|
|
9
|
+
export class ClaudeSessionSynchronizer {
|
|
10
|
+
provider = 'claude';
|
|
11
|
+
claudeHome = path.join(os.homedir(), '.claude');
|
|
12
|
+
/**
|
|
13
|
+
* Scans ~/.claude/projects and upserts discovered sessions into DB.
|
|
14
|
+
*/
|
|
15
|
+
async synchronize(since, ownerUserId) {
|
|
16
|
+
const nameMap = await buildLookupMap(path.join(this.claudeHome, 'history.jsonl'), 'sessionId', 'display');
|
|
17
|
+
const files = await findFilesRecursivelyCreatedAfter(path.join(this.claudeHome, 'projects'), '.jsonl', since ?? null);
|
|
18
|
+
let processed = 0;
|
|
19
|
+
for (const filePath of files) {
|
|
20
|
+
const parsed = await this.processSessionFile(ownerUserId, filePath, nameMap);
|
|
21
|
+
if (!parsed) {
|
|
22
|
+
continue;
|
|
23
|
+
}
|
|
24
|
+
const timestamps = await readFileTimestamps(filePath);
|
|
25
|
+
sessionsDb.createSession(ownerUserId, parsed.sessionId, this.provider, parsed.projectPath, parsed.sessionName, timestamps.createdAt, timestamps.updatedAt, filePath);
|
|
26
|
+
processed += 1;
|
|
27
|
+
}
|
|
28
|
+
return processed;
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Parses and upserts one Claude session JSONL file.
|
|
32
|
+
*/
|
|
33
|
+
async synchronizeFile(filePath, ownerUserId) {
|
|
34
|
+
if (!filePath.endsWith('.jsonl')) {
|
|
35
|
+
return null;
|
|
36
|
+
}
|
|
37
|
+
const nameMap = await buildLookupMap(path.join(this.claudeHome, 'history.jsonl'), 'sessionId', 'display');
|
|
38
|
+
const parsed = await this.processSessionFile(ownerUserId, filePath, nameMap);
|
|
39
|
+
if (!parsed) {
|
|
40
|
+
return null;
|
|
41
|
+
}
|
|
42
|
+
const timestamps = await readFileTimestamps(filePath);
|
|
43
|
+
return sessionsDb.createSession(ownerUserId, parsed.sessionId, this.provider, parsed.projectPath, parsed.sessionName, timestamps.createdAt, timestamps.updatedAt, filePath);
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Extracts session metadata from one Claude JSONL session file.
|
|
47
|
+
*/
|
|
48
|
+
async processSessionFile(ownerUserId, filePath, nameMap) {
|
|
49
|
+
const parsed = await extractFirstValidJsonlData(filePath, (rawData) => {
|
|
50
|
+
const data = rawData;
|
|
51
|
+
const sessionId = typeof data.sessionId === 'string' ? data.sessionId : undefined;
|
|
52
|
+
const projectPath = typeof data.cwd === 'string' ? data.cwd : undefined;
|
|
53
|
+
if (!sessionId || !projectPath) {
|
|
54
|
+
return null;
|
|
55
|
+
}
|
|
56
|
+
return {
|
|
57
|
+
sessionId,
|
|
58
|
+
projectPath,
|
|
59
|
+
};
|
|
60
|
+
});
|
|
61
|
+
if (!parsed) {
|
|
62
|
+
return null;
|
|
63
|
+
}
|
|
64
|
+
const existingSession = sessionsDb.getSessionById(ownerUserId, parsed.sessionId);
|
|
65
|
+
const existingSessionName = existingSession?.custom_name;
|
|
66
|
+
if (existingSessionName && existingSessionName !== 'Untitled Claude Session') {
|
|
67
|
+
return {
|
|
68
|
+
...parsed,
|
|
69
|
+
sessionName: normalizeSessionName(existingSessionName, 'Untitled Claude Session'),
|
|
70
|
+
};
|
|
71
|
+
}
|
|
72
|
+
let sessionName = nameMap.get(parsed.sessionId);
|
|
73
|
+
if (!sessionName) {
|
|
74
|
+
sessionName = await this.extractSessionAiTitleFromEnd(filePath, parsed.sessionId);
|
|
75
|
+
}
|
|
76
|
+
return {
|
|
77
|
+
...parsed,
|
|
78
|
+
sessionName: normalizeSessionName(sessionName, 'Untitled Claude Session'),
|
|
79
|
+
};
|
|
80
|
+
}
|
|
81
|
+
async extractSessionAiTitleFromEnd(filePath, sessionId) {
|
|
82
|
+
try {
|
|
83
|
+
const content = await readFile(filePath, 'utf8');
|
|
84
|
+
const lines = content.split(/\r?\n/);
|
|
85
|
+
for (let index = lines.length - 1; index >= 0; index -= 1) {
|
|
86
|
+
const line = lines[index]?.trim();
|
|
87
|
+
if (!line) {
|
|
88
|
+
continue;
|
|
89
|
+
}
|
|
90
|
+
let parsed;
|
|
91
|
+
try {
|
|
92
|
+
parsed = JSON.parse(line);
|
|
93
|
+
}
|
|
94
|
+
catch {
|
|
95
|
+
continue;
|
|
96
|
+
}
|
|
97
|
+
const data = parsed;
|
|
98
|
+
const eventType = typeof data.type === 'string' ? data.type : undefined;
|
|
99
|
+
const eventSessionId = typeof data.sessionId === 'string' ? data.sessionId : undefined;
|
|
100
|
+
const aiTitle = typeof data.aiTitle === 'string' ? data.aiTitle : undefined;
|
|
101
|
+
const lastPrompt = typeof data.lastPrompt === 'string' ? data.lastPrompt : undefined;
|
|
102
|
+
const claudeRenamedTitle = typeof data.customTitle === 'string' ? data.customTitle : undefined;
|
|
103
|
+
if ((eventType === 'ai-title' && eventSessionId === sessionId && aiTitle?.trim()) ||
|
|
104
|
+
(eventType === 'last-prompt' && eventSessionId === sessionId && lastPrompt?.trim()) ||
|
|
105
|
+
(eventType === "custom-title" && eventSessionId === sessionId && claudeRenamedTitle?.trim())) {
|
|
106
|
+
return aiTitle || lastPrompt || claudeRenamedTitle;
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
catch {
|
|
111
|
+
// Ignore missing/unreadable files so sync can continue.
|
|
112
|
+
}
|
|
113
|
+
return undefined;
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
//# sourceMappingURL=claude-session-synchronizer.provider.js.map
|