mirror of
https://github.com/siteboon/claudecodeui.git
synced 2026-03-05 14:07:40 +00:00
* feat: implement session rename with SQLite storage (closes #72, fixes #358) - Add session_names table to store custom display names per provider - Add PUT /api/sessions/:sessionId/rename endpoint - Replace stub updateSessionSummary with real API call - Apply custom names across all providers (Claude, Codex, Cursor) - Fix project rename destroying config (spread merge instead of overwrite) - Thread provider parameter through sidebar component chain - Add i18n error messages for rename failures (en, ja, ko, zh-CN) * fix: address CodeRabbit review feedback for session rename - Log migration errors instead of swallowing them silently (db.js) - Add try/catch to applyCustomSessionNames to prevent getProjects abort - Move applyCustomSessionNames to db.js as shared helper (DRY) - Fix Cursor getSessionName to check session.summary for custom names - Move edit state clearing to finally block in updateSessionSummary - Sanitize sessionId, add 500-char summary limit, validate provider whitelist - Remove dead applyCustomSessionNames call on empty manual project sessions * fix: reject sessionId on mismatch instead of silent normalization Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com> * fix: enable rename for all providers, add Gemini support, clean up orphans - Enable rename UI (pencil icon) for Codex, Cursor, and Gemini sessions - Keep delete button hidden for Cursor (no backend delete endpoint) - Add 'gemini' to VALID_PROVIDERS and hoist to module scope - Add sessionNamesDb.deleteName on session delete (claude, codex, gemini) - Fix token-usage endpoint sessionId mismatch validation - Remove redundant try/catch in sessionNamesDb methods - Let session_names migration errors propagate to outer handler --------- Co-authored-by: coderabbitai[bot] <136622811+coderabbitai[bot]@users.noreply.github.com> Co-authored-by: Haileyesus <118998054+blackmammoth@users.noreply.github.com>
445 lines
14 KiB
JavaScript
445 lines
14 KiB
JavaScript
import Database from 'better-sqlite3';
|
|
import path from 'path';
|
|
import fs from 'fs';
|
|
import crypto from 'crypto';
|
|
import { fileURLToPath } from 'url';
|
|
import { dirname } from 'path';
|
|
|
|
const __filename = fileURLToPath(import.meta.url);
|
|
const __dirname = dirname(__filename);
|
|
|
|
// ANSI color codes for terminal output
|
|
const colors = {
|
|
reset: '\x1b[0m',
|
|
bright: '\x1b[1m',
|
|
cyan: '\x1b[36m',
|
|
dim: '\x1b[2m',
|
|
};
|
|
|
|
const c = {
|
|
info: (text) => `${colors.cyan}${text}${colors.reset}`,
|
|
bright: (text) => `${colors.bright}${text}${colors.reset}`,
|
|
dim: (text) => `${colors.dim}${text}${colors.reset}`,
|
|
};
|
|
|
|
// Use DATABASE_PATH environment variable if set, otherwise use default location
|
|
const DB_PATH = process.env.DATABASE_PATH || path.join(__dirname, 'auth.db');
|
|
const INIT_SQL_PATH = path.join(__dirname, 'init.sql');
|
|
|
|
// Ensure database directory exists if custom path is provided
|
|
if (process.env.DATABASE_PATH) {
|
|
const dbDir = path.dirname(DB_PATH);
|
|
try {
|
|
if (!fs.existsSync(dbDir)) {
|
|
fs.mkdirSync(dbDir, { recursive: true });
|
|
console.log(`Created database directory: ${dbDir}`);
|
|
}
|
|
} catch (error) {
|
|
console.error(`Failed to create database directory ${dbDir}:`, error.message);
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
// As part of 1.19.2 we are introducing a new location for auth.db. The below handles exisitng moving legacy database from install directory to new location
|
|
const LEGACY_DB_PATH = path.join(__dirname, 'auth.db');
|
|
if (DB_PATH !== LEGACY_DB_PATH && !fs.existsSync(DB_PATH) && fs.existsSync(LEGACY_DB_PATH)) {
|
|
try {
|
|
fs.copyFileSync(LEGACY_DB_PATH, DB_PATH);
|
|
console.log(`[MIGRATION] Copied database from ${LEGACY_DB_PATH} to ${DB_PATH}`);
|
|
for (const suffix of ['-wal', '-shm']) {
|
|
if (fs.existsSync(LEGACY_DB_PATH + suffix)) {
|
|
fs.copyFileSync(LEGACY_DB_PATH + suffix, DB_PATH + suffix);
|
|
}
|
|
}
|
|
} catch (err) {
|
|
console.warn(`[MIGRATION] Could not copy legacy database: ${err.message}`);
|
|
}
|
|
}
|
|
|
|
// Create database connection
|
|
const db = new Database(DB_PATH);
|
|
|
|
// Show app installation path prominently
|
|
const appInstallPath = path.join(__dirname, '../..');
|
|
console.log('');
|
|
console.log(c.dim('═'.repeat(60)));
|
|
console.log(`${c.info('[INFO]')} App Installation: ${c.bright(appInstallPath)}`);
|
|
console.log(`${c.info('[INFO]')} Database: ${c.dim(path.relative(appInstallPath, DB_PATH))}`);
|
|
if (process.env.DATABASE_PATH) {
|
|
console.log(` ${c.dim('(Using custom DATABASE_PATH from environment)')}`);
|
|
}
|
|
console.log(c.dim('═'.repeat(60)));
|
|
console.log('');
|
|
|
|
const runMigrations = () => {
|
|
try {
|
|
const tableInfo = db.prepare("PRAGMA table_info(users)").all();
|
|
const columnNames = tableInfo.map(col => col.name);
|
|
|
|
if (!columnNames.includes('git_name')) {
|
|
console.log('Running migration: Adding git_name column');
|
|
db.exec('ALTER TABLE users ADD COLUMN git_name TEXT');
|
|
}
|
|
|
|
if (!columnNames.includes('git_email')) {
|
|
console.log('Running migration: Adding git_email column');
|
|
db.exec('ALTER TABLE users ADD COLUMN git_email TEXT');
|
|
}
|
|
|
|
if (!columnNames.includes('has_completed_onboarding')) {
|
|
console.log('Running migration: Adding has_completed_onboarding column');
|
|
db.exec('ALTER TABLE users ADD COLUMN has_completed_onboarding BOOLEAN DEFAULT 0');
|
|
}
|
|
|
|
// Create session_names table if it doesn't exist (for existing installations)
|
|
db.exec(`CREATE TABLE IF NOT EXISTS session_names (
|
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
session_id TEXT NOT NULL,
|
|
provider TEXT NOT NULL DEFAULT 'claude',
|
|
custom_name TEXT NOT NULL,
|
|
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
|
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
|
UNIQUE(session_id, provider)
|
|
)`);
|
|
db.exec('CREATE INDEX IF NOT EXISTS idx_session_names_lookup ON session_names(session_id, provider)');
|
|
|
|
console.log('Database migrations completed successfully');
|
|
} catch (error) {
|
|
console.error('Error running migrations:', error.message);
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
// Initialize database with schema
|
|
const initializeDatabase = async () => {
|
|
try {
|
|
const initSQL = fs.readFileSync(INIT_SQL_PATH, 'utf8');
|
|
db.exec(initSQL);
|
|
console.log('Database initialized successfully');
|
|
runMigrations();
|
|
} catch (error) {
|
|
console.error('Error initializing database:', error.message);
|
|
throw error;
|
|
}
|
|
};
|
|
|
|
// User database operations
|
|
const userDb = {
|
|
// Check if any users exist
|
|
hasUsers: () => {
|
|
try {
|
|
const row = db.prepare('SELECT COUNT(*) as count FROM users').get();
|
|
return row.count > 0;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
// Create a new user
|
|
createUser: (username, passwordHash) => {
|
|
try {
|
|
const stmt = db.prepare('INSERT INTO users (username, password_hash) VALUES (?, ?)');
|
|
const result = stmt.run(username, passwordHash);
|
|
return { id: result.lastInsertRowid, username };
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
// Get user by username
|
|
getUserByUsername: (username) => {
|
|
try {
|
|
const row = db.prepare('SELECT * FROM users WHERE username = ? AND is_active = 1').get(username);
|
|
return row;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
// Update last login time (non-fatal — logged but not thrown)
|
|
updateLastLogin: (userId) => {
|
|
try {
|
|
db.prepare('UPDATE users SET last_login = CURRENT_TIMESTAMP WHERE id = ?').run(userId);
|
|
} catch (err) {
|
|
console.warn('Failed to update last login:', err.message);
|
|
}
|
|
},
|
|
|
|
// Get user by ID
|
|
getUserById: (userId) => {
|
|
try {
|
|
const row = db.prepare('SELECT id, username, created_at, last_login FROM users WHERE id = ? AND is_active = 1').get(userId);
|
|
return row;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
getFirstUser: () => {
|
|
try {
|
|
const row = db.prepare('SELECT id, username, created_at, last_login FROM users WHERE is_active = 1 LIMIT 1').get();
|
|
return row;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
updateGitConfig: (userId, gitName, gitEmail) => {
|
|
try {
|
|
const stmt = db.prepare('UPDATE users SET git_name = ?, git_email = ? WHERE id = ?');
|
|
stmt.run(gitName, gitEmail, userId);
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
getGitConfig: (userId) => {
|
|
try {
|
|
const row = db.prepare('SELECT git_name, git_email FROM users WHERE id = ?').get(userId);
|
|
return row;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
completeOnboarding: (userId) => {
|
|
try {
|
|
const stmt = db.prepare('UPDATE users SET has_completed_onboarding = 1 WHERE id = ?');
|
|
stmt.run(userId);
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
hasCompletedOnboarding: (userId) => {
|
|
try {
|
|
const row = db.prepare('SELECT has_completed_onboarding FROM users WHERE id = ?').get(userId);
|
|
return row?.has_completed_onboarding === 1;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
}
|
|
};
|
|
|
|
// API Keys database operations
|
|
const apiKeysDb = {
|
|
// Generate a new API key
|
|
generateApiKey: () => {
|
|
return 'ck_' + crypto.randomBytes(32).toString('hex');
|
|
},
|
|
|
|
// Create a new API key
|
|
createApiKey: (userId, keyName) => {
|
|
try {
|
|
const apiKey = apiKeysDb.generateApiKey();
|
|
const stmt = db.prepare('INSERT INTO api_keys (user_id, key_name, api_key) VALUES (?, ?, ?)');
|
|
const result = stmt.run(userId, keyName, apiKey);
|
|
return { id: result.lastInsertRowid, keyName, apiKey };
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
// Get all API keys for a user
|
|
getApiKeys: (userId) => {
|
|
try {
|
|
const rows = db.prepare('SELECT id, key_name, api_key, created_at, last_used, is_active FROM api_keys WHERE user_id = ? ORDER BY created_at DESC').all(userId);
|
|
return rows;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
// Validate API key and get user
|
|
validateApiKey: (apiKey) => {
|
|
try {
|
|
const row = db.prepare(`
|
|
SELECT u.id, u.username, ak.id as api_key_id
|
|
FROM api_keys ak
|
|
JOIN users u ON ak.user_id = u.id
|
|
WHERE ak.api_key = ? AND ak.is_active = 1 AND u.is_active = 1
|
|
`).get(apiKey);
|
|
|
|
if (row) {
|
|
// Update last_used timestamp
|
|
db.prepare('UPDATE api_keys SET last_used = CURRENT_TIMESTAMP WHERE id = ?').run(row.api_key_id);
|
|
}
|
|
|
|
return row;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
// Delete an API key
|
|
deleteApiKey: (userId, apiKeyId) => {
|
|
try {
|
|
const stmt = db.prepare('DELETE FROM api_keys WHERE id = ? AND user_id = ?');
|
|
const result = stmt.run(apiKeyId, userId);
|
|
return result.changes > 0;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
// Toggle API key active status
|
|
toggleApiKey: (userId, apiKeyId, isActive) => {
|
|
try {
|
|
const stmt = db.prepare('UPDATE api_keys SET is_active = ? WHERE id = ? AND user_id = ?');
|
|
const result = stmt.run(isActive ? 1 : 0, apiKeyId, userId);
|
|
return result.changes > 0;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
}
|
|
};
|
|
|
|
// User credentials database operations (for GitHub tokens, GitLab tokens, etc.)
|
|
const credentialsDb = {
|
|
// Create a new credential
|
|
createCredential: (userId, credentialName, credentialType, credentialValue, description = null) => {
|
|
try {
|
|
const stmt = db.prepare('INSERT INTO user_credentials (user_id, credential_name, credential_type, credential_value, description) VALUES (?, ?, ?, ?, ?)');
|
|
const result = stmt.run(userId, credentialName, credentialType, credentialValue, description);
|
|
return { id: result.lastInsertRowid, credentialName, credentialType };
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
// Get all credentials for a user, optionally filtered by type
|
|
getCredentials: (userId, credentialType = null) => {
|
|
try {
|
|
let query = 'SELECT id, credential_name, credential_type, description, created_at, is_active FROM user_credentials WHERE user_id = ?';
|
|
const params = [userId];
|
|
|
|
if (credentialType) {
|
|
query += ' AND credential_type = ?';
|
|
params.push(credentialType);
|
|
}
|
|
|
|
query += ' ORDER BY created_at DESC';
|
|
|
|
const rows = db.prepare(query).all(...params);
|
|
return rows;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
// Get active credential value for a user by type (returns most recent active)
|
|
getActiveCredential: (userId, credentialType) => {
|
|
try {
|
|
const row = db.prepare('SELECT credential_value FROM user_credentials WHERE user_id = ? AND credential_type = ? AND is_active = 1 ORDER BY created_at DESC LIMIT 1').get(userId, credentialType);
|
|
return row?.credential_value || null;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
// Delete a credential
|
|
deleteCredential: (userId, credentialId) => {
|
|
try {
|
|
const stmt = db.prepare('DELETE FROM user_credentials WHERE id = ? AND user_id = ?');
|
|
const result = stmt.run(credentialId, userId);
|
|
return result.changes > 0;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
},
|
|
|
|
// Toggle credential active status
|
|
toggleCredential: (userId, credentialId, isActive) => {
|
|
try {
|
|
const stmt = db.prepare('UPDATE user_credentials SET is_active = ? WHERE id = ? AND user_id = ?');
|
|
const result = stmt.run(isActive ? 1 : 0, credentialId, userId);
|
|
return result.changes > 0;
|
|
} catch (err) {
|
|
throw err;
|
|
}
|
|
}
|
|
};
|
|
|
|
// Session custom names database operations
|
|
const sessionNamesDb = {
|
|
// Set (insert or update) a custom session name
|
|
setName: (sessionId, provider, customName) => {
|
|
db.prepare(`
|
|
INSERT INTO session_names (session_id, provider, custom_name)
|
|
VALUES (?, ?, ?)
|
|
ON CONFLICT(session_id, provider)
|
|
DO UPDATE SET custom_name = excluded.custom_name, updated_at = CURRENT_TIMESTAMP
|
|
`).run(sessionId, provider, customName);
|
|
},
|
|
|
|
// Get a single custom session name
|
|
getName: (sessionId, provider) => {
|
|
const row = db.prepare(
|
|
'SELECT custom_name FROM session_names WHERE session_id = ? AND provider = ?'
|
|
).get(sessionId, provider);
|
|
return row?.custom_name || null;
|
|
},
|
|
|
|
// Batch lookup — returns Map<sessionId, customName>
|
|
getNames: (sessionIds, provider) => {
|
|
if (!sessionIds.length) return new Map();
|
|
const placeholders = sessionIds.map(() => '?').join(',');
|
|
const rows = db.prepare(
|
|
`SELECT session_id, custom_name FROM session_names
|
|
WHERE session_id IN (${placeholders}) AND provider = ?`
|
|
).all(...sessionIds, provider);
|
|
return new Map(rows.map(r => [r.session_id, r.custom_name]));
|
|
},
|
|
|
|
// Delete a custom session name
|
|
deleteName: (sessionId, provider) => {
|
|
return db.prepare(
|
|
'DELETE FROM session_names WHERE session_id = ? AND provider = ?'
|
|
).run(sessionId, provider).changes > 0;
|
|
},
|
|
};
|
|
|
|
// Apply custom session names from the database (overrides CLI-generated summaries)
|
|
function applyCustomSessionNames(sessions, provider) {
|
|
if (!sessions?.length) return;
|
|
try {
|
|
const ids = sessions.map(s => s.id);
|
|
const customNames = sessionNamesDb.getNames(ids, provider);
|
|
for (const session of sessions) {
|
|
const custom = customNames.get(session.id);
|
|
if (custom) session.summary = custom;
|
|
}
|
|
} catch (error) {
|
|
console.warn(`[DB] Failed to apply custom session names for ${provider}:`, error.message);
|
|
}
|
|
}
|
|
|
|
// Backward compatibility - keep old names pointing to new system
|
|
const githubTokensDb = {
|
|
createGithubToken: (userId, tokenName, githubToken, description = null) => {
|
|
return credentialsDb.createCredential(userId, tokenName, 'github_token', githubToken, description);
|
|
},
|
|
getGithubTokens: (userId) => {
|
|
return credentialsDb.getCredentials(userId, 'github_token');
|
|
},
|
|
getActiveGithubToken: (userId) => {
|
|
return credentialsDb.getActiveCredential(userId, 'github_token');
|
|
},
|
|
deleteGithubToken: (userId, tokenId) => {
|
|
return credentialsDb.deleteCredential(userId, tokenId);
|
|
},
|
|
toggleGithubToken: (userId, tokenId, isActive) => {
|
|
return credentialsDb.toggleCredential(userId, tokenId, isActive);
|
|
}
|
|
};
|
|
|
|
export {
|
|
db,
|
|
initializeDatabase,
|
|
userDb,
|
|
apiKeysDb,
|
|
credentialsDb,
|
|
sessionNamesDb,
|
|
applyCustomSessionNames,
|
|
githubTokensDb // Backward compatibility
|
|
}; |