@pixelbyte-software/pixcode 1.30.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 +248 -0
- package/README.ja.md +240 -0
- package/README.ko.md +240 -0
- package/README.md +285 -0
- package/README.ru.md +248 -0
- package/README.tr.md +250 -0
- package/README.zh-CN.md +240 -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-C2c9QNwK.css +32 -0
- package/dist/assets/index-DyXDZED-.js +1277 -0
- package/dist/assets/vendor-codemirror-NA4v81it.js +41 -0
- package/dist/assets/vendor-react-D7WwDXvu.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 +709 -0
- package/dist-server/server/claude-sdk.js.map +1 -0
- package/dist-server/server/cli.js +854 -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 +277 -0
- package/dist-server/server/cursor-cli.js.map +1 -0
- package/dist-server/server/daemon/manager.js +486 -0
- package/dist-server/server/daemon/manager.js.map +1 -0
- package/dist-server/server/daemon-manager.js +823 -0
- package/dist-server/server/daemon-manager.js.map +1 -0
- package/dist-server/server/database/db.js +535 -0
- package/dist-server/server/database/db.js.map +1 -0
- package/dist-server/server/database/schema.js +97 -0
- package/dist-server/server/database/schema.js.map +1 -0
- package/dist-server/server/gemini-cli.js +408 -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 +2325 -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 +111 -0
- package/dist-server/server/middleware/auth.js.map +1 -0
- package/dist-server/server/modules/providers/list/claude/claude-auth.provider.js +103 -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-sessions.provider.js +266 -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.provider.js +13 -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-sessions.provider.js +282 -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.provider.js +13 -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-sessions.provider.js +369 -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.provider.js +13 -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 +131 -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-sessions.provider.js +207 -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.provider.js +13 -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 +159 -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/sessions.service.js +29 -0
- package/dist-server/server/modules/providers/services/sessions.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/tests/mcp.test.js +250 -0
- package/dist-server/server/modules/providers/tests/mcp.test.js.map +1 -0
- package/dist-server/server/openai-codex.js +373 -0
- package/dist-server/server/openai-codex.js.map +1 -0
- package/dist-server/server/projects.js +2492 -0
- package/dist-server/server/projects.js.map +1 -0
- package/dist-server/server/routes/agent.js +1147 -0
- package/dist-server/server/routes/agent.js.map +1 -0
- package/dist-server/server/routes/auth.js +117 -0
- package/dist-server/server/routes/auth.js.map +1 -0
- package/dist-server/server/routes/cli-auth.js +25 -0
- package/dist-server/server/routes/cli-auth.js.map +1 -0
- package/dist-server/server/routes/codex.js +18 -0
- package/dist-server/server/routes/codex.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 +21 -0
- package/dist-server/server/routes/gemini.js.map +1 -0
- package/dist-server/server/routes/git.js +1259 -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/messages.js +56 -0
- package/dist-server/server/routes/messages.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/projects.js +566 -0
- package/dist-server/server/routes/projects.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 +1373 -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 +26 -0
- package/dist-server/server/services/vapid-keys.js.map +1 -0
- package/dist-server/server/sessionManager.js +194 -0
- package/dist-server/server/sessionManager.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 +3 -0
- package/dist-server/server/shared/types.js.map +1 -0
- package/dist-server/server/shared/utils.js +150 -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/frontmatter.js +16 -0
- package/dist-server/server/utils/frontmatter.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 +118 -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/server/vite-daemon.js +75 -0
- package/dist-server/server/vite-daemon.js.map +1 -0
- package/dist-server/shared/modelConstants.js +90 -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 +168 -0
- package/scripts/fix-node-pty.js +67 -0
- package/server/claude-sdk.js +834 -0
- package/server/cli.js +937 -0
- package/server/constants/config.js +5 -0
- package/server/cursor-cli.js +342 -0
- package/server/daemon/manager.js +564 -0
- package/server/daemon-manager.js +920 -0
- package/server/database/db.js +593 -0
- package/server/database/schema.js +102 -0
- package/server/gemini-cli.js +469 -0
- package/server/gemini-response-handler.js +79 -0
- package/server/index.js +2557 -0
- package/server/load-env.js +34 -0
- package/server/middleware/auth.js +132 -0
- package/server/modules/providers/list/claude/claude-auth.provider.ts +123 -0
- package/server/modules/providers/list/claude/claude-mcp.provider.ts +135 -0
- package/server/modules/providers/list/claude/claude-sessions.provider.ts +306 -0
- package/server/modules/providers/list/claude/claude.provider.ts +15 -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-sessions.provider.ts +319 -0
- package/server/modules/providers/list/codex/codex.provider.ts +15 -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-sessions.provider.ts +421 -0
- package/server/modules/providers/list/cursor/cursor.provider.ts +15 -0
- package/server/modules/providers/list/gemini/gemini-auth.provider.ts +151 -0
- package/server/modules/providers/list/gemini/gemini-mcp.provider.ts +110 -0
- package/server/modules/providers/list/gemini/gemini-sessions.provider.ts +227 -0
- package/server/modules/providers/list/gemini/gemini.provider.ts +15 -0
- package/server/modules/providers/provider.registry.ts +36 -0
- package/server/modules/providers/provider.routes.ts +217 -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/sessions.service.ts +45 -0
- package/server/modules/providers/shared/base/abstract.provider.ts +20 -0
- package/server/modules/providers/shared/mcp/mcp.provider.ts +151 -0
- package/server/modules/providers/tests/mcp.test.ts +293 -0
- package/server/openai-codex.js +426 -0
- package/server/projects.js +2792 -0
- package/server/routes/agent.js +1245 -0
- package/server/routes/auth.js +135 -0
- package/server/routes/cli-auth.js +27 -0
- package/server/routes/codex.js +19 -0
- package/server/routes/commands.js +554 -0
- package/server/routes/cursor.js +52 -0
- package/server/routes/gemini.js +24 -0
- package/server/routes/git.js +1488 -0
- package/server/routes/mcp-utils.js +31 -0
- package/server/routes/messages.js +61 -0
- package/server/routes/plugins.js +307 -0
- package/server/routes/projects.js +627 -0
- package/server/routes/settings.js +286 -0
- package/server/routes/taskmaster.js +1471 -0
- package/server/routes/user.js +123 -0
- package/server/services/notification-orchestrator.js +227 -0
- package/server/services/vapid-keys.js +35 -0
- package/server/sessionManager.js +226 -0
- package/server/shared/interfaces.ts +54 -0
- package/server/shared/types.ts +172 -0
- package/server/shared/utils.ts +193 -0
- package/server/tsconfig.json +36 -0
- package/server/utils/colors.js +21 -0
- package/server/utils/commandParser.js +303 -0
- package/server/utils/frontmatter.js +18 -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 +129 -0
- package/server/utils/url-detection.js +71 -0
- package/server/vite-daemon.js +78 -0
- package/shared/modelConstants.js +97 -0
- package/shared/networkHosts.js +22 -0
|
@@ -0,0 +1,227 @@
|
|
|
1
|
+
import sessionManager from '@/sessionManager.js';
|
|
2
|
+
import { getGeminiCliSessionMessages } from '@/projects.js';
|
|
3
|
+
import type { IProviderSessions } from '@/shared/interfaces.js';
|
|
4
|
+
import type { AnyRecord, FetchHistoryOptions, FetchHistoryResult, NormalizedMessage } from '@/shared/types.js';
|
|
5
|
+
import { createNormalizedMessage, generateMessageId, readObjectRecord } from '@/shared/utils.js';
|
|
6
|
+
|
|
7
|
+
const PROVIDER = 'gemini';
|
|
8
|
+
|
|
9
|
+
export class GeminiSessionsProvider implements IProviderSessions {
|
|
10
|
+
/**
|
|
11
|
+
* Normalizes live Gemini stream-json events into the shared message shape.
|
|
12
|
+
*
|
|
13
|
+
* Gemini history uses a different session file shape, so fetchHistory handles
|
|
14
|
+
* that separately after loading raw persisted messages.
|
|
15
|
+
*/
|
|
16
|
+
normalizeMessage(rawMessage: unknown, sessionId: string | null): NormalizedMessage[] {
|
|
17
|
+
const raw = readObjectRecord(rawMessage);
|
|
18
|
+
if (!raw) {
|
|
19
|
+
return [];
|
|
20
|
+
}
|
|
21
|
+
|
|
22
|
+
const ts = raw.timestamp || new Date().toISOString();
|
|
23
|
+
const baseId = raw.uuid || generateMessageId('gemini');
|
|
24
|
+
|
|
25
|
+
if (raw.type === 'message' && raw.role === 'assistant') {
|
|
26
|
+
const content = raw.content || '';
|
|
27
|
+
const messages: NormalizedMessage[] = [];
|
|
28
|
+
if (content) {
|
|
29
|
+
messages.push(createNormalizedMessage({
|
|
30
|
+
id: baseId,
|
|
31
|
+
sessionId,
|
|
32
|
+
timestamp: ts,
|
|
33
|
+
provider: PROVIDER,
|
|
34
|
+
kind: 'stream_delta',
|
|
35
|
+
content,
|
|
36
|
+
}));
|
|
37
|
+
}
|
|
38
|
+
if (raw.delta !== true) {
|
|
39
|
+
messages.push(createNormalizedMessage({
|
|
40
|
+
sessionId,
|
|
41
|
+
timestamp: ts,
|
|
42
|
+
provider: PROVIDER,
|
|
43
|
+
kind: 'stream_end',
|
|
44
|
+
}));
|
|
45
|
+
}
|
|
46
|
+
return messages;
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
if (raw.type === 'tool_use') {
|
|
50
|
+
return [createNormalizedMessage({
|
|
51
|
+
id: baseId,
|
|
52
|
+
sessionId,
|
|
53
|
+
timestamp: ts,
|
|
54
|
+
provider: PROVIDER,
|
|
55
|
+
kind: 'tool_use',
|
|
56
|
+
toolName: raw.tool_name,
|
|
57
|
+
toolInput: raw.parameters || {},
|
|
58
|
+
toolId: raw.tool_id || baseId,
|
|
59
|
+
})];
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
if (raw.type === 'tool_result') {
|
|
63
|
+
return [createNormalizedMessage({
|
|
64
|
+
id: baseId,
|
|
65
|
+
sessionId,
|
|
66
|
+
timestamp: ts,
|
|
67
|
+
provider: PROVIDER,
|
|
68
|
+
kind: 'tool_result',
|
|
69
|
+
toolId: raw.tool_id || '',
|
|
70
|
+
content: raw.output === undefined ? '' : String(raw.output),
|
|
71
|
+
isError: raw.status === 'error',
|
|
72
|
+
})];
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
if (raw.type === 'result') {
|
|
76
|
+
const messages = [createNormalizedMessage({
|
|
77
|
+
sessionId,
|
|
78
|
+
timestamp: ts,
|
|
79
|
+
provider: PROVIDER,
|
|
80
|
+
kind: 'stream_end',
|
|
81
|
+
})];
|
|
82
|
+
if (raw.stats?.total_tokens) {
|
|
83
|
+
messages.push(createNormalizedMessage({
|
|
84
|
+
sessionId,
|
|
85
|
+
timestamp: ts,
|
|
86
|
+
provider: PROVIDER,
|
|
87
|
+
kind: 'status',
|
|
88
|
+
text: 'Complete',
|
|
89
|
+
tokens: raw.stats.total_tokens,
|
|
90
|
+
canInterrupt: false,
|
|
91
|
+
}));
|
|
92
|
+
}
|
|
93
|
+
return messages;
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
if (raw.type === 'error') {
|
|
97
|
+
return [createNormalizedMessage({
|
|
98
|
+
id: baseId,
|
|
99
|
+
sessionId,
|
|
100
|
+
timestamp: ts,
|
|
101
|
+
provider: PROVIDER,
|
|
102
|
+
kind: 'error',
|
|
103
|
+
content: raw.error || raw.message || 'Unknown Gemini streaming error',
|
|
104
|
+
})];
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
return [];
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
/**
|
|
111
|
+
* Loads Gemini history from the in-memory session manager first, then falls
|
|
112
|
+
* back to Gemini CLI session files on disk.
|
|
113
|
+
*/
|
|
114
|
+
async fetchHistory(
|
|
115
|
+
sessionId: string,
|
|
116
|
+
options: FetchHistoryOptions = {},
|
|
117
|
+
): Promise<FetchHistoryResult> {
|
|
118
|
+
const { limit = null, offset = 0 } = options;
|
|
119
|
+
|
|
120
|
+
let rawMessages: AnyRecord[];
|
|
121
|
+
try {
|
|
122
|
+
rawMessages = sessionManager.getSessionMessages(sessionId) as AnyRecord[];
|
|
123
|
+
|
|
124
|
+
if (rawMessages.length === 0) {
|
|
125
|
+
rawMessages = await getGeminiCliSessionMessages(sessionId) as AnyRecord[];
|
|
126
|
+
}
|
|
127
|
+
} catch (error) {
|
|
128
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
129
|
+
console.warn(`[GeminiProvider] Failed to load session ${sessionId}:`, message);
|
|
130
|
+
return { messages: [], total: 0, hasMore: false, offset: 0, limit: null };
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
const normalized: NormalizedMessage[] = [];
|
|
134
|
+
for (let i = 0; i < rawMessages.length; i++) {
|
|
135
|
+
const raw = rawMessages[i];
|
|
136
|
+
const ts = raw.timestamp || new Date().toISOString();
|
|
137
|
+
const baseId = raw.uuid || generateMessageId('gemini');
|
|
138
|
+
|
|
139
|
+
const role = raw.message?.role || raw.role;
|
|
140
|
+
const content = raw.message?.content || raw.content;
|
|
141
|
+
|
|
142
|
+
if (!role || !content) {
|
|
143
|
+
continue;
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
const normalizedRole = role === 'user' ? 'user' : 'assistant';
|
|
147
|
+
|
|
148
|
+
if (Array.isArray(content)) {
|
|
149
|
+
for (let partIdx = 0; partIdx < content.length; partIdx++) {
|
|
150
|
+
const part = content[partIdx];
|
|
151
|
+
if (part.type === 'text' && part.text) {
|
|
152
|
+
normalized.push(createNormalizedMessage({
|
|
153
|
+
id: `${baseId}_${partIdx}`,
|
|
154
|
+
sessionId,
|
|
155
|
+
timestamp: ts,
|
|
156
|
+
provider: PROVIDER,
|
|
157
|
+
kind: 'text',
|
|
158
|
+
role: normalizedRole,
|
|
159
|
+
content: part.text,
|
|
160
|
+
}));
|
|
161
|
+
} else if (part.type === 'tool_use') {
|
|
162
|
+
normalized.push(createNormalizedMessage({
|
|
163
|
+
id: `${baseId}_${partIdx}`,
|
|
164
|
+
sessionId,
|
|
165
|
+
timestamp: ts,
|
|
166
|
+
provider: PROVIDER,
|
|
167
|
+
kind: 'tool_use',
|
|
168
|
+
toolName: part.name,
|
|
169
|
+
toolInput: part.input,
|
|
170
|
+
toolId: part.id || generateMessageId('gemini_tool'),
|
|
171
|
+
}));
|
|
172
|
+
} else if (part.type === 'tool_result') {
|
|
173
|
+
normalized.push(createNormalizedMessage({
|
|
174
|
+
id: `${baseId}_${partIdx}`,
|
|
175
|
+
sessionId,
|
|
176
|
+
timestamp: ts,
|
|
177
|
+
provider: PROVIDER,
|
|
178
|
+
kind: 'tool_result',
|
|
179
|
+
toolId: part.tool_use_id || '',
|
|
180
|
+
content: part.content === undefined ? '' : String(part.content),
|
|
181
|
+
isError: Boolean(part.is_error),
|
|
182
|
+
}));
|
|
183
|
+
}
|
|
184
|
+
}
|
|
185
|
+
} else if (typeof content === 'string' && content.trim()) {
|
|
186
|
+
normalized.push(createNormalizedMessage({
|
|
187
|
+
id: baseId,
|
|
188
|
+
sessionId,
|
|
189
|
+
timestamp: ts,
|
|
190
|
+
provider: PROVIDER,
|
|
191
|
+
kind: 'text',
|
|
192
|
+
role: normalizedRole,
|
|
193
|
+
content,
|
|
194
|
+
}));
|
|
195
|
+
}
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
const toolResultMap = new Map<string, NormalizedMessage>();
|
|
199
|
+
for (const msg of normalized) {
|
|
200
|
+
if (msg.kind === 'tool_result' && msg.toolId) {
|
|
201
|
+
toolResultMap.set(msg.toolId, msg);
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
for (const msg of normalized) {
|
|
205
|
+
if (msg.kind === 'tool_use' && msg.toolId && toolResultMap.has(msg.toolId)) {
|
|
206
|
+
const toolResult = toolResultMap.get(msg.toolId);
|
|
207
|
+
if (toolResult) {
|
|
208
|
+
msg.toolResult = { content: toolResult.content, isError: toolResult.isError };
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
const start = Math.max(0, offset);
|
|
214
|
+
const pageLimit = limit === null ? null : Math.max(0, limit);
|
|
215
|
+
const messages = pageLimit === null
|
|
216
|
+
? normalized.slice(start)
|
|
217
|
+
: normalized.slice(start, start + pageLimit);
|
|
218
|
+
|
|
219
|
+
return {
|
|
220
|
+
messages,
|
|
221
|
+
total: normalized.length,
|
|
222
|
+
hasMore: pageLimit === null ? false : start + pageLimit < normalized.length,
|
|
223
|
+
offset: start,
|
|
224
|
+
limit: pageLimit,
|
|
225
|
+
};
|
|
226
|
+
}
|
|
227
|
+
}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import { AbstractProvider } from '@/modules/providers/shared/base/abstract.provider.js';
|
|
2
|
+
import { GeminiProviderAuth } from '@/modules/providers/list/gemini/gemini-auth.provider.js';
|
|
3
|
+
import { GeminiMcpProvider } from '@/modules/providers/list/gemini/gemini-mcp.provider.js';
|
|
4
|
+
import { GeminiSessionsProvider } from '@/modules/providers/list/gemini/gemini-sessions.provider.js';
|
|
5
|
+
import type { IProviderAuth, IProviderSessions } from '@/shared/interfaces.js';
|
|
6
|
+
|
|
7
|
+
export class GeminiProvider extends AbstractProvider {
|
|
8
|
+
readonly mcp = new GeminiMcpProvider();
|
|
9
|
+
readonly auth: IProviderAuth = new GeminiProviderAuth();
|
|
10
|
+
readonly sessions: IProviderSessions = new GeminiSessionsProvider();
|
|
11
|
+
|
|
12
|
+
constructor() {
|
|
13
|
+
super('gemini');
|
|
14
|
+
}
|
|
15
|
+
}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { ClaudeProvider } from '@/modules/providers/list/claude/claude.provider.js';
|
|
2
|
+
import { CodexProvider } from '@/modules/providers/list/codex/codex.provider.js';
|
|
3
|
+
import { CursorProvider } from '@/modules/providers/list/cursor/cursor.provider.js';
|
|
4
|
+
import { GeminiProvider } from '@/modules/providers/list/gemini/gemini.provider.js';
|
|
5
|
+
import type { IProvider } from '@/shared/interfaces.js';
|
|
6
|
+
import type { LLMProvider } from '@/shared/types.js';
|
|
7
|
+
import { AppError } from '@/shared/utils.js';
|
|
8
|
+
|
|
9
|
+
const providers: Record<LLMProvider, IProvider> = {
|
|
10
|
+
claude: new ClaudeProvider(),
|
|
11
|
+
codex: new CodexProvider(),
|
|
12
|
+
cursor: new CursorProvider(),
|
|
13
|
+
gemini: new GeminiProvider(),
|
|
14
|
+
};
|
|
15
|
+
|
|
16
|
+
/**
|
|
17
|
+
* Central registry for resolving concrete provider implementations by id.
|
|
18
|
+
*/
|
|
19
|
+
export const providerRegistry = {
|
|
20
|
+
listProviders(): IProvider[] {
|
|
21
|
+
return Object.values(providers);
|
|
22
|
+
},
|
|
23
|
+
|
|
24
|
+
resolveProvider(provider: string): IProvider {
|
|
25
|
+
const key = provider as LLMProvider;
|
|
26
|
+
const resolvedProvider = providers[key];
|
|
27
|
+
if (!resolvedProvider) {
|
|
28
|
+
throw new AppError(`Unsupported provider "${provider}".`, {
|
|
29
|
+
code: 'UNSUPPORTED_PROVIDER',
|
|
30
|
+
statusCode: 400,
|
|
31
|
+
});
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
return resolvedProvider;
|
|
35
|
+
},
|
|
36
|
+
};
|
|
@@ -0,0 +1,217 @@
|
|
|
1
|
+
import express, { type Request, type Response } from 'express';
|
|
2
|
+
|
|
3
|
+
import { providerAuthService } from '@/modules/providers/services/provider-auth.service.js';
|
|
4
|
+
import { providerMcpService } from '@/modules/providers/services/mcp.service.js';
|
|
5
|
+
import type { LLMProvider, McpScope, McpTransport, UpsertProviderMcpServerInput } from '@/shared/types.js';
|
|
6
|
+
import { AppError, asyncHandler, createApiSuccessResponse } from '@/shared/utils.js';
|
|
7
|
+
|
|
8
|
+
const router = express.Router();
|
|
9
|
+
|
|
10
|
+
const readPathParam = (value: unknown, name: string): string => {
|
|
11
|
+
if (typeof value === 'string') {
|
|
12
|
+
return value;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
if (Array.isArray(value) && typeof value[0] === 'string') {
|
|
16
|
+
return value[0];
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
throw new AppError(`${name} path parameter is invalid.`, {
|
|
20
|
+
code: 'INVALID_PATH_PARAMETER',
|
|
21
|
+
statusCode: 400,
|
|
22
|
+
});
|
|
23
|
+
};
|
|
24
|
+
|
|
25
|
+
const normalizeProviderParam = (value: unknown): string =>
|
|
26
|
+
readPathParam(value, 'provider').trim().toLowerCase();
|
|
27
|
+
|
|
28
|
+
const readOptionalQueryString = (value: unknown): string | undefined => {
|
|
29
|
+
if (typeof value !== 'string') {
|
|
30
|
+
return undefined;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
const normalized = value.trim();
|
|
34
|
+
return normalized.length > 0 ? normalized : undefined;
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
const parseMcpScope = (value: unknown): McpScope | undefined => {
|
|
38
|
+
if (value === undefined) {
|
|
39
|
+
return undefined;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
const normalized = readOptionalQueryString(value);
|
|
43
|
+
if (!normalized) {
|
|
44
|
+
return undefined;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
if (normalized === 'user' || normalized === 'local' || normalized === 'project') {
|
|
48
|
+
return normalized;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
throw new AppError(`Unsupported MCP scope "${normalized}".`, {
|
|
52
|
+
code: 'INVALID_MCP_SCOPE',
|
|
53
|
+
statusCode: 400,
|
|
54
|
+
});
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
const parseMcpTransport = (value: unknown): McpTransport => {
|
|
58
|
+
const normalized = readOptionalQueryString(value);
|
|
59
|
+
if (!normalized) {
|
|
60
|
+
throw new AppError('transport is required.', {
|
|
61
|
+
code: 'MCP_TRANSPORT_REQUIRED',
|
|
62
|
+
statusCode: 400,
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
if (normalized === 'stdio' || normalized === 'http' || normalized === 'sse') {
|
|
67
|
+
return normalized;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
throw new AppError(`Unsupported MCP transport "${normalized}".`, {
|
|
71
|
+
code: 'INVALID_MCP_TRANSPORT',
|
|
72
|
+
statusCode: 400,
|
|
73
|
+
});
|
|
74
|
+
};
|
|
75
|
+
|
|
76
|
+
const parseMcpUpsertPayload = (payload: unknown): UpsertProviderMcpServerInput => {
|
|
77
|
+
if (!payload || typeof payload !== 'object') {
|
|
78
|
+
throw new AppError('Request body must be an object.', {
|
|
79
|
+
code: 'INVALID_REQUEST_BODY',
|
|
80
|
+
statusCode: 400,
|
|
81
|
+
});
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
const body = payload as Record<string, unknown>;
|
|
85
|
+
const name = readOptionalQueryString(body.name);
|
|
86
|
+
if (!name) {
|
|
87
|
+
throw new AppError('name is required.', {
|
|
88
|
+
code: 'MCP_NAME_REQUIRED',
|
|
89
|
+
statusCode: 400,
|
|
90
|
+
});
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
const transport = parseMcpTransport(body.transport);
|
|
94
|
+
const scope = parseMcpScope(body.scope);
|
|
95
|
+
const workspacePath = readOptionalQueryString(body.workspacePath);
|
|
96
|
+
|
|
97
|
+
return {
|
|
98
|
+
name,
|
|
99
|
+
transport,
|
|
100
|
+
scope,
|
|
101
|
+
workspacePath,
|
|
102
|
+
command: readOptionalQueryString(body.command),
|
|
103
|
+
args: Array.isArray(body.args) ? body.args.filter((entry): entry is string => typeof entry === 'string') : undefined,
|
|
104
|
+
env: typeof body.env === 'object' && body.env !== null
|
|
105
|
+
? Object.fromEntries(
|
|
106
|
+
Object.entries(body.env as Record<string, unknown>).filter(
|
|
107
|
+
(entry): entry is [string, string] => typeof entry[1] === 'string',
|
|
108
|
+
),
|
|
109
|
+
)
|
|
110
|
+
: undefined,
|
|
111
|
+
cwd: readOptionalQueryString(body.cwd),
|
|
112
|
+
url: readOptionalQueryString(body.url),
|
|
113
|
+
headers: typeof body.headers === 'object' && body.headers !== null
|
|
114
|
+
? Object.fromEntries(
|
|
115
|
+
Object.entries(body.headers as Record<string, unknown>).filter(
|
|
116
|
+
(entry): entry is [string, string] => typeof entry[1] === 'string',
|
|
117
|
+
),
|
|
118
|
+
)
|
|
119
|
+
: undefined,
|
|
120
|
+
envVars: Array.isArray(body.envVars)
|
|
121
|
+
? body.envVars.filter((entry): entry is string => typeof entry === 'string')
|
|
122
|
+
: undefined,
|
|
123
|
+
bearerTokenEnvVar: readOptionalQueryString(body.bearerTokenEnvVar),
|
|
124
|
+
envHttpHeaders: typeof body.envHttpHeaders === 'object' && body.envHttpHeaders !== null
|
|
125
|
+
? Object.fromEntries(
|
|
126
|
+
Object.entries(body.envHttpHeaders as Record<string, unknown>).filter(
|
|
127
|
+
(entry): entry is [string, string] => typeof entry[1] === 'string',
|
|
128
|
+
),
|
|
129
|
+
)
|
|
130
|
+
: undefined,
|
|
131
|
+
};
|
|
132
|
+
};
|
|
133
|
+
|
|
134
|
+
const parseProvider = (value: unknown): LLMProvider => {
|
|
135
|
+
const normalized = normalizeProviderParam(value);
|
|
136
|
+
if (normalized === 'claude' || normalized === 'codex' || normalized === 'cursor' || normalized === 'gemini') {
|
|
137
|
+
return normalized;
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
throw new AppError(`Unsupported provider "${normalized}".`, {
|
|
141
|
+
code: 'UNSUPPORTED_PROVIDER',
|
|
142
|
+
statusCode: 400,
|
|
143
|
+
});
|
|
144
|
+
};
|
|
145
|
+
|
|
146
|
+
router.get(
|
|
147
|
+
'/:provider/auth/status',
|
|
148
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
149
|
+
const provider = parseProvider(req.params.provider);
|
|
150
|
+
const status = await providerAuthService.getProviderAuthStatus(provider);
|
|
151
|
+
res.json(createApiSuccessResponse(status));
|
|
152
|
+
}),
|
|
153
|
+
);
|
|
154
|
+
|
|
155
|
+
router.get(
|
|
156
|
+
'/:provider/mcp/servers',
|
|
157
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
158
|
+
const provider = parseProvider(req.params.provider);
|
|
159
|
+
const workspacePath = readOptionalQueryString(req.query.workspacePath);
|
|
160
|
+
const scope = parseMcpScope(req.query.scope);
|
|
161
|
+
|
|
162
|
+
if (scope) {
|
|
163
|
+
const servers = await providerMcpService.listProviderMcpServersForScope(provider, scope, { workspacePath });
|
|
164
|
+
res.json(createApiSuccessResponse({ provider, scope, servers }));
|
|
165
|
+
return;
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
const groupedServers = await providerMcpService.listProviderMcpServers(provider, { workspacePath });
|
|
169
|
+
res.json(createApiSuccessResponse({ provider, scopes: groupedServers }));
|
|
170
|
+
}),
|
|
171
|
+
);
|
|
172
|
+
|
|
173
|
+
router.post(
|
|
174
|
+
'/:provider/mcp/servers',
|
|
175
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
176
|
+
const provider = parseProvider(req.params.provider);
|
|
177
|
+
const payload = parseMcpUpsertPayload(req.body);
|
|
178
|
+
const server = await providerMcpService.upsertProviderMcpServer(provider, payload);
|
|
179
|
+
res.status(201).json(createApiSuccessResponse({ server }));
|
|
180
|
+
}),
|
|
181
|
+
);
|
|
182
|
+
|
|
183
|
+
router.delete(
|
|
184
|
+
'/:provider/mcp/servers/:name',
|
|
185
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
186
|
+
const provider = parseProvider(req.params.provider);
|
|
187
|
+
const scope = parseMcpScope(req.query.scope);
|
|
188
|
+
const workspacePath = readOptionalQueryString(req.query.workspacePath);
|
|
189
|
+
const result = await providerMcpService.removeProviderMcpServer(provider, {
|
|
190
|
+
name: readPathParam(req.params.name, 'name'),
|
|
191
|
+
scope,
|
|
192
|
+
workspacePath,
|
|
193
|
+
});
|
|
194
|
+
res.json(createApiSuccessResponse(result));
|
|
195
|
+
}),
|
|
196
|
+
);
|
|
197
|
+
|
|
198
|
+
router.post(
|
|
199
|
+
'/mcp/servers/global',
|
|
200
|
+
asyncHandler(async (req: Request, res: Response) => {
|
|
201
|
+
const payload = parseMcpUpsertPayload(req.body);
|
|
202
|
+
if (payload.scope === 'local') {
|
|
203
|
+
throw new AppError('Global MCP add supports only "user" or "project" scopes.', {
|
|
204
|
+
code: 'INVALID_GLOBAL_MCP_SCOPE',
|
|
205
|
+
statusCode: 400,
|
|
206
|
+
});
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
const results = await providerMcpService.addMcpServerToAllProviders({
|
|
210
|
+
...payload,
|
|
211
|
+
scope: payload.scope === 'user' ? 'user' : 'project',
|
|
212
|
+
});
|
|
213
|
+
res.status(201).json(createApiSuccessResponse({ results }));
|
|
214
|
+
}),
|
|
215
|
+
);
|
|
216
|
+
|
|
217
|
+
export default router;
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
import os from 'node:os';
|
|
2
|
+
|
|
3
|
+
import { providerRegistry } from '@/modules/providers/provider.registry.js';
|
|
4
|
+
import type { LLMProvider, McpScope, ProviderMcpServer, UpsertProviderMcpServerInput } from '@/shared/types.js';
|
|
5
|
+
import { AppError } from '@/shared/utils.js';
|
|
6
|
+
|
|
7
|
+
/** Cursor MCP is not supported on Windows hosts (no Cursor CLI integration). */
|
|
8
|
+
function includeProviderInGlobalMcp(providerId: LLMProvider): boolean {
|
|
9
|
+
if (providerId === 'cursor' && os.platform() === 'win32') {
|
|
10
|
+
return false;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
return true;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
export const providerMcpService = {
|
|
18
|
+
/**
|
|
19
|
+
* Lists MCP servers for one provider grouped by supported scopes.
|
|
20
|
+
*/
|
|
21
|
+
async listProviderMcpServers(
|
|
22
|
+
providerName: string,
|
|
23
|
+
options?: { workspacePath?: string },
|
|
24
|
+
): Promise<Record<McpScope, ProviderMcpServer[]>> {
|
|
25
|
+
const provider = providerRegistry.resolveProvider(providerName);
|
|
26
|
+
return provider.mcp.listServers(options);
|
|
27
|
+
},
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Lists MCP servers for one provider scope.
|
|
31
|
+
*/
|
|
32
|
+
async listProviderMcpServersForScope(
|
|
33
|
+
providerName: string,
|
|
34
|
+
scope: McpScope,
|
|
35
|
+
options?: { workspacePath?: string },
|
|
36
|
+
): Promise<ProviderMcpServer[]> {
|
|
37
|
+
const provider = providerRegistry.resolveProvider(providerName);
|
|
38
|
+
return provider.mcp.listServersForScope(scope, options);
|
|
39
|
+
},
|
|
40
|
+
|
|
41
|
+
/**
|
|
42
|
+
* Adds or updates one provider MCP server.
|
|
43
|
+
*/
|
|
44
|
+
async upsertProviderMcpServer(
|
|
45
|
+
providerName: string,
|
|
46
|
+
input: UpsertProviderMcpServerInput,
|
|
47
|
+
): Promise<ProviderMcpServer> {
|
|
48
|
+
const provider = providerRegistry.resolveProvider(providerName);
|
|
49
|
+
return provider.mcp.upsertServer(input);
|
|
50
|
+
},
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Removes one provider MCP server.
|
|
54
|
+
*/
|
|
55
|
+
async removeProviderMcpServer(
|
|
56
|
+
providerName: string,
|
|
57
|
+
input: { name: string; scope?: McpScope; workspacePath?: string },
|
|
58
|
+
): Promise<{ removed: boolean; provider: LLMProvider; name: string; scope: McpScope }> {
|
|
59
|
+
const provider = providerRegistry.resolveProvider(providerName);
|
|
60
|
+
return provider.mcp.removeServer(input);
|
|
61
|
+
},
|
|
62
|
+
|
|
63
|
+
/**
|
|
64
|
+
* Adds one HTTP/stdio MCP server to every provider.
|
|
65
|
+
*/
|
|
66
|
+
async addMcpServerToAllProviders(
|
|
67
|
+
input: Omit<UpsertProviderMcpServerInput, 'scope'> & { scope?: Exclude<McpScope, 'local'> },
|
|
68
|
+
): Promise<Array<{ provider: LLMProvider; created: boolean; error?: string }>> {
|
|
69
|
+
if (input.transport !== 'stdio' && input.transport !== 'http') {
|
|
70
|
+
throw new AppError('Global MCP add supports only "stdio" and "http".', {
|
|
71
|
+
code: 'INVALID_GLOBAL_MCP_TRANSPORT',
|
|
72
|
+
statusCode: 400,
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
const scope = input.scope ?? 'project';
|
|
77
|
+
const results: Array<{ provider: LLMProvider; created: boolean; error?: string }> = [];
|
|
78
|
+
const providers = providerRegistry.listProviders().filter((p) => includeProviderInGlobalMcp(p.id));
|
|
79
|
+
for (const provider of providers) {
|
|
80
|
+
try {
|
|
81
|
+
await provider.mcp.upsertServer({ ...input, scope });
|
|
82
|
+
results.push({ provider: provider.id, created: true });
|
|
83
|
+
} catch (error) {
|
|
84
|
+
results.push({
|
|
85
|
+
provider: provider.id,
|
|
86
|
+
created: false,
|
|
87
|
+
error: error instanceof Error ? error.message : 'Unknown error',
|
|
88
|
+
});
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
return results;
|
|
93
|
+
},
|
|
94
|
+
};
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import { providerRegistry } from '@/modules/providers/provider.registry.js';
|
|
2
|
+
import type { LLMProvider, ProviderAuthStatus } from '@/shared/types.js';
|
|
3
|
+
|
|
4
|
+
export const providerAuthService = {
|
|
5
|
+
/**
|
|
6
|
+
* Resolves a provider and returns its installation/authentication status.
|
|
7
|
+
*/
|
|
8
|
+
async getProviderAuthStatus(providerName: string): Promise<ProviderAuthStatus> {
|
|
9
|
+
const provider = providerRegistry.resolveProvider(providerName);
|
|
10
|
+
return provider.auth.getStatus();
|
|
11
|
+
},
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Returns whether a provider runtime appears installed.
|
|
15
|
+
* Falls back to true if status lookup itself fails so callers preserve the
|
|
16
|
+
* original runtime error instead of replacing it with a status-check failure.
|
|
17
|
+
*/
|
|
18
|
+
async isProviderInstalled(providerName: LLMProvider): Promise<boolean> {
|
|
19
|
+
try {
|
|
20
|
+
const status = await this.getProviderAuthStatus(providerName);
|
|
21
|
+
return status.installed;
|
|
22
|
+
} catch {
|
|
23
|
+
return true;
|
|
24
|
+
}
|
|
25
|
+
},
|
|
26
|
+
};
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { providerRegistry } from '@/modules/providers/provider.registry.js';
|
|
2
|
+
import type {
|
|
3
|
+
FetchHistoryOptions,
|
|
4
|
+
FetchHistoryResult,
|
|
5
|
+
LLMProvider,
|
|
6
|
+
NormalizedMessage,
|
|
7
|
+
} from '@/shared/types.js';
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* Application service for provider-backed session message operations.
|
|
11
|
+
*
|
|
12
|
+
* Callers pass a provider id and this service resolves the concrete provider
|
|
13
|
+
* class, keeping normalization/history call sites decoupled from implementation
|
|
14
|
+
* file layout.
|
|
15
|
+
*/
|
|
16
|
+
export const sessionsService = {
|
|
17
|
+
/**
|
|
18
|
+
* Lists provider ids that can load session history and normalize live messages.
|
|
19
|
+
*/
|
|
20
|
+
listProviderIds(): LLMProvider[] {
|
|
21
|
+
return providerRegistry.listProviders().map((provider) => provider.id);
|
|
22
|
+
},
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Normalizes one provider-native event into frontend session message events.
|
|
26
|
+
*/
|
|
27
|
+
normalizeMessage(
|
|
28
|
+
providerName: string,
|
|
29
|
+
raw: unknown,
|
|
30
|
+
sessionId: string | null,
|
|
31
|
+
): NormalizedMessage[] {
|
|
32
|
+
return providerRegistry.resolveProvider(providerName).sessions.normalizeMessage(raw, sessionId);
|
|
33
|
+
},
|
|
34
|
+
|
|
35
|
+
/**
|
|
36
|
+
* Fetches normalized persisted session history for one provider/session pair.
|
|
37
|
+
*/
|
|
38
|
+
fetchHistory(
|
|
39
|
+
providerName: string,
|
|
40
|
+
sessionId: string,
|
|
41
|
+
options?: FetchHistoryOptions,
|
|
42
|
+
): Promise<FetchHistoryResult> {
|
|
43
|
+
return providerRegistry.resolveProvider(providerName).sessions.fetchHistory(sessionId, options);
|
|
44
|
+
},
|
|
45
|
+
};
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
import type { IProvider, IProviderAuth, IProviderMcp, IProviderSessions } from '@/shared/interfaces.js';
|
|
2
|
+
import type { LLMProvider } from '@/shared/types.js';
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Shared provider base.
|
|
6
|
+
*
|
|
7
|
+
* Concrete providers must expose auth/MCP handlers and implement message
|
|
8
|
+
* normalization/history loading because those behaviors depend on native
|
|
9
|
+
* SDK/CLI formats.
|
|
10
|
+
*/
|
|
11
|
+
export abstract class AbstractProvider implements IProvider {
|
|
12
|
+
readonly id: LLMProvider;
|
|
13
|
+
abstract readonly mcp: IProviderMcp;
|
|
14
|
+
abstract readonly auth: IProviderAuth;
|
|
15
|
+
abstract readonly sessions: IProviderSessions;
|
|
16
|
+
|
|
17
|
+
protected constructor(id: LLMProvider) {
|
|
18
|
+
this.id = id;
|
|
19
|
+
}
|
|
20
|
+
}
|