mirror of
https://github.com/siteboon/claudecodeui.git
synced 2026-05-09 22:18:19 +00:00
Users previously had an all-or-nothing choice for completed sessions: either keep them in the active sidebar or permanently delete them. That made long-lived usage brittle because valuable history stayed in the way unless users destroyed it. This change introduces archiving as a first-class lifecycle so completed work can be hidden without losing transcript history, workspace context, or restoreability. The backend now persists session archive state and excludes archived rows from active session queries by default. Dedicated archive queries and routes make archived sessions and archived workspaces addressable on their own, which is necessary once hidden data can no longer be rebuilt from the active project list. Hard-delete behavior still cleans up transcript files so destructive deletes remain truly destructive. The frontend now mirrors that lifecycle in the sidebar. Delete flows distinguish between archive and permanent delete, archived sessions can be restored, archived workspaces appear beside standalone archived sessions, and archived project sessions open with the correct workspace context instead of routing to a session URL that leaves the main view empty. Follow-up archive UI polish keeps the status affordances explicit without competing with workspace names.
154 lines
5.0 KiB
TypeScript
154 lines
5.0 KiB
TypeScript
const USER_TABLE_SCHEMA_SQL = `
|
|
CREATE TABLE IF NOT EXISTS users (
|
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
username TEXT UNIQUE NOT NULL,
|
|
password_hash TEXT NOT NULL,
|
|
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
|
last_login DATETIME,
|
|
is_active BOOLEAN DEFAULT 1,
|
|
git_name TEXT,
|
|
git_email TEXT,
|
|
has_completed_onboarding BOOLEAN DEFAULT 0
|
|
);
|
|
`;
|
|
|
|
export const API_KEYS_TABLE_SCHEMA_SQL = `
|
|
CREATE TABLE IF NOT EXISTS api_keys (
|
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
user_id INTEGER NOT NULL,
|
|
key_name TEXT NOT NULL,
|
|
api_key TEXT UNIQUE NOT NULL,
|
|
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
|
last_used DATETIME,
|
|
is_active BOOLEAN DEFAULT 1,
|
|
FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE CASCADE
|
|
);
|
|
`;
|
|
|
|
export const USER_CREDENTIALS_TABLE_SCHEMA_SQL = `
|
|
CREATE TABLE IF NOT EXISTS user_credentials (
|
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
user_id INTEGER NOT NULL,
|
|
credential_name TEXT NOT NULL,
|
|
credential_type TEXT NOT NULL, -- 'github_token', 'gitlab_token', 'bitbucket_token', etc.
|
|
credential_value TEXT NOT NULL,
|
|
description TEXT,
|
|
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
|
is_active BOOLEAN DEFAULT 1,
|
|
FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE CASCADE
|
|
);
|
|
`;
|
|
|
|
export const USER_NOTIFICATION_PREFERENCES_TABLE_SCHEMA_SQL = `
|
|
CREATE TABLE IF NOT EXISTS user_notification_preferences (
|
|
user_id INTEGER PRIMARY KEY,
|
|
preferences_json TEXT NOT NULL,
|
|
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
|
FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE CASCADE
|
|
);
|
|
`;
|
|
|
|
export const VAPID_KEYS_TABLE_SCHEMA_SQL = `
|
|
CREATE TABLE IF NOT EXISTS vapid_keys (
|
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
public_key TEXT NOT NULL,
|
|
private_key TEXT NOT NULL,
|
|
created_at DATETIME DEFAULT CURRENT_TIMESTAMP
|
|
);
|
|
`;
|
|
|
|
export const PUSH_SUBSCRIPTIONS_TABLE_SCHEMA_SQL = `
|
|
CREATE TABLE IF NOT EXISTS push_subscriptions (
|
|
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
user_id INTEGER NOT NULL,
|
|
endpoint TEXT NOT NULL UNIQUE,
|
|
keys_p256dh TEXT NOT NULL,
|
|
keys_auth TEXT NOT NULL,
|
|
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
|
FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE CASCADE
|
|
);
|
|
`;
|
|
|
|
export const PROJECTS_TABLE_SCHEMA_SQL = `
|
|
CREATE TABLE IF NOT EXISTS projects (
|
|
project_id TEXT PRIMARY KEY NOT NULL,
|
|
project_path TEXT NOT NULL UNIQUE,
|
|
custom_project_name TEXT DEFAULT NULL,
|
|
isStarred BOOLEAN DEFAULT 0,
|
|
isArchived BOOLEAN DEFAULT 0
|
|
);
|
|
`;
|
|
|
|
export const SESSIONS_TABLE_SCHEMA_SQL = `
|
|
CREATE TABLE IF NOT EXISTS sessions (
|
|
session_id TEXT NOT NULL,
|
|
provider TEXT NOT NULL DEFAULT 'claude',
|
|
custom_name TEXT,
|
|
project_path TEXT,
|
|
jsonl_path TEXT,
|
|
isArchived BOOLEAN DEFAULT 0,
|
|
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
|
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
|
PRIMARY KEY (session_id),
|
|
FOREIGN KEY (project_path) REFERENCES projects(project_path)
|
|
ON DELETE SET NULL
|
|
ON UPDATE CASCADE
|
|
);
|
|
`;
|
|
|
|
export const LAST_SCANNED_AT_SQL = `
|
|
CREATE TABLE IF NOT EXISTS scan_state (
|
|
id INTEGER PRIMARY KEY CHECK (id = 1),
|
|
last_scanned_at TIMESTAMP NULL
|
|
);
|
|
`;
|
|
|
|
export const APP_CONFIG_TABLE_SCHEMA_SQL = `
|
|
CREATE TABLE IF NOT EXISTS app_config (
|
|
key TEXT PRIMARY KEY,
|
|
value TEXT NOT NULL,
|
|
created_at DATETIME DEFAULT CURRENT_TIMESTAMP
|
|
);
|
|
`;
|
|
|
|
export const INIT_SCHEMA_SQL = `
|
|
-- Initialize authentication database
|
|
PRAGMA foreign_keys = ON;
|
|
|
|
${USER_TABLE_SCHEMA_SQL}
|
|
-- Indexes for performance for user lookups
|
|
CREATE INDEX IF NOT EXISTS idx_users_username ON users(username);
|
|
CREATE INDEX IF NOT EXISTS idx_users_active ON users(is_active);
|
|
|
|
${API_KEYS_TABLE_SCHEMA_SQL}
|
|
CREATE INDEX IF NOT EXISTS idx_api_keys_key ON api_keys(api_key);
|
|
CREATE INDEX IF NOT EXISTS idx_api_keys_user_id ON api_keys(user_id);
|
|
CREATE INDEX IF NOT EXISTS idx_api_keys_active ON api_keys(is_active);
|
|
|
|
${USER_CREDENTIALS_TABLE_SCHEMA_SQL}
|
|
CREATE INDEX IF NOT EXISTS idx_user_credentials_user_id ON user_credentials(user_id);
|
|
CREATE INDEX IF NOT EXISTS idx_user_credentials_type ON user_credentials(credential_type);
|
|
CREATE INDEX IF NOT EXISTS idx_user_credentials_active ON user_credentials(is_active);
|
|
|
|
${USER_NOTIFICATION_PREFERENCES_TABLE_SCHEMA_SQL}
|
|
CREATE INDEX IF NOT EXISTS idx_user_notification_preferences_user_id ON user_notification_preferences(user_id);
|
|
|
|
${VAPID_KEYS_TABLE_SCHEMA_SQL}
|
|
|
|
${PUSH_SUBSCRIPTIONS_TABLE_SCHEMA_SQL}
|
|
CREATE INDEX IF NOT EXISTS idx_push_subscriptions_user_id ON push_subscriptions(user_id);
|
|
|
|
${PROJECTS_TABLE_SCHEMA_SQL}
|
|
-- NOTE: These indexes are created in migrations after legacy table-shape repairs.
|
|
-- Creating them here can fail on upgraded installs where projects lacks those columns.
|
|
|
|
${SESSIONS_TABLE_SCHEMA_SQL}
|
|
CREATE INDEX IF NOT EXISTS idx_session_ids_lookup ON sessions(session_id);
|
|
-- NOTE: This index is created in migrations after sessions is rebuilt to include project_path.
|
|
-- Creating it here can fail on upgraded installs where the legacy sessions table has no project_path.
|
|
|
|
${LAST_SCANNED_AT_SQL}
|
|
|
|
${APP_CONFIG_TABLE_SCHEMA_SQL}
|
|
`;
|