mirror of
https://github.com/siteboon/claudecodeui.git
synced 2026-05-17 09:30:05 +00:00
Compare commits
27 Commits
v1.31.5
...
fix/websoc
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ca8345330a | ||
|
|
10528a2bdd | ||
|
|
b2e3a61030 | ||
|
|
17db71c43c | ||
|
|
116f91bc3a | ||
|
|
ded630863f | ||
|
|
54130e8d14 | ||
|
|
de25f6d78e | ||
|
|
e57ce4248e | ||
|
|
5554e4e85e | ||
|
|
13d1d436f8 | ||
|
|
8273dc51c5 | ||
|
|
684e127213 | ||
|
|
29f6d94fbf | ||
|
|
c9413815a4 | ||
|
|
5bd179a36e | ||
|
|
ce36fb85e7 | ||
|
|
f289ce8419 | ||
|
|
6f3e48f4fb | ||
|
|
41e221f1b3 | ||
|
|
59fbddecaa | ||
|
|
1325f18173 | ||
|
|
c064aff568 | ||
|
|
b3fe1b4392 | ||
|
|
beb0a50413 | ||
|
|
e89d2da5df | ||
|
|
392c73b693 |
@@ -157,7 +157,7 @@ export default tseslint.config(
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: "backend-shared-utils", // shared backend runtime helpers that modules may import directly
|
type: "backend-shared-utils", // shared backend runtime helpers that modules may import directly
|
||||||
pattern: ["server/shared/utils.{js,ts}"], // classify the shared utils file so modules can depend on it explicitly
|
pattern: ["server/shared/utils.{js,ts}", "server/shared/claude-cli-path.ts"], // classify the shared utils file so modules can depend on it explicitly
|
||||||
mode: "file",
|
mode: "file",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ import { promises as fs } from 'fs';
|
|||||||
import path from 'path';
|
import path from 'path';
|
||||||
import os from 'os';
|
import os from 'os';
|
||||||
import { CLAUDE_MODELS } from '../shared/modelConstants.js';
|
import { CLAUDE_MODELS } from '../shared/modelConstants.js';
|
||||||
|
import { resolveClaudeCodeExecutablePath } from './shared/claude-cli-path.js';
|
||||||
import {
|
import {
|
||||||
createNotificationEvent,
|
createNotificationEvent,
|
||||||
notifyRunFailed,
|
notifyRunFailed,
|
||||||
@@ -153,11 +154,9 @@ function mapCliOptionsToSDK(options = {}) {
|
|||||||
// Since SDK 0.2.113, options.env replaces process.env instead of overlaying it.
|
// Since SDK 0.2.113, options.env replaces process.env instead of overlaying it.
|
||||||
sdkOptions.env = { ...process.env };
|
sdkOptions.env = { ...process.env };
|
||||||
|
|
||||||
// Use CLAUDE_CLI_PATH if explicitly set, otherwise fall back to 'claude' on PATH.
|
// Resolve the executable eagerly on Windows because the SDK uses raw child_process.spawn,
|
||||||
// The SDK 0.2.113+ looks for a bundled native binary optional dep by default;
|
// which does not reliably follow npm's shell wrappers like cross-spawn does.
|
||||||
// this fallback ensures users who installed via the official installer still work
|
sdkOptions.pathToClaudeCodeExecutable = resolveClaudeCodeExecutablePath(process.env.CLAUDE_CLI_PATH);
|
||||||
// even when npm prune --production has removed those optional deps.
|
|
||||||
sdkOptions.pathToClaudeCodeExecutable = process.env.CLAUDE_CLI_PATH || 'claude';
|
|
||||||
|
|
||||||
// Map working directory
|
// Map working directory
|
||||||
if (cwd) {
|
if (cwd) {
|
||||||
@@ -527,6 +526,12 @@ async function queryClaudeSDK(command, options = {}, ws) {
|
|||||||
}]
|
}]
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Caveat: in 'auto' and 'bypassPermissions' modes the SDK resolves approval
|
||||||
|
// at the permission-mode step and skips this callback, so interactive tools
|
||||||
|
// (AskUserQuestion, ExitPlanMode) won't reach the UI — the classifier/bypass
|
||||||
|
// auto-approves them and the model acts on a generated answer. Move these
|
||||||
|
// tools to a PreToolUse hook (runs before the mode check) if we need them
|
||||||
|
// to work in those modes.
|
||||||
sdkOptions.canUseTool = async (toolName, input, context) => {
|
sdkOptions.canUseTool = async (toolName, input, context) => {
|
||||||
const requiresInteraction = TOOLS_REQUIRING_INTERACTION.has(toolName);
|
const requiresInteraction = TOOLS_REQUIRING_INTERACTION.has(toolName);
|
||||||
|
|
||||||
|
|||||||
@@ -150,7 +150,6 @@ async function spawnCursor(command, options = {}, ws) {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const response = JSON.parse(line);
|
const response = JSON.parse(line);
|
||||||
console.log('Parsed JSON response:', response);
|
|
||||||
|
|
||||||
// Handle different message types
|
// Handle different message types
|
||||||
switch (response.type) {
|
switch (response.type) {
|
||||||
@@ -159,7 +158,6 @@ async function spawnCursor(command, options = {}, ws) {
|
|||||||
// Capture session ID
|
// Capture session ID
|
||||||
if (response.session_id && !capturedSessionId) {
|
if (response.session_id && !capturedSessionId) {
|
||||||
capturedSessionId = response.session_id;
|
capturedSessionId = response.session_id;
|
||||||
console.log('Captured session ID:', capturedSessionId);
|
|
||||||
|
|
||||||
// Update process key with captured session ID
|
// Update process key with captured session ID
|
||||||
if (processKey !== capturedSessionId) {
|
if (processKey !== capturedSessionId) {
|
||||||
@@ -197,7 +195,6 @@ async function spawnCursor(command, options = {}, ws) {
|
|||||||
|
|
||||||
case 'result': {
|
case 'result': {
|
||||||
// Session complete — send stream end + lifecycle complete with result payload
|
// Session complete — send stream end + lifecycle complete with result payload
|
||||||
console.log('Cursor session result:', response);
|
|
||||||
const resultText = typeof response.result === 'string' ? response.result : '';
|
const resultText = typeof response.result === 'string' ? response.result : '';
|
||||||
ws.send(createNormalizedMessage({
|
ws.send(createNormalizedMessage({
|
||||||
kind: 'complete',
|
kind: 'complete',
|
||||||
@@ -213,8 +210,6 @@ async function spawnCursor(command, options = {}, ws) {
|
|||||||
// Unknown message types — ignore.
|
// Unknown message types — ignore.
|
||||||
}
|
}
|
||||||
} catch (parseError) {
|
} catch (parseError) {
|
||||||
console.log('Non-JSON response:', line);
|
|
||||||
|
|
||||||
if (shouldSuppressForTrustRetry(line)) {
|
if (shouldSuppressForTrustRetry(line)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -228,7 +223,6 @@ async function spawnCursor(command, options = {}, ws) {
|
|||||||
// Handle stdout (streaming JSON responses)
|
// Handle stdout (streaming JSON responses)
|
||||||
cursorProcess.stdout.on('data', (data) => {
|
cursorProcess.stdout.on('data', (data) => {
|
||||||
const rawOutput = data.toString();
|
const rawOutput = data.toString();
|
||||||
console.log('Cursor CLI stdout:', rawOutput);
|
|
||||||
|
|
||||||
// Stream chunks can split JSON objects across packets; keep trailing partial line.
|
// Stream chunks can split JSON objects across packets; keep trailing partial line.
|
||||||
stdoutLineBuffer += rawOutput;
|
stdoutLineBuffer += rawOutput;
|
||||||
@@ -254,8 +248,6 @@ async function spawnCursor(command, options = {}, ws) {
|
|||||||
|
|
||||||
// Handle process completion
|
// Handle process completion
|
||||||
cursorProcess.on('close', async (code) => {
|
cursorProcess.on('close', async (code) => {
|
||||||
console.log(`Cursor CLI process exited with code ${code}`);
|
|
||||||
|
|
||||||
const finalSessionId = capturedSessionId || sessionId || processKey;
|
const finalSessionId = capturedSessionId || sessionId || processKey;
|
||||||
activeCursorProcesses.delete(finalSessionId);
|
activeCursorProcesses.delete(finalSessionId);
|
||||||
|
|
||||||
|
|||||||
@@ -1,19 +1,123 @@
|
|||||||
import { spawn } from 'child_process';
|
import { spawn } from 'child_process';
|
||||||
|
import { promises as fs } from 'fs';
|
||||||
|
import os from 'os';
|
||||||
|
import path from 'path';
|
||||||
|
|
||||||
import crossSpawn from 'cross-spawn';
|
import crossSpawn from 'cross-spawn';
|
||||||
|
|
||||||
// Use cross-spawn on Windows for correct .cmd resolution (same pattern as cursor-cli.js)
|
|
||||||
const spawnFunction = process.platform === 'win32' ? crossSpawn : spawn;
|
|
||||||
import { promises as fs } from 'fs';
|
|
||||||
import path from 'path';
|
|
||||||
import os from 'os';
|
|
||||||
import sessionManager from './sessionManager.js';
|
import sessionManager from './sessionManager.js';
|
||||||
import GeminiResponseHandler from './gemini-response-handler.js';
|
import GeminiResponseHandler from './gemini-response-handler.js';
|
||||||
import { notifyRunFailed, notifyRunStopped } from './services/notification-orchestrator.js';
|
import { notifyRunFailed, notifyRunStopped } from './services/notification-orchestrator.js';
|
||||||
import { providerAuthService } from './modules/providers/services/provider-auth.service.js';
|
import { providerAuthService } from './modules/providers/services/provider-auth.service.js';
|
||||||
import { createNormalizedMessage } from './shared/utils.js';
|
import { createNormalizedMessage } from './shared/utils.js';
|
||||||
|
|
||||||
|
// Use cross-spawn on Windows for correct .cmd resolution (same pattern as cursor-cli.js)
|
||||||
|
const spawnFunction = process.platform === 'win32' ? crossSpawn : spawn;
|
||||||
|
|
||||||
let activeGeminiProcesses = new Map(); // Track active processes by session ID
|
let activeGeminiProcesses = new Map(); // Track active processes by session ID
|
||||||
|
|
||||||
|
function mapGeminiExitCodeToMessage(exitCode) {
|
||||||
|
switch (exitCode) {
|
||||||
|
case 42:
|
||||||
|
return 'Gemini rejected the request input (exit code 42).';
|
||||||
|
case 44:
|
||||||
|
return 'Gemini sandbox error (exit code 44). Check local sandbox/container settings.';
|
||||||
|
case 52:
|
||||||
|
return 'Gemini configuration error (exit code 52). Check your Gemini settings files for invalid JSON/config.';
|
||||||
|
case 53:
|
||||||
|
return 'Gemini conversation turn limit reached (exit code 53). Start a new Gemini session.';
|
||||||
|
default:
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const GEMINI_AUTH_ENV_KEYS = [
|
||||||
|
'GEMINI_API_KEY',
|
||||||
|
'GOOGLE_API_KEY',
|
||||||
|
'GOOGLE_CLOUD_PROJECT',
|
||||||
|
'GOOGLE_CLOUD_PROJECT_ID',
|
||||||
|
'GOOGLE_CLOUD_LOCATION',
|
||||||
|
'GOOGLE_APPLICATION_CREDENTIALS'
|
||||||
|
];
|
||||||
|
|
||||||
|
function parseEnvFileContent(content) {
|
||||||
|
const parsed = {};
|
||||||
|
|
||||||
|
for (const rawLine of content.split(/\r?\n/)) {
|
||||||
|
const line = rawLine.trim();
|
||||||
|
if (!line || line.startsWith('#')) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const exportPrefix = 'export ';
|
||||||
|
const normalizedLine = line.startsWith(exportPrefix) ? line.slice(exportPrefix.length).trim() : line;
|
||||||
|
const separatorIndex = normalizedLine.indexOf('=');
|
||||||
|
|
||||||
|
if (separatorIndex <= 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const key = normalizedLine.slice(0, separatorIndex).trim();
|
||||||
|
if (!key) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let value = normalizedLine.slice(separatorIndex + 1).trim();
|
||||||
|
const hasDoubleQuotes = value.startsWith('"') && value.endsWith('"');
|
||||||
|
const hasSingleQuotes = value.startsWith('\'') && value.endsWith('\'');
|
||||||
|
|
||||||
|
if (hasDoubleQuotes || hasSingleQuotes) {
|
||||||
|
value = value.slice(1, -1);
|
||||||
|
} else {
|
||||||
|
// Support inline comments in unquoted values: KEY=value # comment
|
||||||
|
value = value.replace(/\s+#.*$/, '').trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
parsed[key] = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parsed;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadGeminiUserLevelEnv() {
|
||||||
|
const geminiCliHome = (process.env.GEMINI_CLI_HOME || '').trim() || os.homedir();
|
||||||
|
const envCandidates = [
|
||||||
|
path.join(geminiCliHome, '.gemini', '.env'),
|
||||||
|
path.join(geminiCliHome, '.env')
|
||||||
|
];
|
||||||
|
|
||||||
|
for (const envPath of envCandidates) {
|
||||||
|
try {
|
||||||
|
await fs.access(envPath);
|
||||||
|
const content = await fs.readFile(envPath, 'utf8');
|
||||||
|
return parseEnvFileContent(content);
|
||||||
|
} catch {
|
||||||
|
// Keep scanning for the next candidate.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
async function buildGeminiProcessEnv() {
|
||||||
|
const processEnv = { ...process.env };
|
||||||
|
if (processEnv.GEMINI_API_KEY || processEnv.GOOGLE_API_KEY || processEnv.GOOGLE_APPLICATION_CREDENTIALS) {
|
||||||
|
return processEnv;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gemini CLI docs recommend ~/.gemini/.env for persistent headless auth settings.
|
||||||
|
// When the server process was launched without shell profile variables, we still
|
||||||
|
// want the spawned CLI process to inherit those user-level credentials.
|
||||||
|
const userEnv = await loadGeminiUserLevelEnv();
|
||||||
|
for (const key of GEMINI_AUTH_ENV_KEYS) {
|
||||||
|
if (!processEnv[key] && userEnv[key]) {
|
||||||
|
processEnv[key] = userEnv[key];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return processEnv;
|
||||||
|
}
|
||||||
|
|
||||||
async function spawnGemini(command, options = {}, ws) {
|
async function spawnGemini(command, options = {}, ws) {
|
||||||
const { sessionId, projectPath, cwd, toolsSettings, permissionMode, images, sessionSummary } = options;
|
const { sessionId, projectPath, cwd, toolsSettings, permissionMode, images, sessionSummary } = options;
|
||||||
let capturedSessionId = sessionId; // Track session ID throughout the process
|
let capturedSessionId = sessionId; // Track session ID throughout the process
|
||||||
@@ -100,6 +204,11 @@ async function spawnGemini(command, options = {}, ws) {
|
|||||||
args.push('--debug');
|
args.push('--debug');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// This integration runs Gemini in headless mode and cannot answer trust prompts.
|
||||||
|
// Skip folder-trust interactivity so authenticated runs don't fail with
|
||||||
|
// FatalUntrustedWorkspaceError in previously unseen directories.
|
||||||
|
args.push('--skip-trust');
|
||||||
|
|
||||||
// Add MCP config flag only if MCP servers are configured
|
// Add MCP config flag only if MCP servers are configured
|
||||||
try {
|
try {
|
||||||
const geminiConfigPath = path.join(os.homedir(), '.gemini.json');
|
const geminiConfigPath = path.join(os.homedir(), '.gemini.json');
|
||||||
@@ -154,9 +263,6 @@ async function spawnGemini(command, options = {}, ws) {
|
|||||||
|
|
||||||
// Try to find gemini in PATH first, then fall back to environment variable
|
// Try to find gemini in PATH first, then fall back to environment variable
|
||||||
const geminiPath = process.env.GEMINI_PATH || 'gemini';
|
const geminiPath = process.env.GEMINI_PATH || 'gemini';
|
||||||
console.log('Spawning Gemini CLI:', geminiPath, args.join(' '));
|
|
||||||
console.log('Working directory:', workingDir);
|
|
||||||
|
|
||||||
let spawnCmd = geminiPath;
|
let spawnCmd = geminiPath;
|
||||||
let spawnArgs = args;
|
let spawnArgs = args;
|
||||||
|
|
||||||
@@ -168,11 +274,13 @@ async function spawnGemini(command, options = {}, ws) {
|
|||||||
spawnArgs = ['-c', 'exec "$0" "$@"', geminiPath, ...args];
|
spawnArgs = ['-c', 'exec "$0" "$@"', geminiPath, ...args];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const spawnEnv = await buildGeminiProcessEnv();
|
||||||
|
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
const geminiProcess = spawnFunction(spawnCmd, spawnArgs, {
|
const geminiProcess = spawnFunction(spawnCmd, spawnArgs, {
|
||||||
cwd: workingDir,
|
cwd: workingDir,
|
||||||
stdio: ['pipe', 'pipe', 'pipe'],
|
stdio: ['pipe', 'pipe', 'pipe'],
|
||||||
env: { ...process.env } // Inherit all environment variables
|
env: spawnEnv
|
||||||
});
|
});
|
||||||
let terminalNotificationSent = false;
|
let terminalNotificationSent = false;
|
||||||
let terminalFailureReason = null;
|
let terminalFailureReason = null;
|
||||||
@@ -276,12 +384,43 @@ async function spawnGemini(command, options = {}, ws) {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
onInit: (event) => {
|
onInit: (event) => {
|
||||||
if (capturedSessionId) {
|
const discoveredSessionId = event?.session_id;
|
||||||
const sess = sessionManager.getSession(capturedSessionId);
|
if (!discoveredSessionId) {
|
||||||
if (sess && !sess.cliSessionId) {
|
return;
|
||||||
sess.cliSessionId = event.session_id;
|
}
|
||||||
sessionManager.saveSession(capturedSessionId);
|
|
||||||
|
// New Gemini sessions announce their canonical ID asynchronously via the
|
||||||
|
// initial `init` stream event. Avoid synthetic IDs and only register
|
||||||
|
// the session once that real ID is known (same model used by Claude/Codex).
|
||||||
|
if (!capturedSessionId) {
|
||||||
|
capturedSessionId = discoveredSessionId;
|
||||||
|
|
||||||
|
sessionManager.createSession(capturedSessionId, cwd || process.cwd());
|
||||||
|
if (command) {
|
||||||
|
sessionManager.addMessage(capturedSessionId, 'user', command);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (processKey !== capturedSessionId) {
|
||||||
|
activeGeminiProcesses.delete(processKey);
|
||||||
|
activeGeminiProcesses.set(capturedSessionId, geminiProcess);
|
||||||
|
}
|
||||||
|
|
||||||
|
geminiProcess.sessionId = capturedSessionId;
|
||||||
|
|
||||||
|
if (ws.setSessionId && typeof ws.setSessionId === 'function') {
|
||||||
|
ws.setSessionId(capturedSessionId);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!sessionId && !sessionCreatedSent) {
|
||||||
|
sessionCreatedSent = true;
|
||||||
|
ws.send(createNormalizedMessage({ kind: 'session_created', newSessionId: capturedSessionId, sessionId: capturedSessionId, provider: 'gemini' }));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const sess = sessionManager.getSession(capturedSessionId);
|
||||||
|
if (sess && !sess.cliSessionId) {
|
||||||
|
sess.cliSessionId = discoveredSessionId;
|
||||||
|
sessionManager.saveSession(capturedSessionId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -292,30 +431,6 @@ async function spawnGemini(command, options = {}, ws) {
|
|||||||
const rawOutput = data.toString();
|
const rawOutput = data.toString();
|
||||||
startTimeout(); // Re-arm the timeout
|
startTimeout(); // Re-arm the timeout
|
||||||
|
|
||||||
// For new sessions, create a session ID FIRST
|
|
||||||
if (!sessionId && !sessionCreatedSent && !capturedSessionId) {
|
|
||||||
capturedSessionId = `gemini_${Date.now()}`;
|
|
||||||
sessionCreatedSent = true;
|
|
||||||
|
|
||||||
// Create session in session manager
|
|
||||||
sessionManager.createSession(capturedSessionId, cwd || process.cwd());
|
|
||||||
|
|
||||||
// Save the user message now that we have a session ID
|
|
||||||
if (command) {
|
|
||||||
sessionManager.addMessage(capturedSessionId, 'user', command);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update process key with captured session ID
|
|
||||||
if (processKey !== capturedSessionId) {
|
|
||||||
activeGeminiProcesses.delete(processKey);
|
|
||||||
activeGeminiProcesses.set(capturedSessionId, geminiProcess);
|
|
||||||
}
|
|
||||||
|
|
||||||
ws.setSessionId && typeof ws.setSessionId === 'function' && ws.setSessionId(capturedSessionId);
|
|
||||||
|
|
||||||
ws.send(createNormalizedMessage({ kind: 'session_created', newSessionId: capturedSessionId, sessionId: capturedSessionId, provider: 'gemini' }));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (responseHandler) {
|
if (responseHandler) {
|
||||||
responseHandler.processData(rawOutput);
|
responseHandler.processData(rawOutput);
|
||||||
} else if (rawOutput) {
|
} else if (rawOutput) {
|
||||||
@@ -381,12 +496,38 @@ async function spawnGemini(command, options = {}, ws) {
|
|||||||
notifyTerminalState({ code });
|
notifyTerminalState({ code });
|
||||||
resolve();
|
resolve();
|
||||||
} else {
|
} else {
|
||||||
// code 127 = shell "command not found" — check installation
|
const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
|
||||||
|
|
||||||
|
// code 127 = shell "command not found" - check installation
|
||||||
if (code === 127) {
|
if (code === 127) {
|
||||||
const installed = await providerAuthService.isProviderInstalled('gemini');
|
const installed = await providerAuthService.isProviderInstalled('gemini');
|
||||||
if (!installed) {
|
if (!installed) {
|
||||||
const socketSessionId = typeof ws.getSessionId === 'function' ? ws.getSessionId() : finalSessionId;
|
terminalFailureReason = 'Gemini CLI is not installed. Please install it first: https://github.com/google-gemini/gemini-cli';
|
||||||
ws.send(createNormalizedMessage({ kind: 'error', content: 'Gemini CLI is not installed. Please install it first: https://github.com/google-gemini/gemini-cli', sessionId: socketSessionId, provider: 'gemini' }));
|
ws.send(createNormalizedMessage({ kind: 'error', content: terminalFailureReason, sessionId: socketSessionId, provider: 'gemini' }));
|
||||||
|
}
|
||||||
|
} else if (code === 41) {
|
||||||
|
// Gemini CLI documents exit code 41 as FatalAuthenticationError.
|
||||||
|
// Surface an actionable auth error instead of a generic exit-code message.
|
||||||
|
let authErrorSuffix = '';
|
||||||
|
try {
|
||||||
|
const authStatus = await providerAuthService.getProviderAuthStatus('gemini');
|
||||||
|
if (!authStatus?.authenticated && authStatus?.error) {
|
||||||
|
authErrorSuffix = ` Details: ${authStatus.error}`;
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
// Keep base remediation text when auth status lookup fails.
|
||||||
|
}
|
||||||
|
|
||||||
|
terminalFailureReason =
|
||||||
|
'Gemini authentication failed (exit code 41). '
|
||||||
|
+ 'Run `gemini` in a terminal to choose an auth method, or configure a valid `GEMINI_API_KEY`.'
|
||||||
|
+ authErrorSuffix;
|
||||||
|
ws.send(createNormalizedMessage({ kind: 'error', content: terminalFailureReason, sessionId: socketSessionId, provider: 'gemini' }));
|
||||||
|
} else {
|
||||||
|
const mappedError = mapGeminiExitCodeToMessage(code);
|
||||||
|
if (mappedError) {
|
||||||
|
terminalFailureReason = mappedError;
|
||||||
|
ws.send(createNormalizedMessage({ kind: 'error', content: terminalFailureReason, sessionId: socketSessionId, provider: 'gemini' }));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -394,7 +535,14 @@ async function spawnGemini(command, options = {}, ws) {
|
|||||||
code,
|
code,
|
||||||
error: code === null ? 'Gemini CLI process was terminated or timed out' : null
|
error: code === null ? 'Gemini CLI process was terminated or timed out' : null
|
||||||
});
|
});
|
||||||
reject(new Error(code === null ? 'Gemini CLI process was terminated or timed out' : `Gemini CLI exited with code ${code}`));
|
reject(
|
||||||
|
new Error(
|
||||||
|
terminalFailureReason
|
||||||
|
|| (code === null
|
||||||
|
? 'Gemini CLI process was terminated or timed out'
|
||||||
|
: `Gemini CLI exited with code ${code}`)
|
||||||
|
)
|
||||||
|
);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -257,8 +257,10 @@ const rebuildSessionsTableWithProjectSchema = (db: Database): void => {
|
|||||||
|
|
||||||
if (!shouldRebuild) {
|
if (!shouldRebuild) {
|
||||||
addColumnToTableIfNotExists(db, 'sessions', columnNames, 'jsonl_path', 'TEXT');
|
addColumnToTableIfNotExists(db, 'sessions', columnNames, 'jsonl_path', 'TEXT');
|
||||||
|
addColumnToTableIfNotExists(db, 'sessions', columnNames, 'isArchived', 'BOOLEAN DEFAULT 0');
|
||||||
addColumnToTableIfNotExists(db, 'sessions', columnNames, 'created_at', 'DATETIME');
|
addColumnToTableIfNotExists(db, 'sessions', columnNames, 'created_at', 'DATETIME');
|
||||||
addColumnToTableIfNotExists(db, 'sessions', columnNames, 'updated_at', 'DATETIME');
|
addColumnToTableIfNotExists(db, 'sessions', columnNames, 'updated_at', 'DATETIME');
|
||||||
|
db.exec('UPDATE sessions SET isArchived = COALESCE(isArchived, 0)');
|
||||||
db.exec('UPDATE sessions SET created_at = COALESCE(created_at, CURRENT_TIMESTAMP)');
|
db.exec('UPDATE sessions SET created_at = COALESCE(created_at, CURRENT_TIMESTAMP)');
|
||||||
db.exec('UPDATE sessions SET updated_at = COALESCE(updated_at, CURRENT_TIMESTAMP)');
|
db.exec('UPDATE sessions SET updated_at = COALESCE(updated_at, CURRENT_TIMESTAMP)');
|
||||||
return;
|
return;
|
||||||
@@ -284,6 +286,10 @@ const rebuildSessionsTableWithProjectSchema = (db: Database): void => {
|
|||||||
? 'jsonl_path'
|
? 'jsonl_path'
|
||||||
: 'NULL';
|
: 'NULL';
|
||||||
|
|
||||||
|
const isArchivedExpression = columnNames.includes('isArchived')
|
||||||
|
? 'COALESCE(isArchived, 0)'
|
||||||
|
: '0';
|
||||||
|
|
||||||
const createdAtExpression = columnNames.includes('created_at')
|
const createdAtExpression = columnNames.includes('created_at')
|
||||||
? 'COALESCE(created_at, CURRENT_TIMESTAMP)'
|
? 'COALESCE(created_at, CURRENT_TIMESTAMP)'
|
||||||
: 'CURRENT_TIMESTAMP';
|
: 'CURRENT_TIMESTAMP';
|
||||||
@@ -303,6 +309,7 @@ const rebuildSessionsTableWithProjectSchema = (db: Database): void => {
|
|||||||
custom_name TEXT,
|
custom_name TEXT,
|
||||||
project_path TEXT,
|
project_path TEXT,
|
||||||
jsonl_path TEXT,
|
jsonl_path TEXT,
|
||||||
|
isArchived BOOLEAN DEFAULT 0,
|
||||||
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
||||||
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
||||||
PRIMARY KEY (session_id),
|
PRIMARY KEY (session_id),
|
||||||
@@ -319,6 +326,7 @@ const rebuildSessionsTableWithProjectSchema = (db: Database): void => {
|
|||||||
${customNameExpression} AS custom_name,
|
${customNameExpression} AS custom_name,
|
||||||
${projectPathExpression} AS project_path,
|
${projectPathExpression} AS project_path,
|
||||||
${jsonlPathExpression} AS jsonl_path,
|
${jsonlPathExpression} AS jsonl_path,
|
||||||
|
${isArchivedExpression} AS isArchived,
|
||||||
${createdAtExpression} AS created_at,
|
${createdAtExpression} AS created_at,
|
||||||
${updatedAtExpression} AS updated_at,
|
${updatedAtExpression} AS updated_at,
|
||||||
rowid AS source_rowid
|
rowid AS source_rowid
|
||||||
@@ -332,6 +340,7 @@ const rebuildSessionsTableWithProjectSchema = (db: Database): void => {
|
|||||||
custom_name,
|
custom_name,
|
||||||
project_path,
|
project_path,
|
||||||
jsonl_path,
|
jsonl_path,
|
||||||
|
isArchived,
|
||||||
created_at,
|
created_at,
|
||||||
updated_at,
|
updated_at,
|
||||||
ROW_NUMBER() OVER (
|
ROW_NUMBER() OVER (
|
||||||
@@ -346,6 +355,7 @@ const rebuildSessionsTableWithProjectSchema = (db: Database): void => {
|
|||||||
custom_name,
|
custom_name,
|
||||||
project_path,
|
project_path,
|
||||||
jsonl_path,
|
jsonl_path,
|
||||||
|
isArchived,
|
||||||
created_at,
|
created_at,
|
||||||
updated_at
|
updated_at
|
||||||
)
|
)
|
||||||
@@ -355,6 +365,7 @@ const rebuildSessionsTableWithProjectSchema = (db: Database): void => {
|
|||||||
custom_name,
|
custom_name,
|
||||||
project_path,
|
project_path,
|
||||||
jsonl_path,
|
jsonl_path,
|
||||||
|
isArchived,
|
||||||
created_at,
|
created_at,
|
||||||
updated_at
|
updated_at
|
||||||
FROM ranked_rows
|
FROM ranked_rows
|
||||||
@@ -421,6 +432,7 @@ export const runMigrations = (db: Database) => {
|
|||||||
|
|
||||||
db.exec('CREATE INDEX IF NOT EXISTS idx_session_ids_lookup ON sessions(session_id)');
|
db.exec('CREATE INDEX IF NOT EXISTS idx_session_ids_lookup ON sessions(session_id)');
|
||||||
db.exec('CREATE INDEX IF NOT EXISTS idx_sessions_project_path ON sessions(project_path)');
|
db.exec('CREATE INDEX IF NOT EXISTS idx_sessions_project_path ON sessions(project_path)');
|
||||||
|
db.exec('CREATE INDEX IF NOT EXISTS idx_sessions_is_archived ON sessions(isArchived)');
|
||||||
db.exec('CREATE INDEX IF NOT EXISTS idx_projects_is_starred ON projects(isStarred)');
|
db.exec('CREATE INDEX IF NOT EXISTS idx_projects_is_starred ON projects(isStarred)');
|
||||||
db.exec('CREATE INDEX IF NOT EXISTS idx_projects_is_archived ON projects(isArchived)');
|
db.exec('CREATE INDEX IF NOT EXISTS idx_projects_is_archived ON projects(isArchived)');
|
||||||
|
|
||||||
|
|||||||
@@ -95,6 +95,19 @@ export const projectsDb = {
|
|||||||
`).all() as ProjectRepositoryRow[];
|
`).all() as ProjectRepositoryRow[];
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Archived rows are queried separately so archive-focused UIs can present
|
||||||
|
* hidden workspaces without reintroducing them into the active sidebar list.
|
||||||
|
*/
|
||||||
|
getArchivedProjectPaths(): ProjectRepositoryRow[] {
|
||||||
|
const db = getConnection();
|
||||||
|
return db.prepare(`
|
||||||
|
SELECT project_id, project_path, custom_project_name, isStarred, isArchived
|
||||||
|
FROM projects
|
||||||
|
WHERE isArchived = 1
|
||||||
|
`).all() as ProjectRepositoryRow[];
|
||||||
|
},
|
||||||
|
|
||||||
getCustomProjectName(projectPath: string): string | null {
|
getCustomProjectName(projectPath: string): string | null {
|
||||||
const db = getConnection();
|
const db = getConnection();
|
||||||
const normalizedProjectPath = normalizeProjectPath(projectPath);
|
const normalizedProjectPath = normalizeProjectPath(projectPath);
|
||||||
|
|||||||
@@ -0,0 +1,72 @@
|
|||||||
|
import assert from 'node:assert/strict';
|
||||||
|
import { mkdtemp, rm } from 'node:fs/promises';
|
||||||
|
import { tmpdir } from 'node:os';
|
||||||
|
import path from 'node:path';
|
||||||
|
import test from 'node:test';
|
||||||
|
|
||||||
|
import { closeConnection } from '@/modules/database/connection.js';
|
||||||
|
import { initializeDatabase } from '@/modules/database/init-db.js';
|
||||||
|
import { sessionsDb } from '@/modules/database/repositories/sessions.db.js';
|
||||||
|
|
||||||
|
async function withIsolatedDatabase(runTest: () => void | Promise<void>): Promise<void> {
|
||||||
|
const previousDatabasePath = process.env.DATABASE_PATH;
|
||||||
|
const tempDirectory = await mkdtemp(path.join(tmpdir(), 'sessions-db-'));
|
||||||
|
const databasePath = path.join(tempDirectory, 'auth.db');
|
||||||
|
|
||||||
|
closeConnection();
|
||||||
|
process.env.DATABASE_PATH = databasePath;
|
||||||
|
await initializeDatabase();
|
||||||
|
|
||||||
|
try {
|
||||||
|
await runTest();
|
||||||
|
} finally {
|
||||||
|
closeConnection();
|
||||||
|
if (previousDatabasePath === undefined) {
|
||||||
|
delete process.env.DATABASE_PATH;
|
||||||
|
} else {
|
||||||
|
process.env.DATABASE_PATH = previousDatabasePath;
|
||||||
|
}
|
||||||
|
await rm(tempDirectory, { recursive: true, force: true });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
test('session archive queries hide archived rows from active project views', async () => {
|
||||||
|
await withIsolatedDatabase(() => {
|
||||||
|
sessionsDb.createSession('session-active', 'claude', '/workspace/demo-project', 'Active Session');
|
||||||
|
sessionsDb.createSession('session-archived', 'claude', '/workspace/demo-project', 'Archived Session');
|
||||||
|
sessionsDb.updateSessionIsArchived('session-archived', true);
|
||||||
|
|
||||||
|
const activeSessions = sessionsDb.getAllSessions();
|
||||||
|
const archivedSessions = sessionsDb.getArchivedSessions();
|
||||||
|
const activeProjectSessions = sessionsDb.getSessionsByProjectPath('/workspace/demo-project');
|
||||||
|
const allProjectSessions = sessionsDb.getSessionsByProjectPathIncludingArchived('/workspace/demo-project');
|
||||||
|
|
||||||
|
assert.deepEqual(activeSessions.map((session) => session.session_id), ['session-active']);
|
||||||
|
assert.deepEqual(archivedSessions.map((session) => session.session_id), ['session-archived']);
|
||||||
|
assert.deepEqual(activeProjectSessions.map((session) => session.session_id), ['session-active']);
|
||||||
|
assert.deepEqual(
|
||||||
|
allProjectSessions.map((session) => session.session_id).sort(),
|
||||||
|
['session-active', 'session-archived'],
|
||||||
|
);
|
||||||
|
assert.equal(sessionsDb.countSessionsByProjectPath('/workspace/demo-project'), 1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('createSession reactivates archived rows when the session becomes active again', async () => {
|
||||||
|
await withIsolatedDatabase(() => {
|
||||||
|
sessionsDb.createSession('session-reused', 'claude', '/workspace/demo-project', 'First Name');
|
||||||
|
sessionsDb.updateSessionIsArchived('session-reused', true);
|
||||||
|
|
||||||
|
sessionsDb.createSession('session-reused', 'claude', '/workspace/demo-project', 'Updated Name');
|
||||||
|
|
||||||
|
const activeSessions = sessionsDb.getAllSessions();
|
||||||
|
const archivedSessions = sessionsDb.getArchivedSessions();
|
||||||
|
const restoredSession = sessionsDb.getSessionById('session-reused');
|
||||||
|
|
||||||
|
assert.equal(activeSessions.length, 1);
|
||||||
|
assert.equal(activeSessions[0]?.session_id, 'session-reused');
|
||||||
|
assert.equal(activeSessions[0]?.custom_name, 'Updated Name');
|
||||||
|
assert.equal(archivedSessions.length, 0);
|
||||||
|
assert.equal(restoredSession?.isArchived, 0);
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -8,13 +8,14 @@ type SessionRow = {
|
|||||||
project_path: string | null;
|
project_path: string | null;
|
||||||
jsonl_path: string | null;
|
jsonl_path: string | null;
|
||||||
custom_name: string | null;
|
custom_name: string | null;
|
||||||
|
isArchived: number;
|
||||||
created_at: string;
|
created_at: string;
|
||||||
updated_at: string;
|
updated_at: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
type SessionMetadataLookupRow = Pick<
|
type SessionMetadataLookupRow = Pick<
|
||||||
SessionRow,
|
SessionRow,
|
||||||
'session_id' | 'provider' | 'project_path' | 'jsonl_path' | 'custom_name' | 'created_at' | 'updated_at'
|
'session_id' | 'provider' | 'project_path' | 'jsonl_path' | 'custom_name' | 'isArchived' | 'created_at' | 'updated_at'
|
||||||
>;
|
>;
|
||||||
|
|
||||||
function normalizeTimestamp(value?: string): string | null {
|
function normalizeTimestamp(value?: string): string | null {
|
||||||
@@ -53,13 +54,14 @@ export const sessionsDb = {
|
|||||||
projectsDb.createProjectPath(normalizedProjectPath);
|
projectsDb.createProjectPath(normalizedProjectPath);
|
||||||
|
|
||||||
db.prepare(
|
db.prepare(
|
||||||
`INSERT INTO sessions (session_id, provider, custom_name, project_path, jsonl_path, created_at, updated_at)
|
`INSERT INTO sessions (session_id, provider, custom_name, project_path, jsonl_path, isArchived, created_at, updated_at)
|
||||||
VALUES (?, ?, ?, ?, ?, COALESCE(?, CURRENT_TIMESTAMP), COALESCE(?, CURRENT_TIMESTAMP))
|
VALUES (?, ?, ?, ?, ?, 0, COALESCE(?, CURRENT_TIMESTAMP), COALESCE(?, CURRENT_TIMESTAMP))
|
||||||
ON CONFLICT(session_id) DO UPDATE SET
|
ON CONFLICT(session_id) DO UPDATE SET
|
||||||
provider = excluded.provider,
|
provider = excluded.provider,
|
||||||
updated_at = excluded.updated_at,
|
updated_at = excluded.updated_at,
|
||||||
project_path = excluded.project_path,
|
project_path = excluded.project_path,
|
||||||
jsonl_path = excluded.jsonl_path,
|
jsonl_path = excluded.jsonl_path,
|
||||||
|
isArchived = 0,
|
||||||
custom_name = COALESCE(excluded.custom_name, sessions.custom_name)`
|
custom_name = COALESCE(excluded.custom_name, sessions.custom_name)`
|
||||||
).run(
|
).run(
|
||||||
sessionId,
|
sessionId,
|
||||||
@@ -87,7 +89,7 @@ export const sessionsDb = {
|
|||||||
const db = getConnection();
|
const db = getConnection();
|
||||||
const row = db
|
const row = db
|
||||||
.prepare(
|
.prepare(
|
||||||
`SELECT session_id, provider, project_path, jsonl_path, custom_name, created_at, updated_at
|
`SELECT session_id, provider, project_path, jsonl_path, custom_name, isArchived, created_at, updated_at
|
||||||
FROM sessions
|
FROM sessions
|
||||||
WHERE session_id = ?
|
WHERE session_id = ?
|
||||||
ORDER BY updated_at DESC
|
ORDER BY updated_at DESC
|
||||||
@@ -102,8 +104,25 @@ export const sessionsDb = {
|
|||||||
const db = getConnection();
|
const db = getConnection();
|
||||||
return db
|
return db
|
||||||
.prepare(
|
.prepare(
|
||||||
`SELECT session_id, provider, project_path, jsonl_path, custom_name, created_at, updated_at
|
`SELECT session_id, provider, project_path, jsonl_path, custom_name, isArchived, created_at, updated_at
|
||||||
FROM sessions`
|
FROM sessions
|
||||||
|
WHERE isArchived = 0`
|
||||||
|
)
|
||||||
|
.all() as SessionRow[];
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Archived rows are intentionally queried separately so the caller can render
|
||||||
|
* them in a dedicated view without reintroducing them into active session lists.
|
||||||
|
*/
|
||||||
|
getArchivedSessions(): SessionRow[] {
|
||||||
|
const db = getConnection();
|
||||||
|
return db
|
||||||
|
.prepare(
|
||||||
|
`SELECT session_id, provider, project_path, jsonl_path, custom_name, isArchived, created_at, updated_at
|
||||||
|
FROM sessions
|
||||||
|
WHERE isArchived = 1
|
||||||
|
ORDER BY datetime(COALESCE(updated_at, created_at)) DESC, session_id DESC`
|
||||||
)
|
)
|
||||||
.all() as SessionRow[];
|
.all() as SessionRow[];
|
||||||
},
|
},
|
||||||
@@ -113,7 +132,24 @@ export const sessionsDb = {
|
|||||||
const normalizedProjectPath = normalizeProjectPath(projectPath);
|
const normalizedProjectPath = normalizeProjectPath(projectPath);
|
||||||
return db
|
return db
|
||||||
.prepare(
|
.prepare(
|
||||||
`SELECT session_id, provider, project_path, jsonl_path, custom_name, created_at, updated_at
|
`SELECT session_id, provider, project_path, jsonl_path, custom_name, isArchived, created_at, updated_at
|
||||||
|
FROM sessions
|
||||||
|
WHERE project_path = ?
|
||||||
|
AND isArchived = 0`
|
||||||
|
)
|
||||||
|
.all(normalizedProjectPath) as SessionRow[];
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Permanent project deletion must see every session row for the path,
|
||||||
|
* including archived ones, so their transcript files can be cleaned up.
|
||||||
|
*/
|
||||||
|
getSessionsByProjectPathIncludingArchived(projectPath: string): SessionRow[] {
|
||||||
|
const db = getConnection();
|
||||||
|
const normalizedProjectPath = normalizeProjectPath(projectPath);
|
||||||
|
return db
|
||||||
|
.prepare(
|
||||||
|
`SELECT session_id, provider, project_path, jsonl_path, custom_name, isArchived, created_at, updated_at
|
||||||
FROM sessions
|
FROM sessions
|
||||||
WHERE project_path = ?`
|
WHERE project_path = ?`
|
||||||
)
|
)
|
||||||
@@ -125,9 +161,10 @@ export const sessionsDb = {
|
|||||||
const normalizedProjectPath = normalizeProjectPath(projectPath);
|
const normalizedProjectPath = normalizeProjectPath(projectPath);
|
||||||
return db
|
return db
|
||||||
.prepare(
|
.prepare(
|
||||||
`SELECT session_id, provider, project_path, jsonl_path, custom_name, created_at, updated_at
|
`SELECT session_id, provider, project_path, jsonl_path, custom_name, isArchived, created_at, updated_at
|
||||||
FROM sessions
|
FROM sessions
|
||||||
WHERE project_path = ?
|
WHERE project_path = ?
|
||||||
|
AND isArchived = 0
|
||||||
ORDER BY datetime(COALESCE(updated_at, created_at)) DESC, session_id DESC
|
ORDER BY datetime(COALESCE(updated_at, created_at)) DESC, session_id DESC
|
||||||
LIMIT ? OFFSET ?`
|
LIMIT ? OFFSET ?`
|
||||||
)
|
)
|
||||||
@@ -141,7 +178,8 @@ export const sessionsDb = {
|
|||||||
.prepare(
|
.prepare(
|
||||||
`SELECT COUNT(*) AS count
|
`SELECT COUNT(*) AS count
|
||||||
FROM sessions
|
FROM sessions
|
||||||
WHERE project_path = ?`
|
WHERE project_path = ?
|
||||||
|
AND isArchived = 0`
|
||||||
)
|
)
|
||||||
.get(normalizedProjectPath) as { count: number } | undefined;
|
.get(normalizedProjectPath) as { count: number } | undefined;
|
||||||
|
|
||||||
@@ -167,6 +205,19 @@ export const sessionsDb = {
|
|||||||
return row?.custom_name ?? null;
|
return row?.custom_name ?? null;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Soft-delete and restore both use the same flag update so callers keep the
|
||||||
|
* row, metadata, and file path intact while toggling visibility.
|
||||||
|
*/
|
||||||
|
updateSessionIsArchived(sessionId: string, isArchived: boolean): void {
|
||||||
|
const db = getConnection();
|
||||||
|
db.prepare(
|
||||||
|
`UPDATE sessions
|
||||||
|
SET isArchived = ?
|
||||||
|
WHERE session_id = ?`
|
||||||
|
).run(isArchived ? 1 : 0, sessionId);
|
||||||
|
},
|
||||||
|
|
||||||
deleteSessionById(sessionId: string): boolean {
|
deleteSessionById(sessionId: string): boolean {
|
||||||
const db = getConnection();
|
const db = getConnection();
|
||||||
return db.prepare('DELETE FROM sessions WHERE session_id = ?').run(sessionId).changes > 0;
|
return db.prepare('DELETE FROM sessions WHERE session_id = ?').run(sessionId).changes > 0;
|
||||||
|
|||||||
@@ -86,6 +86,7 @@ CREATE TABLE IF NOT EXISTS sessions (
|
|||||||
custom_name TEXT,
|
custom_name TEXT,
|
||||||
project_path TEXT,
|
project_path TEXT,
|
||||||
jsonl_path TEXT,
|
jsonl_path TEXT,
|
||||||
|
isArchived BOOLEAN DEFAULT 0,
|
||||||
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
created_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
||||||
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
updated_at DATETIME DEFAULT CURRENT_TIMESTAMP,
|
||||||
PRIMARY KEY (session_id),
|
PRIMARY KEY (session_id),
|
||||||
|
|||||||
@@ -3,9 +3,9 @@ import express from 'express';
|
|||||||
import { createProject, updateProjectDisplayName } from '@/modules/projects/services/project-management.service.js';
|
import { createProject, updateProjectDisplayName } from '@/modules/projects/services/project-management.service.js';
|
||||||
import { startCloneProject } from '@/modules/projects/services/project-clone.service.js';
|
import { startCloneProject } from '@/modules/projects/services/project-clone.service.js';
|
||||||
import { getProjectTaskMaster } from '@/modules/projects/services/projects-has-taskmaster.service.js';
|
import { getProjectTaskMaster } from '@/modules/projects/services/projects-has-taskmaster.service.js';
|
||||||
import { AppError, asyncHandler } from '@/shared/utils.js';
|
import { AppError, asyncHandler, createApiSuccessResponse } from '@/shared/utils.js';
|
||||||
import { getProjectSessionsPage, getProjectsWithSessions } from '@/modules/projects/services/projects-with-sessions-fetch.service.js';
|
import { getArchivedProjectsWithSessions, getProjectSessionsPage, getProjectsWithSessions } from '@/modules/projects/services/projects-with-sessions-fetch.service.js';
|
||||||
import { deleteOrArchiveProject } from '@/modules/projects/services/project-delete.service.js';
|
import { deleteOrArchiveProject, restoreArchivedProject } from '@/modules/projects/services/project-delete.service.js';
|
||||||
import { applyLegacyStarredProjectIds, toggleProjectStar } from '@/modules/projects/services/project-star.service.js';
|
import { applyLegacyStarredProjectIds, toggleProjectStar } from '@/modules/projects/services/project-star.service.js';
|
||||||
|
|
||||||
const router = express.Router();
|
const router = express.Router();
|
||||||
@@ -73,6 +73,14 @@ router.get(
|
|||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
router.get(
|
||||||
|
'/archived',
|
||||||
|
asyncHandler(async (_req, res) => {
|
||||||
|
const projects = await getArchivedProjectsWithSessions();
|
||||||
|
res.json(createApiSuccessResponse({ projects }));
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
router.get(
|
router.get(
|
||||||
'/:projectId/sessions',
|
'/:projectId/sessions',
|
||||||
asyncHandler(async (req, res) => {
|
asyncHandler(async (req, res) => {
|
||||||
@@ -230,6 +238,15 @@ router.post(
|
|||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
router.post(
|
||||||
|
'/:projectId/restore',
|
||||||
|
asyncHandler(async (req, res) => {
|
||||||
|
const projectId = typeof req.params.projectId === 'string' ? req.params.projectId : '';
|
||||||
|
restoreArchivedProject(projectId);
|
||||||
|
res.json(createApiSuccessResponse({ projectId, isArchived: false }));
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* - `force` not set / false: archive project in DB only (`isArchived` = 1; hidden from active list).
|
* - `force` not set / false: archive project in DB only (`isArchived` = 1; hidden from active list).
|
||||||
* - `force=true`: remove DB row, delete session rows for that path, remove all `*.jsonl` under the Claude project dir.
|
* - `force=true`: remove DB row, delete session rows for that path, remove all `*.jsonl` under the Claude project dir.
|
||||||
|
|||||||
@@ -42,7 +42,7 @@ async function unlinkJsonlIfExists(filePath: string): Promise<void> {
|
|||||||
* Loads all session rows for the project path and removes each distinct `jsonl_path` file on disk.
|
* Loads all session rows for the project path and removes each distinct `jsonl_path` file on disk.
|
||||||
*/
|
*/
|
||||||
export async function deleteSessionJsonlFilesForProjectPath(projectPath: string): Promise<void> {
|
export async function deleteSessionJsonlFilesForProjectPath(projectPath: string): Promise<void> {
|
||||||
const sessions = sessionsDb.getSessionsByProjectPath(projectPath);
|
const sessions = sessionsDb.getSessionsByProjectPathIncludingArchived(projectPath);
|
||||||
const paths = uniqueJsonlPathsFromSessions(sessions);
|
const paths = uniqueJsonlPathsFromSessions(sessions);
|
||||||
|
|
||||||
for (const filePath of paths) {
|
for (const filePath of paths) {
|
||||||
@@ -73,3 +73,18 @@ export async function deleteOrArchiveProject(projectId: string, force: boolean):
|
|||||||
sessionsDb.deleteSessionsByProjectPath(row.project_path);
|
sessionsDb.deleteSessionsByProjectPath(row.project_path);
|
||||||
projectsDb.deleteProjectById(projectId);
|
projectsDb.deleteProjectById(projectId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Restores one archived project row back into the active project list.
|
||||||
|
*/
|
||||||
|
export function restoreArchivedProject(projectId: string): void {
|
||||||
|
const row = projectsDb.getProjectById(projectId);
|
||||||
|
if (!row) {
|
||||||
|
throw new AppError(`Unknown projectId: ${projectId}`, {
|
||||||
|
code: 'PROJECT_NOT_FOUND',
|
||||||
|
statusCode: 404,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
projectsDb.updateProjectIsArchivedById(projectId, false);
|
||||||
|
}
|
||||||
|
|||||||
@@ -40,6 +40,10 @@ export type ProjectListItem = {
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type ArchivedProjectListItem = ProjectListItem & {
|
||||||
|
isArchived: true;
|
||||||
|
};
|
||||||
|
|
||||||
type ProgressUpdate = {
|
type ProgressUpdate = {
|
||||||
phase: 'loading' | 'complete';
|
phase: 'loading' | 'complete';
|
||||||
current: number;
|
current: number;
|
||||||
@@ -150,6 +154,16 @@ function bucketSessionRowsByProvider(rows: SessionRepositoryRow[]): SessionsByPr
|
|||||||
return byProvider;
|
return byProvider;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function readProjectSessionsIncludingArchived(projectPath: string): ProjectSessionsPageResult {
|
||||||
|
const rows = sessionsDb.getSessionsByProjectPathIncludingArchived(projectPath) as SessionRepositoryRow[];
|
||||||
|
|
||||||
|
return {
|
||||||
|
sessionsByProvider: bucketSessionRowsByProvider(rows),
|
||||||
|
total: rows.length,
|
||||||
|
hasMore: false,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Reads one paginated project session slice from the DB and groups rows by provider.
|
* Reads one paginated project session slice from the DB and groups rows by provider.
|
||||||
*/
|
*/
|
||||||
@@ -255,6 +269,56 @@ export async function getProjectsWithSessions(
|
|||||||
return projects;
|
return projects;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads archived projects from DB and includes every session row for each
|
||||||
|
* project path, because an archived workspace should surface all preserved
|
||||||
|
* conversation history in the archive view regardless of each session's flag.
|
||||||
|
*/
|
||||||
|
export async function getArchivedProjectsWithSessions(
|
||||||
|
options: Pick<GetProjectsWithSessionsOptions, 'skipSynchronization'> = {},
|
||||||
|
): Promise<ArchivedProjectListItem[]> {
|
||||||
|
if (!options.skipSynchronization) {
|
||||||
|
await sessionSynchronizerService.synchronizeSessions();
|
||||||
|
}
|
||||||
|
|
||||||
|
const projectRows = projectsDb.getArchivedProjectPaths() as Array<{
|
||||||
|
project_id: string;
|
||||||
|
project_path: string;
|
||||||
|
custom_project_name?: string | null;
|
||||||
|
isStarred?: number;
|
||||||
|
}>;
|
||||||
|
|
||||||
|
const archivedProjects: ArchivedProjectListItem[] = [];
|
||||||
|
|
||||||
|
for (const row of projectRows) {
|
||||||
|
const displayName =
|
||||||
|
row.custom_project_name && row.custom_project_name.trim().length > 0
|
||||||
|
? row.custom_project_name
|
||||||
|
: await generateDisplayName(path.basename(row.project_path) || row.project_path, row.project_path);
|
||||||
|
|
||||||
|
const sessionsPage = readProjectSessionsIncludingArchived(row.project_path);
|
||||||
|
|
||||||
|
archivedProjects.push({
|
||||||
|
projectId: row.project_id,
|
||||||
|
path: row.project_path,
|
||||||
|
displayName,
|
||||||
|
fullPath: row.project_path,
|
||||||
|
isStarred: Boolean(row.isStarred),
|
||||||
|
isArchived: true,
|
||||||
|
sessions: sessionsPage.sessionsByProvider.claude,
|
||||||
|
cursorSessions: sessionsPage.sessionsByProvider.cursor,
|
||||||
|
codexSessions: sessionsPage.sessionsByProvider.codex,
|
||||||
|
geminiSessions: sessionsPage.sessionsByProvider.gemini,
|
||||||
|
sessionMeta: {
|
||||||
|
hasMore: sessionsPage.hasMore,
|
||||||
|
total: sessionsPage.total,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return archivedProjects;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Loads one paginated session slice for a specific project id.
|
* Loads one paginated session slice for a specific project id.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import path from 'node:path';
|
|||||||
|
|
||||||
import spawn from 'cross-spawn';
|
import spawn from 'cross-spawn';
|
||||||
|
|
||||||
|
import { resolveClaudeCodeExecutablePath } from '@/shared/claude-cli-path.js';
|
||||||
import type { IProviderAuth } from '@/shared/interfaces.js';
|
import type { IProviderAuth } from '@/shared/interfaces.js';
|
||||||
import type { ProviderAuthStatus } from '@/shared/types.js';
|
import type { ProviderAuthStatus } from '@/shared/types.js';
|
||||||
import { readObjectRecord, readOptionalString } from '@/shared/utils.js';
|
import { readObjectRecord, readOptionalString } from '@/shared/utils.js';
|
||||||
@@ -20,13 +21,13 @@ export class ClaudeProviderAuth implements IProviderAuth {
|
|||||||
* Checks whether the Claude Code CLI is available on this host.
|
* Checks whether the Claude Code CLI is available on this host.
|
||||||
*/
|
*/
|
||||||
private checkInstalled(): boolean {
|
private checkInstalled(): boolean {
|
||||||
const cliPath = process.env.CLAUDE_CLI_PATH || 'claude';
|
const cliPath = resolveClaudeCodeExecutablePath(process.env.CLAUDE_CLI_PATH);
|
||||||
try {
|
try {
|
||||||
spawn.sync(cliPath, ['--version'], { stdio: 'ignore', timeout: 5000 });
|
spawn.sync(cliPath, ['--version'], { stdio: 'ignore', timeout: 5000 });
|
||||||
return true;
|
return true;
|
||||||
} catch {
|
} catch {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import os from 'node:os';
|
import os from 'node:os';
|
||||||
import path from 'node:path';
|
import path from 'node:path';
|
||||||
|
import { readFile } from 'node:fs/promises';
|
||||||
|
|
||||||
import { sessionsDb } from '@/modules/database/index.js';
|
import { sessionsDb } from '@/modules/database/index.js';
|
||||||
import {
|
import {
|
||||||
@@ -91,7 +92,7 @@ export class ClaudeSessionSynchronizer implements IProviderSessionSynchronizer {
|
|||||||
filePath: string,
|
filePath: string,
|
||||||
nameMap: Map<string, string>
|
nameMap: Map<string, string>
|
||||||
): Promise<ParsedSession | null> {
|
): Promise<ParsedSession | null> {
|
||||||
return extractFirstValidJsonlData(filePath, (rawData) => {
|
const parsed = await extractFirstValidJsonlData(filePath, (rawData) => {
|
||||||
const data = rawData as Record<string, unknown>;
|
const data = rawData as Record<string, unknown>;
|
||||||
const sessionId = typeof data.sessionId === 'string' ? data.sessionId : undefined;
|
const sessionId = typeof data.sessionId === 'string' ? data.sessionId : undefined;
|
||||||
const projectPath = typeof data.cwd === 'string' ? data.cwd : undefined;
|
const projectPath = typeof data.cwd === 'string' ? data.cwd : undefined;
|
||||||
@@ -103,8 +104,73 @@ export class ClaudeSessionSynchronizer implements IProviderSessionSynchronizer {
|
|||||||
return {
|
return {
|
||||||
sessionId,
|
sessionId,
|
||||||
projectPath,
|
projectPath,
|
||||||
sessionName: normalizeSessionName(nameMap.get(sessionId), 'Untitled Claude Session'),
|
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (!parsed) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const existingSession = sessionsDb.getSessionById(parsed.sessionId);
|
||||||
|
const existingSessionName = existingSession?.custom_name;
|
||||||
|
if (existingSessionName && existingSessionName !== 'Untitled Claude Session') {
|
||||||
|
return {
|
||||||
|
...parsed,
|
||||||
|
sessionName: normalizeSessionName(existingSessionName, 'Untitled Claude Session'),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
let sessionName = nameMap.get(parsed.sessionId);
|
||||||
|
if (!sessionName) {
|
||||||
|
sessionName = await this.extractSessionAiTitleFromEnd(filePath, parsed.sessionId);
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
...parsed,
|
||||||
|
sessionName: normalizeSessionName(sessionName, 'Untitled Claude Session'),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private async extractSessionAiTitleFromEnd(
|
||||||
|
filePath: string,
|
||||||
|
sessionId: string
|
||||||
|
): Promise<string | undefined> {
|
||||||
|
try {
|
||||||
|
const content = await readFile(filePath, 'utf8');
|
||||||
|
const lines = content.split(/\r?\n/);
|
||||||
|
|
||||||
|
for (let index = lines.length - 1; index >= 0; index -= 1) {
|
||||||
|
const line = lines[index]?.trim();
|
||||||
|
if (!line) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let parsed: unknown;
|
||||||
|
try {
|
||||||
|
parsed = JSON.parse(line);
|
||||||
|
} catch {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = parsed as Record<string, unknown>;
|
||||||
|
const eventType = typeof data.type === 'string' ? data.type : undefined;
|
||||||
|
const eventSessionId = typeof data.sessionId === 'string' ? data.sessionId : undefined;
|
||||||
|
const aiTitle = typeof data.aiTitle === 'string' ? data.aiTitle : undefined;
|
||||||
|
const lastPrompt = typeof data.lastPrompt === 'string' ? data.lastPrompt : undefined;
|
||||||
|
const claudeRenamedTitle = typeof data.customTitle === 'string' ? data.customTitle : undefined;
|
||||||
|
|
||||||
|
if (
|
||||||
|
(eventType === 'ai-title' && eventSessionId === sessionId && aiTitle?.trim()) ||
|
||||||
|
(eventType === 'last-prompt' && eventSessionId === sessionId && lastPrompt?.trim()) ||
|
||||||
|
(eventType === "custom-title" && eventSessionId === sessionId && claudeRenamedTitle?.trim())
|
||||||
|
) {
|
||||||
|
return aiTitle || lastPrompt || claudeRenamedTitle;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
// Ignore missing/unreadable files so sync can continue.
|
||||||
|
}
|
||||||
|
|
||||||
|
return undefined;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -200,17 +200,18 @@ async function getSessionMessages(
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Claude writes internal command and system reminder entries into history.
|
* Claude writes a mix of truly internal transcript rows and "UI-hidden" local
|
||||||
* Those are useful for the CLI but should not appear in the user-facing chat.
|
* command artifacts into the same JSONL stream.
|
||||||
|
*
|
||||||
|
* Important distinction:
|
||||||
|
* - system reminders / caveats / interruption banners should stay hidden
|
||||||
|
* - local command payloads (`<command-name>...`) and stdout wrappers
|
||||||
|
* (`<local-command-stdout>...`) should be remapped into normal chat messages
|
||||||
|
* instead of being discarded as internal content
|
||||||
*/
|
*/
|
||||||
const INTERNAL_CONTENT_PREFIXES = [
|
const INTERNAL_CONTENT_PREFIXES = [
|
||||||
'<command-name>',
|
|
||||||
'<command-message>',
|
|
||||||
'<command-args>',
|
|
||||||
'<local-command-stdout>',
|
|
||||||
'<system-reminder>',
|
'<system-reminder>',
|
||||||
'Caveat:',
|
'Caveat:',
|
||||||
'This session is being continued from a previous',
|
|
||||||
'[Request interrupted',
|
'[Request interrupted',
|
||||||
] as const;
|
] as const;
|
||||||
|
|
||||||
@@ -218,6 +219,73 @@ function isInternalContent(content: string): boolean {
|
|||||||
return INTERNAL_CONTENT_PREFIXES.some((prefix) => content.startsWith(prefix));
|
return INTERNAL_CONTENT_PREFIXES.some((prefix) => content.startsWith(prefix));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Claude wraps local slash-command metadata in lightweight XML-like tags inside
|
||||||
|
* a plain string payload. We intentionally parse only the small tag surface we
|
||||||
|
* care about instead of introducing a generic XML parser for untrusted history.
|
||||||
|
*/
|
||||||
|
function extractTaggedContent(content: string, tagName: string): string | null {
|
||||||
|
const escapedTagName = tagName.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
|
||||||
|
const match = new RegExp(`<${escapedTagName}>([\\s\\S]*?)<\\/${escapedTagName}>`).exec(content);
|
||||||
|
return match ? match[1] : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
type ClaudeLocalCommandPayload = {
|
||||||
|
commandName: string;
|
||||||
|
commandMessage: string;
|
||||||
|
commandArgs: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Converts Claude's hidden local command wrapper into structured metadata.
|
||||||
|
*
|
||||||
|
* The three tags often coexist in one string payload. Returning `null` lets the
|
||||||
|
* normal text path continue untouched for unrelated messages.
|
||||||
|
*/
|
||||||
|
function parseLocalCommandPayload(content: string): ClaudeLocalCommandPayload | null {
|
||||||
|
const commandName = extractTaggedContent(content, 'command-name');
|
||||||
|
const commandMessage = extractTaggedContent(content, 'command-message');
|
||||||
|
const commandArgs = extractTaggedContent(content, 'command-args');
|
||||||
|
|
||||||
|
if (commandName === null && commandMessage === null && commandArgs === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
commandName: commandName ?? '',
|
||||||
|
commandMessage: commandMessage ?? '',
|
||||||
|
commandArgs: commandArgs ?? '',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Produces the short user-visible command string that should appear in chat.
|
||||||
|
*
|
||||||
|
* We prefer the slash-prefixed command name because that most closely matches
|
||||||
|
* what the user actually typed, and only fall back to the message body when the
|
||||||
|
* command name is unavailable in older transcript variants.
|
||||||
|
*/
|
||||||
|
function buildLocalCommandDisplayText(payload: ClaudeLocalCommandPayload): string {
|
||||||
|
const commandName = payload.commandName.trim();
|
||||||
|
const commandMessage = payload.commandMessage.trim();
|
||||||
|
const commandArgs = payload.commandArgs.trim();
|
||||||
|
const baseCommand = commandName || commandMessage;
|
||||||
|
|
||||||
|
if (!baseCommand) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return commandArgs ? `${baseCommand} ${commandArgs}` : baseCommand;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Claude local-command stdout may contain ANSI styling codes because it was
|
||||||
|
* captured from the terminal. The web chat should receive readable plain text.
|
||||||
|
*/
|
||||||
|
function stripAnsiFormatting(text: string): string {
|
||||||
|
return text.replace(/\u001B\[[0-9;?]*[ -/]*[@-~]/g, '');
|
||||||
|
}
|
||||||
|
|
||||||
export class ClaudeSessionsProvider implements IProviderSessions {
|
export class ClaudeSessionsProvider implements IProviderSessions {
|
||||||
/**
|
/**
|
||||||
* Normalizes one Claude JSONL entry or live SDK stream event into the shared
|
* Normalizes one Claude JSONL entry or live SDK stream event into the shared
|
||||||
@@ -240,7 +308,7 @@ export class ClaudeSessionsProvider implements IProviderSessions {
|
|||||||
const ts = raw.timestamp || new Date().toISOString();
|
const ts = raw.timestamp || new Date().toISOString();
|
||||||
const baseId = raw.uuid || generateMessageId('claude');
|
const baseId = raw.uuid || generateMessageId('claude');
|
||||||
|
|
||||||
if (raw.message?.role === 'user' && raw.message?.content) {
|
if (raw.message?.role === 'user' && raw.message?.content && raw.isMeta !== true) {
|
||||||
if (Array.isArray(raw.message.content)) {
|
if (Array.isArray(raw.message.content)) {
|
||||||
for (let partIndex = 0; partIndex < raw.message.content.length; partIndex++) {
|
for (let partIndex = 0; partIndex < raw.message.content.length; partIndex++) {
|
||||||
const part = raw.message.content[partIndex];
|
const part = raw.message.content[partIndex];
|
||||||
@@ -293,6 +361,80 @@ export class ClaudeSessionsProvider implements IProviderSessions {
|
|||||||
}
|
}
|
||||||
} else if (typeof raw.message.content === 'string') {
|
} else if (typeof raw.message.content === 'string') {
|
||||||
const text = raw.message.content;
|
const text = raw.message.content;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Claude stores compact summaries as synthetic "user" rows so the CLI
|
||||||
|
* can resume the next session turn with the summary in-context.
|
||||||
|
*
|
||||||
|
* For the web UI this is much more useful as assistant-authored summary
|
||||||
|
* text; otherwise it is both filtered by the generic internal-prefix
|
||||||
|
* check and visually mislabeled as a user message.
|
||||||
|
*/
|
||||||
|
if (raw.isCompactSummary === true && text.trim()) {
|
||||||
|
messages.push(createNormalizedMessage({
|
||||||
|
id: baseId,
|
||||||
|
sessionId,
|
||||||
|
timestamp: ts,
|
||||||
|
provider: PROVIDER,
|
||||||
|
kind: 'text',
|
||||||
|
role: 'assistant',
|
||||||
|
content: text,
|
||||||
|
isCompactSummary: true,
|
||||||
|
}));
|
||||||
|
return messages;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Local slash commands are serialized as tagged text even though they
|
||||||
|
* are semantically a user action. Expose the parsed fields to the
|
||||||
|
* frontend and emit a plain user-visible command string so the command
|
||||||
|
* no longer disappears from history.
|
||||||
|
*/
|
||||||
|
const localCommandPayload = parseLocalCommandPayload(text);
|
||||||
|
if (localCommandPayload) {
|
||||||
|
const displayText = buildLocalCommandDisplayText(localCommandPayload);
|
||||||
|
if (displayText) {
|
||||||
|
messages.push(createNormalizedMessage({
|
||||||
|
id: baseId,
|
||||||
|
sessionId,
|
||||||
|
timestamp: ts,
|
||||||
|
provider: PROVIDER,
|
||||||
|
kind: 'text',
|
||||||
|
role: 'user',
|
||||||
|
content: displayText,
|
||||||
|
commandName: localCommandPayload.commandName,
|
||||||
|
commandMessage: localCommandPayload.commandMessage,
|
||||||
|
commandArgs: localCommandPayload.commandArgs,
|
||||||
|
isLocalCommand: true,
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
return messages;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Local command stdout is also written as a "user" row in Claude's
|
||||||
|
* transcript, but it is terminal output produced in response to the
|
||||||
|
* command. Re-label it as assistant text so the chat transcript matches
|
||||||
|
* the actual conversational flow seen by the user.
|
||||||
|
*/
|
||||||
|
const localCommandStdout = extractTaggedContent(text, 'local-command-stdout');
|
||||||
|
if (localCommandStdout !== null) {
|
||||||
|
const stdoutText = stripAnsiFormatting(localCommandStdout).trim();
|
||||||
|
if (stdoutText) {
|
||||||
|
messages.push(createNormalizedMessage({
|
||||||
|
id: baseId,
|
||||||
|
sessionId,
|
||||||
|
timestamp: ts,
|
||||||
|
provider: PROVIDER,
|
||||||
|
kind: 'text',
|
||||||
|
role: 'assistant',
|
||||||
|
content: stdoutText,
|
||||||
|
isLocalCommandStdout: true,
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
return messages;
|
||||||
|
}
|
||||||
|
|
||||||
if (text && !isInternalContent(text)) {
|
if (text && !isInternalContent(text)) {
|
||||||
messages.push(createNormalizedMessage({
|
messages.push(createNormalizedMessage({
|
||||||
id: baseId,
|
id: baseId,
|
||||||
@@ -414,7 +556,9 @@ export class ClaudeSessionsProvider implements IProviderSessions {
|
|||||||
|
|
||||||
let result: ClaudeHistoryResult;
|
let result: ClaudeHistoryResult;
|
||||||
try {
|
try {
|
||||||
result = await getSessionMessages(sessionId, limit, offset);
|
// Load full history first so `total` reflects frontend-normalized messages,
|
||||||
|
// not raw JSONL records.
|
||||||
|
result = await getSessionMessages(sessionId, null, 0);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const message = error instanceof Error ? error.message : String(error);
|
const message = error instanceof Error ? error.message : String(error);
|
||||||
console.warn(`[ClaudeProvider] Failed to load session ${sessionId}:`, message);
|
console.warn(`[ClaudeProvider] Failed to load session ${sessionId}:`, message);
|
||||||
@@ -422,8 +566,6 @@ export class ClaudeSessionsProvider implements IProviderSessions {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const rawMessages = Array.isArray(result) ? result : (result.messages || []);
|
const rawMessages = Array.isArray(result) ? result : (result.messages || []);
|
||||||
const total = Array.isArray(result) ? rawMessages.length : (result.total || 0);
|
|
||||||
const hasMore = Array.isArray(result) ? false : Boolean(result.hasMore);
|
|
||||||
|
|
||||||
const toolResultMap = new Map<string, ClaudeToolResult>();
|
const toolResultMap = new Map<string, ClaudeToolResult>();
|
||||||
for (const raw of rawMessages) {
|
for (const raw of rawMessages) {
|
||||||
@@ -464,12 +606,31 @@ export class ClaudeSessionsProvider implements IProviderSessions {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const totalNormalized = normalized.length;
|
||||||
|
let total = 0;
|
||||||
|
for (const msg of normalized) {
|
||||||
|
if (msg.kind !== 'tool_result') {
|
||||||
|
total += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const normalizedOffset = Math.max(0, offset);
|
||||||
|
const normalizedLimit = limit === null ? null : Math.max(0, limit);
|
||||||
|
const messages = normalizedLimit === null
|
||||||
|
? normalized
|
||||||
|
: normalized.slice(
|
||||||
|
Math.max(0, totalNormalized - normalizedOffset - normalizedLimit),
|
||||||
|
Math.max(0, totalNormalized - normalizedOffset),
|
||||||
|
);
|
||||||
|
const hasMore = normalizedLimit === null
|
||||||
|
? false
|
||||||
|
: Math.max(0, totalNormalized - normalizedOffset - normalizedLimit) > 0;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
messages: normalized,
|
messages,
|
||||||
total,
|
total,
|
||||||
hasMore,
|
hasMore,
|
||||||
offset,
|
offset: normalizedOffset,
|
||||||
limit,
|
limit: normalizedLimit,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import os from 'node:os';
|
import os from 'node:os';
|
||||||
import path from 'node:path';
|
import path from 'node:path';
|
||||||
|
import { readFile } from 'node:fs/promises';
|
||||||
|
|
||||||
import { sessionsDb } from '@/modules/database/index.js';
|
import { sessionsDb } from '@/modules/database/index.js';
|
||||||
import {
|
import {
|
||||||
@@ -99,7 +100,7 @@ export class CodexSessionSynchronizer implements IProviderSessionSynchronizer {
|
|||||||
filePath: string,
|
filePath: string,
|
||||||
nameMap: Map<string, string>
|
nameMap: Map<string, string>
|
||||||
): Promise<ParsedSession | null> {
|
): Promise<ParsedSession | null> {
|
||||||
return extractFirstValidJsonlData(filePath, (rawData) => {
|
const parsed = await extractFirstValidJsonlData(filePath, (rawData) => {
|
||||||
const data = rawData as Record<string, unknown>;
|
const data = rawData as Record<string, unknown>;
|
||||||
const payload = data.payload as Record<string, unknown> | undefined;
|
const payload = data.payload as Record<string, unknown> | undefined;
|
||||||
const sessionId = typeof payload?.id === 'string' ? payload.id : undefined;
|
const sessionId = typeof payload?.id === 'string' ? payload.id : undefined;
|
||||||
@@ -112,8 +113,67 @@ export class CodexSessionSynchronizer implements IProviderSessionSynchronizer {
|
|||||||
return {
|
return {
|
||||||
sessionId,
|
sessionId,
|
||||||
projectPath,
|
projectPath,
|
||||||
sessionName: normalizeSessionName(nameMap.get(sessionId), 'Untitled Codex Session'),
|
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (!parsed) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const existingSession = sessionsDb.getSessionById(parsed.sessionId);
|
||||||
|
const existingSessionName = existingSession?.custom_name;
|
||||||
|
if (existingSessionName && existingSessionName !== 'Untitled Codex Session') {
|
||||||
|
return {
|
||||||
|
...parsed,
|
||||||
|
sessionName: normalizeSessionName(existingSessionName, 'Untitled Codex Session'),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
let sessionName = nameMap.get(parsed.sessionId);
|
||||||
|
if (!sessionName) {
|
||||||
|
sessionName = await this.extractLastAgentMessageFromEnd(filePath);
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
...parsed,
|
||||||
|
sessionName: normalizeSessionName(sessionName, 'Untitled Codex Session'),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private async extractLastAgentMessageFromEnd(filePath: string): Promise<string | undefined> {
|
||||||
|
try {
|
||||||
|
const content = await readFile(filePath, 'utf8');
|
||||||
|
const lines = content.split(/\r?\n/);
|
||||||
|
|
||||||
|
for (let index = lines.length - 1; index >= 0; index -= 1) {
|
||||||
|
const line = lines[index]?.trim();
|
||||||
|
if (!line) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let parsed: unknown;
|
||||||
|
try {
|
||||||
|
parsed = JSON.parse(line);
|
||||||
|
} catch {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = parsed as Record<string, unknown>;
|
||||||
|
const eventType = typeof data.type === 'string' ? data.type : undefined;
|
||||||
|
const payload = data.payload as Record<string, unknown> | undefined;
|
||||||
|
const payloadType = typeof payload?.type === 'string' ? payload.type : undefined;
|
||||||
|
const lastAgentMessage = typeof payload?.last_agent_message === 'string'
|
||||||
|
? payload.last_agent_message
|
||||||
|
: undefined;
|
||||||
|
|
||||||
|
if (eventType === 'event_msg' && payloadType === 'task_complete' && lastAgentMessage?.trim()) {
|
||||||
|
return lastAgentMessage;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
// Ignore missing/unreadable files so sync can continue.
|
||||||
|
}
|
||||||
|
|
||||||
|
return undefined;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -520,7 +520,9 @@ export class CodexSessionsProvider implements IProviderSessions {
|
|||||||
|
|
||||||
let result: CodexHistoryResult;
|
let result: CodexHistoryResult;
|
||||||
try {
|
try {
|
||||||
result = await getCodexSessionMessages(sessionId, limit, offset);
|
// Load full history first so `total` reflects frontend-normalized messages,
|
||||||
|
// not raw JSONL records.
|
||||||
|
result = await getCodexSessionMessages(sessionId, null, 0);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const message = error instanceof Error ? error.message : String(error);
|
const message = error instanceof Error ? error.message : String(error);
|
||||||
console.warn(`[CodexProvider] Failed to load session ${sessionId}:`, message);
|
console.warn(`[CodexProvider] Failed to load session ${sessionId}:`, message);
|
||||||
@@ -528,8 +530,6 @@ export class CodexSessionsProvider implements IProviderSessions {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const rawMessages = Array.isArray(result) ? result : (result.messages || []);
|
const rawMessages = Array.isArray(result) ? result : (result.messages || []);
|
||||||
const total = Array.isArray(result) ? rawMessages.length : (result.total || 0);
|
|
||||||
const hasMore = Array.isArray(result) ? false : Boolean(result.hasMore);
|
|
||||||
const tokenUsage = Array.isArray(result) ? undefined : result.tokenUsage;
|
const tokenUsage = Array.isArray(result) ? undefined : result.tokenUsage;
|
||||||
|
|
||||||
const normalized: NormalizedMessage[] = [];
|
const normalized: NormalizedMessage[] = [];
|
||||||
@@ -552,12 +552,31 @@ export class CodexSessionsProvider implements IProviderSessions {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const totalNormalized = normalized.length;
|
||||||
|
let total = 0;
|
||||||
|
for (const msg of normalized) {
|
||||||
|
if (msg.kind !== 'tool_result') {
|
||||||
|
total += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const normalizedOffset = Math.max(0, offset);
|
||||||
|
const normalizedLimit = limit === null ? null : Math.max(0, limit);
|
||||||
|
const messages = normalizedLimit === null
|
||||||
|
? normalized
|
||||||
|
: normalized.slice(
|
||||||
|
Math.max(0, totalNormalized - normalizedOffset - normalizedLimit),
|
||||||
|
Math.max(0, totalNormalized - normalizedOffset),
|
||||||
|
);
|
||||||
|
const hasMore = normalizedLimit === null
|
||||||
|
? false
|
||||||
|
: Math.max(0, totalNormalized - normalizedOffset - normalizedLimit) > 0;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
messages: normalized,
|
messages,
|
||||||
total,
|
total,
|
||||||
hasMore,
|
hasMore,
|
||||||
offset,
|
offset: normalizedOffset,
|
||||||
limit,
|
limit: normalizedLimit,
|
||||||
tokenUsage,
|
tokenUsage,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -45,44 +45,28 @@ export class CursorSessionSynchronizer implements IProviderSessionSynchronizer {
|
|||||||
*/
|
*/
|
||||||
async synchronize(since?: Date): Promise<number> {
|
async synchronize(since?: Date): Promise<number> {
|
||||||
const projectsDir = path.join(this.cursorHome, 'projects');
|
const projectsDir = path.join(this.cursorHome, 'projects');
|
||||||
const projectEntries = await listDirectoryEntriesSafe(projectsDir);
|
|
||||||
const seenProjectPaths = new Set<string>();
|
|
||||||
|
|
||||||
let processed = 0;
|
let processed = 0;
|
||||||
for (const entry of projectEntries) {
|
|
||||||
if (!entry.isDirectory()) {
|
const files = await findFilesRecursivelyCreatedAfter(projectsDir, '.jsonl', since ?? null);
|
||||||
|
|
||||||
|
for (const filePath of files) {
|
||||||
|
const parsed = await this.processSessionFile(filePath);
|
||||||
|
if (!parsed) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
const workerLogPath = path.join(projectsDir, entry.name, 'worker.log');
|
const timestamps = await readFileTimestamps(filePath);
|
||||||
const projectPath = await this.extractProjectPathFromWorkerLog(workerLogPath);
|
sessionsDb.createSession(
|
||||||
if (!projectPath || seenProjectPaths.has(projectPath)) {
|
parsed.sessionId,
|
||||||
continue;
|
this.provider,
|
||||||
}
|
parsed.projectPath,
|
||||||
|
parsed.sessionName,
|
||||||
seenProjectPaths.add(projectPath);
|
timestamps.createdAt,
|
||||||
const projectHash = this.md5(projectPath);
|
timestamps.updatedAt,
|
||||||
const chatsDir = path.join(this.cursorHome, 'chats', projectHash);
|
filePath
|
||||||
const files = await findFilesRecursivelyCreatedAfter(chatsDir, '.jsonl', since ?? null);
|
);
|
||||||
|
processed += 1;
|
||||||
for (const filePath of files) {
|
|
||||||
const parsed = await this.processSessionFile(filePath);
|
|
||||||
if (!parsed) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
const timestamps = await readFileTimestamps(filePath);
|
|
||||||
sessionsDb.createSession(
|
|
||||||
parsed.sessionId,
|
|
||||||
this.provider,
|
|
||||||
parsed.projectPath,
|
|
||||||
parsed.sessionName,
|
|
||||||
timestamps.createdAt,
|
|
||||||
timestamps.updatedAt,
|
|
||||||
filePath
|
|
||||||
);
|
|
||||||
processed += 1;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return processed;
|
return processed;
|
||||||
@@ -113,13 +97,6 @@ export class CursorSessionSynchronizer implements IProviderSessionSynchronizer {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Produces the same project hash Cursor uses in chat directory names.
|
|
||||||
*/
|
|
||||||
private md5(input: string): string {
|
|
||||||
return crypto.createHash('md5').update(input).digest('hex');
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Extracts project path from Cursor worker.log.
|
* Extracts project path from Cursor worker.log.
|
||||||
*/
|
*/
|
||||||
@@ -149,7 +126,7 @@ export class CursorSessionSynchronizer implements IProviderSessionSynchronizer {
|
|||||||
*/
|
*/
|
||||||
private async processSessionFile(filePath: string): Promise<ParsedSession | null> {
|
private async processSessionFile(filePath: string): Promise<ParsedSession | null> {
|
||||||
const sessionId = path.basename(filePath, '.jsonl');
|
const sessionId = path.basename(filePath, '.jsonl');
|
||||||
const grandparentDir = path.dirname(path.dirname(filePath));
|
const grandparentDir = path.dirname(path.dirname(path.dirname(filePath)));
|
||||||
const workerLogPath = path.join(grandparentDir, 'worker.log');
|
const workerLogPath = path.join(grandparentDir, 'worker.log');
|
||||||
const projectPath = await this.extractProjectPathFromWorkerLog(workerLogPath);
|
const projectPath = await this.extractProjectPathFromWorkerLog(workerLogPath);
|
||||||
|
|
||||||
|
|||||||
@@ -25,6 +25,167 @@ type CursorMessageBlob = {
|
|||||||
content: AnyRecord;
|
content: AnyRecord;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function isInternalCursorText(value: unknown): boolean {
|
||||||
|
if (typeof value !== 'string') {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const normalized = value.trim();
|
||||||
|
return normalized.startsWith('<user_info>') || normalized.startsWith('<system_reminder>');
|
||||||
|
}
|
||||||
|
|
||||||
|
function isInternalCursorPart(part: unknown): boolean {
|
||||||
|
if (!part || typeof part !== 'object') {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const record = part as AnyRecord;
|
||||||
|
const type = typeof record.type === 'string' ? record.type : '';
|
||||||
|
if (type === 'user_info' || type === 'system_reminder') {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return isInternalCursorText(record.text);
|
||||||
|
}
|
||||||
|
|
||||||
|
function unwrapUserQueryText(value: string, role: 'user' | 'assistant'): string {
|
||||||
|
if (role !== 'user') {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
const normalized = value.trimStart();
|
||||||
|
const openTag = '<user_query>';
|
||||||
|
const closeTag = '</user_query>';
|
||||||
|
if (!normalized.startsWith(openTag)) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
const afterOpen = normalized.slice(openTag.length);
|
||||||
|
const closeIndex = afterOpen.lastIndexOf(closeTag);
|
||||||
|
const inner = closeIndex >= 0 ? afterOpen.slice(0, closeIndex) : afterOpen;
|
||||||
|
return inner.trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeToolId(value: unknown): string | null {
|
||||||
|
if (typeof value !== 'string') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const normalized = value.trim();
|
||||||
|
return normalized ? normalized : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
function extractCursorToolResultContent(item: AnyRecord): string {
|
||||||
|
if (typeof item.result === 'string' && item.result.trim()) {
|
||||||
|
return item.result;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof item.output === 'string' && item.output.trim()) {
|
||||||
|
return item.output;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Array.isArray(item.experimental_content)) {
|
||||||
|
const experimentalText = item.experimental_content
|
||||||
|
.map((part: unknown) => {
|
||||||
|
if (typeof part === 'string') {
|
||||||
|
return part;
|
||||||
|
}
|
||||||
|
if (part && typeof part === 'object') {
|
||||||
|
const record = part as AnyRecord;
|
||||||
|
if (typeof record.text === 'string') {
|
||||||
|
return record.text;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return '';
|
||||||
|
})
|
||||||
|
.filter(Boolean)
|
||||||
|
.join('\n');
|
||||||
|
|
||||||
|
if (experimentalText.trim()) {
|
||||||
|
return experimentalText;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return typeof item.result === 'string' ? item.result : '';
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseCursorToolInput(rawInput: unknown): unknown {
|
||||||
|
if (typeof rawInput !== 'string') {
|
||||||
|
return rawInput;
|
||||||
|
}
|
||||||
|
|
||||||
|
const trimmed = rawInput.trim();
|
||||||
|
if (!trimmed) {
|
||||||
|
return rawInput;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
return JSON.parse(trimmed);
|
||||||
|
} catch {
|
||||||
|
return rawInput;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeCursorToolInput(toolName: string, rawInput: unknown): unknown {
|
||||||
|
const parsed = parseCursorToolInput(rawInput);
|
||||||
|
if (!parsed || typeof parsed !== 'object' || Array.isArray(parsed)) {
|
||||||
|
return parsed;
|
||||||
|
}
|
||||||
|
|
||||||
|
const input = parsed as AnyRecord;
|
||||||
|
const normalized: AnyRecord = { ...input };
|
||||||
|
|
||||||
|
const filePath = input.file_path
|
||||||
|
?? input.filePath
|
||||||
|
?? input.path
|
||||||
|
?? input.file
|
||||||
|
?? input.filename;
|
||||||
|
if (typeof filePath === 'string' && filePath.trim()) {
|
||||||
|
normalized.file_path = filePath;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (toolName === 'Write') {
|
||||||
|
const content = input.content
|
||||||
|
?? input.text
|
||||||
|
?? input.value
|
||||||
|
?? input.contents
|
||||||
|
?? input.fileContent
|
||||||
|
?? input.new_string
|
||||||
|
?? input.newString;
|
||||||
|
if (typeof content === 'string') {
|
||||||
|
normalized.content = content;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (toolName === 'Edit') {
|
||||||
|
const oldString = input.old_string
|
||||||
|
?? input.oldString
|
||||||
|
?? input.old
|
||||||
|
?? '';
|
||||||
|
const newString = input.new_string
|
||||||
|
?? input.newString
|
||||||
|
?? input.new
|
||||||
|
?? input.content
|
||||||
|
?? '';
|
||||||
|
|
||||||
|
if (typeof oldString === 'string') {
|
||||||
|
normalized.old_string = oldString;
|
||||||
|
}
|
||||||
|
if (typeof newString === 'string') {
|
||||||
|
normalized.new_string = newString;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (toolName === 'ApplyPatch') {
|
||||||
|
const patch = input.patch ?? input.diff ?? input.content;
|
||||||
|
if (typeof patch === 'string' && !normalized.patch) {
|
||||||
|
normalized.patch = patch;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return normalized;
|
||||||
|
}
|
||||||
|
|
||||||
function sanitizeCursorSessionId(sessionId: string): string {
|
function sanitizeCursorSessionId(sessionId: string): string {
|
||||||
const normalized = sessionId.trim();
|
const normalized = sessionId.trim();
|
||||||
if (!normalized) {
|
if (!normalized) {
|
||||||
@@ -225,13 +386,14 @@ export class CursorSessionsProvider implements IProviderSessions {
|
|||||||
try {
|
try {
|
||||||
const blobs = await this.loadCursorBlobs(sessionId, projectPath);
|
const blobs = await this.loadCursorBlobs(sessionId, projectPath);
|
||||||
const allNormalized = this.normalizeCursorBlobs(blobs, sessionId);
|
const allNormalized = this.normalizeCursorBlobs(blobs, sessionId);
|
||||||
const total = allNormalized.length;
|
const renderableMessages = allNormalized.filter((msg) => msg.kind !== 'tool_result');
|
||||||
|
const total = renderableMessages.length;
|
||||||
|
|
||||||
if (limit !== null) {
|
if (limit !== null) {
|
||||||
const start = offset;
|
const start = offset;
|
||||||
const page = limit === 0
|
const page = limit === 0
|
||||||
? []
|
? []
|
||||||
: allNormalized.slice(start, start + limit);
|
: renderableMessages.slice(start, start + limit);
|
||||||
const hasMore = limit === 0
|
const hasMore = limit === 0
|
||||||
? start < total
|
? start < total
|
||||||
: start + limit < total;
|
: start + limit < total;
|
||||||
@@ -245,7 +407,7 @@ export class CursorSessionsProvider implements IProviderSessions {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
messages: allNormalized,
|
messages: renderableMessages,
|
||||||
total,
|
total,
|
||||||
hasMore: false,
|
hasMore: false,
|
||||||
offset: 0,
|
offset: 0,
|
||||||
@@ -283,11 +445,24 @@ export class CursorSessionsProvider implements IProviderSessions {
|
|||||||
let text = '';
|
let text = '';
|
||||||
if (Array.isArray(content.message.content)) {
|
if (Array.isArray(content.message.content)) {
|
||||||
text = content.message.content
|
text = content.message.content
|
||||||
.map((part: string | AnyRecord) => typeof part === 'string' ? part : part?.text || '')
|
.map((part: string | AnyRecord) => {
|
||||||
|
if (typeof part === 'string') {
|
||||||
|
if (isInternalCursorText(part)) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
return unwrapUserQueryText(part, role);
|
||||||
|
}
|
||||||
|
if (isInternalCursorPart(part)) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
return unwrapUserQueryText(part?.text || '', role);
|
||||||
|
})
|
||||||
.filter(Boolean)
|
.filter(Boolean)
|
||||||
.join('\n');
|
.join('\n');
|
||||||
} else if (typeof content.message.content === 'string') {
|
} else if (typeof content.message.content === 'string') {
|
||||||
text = content.message.content;
|
if (!isInternalCursorText(content.message.content)) {
|
||||||
|
text = unwrapUserQueryText(content.message.content, role);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (text?.trim()) {
|
if (text?.trim()) {
|
||||||
messages.push(createNormalizedMessage({
|
messages.push(createNormalizedMessage({
|
||||||
@@ -316,7 +491,14 @@ export class CursorSessionsProvider implements IProviderSessions {
|
|||||||
if (item?.type !== 'tool-result') {
|
if (item?.type !== 'tool-result') {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
const toolCallId = item.toolCallId || content.id;
|
const cursorOptions = content.providerOptions?.cursor as AnyRecord | undefined;
|
||||||
|
const highLevelToolCallResult = cursorOptions?.highLevelToolCallResult;
|
||||||
|
const toolCallId = normalizeToolId(item.toolCallId)
|
||||||
|
|| normalizeToolId(item.tool_call_id)
|
||||||
|
|| normalizeToolId(highLevelToolCallResult?.toolCallId)
|
||||||
|
|| normalizeToolId(highLevelToolCallResult?.tool_call_id)
|
||||||
|
|| normalizeToolId(content.id)
|
||||||
|
|| '';
|
||||||
messages.push(createNormalizedMessage({
|
messages.push(createNormalizedMessage({
|
||||||
id: `${baseId}_tr`,
|
id: `${baseId}_tr`,
|
||||||
sessionId,
|
sessionId,
|
||||||
@@ -324,8 +506,9 @@ export class CursorSessionsProvider implements IProviderSessions {
|
|||||||
provider: PROVIDER,
|
provider: PROVIDER,
|
||||||
kind: 'tool_result',
|
kind: 'tool_result',
|
||||||
toolId: toolCallId,
|
toolId: toolCallId,
|
||||||
content: item.result || '',
|
content: extractCursorToolResultContent(item),
|
||||||
isError: false,
|
isError: Boolean(item.isError || item.is_error),
|
||||||
|
toolUseResult: highLevelToolCallResult,
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
@@ -336,8 +519,15 @@ export class CursorSessionsProvider implements IProviderSessions {
|
|||||||
if (Array.isArray(content.content)) {
|
if (Array.isArray(content.content)) {
|
||||||
for (let partIdx = 0; partIdx < content.content.length; partIdx++) {
|
for (let partIdx = 0; partIdx < content.content.length; partIdx++) {
|
||||||
const part = content.content[partIdx];
|
const part = content.content[partIdx];
|
||||||
|
if (isInternalCursorPart(part)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (part?.type === 'text' && part?.text) {
|
if (part?.type === 'text' && part?.text) {
|
||||||
|
const normalizedPartText = unwrapUserQueryText(part.text, role);
|
||||||
|
if (!normalizedPartText) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
messages.push(createNormalizedMessage({
|
messages.push(createNormalizedMessage({
|
||||||
id: `${baseId}_${partIdx}`,
|
id: `${baseId}_${partIdx}`,
|
||||||
sessionId,
|
sessionId,
|
||||||
@@ -345,7 +535,7 @@ export class CursorSessionsProvider implements IProviderSessions {
|
|||||||
provider: PROVIDER,
|
provider: PROVIDER,
|
||||||
kind: 'text',
|
kind: 'text',
|
||||||
role,
|
role,
|
||||||
content: part.text,
|
content: normalizedPartText,
|
||||||
sequence: blob.sequence,
|
sequence: blob.sequence,
|
||||||
rowid: blob.rowid,
|
rowid: blob.rowid,
|
||||||
}));
|
}));
|
||||||
@@ -361,7 +551,11 @@ export class CursorSessionsProvider implements IProviderSessions {
|
|||||||
} else if (part?.type === 'tool-call' || part?.type === 'tool_use') {
|
} else if (part?.type === 'tool-call' || part?.type === 'tool_use') {
|
||||||
const rawToolName = part.toolName || part.name || 'Unknown Tool';
|
const rawToolName = part.toolName || part.name || 'Unknown Tool';
|
||||||
const toolName = rawToolName === 'ApplyPatch' ? 'Edit' : rawToolName;
|
const toolName = rawToolName === 'ApplyPatch' ? 'Edit' : rawToolName;
|
||||||
const toolId = part.toolCallId || part.id || `tool_${i}_${partIdx}`;
|
const toolId = normalizeToolId(part.toolCallId)
|
||||||
|
|| normalizeToolId(part.tool_call_id)
|
||||||
|
|| normalizeToolId(part.id)
|
||||||
|
|| `tool_${i}_${partIdx}`;
|
||||||
|
const normalizedToolInput = normalizeCursorToolInput(rawToolName, part.args ?? part.input);
|
||||||
const message = createNormalizedMessage({
|
const message = createNormalizedMessage({
|
||||||
id: `${baseId}_${partIdx}`,
|
id: `${baseId}_${partIdx}`,
|
||||||
sessionId,
|
sessionId,
|
||||||
@@ -369,14 +563,22 @@ export class CursorSessionsProvider implements IProviderSessions {
|
|||||||
provider: PROVIDER,
|
provider: PROVIDER,
|
||||||
kind: 'tool_use',
|
kind: 'tool_use',
|
||||||
toolName,
|
toolName,
|
||||||
toolInput: part.args || part.input,
|
toolInput: normalizedToolInput,
|
||||||
toolId,
|
toolId,
|
||||||
});
|
});
|
||||||
messages.push(message);
|
messages.push(message);
|
||||||
toolUseMap.set(toolId, message);
|
toolUseMap.set(toolId, message);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (typeof content.content === 'string' && content.content.trim()) {
|
} else if (
|
||||||
|
typeof content.content === 'string'
|
||||||
|
&& content.content.trim()
|
||||||
|
&& !isInternalCursorText(content.content)
|
||||||
|
) {
|
||||||
|
const normalizedText = unwrapUserQueryText(content.content, role);
|
||||||
|
if (!normalizedText) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
messages.push(createNormalizedMessage({
|
messages.push(createNormalizedMessage({
|
||||||
id: baseId,
|
id: baseId,
|
||||||
sessionId,
|
sessionId,
|
||||||
@@ -384,7 +586,7 @@ export class CursorSessionsProvider implements IProviderSessions {
|
|||||||
provider: PROVIDER,
|
provider: PROVIDER,
|
||||||
kind: 'text',
|
kind: 'text',
|
||||||
role,
|
role,
|
||||||
content: content.content,
|
content: normalizedText,
|
||||||
sequence: blob.sequence,
|
sequence: blob.sequence,
|
||||||
rowid: blob.rowid,
|
rowid: blob.rowid,
|
||||||
}));
|
}));
|
||||||
@@ -401,6 +603,7 @@ export class CursorSessionsProvider implements IProviderSessions {
|
|||||||
toolUse.toolResult = {
|
toolUse.toolResult = {
|
||||||
content: msg.content,
|
content: msg.content,
|
||||||
isError: msg.isError,
|
isError: msg.isError,
|
||||||
|
toolUseResult: msg.toolUseResult,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,7 +15,24 @@ type GeminiCredentialsStatus = {
|
|||||||
error?: string;
|
error?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
type GeminiAuthType =
|
||||||
|
| 'oauth-personal'
|
||||||
|
| 'gemini-api-key'
|
||||||
|
| 'vertex-ai'
|
||||||
|
| 'compute-default-credentials'
|
||||||
|
| 'gateway'
|
||||||
|
| 'cloud-shell'
|
||||||
|
| null;
|
||||||
|
|
||||||
export class GeminiProviderAuth implements IProviderAuth {
|
export class GeminiProviderAuth implements IProviderAuth {
|
||||||
|
/**
|
||||||
|
* Gemini CLI can override its home root via GEMINI_CLI_HOME.
|
||||||
|
* Use the same resolution so status checks match runtime behavior.
|
||||||
|
*/
|
||||||
|
private getGeminiCliHome(): string {
|
||||||
|
return process.env.GEMINI_CLI_HOME?.trim() || os.homedir();
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks whether the Gemini CLI is available on this host.
|
* Checks whether the Gemini CLI is available on this host.
|
||||||
*/
|
*/
|
||||||
@@ -58,6 +75,88 @@ export class GeminiProviderAuth implements IProviderAuth {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parses dotenv-style key/value pairs.
|
||||||
|
*/
|
||||||
|
private parseEnvFile(content: string): Record<string, string> {
|
||||||
|
const parsed: Record<string, string> = {};
|
||||||
|
|
||||||
|
for (const rawLine of content.split(/\r?\n/)) {
|
||||||
|
const line = rawLine.trim();
|
||||||
|
if (!line || line.startsWith('#')) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const normalizedLine = line.startsWith('export ')
|
||||||
|
? line.slice('export '.length).trim()
|
||||||
|
: line;
|
||||||
|
const separatorIndex = normalizedLine.indexOf('=');
|
||||||
|
if (separatorIndex <= 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const key = normalizedLine.slice(0, separatorIndex).trim();
|
||||||
|
if (!key) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
let value = normalizedLine.slice(separatorIndex + 1).trim();
|
||||||
|
const quoted = (value.startsWith('"') && value.endsWith('"')) || (value.startsWith('\'') && value.endsWith('\''));
|
||||||
|
if (quoted) {
|
||||||
|
value = value.slice(1, -1);
|
||||||
|
} else {
|
||||||
|
value = value.replace(/\s+#.*$/, '').trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
parsed[key] = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
return parsed;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads user-level auth env in Gemini's "first file found" order.
|
||||||
|
*/
|
||||||
|
private async loadUserLevelAuthEnv(): Promise<Record<string, string>> {
|
||||||
|
const geminiCliHome = this.getGeminiCliHome();
|
||||||
|
const envCandidates = [
|
||||||
|
path.join(geminiCliHome, '.gemini', '.env'),
|
||||||
|
path.join(geminiCliHome, '.env'),
|
||||||
|
];
|
||||||
|
|
||||||
|
for (const envPath of envCandidates) {
|
||||||
|
try {
|
||||||
|
const content = await readFile(envPath, 'utf8');
|
||||||
|
return this.parseEnvFile(content);
|
||||||
|
} catch {
|
||||||
|
// Continue to the next fallback.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads Gemini's selected auth type from settings.json when available.
|
||||||
|
*/
|
||||||
|
private async readSelectedAuthType(): Promise<GeminiAuthType> {
|
||||||
|
try {
|
||||||
|
const settingsPath = path.join(this.getGeminiCliHome(), '.gemini', 'settings.json');
|
||||||
|
const content = await readFile(settingsPath, 'utf8');
|
||||||
|
const settings = readObjectRecord(JSON.parse(content));
|
||||||
|
const security = readObjectRecord(settings?.security);
|
||||||
|
const auth = readObjectRecord(security?.auth);
|
||||||
|
const selectedType = readOptionalString(auth?.selectedType);
|
||||||
|
if (!selectedType) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return selectedType as GeminiAuthType;
|
||||||
|
} catch {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks Gemini credentials from API key env vars or local OAuth credential files.
|
* Checks Gemini credentials from API key env vars or local OAuth credential files.
|
||||||
*/
|
*/
|
||||||
@@ -66,8 +165,46 @@ export class GeminiProviderAuth implements IProviderAuth {
|
|||||||
return { authenticated: true, email: 'API Key Auth', method: 'api_key' };
|
return { authenticated: true, email: 'API Key Auth', method: 'api_key' };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const userEnv = await this.loadUserLevelAuthEnv();
|
||||||
|
if (readOptionalString(userEnv.GEMINI_API_KEY)) {
|
||||||
|
return { authenticated: true, email: 'API Key Auth', method: 'api_key' };
|
||||||
|
}
|
||||||
|
|
||||||
|
const selectedType = await this.readSelectedAuthType();
|
||||||
|
if (selectedType === 'vertex-ai') {
|
||||||
|
const hasGoogleApiKey = Boolean(
|
||||||
|
process.env.GOOGLE_API_KEY?.trim()
|
||||||
|
|| readOptionalString(userEnv.GOOGLE_API_KEY)
|
||||||
|
);
|
||||||
|
const hasProject = Boolean(
|
||||||
|
process.env.GOOGLE_CLOUD_PROJECT?.trim()
|
||||||
|
|| process.env.GOOGLE_CLOUD_PROJECT_ID?.trim()
|
||||||
|
|| readOptionalString(userEnv.GOOGLE_CLOUD_PROJECT)
|
||||||
|
|| readOptionalString(userEnv.GOOGLE_CLOUD_PROJECT_ID)
|
||||||
|
);
|
||||||
|
const hasLocation = Boolean(
|
||||||
|
process.env.GOOGLE_CLOUD_LOCATION?.trim()
|
||||||
|
|| readOptionalString(userEnv.GOOGLE_CLOUD_LOCATION)
|
||||||
|
);
|
||||||
|
const hasServiceAccount = Boolean(
|
||||||
|
process.env.GOOGLE_APPLICATION_CREDENTIALS?.trim()
|
||||||
|
|| readOptionalString(userEnv.GOOGLE_APPLICATION_CREDENTIALS)
|
||||||
|
);
|
||||||
|
|
||||||
|
if (hasGoogleApiKey || hasServiceAccount || (hasProject && hasLocation)) {
|
||||||
|
return { authenticated: true, email: 'Vertex AI Auth', method: 'vertex_ai' };
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
authenticated: false,
|
||||||
|
email: null,
|
||||||
|
method: 'vertex_ai',
|
||||||
|
error: 'Gemini is set to Vertex AI, but required env vars are missing',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const credsPath = path.join(os.homedir(), '.gemini', 'oauth_creds.json');
|
const credsPath = path.join(this.getGeminiCliHome(), '.gemini', 'oauth_creds.json');
|
||||||
const content = await readFile(credsPath, 'utf8');
|
const content = await readFile(credsPath, 'utf8');
|
||||||
const creds = readObjectRecord(JSON.parse(content)) ?? {};
|
const creds = readObjectRecord(JSON.parse(content)) ?? {};
|
||||||
const accessToken = readOptionalString(creds.access_token);
|
const accessToken = readOptionalString(creds.access_token);
|
||||||
@@ -106,6 +243,25 @@ export class GeminiProviderAuth implements IProviderAuth {
|
|||||||
method: 'credentials_file',
|
method: 'credentials_file',
|
||||||
};
|
};
|
||||||
} catch {
|
} catch {
|
||||||
|
if (selectedType === 'gemini-api-key') {
|
||||||
|
return {
|
||||||
|
authenticated: false,
|
||||||
|
email: null,
|
||||||
|
method: 'api_key',
|
||||||
|
error: 'Gemini is set to "Use Gemini API key", but GEMINI_API_KEY is unavailable',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (selectedType === 'oauth-personal') {
|
||||||
|
return {
|
||||||
|
authenticated: false,
|
||||||
|
email: null,
|
||||||
|
method: 'credentials_file',
|
||||||
|
error: 'Gemini is set to Google sign-in, but no cached OAuth credentials were found',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// If no explicit auth type was selected, surface the generic "not configured" error.
|
||||||
return {
|
return {
|
||||||
authenticated: false,
|
authenticated: false,
|
||||||
email: null,
|
email: null,
|
||||||
@@ -140,7 +296,7 @@ export class GeminiProviderAuth implements IProviderAuth {
|
|||||||
*/
|
*/
|
||||||
private async getActiveAccountEmail(): Promise<string | null> {
|
private async getActiveAccountEmail(): Promise<string | null> {
|
||||||
try {
|
try {
|
||||||
const accPath = path.join(os.homedir(), '.gemini', 'google_accounts.json');
|
const accPath = path.join(this.getGeminiCliHome(), '.gemini', 'google_accounts.json');
|
||||||
const accContent = await readFile(accPath, 'utf8');
|
const accContent = await readFile(accPath, 'utf8');
|
||||||
const accounts = readObjectRecord(JSON.parse(accContent));
|
const accounts = readObjectRecord(JSON.parse(accContent));
|
||||||
return readOptionalString(accounts?.active) ?? null;
|
return readOptionalString(accounts?.active) ?? null;
|
||||||
|
|||||||
@@ -39,33 +39,37 @@ export class GeminiSessionSynchronizer implements IProviderSessionSynchronizer {
|
|||||||
async synchronize(since?: Date): Promise<number> {
|
async synchronize(since?: Date): Promise<number> {
|
||||||
const projectHashLookup = this.buildProjectHashLookup();
|
const projectHashLookup = this.buildProjectHashLookup();
|
||||||
|
|
||||||
const legacySessionFiles = await findFilesRecursivelyCreatedAfter(
|
// const legacySessionFiles = await findFilesRecursivelyCreatedAfter(
|
||||||
path.join(this.geminiHome, 'sessions'),
|
// path.join(this.geminiHome, 'sessions'),
|
||||||
'.json',
|
// '.json',
|
||||||
since ?? null
|
// since ?? null
|
||||||
);
|
// );
|
||||||
const legacyTempFiles = await findFilesRecursivelyCreatedAfter(
|
// Gemini creates overlapping artifacts across `sessions/` and `tmp/`.
|
||||||
path.join(this.geminiHome, 'tmp'),
|
// We currently index only `tmp/*/chats/*.jsonl` because those files are the
|
||||||
'.json',
|
// live transcript source and avoid duplicate session rows from mirrored files.
|
||||||
since ?? null
|
// const legacyTempFiles = await findFilesRecursivelyCreatedAfter(
|
||||||
);
|
// path.join(this.geminiHome, 'tmp'),
|
||||||
const jsonlSessionFiles = await findFilesRecursivelyCreatedAfter(
|
// '.json',
|
||||||
path.join(this.geminiHome, 'sessions'),
|
// since ?? null
|
||||||
'.jsonl',
|
// );
|
||||||
since ?? null
|
// const jsonlSessionFiles = await findFilesRecursivelyCreatedAfter(
|
||||||
);
|
// path.join(this.geminiHome, 'sessions'),
|
||||||
|
// '.jsonl',
|
||||||
|
// since ?? null
|
||||||
|
// );
|
||||||
const jsonlTempFiles = await findFilesRecursivelyCreatedAfter(
|
const jsonlTempFiles = await findFilesRecursivelyCreatedAfter(
|
||||||
path.join(this.geminiHome, 'tmp'),
|
path.join(this.geminiHome, 'tmp'),
|
||||||
'.jsonl',
|
'.jsonl',
|
||||||
since ?? null
|
since ?? null
|
||||||
);
|
);
|
||||||
|
|
||||||
// Process legacy JSON first, then JSONL. If both exist for a session id,
|
// Current strategy: index only temp chat JSONL artifacts.
|
||||||
// the JSONL artifact becomes the canonical jsonl_path via upsert.
|
|
||||||
const files = [
|
const files = [
|
||||||
...legacySessionFiles,
|
// ...legacySessionFiles,
|
||||||
...legacyTempFiles,
|
// Intentionally disabled to avoid duplicate indexing from mirrored
|
||||||
...jsonlSessionFiles,
|
// `sessions/*.json` and `sessions/*.jsonl` artifacts.
|
||||||
|
// ...legacyTempFiles,
|
||||||
|
// ...jsonlSessionFiles,
|
||||||
...jsonlTempFiles,
|
...jsonlTempFiles,
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|||||||
@@ -528,10 +528,16 @@ export class GeminiSessionsProvider implements IProviderSessions {
|
|||||||
const messages = pageLimit === null
|
const messages = pageLimit === null
|
||||||
? normalized.slice(start)
|
? normalized.slice(start)
|
||||||
: normalized.slice(start, start + pageLimit);
|
: normalized.slice(start, start + pageLimit);
|
||||||
|
let total = 0;
|
||||||
|
for (const msg of normalized) {
|
||||||
|
if (msg.kind !== 'tool_result') {
|
||||||
|
total += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
messages,
|
messages,
|
||||||
total: normalized.length,
|
total,
|
||||||
hasMore: pageLimit === null ? false : start + pageLimit < normalized.length,
|
hasMore: pageLimit === null ? false : start + pageLimit < normalized.length,
|
||||||
offset: start,
|
offset: start,
|
||||||
limit: pageLimit,
|
limit: pageLimit,
|
||||||
|
|||||||
@@ -311,12 +311,33 @@ router.post(
|
|||||||
);
|
);
|
||||||
|
|
||||||
// ----------------- Session routes -----------------
|
// ----------------- Session routes -----------------
|
||||||
|
router.get(
|
||||||
|
'/sessions/archived',
|
||||||
|
asyncHandler(async (_req: Request, res: Response) => {
|
||||||
|
const sessions = sessionsService.listArchivedSessions();
|
||||||
|
res.json(createApiSuccessResponse({ sessions }));
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
router.delete(
|
router.delete(
|
||||||
'/sessions/:sessionId',
|
'/sessions/:sessionId',
|
||||||
asyncHandler(async (req: Request, res: Response) => {
|
asyncHandler(async (req: Request, res: Response) => {
|
||||||
const sessionId = parseSessionId(req.params.sessionId);
|
const sessionId = parseSessionId(req.params.sessionId);
|
||||||
const deletedFromDisk = parseOptionalBooleanQuery(req.query.deletedFromDisk, 'deletedFromDisk') ?? false;
|
const force = parseOptionalBooleanQuery(req.query.force, 'force') ?? false;
|
||||||
const result = await sessionsService.deleteSessionById(sessionId, deletedFromDisk);
|
const deletedFromDisk = parseOptionalBooleanQuery(req.query.deletedFromDisk, 'deletedFromDisk') ?? force;
|
||||||
|
const result = await sessionsService.deleteOrArchiveSessionById(sessionId, {
|
||||||
|
force,
|
||||||
|
deletedFromDisk,
|
||||||
|
});
|
||||||
|
res.json(createApiSuccessResponse(result));
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
router.post(
|
||||||
|
'/sessions/:sessionId/restore',
|
||||||
|
asyncHandler(async (req: Request, res: Response) => {
|
||||||
|
const sessionId = parseSessionId(req.params.sessionId);
|
||||||
|
const result = sessionsService.restoreSessionById(sessionId);
|
||||||
res.json(createApiSuccessResponse(result));
|
res.json(createApiSuccessResponse(result));
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -89,13 +89,8 @@ const RIPGREP_CHUNK_CONCURRENCY = 6;
|
|||||||
const UNKNOWN_PROJECT_KEY = '__unknown_project__';
|
const UNKNOWN_PROJECT_KEY = '__unknown_project__';
|
||||||
|
|
||||||
const INTERNAL_CONTENT_PREFIXES = [
|
const INTERNAL_CONTENT_PREFIXES = [
|
||||||
'<command-name>',
|
|
||||||
'<command-message>',
|
|
||||||
'<command-args>',
|
|
||||||
'<local-command-stdout>',
|
|
||||||
'<system-reminder>',
|
'<system-reminder>',
|
||||||
'Caveat:',
|
'Caveat:',
|
||||||
'This session is being continued from a previous',
|
|
||||||
'Invalid API key',
|
'Invalid API key',
|
||||||
'[Request interrupted',
|
'[Request interrupted',
|
||||||
] as const;
|
] as const;
|
||||||
@@ -302,6 +297,135 @@ function extractClaudeText(content: unknown): string {
|
|||||||
.join(' ');
|
.join(' ');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function extractTaggedContent(content: string, tagName: string): string | null {
|
||||||
|
const escapedTagName = tagName.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
|
||||||
|
const match = new RegExp(`<${escapedTagName}>([\\s\\S]*?)<\\/${escapedTagName}>`).exec(content);
|
||||||
|
return match ? match[1] : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
type ClaudeLocalCommandPayload = {
|
||||||
|
commandName: string;
|
||||||
|
commandMessage: string;
|
||||||
|
commandArgs: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
function parseClaudeLocalCommandPayload(content: string): ClaudeLocalCommandPayload | null {
|
||||||
|
const commandName = extractTaggedContent(content, 'command-name');
|
||||||
|
const commandMessage = extractTaggedContent(content, 'command-message');
|
||||||
|
const commandArgs = extractTaggedContent(content, 'command-args');
|
||||||
|
|
||||||
|
if (commandName === null && commandMessage === null && commandArgs === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
commandName: commandName ?? '',
|
||||||
|
commandMessage: commandMessage ?? '',
|
||||||
|
commandArgs: commandArgs ?? '',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildClaudeLocalCommandDisplayText(payload: ClaudeLocalCommandPayload): string {
|
||||||
|
const commandName = payload.commandName.trim();
|
||||||
|
const commandMessage = payload.commandMessage.trim();
|
||||||
|
const commandArgs = payload.commandArgs.trim();
|
||||||
|
const baseCommand = commandName || commandMessage;
|
||||||
|
|
||||||
|
if (!baseCommand) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return commandArgs ? `${baseCommand} ${commandArgs}` : baseCommand;
|
||||||
|
}
|
||||||
|
|
||||||
|
function stripAnsiFormatting(text: string): string {
|
||||||
|
return text.replace(/\u001B\[[0-9;?]*[ -/]*[@-~]/g, '');
|
||||||
|
}
|
||||||
|
|
||||||
|
type ClaudeSearchableMessage = {
|
||||||
|
text: string;
|
||||||
|
role: 'user' | 'assistant';
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Claude mixes visible chat, compact summaries, and local command wrappers into
|
||||||
|
* the same transcript stream. Search should operate on the user-visible meaning
|
||||||
|
* of those rows rather than the raw wrapper syntax.
|
||||||
|
*/
|
||||||
|
function extractClaudeSearchableMessage(entry: AnyRecord): ClaudeSearchableMessage | null {
|
||||||
|
if (!entry.message?.content || entry.isApiErrorMessage) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const rawRole = entry.message.role;
|
||||||
|
if (rawRole !== 'user' && rawRole !== 'assistant') {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof entry.message.content === 'string') {
|
||||||
|
const content = String(entry.message.content);
|
||||||
|
|
||||||
|
if (entry.isCompactSummary === true && content.trim()) {
|
||||||
|
return {
|
||||||
|
text: content,
|
||||||
|
role: 'assistant',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const localCommand = parseClaudeLocalCommandPayload(content);
|
||||||
|
if (localCommand) {
|
||||||
|
const displayText = buildClaudeLocalCommandDisplayText(localCommand);
|
||||||
|
return displayText
|
||||||
|
? {
|
||||||
|
text: displayText,
|
||||||
|
role: 'user',
|
||||||
|
}
|
||||||
|
: null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const localCommandStdout = extractTaggedContent(content, 'local-command-stdout');
|
||||||
|
if (localCommandStdout !== null) {
|
||||||
|
const stdoutText = stripAnsiFormatting(localCommandStdout).trim();
|
||||||
|
return stdoutText
|
||||||
|
? {
|
||||||
|
text: stdoutText,
|
||||||
|
role: 'assistant',
|
||||||
|
}
|
||||||
|
: null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!content || isInternalContent(content)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
text: content,
|
||||||
|
role: rawRole,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const text = extractClaudeText(entry.message.content);
|
||||||
|
if (!text) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (entry.isCompactSummary === true) {
|
||||||
|
return {
|
||||||
|
text,
|
||||||
|
role: 'assistant',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isInternalContent(text)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
text,
|
||||||
|
role: rawRole,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
function extractCodexText(content: unknown): string {
|
function extractCodexText(content: unknown): string {
|
||||||
if (typeof content === 'string') {
|
if (typeof content === 'string') {
|
||||||
return content;
|
return content;
|
||||||
@@ -348,6 +472,7 @@ function extractGeminiText(content: unknown): string {
|
|||||||
|
|
||||||
function normalizeSearchableSessions(rows: SessionRepositoryRow[]): SearchableSessionRow[] {
|
function normalizeSearchableSessions(rows: SessionRepositoryRow[]): SearchableSessionRow[] {
|
||||||
const normalizedRows: SearchableSessionRow[] = [];
|
const normalizedRows: SearchableSessionRow[] = [];
|
||||||
|
const projectArchiveStateByPath = new Map<string, boolean>();
|
||||||
|
|
||||||
for (const row of rows) {
|
for (const row of rows) {
|
||||||
const provider = row.provider as SearchableProvider;
|
const provider = row.provider as SearchableProvider;
|
||||||
@@ -365,6 +490,27 @@ function normalizeSearchableSessions(rows: SessionRepositoryRow[]): SearchableSe
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Active session rows can still belong to an archived project because
|
||||||
|
* project archiving intentionally preserves the underlying session data.
|
||||||
|
* Global conversation search should follow the visible workspace model,
|
||||||
|
* which means excluding any session whose owning project is archived.
|
||||||
|
*
|
||||||
|
* Cache the archive lookup per normalized project path so one search pass
|
||||||
|
* does not re-query the same project row for every session in that folder.
|
||||||
|
*/
|
||||||
|
const normalizedProjectPath = typeof row.project_path === 'string' ? row.project_path.trim() : '';
|
||||||
|
if (normalizedProjectPath) {
|
||||||
|
if (!projectArchiveStateByPath.has(normalizedProjectPath)) {
|
||||||
|
const projectRow = projectsDb.getProjectPath(normalizedProjectPath);
|
||||||
|
projectArchiveStateByPath.set(normalizedProjectPath, Boolean(projectRow?.isArchived));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (projectArchiveStateByPath.get(normalizedProjectPath) === true) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
normalizedRows.push({
|
normalizedRows.push({
|
||||||
...row,
|
...row,
|
||||||
provider,
|
provider,
|
||||||
@@ -733,18 +879,21 @@ async function parseClaudeSessionMatches(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!entry.message?.content || entry.isApiErrorMessage) {
|
const searchableMessage = extractClaudeSearchableMessage(entry);
|
||||||
|
if (!searchableMessage) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
const role = entry.message.role;
|
const { text, role } = searchableMessage;
|
||||||
if (role !== 'user' && role !== 'assistant') {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
const text = extractClaudeText(entry.message.content);
|
/**
|
||||||
if (!text || isInternalContent(text)) {
|
* Claude compact summaries are the most faithful session-summary source
|
||||||
continue;
|
* after a `/compact` because they describe the post-compaction state that
|
||||||
|
* the resumed session actually continues from. Prefer them over generic
|
||||||
|
* fallback user text when present.
|
||||||
|
*/
|
||||||
|
if (entry.isCompactSummary === true) {
|
||||||
|
state.resolvedSummary = text;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (role === 'user') {
|
if (role === 'user') {
|
||||||
|
|||||||
@@ -18,16 +18,18 @@ const PROVIDER_WATCH_PATHS: Array<{ provider: LLMProvider; rootPath: string }> =
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
provider: 'cursor',
|
provider: 'cursor',
|
||||||
rootPath: path.join(os.homedir(), '.cursor', 'chats'),
|
rootPath: path.join(os.homedir(), '.cursor', 'projects'),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
provider: 'codex',
|
provider: 'codex',
|
||||||
rootPath: path.join(os.homedir(), '.codex', 'sessions'),
|
rootPath: path.join(os.homedir(), '.codex', 'sessions'),
|
||||||
},
|
},
|
||||||
{
|
// {
|
||||||
provider: 'gemini',
|
// provider: 'gemini',
|
||||||
rootPath: path.join(os.homedir(), '.gemini', 'sessions'),
|
// rootPath: path.join(os.homedir(), '.gemini', 'sessions'),
|
||||||
},
|
// },
|
||||||
|
// Keep `sessions/` watcher disabled: Gemini also mirrors artifacts there,
|
||||||
|
// which causes duplicate synchronization events.
|
||||||
{
|
{
|
||||||
provider: 'gemini',
|
provider: 'gemini',
|
||||||
rootPath: path.join(os.homedir(), '.gemini', 'tmp'),
|
rootPath: path.join(os.homedir(), '.gemini', 'tmp'),
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import fsp from 'node:fs/promises';
|
import fsp from 'node:fs/promises';
|
||||||
|
import path from 'node:path';
|
||||||
|
|
||||||
import { sessionsDb } from '@/modules/database/index.js';
|
import { projectsDb, sessionsDb } from '@/modules/database/index.js';
|
||||||
import { providerRegistry } from '@/modules/providers/provider.registry.js';
|
import { providerRegistry } from '@/modules/providers/provider.registry.js';
|
||||||
import type {
|
import type {
|
||||||
FetchHistoryOptions,
|
FetchHistoryOptions,
|
||||||
@@ -10,6 +11,19 @@ import type {
|
|||||||
} from '@/shared/types.js';
|
} from '@/shared/types.js';
|
||||||
import { AppError } from '@/shared/utils.js';
|
import { AppError } from '@/shared/utils.js';
|
||||||
|
|
||||||
|
type ArchivedSessionListItem = {
|
||||||
|
sessionId: string;
|
||||||
|
provider: LLMProvider;
|
||||||
|
projectId: string | null;
|
||||||
|
projectPath: string | null;
|
||||||
|
projectDisplayName: string;
|
||||||
|
sessionTitle: string;
|
||||||
|
createdAt: string | null;
|
||||||
|
updatedAt: string | null;
|
||||||
|
lastActivity: string | null;
|
||||||
|
isProjectArchived: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Removes one file if it exists.
|
* Removes one file if it exists.
|
||||||
*/
|
*/
|
||||||
@@ -26,6 +40,28 @@ async function removeFileIfExists(filePath: string): Promise<boolean> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Archive rows need a stable project label even when the owning project is not
|
||||||
|
* part of the active sidebar payload. This lightweight resolver keeps the
|
||||||
|
* archive API self-contained while still matching the project's stored display
|
||||||
|
* name when one exists.
|
||||||
|
*/
|
||||||
|
function resolveProjectDisplayName(
|
||||||
|
projectPath: string | null,
|
||||||
|
customProjectName: string | null | undefined,
|
||||||
|
): string {
|
||||||
|
const trimmedCustomName = typeof customProjectName === 'string' ? customProjectName.trim() : '';
|
||||||
|
if (trimmedCustomName.length > 0) {
|
||||||
|
return trimmedCustomName;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!projectPath) {
|
||||||
|
return 'Unknown Project';
|
||||||
|
}
|
||||||
|
|
||||||
|
return path.basename(projectPath) || projectPath;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Application service for provider-backed session message operations.
|
* Application service for provider-backed session message operations.
|
||||||
*
|
*
|
||||||
@@ -79,15 +115,53 @@ export const sessionsService = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Deletes one persisted session row by id.
|
* Returns archived sessions with enough project metadata for the sidebar to
|
||||||
*
|
* group, filter, open, and restore them without a per-row follow-up query.
|
||||||
* When `deletedFromDisk` is true and a session `jsonl_path` exists, the path
|
|
||||||
* is deleted from disk before the DB row is removed.
|
|
||||||
*/
|
*/
|
||||||
async deleteSessionById(
|
listArchivedSessions(): ArchivedSessionListItem[] {
|
||||||
|
const archivedSessions = sessionsDb.getArchivedSessions();
|
||||||
|
const projectCache = new Map<string, ReturnType<typeof projectsDb.getProjectPath>>();
|
||||||
|
|
||||||
|
return archivedSessions.map((session) => {
|
||||||
|
const projectPath = session.project_path?.trim() ? session.project_path : null;
|
||||||
|
let project = null;
|
||||||
|
|
||||||
|
if (projectPath) {
|
||||||
|
if (!projectCache.has(projectPath)) {
|
||||||
|
projectCache.set(projectPath, projectsDb.getProjectPath(projectPath));
|
||||||
|
}
|
||||||
|
project = projectCache.get(projectPath) ?? null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
sessionId: session.session_id,
|
||||||
|
provider: session.provider as LLMProvider,
|
||||||
|
projectId: project?.project_id ?? null,
|
||||||
|
projectPath,
|
||||||
|
projectDisplayName: resolveProjectDisplayName(projectPath, project?.custom_project_name),
|
||||||
|
sessionTitle: session.custom_name?.trim() || session.session_id,
|
||||||
|
createdAt: session.created_at ?? null,
|
||||||
|
updatedAt: session.updated_at ?? null,
|
||||||
|
lastActivity: session.updated_at ?? session.created_at ?? null,
|
||||||
|
isProjectArchived: Boolean(project?.isArchived),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Archives or permanently deletes one persisted session row by id.
|
||||||
|
*
|
||||||
|
* Soft-delete mirrors the project behavior by toggling `isArchived` so the
|
||||||
|
* row disappears from active lists but remains restorable. Force-delete
|
||||||
|
* optionally removes the transcript file before deleting the database row.
|
||||||
|
*/
|
||||||
|
async deleteOrArchiveSessionById(
|
||||||
sessionId: string,
|
sessionId: string,
|
||||||
deletedFromDisk = false,
|
options: {
|
||||||
): Promise<{ sessionId: string; deletedFromDisk: boolean }> {
|
force?: boolean;
|
||||||
|
deletedFromDisk?: boolean;
|
||||||
|
} = {},
|
||||||
|
): Promise<{ sessionId: string; action: 'archived' | 'deleted'; deletedFromDisk: boolean }> {
|
||||||
const session = sessionsDb.getSessionById(sessionId);
|
const session = sessionsDb.getSessionById(sessionId);
|
||||||
if (!session) {
|
if (!session) {
|
||||||
throw new AppError(`Session "${sessionId}" was not found.`, {
|
throw new AppError(`Session "${sessionId}" was not found.`, {
|
||||||
@@ -96,8 +170,17 @@ export const sessionsService = {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!options.force) {
|
||||||
|
sessionsDb.updateSessionIsArchived(sessionId, true);
|
||||||
|
return {
|
||||||
|
sessionId,
|
||||||
|
action: 'archived',
|
||||||
|
deletedFromDisk: false,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
let removedFromDisk = false;
|
let removedFromDisk = false;
|
||||||
if (deletedFromDisk && session.jsonl_path) {
|
if (options.deletedFromDisk && session.jsonl_path) {
|
||||||
removedFromDisk = await removeFileIfExists(session.jsonl_path);
|
removedFromDisk = await removeFileIfExists(session.jsonl_path);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -109,7 +192,27 @@ export const sessionsService = {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
return { sessionId, deletedFromDisk: removedFromDisk };
|
return {
|
||||||
|
sessionId,
|
||||||
|
action: 'deleted',
|
||||||
|
deletedFromDisk: removedFromDisk,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Restores one archived session back into the active sidebar lists.
|
||||||
|
*/
|
||||||
|
restoreSessionById(sessionId: string): { sessionId: string; isArchived: false } {
|
||||||
|
const session = sessionsDb.getSessionById(sessionId);
|
||||||
|
if (!session) {
|
||||||
|
throw new AppError(`Session "${sessionId}" was not found.`, {
|
||||||
|
code: 'SESSION_NOT_FOUND',
|
||||||
|
statusCode: 404,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
sessionsDb.updateSessionIsArchived(sessionId, false);
|
||||||
|
return { sessionId, isArchived: false };
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -143,7 +143,7 @@ function transformCodexEvent(event) {
|
|||||||
case 'thread.started':
|
case 'thread.started':
|
||||||
return {
|
return {
|
||||||
type: 'thread_started',
|
type: 'thread_started',
|
||||||
threadId: event.id
|
threadId: event.thread_id || event.id
|
||||||
};
|
};
|
||||||
|
|
||||||
case 'error':
|
case 'error':
|
||||||
@@ -207,7 +207,8 @@ export async function queryCodex(command, options = {}, ws) {
|
|||||||
|
|
||||||
let codex;
|
let codex;
|
||||||
let thread;
|
let thread;
|
||||||
let currentSessionId = sessionId;
|
let capturedSessionId = sessionId;
|
||||||
|
let sessionCreatedSent = false;
|
||||||
let terminalFailure = null;
|
let terminalFailure = null;
|
||||||
const abortController = new AbortController();
|
const abortController = new AbortController();
|
||||||
|
|
||||||
@@ -231,20 +232,23 @@ export async function queryCodex(command, options = {}, ws) {
|
|||||||
thread = codex.startThread(threadOptions);
|
thread = codex.startThread(threadOptions);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the thread ID
|
const registerSession = (id) => {
|
||||||
currentSessionId = thread.id || sessionId || `codex-${Date.now()}`;
|
if (!id) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
activeCodexSessions.set(id, {
|
||||||
|
thread,
|
||||||
|
codex,
|
||||||
|
status: 'running',
|
||||||
|
abortController,
|
||||||
|
startedAt: new Date().toISOString()
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
// Track the session
|
// Existing sessions can be tracked immediately; new sessions are tracked after thread.started.
|
||||||
activeCodexSessions.set(currentSessionId, {
|
if (capturedSessionId) {
|
||||||
thread,
|
registerSession(capturedSessionId);
|
||||||
codex,
|
}
|
||||||
status: 'running',
|
|
||||||
abortController,
|
|
||||||
startedAt: new Date().toISOString()
|
|
||||||
});
|
|
||||||
|
|
||||||
// Send session created event
|
|
||||||
sendMessage(ws, createNormalizedMessage({ kind: 'session_created', newSessionId: currentSessionId, sessionId: currentSessionId, provider: 'codex' }));
|
|
||||||
|
|
||||||
// Execute with streaming
|
// Execute with streaming
|
||||||
const streamedTurn = await thread.runStreamed(command, {
|
const streamedTurn = await thread.runStreamed(command, {
|
||||||
@@ -252,11 +256,34 @@ export async function queryCodex(command, options = {}, ws) {
|
|||||||
});
|
});
|
||||||
|
|
||||||
for await (const event of streamedTurn.events) {
|
for await (const event of streamedTurn.events) {
|
||||||
|
// Capture thread/session id lazily from the stream (Codex emits this asynchronously).
|
||||||
|
if (event.type === 'thread.started') {
|
||||||
|
const discoveredSessionId = event.thread_id || event.id || null;
|
||||||
|
if (discoveredSessionId && !capturedSessionId) {
|
||||||
|
capturedSessionId = discoveredSessionId;
|
||||||
|
registerSession(capturedSessionId);
|
||||||
|
|
||||||
|
if (ws.setSessionId && typeof ws.setSessionId === 'function') {
|
||||||
|
ws.setSessionId(capturedSessionId);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!sessionId && !sessionCreatedSent) {
|
||||||
|
sessionCreatedSent = true;
|
||||||
|
sendMessage(ws, createNormalizedMessage({ kind: 'session_created', newSessionId: capturedSessionId, sessionId: capturedSessionId, provider: 'codex' }));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Check if session was aborted
|
// Check if session was aborted
|
||||||
const session = activeCodexSessions.get(currentSessionId);
|
if (abortController.signal.aborted) {
|
||||||
if (!session || session.status === 'aborted') {
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
if (capturedSessionId) {
|
||||||
|
const session = activeCodexSessions.get(capturedSessionId);
|
||||||
|
if (session?.status === 'aborted') {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (event.type === 'item.started' || event.type === 'item.updated') {
|
if (event.type === 'item.started' || event.type === 'item.updated') {
|
||||||
continue;
|
continue;
|
||||||
@@ -265,7 +292,7 @@ export async function queryCodex(command, options = {}, ws) {
|
|||||||
const transformed = transformCodexEvent(event);
|
const transformed = transformCodexEvent(event);
|
||||||
|
|
||||||
// Normalize the transformed event into NormalizedMessage(s) via adapter
|
// Normalize the transformed event into NormalizedMessage(s) via adapter
|
||||||
const normalizedMsgs = sessionsService.normalizeMessage('codex', transformed, currentSessionId);
|
const normalizedMsgs = sessionsService.normalizeMessage('codex', transformed, capturedSessionId || sessionId || null);
|
||||||
for (const msg of normalizedMsgs) {
|
for (const msg of normalizedMsgs) {
|
||||||
sendMessage(ws, msg);
|
sendMessage(ws, msg);
|
||||||
}
|
}
|
||||||
@@ -275,7 +302,7 @@ export async function queryCodex(command, options = {}, ws) {
|
|||||||
notifyRunFailed({
|
notifyRunFailed({
|
||||||
userId: ws?.userId || null,
|
userId: ws?.userId || null,
|
||||||
provider: 'codex',
|
provider: 'codex',
|
||||||
sessionId: currentSessionId,
|
sessionId: capturedSessionId || sessionId || null,
|
||||||
sessionName: sessionSummary,
|
sessionName: sessionSummary,
|
||||||
error: terminalFailure
|
error: terminalFailure
|
||||||
});
|
});
|
||||||
@@ -284,24 +311,29 @@ export async function queryCodex(command, options = {}, ws) {
|
|||||||
// Extract and send token usage if available (normalized to match Claude format)
|
// Extract and send token usage if available (normalized to match Claude format)
|
||||||
if (event.type === 'turn.completed' && event.usage) {
|
if (event.type === 'turn.completed' && event.usage) {
|
||||||
const totalTokens = (event.usage.input_tokens || 0) + (event.usage.output_tokens || 0);
|
const totalTokens = (event.usage.input_tokens || 0) + (event.usage.output_tokens || 0);
|
||||||
sendMessage(ws, createNormalizedMessage({ kind: 'status', text: 'token_budget', tokenBudget: { used: totalTokens, total: 200000 }, sessionId: currentSessionId, provider: 'codex' }));
|
sendMessage(ws, createNormalizedMessage({ kind: 'status', text: 'token_budget', tokenBudget: { used: totalTokens, total: 200000 }, sessionId: capturedSessionId || sessionId || null, provider: 'codex' }));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Send completion event
|
// Send completion event
|
||||||
if (!terminalFailure) {
|
if (!terminalFailure) {
|
||||||
sendMessage(ws, createNormalizedMessage({ kind: 'complete', actualSessionId: thread.id, sessionId: currentSessionId, provider: 'codex' }));
|
sendMessage(ws, createNormalizedMessage({
|
||||||
|
kind: 'complete',
|
||||||
|
actualSessionId: capturedSessionId || thread.id || sessionId || null,
|
||||||
|
sessionId: capturedSessionId || sessionId || null,
|
||||||
|
provider: 'codex'
|
||||||
|
}));
|
||||||
notifyRunStopped({
|
notifyRunStopped({
|
||||||
userId: ws?.userId || null,
|
userId: ws?.userId || null,
|
||||||
provider: 'codex',
|
provider: 'codex',
|
||||||
sessionId: currentSessionId,
|
sessionId: capturedSessionId || sessionId || null,
|
||||||
sessionName: sessionSummary,
|
sessionName: sessionSummary,
|
||||||
stopReason: 'completed'
|
stopReason: 'completed'
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const session = currentSessionId ? activeCodexSessions.get(currentSessionId) : null;
|
const session = capturedSessionId ? activeCodexSessions.get(capturedSessionId) : null;
|
||||||
const wasAborted =
|
const wasAborted =
|
||||||
session?.status === 'aborted' ||
|
session?.status === 'aborted' ||
|
||||||
error?.name === 'AbortError' ||
|
error?.name === 'AbortError' ||
|
||||||
@@ -316,12 +348,12 @@ export async function queryCodex(command, options = {}, ws) {
|
|||||||
? 'Codex CLI is not configured. Please set up authentication first.'
|
? 'Codex CLI is not configured. Please set up authentication first.'
|
||||||
: error.message;
|
: error.message;
|
||||||
|
|
||||||
sendMessage(ws, createNormalizedMessage({ kind: 'error', content: errorContent, sessionId: currentSessionId, provider: 'codex' }));
|
sendMessage(ws, createNormalizedMessage({ kind: 'error', content: errorContent, sessionId: capturedSessionId || sessionId || null, provider: 'codex' }));
|
||||||
if (!terminalFailure) {
|
if (!terminalFailure) {
|
||||||
notifyRunFailed({
|
notifyRunFailed({
|
||||||
userId: ws?.userId || null,
|
userId: ws?.userId || null,
|
||||||
provider: 'codex',
|
provider: 'codex',
|
||||||
sessionId: currentSessionId,
|
sessionId: capturedSessionId || sessionId || null,
|
||||||
sessionName: sessionSummary,
|
sessionName: sessionSummary,
|
||||||
error
|
error
|
||||||
});
|
});
|
||||||
@@ -330,8 +362,8 @@ export async function queryCodex(command, options = {}, ws) {
|
|||||||
|
|
||||||
} finally {
|
} finally {
|
||||||
// Update session status
|
// Update session status
|
||||||
if (currentSessionId) {
|
if (capturedSessionId) {
|
||||||
const session = activeCodexSessions.get(currentSessionId);
|
const session = activeCodexSessions.get(capturedSessionId);
|
||||||
if (session) {
|
if (session) {
|
||||||
session.status = session.status === 'aborted' ? 'aborted' : 'completed';
|
session.status = session.status === 'aborted' ? 'aborted' : 'completed';
|
||||||
}
|
}
|
||||||
|
|||||||
61
server/shared/claude-cli-path.test.ts
Normal file
61
server/shared/claude-cli-path.test.ts
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
import assert from 'node:assert/strict';
|
||||||
|
import test from 'node:test';
|
||||||
|
|
||||||
|
import {
|
||||||
|
resolveClaudeCodeExecutablePath,
|
||||||
|
type ResolveClaudeCodeExecutablePathDependencies,
|
||||||
|
} from '@/shared/claude-cli-path.js';
|
||||||
|
|
||||||
|
test('resolveClaudeCodeExecutablePath resolves the npm Claude wrapper to its native exe on Windows', () => {
|
||||||
|
const wrapperDir = 'C:\\nvm4w\\nodejs';
|
||||||
|
const nativePath = `${wrapperDir}\\node_modules\\@anthropic-ai\\claude-code\\bin\\claude.exe`;
|
||||||
|
const execFileSync =
|
||||||
|
(() => `${wrapperDir}\\claude\r\n${wrapperDir}\\claude.cmd\r\n`) as unknown as ResolveClaudeCodeExecutablePathDependencies['execFileSync'];
|
||||||
|
const readFileSync = (() => '') as unknown as ResolveClaudeCodeExecutablePathDependencies['readFileSync'];
|
||||||
|
|
||||||
|
const resolved = resolveClaudeCodeExecutablePath('claude', {
|
||||||
|
platform: 'win32',
|
||||||
|
execFileSync,
|
||||||
|
existsSync: (candidate) => candidate === nativePath,
|
||||||
|
readFileSync,
|
||||||
|
});
|
||||||
|
|
||||||
|
assert.equal(resolved, nativePath);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('resolveClaudeCodeExecutablePath keeps an explicit JavaScript launcher path unchanged', () => {
|
||||||
|
const scriptPath = 'C:\\tools\\claude.js';
|
||||||
|
|
||||||
|
const resolved = resolveClaudeCodeExecutablePath(scriptPath, {
|
||||||
|
platform: 'win32',
|
||||||
|
});
|
||||||
|
|
||||||
|
assert.equal(resolved, scriptPath);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('resolveClaudeCodeExecutablePath can parse a wrapper file path containing letters r and n before claude.exe', () => {
|
||||||
|
const wrapperPath = 'C:\\tools\\claude';
|
||||||
|
const nativePath = 'C:\\tools\\custom\\bin\\node_modules\\@anthropic-ai\\claude-code\\bin\\claude.exe';
|
||||||
|
const readFileSync = (() => `exec "$basedir/custom/bin/node_modules/@anthropic-ai/claude-code/bin/claude.exe" "$@"`) as unknown as ResolveClaudeCodeExecutablePathDependencies['readFileSync'];
|
||||||
|
|
||||||
|
const resolved = resolveClaudeCodeExecutablePath(wrapperPath, {
|
||||||
|
platform: 'win32',
|
||||||
|
existsSync: (candidate) => candidate === nativePath,
|
||||||
|
readFileSync,
|
||||||
|
});
|
||||||
|
|
||||||
|
assert.equal(resolved, nativePath);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('resolveClaudeCodeExecutablePath falls back to the configured command when PATH lookup fails', () => {
|
||||||
|
const execFileSync = (() => {
|
||||||
|
throw new Error('not found');
|
||||||
|
}) as unknown as ResolveClaudeCodeExecutablePathDependencies['execFileSync'];
|
||||||
|
|
||||||
|
const resolved = resolveClaudeCodeExecutablePath('claude', {
|
||||||
|
platform: 'win32',
|
||||||
|
execFileSync,
|
||||||
|
});
|
||||||
|
|
||||||
|
assert.equal(resolved, 'claude');
|
||||||
|
});
|
||||||
139
server/shared/claude-cli-path.ts
Normal file
139
server/shared/claude-cli-path.ts
Normal file
@@ -0,0 +1,139 @@
|
|||||||
|
import { execFileSync } from 'node:child_process';
|
||||||
|
import fs from 'node:fs';
|
||||||
|
import path from 'node:path';
|
||||||
|
|
||||||
|
const DEFAULT_CLAUDE_COMMAND = 'claude';
|
||||||
|
const CLAUDE_SCRIPT_EXTENSIONS = new Set(['.cjs', '.js', '.jsx', '.mjs', '.ts', '.tsx']);
|
||||||
|
const CLAUDE_WRAPPER_SEGMENTS = ['node_modules', '@anthropic-ai', 'claude-code', 'bin', 'claude.exe'] as const;
|
||||||
|
|
||||||
|
export type ResolveClaudeCodeExecutablePathDependencies = {
|
||||||
|
execFileSync?: typeof execFileSync;
|
||||||
|
existsSync?: typeof fs.existsSync;
|
||||||
|
platform?: NodeJS.Platform;
|
||||||
|
readFileSync?: typeof fs.readFileSync;
|
||||||
|
};
|
||||||
|
|
||||||
|
function getPathApi(platform: NodeJS.Platform) {
|
||||||
|
return platform === 'win32' ? path.win32 : path;
|
||||||
|
}
|
||||||
|
|
||||||
|
function stripWrappingQuotes(value: string): string {
|
||||||
|
const trimmed = value.trim();
|
||||||
|
if (
|
||||||
|
(trimmed.startsWith('"') && trimmed.endsWith('"')) ||
|
||||||
|
(trimmed.startsWith("'") && trimmed.endsWith("'"))
|
||||||
|
) {
|
||||||
|
return trimmed.slice(1, -1);
|
||||||
|
}
|
||||||
|
return trimmed;
|
||||||
|
}
|
||||||
|
|
||||||
|
function isPathLike(value: string): boolean {
|
||||||
|
return value.includes('/') || value.includes('\\');
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveClaudeWrapperBinary(
|
||||||
|
wrapperPath: string,
|
||||||
|
deps: Required<ResolveClaudeCodeExecutablePathDependencies>,
|
||||||
|
): string | null {
|
||||||
|
const pathApi = getPathApi(deps.platform);
|
||||||
|
const directCandidate = pathApi.resolve(pathApi.dirname(wrapperPath), ...CLAUDE_WRAPPER_SEGMENTS);
|
||||||
|
|
||||||
|
if (deps.existsSync(directCandidate)) {
|
||||||
|
return directCandidate;
|
||||||
|
}
|
||||||
|
|
||||||
|
let content: string;
|
||||||
|
try {
|
||||||
|
content = deps.readFileSync(wrapperPath, 'utf8');
|
||||||
|
} catch {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const matches = content.matchAll(/["']([^"'\\\r\n]*claude\.exe)["']/gi);
|
||||||
|
for (const match of matches) {
|
||||||
|
const rawTarget = match[1]
|
||||||
|
.replace(/^\$basedir[\\/]/i, '')
|
||||||
|
.replace(/^%dp0%[\\/]/i, '')
|
||||||
|
.replace(/^%~dp0[\\/]/i, '');
|
||||||
|
const normalizedTarget = rawTarget.replace(/[\\/]/g, pathApi.sep);
|
||||||
|
const candidate = pathApi.isAbsolute(normalizedTarget)
|
||||||
|
? normalizedTarget
|
||||||
|
: pathApi.resolve(pathApi.dirname(wrapperPath), normalizedTarget);
|
||||||
|
|
||||||
|
if (deps.existsSync(candidate)) {
|
||||||
|
return candidate;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveWindowsClaudeExecutablePath(
|
||||||
|
configuredPath: string,
|
||||||
|
deps: Required<ResolveClaudeCodeExecutablePathDependencies>,
|
||||||
|
): string {
|
||||||
|
const pathApi = getPathApi(deps.platform);
|
||||||
|
const extension = pathApi.extname(configuredPath).toLowerCase();
|
||||||
|
const explicitPath = isPathLike(configuredPath) || pathApi.isAbsolute(configuredPath);
|
||||||
|
|
||||||
|
if (CLAUDE_SCRIPT_EXTENSIONS.has(extension)) {
|
||||||
|
return configuredPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (explicitPath && extension === '.exe') {
|
||||||
|
return configuredPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (explicitPath) {
|
||||||
|
return resolveClaudeWrapperBinary(configuredPath, deps) ?? configuredPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const stdout = deps.execFileSync('where.exe', [configuredPath], {
|
||||||
|
encoding: 'utf8',
|
||||||
|
stdio: ['ignore', 'pipe', 'ignore'],
|
||||||
|
windowsHide: true,
|
||||||
|
});
|
||||||
|
const candidates = stdout
|
||||||
|
.split(/\r?\n/)
|
||||||
|
.map((entry) => entry.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
|
||||||
|
for (const candidate of candidates) {
|
||||||
|
if (pathApi.extname(candidate).toLowerCase() === '.exe') {
|
||||||
|
return candidate;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const candidate of candidates) {
|
||||||
|
const resolved = resolveClaudeWrapperBinary(candidate, deps);
|
||||||
|
if (resolved) {
|
||||||
|
return resolved;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
return configuredPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
return configuredPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function resolveClaudeCodeExecutablePath(
|
||||||
|
configuredPath: string | undefined = process.env.CLAUDE_CLI_PATH,
|
||||||
|
dependencies: ResolveClaudeCodeExecutablePathDependencies = {},
|
||||||
|
): string {
|
||||||
|
const deps: Required<ResolveClaudeCodeExecutablePathDependencies> = {
|
||||||
|
execFileSync: dependencies.execFileSync ?? execFileSync,
|
||||||
|
existsSync: dependencies.existsSync ?? fs.existsSync,
|
||||||
|
platform: dependencies.platform ?? process.platform,
|
||||||
|
readFileSync: dependencies.readFileSync ?? fs.readFileSync,
|
||||||
|
};
|
||||||
|
|
||||||
|
const normalizedPath = stripWrappingQuotes(configuredPath || DEFAULT_CLAUDE_COMMAND);
|
||||||
|
if (deps.platform !== 'win32') {
|
||||||
|
return normalizedPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
return resolveWindowsClaudeExecutablePath(normalizedPath, deps);
|
||||||
|
}
|
||||||
@@ -102,6 +102,21 @@ export type NormalizedMessage = {
|
|||||||
kind: MessageKind;
|
kind: MessageKind;
|
||||||
role?: 'user' | 'assistant';
|
role?: 'user' | 'assistant';
|
||||||
content?: string;
|
content?: string;
|
||||||
|
/**
|
||||||
|
* Optional display-oriented metadata used by providers that need to expose
|
||||||
|
* richer transcript artifacts without introducing a brand-new message kind.
|
||||||
|
*
|
||||||
|
* Current Claude usage:
|
||||||
|
* - local slash commands expose parsed command fields
|
||||||
|
* - compact summaries are flagged so the UI can treat them differently later
|
||||||
|
*/
|
||||||
|
displayText?: string;
|
||||||
|
commandName?: string;
|
||||||
|
commandMessage?: string;
|
||||||
|
commandArgs?: string;
|
||||||
|
isLocalCommand?: boolean;
|
||||||
|
isLocalCommandStdout?: boolean;
|
||||||
|
isCompactSummary?: boolean;
|
||||||
images?: unknown;
|
images?: unknown;
|
||||||
toolName?: string;
|
toolName?: string;
|
||||||
toolInput?: unknown;
|
toolInput?: unknown;
|
||||||
|
|||||||
@@ -84,6 +84,7 @@ export const GEMINI_MODELS = {
|
|||||||
{ value: "gemini-2.5-flash", label: "Gemini 2.5 Flash" },
|
{ value: "gemini-2.5-flash", label: "Gemini 2.5 Flash" },
|
||||||
{ value: "gemini-2.5-pro", label: "Gemini 2.5 Pro" },
|
{ value: "gemini-2.5-pro", label: "Gemini 2.5 Pro" },
|
||||||
{ value: "gemini-2.0-flash-lite", label: "Gemini 2.0 Flash Lite" },
|
{ value: "gemini-2.0-flash-lite", label: "Gemini 2.0 Flash Lite" },
|
||||||
|
{ value: "gemini-2.5-flash-lite", label: "Gemini 2.5 Flash Lite" },
|
||||||
{ value: "gemini-2.0-flash", label: "Gemini 2.0 Flash" },
|
{ value: "gemini-2.0-flash", label: "Gemini 2.0 Flash" },
|
||||||
{ value: "gemini-2.0-pro-exp", label: "Gemini 2.0 Pro Experimental" },
|
{ value: "gemini-2.0-pro-exp", label: "Gemini 2.0 Pro Experimental" },
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -34,7 +34,6 @@ function AppContentInner() {
|
|||||||
markSessionAsInactive,
|
markSessionAsInactive,
|
||||||
markSessionAsProcessing,
|
markSessionAsProcessing,
|
||||||
markSessionAsNotProcessing,
|
markSessionAsNotProcessing,
|
||||||
replaceTemporarySession,
|
|
||||||
} = useSessionProtection();
|
} = useSessionProtection();
|
||||||
|
|
||||||
const {
|
const {
|
||||||
@@ -44,6 +43,7 @@ function AppContentInner() {
|
|||||||
sidebarOpen,
|
sidebarOpen,
|
||||||
isLoadingProjects,
|
isLoadingProjects,
|
||||||
externalMessageUpdate,
|
externalMessageUpdate,
|
||||||
|
newSessionTrigger,
|
||||||
setActiveTab,
|
setActiveTab,
|
||||||
setSidebarOpen,
|
setSidebarOpen,
|
||||||
setIsInputFocused,
|
setIsInputFocused,
|
||||||
@@ -190,10 +190,12 @@ function AppContentInner() {
|
|||||||
onSessionProcessing={markSessionAsProcessing}
|
onSessionProcessing={markSessionAsProcessing}
|
||||||
onSessionNotProcessing={markSessionAsNotProcessing}
|
onSessionNotProcessing={markSessionAsNotProcessing}
|
||||||
processingSessions={processingSessions}
|
processingSessions={processingSessions}
|
||||||
onReplaceTemporarySession={replaceTemporarySession}
|
onNavigateToSession={(targetSessionId: string, options) =>
|
||||||
onNavigateToSession={(targetSessionId: string) => navigate(`/session/${targetSessionId}`)}
|
navigate(`/session/${targetSessionId}`, { replace: Boolean(options?.replace) })
|
||||||
|
}
|
||||||
onShowSettings={() => setShowSettings(true)}
|
onShowSettings={() => setShowSettings(true)}
|
||||||
externalMessageUpdate={externalMessageUpdate}
|
externalMessageUpdate={externalMessageUpdate}
|
||||||
|
newSessionTrigger={newSessionTrigger}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ import type {
|
|||||||
TouchEvent,
|
TouchEvent,
|
||||||
} from 'react';
|
} from 'react';
|
||||||
import { useDropzone } from 'react-dropzone';
|
import { useDropzone } from 'react-dropzone';
|
||||||
|
|
||||||
import { authenticatedFetch } from '../../../utils/api';
|
import { authenticatedFetch } from '../../../utils/api';
|
||||||
import { thinkingModes } from '../constants/thinkingModes';
|
import { thinkingModes } from '../constants/thinkingModes';
|
||||||
import { grantClaudeToolPermission } from '../utils/chatPermissions';
|
import { grantClaudeToolPermission } from '../utils/chatPermissions';
|
||||||
@@ -21,6 +22,7 @@ import type {
|
|||||||
} from '../types/types';
|
} from '../types/types';
|
||||||
import type { Project, ProjectSession, LLMProvider } from '../../../types/app';
|
import type { Project, ProjectSession, LLMProvider } from '../../../types/app';
|
||||||
import { escapeRegExp } from '../utils/chatFormatting';
|
import { escapeRegExp } from '../utils/chatFormatting';
|
||||||
|
|
||||||
import { useFileMentions } from './useFileMentions';
|
import { useFileMentions } from './useFileMentions';
|
||||||
import { type SlashCommand, useSlashCommands } from './useSlashCommands';
|
import { type SlashCommand, useSlashCommands } from './useSlashCommands';
|
||||||
|
|
||||||
@@ -80,9 +82,6 @@ const createFakeSubmitEvent = () => {
|
|||||||
return { preventDefault: () => undefined } as unknown as FormEvent<HTMLFormElement>;
|
return { preventDefault: () => undefined } as unknown as FormEvent<HTMLFormElement>;
|
||||||
};
|
};
|
||||||
|
|
||||||
const isTemporarySessionId = (sessionId: string | null | undefined) =>
|
|
||||||
Boolean(sessionId && sessionId.startsWith('new-session-'));
|
|
||||||
|
|
||||||
const getNotificationSessionSummary = (
|
const getNotificationSessionSummary = (
|
||||||
selectedSession: ProjectSession | null,
|
selectedSession: ProjectSession | null,
|
||||||
fallbackInput: string,
|
fallbackInput: string,
|
||||||
@@ -533,7 +532,6 @@ export function useChatComposerState({
|
|||||||
|
|
||||||
const effectiveSessionId =
|
const effectiveSessionId =
|
||||||
currentSessionId || selectedSession?.id || sessionStorage.getItem('cursorSessionId');
|
currentSessionId || selectedSession?.id || sessionStorage.getItem('cursorSessionId');
|
||||||
const sessionToActivate = effectiveSessionId || `new-session-${Date.now()}`;
|
|
||||||
|
|
||||||
const userMessage: ChatMessage = {
|
const userMessage: ChatMessage = {
|
||||||
type: 'user',
|
type: 'user',
|
||||||
@@ -559,10 +557,12 @@ export function useChatComposerState({
|
|||||||
// Reset stale pending IDs from previous interrupted runs before creating a new one.
|
// Reset stale pending IDs from previous interrupted runs before creating a new one.
|
||||||
sessionStorage.removeItem('pendingSessionId');
|
sessionStorage.removeItem('pendingSessionId');
|
||||||
}
|
}
|
||||||
|
// For new sessions we intentionally keep this as `null` until the backend
|
||||||
|
// emits `session_created` with the canonical provider session id.
|
||||||
pendingViewSessionRef.current = { sessionId: null, startedAt: Date.now() };
|
pendingViewSessionRef.current = { sessionId: null, startedAt: Date.now() };
|
||||||
}
|
}
|
||||||
onSessionActive?.(sessionToActivate);
|
if (effectiveSessionId) {
|
||||||
if (effectiveSessionId && !isTemporarySessionId(effectiveSessionId)) {
|
onSessionActive?.(effectiveSessionId);
|
||||||
onSessionProcessing?.(effectiveSessionId);
|
onSessionProcessing?.(effectiveSessionId);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -868,7 +868,7 @@ export function useChatComposerState({
|
|||||||
];
|
];
|
||||||
|
|
||||||
const targetSessionId =
|
const targetSessionId =
|
||||||
candidateSessionIds.find((sessionId) => Boolean(sessionId) && !isTemporarySessionId(sessionId)) || null;
|
candidateSessionIds.find((sessionId) => Boolean(sessionId)) || null;
|
||||||
|
|
||||||
if (!targetSessionId) {
|
if (!targetSessionId) {
|
||||||
console.warn('Abort requested but no concrete session ID is available yet.');
|
console.warn('Abort requested but no concrete session ID is available yet.');
|
||||||
|
|||||||
@@ -11,8 +11,9 @@ import { decodeHtmlEntities, unescapeWithMathProtection, formatUsageLimitText }
|
|||||||
* Convert NormalizedMessage[] from the session store into ChatMessage[]
|
* Convert NormalizedMessage[] from the session store into ChatMessage[]
|
||||||
* that the existing UI components expect.
|
* that the existing UI components expect.
|
||||||
*
|
*
|
||||||
* Internal/system content (e.g. <system-reminder>, <command-name>) is already
|
* Truly internal/system content is already filtered server-side. Some Claude
|
||||||
* filtered server-side by the Claude provider module.
|
* transcript artifacts such as local slash commands and compact summaries are
|
||||||
|
* intentionally preserved and annotated so they can render like normal chat.
|
||||||
*/
|
*/
|
||||||
export function normalizedToChatMessages(messages: NormalizedMessage[]): ChatMessage[] {
|
export function normalizedToChatMessages(messages: NormalizedMessage[]): ChatMessage[] {
|
||||||
const converted: ChatMessage[] = [];
|
const converted: ChatMessage[] = [];
|
||||||
@@ -26,6 +27,16 @@ export function normalizedToChatMessages(messages: NormalizedMessage[]): ChatMes
|
|||||||
}
|
}
|
||||||
|
|
||||||
for (const msg of messages) {
|
for (const msg of messages) {
|
||||||
|
const sharedMetadata = {
|
||||||
|
displayText: msg.displayText,
|
||||||
|
commandName: msg.commandName,
|
||||||
|
commandMessage: msg.commandMessage,
|
||||||
|
commandArgs: msg.commandArgs,
|
||||||
|
isLocalCommand: msg.isLocalCommand,
|
||||||
|
isLocalCommandStdout: msg.isLocalCommandStdout,
|
||||||
|
isCompactSummary: msg.isCompactSummary,
|
||||||
|
};
|
||||||
|
|
||||||
switch (msg.kind) {
|
switch (msg.kind) {
|
||||||
case 'text': {
|
case 'text': {
|
||||||
const content = msg.content || '';
|
const content = msg.content || '';
|
||||||
@@ -42,12 +53,14 @@ export function normalizedToChatMessages(messages: NormalizedMessage[]): ChatMes
|
|||||||
timestamp: msg.timestamp,
|
timestamp: msg.timestamp,
|
||||||
isTaskNotification: true,
|
isTaskNotification: true,
|
||||||
taskStatus: taskNotifMatch[1]?.trim() || 'completed',
|
taskStatus: taskNotifMatch[1]?.trim() || 'completed',
|
||||||
|
...sharedMetadata,
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
converted.push({
|
converted.push({
|
||||||
type: 'user',
|
type: 'user',
|
||||||
content: unescapeWithMathProtection(decodeHtmlEntities(content)),
|
content: unescapeWithMathProtection(decodeHtmlEntities(content)),
|
||||||
timestamp: msg.timestamp,
|
timestamp: msg.timestamp,
|
||||||
|
...sharedMetadata,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@@ -58,6 +71,7 @@ export function normalizedToChatMessages(messages: NormalizedMessage[]): ChatMes
|
|||||||
type: 'assistant',
|
type: 'assistant',
|
||||||
content: text,
|
content: text,
|
||||||
timestamp: msg.timestamp,
|
timestamp: msg.timestamp,
|
||||||
|
...sharedMetadata,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -106,6 +120,7 @@ export function normalizedToChatMessages(messages: NormalizedMessage[]): ChatMes
|
|||||||
isComplete: Boolean(toolResult),
|
isComplete: Boolean(toolResult),
|
||||||
}
|
}
|
||||||
: undefined,
|
: undefined,
|
||||||
|
...sharedMetadata,
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -117,6 +132,7 @@ export function normalizedToChatMessages(messages: NormalizedMessage[]): ChatMes
|
|||||||
content: unescapeWithMathProtection(msg.content),
|
content: unescapeWithMathProtection(msg.content),
|
||||||
timestamp: msg.timestamp,
|
timestamp: msg.timestamp,
|
||||||
isThinking: true,
|
isThinking: true,
|
||||||
|
...sharedMetadata,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -126,6 +142,7 @@ export function normalizedToChatMessages(messages: NormalizedMessage[]): ChatMes
|
|||||||
type: 'error',
|
type: 'error',
|
||||||
content: msg.content || 'Unknown error',
|
content: msg.content || 'Unknown error',
|
||||||
timestamp: msg.timestamp,
|
timestamp: msg.timestamp,
|
||||||
|
...sharedMetadata,
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@@ -135,6 +152,7 @@ export function normalizedToChatMessages(messages: NormalizedMessage[]): ChatMes
|
|||||||
content: msg.content || '',
|
content: msg.content || '',
|
||||||
timestamp: msg.timestamp,
|
timestamp: msg.timestamp,
|
||||||
isInteractivePrompt: true,
|
isInteractivePrompt: true,
|
||||||
|
...sharedMetadata,
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@@ -145,6 +163,7 @@ export function normalizedToChatMessages(messages: NormalizedMessage[]): ChatMes
|
|||||||
timestamp: msg.timestamp,
|
timestamp: msg.timestamp,
|
||||||
isTaskNotification: true,
|
isTaskNotification: true,
|
||||||
taskStatus: msg.status || 'completed',
|
taskStatus: msg.status || 'completed',
|
||||||
|
...sharedMetadata,
|
||||||
});
|
});
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@@ -155,6 +174,7 @@ export function normalizedToChatMessages(messages: NormalizedMessage[]): ChatMes
|
|||||||
content: msg.content,
|
content: msg.content,
|
||||||
timestamp: msg.timestamp,
|
timestamp: msg.timestamp,
|
||||||
isStreaming: true,
|
isStreaming: true,
|
||||||
|
...sharedMetadata,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
import { useEffect, useRef } from 'react';
|
import { useEffect, useRef } from 'react';
|
||||||
import type { Dispatch, MutableRefObject, SetStateAction } from 'react';
|
import type { Dispatch, MutableRefObject, SetStateAction } from 'react';
|
||||||
|
|
||||||
import { usePaletteOps } from '../../../contexts/PaletteOpsContext';
|
import { usePaletteOps } from '../../../contexts/PaletteOpsContext';
|
||||||
import type { PendingPermissionRequest } from '../types/types';
|
import type { PendingPermissionRequest, SessionNavigationOptions } from '../types/types';
|
||||||
import type { Project, ProjectSession, LLMProvider } from '../../../types/app';
|
import type { ProjectSession, LLMProvider } from '../../../types/app';
|
||||||
import type { SessionStore, NormalizedMessage } from '../../../stores/useSessionStore';
|
import type { SessionStore, NormalizedMessage } from '../../../stores/useSessionStore';
|
||||||
|
|
||||||
type PendingViewSession = {
|
type PendingViewSession = {
|
||||||
@@ -50,7 +51,6 @@ type LatestChatMessage = {
|
|||||||
interface UseChatRealtimeHandlersArgs {
|
interface UseChatRealtimeHandlersArgs {
|
||||||
latestMessage: LatestChatMessage | null;
|
latestMessage: LatestChatMessage | null;
|
||||||
provider: LLMProvider;
|
provider: LLMProvider;
|
||||||
selectedProject: Project | null;
|
|
||||||
selectedSession: ProjectSession | null;
|
selectedSession: ProjectSession | null;
|
||||||
currentSessionId: string | null;
|
currentSessionId: string | null;
|
||||||
setCurrentSessionId: (sessionId: string | null) => void;
|
setCurrentSessionId: (sessionId: string | null) => void;
|
||||||
@@ -60,14 +60,12 @@ interface UseChatRealtimeHandlersArgs {
|
|||||||
setTokenBudget: (budget: Record<string, unknown> | null) => void;
|
setTokenBudget: (budget: Record<string, unknown> | null) => void;
|
||||||
setPendingPermissionRequests: Dispatch<SetStateAction<PendingPermissionRequest[]>>;
|
setPendingPermissionRequests: Dispatch<SetStateAction<PendingPermissionRequest[]>>;
|
||||||
pendingViewSessionRef: MutableRefObject<PendingViewSession | null>;
|
pendingViewSessionRef: MutableRefObject<PendingViewSession | null>;
|
||||||
streamBufferRef: MutableRefObject<string>;
|
|
||||||
streamTimerRef: MutableRefObject<number | null>;
|
streamTimerRef: MutableRefObject<number | null>;
|
||||||
accumulatedStreamRef: MutableRefObject<string>;
|
accumulatedStreamRef: MutableRefObject<string>;
|
||||||
onSessionInactive?: (sessionId?: string | null) => void;
|
onSessionInactive?: (sessionId?: string | null) => void;
|
||||||
onSessionProcessing?: (sessionId?: string | null) => void;
|
onSessionProcessing?: (sessionId?: string | null) => void;
|
||||||
onSessionNotProcessing?: (sessionId?: string | null) => void;
|
onSessionNotProcessing?: (sessionId?: string | null) => void;
|
||||||
onReplaceTemporarySession?: (sessionId?: string | null) => void;
|
onNavigateToSession?: (sessionId: string, options?: SessionNavigationOptions) => void;
|
||||||
onNavigateToSession?: (sessionId: string) => void;
|
|
||||||
onWebSocketReconnect?: () => void;
|
onWebSocketReconnect?: () => void;
|
||||||
sessionStore: SessionStore;
|
sessionStore: SessionStore;
|
||||||
}
|
}
|
||||||
@@ -79,7 +77,6 @@ interface UseChatRealtimeHandlersArgs {
|
|||||||
export function useChatRealtimeHandlers({
|
export function useChatRealtimeHandlers({
|
||||||
latestMessage,
|
latestMessage,
|
||||||
provider,
|
provider,
|
||||||
selectedProject,
|
|
||||||
selectedSession,
|
selectedSession,
|
||||||
currentSessionId,
|
currentSessionId,
|
||||||
setCurrentSessionId,
|
setCurrentSessionId,
|
||||||
@@ -89,13 +86,11 @@ export function useChatRealtimeHandlers({
|
|||||||
setTokenBudget,
|
setTokenBudget,
|
||||||
setPendingPermissionRequests,
|
setPendingPermissionRequests,
|
||||||
pendingViewSessionRef,
|
pendingViewSessionRef,
|
||||||
streamBufferRef,
|
|
||||||
streamTimerRef,
|
streamTimerRef,
|
||||||
accumulatedStreamRef,
|
accumulatedStreamRef,
|
||||||
onSessionInactive,
|
onSessionInactive,
|
||||||
onSessionProcessing,
|
onSessionProcessing,
|
||||||
onSessionNotProcessing,
|
onSessionNotProcessing,
|
||||||
onReplaceTemporarySession,
|
|
||||||
onNavigateToSession,
|
onNavigateToSession,
|
||||||
onWebSocketReconnect,
|
onWebSocketReconnect,
|
||||||
sessionStore,
|
sessionStore,
|
||||||
@@ -186,7 +181,6 @@ export function useChatRealtimeHandlers({
|
|||||||
if (msg.kind === 'stream_delta') {
|
if (msg.kind === 'stream_delta') {
|
||||||
const text = msg.content || '';
|
const text = msg.content || '';
|
||||||
if (!text) return;
|
if (!text) return;
|
||||||
streamBufferRef.current += text;
|
|
||||||
accumulatedStreamRef.current += text;
|
accumulatedStreamRef.current += text;
|
||||||
if (!streamTimerRef.current) {
|
if (!streamTimerRef.current) {
|
||||||
streamTimerRef.current = window.setTimeout(() => {
|
streamTimerRef.current = window.setTimeout(() => {
|
||||||
@@ -215,12 +209,18 @@ export function useChatRealtimeHandlers({
|
|||||||
sessionStore.finalizeStreaming(sid);
|
sessionStore.finalizeStreaming(sid);
|
||||||
}
|
}
|
||||||
accumulatedStreamRef.current = '';
|
accumulatedStreamRef.current = '';
|
||||||
streamBufferRef.current = '';
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// --- All other messages: route to store ---
|
// --- All other messages: route to store ---
|
||||||
if (sid) {
|
const shouldPersist =
|
||||||
|
msg.kind !== 'session_created'
|
||||||
|
&& msg.kind !== 'complete'
|
||||||
|
&& msg.kind !== 'status'
|
||||||
|
&& msg.kind !== 'permission_request'
|
||||||
|
&& msg.kind !== 'permission_cancelled';
|
||||||
|
|
||||||
|
if (sid && shouldPersist) {
|
||||||
sessionStore.appendRealtime(sid, msg as NormalizedMessage);
|
sessionStore.appendRealtime(sid, msg as NormalizedMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -230,13 +230,16 @@ export function useChatRealtimeHandlers({
|
|||||||
const newSessionId = msg.newSessionId;
|
const newSessionId = msg.newSessionId;
|
||||||
if (!newSessionId) break;
|
if (!newSessionId) break;
|
||||||
|
|
||||||
if (!currentSessionId || currentSessionId.startsWith('new-session-')) {
|
// We no longer synthesize client-side placeholder IDs. Until the provider
|
||||||
|
// announces `session_created`, the active id is expected to be null.
|
||||||
|
if (!currentSessionId) {
|
||||||
|
console.log('Session created with ID:', newSessionId);
|
||||||
|
console.log('Existing session ID:', currentSessionId);
|
||||||
sessionStorage.setItem('pendingSessionId', newSessionId);
|
sessionStorage.setItem('pendingSessionId', newSessionId);
|
||||||
if (pendingViewSessionRef.current && !pendingViewSessionRef.current.sessionId) {
|
if (pendingViewSessionRef.current && !pendingViewSessionRef.current.sessionId) {
|
||||||
pendingViewSessionRef.current.sessionId = newSessionId;
|
pendingViewSessionRef.current.sessionId = newSessionId;
|
||||||
}
|
}
|
||||||
setCurrentSessionId(newSessionId);
|
setCurrentSessionId(newSessionId);
|
||||||
onReplaceTemporarySession?.(newSessionId);
|
|
||||||
setPendingPermissionRequests((prev) =>
|
setPendingPermissionRequests((prev) =>
|
||||||
prev.map((r) => (r.sessionId ? r : { ...r, sessionId: newSessionId })),
|
prev.map((r) => (r.sessionId ? r : { ...r, sessionId: newSessionId })),
|
||||||
);
|
);
|
||||||
@@ -256,7 +259,6 @@ export function useChatRealtimeHandlers({
|
|||||||
sessionStore.finalizeStreaming(sid);
|
sessionStore.finalizeStreaming(sid);
|
||||||
}
|
}
|
||||||
accumulatedStreamRef.current = '';
|
accumulatedStreamRef.current = '';
|
||||||
streamBufferRef.current = '';
|
|
||||||
|
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
setCanAbortSession(false);
|
setCanAbortSession(false);
|
||||||
@@ -273,13 +275,53 @@ export function useChatRealtimeHandlers({
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Clear pending session
|
const actualSessionId =
|
||||||
|
typeof msg.actualSessionId === 'string' && msg.actualSessionId.trim().length > 0
|
||||||
|
? msg.actualSessionId
|
||||||
|
: null;
|
||||||
const pendingSessionId = sessionStorage.getItem('pendingSessionId');
|
const pendingSessionId = sessionStorage.getItem('pendingSessionId');
|
||||||
if (pendingSessionId && !currentSessionId && msg.exitCode === 0) {
|
const completedSuccessfully = msg.exitCode === undefined || msg.exitCode === 0;
|
||||||
const actualId = msg.actualSessionId || pendingSessionId;
|
const isVisibleSession =
|
||||||
setCurrentSessionId(actualId);
|
Boolean(
|
||||||
if (msg.actualSessionId) {
|
sid
|
||||||
onNavigateToSession?.(actualId);
|
&& (
|
||||||
|
sid === activeViewSessionId
|
||||||
|
|| sid === pendingSessionId
|
||||||
|
|| pendingViewSessionRef.current?.sessionId === sid
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (actualSessionId && sid && actualSessionId !== sid) {
|
||||||
|
sessionStore.replaceSessionId(sid, actualSessionId);
|
||||||
|
|
||||||
|
if (isVisibleSession) {
|
||||||
|
setCurrentSessionId(actualSessionId);
|
||||||
|
|
||||||
|
if (pendingViewSessionRef.current) {
|
||||||
|
const pendingSession = pendingViewSessionRef.current.sessionId;
|
||||||
|
if (!pendingSession || pendingSession === sid) {
|
||||||
|
pendingViewSessionRef.current.sessionId = actualSessionId;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (completedSuccessfully && pendingSessionId === sid) {
|
||||||
|
sessionStorage.removeItem('pendingSessionId');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isVisibleSession) {
|
||||||
|
onNavigateToSession?.(actualSessionId, { replace: true });
|
||||||
|
setTimeout(() => { void paletteOps.refreshProjects(); }, 500);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Clear pending session
|
||||||
|
if (pendingSessionId && !currentSessionId && completedSuccessfully) {
|
||||||
|
const resolvedSessionId = actualSessionId || pendingSessionId;
|
||||||
|
setCurrentSessionId(resolvedSessionId);
|
||||||
|
if (actualSessionId) {
|
||||||
|
onNavigateToSession?.(resolvedSessionId, { replace: true });
|
||||||
}
|
}
|
||||||
sessionStorage.removeItem('pendingSessionId');
|
sessionStorage.removeItem('pendingSessionId');
|
||||||
setTimeout(() => { void paletteOps.refreshProjects(); }, 500);
|
setTimeout(() => { void paletteOps.refreshProjects(); }, 500);
|
||||||
@@ -345,7 +387,6 @@ export function useChatRealtimeHandlers({
|
|||||||
}, [
|
}, [
|
||||||
latestMessage,
|
latestMessage,
|
||||||
provider,
|
provider,
|
||||||
selectedProject,
|
|
||||||
selectedSession,
|
selectedSession,
|
||||||
currentSessionId,
|
currentSessionId,
|
||||||
setCurrentSessionId,
|
setCurrentSessionId,
|
||||||
@@ -355,13 +396,11 @@ export function useChatRealtimeHandlers({
|
|||||||
setTokenBudget,
|
setTokenBudget,
|
||||||
setPendingPermissionRequests,
|
setPendingPermissionRequests,
|
||||||
pendingViewSessionRef,
|
pendingViewSessionRef,
|
||||||
streamBufferRef,
|
|
||||||
streamTimerRef,
|
streamTimerRef,
|
||||||
accumulatedStreamRef,
|
accumulatedStreamRef,
|
||||||
onSessionInactive,
|
onSessionInactive,
|
||||||
onSessionProcessing,
|
onSessionProcessing,
|
||||||
onSessionNotProcessing,
|
onSessionNotProcessing,
|
||||||
onReplaceTemporarySession,
|
|
||||||
onNavigateToSession,
|
onNavigateToSession,
|
||||||
onWebSocketReconnect,
|
onWebSocketReconnect,
|
||||||
sessionStore,
|
sessionStore,
|
||||||
|
|||||||
@@ -1,11 +1,13 @@
|
|||||||
import { useCallback, useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react';
|
import { useCallback, useEffect, useLayoutEffect, useMemo, useRef, useState } from 'react';
|
||||||
import type { MutableRefObject } from 'react';
|
import type { MutableRefObject } from 'react';
|
||||||
|
|
||||||
import { authenticatedFetch } from '../../../utils/api';
|
import { authenticatedFetch } from '../../../utils/api';
|
||||||
import type { ChatMessage, Provider } from '../types/types';
|
|
||||||
import type { Project, ProjectSession, LLMProvider } from '../../../types/app';
|
import type { Project, ProjectSession, LLMProvider } from '../../../types/app';
|
||||||
import { createCachedDiffCalculator, type DiffCalculator } from '../utils/messageTransforms';
|
|
||||||
import { normalizedToChatMessages } from './useChatMessages';
|
|
||||||
import type { SessionStore, NormalizedMessage } from '../../../stores/useSessionStore';
|
import type { SessionStore, NormalizedMessage } from '../../../stores/useSessionStore';
|
||||||
|
import type { ChatMessage, Provider } from '../types/types';
|
||||||
|
import { createCachedDiffCalculator, type DiffCalculator } from '../utils/messageTransforms';
|
||||||
|
|
||||||
|
import { normalizedToChatMessages } from './useChatMessages';
|
||||||
|
|
||||||
const MESSAGES_PER_PAGE = 20;
|
const MESSAGES_PER_PAGE = 20;
|
||||||
const INITIAL_VISIBLE_MESSAGES = 100;
|
const INITIAL_VISIBLE_MESSAGES = 100;
|
||||||
@@ -22,6 +24,7 @@ interface UseChatSessionStateArgs {
|
|||||||
sendMessage: (message: unknown) => void;
|
sendMessage: (message: unknown) => void;
|
||||||
autoScrollToBottom?: boolean;
|
autoScrollToBottom?: boolean;
|
||||||
externalMessageUpdate?: number;
|
externalMessageUpdate?: number;
|
||||||
|
newSessionTrigger?: number;
|
||||||
processingSessions?: Set<string>;
|
processingSessions?: Set<string>;
|
||||||
resetStreamingState: () => void;
|
resetStreamingState: () => void;
|
||||||
pendingViewSessionRef: MutableRefObject<PendingViewSession | null>;
|
pendingViewSessionRef: MutableRefObject<PendingViewSession | null>;
|
||||||
@@ -95,6 +98,7 @@ export function useChatSessionState({
|
|||||||
sendMessage,
|
sendMessage,
|
||||||
autoScrollToBottom,
|
autoScrollToBottom,
|
||||||
externalMessageUpdate,
|
externalMessageUpdate,
|
||||||
|
newSessionTrigger,
|
||||||
processingSessions,
|
processingSessions,
|
||||||
resetStreamingState,
|
resetStreamingState,
|
||||||
pendingViewSessionRef,
|
pendingViewSessionRef,
|
||||||
@@ -131,15 +135,86 @@ export function useChatSessionState({
|
|||||||
const loadAllFinishedTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);
|
const loadAllFinishedTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);
|
||||||
const loadAllOverlayTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);
|
const loadAllOverlayTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);
|
||||||
const lastLoadedSessionKeyRef = useRef<string | null>(null);
|
const lastLoadedSessionKeyRef = useRef<string | null>(null);
|
||||||
|
/**
|
||||||
|
* Tracks the last processed value from `useProjectsState.newSessionTrigger`.
|
||||||
|
*
|
||||||
|
* The trigger itself is intentionally increment-only and routed via:
|
||||||
|
* useProjectsState -> AppContent -> MainContent -> ChatInterface -> this hook.
|
||||||
|
* We compare values to ensure each explicit New Session click runs exactly one
|
||||||
|
* reset pass in this local chat state domain.
|
||||||
|
*/
|
||||||
|
const previousNewSessionTriggerRef = useRef(newSessionTrigger ?? 0);
|
||||||
|
|
||||||
const createDiff = useMemo<DiffCalculator>(() => createCachedDiffCalculator(), []);
|
const createDiff = useMemo<DiffCalculator>(() => createCachedDiffCalculator(), []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const trigger = newSessionTrigger ?? 0;
|
||||||
|
if (trigger === previousNewSessionTriggerRef.current) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
previousNewSessionTriggerRef.current = trigger;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Consumer-side reset for explicit New Session intent.
|
||||||
|
*
|
||||||
|
* Why this is essential:
|
||||||
|
* - Chat keeps local state that is not fully derived from `selectedSession`:
|
||||||
|
* `currentSessionId`, `pendingUserMessage`, streaming/status flags, message
|
||||||
|
* pagination/scroll bookkeeping, and pending session IDs in sessionStorage.
|
||||||
|
* - If the user clicks New Session while already on the same route with no
|
||||||
|
* selected session, parent state updates can be idempotent and this local
|
||||||
|
* state would otherwise persist, making the click appear to "do nothing".
|
||||||
|
*
|
||||||
|
* What this reset guarantees:
|
||||||
|
* - A deterministic clean draft state on every New Session click.
|
||||||
|
* - No dependence on route/tab/session-object identity changes.
|
||||||
|
* - No coupling to unrelated external update signals.
|
||||||
|
*/
|
||||||
|
resetStreamingState();
|
||||||
|
pendingViewSessionRef.current = null;
|
||||||
|
setClaudeStatus(null);
|
||||||
|
setCanAbortSession(false);
|
||||||
|
setIsLoading(false);
|
||||||
|
setCurrentSessionId(null);
|
||||||
|
setPendingUserMessage(null);
|
||||||
|
sessionStorage.removeItem('pendingSessionId');
|
||||||
|
sessionStorage.removeItem('cursorSessionId');
|
||||||
|
messagesOffsetRef.current = 0;
|
||||||
|
setHasMoreMessages(false);
|
||||||
|
setTotalMessages(0);
|
||||||
|
|
||||||
|
setTokenBudget(null);
|
||||||
|
setVisibleMessageCount(INITIAL_VISIBLE_MESSAGES);
|
||||||
|
setAllMessagesLoaded(false);
|
||||||
|
allMessagesLoadedRef.current = false;
|
||||||
|
setIsLoadingAllMessages(false);
|
||||||
|
setLoadAllJustFinished(false);
|
||||||
|
setShowLoadAllOverlay(false);
|
||||||
|
setViewHiddenCount(0);
|
||||||
|
setSearchTarget(null);
|
||||||
|
searchScrollActiveRef.current = false;
|
||||||
|
topLoadLockRef.current = false;
|
||||||
|
pendingScrollRestoreRef.current = null;
|
||||||
|
pendingInitialScrollRef.current = true;
|
||||||
|
lastLoadedSessionKeyRef.current = null;
|
||||||
|
|
||||||
|
if (loadAllOverlayTimerRef.current) {
|
||||||
|
clearTimeout(loadAllOverlayTimerRef.current);
|
||||||
|
loadAllOverlayTimerRef.current = null;
|
||||||
|
}
|
||||||
|
if (loadAllFinishedTimerRef.current) {
|
||||||
|
clearTimeout(loadAllFinishedTimerRef.current);
|
||||||
|
loadAllFinishedTimerRef.current = null;
|
||||||
|
}
|
||||||
|
}, [newSessionTrigger, pendingViewSessionRef, resetStreamingState]);
|
||||||
|
|
||||||
/* ---------------------------------------------------------------- */
|
/* ---------------------------------------------------------------- */
|
||||||
/* Derive chatMessages from the store */
|
/* Derive chatMessages from the store */
|
||||||
/* ---------------------------------------------------------------- */
|
/* ---------------------------------------------------------------- */
|
||||||
|
|
||||||
const activeSessionId = selectedSession?.id || currentSessionId || null;
|
const activeSessionId = selectedSession?.id || currentSessionId || null;
|
||||||
const [pendingUserMessage, setPendingUserMessage] = useState<ChatMessage | null>(null);
|
const [pendingUserMessage, setPendingUserMessage] = useState<ChatMessage | null>(null);
|
||||||
|
const flushedPendingUserMessageRef = useRef<ChatMessage | null>(null);
|
||||||
|
|
||||||
// Tell the store which session we're viewing so it only re-renders for this one
|
// Tell the store which session we're viewing so it only re-renders for this one
|
||||||
const prevActiveForStoreRef = useRef<string | null>(null);
|
const prevActiveForStoreRef = useRef<string | null>(null);
|
||||||
@@ -148,17 +223,29 @@ export function useChatSessionState({
|
|||||||
sessionStore.setActiveSession(activeSessionId);
|
sessionStore.setActiveSession(activeSessionId);
|
||||||
}
|
}
|
||||||
|
|
||||||
// When a real session ID arrives and we have a pending user message, flush it to the store
|
useEffect(() => {
|
||||||
const prevActiveSessionRef = useRef<string | null>(null);
|
if (!pendingUserMessage) {
|
||||||
if (activeSessionId && activeSessionId !== prevActiveSessionRef.current && pendingUserMessage) {
|
flushedPendingUserMessageRef.current = null;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!activeSessionId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (flushedPendingUserMessageRef.current === pendingUserMessage) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const prov = (localStorage.getItem('selected-provider') as LLMProvider) || 'claude';
|
const prov = (localStorage.getItem('selected-provider') as LLMProvider) || 'claude';
|
||||||
const normalized = chatMessageToNormalized(pendingUserMessage, activeSessionId, prov);
|
const normalized = chatMessageToNormalized(pendingUserMessage, activeSessionId, prov);
|
||||||
if (normalized) {
|
if (normalized) {
|
||||||
sessionStore.appendRealtime(activeSessionId, normalized);
|
sessionStore.appendRealtime(activeSessionId, normalized);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
flushedPendingUserMessageRef.current = pendingUserMessage;
|
||||||
setPendingUserMessage(null);
|
setPendingUserMessage(null);
|
||||||
}
|
}, [activeSessionId, pendingUserMessage, sessionStore]);
|
||||||
prevActiveSessionRef.current = activeSessionId;
|
|
||||||
|
|
||||||
const storeMessages = activeSessionId ? sessionStore.getMessages(activeSessionId) : [];
|
const storeMessages = activeSessionId ? sessionStore.getMessages(activeSessionId) : [];
|
||||||
|
|
||||||
@@ -232,7 +319,6 @@ export function useChatSessionState({
|
|||||||
if (!hasMoreMessages || !selectedSession || !selectedProject) return false;
|
if (!hasMoreMessages || !selectedSession || !selectedProject) return false;
|
||||||
|
|
||||||
const sessionProvider = selectedSession.__provider || 'claude';
|
const sessionProvider = selectedSession.__provider || 'claude';
|
||||||
if (sessionProvider === 'cursor') return false;
|
|
||||||
|
|
||||||
isLoadingMoreRef.current = true;
|
isLoadingMoreRef.current = true;
|
||||||
const previousScrollHeight = container.scrollHeight;
|
const previousScrollHeight = container.scrollHeight;
|
||||||
@@ -465,7 +551,6 @@ export function useChatSessionState({
|
|||||||
const scrollToTarget = async () => {
|
const scrollToTarget = async () => {
|
||||||
if (!allMessagesLoadedRef.current && selectedSession && selectedProject) {
|
if (!allMessagesLoadedRef.current && selectedSession && selectedProject) {
|
||||||
const sessionProvider = selectedSession.__provider || 'claude';
|
const sessionProvider = selectedSession.__provider || 'claude';
|
||||||
if (sessionProvider !== 'cursor') {
|
|
||||||
try {
|
try {
|
||||||
// Load all messages into the store for search navigation
|
// Load all messages into the store for search navigation
|
||||||
const slot = await sessionStore.fetchFromServer(selectedSession.id, {
|
const slot = await sessionStore.fetchFromServer(selectedSession.id, {
|
||||||
@@ -487,7 +572,6 @@ export function useChatSessionState({
|
|||||||
} catch {
|
} catch {
|
||||||
// Fall through and scroll in current messages
|
// Fall through and scroll in current messages
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
setVisibleMessageCount(Infinity);
|
setVisibleMessageCount(Infinity);
|
||||||
|
|
||||||
@@ -542,7 +626,7 @@ export function useChatSessionState({
|
|||||||
|
|
||||||
// Token usage fetch for Claude
|
// Token usage fetch for Claude
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!selectedProject || !selectedSession?.id || selectedSession.id.startsWith('new-session-')) {
|
if (!selectedProject || !selectedSession?.id) {
|
||||||
setTokenBudget(null);
|
setTokenBudget(null);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -635,15 +719,6 @@ export function useChatSessionState({
|
|||||||
if (!selectedSession || !selectedProject) return;
|
if (!selectedSession || !selectedProject) return;
|
||||||
if (isLoadingAllMessages) return;
|
if (isLoadingAllMessages) return;
|
||||||
const sessionProvider = selectedSession.__provider || 'claude';
|
const sessionProvider = selectedSession.__provider || 'claude';
|
||||||
if (sessionProvider === 'cursor') {
|
|
||||||
setVisibleMessageCount(Infinity);
|
|
||||||
setAllMessagesLoaded(true);
|
|
||||||
allMessagesLoadedRef.current = true;
|
|
||||||
setLoadAllJustFinished(true);
|
|
||||||
if (loadAllFinishedTimerRef.current) clearTimeout(loadAllFinishedTimerRef.current);
|
|
||||||
loadAllFinishedTimerRef.current = setTimeout(() => { setLoadAllJustFinished(false); setShowLoadAllOverlay(false); }, 1000);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const requestSessionId = selectedSession.id;
|
const requestSessionId = selectedSession.id;
|
||||||
allMessagesLoadedRef.current = true;
|
allMessagesLoadedRef.current = true;
|
||||||
|
|||||||
@@ -28,6 +28,7 @@ export interface SubagentChildTool {
|
|||||||
export interface ChatMessage {
|
export interface ChatMessage {
|
||||||
type: string;
|
type: string;
|
||||||
content?: string;
|
content?: string;
|
||||||
|
displayText?: string;
|
||||||
timestamp: string | number | Date;
|
timestamp: string | number | Date;
|
||||||
images?: ChatImage[];
|
images?: ChatImage[];
|
||||||
reasoning?: string;
|
reasoning?: string;
|
||||||
@@ -40,6 +41,12 @@ export interface ChatMessage {
|
|||||||
toolResult?: ToolResult | null;
|
toolResult?: ToolResult | null;
|
||||||
toolId?: string;
|
toolId?: string;
|
||||||
toolCallId?: string;
|
toolCallId?: string;
|
||||||
|
commandName?: string;
|
||||||
|
commandMessage?: string;
|
||||||
|
commandArgs?: string;
|
||||||
|
isLocalCommand?: boolean;
|
||||||
|
isLocalCommandStdout?: boolean;
|
||||||
|
isCompactSummary?: boolean;
|
||||||
isSubagentContainer?: boolean;
|
isSubagentContainer?: boolean;
|
||||||
subagentState?: {
|
subagentState?: {
|
||||||
childTools: SubagentChildTool[];
|
childTools: SubagentChildTool[];
|
||||||
@@ -91,6 +98,10 @@ export interface Question {
|
|||||||
multiSelect?: boolean;
|
multiSelect?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type SessionNavigationOptions = {
|
||||||
|
replace?: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
export interface ChatInterfaceProps {
|
export interface ChatInterfaceProps {
|
||||||
selectedProject: Project | null;
|
selectedProject: Project | null;
|
||||||
selectedSession: ProjectSession | null;
|
selectedSession: ProjectSession | null;
|
||||||
@@ -104,8 +115,7 @@ export interface ChatInterfaceProps {
|
|||||||
onSessionProcessing?: (sessionId?: string | null) => void;
|
onSessionProcessing?: (sessionId?: string | null) => void;
|
||||||
onSessionNotProcessing?: (sessionId?: string | null) => void;
|
onSessionNotProcessing?: (sessionId?: string | null) => void;
|
||||||
processingSessions?: Set<string>;
|
processingSessions?: Set<string>;
|
||||||
onReplaceTemporarySession?: (sessionId?: string | null) => void;
|
onNavigateToSession?: (targetSessionId: string, options?: SessionNavigationOptions) => void;
|
||||||
onNavigateToSession?: (targetSessionId: string) => void;
|
|
||||||
onShowSettings?: () => void;
|
onShowSettings?: () => void;
|
||||||
autoExpandTools?: boolean;
|
autoExpandTools?: boolean;
|
||||||
showRawParameters?: boolean;
|
showRawParameters?: boolean;
|
||||||
@@ -113,6 +123,7 @@ export interface ChatInterfaceProps {
|
|||||||
autoScrollToBottom?: boolean;
|
autoScrollToBottom?: boolean;
|
||||||
sendByCtrlEnter?: boolean;
|
sendByCtrlEnter?: boolean;
|
||||||
externalMessageUpdate?: number;
|
externalMessageUpdate?: number;
|
||||||
|
newSessionTrigger?: number;
|
||||||
onTaskClick?: (...args: unknown[]) => void;
|
onTaskClick?: (...args: unknown[]) => void;
|
||||||
onShowAllTasks?: (() => void) | null;
|
onShowAllTasks?: (() => void) | null;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -34,7 +34,6 @@ function ChatInterface({
|
|||||||
onSessionProcessing,
|
onSessionProcessing,
|
||||||
onSessionNotProcessing,
|
onSessionNotProcessing,
|
||||||
processingSessions,
|
processingSessions,
|
||||||
onReplaceTemporarySession,
|
|
||||||
onNavigateToSession,
|
onNavigateToSession,
|
||||||
onShowSettings,
|
onShowSettings,
|
||||||
autoExpandTools,
|
autoExpandTools,
|
||||||
@@ -43,13 +42,13 @@ function ChatInterface({
|
|||||||
autoScrollToBottom,
|
autoScrollToBottom,
|
||||||
sendByCtrlEnter,
|
sendByCtrlEnter,
|
||||||
externalMessageUpdate,
|
externalMessageUpdate,
|
||||||
|
newSessionTrigger,
|
||||||
onShowAllTasks,
|
onShowAllTasks,
|
||||||
}: ChatInterfaceProps) {
|
}: ChatInterfaceProps) {
|
||||||
const { tasksEnabled, isTaskMasterInstalled } = useTasksSettings();
|
const { tasksEnabled, isTaskMasterInstalled } = useTasksSettings();
|
||||||
const { t } = useTranslation('chat');
|
const { t } = useTranslation('chat');
|
||||||
|
|
||||||
const sessionStore = useSessionStore();
|
const sessionStore = useSessionStore();
|
||||||
const streamBufferRef = useRef('');
|
|
||||||
const streamTimerRef = useRef<number | null>(null);
|
const streamTimerRef = useRef<number | null>(null);
|
||||||
const accumulatedStreamRef = useRef('');
|
const accumulatedStreamRef = useRef('');
|
||||||
const pendingViewSessionRef = useRef<PendingViewSession | null>(null);
|
const pendingViewSessionRef = useRef<PendingViewSession | null>(null);
|
||||||
@@ -59,7 +58,6 @@ function ChatInterface({
|
|||||||
clearTimeout(streamTimerRef.current);
|
clearTimeout(streamTimerRef.current);
|
||||||
streamTimerRef.current = null;
|
streamTimerRef.current = null;
|
||||||
}
|
}
|
||||||
streamBufferRef.current = '';
|
|
||||||
accumulatedStreamRef.current = '';
|
accumulatedStreamRef.current = '';
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
@@ -123,6 +121,7 @@ function ChatInterface({
|
|||||||
sendMessage,
|
sendMessage,
|
||||||
autoScrollToBottom,
|
autoScrollToBottom,
|
||||||
externalMessageUpdate,
|
externalMessageUpdate,
|
||||||
|
newSessionTrigger,
|
||||||
processingSessions,
|
processingSessions,
|
||||||
resetStreamingState,
|
resetStreamingState,
|
||||||
pendingViewSessionRef,
|
pendingViewSessionRef,
|
||||||
@@ -223,7 +222,6 @@ function ChatInterface({
|
|||||||
useChatRealtimeHandlers({
|
useChatRealtimeHandlers({
|
||||||
latestMessage,
|
latestMessage,
|
||||||
provider,
|
provider,
|
||||||
selectedProject,
|
|
||||||
selectedSession,
|
selectedSession,
|
||||||
currentSessionId,
|
currentSessionId,
|
||||||
setCurrentSessionId,
|
setCurrentSessionId,
|
||||||
@@ -233,13 +231,11 @@ function ChatInterface({
|
|||||||
setTokenBudget,
|
setTokenBudget,
|
||||||
setPendingPermissionRequests,
|
setPendingPermissionRequests,
|
||||||
pendingViewSessionRef,
|
pendingViewSessionRef,
|
||||||
streamBufferRef,
|
|
||||||
streamTimerRef,
|
streamTimerRef,
|
||||||
accumulatedStreamRef,
|
accumulatedStreamRef,
|
||||||
onSessionInactive,
|
onSessionInactive,
|
||||||
onSessionProcessing,
|
onSessionProcessing,
|
||||||
onSessionNotProcessing,
|
onSessionNotProcessing,
|
||||||
onReplaceTemporarySession,
|
|
||||||
onNavigateToSession,
|
onNavigateToSession,
|
||||||
onWebSocketReconnect: handleWebSocketReconnect,
|
onWebSocketReconnect: handleWebSocketReconnect,
|
||||||
sessionStore,
|
sessionStore,
|
||||||
|
|||||||
@@ -326,7 +326,7 @@ export default function ChatComposer({
|
|||||||
: permissionMode === 'acceptEdits'
|
: permissionMode === 'acceptEdits'
|
||||||
? 'border-green-300/60 bg-green-50 text-green-700 hover:bg-green-100 dark:border-green-600/40 dark:bg-green-900/15 dark:text-green-300 dark:hover:bg-green-900/25'
|
? 'border-green-300/60 bg-green-50 text-green-700 hover:bg-green-100 dark:border-green-600/40 dark:bg-green-900/15 dark:text-green-300 dark:hover:bg-green-900/25'
|
||||||
: permissionMode === 'auto'
|
: permissionMode === 'auto'
|
||||||
? 'border-amber-300/60 bg-amber-50 text-amber-700 hover:bg-amber-100 dark:border-amber-600/40 dark:bg-amber-900/15 dark:text-amber-300 dark:hover:bg-amber-900/25'
|
? 'border-blue-300/60 bg-blue-50 text-blue-700 hover:bg-blue-100 dark:border-blue-600/40 dark:bg-blue-900/15 dark:text-blue-300 dark:hover:bg-blue-900/25'
|
||||||
: permissionMode === 'bypassPermissions'
|
: permissionMode === 'bypassPermissions'
|
||||||
? 'border-orange-300/60 bg-orange-50 text-orange-700 hover:bg-orange-100 dark:border-orange-600/40 dark:bg-orange-900/15 dark:text-orange-300 dark:hover:bg-orange-900/25'
|
? 'border-orange-300/60 bg-orange-50 text-orange-700 hover:bg-orange-100 dark:border-orange-600/40 dark:bg-orange-900/15 dark:text-orange-300 dark:hover:bg-orange-900/25'
|
||||||
: 'border-primary/20 bg-primary/5 text-primary hover:bg-primary/10'
|
: 'border-primary/20 bg-primary/5 text-primary hover:bg-primary/10'
|
||||||
@@ -341,7 +341,7 @@ export default function ChatComposer({
|
|||||||
: permissionMode === 'acceptEdits'
|
: permissionMode === 'acceptEdits'
|
||||||
? 'bg-green-500'
|
? 'bg-green-500'
|
||||||
: permissionMode === 'auto'
|
: permissionMode === 'auto'
|
||||||
? 'bg-amber-500'
|
? 'bg-blue-500'
|
||||||
: permissionMode === 'bypassPermissions'
|
: permissionMode === 'bypassPermissions'
|
||||||
? 'bg-orange-500'
|
? 'bg-orange-500'
|
||||||
: 'bg-primary'
|
: 'bg-primary'
|
||||||
|
|||||||
@@ -213,13 +213,6 @@ export default function ChatMessagesPane({
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{/* Performance warning when all messages are loaded */}
|
|
||||||
{allMessagesLoaded && (
|
|
||||||
<div className="border-b border-amber-200 bg-amber-50 py-1.5 text-center text-xs text-amber-600 dark:border-amber-800 dark:bg-amber-900/20 dark:text-amber-400">
|
|
||||||
{t('session.messages.perfWarning')}
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
|
|
||||||
{/* Legacy message count indicator (for non-paginated view) */}
|
{/* Legacy message count indicator (for non-paginated view) */}
|
||||||
{!hasMoreMessages && chatMessages.length > visibleMessageCount && (
|
{!hasMoreMessages && chatMessages.length > visibleMessageCount && (
|
||||||
<div className="border-b border-gray-200 py-2 text-center text-sm text-gray-500 dark:border-gray-700 dark:text-gray-400">
|
<div className="border-b border-gray-200 py-2 text-center text-sm text-gray-500 dark:border-gray-700 dark:text-gray-400">
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
import type { Dispatch, SetStateAction } from 'react';
|
import type { Dispatch, SetStateAction } from 'react';
|
||||||
|
|
||||||
import type { AppTab, Project, ProjectSession } from '../../../types/app';
|
import type { AppTab, Project, ProjectSession } from '../../../types/app';
|
||||||
|
import type { SessionNavigationOptions } from '../../chat/types/types';
|
||||||
|
|
||||||
export type SessionLifecycleHandler = (sessionId?: string | null) => void;
|
export type SessionLifecycleHandler = (sessionId?: string | null) => void;
|
||||||
|
|
||||||
@@ -49,10 +51,10 @@ export type MainContentProps = {
|
|||||||
onSessionProcessing: SessionLifecycleHandler;
|
onSessionProcessing: SessionLifecycleHandler;
|
||||||
onSessionNotProcessing: SessionLifecycleHandler;
|
onSessionNotProcessing: SessionLifecycleHandler;
|
||||||
processingSessions: Set<string>;
|
processingSessions: Set<string>;
|
||||||
onReplaceTemporarySession: SessionLifecycleHandler;
|
onNavigateToSession: (targetSessionId: string, options?: SessionNavigationOptions) => void;
|
||||||
onNavigateToSession: (targetSessionId: string) => void;
|
|
||||||
onShowSettings: () => void;
|
onShowSettings: () => void;
|
||||||
externalMessageUpdate: number;
|
externalMessageUpdate: number;
|
||||||
|
newSessionTrigger: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type MainContentHeaderProps = {
|
export type MainContentHeaderProps = {
|
||||||
|
|||||||
@@ -47,10 +47,10 @@ function MainContent({
|
|||||||
onSessionProcessing,
|
onSessionProcessing,
|
||||||
onSessionNotProcessing,
|
onSessionNotProcessing,
|
||||||
processingSessions,
|
processingSessions,
|
||||||
onReplaceTemporarySession,
|
|
||||||
onNavigateToSession,
|
onNavigateToSession,
|
||||||
onShowSettings,
|
onShowSettings,
|
||||||
externalMessageUpdate,
|
externalMessageUpdate,
|
||||||
|
newSessionTrigger,
|
||||||
}: MainContentProps) {
|
}: MainContentProps) {
|
||||||
const { preferences } = useUiPreferences();
|
const { preferences } = useUiPreferences();
|
||||||
const { autoExpandTools, showRawParameters, showThinking, autoScrollToBottom, sendByCtrlEnter } = preferences;
|
const { autoExpandTools, showRawParameters, showThinking, autoScrollToBottom, sendByCtrlEnter } = preferences;
|
||||||
@@ -136,7 +136,6 @@ function MainContent({
|
|||||||
onSessionProcessing={onSessionProcessing}
|
onSessionProcessing={onSessionProcessing}
|
||||||
onSessionNotProcessing={onSessionNotProcessing}
|
onSessionNotProcessing={onSessionNotProcessing}
|
||||||
processingSessions={processingSessions}
|
processingSessions={processingSessions}
|
||||||
onReplaceTemporarySession={onReplaceTemporarySession}
|
|
||||||
onNavigateToSession={onNavigateToSession}
|
onNavigateToSession={onNavigateToSession}
|
||||||
onShowSettings={onShowSettings}
|
onShowSettings={onShowSettings}
|
||||||
autoExpandTools={autoExpandTools}
|
autoExpandTools={autoExpandTools}
|
||||||
@@ -145,6 +144,7 @@ function MainContent({
|
|||||||
autoScrollToBottom={autoScrollToBottom}
|
autoScrollToBottom={autoScrollToBottom}
|
||||||
sendByCtrlEnter={sendByCtrlEnter}
|
sendByCtrlEnter={sendByCtrlEnter}
|
||||||
externalMessageUpdate={externalMessageUpdate}
|
externalMessageUpdate={externalMessageUpdate}
|
||||||
|
newSessionTrigger={newSessionTrigger}
|
||||||
onShowAllTasks={tasksEnabled ? () => setActiveTab('tasks') : null}
|
onShowAllTasks={tasksEnabled ? () => setActiveTab('tasks') : null}
|
||||||
/>
|
/>
|
||||||
</ErrorBoundary>
|
</ErrorBoundary>
|
||||||
|
|||||||
@@ -5,8 +5,11 @@ import { api } from '../../../utils/api';
|
|||||||
import { usePaletteOps } from '../../../contexts/PaletteOpsContext';
|
import { usePaletteOps } from '../../../contexts/PaletteOpsContext';
|
||||||
import type { Project, ProjectSession, LLMProvider } from '../../../types/app';
|
import type { Project, ProjectSession, LLMProvider } from '../../../types/app';
|
||||||
import type {
|
import type {
|
||||||
|
ArchivedProjectListItem,
|
||||||
|
ArchivedSessionListItem,
|
||||||
DeleteProjectConfirmation,
|
DeleteProjectConfirmation,
|
||||||
ProjectSortOrder,
|
ProjectSortOrder,
|
||||||
|
SidebarSearchMode,
|
||||||
SessionDeleteConfirmation,
|
SessionDeleteConfirmation,
|
||||||
SessionWithProvider,
|
SessionWithProvider,
|
||||||
} from '../types/types';
|
} from '../types/types';
|
||||||
@@ -60,6 +63,20 @@ export type SearchProgress = {
|
|||||||
totalProjects: number;
|
totalProjects: number;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
type ArchivedSessionsApiPayload = {
|
||||||
|
success?: boolean;
|
||||||
|
data?: {
|
||||||
|
sessions?: ArchivedSessionListItem[];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
type ArchivedProjectsApiPayload = {
|
||||||
|
success?: boolean;
|
||||||
|
data?: {
|
||||||
|
projects?: ArchivedProjectListItem[];
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
type UseSidebarControllerArgs = {
|
type UseSidebarControllerArgs = {
|
||||||
projects: Project[];
|
projects: Project[];
|
||||||
selectedProject: Project | null;
|
selectedProject: Project | null;
|
||||||
@@ -112,10 +129,13 @@ export function useSidebarController({
|
|||||||
const [deleteConfirmation, setDeleteConfirmation] = useState<DeleteProjectConfirmation | null>(null);
|
const [deleteConfirmation, setDeleteConfirmation] = useState<DeleteProjectConfirmation | null>(null);
|
||||||
const [sessionDeleteConfirmation, setSessionDeleteConfirmation] = useState<SessionDeleteConfirmation | null>(null);
|
const [sessionDeleteConfirmation, setSessionDeleteConfirmation] = useState<SessionDeleteConfirmation | null>(null);
|
||||||
const [showVersionModal, setShowVersionModal] = useState(false);
|
const [showVersionModal, setShowVersionModal] = useState(false);
|
||||||
const [searchMode, setSearchMode] = useState<'projects' | 'conversations'>('projects');
|
const [searchMode, setSearchMode] = useState<SidebarSearchMode>('projects');
|
||||||
const [conversationResults, setConversationResults] = useState<ConversationSearchResults | null>(null);
|
const [conversationResults, setConversationResults] = useState<ConversationSearchResults | null>(null);
|
||||||
const [isSearching, setIsSearching] = useState(false);
|
const [isSearching, setIsSearching] = useState(false);
|
||||||
const [searchProgress, setSearchProgress] = useState<SearchProgress | null>(null);
|
const [searchProgress, setSearchProgress] = useState<SearchProgress | null>(null);
|
||||||
|
const [archivedProjects, setArchivedProjects] = useState<ArchivedProjectListItem[]>([]);
|
||||||
|
const [archivedSessions, setArchivedSessions] = useState<ArchivedSessionListItem[]>([]);
|
||||||
|
const [isArchivedSessionsLoading, setIsArchivedSessionsLoading] = useState(false);
|
||||||
const [debouncedSearchQuery, setDebouncedSearchQuery] = useState('');
|
const [debouncedSearchQuery, setDebouncedSearchQuery] = useState('');
|
||||||
const [optimisticStarByProjectId, setOptimisticStarByProjectId] = useState<Map<string, boolean>>(new Map());
|
const [optimisticStarByProjectId, setOptimisticStarByProjectId] = useState<Map<string, boolean>>(new Map());
|
||||||
const [loadingMoreProjects, setLoadingMoreProjects] = useState<Set<string>>(new Set());
|
const [loadingMoreProjects, setLoadingMoreProjects] = useState<Set<string>>(new Set());
|
||||||
@@ -201,6 +221,40 @@ export function useSidebarController({
|
|||||||
onRefreshRef.current = onRefresh;
|
onRefreshRef.current = onRefresh;
|
||||||
}, [onRefresh]);
|
}, [onRefresh]);
|
||||||
|
|
||||||
|
const fetchArchivedSessions = useCallback(async () => {
|
||||||
|
setIsArchivedSessionsLoading(true);
|
||||||
|
|
||||||
|
try {
|
||||||
|
const [archivedProjectsResponse, archivedSessionsResponse] = await Promise.all([
|
||||||
|
api.archivedProjects(),
|
||||||
|
api.getArchivedSessions(),
|
||||||
|
]);
|
||||||
|
|
||||||
|
if (!archivedProjectsResponse.ok) {
|
||||||
|
throw new Error(`Failed to load archived projects: ${archivedProjectsResponse.status}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!archivedSessionsResponse.ok) {
|
||||||
|
throw new Error(`Failed to load archived sessions: ${archivedSessionsResponse.status}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const archivedProjectsPayload = (await archivedProjectsResponse.json()) as ArchivedProjectsApiPayload;
|
||||||
|
const archivedSessionsPayload = (await archivedSessionsResponse.json()) as ArchivedSessionsApiPayload;
|
||||||
|
const nextProjects = Array.isArray(archivedProjectsPayload.data?.projects) ? archivedProjectsPayload.data.projects : [];
|
||||||
|
const archivedProjectIds = new Set(nextProjects.map((project) => project.projectId));
|
||||||
|
const nextStandaloneSessions = Array.isArray(archivedSessionsPayload.data?.sessions)
|
||||||
|
? archivedSessionsPayload.data.sessions.filter((session) => !session.projectId || !archivedProjectIds.has(session.projectId))
|
||||||
|
: [];
|
||||||
|
|
||||||
|
setArchivedProjects(nextProjects);
|
||||||
|
setArchivedSessions(nextStandaloneSessions);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('[Sidebar] Failed to load archived sessions:', error);
|
||||||
|
} finally {
|
||||||
|
setIsArchivedSessionsLoading(false);
|
||||||
|
}
|
||||||
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (migrationStartedRef.current) {
|
if (migrationStartedRef.current) {
|
||||||
return;
|
return;
|
||||||
@@ -227,6 +281,20 @@ export function useSidebarController({
|
|||||||
void migrateLegacyStars();
|
void migrateLegacyStars();
|
||||||
}, [onRefresh]);
|
}, [onRefresh]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
void fetchArchivedSessions();
|
||||||
|
}, [fetchArchivedSessions]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (searchMode !== 'archived') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Refresh archive contents when the archived tab opens so restore actions
|
||||||
|
// and background synchronizer updates are reflected without a full reload.
|
||||||
|
void fetchArchivedSessions();
|
||||||
|
}, [fetchArchivedSessions, searchMode]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setOptimisticStarByProjectId((previous) => {
|
setOptimisticStarByProjectId((previous) => {
|
||||||
if (previous.size === 0) {
|
if (previous.size === 0) {
|
||||||
@@ -519,6 +587,56 @@ export function useSidebarController({
|
|||||||
[debouncedSearchQuery, sortedProjects],
|
[debouncedSearchQuery, sortedProjects],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const filteredArchivedSessions = useMemo(() => {
|
||||||
|
const normalizedSearch = debouncedSearchQuery.trim().toLowerCase();
|
||||||
|
if (!normalizedSearch) {
|
||||||
|
return archivedSessions;
|
||||||
|
}
|
||||||
|
|
||||||
|
return archivedSessions.filter((session) => {
|
||||||
|
const searchableFields = [
|
||||||
|
session.sessionTitle,
|
||||||
|
session.projectDisplayName,
|
||||||
|
session.projectPath ?? '',
|
||||||
|
session.provider,
|
||||||
|
];
|
||||||
|
|
||||||
|
return searchableFields.some((value) => value.toLowerCase().includes(normalizedSearch));
|
||||||
|
});
|
||||||
|
}, [archivedSessions, debouncedSearchQuery]);
|
||||||
|
|
||||||
|
const filteredArchivedProjects = useMemo(() => {
|
||||||
|
const normalizedSearch = debouncedSearchQuery.trim().toLowerCase();
|
||||||
|
if (!normalizedSearch) {
|
||||||
|
return archivedProjects;
|
||||||
|
}
|
||||||
|
|
||||||
|
return archivedProjects.filter((project) => {
|
||||||
|
const projectMatches = [
|
||||||
|
project.displayName,
|
||||||
|
project.fullPath || '',
|
||||||
|
].some((value) => value.toLowerCase().includes(normalizedSearch));
|
||||||
|
|
||||||
|
if (projectMatches) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return getAllSessions(project).some((session) => {
|
||||||
|
const sessionSummary =
|
||||||
|
typeof session.summary === 'string' && session.summary.trim().length > 0
|
||||||
|
? session.summary
|
||||||
|
: typeof session.name === 'string'
|
||||||
|
? session.name
|
||||||
|
: '';
|
||||||
|
|
||||||
|
return [
|
||||||
|
sessionSummary,
|
||||||
|
session.__provider,
|
||||||
|
].some((value) => value.toLowerCase().includes(normalizedSearch));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}, [archivedProjects, debouncedSearchQuery]);
|
||||||
|
|
||||||
const startEditing = useCallback((project: Project) => {
|
const startEditing = useCallback((project: Project) => {
|
||||||
// `editingProject` is keyed by projectId so it stays stable across
|
// `editingProject` is keyed by projectId so it stays stable across
|
||||||
// display-name mutations that happen while the input is open.
|
// display-name mutations that happen while the input is open.
|
||||||
@@ -556,17 +674,26 @@ export function useSidebarController({
|
|||||||
// Kept with project/provider arguments for component wiring compatibility;
|
// Kept with project/provider arguments for component wiring compatibility;
|
||||||
// deletion now uses only `sessionId` via /api/providers/sessions/:sessionId.
|
// deletion now uses only `sessionId` via /api/providers/sessions/:sessionId.
|
||||||
(
|
(
|
||||||
projectId: string,
|
projectId: string | null,
|
||||||
sessionId: string,
|
sessionId: string,
|
||||||
sessionTitle: string,
|
sessionTitle: string,
|
||||||
provider: SessionDeleteConfirmation['provider'] = 'claude',
|
provider: SessionDeleteConfirmation['provider'] = 'claude',
|
||||||
|
options: {
|
||||||
|
isArchived?: boolean;
|
||||||
|
} = {},
|
||||||
) => {
|
) => {
|
||||||
setSessionDeleteConfirmation({ projectId, sessionId, sessionTitle, provider });
|
setSessionDeleteConfirmation({
|
||||||
|
projectId,
|
||||||
|
sessionId,
|
||||||
|
sessionTitle,
|
||||||
|
provider,
|
||||||
|
isArchived: Boolean(options.isArchived),
|
||||||
|
});
|
||||||
},
|
},
|
||||||
[],
|
[],
|
||||||
);
|
);
|
||||||
|
|
||||||
const confirmDeleteSession = useCallback(async () => {
|
const confirmDeleteSession = useCallback(async (hardDelete = false) => {
|
||||||
if (!sessionDeleteConfirmation) {
|
if (!sessionDeleteConfirmation) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -575,10 +702,11 @@ export function useSidebarController({
|
|||||||
setSessionDeleteConfirmation(null);
|
setSessionDeleteConfirmation(null);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const response = await api.deleteSession(sessionId);
|
const response = await api.deleteSession(sessionId, hardDelete);
|
||||||
|
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
onSessionDelete?.(sessionId);
|
onSessionDelete?.(sessionId);
|
||||||
|
await fetchArchivedSessions();
|
||||||
} else {
|
} else {
|
||||||
const errorText = await response.text();
|
const errorText = await response.text();
|
||||||
console.error('[Sidebar] Failed to delete session:', {
|
console.error('[Sidebar] Failed to delete session:', {
|
||||||
@@ -591,7 +719,7 @@ export function useSidebarController({
|
|||||||
console.error('[Sidebar] Error deleting session:', error);
|
console.error('[Sidebar] Error deleting session:', error);
|
||||||
alert(t('messages.deleteSessionError'));
|
alert(t('messages.deleteSessionError'));
|
||||||
}
|
}
|
||||||
}, [onSessionDelete, sessionDeleteConfirmation, t]);
|
}, [fetchArchivedSessions, onSessionDelete, sessionDeleteConfirmation, t]);
|
||||||
|
|
||||||
const requestProjectDelete = useCallback(
|
const requestProjectDelete = useCallback(
|
||||||
(project: Project) => {
|
(project: Project) => {
|
||||||
@@ -647,14 +775,88 @@ export function useSidebarController({
|
|||||||
[onProjectSelect, setCurrentProject],
|
[onProjectSelect, setCurrentProject],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const openArchivedSession = useCallback((session: ArchivedSessionListItem) => {
|
||||||
|
const activeProject = session.projectId
|
||||||
|
? projects.find((candidate) => candidate.projectId === session.projectId)
|
||||||
|
: null;
|
||||||
|
const archivedProject = session.projectId
|
||||||
|
? archivedProjects.find((candidate) => candidate.projectId === session.projectId)
|
||||||
|
: null;
|
||||||
|
const matchingProject = activeProject ?? archivedProject ?? null;
|
||||||
|
const sessionPayload: ProjectSession = {
|
||||||
|
id: session.sessionId,
|
||||||
|
summary: session.sessionTitle,
|
||||||
|
__provider: session.provider,
|
||||||
|
__projectId: matchingProject?.projectId ?? session.projectId ?? undefined,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Archived sessions still need a selected project context. Active projects
|
||||||
|
// come from the normal sidebar list, while archived-project sessions resolve
|
||||||
|
// through the archive payload loaded by this controller.
|
||||||
|
if (matchingProject) {
|
||||||
|
handleProjectSelect(matchingProject);
|
||||||
|
}
|
||||||
|
|
||||||
|
onSessionSelect(sessionPayload);
|
||||||
|
}, [archivedProjects, handleProjectSelect, onSessionSelect, projects]);
|
||||||
|
|
||||||
|
const restoreArchivedProject = useCallback(async (projectId: string) => {
|
||||||
|
try {
|
||||||
|
const response = await api.restoreProject(projectId);
|
||||||
|
if (!response.ok) {
|
||||||
|
const errorText = await response.text();
|
||||||
|
console.error('[Sidebar] Failed to restore project:', {
|
||||||
|
status: response.status,
|
||||||
|
error: errorText,
|
||||||
|
});
|
||||||
|
alert(t('messages.restoreProjectFailed', 'Failed to restore project. Please try again.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
Promise.resolve(onRefresh()),
|
||||||
|
fetchArchivedSessions(),
|
||||||
|
]);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('[Sidebar] Error restoring project:', error);
|
||||||
|
alert(t('messages.restoreProjectError', 'Error restoring project. Please try again.'));
|
||||||
|
}
|
||||||
|
}, [fetchArchivedSessions, onRefresh, t]);
|
||||||
|
|
||||||
|
const restoreArchivedSession = useCallback(async (sessionId: string) => {
|
||||||
|
try {
|
||||||
|
const response = await api.restoreSession(sessionId);
|
||||||
|
if (!response.ok) {
|
||||||
|
const errorText = await response.text();
|
||||||
|
console.error('[Sidebar] Failed to restore session:', {
|
||||||
|
status: response.status,
|
||||||
|
error: errorText,
|
||||||
|
});
|
||||||
|
alert(t('messages.restoreSessionFailed', 'Failed to restore session. Please try again.'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await Promise.all([
|
||||||
|
Promise.resolve(onRefresh()),
|
||||||
|
fetchArchivedSessions(),
|
||||||
|
]);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('[Sidebar] Error restoring session:', error);
|
||||||
|
alert(t('messages.restoreSessionError', 'Error restoring session. Please try again.'));
|
||||||
|
}
|
||||||
|
}, [fetchArchivedSessions, onRefresh, t]);
|
||||||
|
|
||||||
const refreshProjects = useCallback(async () => {
|
const refreshProjects = useCallback(async () => {
|
||||||
setIsRefreshing(true);
|
setIsRefreshing(true);
|
||||||
try {
|
try {
|
||||||
await onRefresh();
|
await Promise.all([
|
||||||
|
Promise.resolve(onRefresh()),
|
||||||
|
fetchArchivedSessions(),
|
||||||
|
]);
|
||||||
} finally {
|
} finally {
|
||||||
setIsRefreshing(false);
|
setIsRefreshing(false);
|
||||||
}
|
}
|
||||||
}, [onRefresh]);
|
}, [fetchArchivedSessions, onRefresh]);
|
||||||
|
|
||||||
const updateSessionSummary = useCallback(
|
const updateSessionSummary = useCallback(
|
||||||
// `_projectId` and `_provider` are preserved for compatibility with
|
// `_projectId` and `_provider` are preserved for compatibility with
|
||||||
@@ -712,6 +914,10 @@ export function useSidebarController({
|
|||||||
sessionDeleteConfirmation,
|
sessionDeleteConfirmation,
|
||||||
showVersionModal,
|
showVersionModal,
|
||||||
filteredProjects,
|
filteredProjects,
|
||||||
|
archivedProjects: filteredArchivedProjects,
|
||||||
|
archivedSessions: filteredArchivedSessions,
|
||||||
|
archivedSessionsCount: archivedProjects.length + archivedSessions.length,
|
||||||
|
isArchivedSessionsLoading,
|
||||||
toggleProject,
|
toggleProject,
|
||||||
handleSessionClick,
|
handleSessionClick,
|
||||||
toggleStarProject,
|
toggleStarProject,
|
||||||
@@ -726,6 +932,9 @@ export function useSidebarController({
|
|||||||
requestProjectDelete,
|
requestProjectDelete,
|
||||||
confirmDeleteProject,
|
confirmDeleteProject,
|
||||||
handleProjectSelect,
|
handleProjectSelect,
|
||||||
|
openArchivedSession,
|
||||||
|
restoreArchivedProject,
|
||||||
|
restoreArchivedSession,
|
||||||
refreshProjects,
|
refreshProjects,
|
||||||
updateSessionSummary,
|
updateSessionSummary,
|
||||||
collapseSidebar,
|
collapseSidebar,
|
||||||
|
|||||||
@@ -1,11 +1,26 @@
|
|||||||
import type { LoadingProgress, Project, ProjectSession, LLMProvider } from '../../../types/app';
|
import type { LoadingProgress, Project, ProjectSession, LLMProvider } from '../../../types/app';
|
||||||
|
|
||||||
export type ProjectSortOrder = 'name' | 'date';
|
export type ProjectSortOrder = 'name' | 'date';
|
||||||
|
export type SidebarSearchMode = 'projects' | 'conversations' | 'archived';
|
||||||
|
export type ArchivedProjectListItem = Project & { isArchived: true };
|
||||||
|
|
||||||
export type SessionWithProvider = ProjectSession & {
|
export type SessionWithProvider = ProjectSession & {
|
||||||
__provider: LLMProvider;
|
__provider: LLMProvider;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export type ArchivedSessionListItem = {
|
||||||
|
sessionId: string;
|
||||||
|
provider: LLMProvider;
|
||||||
|
projectId: string | null;
|
||||||
|
projectPath: string | null;
|
||||||
|
projectDisplayName: string;
|
||||||
|
sessionTitle: string;
|
||||||
|
createdAt: string | null;
|
||||||
|
updatedAt: string | null;
|
||||||
|
lastActivity: string | null;
|
||||||
|
isProjectArchived: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
export type DeleteProjectConfirmation = {
|
export type DeleteProjectConfirmation = {
|
||||||
project: Project;
|
project: Project;
|
||||||
sessionCount: number;
|
sessionCount: number;
|
||||||
@@ -14,10 +29,11 @@ export type DeleteProjectConfirmation = {
|
|||||||
// Delete confirmation payload used by sidebar UX. `projectId`/`provider` are
|
// Delete confirmation payload used by sidebar UX. `projectId`/`provider` are
|
||||||
// kept for wiring compatibility, while API deletion now keys only by sessionId.
|
// kept for wiring compatibility, while API deletion now keys only by sessionId.
|
||||||
export type SessionDeleteConfirmation = {
|
export type SessionDeleteConfirmation = {
|
||||||
projectId: string;
|
projectId: string | null;
|
||||||
sessionId: string;
|
sessionId: string;
|
||||||
sessionTitle: string;
|
sessionTitle: string;
|
||||||
provider: LLMProvider;
|
provider: LLMProvider;
|
||||||
|
isArchived: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type SidebarProps = {
|
export type SidebarProps = {
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import type { TFunction } from 'i18next';
|
import type { TFunction } from 'i18next';
|
||||||
|
|
||||||
import type { Project } from '../../../types/app';
|
import type { Project } from '../../../types/app';
|
||||||
import type { ProjectSortOrder, SettingsProject, SessionViewModel, SessionWithProvider } from '../types/types';
|
import type { ProjectSortOrder, SettingsProject, SessionViewModel, SessionWithProvider } from '../types/types';
|
||||||
|
|
||||||
@@ -52,44 +53,24 @@ export const clearLegacyStarredProjectIds = () => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const getCreatedTimestamp = (session: SessionWithProvider): string => {
|
||||||
|
return String(session.createdAt || session.created_at || '');
|
||||||
|
};
|
||||||
|
|
||||||
|
const getUpdatedTimestamp = (session: SessionWithProvider): string => {
|
||||||
|
return String(session.lastActivity || '');
|
||||||
|
};
|
||||||
|
|
||||||
export const getSessionDate = (session: SessionWithProvider): Date => {
|
export const getSessionDate = (session: SessionWithProvider): Date => {
|
||||||
if (session.__provider === 'cursor') {
|
return new Date(getUpdatedTimestamp(session) || getCreatedTimestamp(session) || 0);
|
||||||
return new Date(session.createdAt || 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (session.__provider === 'codex') {
|
|
||||||
return new Date(session.createdAt || session.lastActivity || 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
return new Date(session.lastActivity || session.createdAt || 0);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getSessionName = (session: SessionWithProvider, t: TFunction): string => {
|
export const getSessionName = (session: SessionWithProvider, t: TFunction): string => {
|
||||||
if (session.__provider === 'cursor') {
|
return session.summary || session.name || t('projects.newSession');
|
||||||
return session.summary || session.name || t('projects.untitledSession');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (session.__provider === 'codex') {
|
|
||||||
return session.summary || session.name || t('projects.codexSession');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (session.__provider === 'gemini') {
|
|
||||||
return session.summary || session.name || t('projects.newSession');
|
|
||||||
}
|
|
||||||
|
|
||||||
return session.summary || t('projects.newSession');
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getSessionTime = (session: SessionWithProvider): string => {
|
export const getSessionTime = (session: SessionWithProvider): string => {
|
||||||
if (session.__provider === 'cursor') {
|
return getUpdatedTimestamp(session) || getCreatedTimestamp(session);
|
||||||
return String(session.createdAt || '');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (session.__provider === 'codex') {
|
|
||||||
return String(session.createdAt || session.lastActivity || '');
|
|
||||||
}
|
|
||||||
|
|
||||||
return String(session.lastActivity || session.createdAt || '');
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export const createSessionViewModel = (
|
export const createSessionViewModel = (
|
||||||
|
|||||||
@@ -75,6 +75,10 @@ function Sidebar({
|
|||||||
sessionDeleteConfirmation,
|
sessionDeleteConfirmation,
|
||||||
showVersionModal,
|
showVersionModal,
|
||||||
filteredProjects,
|
filteredProjects,
|
||||||
|
archivedProjects,
|
||||||
|
archivedSessions,
|
||||||
|
archivedSessionsCount,
|
||||||
|
isArchivedSessionsLoading,
|
||||||
toggleProject,
|
toggleProject,
|
||||||
handleSessionClick,
|
handleSessionClick,
|
||||||
toggleStarProject,
|
toggleStarProject,
|
||||||
@@ -90,6 +94,9 @@ function Sidebar({
|
|||||||
requestProjectDelete,
|
requestProjectDelete,
|
||||||
confirmDeleteProject,
|
confirmDeleteProject,
|
||||||
handleProjectSelect,
|
handleProjectSelect,
|
||||||
|
openArchivedSession,
|
||||||
|
restoreArchivedProject,
|
||||||
|
restoreArchivedSession,
|
||||||
refreshProjects,
|
refreshProjects,
|
||||||
updateSessionSummary,
|
updateSessionSummary,
|
||||||
collapseSidebar: handleCollapseSidebar,
|
collapseSidebar: handleCollapseSidebar,
|
||||||
@@ -184,8 +191,8 @@ function Sidebar({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<SidebarModals
|
<SidebarModals
|
||||||
projects={projects}
|
projects={projects}
|
||||||
showSettings={showSettings}
|
showSettings={showSettings}
|
||||||
settingsInitialTab={settingsInitialTab}
|
settingsInitialTab={settingsInitialTab}
|
||||||
onCloseSettings={onCloseSettings}
|
onCloseSettings={onCloseSettings}
|
||||||
@@ -217,22 +224,38 @@ function Sidebar({
|
|||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
<SidebarContent
|
<SidebarContent
|
||||||
isPWA={isPWA}
|
isPWA={isPWA}
|
||||||
isMobile={isMobile}
|
isMobile={isMobile}
|
||||||
isLoading={isLoading}
|
isLoading={isLoading}
|
||||||
projects={projects}
|
projects={projects}
|
||||||
|
archivedProjects={archivedProjects}
|
||||||
|
archivedSessions={archivedSessions}
|
||||||
|
archivedSessionsCount={archivedSessionsCount}
|
||||||
|
isArchivedSessionsLoading={isArchivedSessionsLoading}
|
||||||
searchFilter={searchFilter}
|
searchFilter={searchFilter}
|
||||||
onSearchFilterChange={setSearchFilter}
|
onSearchFilterChange={setSearchFilter}
|
||||||
onClearSearchFilter={() => setSearchFilter('')}
|
onClearSearchFilter={() => setSearchFilter('')}
|
||||||
searchMode={searchMode}
|
searchMode={searchMode}
|
||||||
onSearchModeChange={(mode: 'projects' | 'conversations') => {
|
onSearchModeChange={(mode) => {
|
||||||
setSearchMode(mode);
|
setSearchMode(mode);
|
||||||
if (mode === 'projects') clearConversationResults();
|
if (mode === 'projects') clearConversationResults();
|
||||||
}}
|
}}
|
||||||
conversationResults={conversationResults}
|
conversationResults={conversationResults}
|
||||||
isSearching={isSearching}
|
isSearching={isSearching}
|
||||||
searchProgress={searchProgress}
|
searchProgress={searchProgress}
|
||||||
|
onRestoreArchivedProject={restoreArchivedProject}
|
||||||
|
onArchivedSessionClick={openArchivedSession}
|
||||||
|
onRestoreArchivedSession={restoreArchivedSession}
|
||||||
|
onDeleteArchivedSession={(session) => {
|
||||||
|
showDeleteSessionConfirmation(
|
||||||
|
session.projectId,
|
||||||
|
session.sessionId,
|
||||||
|
session.sessionTitle,
|
||||||
|
session.provider,
|
||||||
|
{ isArchived: true },
|
||||||
|
);
|
||||||
|
}}
|
||||||
onConversationResultClick={(projectId: string | null, sessionId: string, provider: string, messageTimestamp?: string | null, messageSnippet?: string | null) => {
|
onConversationResultClick={(projectId: string | null, sessionId: string, provider: string, messageTimestamp?: string | null, messageSnippet?: string | null) => {
|
||||||
// `projectId` (DB key) is the canonical identifier post-migration.
|
// `projectId` (DB key) is the canonical identifier post-migration.
|
||||||
// The server emits null when it can't resolve a project row for
|
// The server emits null when it can't resolve a project row for
|
||||||
|
|||||||
@@ -1,15 +1,16 @@
|
|||||||
import { type ReactNode } from 'react';
|
import { type ReactNode } from 'react';
|
||||||
import { Folder, MessageSquare, Search } from 'lucide-react';
|
import { Archive, Folder, MessageSquare, RotateCcw, Search, Trash2 } from 'lucide-react';
|
||||||
import type { TFunction } from 'i18next';
|
import type { TFunction } from 'i18next';
|
||||||
import { ScrollArea } from '../../../../shared/view/ui';
|
import { ScrollArea } from '../../../../shared/view/ui';
|
||||||
import type { Project } from '../../../../types/app';
|
import type { Project } from '../../../../types/app';
|
||||||
import type { ReleaseInfo } from '../../../../types/sharedTypes';
|
import type { ReleaseInfo } from '../../../../types/sharedTypes';
|
||||||
import type { ConversationSearchResults, SearchProgress } from '../../hooks/useSidebarController';
|
import type { ConversationSearchResults, SearchProgress } from '../../hooks/useSidebarController';
|
||||||
|
import type { ArchivedProjectListItem, ArchivedSessionListItem, SidebarSearchMode } from '../../types/types';
|
||||||
|
import SessionProviderLogo from '../../../llm-logo-provider/SessionProviderLogo';
|
||||||
import SidebarFooter from './SidebarFooter';
|
import SidebarFooter from './SidebarFooter';
|
||||||
import SidebarHeader from './SidebarHeader';
|
import SidebarHeader from './SidebarHeader';
|
||||||
import SidebarProjectList, { type SidebarProjectListProps } from './SidebarProjectList';
|
import SidebarProjectList, { type SidebarProjectListProps } from './SidebarProjectList';
|
||||||
|
import { getAllSessions } from '../../utils/utils';
|
||||||
type SearchMode = 'projects' | 'conversations';
|
|
||||||
|
|
||||||
function HighlightedSnippet({ snippet, highlights }: { snippet: string; highlights: { start: number; end: number }[] }) {
|
function HighlightedSnippet({ snippet, highlights }: { snippet: string; highlights: { start: number; end: number }[] }) {
|
||||||
const parts: ReactNode[] = [];
|
const parts: ReactNode[] = [];
|
||||||
@@ -35,19 +36,100 @@ function HighlightedSnippet({ snippet, highlights }: { snippet: string; highligh
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type ArchivedSessionGroup = {
|
||||||
|
key: string;
|
||||||
|
projectId: string | null;
|
||||||
|
projectDisplayName: string;
|
||||||
|
projectPath: string | null;
|
||||||
|
isProjectArchived: boolean;
|
||||||
|
sessions: ArchivedSessionListItem[];
|
||||||
|
latestActivity: string | null;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Groups archived sessions by project metadata so the archive view preserves
|
||||||
|
* the same mental model as the active sidebar: projects first, then sessions.
|
||||||
|
*/
|
||||||
|
function groupArchivedSessionsByProject(sessions: ArchivedSessionListItem[]): ArchivedSessionGroup[] {
|
||||||
|
const groups = new Map<string, ArchivedSessionGroup>();
|
||||||
|
|
||||||
|
for (const session of sessions) {
|
||||||
|
const key = session.projectId ?? session.projectPath ?? `session:${session.sessionId}`;
|
||||||
|
const existingGroup = groups.get(key);
|
||||||
|
|
||||||
|
if (existingGroup) {
|
||||||
|
existingGroup.sessions.push(session);
|
||||||
|
if (!existingGroup.latestActivity || (session.lastActivity && session.lastActivity > existingGroup.latestActivity)) {
|
||||||
|
existingGroup.latestActivity = session.lastActivity;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
groups.set(key, {
|
||||||
|
key,
|
||||||
|
projectId: session.projectId,
|
||||||
|
projectDisplayName: session.projectDisplayName,
|
||||||
|
projectPath: session.projectPath,
|
||||||
|
isProjectArchived: session.isProjectArchived,
|
||||||
|
sessions: [session],
|
||||||
|
latestActivity: session.lastActivity,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return [...groups.values()].sort((groupA, groupB) => {
|
||||||
|
const a = groupA.latestActivity ?? '';
|
||||||
|
const b = groupB.latestActivity ?? '';
|
||||||
|
return b.localeCompare(a);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function formatCompactArchivedAge(dateString: string | null): string {
|
||||||
|
if (!dateString) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
const date = new Date(dateString);
|
||||||
|
if (Number.isNaN(date.getTime())) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
const diffInMinutes = Math.floor(Math.max(0, Date.now() - date.getTime()) / (1000 * 60));
|
||||||
|
if (diffInMinutes < 1) {
|
||||||
|
return '<1m';
|
||||||
|
}
|
||||||
|
if (diffInMinutes < 60) {
|
||||||
|
return `${diffInMinutes}m`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const diffInHours = Math.floor(diffInMinutes / 60);
|
||||||
|
if (diffInHours < 24) {
|
||||||
|
return `${diffInHours}hr`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return `${Math.floor(diffInHours / 24)}d`;
|
||||||
|
}
|
||||||
|
|
||||||
type SidebarContentProps = {
|
type SidebarContentProps = {
|
||||||
isPWA: boolean;
|
isPWA: boolean;
|
||||||
isMobile: boolean;
|
isMobile: boolean;
|
||||||
isLoading: boolean;
|
isLoading: boolean;
|
||||||
projects: Project[];
|
projects: Project[];
|
||||||
|
archivedProjects: ArchivedProjectListItem[];
|
||||||
|
archivedSessions: ArchivedSessionListItem[];
|
||||||
|
archivedSessionsCount: number;
|
||||||
|
isArchivedSessionsLoading: boolean;
|
||||||
searchFilter: string;
|
searchFilter: string;
|
||||||
onSearchFilterChange: (value: string) => void;
|
onSearchFilterChange: (value: string) => void;
|
||||||
onClearSearchFilter: () => void;
|
onClearSearchFilter: () => void;
|
||||||
searchMode: SearchMode;
|
searchMode: SidebarSearchMode;
|
||||||
onSearchModeChange: (mode: SearchMode) => void;
|
onSearchModeChange: (mode: SidebarSearchMode) => void;
|
||||||
conversationResults: ConversationSearchResults | null;
|
conversationResults: ConversationSearchResults | null;
|
||||||
isSearching: boolean;
|
isSearching: boolean;
|
||||||
searchProgress: SearchProgress | null;
|
searchProgress: SearchProgress | null;
|
||||||
|
onRestoreArchivedProject: (projectId: string) => void;
|
||||||
|
onArchivedSessionClick: (session: ArchivedSessionListItem) => void;
|
||||||
|
onRestoreArchivedSession: (sessionId: string) => void;
|
||||||
|
onDeleteArchivedSession: (session: ArchivedSessionListItem) => void;
|
||||||
// Conversation result clicks pass back the DB projectId (or null when the
|
// Conversation result clicks pass back the DB projectId (or null when the
|
||||||
// server couldn't resolve it). Consumers must handle the null case.
|
// server couldn't resolve it). Consumers must handle the null case.
|
||||||
onConversationResultClick: (projectId: string | null, sessionId: string, provider: string, messageTimestamp?: string | null, messageSnippet?: string | null) => void;
|
onConversationResultClick: (projectId: string | null, sessionId: string, provider: string, messageTimestamp?: string | null, messageSnippet?: string | null) => void;
|
||||||
@@ -70,6 +152,10 @@ export default function SidebarContent({
|
|||||||
isMobile,
|
isMobile,
|
||||||
isLoading,
|
isLoading,
|
||||||
projects,
|
projects,
|
||||||
|
archivedProjects,
|
||||||
|
archivedSessions,
|
||||||
|
archivedSessionsCount,
|
||||||
|
isArchivedSessionsLoading,
|
||||||
searchFilter,
|
searchFilter,
|
||||||
onSearchFilterChange,
|
onSearchFilterChange,
|
||||||
onClearSearchFilter,
|
onClearSearchFilter,
|
||||||
@@ -78,6 +164,10 @@ export default function SidebarContent({
|
|||||||
conversationResults,
|
conversationResults,
|
||||||
isSearching,
|
isSearching,
|
||||||
searchProgress,
|
searchProgress,
|
||||||
|
onRestoreArchivedProject,
|
||||||
|
onArchivedSessionClick,
|
||||||
|
onRestoreArchivedSession,
|
||||||
|
onDeleteArchivedSession,
|
||||||
onConversationResultClick,
|
onConversationResultClick,
|
||||||
onRefresh,
|
onRefresh,
|
||||||
isRefreshing,
|
isRefreshing,
|
||||||
@@ -94,6 +184,7 @@ export default function SidebarContent({
|
|||||||
}: SidebarContentProps) {
|
}: SidebarContentProps) {
|
||||||
const showConversationSearch = searchMode === 'conversations' && searchFilter.trim().length >= 2;
|
const showConversationSearch = searchMode === 'conversations' && searchFilter.trim().length >= 2;
|
||||||
const hasPartialResults = conversationResults && conversationResults.results.length > 0;
|
const hasPartialResults = conversationResults && conversationResults.results.length > 0;
|
||||||
|
const groupedArchivedSessions = groupArchivedSessionsByProject(archivedSessions);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
@@ -105,6 +196,8 @@ export default function SidebarContent({
|
|||||||
isMobile={isMobile}
|
isMobile={isMobile}
|
||||||
isLoading={isLoading}
|
isLoading={isLoading}
|
||||||
projectsCount={projects.length}
|
projectsCount={projects.length}
|
||||||
|
archivedSessionsCount={archivedSessionsCount}
|
||||||
|
isArchivedSessionsLoading={isArchivedSessionsLoading}
|
||||||
searchFilter={searchFilter}
|
searchFilter={searchFilter}
|
||||||
onSearchFilterChange={onSearchFilterChange}
|
onSearchFilterChange={onSearchFilterChange}
|
||||||
onClearSearchFilter={onClearSearchFilter}
|
onClearSearchFilter={onClearSearchFilter}
|
||||||
@@ -214,6 +307,207 @@ export default function SidebarContent({
|
|||||||
))}
|
))}
|
||||||
</div>
|
</div>
|
||||||
) : null
|
) : null
|
||||||
|
) : searchMode === 'archived' ? (
|
||||||
|
isArchivedSessionsLoading ? (
|
||||||
|
<div className="px-4 py-12 text-center md:py-8">
|
||||||
|
<div className="mx-auto mb-4 flex h-12 w-12 items-center justify-center rounded-lg bg-muted md:mb-3">
|
||||||
|
<div className="h-6 w-6 animate-spin rounded-full border-2 border-muted-foreground border-t-transparent" />
|
||||||
|
</div>
|
||||||
|
<h3 className="mb-2 text-base font-medium text-foreground md:mb-1">
|
||||||
|
{t('archived.loadingTitle', 'Loading archive...')}
|
||||||
|
</h3>
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
{t('archived.loadingDescription', 'Fetching hidden workspaces and sessions you can restore later.')}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
) : archivedProjects.length === 0 && groupedArchivedSessions.length === 0 ? (
|
||||||
|
<div className="px-4 py-12 text-center md:py-8">
|
||||||
|
<div className="mx-auto mb-4 flex h-12 w-12 items-center justify-center rounded-lg bg-muted md:mb-3">
|
||||||
|
<Archive className="h-6 w-6 text-muted-foreground" />
|
||||||
|
</div>
|
||||||
|
<h3 className="mb-2 text-base font-medium text-foreground md:mb-1">
|
||||||
|
{archivedSessionsCount > 0
|
||||||
|
? t('archived.noMatchingSessions', 'No matching archived items')
|
||||||
|
: t('archived.emptyTitle', 'No archived items')}
|
||||||
|
</h3>
|
||||||
|
<p className="text-sm text-muted-foreground">
|
||||||
|
{archivedSessionsCount > 0
|
||||||
|
? t('archived.tryDifferentSearch', 'Try a different search term.')
|
||||||
|
: t('archived.emptyDescription', 'Archived workspaces and sessions will appear here when you hide them from the active list.')}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="space-y-3 px-2">
|
||||||
|
<div className="flex items-center justify-between px-1">
|
||||||
|
<p className="text-xs text-muted-foreground">
|
||||||
|
{`${archivedSessionsCount} ${t(
|
||||||
|
archivedSessionsCount === 1 ? 'archived.sessionCountOne' : 'archived.sessionCountOther',
|
||||||
|
archivedSessionsCount === 1 ? 'archived item' : 'archived items',
|
||||||
|
)}`}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
{archivedProjects.map((project) => {
|
||||||
|
const projectSessions = getAllSessions(project);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div key={project.projectId} className="overflow-hidden rounded-xl border border-border/70 bg-card/60 shadow-sm">
|
||||||
|
<div className="flex items-start justify-between gap-3 border-b border-border/60 px-3 py-2.5">
|
||||||
|
<div className="min-w-0">
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<Folder className="h-3.5 w-3.5 flex-shrink-0 text-muted-foreground" />
|
||||||
|
<span className="truncate text-sm font-medium text-foreground">
|
||||||
|
{project.displayName}
|
||||||
|
</span>
|
||||||
|
<span className="inline-flex items-center justify-center rounded-full bg-muted px-1 py-px text-[7px] font-medium uppercase leading-none tracking-[0.02em] text-center text-muted-foreground">
|
||||||
|
{t('archived.projectArchived', 'Project archived')}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<p className="mt-1 truncate text-xs text-muted-foreground/70" title={project.fullPath}>
|
||||||
|
{project.fullPath}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<button
|
||||||
|
className="flex h-7 w-7 flex-shrink-0 items-center justify-center rounded-lg bg-emerald-50 text-emerald-700 transition-colors hover:bg-emerald-100 dark:bg-emerald-900/20 dark:text-emerald-300 dark:hover:bg-emerald-900/30"
|
||||||
|
onClick={() => onRestoreArchivedProject(project.projectId)}
|
||||||
|
title={t('archived.restoreProject', 'Restore workspace')}
|
||||||
|
>
|
||||||
|
<RotateCcw className="h-3.5 w-3.5" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{projectSessions.length > 0 && (
|
||||||
|
<div className="divide-y divide-border/50">
|
||||||
|
{projectSessions.map((session) => (
|
||||||
|
<button
|
||||||
|
key={String(session.id)}
|
||||||
|
className="flex w-full items-center gap-2 px-3 py-2.5 text-left transition-colors hover:bg-accent/40"
|
||||||
|
onClick={() => onArchivedSessionClick({
|
||||||
|
sessionId: String(session.id),
|
||||||
|
provider: session.__provider,
|
||||||
|
projectId: project.projectId,
|
||||||
|
projectPath: project.fullPath,
|
||||||
|
projectDisplayName: project.displayName,
|
||||||
|
sessionTitle:
|
||||||
|
(typeof session.summary === 'string' && session.summary.trim().length > 0
|
||||||
|
? session.summary
|
||||||
|
: typeof session.name === 'string' && session.name.trim().length > 0
|
||||||
|
? session.name
|
||||||
|
: String(session.id)),
|
||||||
|
createdAt: typeof session.created_at === 'string' ? session.created_at : null,
|
||||||
|
updatedAt: typeof session.updated_at === 'string' ? session.updated_at : null,
|
||||||
|
lastActivity:
|
||||||
|
typeof session.lastActivity === 'string'
|
||||||
|
? session.lastActivity
|
||||||
|
: typeof session.updated_at === 'string'
|
||||||
|
? session.updated_at
|
||||||
|
: typeof session.created_at === 'string'
|
||||||
|
? session.created_at
|
||||||
|
: null,
|
||||||
|
isProjectArchived: true,
|
||||||
|
})}
|
||||||
|
>
|
||||||
|
<SessionProviderLogo provider={session.__provider} className="h-3.5 w-3.5 flex-shrink-0" />
|
||||||
|
<div className="min-w-0 flex-1">
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className="truncate text-xs font-medium text-foreground">
|
||||||
|
{(typeof session.summary === 'string' && session.summary.trim().length > 0
|
||||||
|
? session.summary
|
||||||
|
: typeof session.name === 'string' && session.name.trim().length > 0
|
||||||
|
? session.name
|
||||||
|
: String(session.id))}
|
||||||
|
</span>
|
||||||
|
<span className="ml-auto flex-shrink-0 text-[11px] text-muted-foreground">
|
||||||
|
{formatCompactArchivedAge(
|
||||||
|
typeof session.lastActivity === 'string'
|
||||||
|
? session.lastActivity
|
||||||
|
: typeof session.updated_at === 'string'
|
||||||
|
? session.updated_at
|
||||||
|
: typeof session.created_at === 'string'
|
||||||
|
? session.created_at
|
||||||
|
: null,
|
||||||
|
)}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<p className="mt-0.5 text-[11px] uppercase tracking-wide text-muted-foreground/70">
|
||||||
|
{session.__provider}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
{groupedArchivedSessions.map((group) => (
|
||||||
|
<div key={group.key} className="overflow-hidden rounded-xl border border-border/70 bg-card/60 shadow-sm">
|
||||||
|
<div className="flex items-start justify-between gap-3 border-b border-border/60 px-3 py-2.5">
|
||||||
|
<div className="min-w-0">
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<Folder className="h-3.5 w-3.5 flex-shrink-0 text-muted-foreground" />
|
||||||
|
<span className="truncate text-sm font-medium text-foreground">
|
||||||
|
{group.projectDisplayName}
|
||||||
|
</span>
|
||||||
|
{group.isProjectArchived && (
|
||||||
|
<span className="inline-flex items-center justify-center rounded-full bg-muted px-1 py-px text-[7px] font-medium uppercase leading-none tracking-[0.02em] text-center text-muted-foreground">
|
||||||
|
{t('archived.projectArchived', 'Project archived')}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
{group.projectPath && (
|
||||||
|
<p className="mt-1 truncate text-xs text-muted-foreground/70" title={group.projectPath}>
|
||||||
|
{group.projectPath}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<span className="flex-shrink-0 text-[11px] text-muted-foreground">
|
||||||
|
{group.sessions.length}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className="divide-y divide-border/50">
|
||||||
|
{group.sessions.map((session) => (
|
||||||
|
<div key={session.sessionId} className="flex items-center gap-2 px-3 py-2.5">
|
||||||
|
<button
|
||||||
|
className="flex min-w-0 flex-1 items-center gap-2 text-left transition-colors hover:text-foreground"
|
||||||
|
onClick={() => onArchivedSessionClick(session)}
|
||||||
|
>
|
||||||
|
<SessionProviderLogo provider={session.provider} className="h-3.5 w-3.5 flex-shrink-0" />
|
||||||
|
<div className="min-w-0 flex-1">
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<span className="truncate text-xs font-medium text-foreground">
|
||||||
|
{session.sessionTitle}
|
||||||
|
</span>
|
||||||
|
{session.lastActivity && (
|
||||||
|
<span className="ml-auto flex-shrink-0 text-[11px] text-muted-foreground">
|
||||||
|
{formatCompactArchivedAge(session.lastActivity)}
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<p className="mt-0.5 text-[11px] uppercase tracking-wide text-muted-foreground/70">
|
||||||
|
{session.provider}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
className="flex h-7 w-7 flex-shrink-0 items-center justify-center rounded-lg bg-emerald-50 text-emerald-700 transition-colors hover:bg-emerald-100 dark:bg-emerald-900/20 dark:text-emerald-300 dark:hover:bg-emerald-900/30"
|
||||||
|
onClick={() => onRestoreArchivedSession(session.sessionId)}
|
||||||
|
title={t('archived.restore', 'Restore session')}
|
||||||
|
>
|
||||||
|
<RotateCcw className="h-3.5 w-3.5" />
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
className="flex h-7 w-7 flex-shrink-0 items-center justify-center rounded-lg bg-red-50 text-red-700 transition-colors hover:bg-red-100 dark:bg-red-900/20 dark:text-red-300 dark:hover:bg-red-900/30"
|
||||||
|
onClick={() => onDeleteArchivedSession(session)}
|
||||||
|
title={t('archived.deletePermanently', 'Delete permanently')}
|
||||||
|
>
|
||||||
|
<Trash2 className="h-3.5 w-3.5" />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)
|
||||||
) : (
|
) : (
|
||||||
<SidebarProjectList {...projectListProps} />
|
<SidebarProjectList {...projectListProps} />
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -1,25 +1,26 @@
|
|||||||
import { Folder, FolderPlus, MessageSquare, Plus, RefreshCw, Search, X, PanelLeftClose } from 'lucide-react';
|
import { Archive, Folder, FolderPlus, MessageSquare, Plus, RefreshCw, Search, X, PanelLeftClose } from 'lucide-react';
|
||||||
import type { TFunction } from 'i18next';
|
import type { TFunction } from 'i18next';
|
||||||
import { Button, Input } from '../../../../shared/view/ui';
|
import { Button, Input, Tooltip } from '../../../../shared/view/ui';
|
||||||
import { IS_PLATFORM } from '../../../../constants/config';
|
import { IS_PLATFORM } from '../../../../constants/config';
|
||||||
import { cn } from '../../../../lib/utils';
|
import { cn } from '../../../../lib/utils';
|
||||||
|
import type { SidebarSearchMode } from '../../types/types';
|
||||||
import GitHubStarBadge from './GitHubStarBadge';
|
import GitHubStarBadge from './GitHubStarBadge';
|
||||||
|
|
||||||
const MOD_KEY =
|
const MOD_KEY =
|
||||||
typeof navigator !== 'undefined' && /Mac|iPhone|iPad/.test(navigator.platform) ? '⌘' : 'Ctrl';
|
typeof navigator !== 'undefined' && /Mac|iPhone|iPad/.test(navigator.platform) ? '⌘' : 'Ctrl';
|
||||||
|
|
||||||
type SearchMode = 'projects' | 'conversations';
|
|
||||||
|
|
||||||
type SidebarHeaderProps = {
|
type SidebarHeaderProps = {
|
||||||
isPWA: boolean;
|
isPWA: boolean;
|
||||||
isMobile: boolean;
|
isMobile: boolean;
|
||||||
isLoading: boolean;
|
isLoading: boolean;
|
||||||
projectsCount: number;
|
projectsCount: number;
|
||||||
|
archivedSessionsCount: number;
|
||||||
|
isArchivedSessionsLoading: boolean;
|
||||||
searchFilter: string;
|
searchFilter: string;
|
||||||
onSearchFilterChange: (value: string) => void;
|
onSearchFilterChange: (value: string) => void;
|
||||||
onClearSearchFilter: () => void;
|
onClearSearchFilter: () => void;
|
||||||
searchMode: SearchMode;
|
searchMode: SidebarSearchMode;
|
||||||
onSearchModeChange: (mode: SearchMode) => void;
|
onSearchModeChange: (mode: SidebarSearchMode) => void;
|
||||||
onRefresh: () => void;
|
onRefresh: () => void;
|
||||||
isRefreshing: boolean;
|
isRefreshing: boolean;
|
||||||
onCreateProject: () => void;
|
onCreateProject: () => void;
|
||||||
@@ -32,6 +33,8 @@ export default function SidebarHeader({
|
|||||||
isMobile,
|
isMobile,
|
||||||
isLoading,
|
isLoading,
|
||||||
projectsCount,
|
projectsCount,
|
||||||
|
archivedSessionsCount,
|
||||||
|
isArchivedSessionsLoading,
|
||||||
searchFilter,
|
searchFilter,
|
||||||
onSearchFilterChange,
|
onSearchFilterChange,
|
||||||
onClearSearchFilter,
|
onClearSearchFilter,
|
||||||
@@ -43,6 +46,13 @@ export default function SidebarHeader({
|
|||||||
onCollapseSidebar,
|
onCollapseSidebar,
|
||||||
t,
|
t,
|
||||||
}: SidebarHeaderProps) {
|
}: SidebarHeaderProps) {
|
||||||
|
const showSearchTools = (projectsCount > 0 || archivedSessionsCount > 0 || isArchivedSessionsLoading) && !isLoading;
|
||||||
|
const searchPlaceholder = searchMode === 'conversations'
|
||||||
|
? t('search.conversationsPlaceholder')
|
||||||
|
: searchMode === 'archived'
|
||||||
|
? t('search.archivedPlaceholder', 'Search archived sessions...')
|
||||||
|
: t('projects.searchPlaceholder');
|
||||||
|
|
||||||
const LogoBlock = () => (
|
const LogoBlock = () => (
|
||||||
<div className="flex min-w-0 items-center gap-2.5">
|
<div className="flex min-w-0 items-center gap-2.5">
|
||||||
<div className="flex h-7 w-7 flex-shrink-0 items-center justify-center rounded-lg bg-primary/90 shadow-sm">
|
<div className="flex h-7 w-7 flex-shrink-0 items-center justify-center rounded-lg bg-primary/90 shadow-sm">
|
||||||
@@ -113,7 +123,7 @@ export default function SidebarHeader({
|
|||||||
<GitHubStarBadge />
|
<GitHubStarBadge />
|
||||||
|
|
||||||
{/* Search bar */}
|
{/* Search bar */}
|
||||||
{projectsCount > 0 && !isLoading && (
|
{showSearchTools && (
|
||||||
<div className="mt-2.5 space-y-2">
|
<div className="mt-2.5 space-y-2">
|
||||||
{/* Search mode toggle */}
|
{/* Search mode toggle */}
|
||||||
<div className="flex rounded-lg bg-muted/50 p-0.5">
|
<div className="flex rounded-lg bg-muted/50 p-0.5">
|
||||||
@@ -143,12 +153,28 @@ export default function SidebarHeader({
|
|||||||
<MessageSquare className="h-3 w-3" />
|
<MessageSquare className="h-3 w-3" />
|
||||||
{t('search.modeConversations')}
|
{t('search.modeConversations')}
|
||||||
</button>
|
</button>
|
||||||
|
<Tooltip content={t('search.archiveOnlyTooltip', 'Archive only')} position="top">
|
||||||
|
<button
|
||||||
|
onClick={() => onSearchModeChange('archived')}
|
||||||
|
aria-pressed={searchMode === 'archived'}
|
||||||
|
aria-label={t('search.archiveOnlyTooltip', 'Archive only')}
|
||||||
|
title={t('search.archiveOnlyTooltip', 'Archive only')}
|
||||||
|
className={cn(
|
||||||
|
"flex items-center justify-center rounded-md px-2.5 py-1.5 text-xs font-medium transition-all",
|
||||||
|
searchMode === 'archived'
|
||||||
|
? "bg-background shadow-sm text-foreground"
|
||||||
|
: "text-muted-foreground hover:text-foreground"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<Archive className="h-3 w-3" />
|
||||||
|
</button>
|
||||||
|
</Tooltip>
|
||||||
</div>
|
</div>
|
||||||
<div className="relative">
|
<div className="relative">
|
||||||
<Search className="pointer-events-none absolute left-3 top-1/2 h-3.5 w-3.5 -translate-y-1/2 text-muted-foreground/50" />
|
<Search className="pointer-events-none absolute left-3 top-1/2 h-3.5 w-3.5 -translate-y-1/2 text-muted-foreground/50" />
|
||||||
<Input
|
<Input
|
||||||
type="text"
|
type="text"
|
||||||
placeholder={searchMode === 'conversations' ? t('search.conversationsPlaceholder') : t('projects.searchPlaceholder')}
|
placeholder={searchPlaceholder}
|
||||||
value={searchFilter}
|
value={searchFilter}
|
||||||
onChange={(event) => onSearchFilterChange(event.target.value)}
|
onChange={(event) => onSearchFilterChange(event.target.value)}
|
||||||
className="nav-search-input h-9 rounded-xl border-0 pl-9 pr-14 text-sm transition-all duration-200 placeholder:text-muted-foreground/40 focus-visible:ring-0 focus-visible:ring-offset-0"
|
className="nav-search-input h-9 rounded-xl border-0 pl-9 pr-14 text-sm transition-all duration-200 placeholder:text-muted-foreground/40 focus-visible:ring-0 focus-visible:ring-offset-0"
|
||||||
@@ -215,7 +241,7 @@ export default function SidebarHeader({
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
{/* Mobile search */}
|
{/* Mobile search */}
|
||||||
{projectsCount > 0 && !isLoading && (
|
{showSearchTools && (
|
||||||
<div className="mt-2.5 space-y-2">
|
<div className="mt-2.5 space-y-2">
|
||||||
<div className="flex rounded-lg bg-muted/50 p-0.5">
|
<div className="flex rounded-lg bg-muted/50 p-0.5">
|
||||||
<button
|
<button
|
||||||
@@ -244,12 +270,28 @@ export default function SidebarHeader({
|
|||||||
<MessageSquare className="h-3 w-3" />
|
<MessageSquare className="h-3 w-3" />
|
||||||
{t('search.modeConversations')}
|
{t('search.modeConversations')}
|
||||||
</button>
|
</button>
|
||||||
|
<Tooltip content={t('search.archiveOnlyTooltip', 'Archive only')} position="top">
|
||||||
|
<button
|
||||||
|
onClick={() => onSearchModeChange('archived')}
|
||||||
|
aria-pressed={searchMode === 'archived'}
|
||||||
|
aria-label={t('search.archiveOnlyTooltip', 'Archive only')}
|
||||||
|
title={t('search.archiveOnlyTooltip', 'Archive only')}
|
||||||
|
className={cn(
|
||||||
|
"flex items-center justify-center rounded-md px-2.5 py-1.5 text-xs font-medium transition-all",
|
||||||
|
searchMode === 'archived'
|
||||||
|
? "bg-background shadow-sm text-foreground"
|
||||||
|
: "text-muted-foreground hover:text-foreground"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<Archive className="h-3 w-3" />
|
||||||
|
</button>
|
||||||
|
</Tooltip>
|
||||||
</div>
|
</div>
|
||||||
<div className="relative">
|
<div className="relative">
|
||||||
<Search className="pointer-events-none absolute left-3 top-1/2 h-4 w-4 -translate-y-1/2 text-muted-foreground/50" />
|
<Search className="pointer-events-none absolute left-3 top-1/2 h-4 w-4 -translate-y-1/2 text-muted-foreground/50" />
|
||||||
<Input
|
<Input
|
||||||
type="text"
|
type="text"
|
||||||
placeholder={searchMode === 'conversations' ? t('search.conversationsPlaceholder') : t('projects.searchPlaceholder')}
|
placeholder={searchPlaceholder}
|
||||||
value={searchFilter}
|
value={searchFilter}
|
||||||
onChange={(event) => onSearchFilterChange(event.target.value)}
|
onChange={(event) => onSearchFilterChange(event.target.value)}
|
||||||
className="nav-search-input h-10 rounded-xl border-0 pl-10 pr-9 text-sm transition-all duration-200 placeholder:text-muted-foreground/40 focus-visible:ring-0 focus-visible:ring-offset-0"
|
className="nav-search-input h-10 rounded-xl border-0 pl-10 pr-9 text-sm transition-all duration-200 placeholder:text-muted-foreground/40 focus-visible:ring-0 focus-visible:ring-offset-0"
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ type SidebarModalsProps = {
|
|||||||
onConfirmDeleteProject: (deleteData?: boolean) => void;
|
onConfirmDeleteProject: (deleteData?: boolean) => void;
|
||||||
sessionDeleteConfirmation: SessionDeleteConfirmation | null;
|
sessionDeleteConfirmation: SessionDeleteConfirmation | null;
|
||||||
onCancelDeleteSession: () => void;
|
onCancelDeleteSession: () => void;
|
||||||
onConfirmDeleteSession: () => void;
|
onConfirmDeleteSession: (hardDelete?: boolean) => void;
|
||||||
showVersionModal: boolean;
|
showVersionModal: boolean;
|
||||||
onCloseVersionModal: () => void;
|
onCloseVersionModal: () => void;
|
||||||
releaseInfo: ReleaseInfo | null;
|
releaseInfo: ReleaseInfo | null;
|
||||||
@@ -133,7 +133,7 @@ export default function SidebarModals({
|
|||||||
onClick={() => onConfirmDeleteProject(false)}
|
onClick={() => onConfirmDeleteProject(false)}
|
||||||
>
|
>
|
||||||
<EyeOff className="mr-2 h-4 w-4" />
|
<EyeOff className="mr-2 h-4 w-4" />
|
||||||
{t('deleteConfirmation.removeFromSidebar')}
|
{t('deleteConfirmation.archiveProject', 'Archive project')}
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
variant="destructive"
|
variant="destructive"
|
||||||
@@ -173,22 +173,34 @@ export default function SidebarModals({
|
|||||||
?
|
?
|
||||||
</p>
|
</p>
|
||||||
<p className="mt-3 text-xs text-muted-foreground">
|
<p className="mt-3 text-xs text-muted-foreground">
|
||||||
{t('deleteConfirmation.cannotUndo')}
|
{sessionDeleteConfirmation.isArchived
|
||||||
|
? t('deleteConfirmation.archivedSessionNotice', 'This session is already archived. You can keep it hidden or delete it permanently.')
|
||||||
|
: t('deleteConfirmation.archiveSessionNotice', 'Archive keeps the session out of the active list while preserving its history.')}
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex gap-3 border-t border-border bg-muted/30 p-4">
|
<div className="flex flex-col gap-2 border-t border-border bg-muted/30 p-4">
|
||||||
<Button variant="outline" className="flex-1" onClick={onCancelDeleteSession}>
|
{!sessionDeleteConfirmation.isArchived && (
|
||||||
{t('actions.cancel')}
|
<Button
|
||||||
</Button>
|
variant="outline"
|
||||||
|
className="w-full justify-start"
|
||||||
|
onClick={() => onConfirmDeleteSession(false)}
|
||||||
|
>
|
||||||
|
<EyeOff className="mr-2 h-4 w-4" />
|
||||||
|
{t('deleteConfirmation.archiveSession', 'Archive session')}
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
<Button
|
<Button
|
||||||
variant="destructive"
|
variant="destructive"
|
||||||
className="flex-1 bg-red-600 text-white hover:bg-red-700"
|
className="w-full justify-start bg-red-600 text-white hover:bg-red-700"
|
||||||
onClick={onConfirmDeleteSession}
|
onClick={() => onConfirmDeleteSession(true)}
|
||||||
>
|
>
|
||||||
<Trash2 className="mr-2 h-4 w-4" />
|
<Trash2 className="mr-2 h-4 w-4" />
|
||||||
{t('actions.delete')}
|
{t('deleteConfirmation.deleteSessionPermanently', 'Delete permanently')}
|
||||||
|
</Button>
|
||||||
|
<Button variant="ghost" className="w-full" onClick={onCancelDeleteSession}>
|
||||||
|
{t('actions.cancel')}
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -239,7 +239,7 @@ export default function SidebarSessionItem({
|
|||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
requestDeleteSession();
|
requestDeleteSession();
|
||||||
}}
|
}}
|
||||||
title={t('tooltips.deleteSession')}
|
title={t('tooltips.deleteSessionOptions', 'Archive or permanently delete this session')}
|
||||||
>
|
>
|
||||||
<Trash2 className="h-3 w-3 text-red-600 dark:text-red-400" />
|
<Trash2 className="h-3 w-3 text-red-600 dark:text-red-400" />
|
||||||
</button>
|
</button>
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import { api } from '../utils/api';
|
|||||||
import type {
|
import type {
|
||||||
AppSocketMessage,
|
AppSocketMessage,
|
||||||
AppTab,
|
AppTab,
|
||||||
|
LLMProvider,
|
||||||
LoadingProgress,
|
LoadingProgress,
|
||||||
Project,
|
Project,
|
||||||
ProjectSession,
|
ProjectSession,
|
||||||
@@ -261,6 +262,27 @@ export function useProjectsState({
|
|||||||
const [showSettings, setShowSettings] = useState(false);
|
const [showSettings, setShowSettings] = useState(false);
|
||||||
const [settingsInitialTab, setSettingsInitialTab] = useState('agents');
|
const [settingsInitialTab, setSettingsInitialTab] = useState('agents');
|
||||||
const [externalMessageUpdate, setExternalMessageUpdate] = useState(0);
|
const [externalMessageUpdate, setExternalMessageUpdate] = useState(0);
|
||||||
|
/**
|
||||||
|
* `newSessionTrigger` is an explicit, monotonic intent signal for user-driven
|
||||||
|
* New Session actions.
|
||||||
|
*
|
||||||
|
* It exists because `handleNewSession` can be invoked while the app is already in
|
||||||
|
* the same visible state (`selectedSession === null`, `activeTab === 'chat'`,
|
||||||
|
* route already `/`). In that case, React/router updates are idempotent and no
|
||||||
|
* downstream reset logic runs.
|
||||||
|
*
|
||||||
|
* Usage across the codebase:
|
||||||
|
* 1) Produced here in `handleNewSession` via increment (always changes).
|
||||||
|
* 2) Returned from this hook and threaded through:
|
||||||
|
* useProjectsState -> AppContent -> MainContent -> ChatInterface.
|
||||||
|
* 3) Consumed in `useChatSessionState` as an effect dependency to forcibly clear
|
||||||
|
* chat-local state (`currentSessionId`, pending draft message, streaming flags,
|
||||||
|
* pending session storage keys, pagination/scroll artifacts).
|
||||||
|
*
|
||||||
|
* Keeping this signal dedicated avoids coupling resets to unrelated counters/events
|
||||||
|
* (for example websocket/project refresh updates) that could cause accidental resets.
|
||||||
|
*/
|
||||||
|
const [newSessionTrigger, setNewSessionTrigger] = useState(0);
|
||||||
|
|
||||||
const loadingProgressTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
|
const loadingProgressTimeoutRef = useRef<ReturnType<typeof setTimeout> | null>(null);
|
||||||
const lastHandledMessageRef = useRef<AppSocketMessage | null>(null);
|
const lastHandledMessageRef = useRef<AppSocketMessage | null>(null);
|
||||||
@@ -413,9 +435,7 @@ export function useProjectsState({
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const hasActiveSession =
|
const hasActiveSession = Boolean(selectedSession && activeSessions.has(selectedSession.id));
|
||||||
(selectedSession && activeSessions.has(selectedSession.id)) ||
|
|
||||||
(activeSessions.size > 0 && Array.from(activeSessions).some((id) => id.startsWith('new-session-')));
|
|
||||||
|
|
||||||
const updatedProjectsWithTaskMaster = mergeTaskMasterCache(projectsMessage.projects, projects);
|
const updatedProjectsWithTaskMaster = mergeTaskMasterCache(projectsMessage.projects, projects);
|
||||||
const updatedProjects = mergeExpandedSessionPages(projects, updatedProjectsWithTaskMaster);
|
const updatedProjects = mergeExpandedSessionPages(projects, updatedProjectsWithTaskMaster);
|
||||||
@@ -536,7 +556,42 @@ export function useProjectsState({
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, [sessionId, projects, selectedProject?.projectId, selectedSession?.id, selectedSession?.__provider]);
|
|
||||||
|
// Session id is in the URL but not yet present on any project payload (common
|
||||||
|
// right after `session_created` + navigate, before the next projects refresh).
|
||||||
|
// Without a `selectedSession`, chat state clears `currentSessionId` and the
|
||||||
|
// UI stops reading the session store even though messages stream under this id.
|
||||||
|
if (selectedSession?.id === sessionId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!selectedProject) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let providerFromStorage: string | null = null;
|
||||||
|
try {
|
||||||
|
providerFromStorage = localStorage.getItem('selected-provider');
|
||||||
|
} catch {
|
||||||
|
providerFromStorage = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const normalizedProvider: LLMProvider =
|
||||||
|
providerFromStorage === 'cursor'
|
||||||
|
? 'cursor'
|
||||||
|
: providerFromStorage === 'codex'
|
||||||
|
? 'codex'
|
||||||
|
: providerFromStorage === 'gemini'
|
||||||
|
? 'gemini'
|
||||||
|
: 'claude';
|
||||||
|
|
||||||
|
setSelectedSession({
|
||||||
|
id: sessionId,
|
||||||
|
__provider: normalizedProvider,
|
||||||
|
__projectId: selectedProject.projectId,
|
||||||
|
summary: '',
|
||||||
|
});
|
||||||
|
}, [sessionId, projects, selectedProject, selectedSession?.id, selectedSession?.__provider]);
|
||||||
|
|
||||||
const handleProjectSelect = useCallback(
|
const handleProjectSelect = useCallback(
|
||||||
(project: Project) => {
|
(project: Project) => {
|
||||||
@@ -587,6 +642,7 @@ export function useProjectsState({
|
|||||||
setSelectedProject(project);
|
setSelectedProject(project);
|
||||||
setSelectedSession(null);
|
setSelectedSession(null);
|
||||||
setActiveTab('chat');
|
setActiveTab('chat');
|
||||||
|
setNewSessionTrigger((previous) => previous + 1);
|
||||||
navigate('/');
|
navigate('/');
|
||||||
|
|
||||||
if (isMobile) {
|
if (isMobile) {
|
||||||
@@ -806,6 +862,7 @@ export function useProjectsState({
|
|||||||
showSettings,
|
showSettings,
|
||||||
settingsInitialTab,
|
settingsInitialTab,
|
||||||
externalMessageUpdate,
|
externalMessageUpdate,
|
||||||
|
newSessionTrigger,
|
||||||
setActiveTab,
|
setActiveTab,
|
||||||
setSidebarOpen,
|
setSidebarOpen,
|
||||||
setIsInputFocused,
|
setIsInputFocused,
|
||||||
|
|||||||
@@ -44,23 +44,6 @@ export function useSessionProtection() {
|
|||||||
});
|
});
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const replaceTemporarySession = useCallback((realSessionId?: string | null) => {
|
|
||||||
if (!realSessionId) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
setActiveSessions((prev) => {
|
|
||||||
const next = new Set<string>();
|
|
||||||
for (const sessionId of prev) {
|
|
||||||
if (!sessionId.startsWith('new-session-')) {
|
|
||||||
next.add(sessionId);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
next.add(realSessionId);
|
|
||||||
return next;
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
activeSessions,
|
activeSessions,
|
||||||
processingSessions,
|
processingSessions,
|
||||||
@@ -68,6 +51,5 @@ export function useSessionProtection() {
|
|||||||
markSessionAsInactive,
|
markSessionAsInactive,
|
||||||
markSessionAsProcessing,
|
markSessionAsProcessing,
|
||||||
markSessionAsNotProcessing,
|
markSessionAsNotProcessing,
|
||||||
replaceTemporarySession,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -40,6 +40,20 @@ export interface NormalizedMessage {
|
|||||||
// kind-specific fields (flat for simplicity)
|
// kind-specific fields (flat for simplicity)
|
||||||
role?: 'user' | 'assistant';
|
role?: 'user' | 'assistant';
|
||||||
content?: string;
|
content?: string;
|
||||||
|
/**
|
||||||
|
* Mirrors optional transcript metadata from the server.
|
||||||
|
*
|
||||||
|
* These fields are currently used by Claude history normalization so local
|
||||||
|
* slash commands, local stdout, and compact summaries do not disappear when
|
||||||
|
* the session store hydrates from REST history.
|
||||||
|
*/
|
||||||
|
displayText?: string;
|
||||||
|
commandName?: string;
|
||||||
|
commandMessage?: string;
|
||||||
|
commandArgs?: string;
|
||||||
|
isLocalCommand?: boolean;
|
||||||
|
isLocalCommandStdout?: boolean;
|
||||||
|
isCompactSummary?: boolean;
|
||||||
images?: string[];
|
images?: string[];
|
||||||
toolName?: string;
|
toolName?: string;
|
||||||
toolInput?: unknown;
|
toolInput?: unknown;
|
||||||
@@ -104,17 +118,126 @@ function createEmptySlot(): SessionSlot {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Compute merged messages: server + realtime, deduped by id.
|
* Compute merged messages: server + realtime, deduped by id and adjacent
|
||||||
* Server messages take priority (they're the persisted source of truth).
|
* assistant echo (same trimmed text), so finalized stream rows do not stack
|
||||||
* Realtime messages that aren't yet in server stay (in-flight streaming).
|
* on top of the persisted copy before realtime is cleared.
|
||||||
*/
|
*/
|
||||||
|
function userTextFingerprint(m: NormalizedMessage): string | null {
|
||||||
|
if (m.kind !== 'text' || m.role !== 'user') return null;
|
||||||
|
const t = (m.content || '').trim();
|
||||||
|
return t.length > 0 ? t : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* After `finalizeStreaming`, the client holds a synthetic assistant `text` row
|
||||||
|
* while the sessions API soon returns the same reply with a different id.
|
||||||
|
* Those sit back-to-back in merged order and look like duplicate bubbles until
|
||||||
|
* `refreshFromServer` clears realtime. Collapse same-text assistant rows and
|
||||||
|
* stream_placeholder → text when content matches.
|
||||||
|
*/
|
||||||
|
function dedupeAdjacentAssistantEchoes(merged: NormalizedMessage[]): NormalizedMessage[] {
|
||||||
|
const out: NormalizedMessage[] = [];
|
||||||
|
for (const m of merged) {
|
||||||
|
const prev = out[out.length - 1];
|
||||||
|
if (prev) {
|
||||||
|
if (prev.kind === 'stream_delta' && m.kind === 'text' && m.role === 'assistant') {
|
||||||
|
const ps = (prev.content || '').trim();
|
||||||
|
const ms = (m.content || '').trim();
|
||||||
|
if (ps.length > 0 && ps === ms) {
|
||||||
|
out[out.length - 1] = m;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
prev.kind === 'text'
|
||||||
|
&& m.kind === 'text'
|
||||||
|
&& prev.role === 'assistant'
|
||||||
|
&& m.role === 'assistant'
|
||||||
|
) {
|
||||||
|
const ms = (m.content || '').trim();
|
||||||
|
if (ms.length > 0 && ms === (prev.content || '').trim()) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
out.push(m);
|
||||||
|
}
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
function computeMerged(server: NormalizedMessage[], realtime: NormalizedMessage[]): NormalizedMessage[] {
|
function computeMerged(server: NormalizedMessage[], realtime: NormalizedMessage[]): NormalizedMessage[] {
|
||||||
if (realtime.length === 0) return server;
|
if (realtime.length === 0) return server;
|
||||||
if (server.length === 0) return realtime;
|
if (server.length === 0) return dedupeAdjacentAssistantEchoes(realtime);
|
||||||
const serverIds = new Set(server.map(m => m.id));
|
const serverIds = new Set(server.map(m => m.id));
|
||||||
const extra = realtime.filter(m => !serverIds.has(m.id));
|
const serverUserTexts = new Set(
|
||||||
|
server.map(userTextFingerprint).filter((t): t is string => t !== null),
|
||||||
|
);
|
||||||
|
const extra = realtime.filter((m) => {
|
||||||
|
if (serverIds.has(m.id)) return false;
|
||||||
|
// Optimistic user rows use `local_*` ids; once the same text exists on the
|
||||||
|
// server-backed copy, drop the realtime echo to avoid duplicate bubbles.
|
||||||
|
if (m.id.startsWith('local_')) {
|
||||||
|
const fp = userTextFingerprint(m);
|
||||||
|
if (fp && serverUserTexts.has(fp)) return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
});
|
||||||
if (extra.length === 0) return server;
|
if (extra.length === 0) return server;
|
||||||
return [...server, ...extra];
|
return dedupeAdjacentAssistantEchoes([...server, ...extra]);
|
||||||
|
}
|
||||||
|
|
||||||
|
function compareMessagesByTimestamp(left: NormalizedMessage, right: NormalizedMessage): number {
|
||||||
|
const leftTime = Date.parse(left.timestamp);
|
||||||
|
const rightTime = Date.parse(right.timestamp);
|
||||||
|
|
||||||
|
if (Number.isNaN(leftTime) || Number.isNaN(rightTime) || leftTime === rightTime) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return leftTime - rightTime;
|
||||||
|
}
|
||||||
|
|
||||||
|
function rewriteMessageSessionId(
|
||||||
|
msg: NormalizedMessage,
|
||||||
|
fromSessionId: string,
|
||||||
|
toSessionId: string,
|
||||||
|
): NormalizedMessage {
|
||||||
|
const streamingSourceId = `__streaming_${fromSessionId}`;
|
||||||
|
const nextId = msg.id === streamingSourceId ? `__streaming_${toSessionId}` : msg.id;
|
||||||
|
|
||||||
|
if (msg.sessionId === toSessionId && nextId === msg.id) {
|
||||||
|
return msg;
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
...msg,
|
||||||
|
id: nextId,
|
||||||
|
sessionId: toSessionId,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function mergeMessagesById(
|
||||||
|
existing: NormalizedMessage[],
|
||||||
|
incoming: NormalizedMessage[],
|
||||||
|
): NormalizedMessage[] {
|
||||||
|
if (existing.length === 0) return incoming;
|
||||||
|
if (incoming.length === 0) return existing;
|
||||||
|
|
||||||
|
const merged = [...existing, ...incoming];
|
||||||
|
const deduped: NormalizedMessage[] = [];
|
||||||
|
const seen = new Set<string>();
|
||||||
|
|
||||||
|
for (const msg of merged) {
|
||||||
|
if (seen.has(msg.id)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
seen.add(msg.id);
|
||||||
|
deduped.push(msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
deduped.sort(compareMessagesByTimestamp);
|
||||||
|
return deduped;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -141,28 +264,59 @@ const MAX_REALTIME_MESSAGES = 500;
|
|||||||
|
|
||||||
export function useSessionStore() {
|
export function useSessionStore() {
|
||||||
const storeRef = useRef(new Map<string, SessionSlot>());
|
const storeRef = useRef(new Map<string, SessionSlot>());
|
||||||
|
const sessionAliasesRef = useRef(new Map<string, string>());
|
||||||
const activeSessionIdRef = useRef<string | null>(null);
|
const activeSessionIdRef = useRef<string | null>(null);
|
||||||
// Bump to force re-render — only when the active session's data changes
|
// Bump to force re-render — only when the active session's data changes
|
||||||
const [, setTick] = useState(0);
|
const [, setTick] = useState(0);
|
||||||
const notify = useCallback((sessionId: string) => {
|
const notify = useCallback((sessionId: string) => {
|
||||||
if (sessionId === activeSessionIdRef.current) {
|
const aliases = sessionAliasesRef.current;
|
||||||
|
let resolvedSessionId = sessionId;
|
||||||
|
const visited = new Set<string>();
|
||||||
|
|
||||||
|
while (aliases.has(resolvedSessionId) && !visited.has(resolvedSessionId)) {
|
||||||
|
visited.add(resolvedSessionId);
|
||||||
|
resolvedSessionId = aliases.get(resolvedSessionId)!;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (resolvedSessionId === activeSessionIdRef.current) {
|
||||||
setTick(n => n + 1);
|
setTick(n => n + 1);
|
||||||
}
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const setActiveSession = useCallback((sessionId: string | null) => {
|
const resolveSessionId = useCallback((sessionId: string | null | undefined): string | null => {
|
||||||
activeSessionIdRef.current = sessionId;
|
if (!sessionId) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const aliases = sessionAliasesRef.current;
|
||||||
|
let resolvedSessionId = sessionId;
|
||||||
|
const visited = new Set<string>();
|
||||||
|
|
||||||
|
while (aliases.has(resolvedSessionId) && !visited.has(resolvedSessionId)) {
|
||||||
|
visited.add(resolvedSessionId);
|
||||||
|
resolvedSessionId = aliases.get(resolvedSessionId)!;
|
||||||
|
}
|
||||||
|
|
||||||
|
return resolvedSessionId;
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
|
const setActiveSession = useCallback((sessionId: string | null) => {
|
||||||
|
activeSessionIdRef.current = resolveSessionId(sessionId);
|
||||||
|
}, [resolveSessionId]);
|
||||||
|
|
||||||
const getSlot = useCallback((sessionId: string): SessionSlot => {
|
const getSlot = useCallback((sessionId: string): SessionSlot => {
|
||||||
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
const store = storeRef.current;
|
const store = storeRef.current;
|
||||||
if (!store.has(sessionId)) {
|
if (!store.has(resolvedSessionId)) {
|
||||||
store.set(sessionId, createEmptySlot());
|
store.set(resolvedSessionId, createEmptySlot());
|
||||||
}
|
}
|
||||||
return store.get(sessionId)!;
|
return store.get(resolvedSessionId)!;
|
||||||
}, []);
|
}, [resolveSessionId]);
|
||||||
|
|
||||||
const has = useCallback((sessionId: string) => storeRef.current.has(sessionId), []);
|
const has = useCallback((sessionId: string) => {
|
||||||
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
|
return storeRef.current.has(resolvedSessionId);
|
||||||
|
}, [resolveSessionId]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fetch messages from the provider sessions endpoint and populate serverMessages.
|
* Fetch messages from the provider sessions endpoint and populate serverMessages.
|
||||||
@@ -179,9 +333,10 @@ export function useSessionStore() {
|
|||||||
offset?: number;
|
offset?: number;
|
||||||
} = {},
|
} = {},
|
||||||
) => {
|
) => {
|
||||||
const slot = getSlot(sessionId);
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
|
const slot = getSlot(resolvedSessionId);
|
||||||
slot.status = 'loading';
|
slot.status = 'loading';
|
||||||
notify(sessionId);
|
notify(resolvedSessionId);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const params = new URLSearchParams();
|
const params = new URLSearchParams();
|
||||||
@@ -191,7 +346,7 @@ export function useSessionStore() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const qs = params.toString();
|
const qs = params.toString();
|
||||||
const url = `/api/providers/sessions/${encodeURIComponent(sessionId)}/messages${qs ? `?${qs}` : ''}`;
|
const url = `/api/providers/sessions/${encodeURIComponent(resolvedSessionId)}/messages${qs ? `?${qs}` : ''}`;
|
||||||
const response = await authenticatedFetch(url);
|
const response = await authenticatedFetch(url);
|
||||||
|
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
@@ -212,15 +367,15 @@ export function useSessionStore() {
|
|||||||
slot.tokenUsage = data.tokenUsage;
|
slot.tokenUsage = data.tokenUsage;
|
||||||
}
|
}
|
||||||
|
|
||||||
notify(sessionId);
|
notify(resolvedSessionId);
|
||||||
return slot;
|
return slot;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(`[SessionStore] fetch failed for ${sessionId}:`, error);
|
console.error(`[SessionStore] fetch failed for ${resolvedSessionId}:`, error);
|
||||||
slot.status = 'error';
|
slot.status = 'error';
|
||||||
notify(sessionId);
|
notify(resolvedSessionId);
|
||||||
return slot;
|
return slot;
|
||||||
}
|
}
|
||||||
}, [getSlot, notify]);
|
}, [getSlot, notify, resolveSessionId]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load older (paginated) messages and prepend to serverMessages.
|
* Load older (paginated) messages and prepend to serverMessages.
|
||||||
@@ -234,7 +389,8 @@ export function useSessionStore() {
|
|||||||
limit?: number;
|
limit?: number;
|
||||||
} = {},
|
} = {},
|
||||||
) => {
|
) => {
|
||||||
const slot = getSlot(sessionId);
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
|
const slot = getSlot(resolvedSessionId);
|
||||||
if (!slot.hasMore) return slot;
|
if (!slot.hasMore) return slot;
|
||||||
|
|
||||||
const params = new URLSearchParams();
|
const params = new URLSearchParams();
|
||||||
@@ -243,7 +399,7 @@ export function useSessionStore() {
|
|||||||
params.append('offset', String(slot.offset));
|
params.append('offset', String(slot.offset));
|
||||||
|
|
||||||
const qs = params.toString();
|
const qs = params.toString();
|
||||||
const url = `/api/providers/sessions/${encodeURIComponent(sessionId)}/messages${qs ? `?${qs}` : ''}`;
|
const url = `/api/providers/sessions/${encodeURIComponent(resolvedSessionId)}/messages${qs ? `?${qs}` : ''}`;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const response = await authenticatedFetch(url);
|
const response = await authenticatedFetch(url);
|
||||||
@@ -256,43 +412,54 @@ export function useSessionStore() {
|
|||||||
slot.hasMore = Boolean(data.hasMore);
|
slot.hasMore = Boolean(data.hasMore);
|
||||||
slot.offset = slot.offset + olderMessages.length;
|
slot.offset = slot.offset + olderMessages.length;
|
||||||
recomputeMergedIfNeeded(slot);
|
recomputeMergedIfNeeded(slot);
|
||||||
notify(sessionId);
|
notify(resolvedSessionId);
|
||||||
return slot;
|
return slot;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(`[SessionStore] fetchMore failed for ${sessionId}:`, error);
|
console.error(`[SessionStore] fetchMore failed for ${resolvedSessionId}:`, error);
|
||||||
return slot;
|
return slot;
|
||||||
}
|
}
|
||||||
}, [getSlot, notify]);
|
}, [getSlot, notify, resolveSessionId]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Append a realtime (WebSocket) message to the correct session slot.
|
* Append a realtime (WebSocket) message to the correct session slot.
|
||||||
* This works regardless of which session is actively viewed.
|
* This works regardless of which session is actively viewed.
|
||||||
*/
|
*/
|
||||||
const appendRealtime = useCallback((sessionId: string, msg: NormalizedMessage) => {
|
const appendRealtime = useCallback((sessionId: string, msg: NormalizedMessage) => {
|
||||||
const slot = getSlot(sessionId);
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
let updated = [...slot.realtimeMessages, msg];
|
const slot = getSlot(resolvedSessionId);
|
||||||
|
const normalizedMessage =
|
||||||
|
msg.sessionId === resolvedSessionId
|
||||||
|
? msg
|
||||||
|
: { ...msg, sessionId: resolvedSessionId };
|
||||||
|
let updated = [...slot.realtimeMessages, normalizedMessage];
|
||||||
if (updated.length > MAX_REALTIME_MESSAGES) {
|
if (updated.length > MAX_REALTIME_MESSAGES) {
|
||||||
updated = updated.slice(-MAX_REALTIME_MESSAGES);
|
updated = updated.slice(-MAX_REALTIME_MESSAGES);
|
||||||
}
|
}
|
||||||
slot.realtimeMessages = updated;
|
slot.realtimeMessages = updated;
|
||||||
recomputeMergedIfNeeded(slot);
|
recomputeMergedIfNeeded(slot);
|
||||||
notify(sessionId);
|
notify(resolvedSessionId);
|
||||||
}, [getSlot, notify]);
|
}, [getSlot, notify, resolveSessionId]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Append multiple realtime messages at once (batch).
|
* Append multiple realtime messages at once (batch).
|
||||||
*/
|
*/
|
||||||
const appendRealtimeBatch = useCallback((sessionId: string, msgs: NormalizedMessage[]) => {
|
const appendRealtimeBatch = useCallback((sessionId: string, msgs: NormalizedMessage[]) => {
|
||||||
if (msgs.length === 0) return;
|
if (msgs.length === 0) return;
|
||||||
const slot = getSlot(sessionId);
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
let updated = [...slot.realtimeMessages, ...msgs];
|
const slot = getSlot(resolvedSessionId);
|
||||||
|
const normalizedMessages = msgs.map((msg) =>
|
||||||
|
msg.sessionId === resolvedSessionId
|
||||||
|
? msg
|
||||||
|
: { ...msg, sessionId: resolvedSessionId },
|
||||||
|
);
|
||||||
|
let updated = [...slot.realtimeMessages, ...normalizedMessages];
|
||||||
if (updated.length > MAX_REALTIME_MESSAGES) {
|
if (updated.length > MAX_REALTIME_MESSAGES) {
|
||||||
updated = updated.slice(-MAX_REALTIME_MESSAGES);
|
updated = updated.slice(-MAX_REALTIME_MESSAGES);
|
||||||
}
|
}
|
||||||
slot.realtimeMessages = updated;
|
slot.realtimeMessages = updated;
|
||||||
recomputeMergedIfNeeded(slot);
|
recomputeMergedIfNeeded(slot);
|
||||||
notify(sessionId);
|
notify(resolvedSessionId);
|
||||||
}, [getSlot, notify]);
|
}, [getSlot, notify, resolveSessionId]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Re-fetch serverMessages from the provider sessions endpoint.
|
* Re-fetch serverMessages from the provider sessions endpoint.
|
||||||
@@ -305,12 +472,13 @@ export function useSessionStore() {
|
|||||||
projectPath?: string;
|
projectPath?: string;
|
||||||
} = {},
|
} = {},
|
||||||
) => {
|
) => {
|
||||||
const slot = getSlot(sessionId);
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
|
const slot = getSlot(resolvedSessionId);
|
||||||
try {
|
try {
|
||||||
const params = new URLSearchParams();
|
const params = new URLSearchParams();
|
||||||
|
|
||||||
const qs = params.toString();
|
const qs = params.toString();
|
||||||
const url = `/api/providers/sessions/${encodeURIComponent(sessionId)}/messages${qs ? `?${qs}` : ''}`;
|
const url = `/api/providers/sessions/${encodeURIComponent(resolvedSessionId)}/messages${qs ? `?${qs}` : ''}`;
|
||||||
const response = await authenticatedFetch(url);
|
const response = await authenticatedFetch(url);
|
||||||
|
|
||||||
if (!response.ok) throw new Error(`HTTP ${response.status}`);
|
if (!response.ok) throw new Error(`HTTP ${response.status}`);
|
||||||
@@ -323,40 +491,43 @@ export function useSessionStore() {
|
|||||||
// drop realtime messages that the server has caught up with to prevent unbounded growth.
|
// drop realtime messages that the server has caught up with to prevent unbounded growth.
|
||||||
slot.realtimeMessages = [];
|
slot.realtimeMessages = [];
|
||||||
recomputeMergedIfNeeded(slot);
|
recomputeMergedIfNeeded(slot);
|
||||||
notify(sessionId);
|
notify(resolvedSessionId);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error(`[SessionStore] refresh failed for ${sessionId}:`, error);
|
console.error(`[SessionStore] refresh failed for ${resolvedSessionId}:`, error);
|
||||||
}
|
}
|
||||||
}, [getSlot, notify]);
|
}, [getSlot, notify, resolveSessionId]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Update session status.
|
* Update session status.
|
||||||
*/
|
*/
|
||||||
const setStatus = useCallback((sessionId: string, status: SessionStatus) => {
|
const setStatus = useCallback((sessionId: string, status: SessionStatus) => {
|
||||||
const slot = getSlot(sessionId);
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
|
const slot = getSlot(resolvedSessionId);
|
||||||
slot.status = status;
|
slot.status = status;
|
||||||
notify(sessionId);
|
notify(resolvedSessionId);
|
||||||
}, [getSlot, notify]);
|
}, [getSlot, notify, resolveSessionId]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if a session's data is stale (>30s old).
|
* Check if a session's data is stale (>30s old).
|
||||||
*/
|
*/
|
||||||
const isStale = useCallback((sessionId: string) => {
|
const isStale = useCallback((sessionId: string) => {
|
||||||
const slot = storeRef.current.get(sessionId);
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
|
const slot = storeRef.current.get(resolvedSessionId);
|
||||||
if (!slot) return true;
|
if (!slot) return true;
|
||||||
return Date.now() - slot.fetchedAt > STALE_THRESHOLD_MS;
|
return Date.now() - slot.fetchedAt > STALE_THRESHOLD_MS;
|
||||||
}, []);
|
}, [resolveSessionId]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Update or create a streaming message (accumulated text so far).
|
* Update or create a streaming message (accumulated text so far).
|
||||||
* Uses a well-known ID so subsequent calls replace the same message.
|
* Uses a well-known ID so subsequent calls replace the same message.
|
||||||
*/
|
*/
|
||||||
const updateStreaming = useCallback((sessionId: string, accumulatedText: string, msgProvider: LLMProvider) => {
|
const updateStreaming = useCallback((sessionId: string, accumulatedText: string, msgProvider: LLMProvider) => {
|
||||||
const slot = getSlot(sessionId);
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
const streamId = `__streaming_${sessionId}`;
|
const slot = getSlot(resolvedSessionId);
|
||||||
|
const streamId = `__streaming_${resolvedSessionId}`;
|
||||||
const msg: NormalizedMessage = {
|
const msg: NormalizedMessage = {
|
||||||
id: streamId,
|
id: streamId,
|
||||||
sessionId,
|
sessionId: resolvedSessionId,
|
||||||
timestamp: new Date().toISOString(),
|
timestamp: new Date().toISOString(),
|
||||||
provider: msgProvider,
|
provider: msgProvider,
|
||||||
kind: 'stream_delta',
|
kind: 'stream_delta',
|
||||||
@@ -370,17 +541,18 @@ export function useSessionStore() {
|
|||||||
slot.realtimeMessages = [...slot.realtimeMessages, msg];
|
slot.realtimeMessages = [...slot.realtimeMessages, msg];
|
||||||
}
|
}
|
||||||
recomputeMergedIfNeeded(slot);
|
recomputeMergedIfNeeded(slot);
|
||||||
notify(sessionId);
|
notify(resolvedSessionId);
|
||||||
}, [getSlot, notify]);
|
}, [getSlot, notify, resolveSessionId]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Finalize streaming: convert the streaming message to a regular text message.
|
* Finalize streaming: convert the streaming message to a regular text message.
|
||||||
* The well-known streaming ID is replaced with a unique text message ID.
|
* The well-known streaming ID is replaced with a unique text message ID.
|
||||||
*/
|
*/
|
||||||
const finalizeStreaming = useCallback((sessionId: string) => {
|
const finalizeStreaming = useCallback((sessionId: string) => {
|
||||||
const slot = storeRef.current.get(sessionId);
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
|
const slot = storeRef.current.get(resolvedSessionId);
|
||||||
if (!slot) return;
|
if (!slot) return;
|
||||||
const streamId = `__streaming_${sessionId}`;
|
const streamId = `__streaming_${resolvedSessionId}`;
|
||||||
const idx = slot.realtimeMessages.findIndex(m => m.id === streamId);
|
const idx = slot.realtimeMessages.findIndex(m => m.id === streamId);
|
||||||
if (idx >= 0) {
|
if (idx >= 0) {
|
||||||
const stream = slot.realtimeMessages[idx];
|
const stream = slot.realtimeMessages[idx];
|
||||||
@@ -392,35 +564,104 @@ export function useSessionStore() {
|
|||||||
role: 'assistant',
|
role: 'assistant',
|
||||||
};
|
};
|
||||||
recomputeMergedIfNeeded(slot);
|
recomputeMergedIfNeeded(slot);
|
||||||
notify(sessionId);
|
notify(resolvedSessionId);
|
||||||
}
|
}
|
||||||
}, [notify]);
|
}, [notify, resolveSessionId]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Clear realtime messages for a session (e.g., after stream completes and server fetch catches up).
|
* Clear realtime messages for a session (e.g., after stream completes and server fetch catches up).
|
||||||
*/
|
*/
|
||||||
const clearRealtime = useCallback((sessionId: string) => {
|
const clearRealtime = useCallback((sessionId: string) => {
|
||||||
const slot = storeRef.current.get(sessionId);
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
|
const slot = storeRef.current.get(resolvedSessionId);
|
||||||
if (slot) {
|
if (slot) {
|
||||||
slot.realtimeMessages = [];
|
slot.realtimeMessages = [];
|
||||||
recomputeMergedIfNeeded(slot);
|
recomputeMergedIfNeeded(slot);
|
||||||
notify(sessionId);
|
notify(resolvedSessionId);
|
||||||
}
|
}
|
||||||
}, [notify]);
|
}, [notify, resolveSessionId]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get merged messages for a session (for rendering).
|
* Get merged messages for a session (for rendering).
|
||||||
*/
|
*/
|
||||||
const getMessages = useCallback((sessionId: string): NormalizedMessage[] => {
|
const getMessages = useCallback((sessionId: string): NormalizedMessage[] => {
|
||||||
return storeRef.current.get(sessionId)?.merged ?? [];
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
}, []);
|
return storeRef.current.get(resolvedSessionId)?.merged ?? [];
|
||||||
|
}, [resolveSessionId]);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get session slot (for status, pagination info, etc.).
|
* Get session slot (for status, pagination info, etc.).
|
||||||
*/
|
*/
|
||||||
const getSessionSlot = useCallback((sessionId: string): SessionSlot | undefined => {
|
const getSessionSlot = useCallback((sessionId: string): SessionSlot | undefined => {
|
||||||
return storeRef.current.get(sessionId);
|
const resolvedSessionId = resolveSessionId(sessionId) ?? sessionId;
|
||||||
}, []);
|
return storeRef.current.get(resolvedSessionId);
|
||||||
|
}, [resolveSessionId]);
|
||||||
|
|
||||||
|
const replaceSessionId = useCallback((fromSessionId: string, toSessionId: string) => {
|
||||||
|
const resolvedFromSessionId = resolveSessionId(fromSessionId) ?? fromSessionId;
|
||||||
|
const resolvedToSessionId = resolveSessionId(toSessionId) ?? toSessionId;
|
||||||
|
|
||||||
|
if (resolvedFromSessionId === resolvedToSessionId) {
|
||||||
|
sessionAliasesRef.current.set(fromSessionId, resolvedToSessionId);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const store = storeRef.current;
|
||||||
|
const sourceSlot = store.get(resolvedFromSessionId);
|
||||||
|
const targetSlot = store.get(resolvedToSessionId) ?? createEmptySlot();
|
||||||
|
|
||||||
|
if (sourceSlot) {
|
||||||
|
const migratedServerMessages = sourceSlot.serverMessages.map((msg) =>
|
||||||
|
rewriteMessageSessionId(msg, resolvedFromSessionId, resolvedToSessionId),
|
||||||
|
);
|
||||||
|
const migratedRealtimeMessages = sourceSlot.realtimeMessages.map((msg) =>
|
||||||
|
rewriteMessageSessionId(msg, resolvedFromSessionId, resolvedToSessionId),
|
||||||
|
);
|
||||||
|
|
||||||
|
targetSlot.serverMessages = mergeMessagesById(targetSlot.serverMessages, migratedServerMessages);
|
||||||
|
targetSlot.realtimeMessages = mergeMessagesById(targetSlot.realtimeMessages, migratedRealtimeMessages);
|
||||||
|
if (targetSlot.realtimeMessages.length > MAX_REALTIME_MESSAGES) {
|
||||||
|
targetSlot.realtimeMessages = targetSlot.realtimeMessages.slice(-MAX_REALTIME_MESSAGES);
|
||||||
|
}
|
||||||
|
targetSlot.status =
|
||||||
|
sourceSlot.status === 'error'
|
||||||
|
? 'error'
|
||||||
|
: sourceSlot.status === 'streaming' || targetSlot.status === 'streaming'
|
||||||
|
? 'streaming'
|
||||||
|
: sourceSlot.status === 'loading' || targetSlot.status === 'loading'
|
||||||
|
? 'loading'
|
||||||
|
: targetSlot.status;
|
||||||
|
targetSlot.fetchedAt = Math.max(targetSlot.fetchedAt, sourceSlot.fetchedAt, Date.now());
|
||||||
|
targetSlot.total = Math.max(
|
||||||
|
targetSlot.total,
|
||||||
|
sourceSlot.total,
|
||||||
|
targetSlot.serverMessages.length,
|
||||||
|
targetSlot.realtimeMessages.length,
|
||||||
|
);
|
||||||
|
targetSlot.hasMore = targetSlot.hasMore || sourceSlot.hasMore;
|
||||||
|
targetSlot.offset = Math.max(targetSlot.offset, sourceSlot.offset);
|
||||||
|
targetSlot.tokenUsage = targetSlot.tokenUsage ?? sourceSlot.tokenUsage;
|
||||||
|
recomputeMergedIfNeeded(targetSlot);
|
||||||
|
|
||||||
|
store.set(resolvedToSessionId, targetSlot);
|
||||||
|
store.delete(resolvedFromSessionId);
|
||||||
|
}
|
||||||
|
|
||||||
|
sessionAliasesRef.current.set(resolvedFromSessionId, resolvedToSessionId);
|
||||||
|
sessionAliasesRef.current.set(fromSessionId, resolvedToSessionId);
|
||||||
|
|
||||||
|
for (const [aliasSessionId, targetSessionId] of sessionAliasesRef.current.entries()) {
|
||||||
|
if (targetSessionId === resolvedFromSessionId) {
|
||||||
|
sessionAliasesRef.current.set(aliasSessionId, resolvedToSessionId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (activeSessionIdRef.current === resolvedFromSessionId) {
|
||||||
|
activeSessionIdRef.current = resolvedToSessionId;
|
||||||
|
}
|
||||||
|
|
||||||
|
notify(resolvedToSessionId);
|
||||||
|
}, [notify, resolveSessionId]);
|
||||||
|
|
||||||
return useMemo(() => ({
|
return useMemo(() => ({
|
||||||
getSlot,
|
getSlot,
|
||||||
@@ -438,11 +679,12 @@ export function useSessionStore() {
|
|||||||
clearRealtime,
|
clearRealtime,
|
||||||
getMessages,
|
getMessages,
|
||||||
getSessionSlot,
|
getSessionSlot,
|
||||||
|
replaceSessionId,
|
||||||
}), [
|
}), [
|
||||||
getSlot, has, fetchFromServer, fetchMore,
|
getSlot, has, fetchFromServer, fetchMore,
|
||||||
appendRealtime, appendRealtimeBatch, refreshFromServer,
|
appendRealtime, appendRealtimeBatch, refreshFromServer,
|
||||||
setActiveSession, setStatus, isStale, updateStreaming, finalizeStreaming,
|
setActiveSession, setStatus, isStale, updateStreaming, finalizeStreaming,
|
||||||
clearRealtime, getMessages, getSessionSlot,
|
clearRealtime, getMessages, getSessionSlot, replaceSessionId,
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -54,6 +54,7 @@ export const api = {
|
|||||||
// After the projectName → projectId migration the path/query identifier is
|
// After the projectName → projectId migration the path/query identifier is
|
||||||
// the DB-assigned `projectId`; parameter names reflect that for clarity.
|
// the DB-assigned `projectId`; parameter names reflect that for clarity.
|
||||||
projects: () => authenticatedFetch('/api/projects'),
|
projects: () => authenticatedFetch('/api/projects'),
|
||||||
|
archivedProjects: () => authenticatedFetch('/api/projects/archived'),
|
||||||
projectSessions: (projectId, { limit = 20, offset = 0 } = {}) => {
|
projectSessions: (projectId, { limit = 20, offset = 0 } = {}) => {
|
||||||
const params = new URLSearchParams();
|
const params = new URLSearchParams();
|
||||||
params.set('limit', String(limit));
|
params.set('limit', String(limit));
|
||||||
@@ -78,9 +79,28 @@ export const api = {
|
|||||||
method: 'PUT',
|
method: 'PUT',
|
||||||
body: JSON.stringify({ displayName }),
|
body: JSON.stringify({ displayName }),
|
||||||
}),
|
}),
|
||||||
deleteSession: (sessionId) =>
|
restoreProject: (projectId) =>
|
||||||
authenticatedFetch(`/api/providers/sessions/${sessionId}`, {
|
authenticatedFetch(`/api/projects/${encodeURIComponent(projectId)}/restore`, {
|
||||||
|
method: 'POST',
|
||||||
|
}),
|
||||||
|
// Session deletion now mirrors project deletion:
|
||||||
|
// - default: archive only (`isArchived = 1`)
|
||||||
|
// - hardDelete: remove the row and, by default, its persisted transcript file
|
||||||
|
deleteSession: (sessionId, hardDelete = false) => {
|
||||||
|
const params = new URLSearchParams();
|
||||||
|
if (hardDelete) {
|
||||||
|
params.set('force', 'true');
|
||||||
|
}
|
||||||
|
const qs = params.toString();
|
||||||
|
return authenticatedFetch(`/api/providers/sessions/${sessionId}${qs ? `?${qs}` : ''}`, {
|
||||||
method: 'DELETE',
|
method: 'DELETE',
|
||||||
|
});
|
||||||
|
},
|
||||||
|
getArchivedSessions: () =>
|
||||||
|
authenticatedFetch('/api/providers/sessions/archived'),
|
||||||
|
restoreSession: (sessionId) =>
|
||||||
|
authenticatedFetch(`/api/providers/sessions/${sessionId}/restore`, {
|
||||||
|
method: 'POST',
|
||||||
}),
|
}),
|
||||||
renameSession: (sessionId, summary) =>
|
renameSession: (sessionId, summary) =>
|
||||||
authenticatedFetch(`/api/providers/sessions/${sessionId}`, {
|
authenticatedFetch(`/api/providers/sessions/${sessionId}`, {
|
||||||
|
|||||||
Reference in New Issue
Block a user