/ src / commands / resume / resume.tsx
resume.tsx
  1  import { c as _c } from "react/compiler-runtime";
  2  import chalk from 'chalk';
  3  import type { UUID } from 'crypto';
  4  import figures from 'figures';
  5  import * as React from 'react';
  6  import { getOriginalCwd, getSessionId } from '../../bootstrap/state.js';
  7  import type { CommandResultDisplay, ResumeEntrypoint } from '../../commands.js';
  8  import { LogSelector } from '../../components/LogSelector.js';
  9  import { MessageResponse } from '../../components/MessageResponse.js';
 10  import { Spinner } from '../../components/Spinner.js';
 11  import { useIsInsideModal } from '../../context/modalContext.js';
 12  import { useTerminalSize } from '../../hooks/useTerminalSize.js';
 13  import { setClipboard } from '../../ink/termio/osc.js';
 14  import { Box, Text } from '../../ink.js';
 15  import type { LocalJSXCommandCall } from '../../types/command.js';
 16  import type { LogOption } from '../../types/logs.js';
 17  import { agenticSessionSearch } from '../../utils/agenticSessionSearch.js';
 18  import { checkCrossProjectResume } from '../../utils/crossProjectResume.js';
 19  import { getWorktreePaths } from '../../utils/getWorktreePaths.js';
 20  import { logError } from '../../utils/log.js';
 21  import { getLastSessionLog, getSessionIdFromLog, isCustomTitleEnabled, isLiteLog, loadAllProjectsMessageLogs, loadFullLog, loadSameRepoMessageLogs, searchSessionsByCustomTitle } from '../../utils/sessionStorage.js';
 22  import { validateUuid } from '../../utils/uuid.js';
 23  type ResumeResult = {
 24    resultType: 'sessionNotFound';
 25    arg: string;
 26  } | {
 27    resultType: 'multipleMatches';
 28    arg: string;
 29    count: number;
 30  };
 31  function resumeHelpMessage(result: ResumeResult): string {
 32    switch (result.resultType) {
 33      case 'sessionNotFound':
 34        return `Session ${chalk.bold(result.arg)} was not found.`;
 35      case 'multipleMatches':
 36        return `Found ${result.count} sessions matching ${chalk.bold(result.arg)}. Please use /resume to pick a specific session.`;
 37    }
 38  }
 39  function ResumeError(t0) {
 40    const $ = _c(10);
 41    const {
 42      message,
 43      args,
 44      onDone
 45    } = t0;
 46    let t1;
 47    let t2;
 48    if ($[0] !== onDone) {
 49      t1 = () => {
 50        const timer = setTimeout(onDone, 0);
 51        return () => clearTimeout(timer);
 52      };
 53      t2 = [onDone];
 54      $[0] = onDone;
 55      $[1] = t1;
 56      $[2] = t2;
 57    } else {
 58      t1 = $[1];
 59      t2 = $[2];
 60    }
 61    React.useEffect(t1, t2);
 62    let t3;
 63    if ($[3] !== args) {
 64      t3 = <Text dimColor={true}>{figures.pointer} /resume {args}</Text>;
 65      $[3] = args;
 66      $[4] = t3;
 67    } else {
 68      t3 = $[4];
 69    }
 70    let t4;
 71    if ($[5] !== message) {
 72      t4 = <MessageResponse><Text>{message}</Text></MessageResponse>;
 73      $[5] = message;
 74      $[6] = t4;
 75    } else {
 76      t4 = $[6];
 77    }
 78    let t5;
 79    if ($[7] !== t3 || $[8] !== t4) {
 80      t5 = <Box flexDirection="column">{t3}{t4}</Box>;
 81      $[7] = t3;
 82      $[8] = t4;
 83      $[9] = t5;
 84    } else {
 85      t5 = $[9];
 86    }
 87    return t5;
 88  }
 89  function ResumeCommand({
 90    onDone,
 91    onResume
 92  }: {
 93    onDone: (result?: string, options?: {
 94      display?: CommandResultDisplay;
 95    }) => void;
 96    onResume: (sessionId: UUID, log: LogOption, entrypoint: ResumeEntrypoint) => Promise<void>;
 97  }): React.ReactNode {
 98    const [logs, setLogs] = React.useState<LogOption[]>([]);
 99    const [worktreePaths, setWorktreePaths] = React.useState<string[]>([]);
100    const [loading, setLoading] = React.useState(true);
101    const [resuming, setResuming] = React.useState(false);
102    const [showAllProjects, setShowAllProjects] = React.useState(false);
103    const {
104      rows
105    } = useTerminalSize();
106    const insideModal = useIsInsideModal();
107    const loadLogs = React.useCallback(async (allProjects: boolean, paths: string[]) => {
108      setLoading(true);
109      try {
110        const allLogs = allProjects ? await loadAllProjectsMessageLogs() : await loadSameRepoMessageLogs(paths);
111        const resumable = filterResumableSessions(allLogs, getSessionId());
112        if (resumable.length === 0) {
113          onDone('No conversations found to resume');
114          return;
115        }
116        setLogs(resumable);
117      } catch (_err) {
118        onDone('Failed to load conversations');
119      } finally {
120        setLoading(false);
121      }
122    }, [onDone]);
123    React.useEffect(() => {
124      async function init() {
125        const paths_0 = await getWorktreePaths(getOriginalCwd());
126        setWorktreePaths(paths_0);
127        void loadLogs(false, paths_0);
128      }
129      void init();
130    }, [loadLogs]);
131    const handleToggleAllProjects = React.useCallback(() => {
132      const newValue = !showAllProjects;
133      setShowAllProjects(newValue);
134      void loadLogs(newValue, worktreePaths);
135    }, [showAllProjects, loadLogs, worktreePaths]);
136    async function handleSelect(log: LogOption) {
137      const sessionId = validateUuid(getSessionIdFromLog(log));
138      if (!sessionId) {
139        onDone('Failed to resume conversation');
140        return;
141      }
142  
143      // Load full messages for lite logs
144      const fullLog = isLiteLog(log) ? await loadFullLog(log) : log;
145  
146      // Check if this conversation is from a different directory
147      const crossProjectCheck = checkCrossProjectResume(fullLog, showAllProjects, worktreePaths);
148      if (crossProjectCheck.isCrossProject) {
149        if (crossProjectCheck.isSameRepoWorktree) {
150          // Same repo worktree - can resume directly
151          setResuming(true);
152          void onResume(sessionId, fullLog, 'slash_command_picker');
153          return;
154        }
155  
156        // Different project - show command instead of resuming
157        const raw = await setClipboard(crossProjectCheck.command);
158        if (raw) process.stdout.write(raw);
159  
160        // Format the output message
161        const message = ['', 'This conversation is from a different directory.', '', 'To resume, run:', `  ${crossProjectCheck.command}`, '', '(Command copied to clipboard)', ''].join('\n');
162        onDone(message, {
163          display: 'user'
164        });
165        return;
166      }
167  
168      // Same directory - proceed with resume
169      setResuming(true);
170      void onResume(sessionId, fullLog, 'slash_command_picker');
171    }
172    function handleCancel() {
173      onDone('Resume cancelled', {
174        display: 'system'
175      });
176    }
177    if (loading) {
178      return <Box>
179          <Spinner />
180          <Text> Loading conversations…</Text>
181        </Box>;
182    }
183    if (resuming) {
184      return <Box>
185          <Spinner />
186          <Text> Resuming conversation…</Text>
187        </Box>;
188    }
189    return <LogSelector logs={logs} maxHeight={insideModal ? Math.floor(rows / 2) : rows - 2} onCancel={handleCancel} onSelect={handleSelect} onLogsChanged={() => loadLogs(showAllProjects, worktreePaths)} showAllProjects={showAllProjects} onToggleAllProjects={handleToggleAllProjects} onAgenticSearch={agenticSessionSearch} />;
190  }
191  export function filterResumableSessions(logs: LogOption[], currentSessionId: string): LogOption[] {
192    return logs.filter(l => !l.isSidechain && getSessionIdFromLog(l) !== currentSessionId);
193  }
194  export const call: LocalJSXCommandCall = async (onDone, context, args) => {
195    const onResume = async (sessionId: UUID, log: LogOption, entrypoint: ResumeEntrypoint) => {
196      try {
197        await context.resume?.(sessionId, log, entrypoint);
198        onDone(undefined, {
199          display: 'skip'
200        });
201      } catch (error) {
202        logError(error as Error);
203        onDone(`Failed to resume: ${(error as Error).message}`);
204      }
205    };
206    const arg = args?.trim();
207  
208    // No argument provided - show picker
209    if (!arg) {
210      return <ResumeCommand key={Date.now()} onDone={onDone} onResume={onResume} />;
211    }
212  
213    // Load logs to search (includes same-repo worktrees)
214    const worktreePaths = await getWorktreePaths(getOriginalCwd());
215    const logs = await loadSameRepoMessageLogs(worktreePaths);
216    if (logs.length === 0) {
217      const message = 'No conversations found to resume.';
218      return <ResumeError message={message} args={arg} onDone={() => onDone(message)} />;
219    }
220  
221    // First, check if arg is a valid UUID
222    const maybeSessionId = validateUuid(arg);
223    if (maybeSessionId) {
224      const matchingLogs = logs.filter(l => getSessionIdFromLog(l) === maybeSessionId).sort((a, b) => b.modified.getTime() - a.modified.getTime());
225      if (matchingLogs.length > 0) {
226        const log = matchingLogs[0]!;
227        const fullLog = isLiteLog(log) ? await loadFullLog(log) : log;
228        void onResume(maybeSessionId, fullLog, 'slash_command_session_id');
229        return null;
230      }
231  
232      // Enriched logs didn't find it — try direct file lookup. This handles
233      // sessions filtered out by enrichLogs (e.g., first message >16KB makes
234      // firstPrompt extraction fail, causing the session to be dropped).
235      const directLog = await getLastSessionLog(maybeSessionId);
236      if (directLog) {
237        void onResume(maybeSessionId, directLog, 'slash_command_session_id');
238        return null;
239      }
240    }
241  
242    // Next, try exact custom title match (only if feature is enabled)
243    if (isCustomTitleEnabled()) {
244      const titleMatches = await searchSessionsByCustomTitle(arg, {
245        exact: true
246      });
247      if (titleMatches.length === 1) {
248        const log = titleMatches[0]!;
249        const sessionId = getSessionIdFromLog(log);
250        if (sessionId) {
251          const fullLog = isLiteLog(log) ? await loadFullLog(log) : log;
252          void onResume(sessionId, fullLog, 'slash_command_title');
253          return null;
254        }
255      }
256  
257      // Multiple matches - show error
258      if (titleMatches.length > 1) {
259        const message = resumeHelpMessage({
260          resultType: 'multipleMatches',
261          arg,
262          count: titleMatches.length
263        });
264        return <ResumeError message={message} args={arg} onDone={() => onDone(message)} />;
265      }
266    }
267  
268    // No match found - show error
269    const message = resumeHelpMessage({
270      resultType: 'sessionNotFound',
271      arg
272    });
273    return <ResumeError message={message} args={arg} onDone={() => onDone(message)} />;
274  };
275  //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["chalk","UUID","figures","React","getOriginalCwd","getSessionId","CommandResultDisplay","ResumeEntrypoint","LogSelector","MessageResponse","Spinner","useIsInsideModal","useTerminalSize","setClipboard","Box","Text","LocalJSXCommandCall","LogOption","agenticSessionSearch","checkCrossProjectResume","getWorktreePaths","logError","getLastSessionLog","getSessionIdFromLog","isCustomTitleEnabled","isLiteLog","loadAllProjectsMessageLogs","loadFullLog","loadSameRepoMessageLogs","searchSessionsByCustomTitle","validateUuid","ResumeResult","resultType","arg","count","resumeHelpMessage","result","bold","ResumeError","t0","$","_c","message","args","onDone","t1","t2","timer","setTimeout","clearTimeout","useEffect","t3","pointer","t4","t5","ResumeCommand","onResume","options","display","sessionId","log","entrypoint","Promise","ReactNode","logs","setLogs","useState","worktreePaths","setWorktreePaths","loading","setLoading","resuming","setResuming","showAllProjects","setShowAllProjects","rows","insideModal","loadLogs","useCallback","allProjects","paths","allLogs","resumable","filterResumableSessions","length","_err","init","handleToggleAllProjects","newValue","handleSelect","fullLog","crossProjectCheck","isCrossProject","isSameRepoWorktree","raw","command","process","stdout","write","join","handleCancel","Math","floor","currentSessionId","filter","l","isSidechain","call","context","resume","undefined","error","Error","trim","Date","now","maybeSessionId","matchingLogs","sort","a","b","modified","getTime","directLog","titleMatches","exact"],"sources":["resume.tsx"],"sourcesContent":["import chalk from 'chalk'\nimport type { UUID } from 'crypto'\nimport figures from 'figures'\nimport * as React from 'react'\nimport { getOriginalCwd, getSessionId } from '../../bootstrap/state.js'\nimport type { CommandResultDisplay, ResumeEntrypoint } from '../../commands.js'\nimport { LogSelector } from '../../components/LogSelector.js'\nimport { MessageResponse } from '../../components/MessageResponse.js'\nimport { Spinner } from '../../components/Spinner.js'\nimport { useIsInsideModal } from '../../context/modalContext.js'\nimport { useTerminalSize } from '../../hooks/useTerminalSize.js'\nimport { setClipboard } from '../../ink/termio/osc.js'\nimport { Box, Text } from '../../ink.js'\nimport type { LocalJSXCommandCall } from '../../types/command.js'\nimport type { LogOption } from '../../types/logs.js'\nimport { agenticSessionSearch } from '../../utils/agenticSessionSearch.js'\nimport { checkCrossProjectResume } from '../../utils/crossProjectResume.js'\nimport { getWorktreePaths } from '../../utils/getWorktreePaths.js'\nimport { logError } from '../../utils/log.js'\nimport {\n  getLastSessionLog,\n  getSessionIdFromLog,\n  isCustomTitleEnabled,\n  isLiteLog,\n  loadAllProjectsMessageLogs,\n  loadFullLog,\n  loadSameRepoMessageLogs,\n  searchSessionsByCustomTitle,\n} from '../../utils/sessionStorage.js'\nimport { validateUuid } from '../../utils/uuid.js'\n\ntype ResumeResult =\n  | { resultType: 'sessionNotFound'; arg: string }\n  | { resultType: 'multipleMatches'; arg: string; count: number }\n\nfunction resumeHelpMessage(result: ResumeResult): string {\n  switch (result.resultType) {\n    case 'sessionNotFound':\n      return `Session ${chalk.bold(result.arg)} was not found.`\n    case 'multipleMatches':\n      return `Found ${result.count} sessions matching ${chalk.bold(result.arg)}. Please use /resume to pick a specific session.`\n  }\n}\n\nfunction ResumeError({\n  message,\n  args,\n  onDone,\n}: {\n  message: string\n  args: string\n  onDone: () => void\n}): React.ReactNode {\n  React.useEffect(() => {\n    const timer = setTimeout(onDone, 0)\n    return () => clearTimeout(timer)\n  }, [onDone])\n\n  return (\n    <Box flexDirection=\"column\">\n      <Text dimColor>\n        {figures.pointer} /resume {args}\n      </Text>\n      <MessageResponse>\n        <Text>{message}</Text>\n      </MessageResponse>\n    </Box>\n  )\n}\n\nfunction ResumeCommand({\n  onDone,\n  onResume,\n}: {\n  onDone: (\n    result?: string,\n    options?: { display?: CommandResultDisplay },\n  ) => void\n  onResume: (\n    sessionId: UUID,\n    log: LogOption,\n    entrypoint: ResumeEntrypoint,\n  ) => Promise<void>\n}): React.ReactNode {\n  const [logs, setLogs] = React.useState<LogOption[]>([])\n  const [worktreePaths, setWorktreePaths] = React.useState<string[]>([])\n  const [loading, setLoading] = React.useState(true)\n  const [resuming, setResuming] = React.useState(false)\n  const [showAllProjects, setShowAllProjects] = React.useState(false)\n  const { rows } = useTerminalSize()\n  const insideModal = useIsInsideModal()\n\n  const loadLogs = React.useCallback(\n    async (allProjects: boolean, paths: string[]) => {\n      setLoading(true)\n      try {\n        const allLogs = allProjects\n          ? await loadAllProjectsMessageLogs()\n          : await loadSameRepoMessageLogs(paths)\n        const resumable = filterResumableSessions(allLogs, getSessionId())\n        if (resumable.length === 0) {\n          onDone('No conversations found to resume')\n          return\n        }\n        setLogs(resumable)\n      } catch (_err) {\n        onDone('Failed to load conversations')\n      } finally {\n        setLoading(false)\n      }\n    },\n    [onDone],\n  )\n\n  React.useEffect(() => {\n    async function init() {\n      const paths = await getWorktreePaths(getOriginalCwd())\n      setWorktreePaths(paths)\n      void loadLogs(false, paths)\n    }\n    void init()\n  }, [loadLogs])\n\n  const handleToggleAllProjects = React.useCallback(() => {\n    const newValue = !showAllProjects\n    setShowAllProjects(newValue)\n    void loadLogs(newValue, worktreePaths)\n  }, [showAllProjects, loadLogs, worktreePaths])\n\n  async function handleSelect(log: LogOption) {\n    const sessionId = validateUuid(getSessionIdFromLog(log))\n    if (!sessionId) {\n      onDone('Failed to resume conversation')\n      return\n    }\n\n    // Load full messages for lite logs\n    const fullLog = isLiteLog(log) ? await loadFullLog(log) : log\n\n    // Check if this conversation is from a different directory\n    const crossProjectCheck = checkCrossProjectResume(\n      fullLog,\n      showAllProjects,\n      worktreePaths,\n    )\n    if (crossProjectCheck.isCrossProject) {\n      if (crossProjectCheck.isSameRepoWorktree) {\n        // Same repo worktree - can resume directly\n        setResuming(true)\n        void onResume(sessionId, fullLog, 'slash_command_picker')\n        return\n      }\n\n      // Different project - show command instead of resuming\n      const raw = await setClipboard(crossProjectCheck.command)\n      if (raw) process.stdout.write(raw)\n\n      // Format the output message\n      const message = [\n        '',\n        'This conversation is from a different directory.',\n        '',\n        'To resume, run:',\n        `  ${crossProjectCheck.command}`,\n        '',\n        '(Command copied to clipboard)',\n        '',\n      ].join('\\n')\n\n      onDone(message, { display: 'user' })\n      return\n    }\n\n    // Same directory - proceed with resume\n    setResuming(true)\n    void onResume(sessionId, fullLog, 'slash_command_picker')\n  }\n\n  function handleCancel() {\n    onDone('Resume cancelled', { display: 'system' })\n  }\n\n  if (loading) {\n    return (\n      <Box>\n        <Spinner />\n        <Text> Loading conversations…</Text>\n      </Box>\n    )\n  }\n\n  if (resuming) {\n    return (\n      <Box>\n        <Spinner />\n        <Text> Resuming conversation…</Text>\n      </Box>\n    )\n  }\n\n  return (\n    <LogSelector\n      logs={logs}\n      maxHeight={insideModal ? Math.floor(rows / 2) : rows - 2}\n      onCancel={handleCancel}\n      onSelect={handleSelect}\n      onLogsChanged={() => loadLogs(showAllProjects, worktreePaths)}\n      showAllProjects={showAllProjects}\n      onToggleAllProjects={handleToggleAllProjects}\n      onAgenticSearch={agenticSessionSearch}\n    />\n  )\n}\n\nexport function filterResumableSessions(\n  logs: LogOption[],\n  currentSessionId: string,\n): LogOption[] {\n  return logs.filter(\n    l => !l.isSidechain && getSessionIdFromLog(l) !== currentSessionId,\n  )\n}\n\nexport const call: LocalJSXCommandCall = async (onDone, context, args) => {\n  const onResume = async (\n    sessionId: UUID,\n    log: LogOption,\n    entrypoint: ResumeEntrypoint,\n  ) => {\n    try {\n      await context.resume?.(sessionId, log, entrypoint)\n      onDone(undefined, { display: 'skip' })\n    } catch (error) {\n      logError(error as Error)\n      onDone(`Failed to resume: ${(error as Error).message}`)\n    }\n  }\n\n  const arg = args?.trim()\n\n  // No argument provided - show picker\n  if (!arg) {\n    return (\n      <ResumeCommand key={Date.now()} onDone={onDone} onResume={onResume} />\n    )\n  }\n\n  // Load logs to search (includes same-repo worktrees)\n  const worktreePaths = await getWorktreePaths(getOriginalCwd())\n  const logs = await loadSameRepoMessageLogs(worktreePaths)\n  if (logs.length === 0) {\n    const message = 'No conversations found to resume.'\n    return (\n      <ResumeError\n        message={message}\n        args={arg}\n        onDone={() => onDone(message)}\n      />\n    )\n  }\n\n  // First, check if arg is a valid UUID\n  const maybeSessionId = validateUuid(arg)\n  if (maybeSessionId) {\n    const matchingLogs = logs\n      .filter(l => getSessionIdFromLog(l) === maybeSessionId)\n      .sort((a, b) => b.modified.getTime() - a.modified.getTime())\n\n    if (matchingLogs.length > 0) {\n      const log = matchingLogs[0]!\n      const fullLog = isLiteLog(log) ? await loadFullLog(log) : log\n      void onResume(maybeSessionId, fullLog, 'slash_command_session_id')\n      return null\n    }\n\n    // Enriched logs didn't find it — try direct file lookup. This handles\n    // sessions filtered out by enrichLogs (e.g., first message >16KB makes\n    // firstPrompt extraction fail, causing the session to be dropped).\n    const directLog = await getLastSessionLog(maybeSessionId)\n    if (directLog) {\n      void onResume(maybeSessionId, directLog, 'slash_command_session_id')\n      return null\n    }\n  }\n\n  // Next, try exact custom title match (only if feature is enabled)\n  if (isCustomTitleEnabled()) {\n    const titleMatches = await searchSessionsByCustomTitle(arg, {\n      exact: true,\n    })\n    if (titleMatches.length === 1) {\n      const log = titleMatches[0]!\n      const sessionId = getSessionIdFromLog(log)\n      if (sessionId) {\n        const fullLog = isLiteLog(log) ? await loadFullLog(log) : log\n        void onResume(sessionId, fullLog, 'slash_command_title')\n        return null\n      }\n    }\n\n    // Multiple matches - show error\n    if (titleMatches.length > 1) {\n      const message = resumeHelpMessage({\n        resultType: 'multipleMatches',\n        arg,\n        count: titleMatches.length,\n      })\n      return (\n        <ResumeError\n          message={message}\n          args={arg}\n          onDone={() => onDone(message)}\n        />\n      )\n    }\n  }\n\n  // No match found - show error\n  const message = resumeHelpMessage({ resultType: 'sessionNotFound', arg })\n  return (\n    <ResumeError message={message} args={arg} onDone={() => onDone(message)} />\n  )\n}\n"],"mappings":"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","ignoreList":[]}