mirror of
https://github.com/siteboon/claudecodeui.git
synced 2026-02-28 03:27:40 +00:00
* feat: integrate Gemini AI agent provider - Core Backend: Ported gemini-cli.js and gemini-response-handler.js to establish the CLI bridge. Registered 'gemini' as an active provider within index.js. - Core Frontend: Extended QuickSettingsPanel.jsx, Settings.jsx, and AgentListItem.jsx to render the Gemini provider option, models (gemini-pro, gemini-flash, etc.), and handle OAuth states. - WebSocket Pipeline: Added support for gemini-command executions in backend and payload processing of gemini-response and gemini-error streams in useChatRealtimeHandlers.ts. Resolved JSON double-stringification and sessionId stripping issues in the transmission handler. - Platform Compatibility: Added scripts/fix-node-pty.js postinstall script and modified posix_spawnp calls with sh -c wrapper to prevent ENOEXEC and MacOS permission errors when spawning the gemini headless binary. - UX & Design: Imported official Google Gemini branding via GeminiLogo.jsx and gemini-ai-icon.svg. Updated translations (chat.json) for en, zh-CN, and ko locales. * fix: propagate gemini permission mode from settings to cli - Added Gemini Permissions UI in Settings to toggle Auto Edit and YOLO modes - Synced gemini permission mode to localStorage - Passed permissionMode in useChatComposerState for Gemini commands - Mapped frontend permission modes to --yolo and --approval-mode options in gemini-cli.js * feat(gemini): Refactor Gemini CLI integration to use stream-json - Replaced regex buffering text-system with NDJSON stream parsing - Added fallback for restricted models like gemini-3.1-pro-preview * feat(gemini): Render tool_use and tool_result UI bubbles - Forwarded gemini tool NDJSON objects to the websocket - Added React state handlers in useChatRealtimeHandlers to match Claude's tool UI behavior * feat(gemini): Add native session resumption and UI token tracking - Captured cliSessionId from init events to map ClaudeCodeUI's chat sessionId directly into Gemini's internal session manager. - Updated gemini-cli.js spawn arguments to append the --resume proxy flag instead of naively dumping the accumulated chat history into the command prompt. - Handled result stream objects by proxying total_tokens back into the frontend's claude-status tracker to natively populate the UI label. - Eliminated gemini-3 model proxy filter entirely. * fix(gemini): Fix static 'Claude' name rendering in chat UI header - Added "gemini": "Gemini" translation strings to messageTypes across English, Korean, and Chinese loc dictionaries. - Updated AssistantThinkingIndicator and MessageComponent ternary checks to identify provider === 'gemini' and render the appropriate brand label instead of statically defaulting to Claude. * feat: Add Gemini session persistence API mapping and Sidebar UI * fix(gemini): Watch ~/.gemini/sessions for live UI updates Added the .gemini/sessions directory to PROVIDER_WATCH_PATHS so that Chokidar emits projects_updated websocket events when new Gemini sessions are created or modified, fixing live sidebar updates. * fix(gemini): Fix Gemini authentication status display in Settings UI - Injected 'checkGeminiAuthStatus' into the Settings.jsx React effect hook so that the UI can poll and render the 'geminiAuthStatus' state. - Updated 'checkGeminiCredentials()' inside server/routes/cli-auth.js to read from '~/.gemini/oauth_creds.json' and '~/.gemini/google_accounts.json', resolving the email address correctly. * Use logo-only icon for gemini * feat(gemini): Add Gemini 3 preview models to UI selection list * Fix Gemini CLI session resume bug and PR #422 review nitpicks * Fix Gemini tool calls disappearing from UI after completion * fix(gemini): resolve outstanding PR #422 feedback and stabilize gemini CLI timeouts * fix(gemini): resolve resume flag and shell session initialization issues This commit addresses the remaining PR comments for the Gemini CLI integration: - Moves the `--resume` flag logic outside the prompt command block, ensuring Gemini sessions correctly resume even when a new prompt isn't passed. - Updates `handleShellConnection` to correctly lookup the native `cliSessionId` from the internal `sessionId` when spawning Gemini sessions in a plain shell. - Refactors dynamic import of `sessionManager.js` back to a native static import for code consistency. * chore: fix TypeScript errors and remove gemini CLI dependency * fix: use cross-spawn on Windows to resolve gemini.cmd correctly --------- Co-authored-by: Haileyesus <118998054+blackmammoth@users.noreply.github.com>
226 lines
5.8 KiB
JavaScript
226 lines
5.8 KiB
JavaScript
import { promises as fs } from 'fs';
|
|
import path from 'path';
|
|
import os from 'os';
|
|
|
|
class SessionManager {
|
|
constructor() {
|
|
// Store sessions in memory with conversation history
|
|
this.sessions = new Map();
|
|
this.maxSessions = 100;
|
|
this.sessionsDir = path.join(os.homedir(), '.gemini', 'sessions');
|
|
this.ready = this.init();
|
|
}
|
|
|
|
async init() {
|
|
await this.initSessionsDir();
|
|
await this.loadSessions();
|
|
}
|
|
|
|
async initSessionsDir() {
|
|
try {
|
|
await fs.mkdir(this.sessionsDir, { recursive: true });
|
|
} catch (error) {
|
|
// console.error('Error creating sessions directory:', error);
|
|
}
|
|
}
|
|
|
|
// Create a new session
|
|
createSession(sessionId, projectPath) {
|
|
const session = {
|
|
id: sessionId,
|
|
projectPath: projectPath,
|
|
messages: [],
|
|
createdAt: new Date(),
|
|
lastActivity: new Date()
|
|
};
|
|
|
|
// Evict oldest session from memory if we exceed limit
|
|
if (this.sessions.size >= this.maxSessions) {
|
|
const oldestKey = this.sessions.keys().next().value;
|
|
if (oldestKey) this.sessions.delete(oldestKey);
|
|
}
|
|
|
|
this.sessions.set(sessionId, session);
|
|
this.saveSession(sessionId);
|
|
|
|
return session;
|
|
}
|
|
|
|
// Add a message to session
|
|
addMessage(sessionId, role, content) {
|
|
let session = this.sessions.get(sessionId);
|
|
|
|
if (!session) {
|
|
// Create session if it doesn't exist
|
|
session = this.createSession(sessionId, '');
|
|
}
|
|
|
|
const message = {
|
|
role: role, // 'user' or 'assistant'
|
|
content: content,
|
|
timestamp: new Date()
|
|
};
|
|
|
|
session.messages.push(message);
|
|
session.lastActivity = new Date();
|
|
|
|
this.saveSession(sessionId);
|
|
|
|
return session;
|
|
}
|
|
|
|
// Get session by ID
|
|
getSession(sessionId) {
|
|
return this.sessions.get(sessionId);
|
|
}
|
|
|
|
// Get all sessions for a project
|
|
getProjectSessions(projectPath) {
|
|
const sessions = [];
|
|
|
|
for (const [id, session] of this.sessions) {
|
|
if (session.projectPath === projectPath) {
|
|
sessions.push({
|
|
id: session.id,
|
|
summary: this.getSessionSummary(session),
|
|
messageCount: session.messages.length,
|
|
lastActivity: session.lastActivity
|
|
});
|
|
}
|
|
}
|
|
|
|
return sessions.sort((a, b) =>
|
|
new Date(b.lastActivity) - new Date(a.lastActivity)
|
|
);
|
|
}
|
|
|
|
// Get session summary
|
|
getSessionSummary(session) {
|
|
if (session.messages.length === 0) {
|
|
return 'New Session';
|
|
}
|
|
|
|
// Find first user message
|
|
const firstUserMessage = session.messages.find(m => m.role === 'user');
|
|
if (firstUserMessage) {
|
|
const content = firstUserMessage.content;
|
|
return content.length > 50 ? content.substring(0, 50) + '...' : content;
|
|
}
|
|
|
|
return 'New Session';
|
|
}
|
|
|
|
// Build conversation context for Gemini
|
|
buildConversationContext(sessionId, maxMessages = 10) {
|
|
const session = this.sessions.get(sessionId);
|
|
|
|
if (!session || session.messages.length === 0) {
|
|
return '';
|
|
}
|
|
|
|
// Get last N messages for context
|
|
const recentMessages = session.messages.slice(-maxMessages);
|
|
|
|
let context = 'Here is the conversation history:\n\n';
|
|
|
|
for (const msg of recentMessages) {
|
|
if (msg.role === 'user') {
|
|
context += `User: ${msg.content}\n`;
|
|
} else {
|
|
context += `Assistant: ${msg.content}\n`;
|
|
}
|
|
}
|
|
|
|
context += '\nBased on the conversation history above, please answer the following:\n';
|
|
|
|
return context;
|
|
}
|
|
|
|
// Prevent path traversal
|
|
_safeFilePath(sessionId) {
|
|
const safeId = String(sessionId).replace(/[/\\]|\.\./g, '');
|
|
return path.join(this.sessionsDir, `${safeId}.json`);
|
|
}
|
|
|
|
// Save session to disk
|
|
async saveSession(sessionId) {
|
|
const session = this.sessions.get(sessionId);
|
|
if (!session) return;
|
|
|
|
try {
|
|
const filePath = this._safeFilePath(sessionId);
|
|
await fs.writeFile(filePath, JSON.stringify(session, null, 2));
|
|
} catch (error) {
|
|
// console.error('Error saving session:', error);
|
|
}
|
|
}
|
|
|
|
// Load sessions from disk
|
|
async loadSessions() {
|
|
try {
|
|
const files = await fs.readdir(this.sessionsDir);
|
|
|
|
for (const file of files) {
|
|
if (file.endsWith('.json')) {
|
|
try {
|
|
const filePath = path.join(this.sessionsDir, file);
|
|
const data = await fs.readFile(filePath, 'utf8');
|
|
const session = JSON.parse(data);
|
|
|
|
// Convert dates
|
|
session.createdAt = new Date(session.createdAt);
|
|
session.lastActivity = new Date(session.lastActivity);
|
|
session.messages.forEach(msg => {
|
|
msg.timestamp = new Date(msg.timestamp);
|
|
});
|
|
|
|
this.sessions.set(session.id, session);
|
|
} catch (error) {
|
|
// console.error(`Error loading session ${file}:`, error);
|
|
}
|
|
}
|
|
}
|
|
|
|
// Enforce eviction after loading to prevent massive memory usage
|
|
while (this.sessions.size > this.maxSessions) {
|
|
const oldestKey = this.sessions.keys().next().value;
|
|
if (oldestKey) this.sessions.delete(oldestKey);
|
|
}
|
|
} catch (error) {
|
|
// console.error('Error loading sessions:', error);
|
|
}
|
|
}
|
|
|
|
// Delete a session
|
|
async deleteSession(sessionId) {
|
|
this.sessions.delete(sessionId);
|
|
|
|
try {
|
|
const filePath = this._safeFilePath(sessionId);
|
|
await fs.unlink(filePath);
|
|
} catch (error) {
|
|
// console.error('Error deleting session file:', error);
|
|
}
|
|
}
|
|
|
|
// Get session messages for display
|
|
getSessionMessages(sessionId) {
|
|
const session = this.sessions.get(sessionId);
|
|
if (!session) return [];
|
|
|
|
return session.messages.map(msg => ({
|
|
type: 'message',
|
|
message: {
|
|
role: msg.role,
|
|
content: msg.content
|
|
},
|
|
timestamp: msg.timestamp.toISOString()
|
|
}));
|
|
}
|
|
}
|
|
|
|
// Singleton instance
|
|
const sessionManager = new SessionManager();
|
|
|
|
export const ready = sessionManager.ready;
|
|
export default sessionManager; |