Files
cc-switch/tests/msw/state.ts
Jason 8c3f18a9bd feat: add session deletion with per-provider cleanup and path safety
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.
2026-03-08 19:42:18 +08:00

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[]
>;
};