mirror of
https://github.com/farion1231/cc-switch.git
synced 2026-04-02 18:12:05 +08:00
Add delete_session Tauri command dispatching to provider-specific deletion logic for all 5 providers (Claude, Codex, Gemini, OpenCode, OpenClaw). Includes path traversal protection via canonicalize + starts_with validation, session ID verification against file contents, frontend confirmation dialog with optimistic cache updates, i18n keys (zh/en/ja), and component tests.
408 lines
9.6 KiB
TypeScript
408 lines
9.6 KiB
TypeScript
import type { AppId } from "@/lib/api/types";
|
|
import type {
|
|
McpServer,
|
|
Provider,
|
|
SessionMessage,
|
|
SessionMeta,
|
|
Settings,
|
|
} from "@/types";
|
|
|
|
type ProvidersByApp = Record<AppId, Record<string, Provider>>;
|
|
type CurrentProviderState = Record<AppId, string>;
|
|
type McpConfigState = Record<AppId, Record<string, McpServer>>;
|
|
|
|
const createDefaultProviders = (): ProvidersByApp => ({
|
|
claude: {
|
|
"claude-1": {
|
|
id: "claude-1",
|
|
name: "Claude Default",
|
|
settingsConfig: {},
|
|
category: "official",
|
|
sortIndex: 0,
|
|
createdAt: Date.now(),
|
|
},
|
|
"claude-2": {
|
|
id: "claude-2",
|
|
name: "Claude Custom",
|
|
settingsConfig: {},
|
|
category: "custom",
|
|
sortIndex: 1,
|
|
createdAt: Date.now() + 1,
|
|
},
|
|
},
|
|
codex: {
|
|
"codex-1": {
|
|
id: "codex-1",
|
|
name: "Codex Default",
|
|
settingsConfig: {},
|
|
category: "official",
|
|
sortIndex: 0,
|
|
createdAt: Date.now(),
|
|
},
|
|
"codex-2": {
|
|
id: "codex-2",
|
|
name: "Codex Secondary",
|
|
settingsConfig: {},
|
|
category: "custom",
|
|
sortIndex: 1,
|
|
createdAt: Date.now() + 1,
|
|
},
|
|
},
|
|
gemini: {
|
|
"gemini-1": {
|
|
id: "gemini-1",
|
|
name: "Gemini Default",
|
|
settingsConfig: {
|
|
env: {
|
|
GEMINI_API_KEY: "test-key",
|
|
GOOGLE_GEMINI_BASE_URL: "https://generativelanguage.googleapis.com",
|
|
},
|
|
},
|
|
category: "official",
|
|
sortIndex: 0,
|
|
createdAt: Date.now(),
|
|
},
|
|
},
|
|
opencode: {},
|
|
openclaw: {},
|
|
});
|
|
|
|
const createDefaultCurrent = (): CurrentProviderState => ({
|
|
claude: "claude-1",
|
|
codex: "codex-1",
|
|
gemini: "gemini-1",
|
|
opencode: "",
|
|
openclaw: "",
|
|
});
|
|
|
|
let providers = createDefaultProviders();
|
|
let current = createDefaultCurrent();
|
|
let settingsState: Settings = {
|
|
showInTray: true,
|
|
minimizeToTrayOnClose: true,
|
|
enableClaudePluginIntegration: false,
|
|
claudeConfigDir: "/default/claude",
|
|
codexConfigDir: "/default/codex",
|
|
language: "zh",
|
|
};
|
|
let appConfigDirOverride: string | null = null;
|
|
const sessionMessageKey = (providerId: string, sourcePath: string) =>
|
|
`${providerId}:${sourcePath}`;
|
|
|
|
const createDefaultSessions = (): SessionMeta[] => {
|
|
const now = Date.now();
|
|
return [
|
|
{
|
|
providerId: "codex",
|
|
sessionId: "codex-session-1",
|
|
title: "Codex Session One",
|
|
summary: "Codex summary",
|
|
projectDir: "/mock/codex",
|
|
createdAt: now - 2000,
|
|
lastActiveAt: now - 1000,
|
|
sourcePath: "/mock/codex/session-1.jsonl",
|
|
resumeCommand: "codex resume codex-session-1",
|
|
},
|
|
{
|
|
providerId: "claude",
|
|
sessionId: "claude-session-1",
|
|
title: "Claude Session One",
|
|
summary: "Claude summary",
|
|
projectDir: "/mock/claude",
|
|
createdAt: now - 4000,
|
|
lastActiveAt: now - 3000,
|
|
sourcePath: "/mock/claude/session-1.jsonl",
|
|
resumeCommand: "claude --resume claude-session-1",
|
|
},
|
|
];
|
|
};
|
|
|
|
const createDefaultSessionMessages = (): Record<string, SessionMessage[]> => ({
|
|
[sessionMessageKey("codex", "/mock/codex/session-1.jsonl")]: [
|
|
{
|
|
role: "user",
|
|
content: "First codex message",
|
|
ts: Date.now() - 1000,
|
|
},
|
|
],
|
|
[sessionMessageKey("claude", "/mock/claude/session-1.jsonl")]: [
|
|
{
|
|
role: "user",
|
|
content: "First claude message",
|
|
ts: Date.now() - 3000,
|
|
},
|
|
],
|
|
});
|
|
|
|
let sessionsState = createDefaultSessions();
|
|
let sessionMessagesState = createDefaultSessionMessages();
|
|
let mcpConfigs: McpConfigState = {
|
|
claude: {
|
|
sample: {
|
|
id: "sample",
|
|
name: "Sample Claude Server",
|
|
enabled: true,
|
|
apps: {
|
|
claude: true,
|
|
codex: false,
|
|
gemini: false,
|
|
opencode: false,
|
|
openclaw: false,
|
|
},
|
|
server: {
|
|
type: "stdio",
|
|
command: "claude-server",
|
|
},
|
|
},
|
|
},
|
|
codex: {
|
|
httpServer: {
|
|
id: "httpServer",
|
|
name: "HTTP Codex Server",
|
|
enabled: false,
|
|
apps: {
|
|
claude: false,
|
|
codex: true,
|
|
gemini: false,
|
|
opencode: false,
|
|
openclaw: false,
|
|
},
|
|
server: {
|
|
type: "http",
|
|
url: "http://localhost:3000",
|
|
},
|
|
},
|
|
},
|
|
gemini: {},
|
|
opencode: {},
|
|
openclaw: {},
|
|
};
|
|
|
|
const cloneProviders = (value: ProvidersByApp) =>
|
|
JSON.parse(JSON.stringify(value)) as ProvidersByApp;
|
|
|
|
export const resetProviderState = () => {
|
|
providers = createDefaultProviders();
|
|
current = createDefaultCurrent();
|
|
sessionsState = createDefaultSessions();
|
|
sessionMessagesState = createDefaultSessionMessages();
|
|
settingsState = {
|
|
showInTray: true,
|
|
minimizeToTrayOnClose: true,
|
|
enableClaudePluginIntegration: false,
|
|
claudeConfigDir: "/default/claude",
|
|
codexConfigDir: "/default/codex",
|
|
language: "zh",
|
|
};
|
|
appConfigDirOverride = null;
|
|
mcpConfigs = {
|
|
claude: {
|
|
sample: {
|
|
id: "sample",
|
|
name: "Sample Claude Server",
|
|
enabled: true,
|
|
apps: {
|
|
claude: true,
|
|
codex: false,
|
|
gemini: false,
|
|
opencode: false,
|
|
openclaw: false,
|
|
},
|
|
server: {
|
|
type: "stdio",
|
|
command: "claude-server",
|
|
},
|
|
},
|
|
},
|
|
codex: {
|
|
httpServer: {
|
|
id: "httpServer",
|
|
name: "HTTP Codex Server",
|
|
enabled: false,
|
|
apps: {
|
|
claude: false,
|
|
codex: true,
|
|
gemini: false,
|
|
opencode: false,
|
|
openclaw: false,
|
|
},
|
|
server: {
|
|
type: "http",
|
|
url: "http://localhost:3000",
|
|
},
|
|
},
|
|
},
|
|
gemini: {},
|
|
opencode: {},
|
|
openclaw: {},
|
|
};
|
|
};
|
|
|
|
export const getProviders = (appType: AppId) =>
|
|
cloneProviders(providers)[appType] ?? {};
|
|
|
|
export const getCurrentProviderId = (appType: AppId) => current[appType] ?? "";
|
|
|
|
export const setCurrentProviderId = (appType: AppId, providerId: string) => {
|
|
current[appType] = providerId;
|
|
};
|
|
|
|
export const updateProviders = (
|
|
appType: AppId,
|
|
data: Record<string, Provider>,
|
|
) => {
|
|
providers[appType] = cloneProviders({ [appType]: data } as ProvidersByApp)[
|
|
appType
|
|
];
|
|
};
|
|
|
|
export const setProviders = (
|
|
appType: AppId,
|
|
data: Record<string, Provider>,
|
|
) => {
|
|
providers[appType] = JSON.parse(JSON.stringify(data)) as Record<
|
|
string,
|
|
Provider
|
|
>;
|
|
};
|
|
|
|
export const addProvider = (appType: AppId, provider: Provider) => {
|
|
providers[appType] = providers[appType] ?? {};
|
|
providers[appType][provider.id] = provider;
|
|
};
|
|
|
|
export const updateProvider = (appType: AppId, provider: Provider) => {
|
|
if (!providers[appType]) return;
|
|
providers[appType][provider.id] = {
|
|
...providers[appType][provider.id],
|
|
...provider,
|
|
};
|
|
};
|
|
|
|
export const deleteProvider = (appType: AppId, providerId: string) => {
|
|
if (!providers[appType]) return;
|
|
delete providers[appType][providerId];
|
|
if (current[appType] === providerId) {
|
|
const fallback = Object.keys(providers[appType])[0] ?? "";
|
|
current[appType] = fallback;
|
|
}
|
|
};
|
|
|
|
export const updateSortOrder = (
|
|
appType: AppId,
|
|
updates: { id: string; sortIndex: number }[],
|
|
) => {
|
|
if (!providers[appType]) return;
|
|
updates.forEach(({ id, sortIndex }) => {
|
|
const provider = providers[appType][id];
|
|
if (provider) {
|
|
providers[appType][id] = { ...provider, sortIndex };
|
|
}
|
|
});
|
|
};
|
|
|
|
export const listProviders = (appType: AppId) =>
|
|
JSON.parse(JSON.stringify(providers[appType] ?? {})) as Record<
|
|
string,
|
|
Provider
|
|
>;
|
|
|
|
export const getSettings = () =>
|
|
JSON.parse(JSON.stringify(settingsState)) as Settings;
|
|
|
|
export const setSettings = (data: Partial<Settings>) => {
|
|
settingsState = { ...settingsState, ...data };
|
|
};
|
|
|
|
export const getAppConfigDirOverride = () => appConfigDirOverride;
|
|
|
|
export const setAppConfigDirOverrideState = (value: string | null) => {
|
|
appConfigDirOverride = value;
|
|
};
|
|
|
|
export const getMcpConfig = (appType: AppId) => {
|
|
const servers = JSON.parse(
|
|
JSON.stringify(mcpConfigs[appType] ?? {}),
|
|
) as Record<string, McpServer>;
|
|
return {
|
|
configPath: `/mock/${appType}.mcp.json`,
|
|
servers,
|
|
};
|
|
};
|
|
|
|
export const setMcpConfig = (
|
|
appType: AppId,
|
|
value: Record<string, McpServer>,
|
|
) => {
|
|
mcpConfigs[appType] = JSON.parse(JSON.stringify(value)) as Record<
|
|
string,
|
|
McpServer
|
|
>;
|
|
};
|
|
|
|
export const setMcpServerEnabled = (
|
|
appType: AppId,
|
|
id: string,
|
|
enabled: boolean,
|
|
) => {
|
|
if (!mcpConfigs[appType]?.[id]) return;
|
|
mcpConfigs[appType][id] = {
|
|
...mcpConfigs[appType][id],
|
|
enabled,
|
|
};
|
|
};
|
|
|
|
export const upsertMcpServer = (
|
|
appType: AppId,
|
|
id: string,
|
|
server: McpServer,
|
|
) => {
|
|
if (!mcpConfigs[appType]) {
|
|
mcpConfigs[appType] = {};
|
|
}
|
|
mcpConfigs[appType][id] = JSON.parse(JSON.stringify(server)) as McpServer;
|
|
};
|
|
|
|
export const deleteMcpServer = (appType: AppId, id: string) => {
|
|
if (!mcpConfigs[appType]) return;
|
|
delete mcpConfigs[appType][id];
|
|
};
|
|
|
|
export const listSessions = () =>
|
|
JSON.parse(JSON.stringify(sessionsState)) as SessionMeta[];
|
|
|
|
export const getSessionMessages = (providerId: string, sourcePath: string) =>
|
|
JSON.parse(
|
|
JSON.stringify(
|
|
sessionMessagesState[sessionMessageKey(providerId, sourcePath)] ?? [],
|
|
),
|
|
) as SessionMessage[];
|
|
|
|
export const deleteSession = (
|
|
providerId: string,
|
|
sessionId: string,
|
|
sourcePath: string,
|
|
) => {
|
|
sessionsState = sessionsState.filter(
|
|
(session) =>
|
|
!(
|
|
session.providerId === providerId &&
|
|
session.sessionId === sessionId &&
|
|
session.sourcePath === sourcePath
|
|
),
|
|
);
|
|
delete sessionMessagesState[sessionMessageKey(providerId, sourcePath)];
|
|
return true;
|
|
};
|
|
|
|
export const setSessionFixtures = (
|
|
sessions: SessionMeta[],
|
|
messages: Record<string, SessionMessage[]>,
|
|
) => {
|
|
sessionsState = JSON.parse(JSON.stringify(sessions)) as SessionMeta[];
|
|
sessionMessagesState = JSON.parse(JSON.stringify(messages)) as Record<
|
|
string,
|
|
SessionMessage[]
|
|
>;
|
|
};
|