/ 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,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJjaGFsayIsIlVVSUQiLCJmaWd1cmVzIiwiUmVhY3QiLCJnZXRPcmlnaW5hbEN3ZCIsImdldFNlc3Npb25JZCIsIkNvbW1hbmRSZXN1bHREaXNwbGF5IiwiUmVzdW1lRW50cnlwb2ludCIsIkxvZ1NlbGVjdG9yIiwiTWVzc2FnZVJlc3BvbnNlIiwiU3Bpbm5lciIsInVzZUlzSW5zaWRlTW9kYWwiLCJ1c2VUZXJtaW5hbFNpemUiLCJzZXRDbGlwYm9hcmQiLCJCb3giLCJUZXh0IiwiTG9jYWxKU1hDb21tYW5kQ2FsbCIsIkxvZ09wdGlvbiIsImFnZW50aWNTZXNzaW9uU2VhcmNoIiwiY2hlY2tDcm9zc1Byb2plY3RSZXN1bWUiLCJnZXRXb3JrdHJlZVBhdGhzIiwibG9nRXJyb3IiLCJnZXRMYXN0U2Vzc2lvbkxvZyIsImdldFNlc3Npb25JZEZyb21Mb2ciLCJpc0N1c3RvbVRpdGxlRW5hYmxlZCIsImlzTGl0ZUxvZyIsImxvYWRBbGxQcm9qZWN0c01lc3NhZ2VMb2dzIiwibG9hZEZ1bGxMb2ciLCJsb2FkU2FtZVJlcG9NZXNzYWdlTG9ncyIsInNlYXJjaFNlc3Npb25zQnlDdXN0b21UaXRsZSIsInZhbGlkYXRlVXVpZCIsIlJlc3VtZVJlc3VsdCIsInJlc3VsdFR5cGUiLCJhcmciLCJjb3VudCIsInJlc3VtZUhlbHBNZXNzYWdlIiwicmVzdWx0IiwiYm9sZCIsIlJlc3VtZUVycm9yIiwidDAiLCIkIiwiX2MiLCJtZXNzYWdlIiwiYXJncyIsIm9uRG9uZSIsInQxIiwidDIiLCJ0aW1lciIsInNldFRpbWVvdXQiLCJjbGVhclRpbWVvdXQiLCJ1c2VFZmZlY3QiLCJ0MyIsInBvaW50ZXIiLCJ0NCIsInQ1IiwiUmVzdW1lQ29tbWFuZCIsIm9uUmVzdW1lIiwib3B0aW9ucyIsImRpc3BsYXkiLCJzZXNzaW9uSWQiLCJsb2ciLCJlbnRyeXBvaW50IiwiUHJvbWlzZSIsIlJlYWN0Tm9kZSIsImxvZ3MiLCJzZXRMb2dzIiwidXNlU3RhdGUiLCJ3b3JrdHJlZVBhdGhzIiwic2V0V29ya3RyZWVQYXRocyIsImxvYWRpbmciLCJzZXRMb2FkaW5nIiwicmVzdW1pbmciLCJzZXRSZXN1bWluZyIsInNob3dBbGxQcm9qZWN0cyIsInNldFNob3dBbGxQcm9qZWN0cyIsInJvd3MiLCJpbnNpZGVNb2RhbCIsImxvYWRMb2dzIiwidXNlQ2FsbGJhY2siLCJhbGxQcm9qZWN0cyIsInBhdGhzIiwiYWxsTG9ncyIsInJlc3VtYWJsZSIsImZpbHRlclJlc3VtYWJsZVNlc3Npb25zIiwibGVuZ3RoIiwiX2VyciIsImluaXQiLCJoYW5kbGVUb2dnbGVBbGxQcm9qZWN0cyIsIm5ld1ZhbHVlIiwiaGFuZGxlU2VsZWN0IiwiZnVsbExvZyIsImNyb3NzUHJvamVjdENoZWNrIiwiaXNDcm9zc1Byb2plY3QiLCJpc1NhbWVSZXBvV29ya3RyZWUiLCJyYXciLCJjb21tYW5kIiwicHJvY2VzcyIsInN0ZG91dCIsIndyaXRlIiwiam9pbiIsImhhbmRsZUNhbmNlbCIsIk1hdGgiLCJmbG9vciIsImN1cnJlbnRTZXNzaW9uSWQiLCJmaWx0ZXIiLCJsIiwiaXNTaWRlY2hhaW4iLCJjYWxsIiwiY29udGV4dCIsInJlc3VtZSIsInVuZGVmaW5lZCIsImVycm9yIiwiRXJyb3IiLCJ0cmltIiwiRGF0ZSIsIm5vdyIsIm1heWJlU2Vzc2lvbklkIiwibWF0Y2hpbmdMb2dzIiwic29ydCIsImEiLCJiIiwibW9kaWZpZWQiLCJnZXRUaW1lIiwiZGlyZWN0TG9nIiwidGl0bGVNYXRjaGVzIiwiZXhhY3QiXSwic291cmNlcyI6WyJyZXN1bWUudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBjaGFsayBmcm9tICdjaGFsaydcbmltcG9ydCB0eXBlIHsgVVVJRCB9IGZyb20gJ2NyeXB0bydcbmltcG9ydCBmaWd1cmVzIGZyb20gJ2ZpZ3VyZXMnXG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IGdldE9yaWdpbmFsQ3dkLCBnZXRTZXNzaW9uSWQgfSBmcm9tICcuLi8uLi9ib290c3RyYXAvc3RhdGUuanMnXG5pbXBvcnQgdHlwZSB7IENvbW1hbmRSZXN1bHREaXNwbGF5LCBSZXN1bWVFbnRyeXBvaW50IH0gZnJvbSAnLi4vLi4vY29tbWFuZHMuanMnXG5pbXBvcnQgeyBMb2dTZWxlY3RvciB9IGZyb20gJy4uLy4uL2NvbXBvbmVudHMvTG9nU2VsZWN0b3IuanMnXG5pbXBvcnQgeyBNZXNzYWdlUmVzcG9uc2UgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL01lc3NhZ2VSZXNwb25zZS5qcydcbmltcG9ydCB7IFNwaW5uZXIgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL1NwaW5uZXIuanMnXG5pbXBvcnQgeyB1c2VJc0luc2lkZU1vZGFsIH0gZnJvbSAnLi4vLi4vY29udGV4dC9tb2RhbENvbnRleHQuanMnXG5pbXBvcnQgeyB1c2VUZXJtaW5hbFNpemUgfSBmcm9tICcuLi8uLi9ob29rcy91c2VUZXJtaW5hbFNpemUuanMnXG5pbXBvcnQgeyBzZXRDbGlwYm9hcmQgfSBmcm9tICcuLi8uLi9pbmsvdGVybWlvL29zYy5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB0eXBlIHsgTG9jYWxKU1hDb21tYW5kQ2FsbCB9IGZyb20gJy4uLy4uL3R5cGVzL2NvbW1hbmQuanMnXG5pbXBvcnQgdHlwZSB7IExvZ09wdGlvbiB9IGZyb20gJy4uLy4uL3R5cGVzL2xvZ3MuanMnXG5pbXBvcnQgeyBhZ2VudGljU2Vzc2lvblNlYXJjaCB9IGZyb20gJy4uLy4uL3V0aWxzL2FnZW50aWNTZXNzaW9uU2VhcmNoLmpzJ1xuaW1wb3J0IHsgY2hlY2tDcm9zc1Byb2plY3RSZXN1bWUgfSBmcm9tICcuLi8uLi91dGlscy9jcm9zc1Byb2plY3RSZXN1bWUuanMnXG5pbXBvcnQgeyBnZXRXb3JrdHJlZVBhdGhzIH0gZnJvbSAnLi4vLi4vdXRpbHMvZ2V0V29ya3RyZWVQYXRocy5qcydcbmltcG9ydCB7IGxvZ0Vycm9yIH0gZnJvbSAnLi4vLi4vdXRpbHMvbG9nLmpzJ1xuaW1wb3J0IHtcbiAgZ2V0TGFzdFNlc3Npb25Mb2csXG4gIGdldFNlc3Npb25JZEZyb21Mb2csXG4gIGlzQ3VzdG9tVGl0bGVFbmFibGVkLFxuICBpc0xpdGVMb2csXG4gIGxvYWRBbGxQcm9qZWN0c01lc3NhZ2VMb2dzLFxuICBsb2FkRnVsbExvZyxcbiAgbG9hZFNhbWVSZXBvTWVzc2FnZUxvZ3MsXG4gIHNlYXJjaFNlc3Npb25zQnlDdXN0b21UaXRsZSxcbn0gZnJvbSAnLi4vLi4vdXRpbHMvc2Vzc2lvblN0b3JhZ2UuanMnXG5pbXBvcnQgeyB2YWxpZGF0ZVV1aWQgfSBmcm9tICcuLi8uLi91dGlscy91dWlkLmpzJ1xuXG50eXBlIFJlc3VtZVJlc3VsdCA9XG4gIHwgeyByZXN1bHRUeXBlOiAnc2Vzc2lvbk5vdEZvdW5kJzsgYXJnOiBzdHJpbmcgfVxuICB8IHsgcmVzdWx0VHlwZTogJ211bHRpcGxlTWF0Y2hlcyc7IGFyZzogc3RyaW5nOyBjb3VudDogbnVtYmVyIH1cblxuZnVuY3Rpb24gcmVzdW1lSGVscE1lc3NhZ2UocmVzdWx0OiBSZXN1bWVSZXN1bHQpOiBzdHJpbmcge1xuICBzd2l0Y2ggKHJlc3VsdC5yZXN1bHRUeXBlKSB7XG4gICAgY2FzZSAnc2Vzc2lvbk5vdEZvdW5kJzpcbiAgICAgIHJldHVybiBgU2Vzc2lvbiAke2NoYWxrLmJvbGQocmVzdWx0LmFyZyl9IHdhcyBub3QgZm91bmQuYFxuICAgIGNhc2UgJ211bHRpcGxlTWF0Y2hlcyc6XG4gICAgICByZXR1cm4gYEZvdW5kICR7cmVzdWx0LmNvdW50fSBzZXNzaW9ucyBtYXRjaGluZyAke2NoYWxrLmJvbGQocmVzdWx0LmFyZyl9LiBQbGVhc2UgdXNlIC9yZXN1bWUgdG8gcGljayBhIHNwZWNpZmljIHNlc3Npb24uYFxuICB9XG59XG5cbmZ1bmN0aW9uIFJlc3VtZUVycm9yKHtcbiAgbWVzc2FnZSxcbiAgYXJncyxcbiAgb25Eb25lLFxufToge1xuICBtZXNzYWdlOiBzdHJpbmdcbiAgYXJnczogc3RyaW5nXG4gIG9uRG9uZTogKCkgPT4gdm9pZFxufSk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIFJlYWN0LnVzZUVmZmVjdCgoKSA9PiB7XG4gICAgY29uc3QgdGltZXIgPSBzZXRUaW1lb3V0KG9uRG9uZSwgMClcbiAgICByZXR1cm4gKCkgPT4gY2xlYXJUaW1lb3V0KHRpbWVyKVxuICB9LCBbb25Eb25lXSlcblxuICByZXR1cm4gKFxuICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgIHtmaWd1cmVzLnBvaW50ZXJ9IC9yZXN1bWUge2FyZ3N9XG4gICAgICA8L1RleHQ+XG4gICAgICA8TWVzc2FnZVJlc3BvbnNlPlxuICAgICAgICA8VGV4dD57bWVzc2FnZX08L1RleHQ+XG4gICAgICA8L01lc3NhZ2VSZXNwb25zZT5cbiAgICA8L0JveD5cbiAgKVxufVxuXG5mdW5jdGlvbiBSZXN1bWVDb21tYW5kKHtcbiAgb25Eb25lLFxuICBvblJlc3VtZSxcbn06IHtcbiAgb25Eb25lOiAoXG4gICAgcmVzdWx0Pzogc3RyaW5nLFxuICAgIG9wdGlvbnM/OiB7IGRpc3BsYXk/OiBDb21tYW5kUmVzdWx0RGlzcGxheSB9LFxuICApID0+IHZvaWRcbiAgb25SZXN1bWU6IChcbiAgICBzZXNzaW9uSWQ6IFVVSUQsXG4gICAgbG9nOiBMb2dPcHRpb24sXG4gICAgZW50cnlwb2ludDogUmVzdW1lRW50cnlwb2ludCxcbiAgKSA9PiBQcm9taXNlPHZvaWQ+XG59KTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgW2xvZ3MsIHNldExvZ3NdID0gUmVhY3QudXNlU3RhdGU8TG9nT3B0aW9uW10+KFtdKVxuICBjb25zdCBbd29ya3RyZWVQYXRocywgc2V0V29ya3RyZWVQYXRoc10gPSBSZWFjdC51c2VTdGF0ZTxzdHJpbmdbXT4oW10pXG4gIGNvbnN0IFtsb2FkaW5nLCBzZXRMb2FkaW5nXSA9IFJlYWN0LnVzZVN0YXRlKHRydWUpXG4gIGNvbnN0IFtyZXN1bWluZywgc2V0UmVzdW1pbmddID0gUmVhY3QudXNlU3RhdGUoZmFsc2UpXG4gIGNvbnN0IFtzaG93QWxsUHJvamVjdHMsIHNldFNob3dBbGxQcm9qZWN0c10gPSBSZWFjdC51c2VTdGF0ZShmYWxzZSlcbiAgY29uc3QgeyByb3dzIH0gPSB1c2VUZXJtaW5hbFNpemUoKVxuICBjb25zdCBpbnNpZGVNb2RhbCA9IHVzZUlzSW5zaWRlTW9kYWwoKVxuXG4gIGNvbnN0IGxvYWRMb2dzID0gUmVhY3QudXNlQ2FsbGJhY2soXG4gICAgYXN5bmMgKGFsbFByb2plY3RzOiBib29sZWFuLCBwYXRoczogc3RyaW5nW10pID0+IHtcbiAgICAgIHNldExvYWRpbmcodHJ1ZSlcbiAgICAgIHRyeSB7XG4gICAgICAgIGNvbnN0IGFsbExvZ3MgPSBhbGxQcm9qZWN0c1xuICAgICAgICAgID8gYXdhaXQgbG9hZEFsbFByb2plY3RzTWVzc2FnZUxvZ3MoKVxuICAgICAgICAgIDogYXdhaXQgbG9hZFNhbWVSZXBvTWVzc2FnZUxvZ3MocGF0aHMpXG4gICAgICAgIGNvbnN0IHJlc3VtYWJsZSA9IGZpbHRlclJlc3VtYWJsZVNlc3Npb25zKGFsbExvZ3MsIGdldFNlc3Npb25JZCgpKVxuICAgICAgICBpZiAocmVzdW1hYmxlLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICAgIG9uRG9uZSgnTm8gY29udmVyc2F0aW9ucyBmb3VuZCB0byByZXN1bWUnKVxuICAgICAgICAgIHJldHVyblxuICAgICAgICB9XG4gICAgICAgIHNldExvZ3MocmVzdW1hYmxlKVxuICAgICAgfSBjYXRjaCAoX2Vycikge1xuICAgICAgICBvbkRvbmUoJ0ZhaWxlZCB0byBsb2FkIGNvbnZlcnNhdGlvbnMnKVxuICAgICAgfSBmaW5hbGx5IHtcbiAgICAgICAgc2V0TG9hZGluZyhmYWxzZSlcbiAgICAgIH1cbiAgICB9LFxuICAgIFtvbkRvbmVdLFxuICApXG5cbiAgUmVhY3QudXNlRWZmZWN0KCgpID0+IHtcbiAgICBhc3luYyBmdW5jdGlvbiBpbml0KCkge1xuICAgICAgY29uc3QgcGF0aHMgPSBhd2FpdCBnZXRXb3JrdHJlZVBhdGhzKGdldE9yaWdpbmFsQ3dkKCkpXG4gICAgICBzZXRXb3JrdHJlZVBhdGhzKHBhdGhzKVxuICAgICAgdm9pZCBsb2FkTG9ncyhmYWxzZSwgcGF0aHMpXG4gICAgfVxuICAgIHZvaWQgaW5pdCgpXG4gIH0sIFtsb2FkTG9nc10pXG5cbiAgY29uc3QgaGFuZGxlVG9nZ2xlQWxsUHJvamVjdHMgPSBSZWFjdC51c2VDYWxsYmFjaygoKSA9PiB7XG4gICAgY29uc3QgbmV3VmFsdWUgPSAhc2hvd0FsbFByb2plY3RzXG4gICAgc2V0U2hvd0FsbFByb2plY3RzKG5ld1ZhbHVlKVxuICAgIHZvaWQgbG9hZExvZ3MobmV3VmFsdWUsIHdvcmt0cmVlUGF0aHMpXG4gIH0sIFtzaG93QWxsUHJvamVjdHMsIGxvYWRMb2dzLCB3b3JrdHJlZVBhdGhzXSlcblxuICBhc3luYyBmdW5jdGlvbiBoYW5kbGVTZWxlY3QobG9nOiBMb2dPcHRpb24pIHtcbiAgICBjb25zdCBzZXNzaW9uSWQgPSB2YWxpZGF0ZVV1aWQoZ2V0U2Vzc2lvbklkRnJvbUxvZyhsb2cpKVxuICAgIGlmICghc2Vzc2lvbklkKSB7XG4gICAgICBvbkRvbmUoJ0ZhaWxlZCB0byByZXN1bWUgY29udmVyc2F0aW9uJylcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIC8vIExvYWQgZnVsbCBtZXNzYWdlcyBmb3IgbGl0ZSBsb2dzXG4gICAgY29uc3QgZnVsbExvZyA9IGlzTGl0ZUxvZyhsb2cpID8gYXdhaXQgbG9hZEZ1bGxMb2cobG9nKSA6IGxvZ1xuXG4gICAgLy8gQ2hlY2sgaWYgdGhpcyBjb252ZXJzYXRpb24gaXMgZnJvbSBhIGRpZmZlcmVudCBkaXJlY3RvcnlcbiAgICBjb25zdCBjcm9zc1Byb2plY3RDaGVjayA9IGNoZWNrQ3Jvc3NQcm9qZWN0UmVzdW1lKFxuICAgICAgZnVsbExvZyxcbiAgICAgIHNob3dBbGxQcm9qZWN0cyxcbiAgICAgIHdvcmt0cmVlUGF0aHMsXG4gICAgKVxuICAgIGlmIChjcm9zc1Byb2plY3RDaGVjay5pc0Nyb3NzUHJvamVjdCkge1xuICAgICAgaWYgKGNyb3NzUHJvamVjdENoZWNrLmlzU2FtZVJlcG9Xb3JrdHJlZSkge1xuICAgICAgICAvLyBTYW1lIHJlcG8gd29ya3RyZWUgLSBjYW4gcmVzdW1lIGRpcmVjdGx5XG4gICAgICAgIHNldFJlc3VtaW5nKHRydWUpXG4gICAgICAgIHZvaWQgb25SZXN1bWUoc2Vzc2lvbklkLCBmdWxsTG9nLCAnc2xhc2hfY29tbWFuZF9waWNrZXInKVxuICAgICAgICByZXR1cm5cbiAgICAgIH1cblxuICAgICAgLy8gRGlmZmVyZW50IHByb2plY3QgLSBzaG93IGNvbW1hbmQgaW5zdGVhZCBvZiByZXN1bWluZ1xuICAgICAgY29uc3QgcmF3ID0gYXdhaXQgc2V0Q2xpcGJvYXJkKGNyb3NzUHJvamVjdENoZWNrLmNvbW1hbmQpXG4gICAgICBpZiAocmF3KSBwcm9jZXNzLnN0ZG91dC53cml0ZShyYXcpXG5cbiAgICAgIC8vIEZvcm1hdCB0aGUgb3V0cHV0IG1lc3NhZ2VcbiAgICAgIGNvbnN0IG1lc3NhZ2UgPSBbXG4gICAgICAgICcnLFxuICAgICAgICAnVGhpcyBjb252ZXJzYXRpb24gaXMgZnJvbSBhIGRpZmZlcmVudCBkaXJlY3RvcnkuJyxcbiAgICAgICAgJycsXG4gICAgICAgICdUbyByZXN1bWUsIHJ1bjonLFxuICAgICAgICBgICAke2Nyb3NzUHJvamVjdENoZWNrLmNvbW1hbmR9YCxcbiAgICAgICAgJycsXG4gICAgICAgICcoQ29tbWFuZCBjb3BpZWQgdG8gY2xpcGJvYXJkKScsXG4gICAgICAgICcnLFxuICAgICAgXS5qb2luKCdcXG4nKVxuXG4gICAgICBvbkRvbmUobWVzc2FnZSwgeyBkaXNwbGF5OiAndXNlcicgfSlcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIC8vIFNhbWUgZGlyZWN0b3J5IC0gcHJvY2VlZCB3aXRoIHJlc3VtZVxuICAgIHNldFJlc3VtaW5nKHRydWUpXG4gICAgdm9pZCBvblJlc3VtZShzZXNzaW9uSWQsIGZ1bGxMb2csICdzbGFzaF9jb21tYW5kX3BpY2tlcicpXG4gIH1cblxuICBmdW5jdGlvbiBoYW5kbGVDYW5jZWwoKSB7XG4gICAgb25Eb25lKCdSZXN1bWUgY2FuY2VsbGVkJywgeyBkaXNwbGF5OiAnc3lzdGVtJyB9KVxuICB9XG5cbiAgaWYgKGxvYWRpbmcpIHtcbiAgICByZXR1cm4gKFxuICAgICAgPEJveD5cbiAgICAgICAgPFNwaW5uZXIgLz5cbiAgICAgICAgPFRleHQ+IExvYWRpbmcgY29udmVyc2F0aW9uc+KApjwvVGV4dD5cbiAgICAgIDwvQm94PlxuICAgIClcbiAgfVxuXG4gIGlmIChyZXN1bWluZykge1xuICAgIHJldHVybiAoXG4gICAgICA8Qm94PlxuICAgICAgICA8U3Bpbm5lciAvPlxuICAgICAgICA8VGV4dD4gUmVzdW1pbmcgY29udmVyc2F0aW9u4oCmPC9UZXh0PlxuICAgICAgPC9Cb3g+XG4gICAgKVxuICB9XG5cbiAgcmV0dXJuIChcbiAgICA8TG9nU2VsZWN0b3JcbiAgICAgIGxvZ3M9e2xvZ3N9XG4gICAgICBtYXhIZWlnaHQ9e2luc2lkZU1vZGFsID8gTWF0aC5mbG9vcihyb3dzIC8gMikgOiByb3dzIC0gMn1cbiAgICAgIG9uQ2FuY2VsPXtoYW5kbGVDYW5jZWx9XG4gICAgICBvblNlbGVjdD17aGFuZGxlU2VsZWN0fVxuICAgICAgb25Mb2dzQ2hhbmdlZD17KCkgPT4gbG9hZExvZ3Moc2hvd0FsbFByb2plY3RzLCB3b3JrdHJlZVBhdGhzKX1cbiAgICAgIHNob3dBbGxQcm9qZWN0cz17c2hvd0FsbFByb2plY3RzfVxuICAgICAgb25Ub2dnbGVBbGxQcm9qZWN0cz17aGFuZGxlVG9nZ2xlQWxsUHJvamVjdHN9XG4gICAgICBvbkFnZW50aWNTZWFyY2g9e2FnZW50aWNTZXNzaW9uU2VhcmNofVxuICAgIC8+XG4gIClcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGZpbHRlclJlc3VtYWJsZVNlc3Npb25zKFxuICBsb2dzOiBMb2dPcHRpb25bXSxcbiAgY3VycmVudFNlc3Npb25JZDogc3RyaW5nLFxuKTogTG9nT3B0aW9uW10ge1xuICByZXR1cm4gbG9ncy5maWx0ZXIoXG4gICAgbCA9PiAhbC5pc1NpZGVjaGFpbiAmJiBnZXRTZXNzaW9uSWRGcm9tTG9nKGwpICE9PSBjdXJyZW50U2Vzc2lvbklkLFxuICApXG59XG5cbmV4cG9ydCBjb25zdCBjYWxsOiBMb2NhbEpTWENvbW1hbmRDYWxsID0gYXN5bmMgKG9uRG9uZSwgY29udGV4dCwgYXJncykgPT4ge1xuICBjb25zdCBvblJlc3VtZSA9IGFzeW5jIChcbiAgICBzZXNzaW9uSWQ6IFVVSUQsXG4gICAgbG9nOiBMb2dPcHRpb24sXG4gICAgZW50cnlwb2ludDogUmVzdW1lRW50cnlwb2ludCxcbiAgKSA9PiB7XG4gICAgdHJ5IHtcbiAgICAgIGF3YWl0IGNvbnRleHQucmVzdW1lPy4oc2Vzc2lvbklkLCBsb2csIGVudHJ5cG9pbnQpXG4gICAgICBvbkRvbmUodW5kZWZpbmVkLCB7IGRpc3BsYXk6ICdza2lwJyB9KVxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBsb2dFcnJvcihlcnJvciBhcyBFcnJvcilcbiAgICAgIG9uRG9uZShgRmFpbGVkIHRvIHJlc3VtZTogJHsoZXJyb3IgYXMgRXJyb3IpLm1lc3NhZ2V9YClcbiAgICB9XG4gIH1cblxuICBjb25zdCBhcmcgPSBhcmdzPy50cmltKClcblxuICAvLyBObyBhcmd1bWVudCBwcm92aWRlZCAtIHNob3cgcGlja2VyXG4gIGlmICghYXJnKSB7XG4gICAgcmV0dXJuIChcbiAgICAgIDxSZXN1bWVDb21tYW5kIGtleT17RGF0ZS5ub3coKX0gb25Eb25lPXtvbkRvbmV9IG9uUmVzdW1lPXtvblJlc3VtZX0gLz5cbiAgICApXG4gIH1cblxuICAvLyBMb2FkIGxvZ3MgdG8gc2VhcmNoIChpbmNsdWRlcyBzYW1lLXJlcG8gd29ya3RyZWVzKVxuICBjb25zdCB3b3JrdHJlZVBhdGhzID0gYXdhaXQgZ2V0V29ya3RyZWVQYXRocyhnZXRPcmlnaW5hbEN3ZCgpKVxuICBjb25zdCBsb2dzID0gYXdhaXQgbG9hZFNhbWVSZXBvTWVzc2FnZUxvZ3Mod29ya3RyZWVQYXRocylcbiAgaWYgKGxvZ3MubGVuZ3RoID09PSAwKSB7XG4gICAgY29uc3QgbWVzc2FnZSA9ICdObyBjb252ZXJzYXRpb25zIGZvdW5kIHRvIHJlc3VtZS4nXG4gICAgcmV0dXJuIChcbiAgICAgIDxSZXN1bWVFcnJvclxuICAgICAgICBtZXNzYWdlPXttZXNzYWdlfVxuICAgICAgICBhcmdzPXthcmd9XG4gICAgICAgIG9uRG9uZT17KCkgPT4gb25Eb25lKG1lc3NhZ2UpfVxuICAgICAgLz5cbiAgICApXG4gIH1cblxuICAvLyBGaXJzdCwgY2hlY2sgaWYgYXJnIGlzIGEgdmFsaWQgVVVJRFxuICBjb25zdCBtYXliZVNlc3Npb25JZCA9IHZhbGlkYXRlVXVpZChhcmcpXG4gIGlmIChtYXliZVNlc3Npb25JZCkge1xuICAgIGNvbnN0IG1hdGNoaW5nTG9ncyA9IGxvZ3NcbiAgICAgIC5maWx0ZXIobCA9PiBnZXRTZXNzaW9uSWRGcm9tTG9nKGwpID09PSBtYXliZVNlc3Npb25JZClcbiAgICAgIC5zb3J0KChhLCBiKSA9PiBiLm1vZGlmaWVkLmdldFRpbWUoKSAtIGEubW9kaWZpZWQuZ2V0VGltZSgpKVxuXG4gICAgaWYgKG1hdGNoaW5nTG9ncy5sZW5ndGggPiAwKSB7XG4gICAgICBjb25zdCBsb2cgPSBtYXRjaGluZ0xvZ3NbMF0hXG4gICAgICBjb25zdCBmdWxsTG9nID0gaXNMaXRlTG9nKGxvZykgPyBhd2FpdCBsb2FkRnVsbExvZyhsb2cpIDogbG9nXG4gICAgICB2b2lkIG9uUmVzdW1lKG1heWJlU2Vzc2lvbklkLCBmdWxsTG9nLCAnc2xhc2hfY29tbWFuZF9zZXNzaW9uX2lkJylcbiAgICAgIHJldHVybiBudWxsXG4gICAgfVxuXG4gICAgLy8gRW5yaWNoZWQgbG9ncyBkaWRuJ3QgZmluZCBpdCDigJQgdHJ5IGRpcmVjdCBmaWxlIGxvb2t1cC4gVGhpcyBoYW5kbGVzXG4gICAgLy8gc2Vzc2lvbnMgZmlsdGVyZWQgb3V0IGJ5IGVucmljaExvZ3MgKGUuZy4sIGZpcnN0IG1lc3NhZ2UgPjE2S0IgbWFrZXNcbiAgICAvLyBmaXJzdFByb21wdCBleHRyYWN0aW9uIGZhaWwsIGNhdXNpbmcgdGhlIHNlc3Npb24gdG8gYmUgZHJvcHBlZCkuXG4gICAgY29uc3QgZGlyZWN0TG9nID0gYXdhaXQgZ2V0TGFzdFNlc3Npb25Mb2cobWF5YmVTZXNzaW9uSWQpXG4gICAgaWYgKGRpcmVjdExvZykge1xuICAgICAgdm9pZCBvblJlc3VtZShtYXliZVNlc3Npb25JZCwgZGlyZWN0TG9nLCAnc2xhc2hfY29tbWFuZF9zZXNzaW9uX2lkJylcbiAgICAgIHJldHVybiBudWxsXG4gICAgfVxuICB9XG5cbiAgLy8gTmV4dCwgdHJ5IGV4YWN0IGN1c3RvbSB0aXRsZSBtYXRjaCAob25seSBpZiBmZWF0dXJlIGlzIGVuYWJsZWQpXG4gIGlmIChpc0N1c3RvbVRpdGxlRW5hYmxlZCgpKSB7XG4gICAgY29uc3QgdGl0bGVNYXRjaGVzID0gYXdhaXQgc2VhcmNoU2Vzc2lvbnNCeUN1c3RvbVRpdGxlKGFyZywge1xuICAgICAgZXhhY3Q6IHRydWUsXG4gICAgfSlcbiAgICBpZiAodGl0bGVNYXRjaGVzLmxlbmd0aCA9PT0gMSkge1xuICAgICAgY29uc3QgbG9nID0gdGl0bGVNYXRjaGVzWzBdIVxuICAgICAgY29uc3Qgc2Vzc2lvbklkID0gZ2V0U2Vzc2lvbklkRnJvbUxvZyhsb2cpXG4gICAgICBpZiAoc2Vzc2lvbklkKSB7XG4gICAgICAgIGNvbnN0IGZ1bGxMb2cgPSBpc0xpdGVMb2cobG9nKSA/IGF3YWl0IGxvYWRGdWxsTG9nKGxvZykgOiBsb2dcbiAgICAgICAgdm9pZCBvblJlc3VtZShzZXNzaW9uSWQsIGZ1bGxMb2csICdzbGFzaF9jb21tYW5kX3RpdGxlJylcbiAgICAgICAgcmV0dXJuIG51bGxcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBNdWx0aXBsZSBtYXRjaGVzIC0gc2hvdyBlcnJvclxuICAgIGlmICh0aXRsZU1hdGNoZXMubGVuZ3RoID4gMSkge1xuICAgICAgY29uc3QgbWVzc2FnZSA9IHJlc3VtZUhlbHBNZXNzYWdlKHtcbiAgICAgICAgcmVzdWx0VHlwZTogJ211bHRpcGxlTWF0Y2hlcycsXG4gICAgICAgIGFyZyxcbiAgICAgICAgY291bnQ6IHRpdGxlTWF0Y2hlcy5sZW5ndGgsXG4gICAgICB9KVxuICAgICAgcmV0dXJuIChcbiAgICAgICAgPFJlc3VtZUVycm9yXG4gICAgICAgICAgbWVzc2FnZT17bWVzc2FnZX1cbiAgICAgICAgICBhcmdzPXthcmd9XG4gICAgICAgICAgb25Eb25lPXsoKSA9PiBvbkRvbmUobWVzc2FnZSl9XG4gICAgICAgIC8+XG4gICAgICApXG4gICAgfVxuICB9XG5cbiAgLy8gTm8gbWF0Y2ggZm91bmQgLSBzaG93IGVycm9yXG4gIGNvbnN0IG1lc3NhZ2UgPSByZXN1bWVIZWxwTWVzc2FnZSh7IHJlc3VsdFR5cGU6ICdzZXNzaW9uTm90Rm91bmQnLCBhcmcgfSlcbiAgcmV0dXJuIChcbiAgICA8UmVzdW1lRXJyb3IgbWVzc2FnZT17bWVzc2FnZX0gYXJncz17YXJnfSBvbkRvbmU9eygpID0+IG9uRG9uZShtZXNzYWdlKX0gLz5cbiAgKVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsT0FBT0EsS0FBSyxNQUFNLE9BQU87QUFDekIsY0FBY0MsSUFBSSxRQUFRLFFBQVE7QUFDbEMsT0FBT0MsT0FBTyxNQUFNLFNBQVM7QUFDN0IsT0FBTyxLQUFLQyxLQUFLLE1BQU0sT0FBTztBQUM5QixTQUFTQyxjQUFjLEVBQUVDLFlBQVksUUFBUSwwQkFBMEI7QUFDdkUsY0FBY0Msb0JBQW9CLEVBQUVDLGdCQUFnQixRQUFRLG1CQUFtQjtBQUMvRSxTQUFTQyxXQUFXLFFBQVEsaUNBQWlDO0FBQzdELFNBQVNDLGVBQWUsUUFBUSxxQ0FBcUM7QUFDckUsU0FBU0MsT0FBTyxRQUFRLDZCQUE2QjtBQUNyRCxTQUFTQyxnQkFBZ0IsUUFBUSwrQkFBK0I7QUFDaEUsU0FBU0MsZUFBZSxR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