/ tools / FileWriteTool / UI.tsx
UI.tsx
  1  import { c as _c } from "react/compiler-runtime";
  2  import type { ToolResultBlockParam } from '@anthropic-ai/sdk/resources/index.mjs';
  3  import type { StructuredPatchHunk } from 'diff';
  4  import { isAbsolute, relative, resolve } from 'path';
  5  import * as React from 'react';
  6  import { Suspense, use, useState } from 'react';
  7  import { MessageResponse } from 'src/components/MessageResponse.js';
  8  import { extractTag } from 'src/utils/messages.js';
  9  import { CtrlOToExpand } from '../../components/CtrlOToExpand.js';
 10  import { FallbackToolUseErrorMessage } from '../../components/FallbackToolUseErrorMessage.js';
 11  import { FileEditToolUpdatedMessage } from '../../components/FileEditToolUpdatedMessage.js';
 12  import { FileEditToolUseRejectedMessage } from '../../components/FileEditToolUseRejectedMessage.js';
 13  import { FilePathLink } from '../../components/FilePathLink.js';
 14  import { HighlightedCode } from '../../components/HighlightedCode.js';
 15  import { useTerminalSize } from '../../hooks/useTerminalSize.js';
 16  import { Box, Text } from '../../ink.js';
 17  import type { ToolProgressData } from '../../Tool.js';
 18  import type { ProgressMessage } from '../../types/message.js';
 19  import { getCwd } from '../../utils/cwd.js';
 20  import { getPatchForDisplay } from '../../utils/diff.js';
 21  import { getDisplayPath } from '../../utils/file.js';
 22  import { logError } from '../../utils/log.js';
 23  import { getPlansDirectory } from '../../utils/plans.js';
 24  import { openForScan, readCapped } from '../../utils/readEditContext.js';
 25  import type { Output } from './FileWriteTool.js';
 26  const MAX_LINES_TO_RENDER = 10;
 27  // Model output uses \n regardless of platform, so always split on \n.
 28  // os.EOL is \r\n on Windows, which would give numLines=1 for all files.
 29  const EOL = '\n';
 30  
 31  /**
 32   * Count visible lines in file content. A trailing newline is treated as a
 33   * line terminator (not a new empty line), matching editor line numbering.
 34   */
 35  export function countLines(content: string): number {
 36    const parts = content.split(EOL);
 37    return content.endsWith(EOL) ? parts.length - 1 : parts.length;
 38  }
 39  function FileWriteToolCreatedMessage(t0) {
 40    const $ = _c(25);
 41    const {
 42      filePath,
 43      content,
 44      verbose
 45    } = t0;
 46    const {
 47      columns
 48    } = useTerminalSize();
 49    const contentWithFallback = content || "(No content)";
 50    const numLines = countLines(content);
 51    const plusLines = numLines - MAX_LINES_TO_RENDER;
 52    let t1;
 53    if ($[0] !== numLines) {
 54      t1 = <Text bold={true}>{numLines}</Text>;
 55      $[0] = numLines;
 56      $[1] = t1;
 57    } else {
 58      t1 = $[1];
 59    }
 60    let t2;
 61    if ($[2] !== filePath || $[3] !== verbose) {
 62      t2 = verbose ? filePath : relative(getCwd(), filePath);
 63      $[2] = filePath;
 64      $[3] = verbose;
 65      $[4] = t2;
 66    } else {
 67      t2 = $[4];
 68    }
 69    let t3;
 70    if ($[5] !== t2) {
 71      t3 = <Text bold={true}>{t2}</Text>;
 72      $[5] = t2;
 73      $[6] = t3;
 74    } else {
 75      t3 = $[6];
 76    }
 77    let t4;
 78    if ($[7] !== t1 || $[8] !== t3) {
 79      t4 = <Text>Wrote {t1} lines to{" "}{t3}</Text>;
 80      $[7] = t1;
 81      $[8] = t3;
 82      $[9] = t4;
 83    } else {
 84      t4 = $[9];
 85    }
 86    let t5;
 87    if ($[10] !== contentWithFallback || $[11] !== verbose) {
 88      t5 = verbose ? contentWithFallback : contentWithFallback.split("\n").slice(0, MAX_LINES_TO_RENDER).join("\n");
 89      $[10] = contentWithFallback;
 90      $[11] = verbose;
 91      $[12] = t5;
 92    } else {
 93      t5 = $[12];
 94    }
 95    const t6 = columns - 12;
 96    let t7;
 97    if ($[13] !== filePath || $[14] !== t5 || $[15] !== t6) {
 98      t7 = <Box flexDirection="column"><HighlightedCode code={t5} filePath={filePath} width={t6} /></Box>;
 99      $[13] = filePath;
100      $[14] = t5;
101      $[15] = t6;
102      $[16] = t7;
103    } else {
104      t7 = $[16];
105    }
106    let t8;
107    if ($[17] !== numLines || $[18] !== plusLines || $[19] !== verbose) {
108      t8 = !verbose && plusLines > 0 && <Text dimColor={true}>… +{plusLines} {plusLines === 1 ? "line" : "lines"}{" "}{numLines > 0 && <CtrlOToExpand />}</Text>;
109      $[17] = numLines;
110      $[18] = plusLines;
111      $[19] = verbose;
112      $[20] = t8;
113    } else {
114      t8 = $[20];
115    }
116    let t9;
117    if ($[21] !== t4 || $[22] !== t7 || $[23] !== t8) {
118      t9 = <MessageResponse><Box flexDirection="column">{t4}{t7}{t8}</Box></MessageResponse>;
119      $[21] = t4;
120      $[22] = t7;
121      $[23] = t8;
122      $[24] = t9;
123    } else {
124      t9 = $[24];
125    }
126    return t9;
127  }
128  export function userFacingName(input: Partial<{
129    file_path: string;
130    content: string;
131  }> | undefined): string {
132    if (input?.file_path?.startsWith(getPlansDirectory())) {
133      return 'Updated plan';
134    }
135    return 'Write';
136  }
137  
138  /** Gates fullscreen click-to-expand. Only `create` truncates (to
139   *  MAX_LINES_TO_RENDER); `update` renders the full diff regardless of verbose.
140   *  Called per visible message on hover/scroll, so early-exit after finding the
141   *  (MAX+1)th line instead of splitting the whole (possibly huge) content. */
142  export function isResultTruncated({
143    type,
144    content
145  }: Output): boolean {
146    if (type !== 'create') return false;
147    let pos = 0;
148    for (let i = 0; i < MAX_LINES_TO_RENDER; i++) {
149      pos = content.indexOf(EOL, pos);
150      if (pos === -1) return false;
151      pos++;
152    }
153    // countLines treats a trailing EOL as a terminator, not a new line
154    return pos < content.length;
155  }
156  export function getToolUseSummary(input: Partial<{
157    file_path: string;
158    content: string;
159  }> | undefined): string | null {
160    if (!input?.file_path) {
161      return null;
162    }
163    return getDisplayPath(input.file_path);
164  }
165  export function renderToolUseMessage(input: Partial<{
166    file_path: string;
167    content: string;
168  }>, {
169    verbose
170  }: {
171    verbose: boolean;
172  }): React.ReactNode {
173    if (!input.file_path) {
174      return null;
175    }
176    // For plan files, path is already in userFacingName
177    if (input.file_path.startsWith(getPlansDirectory())) {
178      return '';
179    }
180    return <FilePathLink filePath={input.file_path}>
181        {verbose ? input.file_path : getDisplayPath(input.file_path)}
182      </FilePathLink>;
183  }
184  export function renderToolUseRejectedMessage({
185    file_path,
186    content
187  }: {
188    file_path: string;
189    content: string;
190  }, {
191    style,
192    verbose
193  }: {
194    style?: 'condensed';
195    verbose: boolean;
196  }): React.ReactNode {
197    return <WriteRejectionDiff filePath={file_path} content={content} style={style} verbose={verbose} />;
198  }
199  type RejectionDiffData = {
200    type: 'create';
201  } | {
202    type: 'update';
203    patch: StructuredPatchHunk[];
204    oldContent: string;
205  } | {
206    type: 'error';
207  };
208  function WriteRejectionDiff(t0) {
209    const $ = _c(20);
210    const {
211      filePath,
212      content,
213      style,
214      verbose
215    } = t0;
216    let t1;
217    if ($[0] !== content || $[1] !== filePath) {
218      t1 = () => loadRejectionDiff(filePath, content);
219      $[0] = content;
220      $[1] = filePath;
221      $[2] = t1;
222    } else {
223      t1 = $[2];
224    }
225    const [dataPromise] = useState(t1);
226    let t2;
227    if ($[3] !== content) {
228      t2 = content.split("\n")[0] ?? null;
229      $[3] = content;
230      $[4] = t2;
231    } else {
232      t2 = $[4];
233    }
234    const firstLine = t2;
235    let t3;
236    if ($[5] !== content || $[6] !== filePath || $[7] !== firstLine || $[8] !== verbose) {
237      t3 = <FileEditToolUseRejectedMessage file_path={filePath} operation="write" content={content} firstLine={firstLine} verbose={verbose} />;
238      $[5] = content;
239      $[6] = filePath;
240      $[7] = firstLine;
241      $[8] = verbose;
242      $[9] = t3;
243    } else {
244      t3 = $[9];
245    }
246    const createFallback = t3;
247    let t4;
248    if ($[10] !== createFallback || $[11] !== dataPromise || $[12] !== filePath || $[13] !== firstLine || $[14] !== style || $[15] !== verbose) {
249      t4 = <WriteRejectionBody promise={dataPromise} filePath={filePath} firstLine={firstLine} createFallback={createFallback} style={style} verbose={verbose} />;
250      $[10] = createFallback;
251      $[11] = dataPromise;
252      $[12] = filePath;
253      $[13] = firstLine;
254      $[14] = style;
255      $[15] = verbose;
256      $[16] = t4;
257    } else {
258      t4 = $[16];
259    }
260    let t5;
261    if ($[17] !== createFallback || $[18] !== t4) {
262      t5 = <Suspense fallback={createFallback}>{t4}</Suspense>;
263      $[17] = createFallback;
264      $[18] = t4;
265      $[19] = t5;
266    } else {
267      t5 = $[19];
268    }
269    return t5;
270  }
271  function WriteRejectionBody(t0) {
272    const $ = _c(8);
273    const {
274      promise,
275      filePath,
276      firstLine,
277      createFallback,
278      style,
279      verbose
280    } = t0;
281    const data = use(promise);
282    if (data.type === "create") {
283      return createFallback;
284    }
285    if (data.type === "error") {
286      let t1;
287      if ($[0] === Symbol.for("react.memo_cache_sentinel")) {
288        t1 = <MessageResponse><Text>(No changes)</Text></MessageResponse>;
289        $[0] = t1;
290      } else {
291        t1 = $[0];
292      }
293      return t1;
294    }
295    let t1;
296    if ($[1] !== data.oldContent || $[2] !== data.patch || $[3] !== filePath || $[4] !== firstLine || $[5] !== style || $[6] !== verbose) {
297      t1 = <FileEditToolUseRejectedMessage file_path={filePath} operation="update" patch={data.patch} firstLine={firstLine} fileContent={data.oldContent} style={style} verbose={verbose} />;
298      $[1] = data.oldContent;
299      $[2] = data.patch;
300      $[3] = filePath;
301      $[4] = firstLine;
302      $[5] = style;
303      $[6] = verbose;
304      $[7] = t1;
305    } else {
306      t1 = $[7];
307    }
308    return t1;
309  }
310  async function loadRejectionDiff(filePath: string, content: string): Promise<RejectionDiffData> {
311    try {
312      const fullFilePath = isAbsolute(filePath) ? filePath : resolve(getCwd(), filePath);
313      const handle = await openForScan(fullFilePath);
314      if (handle === null) return {
315        type: 'create'
316      };
317      let oldContent: string | null;
318      try {
319        oldContent = await readCapped(handle);
320      } finally {
321        await handle.close();
322      }
323      // File exceeds MAX_SCAN_BYTES — fall back to the create view rather than
324      // OOMing on a diff of a multi-GB file.
325      if (oldContent === null) return {
326        type: 'create'
327      };
328      const patch = getPatchForDisplay({
329        filePath,
330        fileContents: oldContent,
331        edits: [{
332          old_string: oldContent,
333          new_string: content,
334          replace_all: false
335        }]
336      });
337      return {
338        type: 'update',
339        patch,
340        oldContent
341      };
342    } catch (e) {
343      // User may have manually applied the change while the diff was shown.
344      logError(e as Error);
345      return {
346        type: 'error'
347      };
348    }
349  }
350  export function renderToolUseErrorMessage(result: ToolResultBlockParam['content'], {
351    verbose
352  }: {
353    verbose: boolean;
354  }): React.ReactNode {
355    if (!verbose && typeof result === 'string' && extractTag(result, 'tool_use_error')) {
356      return <MessageResponse>
357          <Text color="error">Error writing file</Text>
358        </MessageResponse>;
359    }
360    return <FallbackToolUseErrorMessage result={result} verbose={verbose} />;
361  }
362  export function renderToolResultMessage({
363    filePath,
364    content,
365    structuredPatch,
366    type,
367    originalFile
368  }: Output, _progressMessagesForMessage: ProgressMessage<ToolProgressData>[], {
369    style,
370    verbose
371  }: {
372    style?: 'condensed';
373    verbose: boolean;
374  }): React.ReactNode {
375    switch (type) {
376      case 'create':
377        {
378          const isPlanFile = filePath.startsWith(getPlansDirectory());
379  
380          // Plan files: invert condensed behavior
381          // - Regular mode: just show hint (user can type /plan to see full content)
382          // - Condensed mode (subagent view): show full content
383          if (isPlanFile && !verbose) {
384            if (style !== 'condensed') {
385              return <MessageResponse>
386                <Text dimColor>/plan to preview</Text>
387              </MessageResponse>;
388            }
389          } else if (style === 'condensed' && !verbose) {
390            const numLines = countLines(content);
391            return <Text>
392              Wrote <Text bold>{numLines}</Text> lines to{' '}
393              <Text bold>{relative(getCwd(), filePath)}</Text>
394            </Text>;
395          }
396          return <FileWriteToolCreatedMessage filePath={filePath} content={content} verbose={verbose} />;
397        }
398      case 'update':
399        {
400          const isPlanFile = filePath.startsWith(getPlansDirectory());
401          return <FileEditToolUpdatedMessage filePath={filePath} structuredPatch={structuredPatch} firstLine={content.split('\n')[0] ?? null} fileContent={originalFile ?? undefined} style={style} verbose={verbose} previewHint={isPlanFile ? '/plan to preview' : undefined} />;
402        }
403    }
404  }
405  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJUb29sUmVzdWx0QmxvY2tQYXJhbSIsIlN0cnVjdHVyZWRQYXRjaEh1bmsiLCJpc0Fic29sdXRlIiwicmVsYXRpdmUiLCJyZXNvbHZlIiwiUmVhY3QiLCJTdXNwZW5zZSIsInVzZSIsInVzZVN0YXRlIiwiTWVzc2FnZVJlc3BvbnNlIiwiZXh0cmFjdFRhZyIsIkN0cmxPVG9FeHBhbmQiLCJGYWxsYmFja1Rvb2xVc2VFcnJvck1lc3NhZ2UiLCJGaWxlRWRpdFRvb2xVcGRhdGVkTWVzc2FnZSIsIkZpbGVFZGl0VG9vbFVzZVJlamVjdGVkTWVzc2FnZSIsIkZpbGVQYXRoTGluayIsIkhpZ2hsaWdodGVkQ29kZSIsInVzZVRlcm1pbmFsU2l6ZSIsIkJveCIsIlRleHQiLCJUb29sUHJvZ3Jlc3NEYXRhIiwiUHJvZ3Jlc3NNZXNzYWdlIiwiZ2V0Q3dkIiwiZ2V0UGF0Y2hGb3JEaXNwbGF5IiwiZ2V0RGlzcGxheVBhdGgiLCJsb2dFcnJvciIsImdldFBsYW5zRGlyZWN0b3J5Iiwib3BlbkZvclNjYW4iLCJyZWFkQ2FwcGVkIiwiT3V0cHV0IiwiTUFYX0xJTkVTX1RPX1JFTkRFUiIsIkVPTCIsImNvdW50TGluZXMiLCJjb250ZW50IiwicGFydHMiLCJzcGxpdCIsImVuZHNXaXRoIiwibGVuZ3RoIiwiRmlsZVdyaXRlVG9vbENyZWF0ZWRNZXNzYWdlIiwidDAiLCIkIiwiX2MiLCJmaWxlUGF0aCIsInZlcmJvc2UiLCJjb2x1bW5zIiwiY29udGVudFdpdGhGYWxsYmFjayIsIm51bUxpbmVzIiwicGx1c0xpbmVzIiwidDEiLCJ0MiIsInQzIiwidDQiLCJ0NSIsInNsaWNlIiwiam9pbiIsInQ2IiwidDciLCJ0OCIsInQ5IiwidXNlckZhY2luZ05hbWUiLCJpbnB1dCIsIlBhcnRpYWwiLCJmaWxlX3BhdGgiLCJzdGFydHNXaXRoIiwiaXNSZXN1bHRUcnVuY2F0ZWQiLCJ0eXBlIiwicG9zIiwiaSIsImluZGV4T2YiLCJnZXRUb29sVXNlU3VtbWFyeSIsInJlbmRlclRvb2xVc2VNZXNzYWdlIiwiUmVhY3ROb2RlIiwicmVuZGVyVG9vbFVzZVJlamVjdGVkTWVzc2FnZSIsInN0eWxlIiwiUmVqZWN0aW9uRGlmZkRhdGEiLCJwYXRjaCIsIm9sZENvbnRlbnQiLCJXcml0ZVJlamVjdGlvbkRpZmYiLCJsb2FkUmVqZWN0aW9uRGlmZiIsImRhdGFQcm9taXNlIiwiZmlyc3RMaW5lIiwiY3JlYXRlRmFsbGJhY2siLCJXcml0ZVJlamVjdGlvbkJvZHkiLCJwcm9taXNlIiwiZGF0YSIsIlN5bWJvbCIsImZvciIsIlByb21pc2UiLCJmdWxsRmlsZVBhdGgiLCJoYW5kbGUiLCJjbG9zZSIsImZpbGVDb250ZW50cyIsImVkaXRzIiwib2xkX3N0cmluZyIsIm5ld19zdHJpbmciLCJyZXBsYWNlX2FsbCIsImUiLCJFcnJvciIsInJlbmRlclRvb2xVc2VFcnJvck1lc3NhZ2UiLCJyZXN1bHQiLCJyZW5kZXJUb29sUmVzdWx0TWVzc2FnZSIsInN0cnVjdHVyZWRQYXRjaCIsIm9yaWdpbmFsRmlsZSIsIl9wcm9ncmVzc01lc3NhZ2VzRm9yTWVzc2FnZSIsImlzUGxhbkZpbGUiLCJ1bmRlZmluZWQiXSwic291cmNlcyI6WyJVSS50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBUb29sUmVzdWx0QmxvY2tQYXJhbSB9IGZyb20gJ0BhbnRocm9waWMtYWkvc2RrL3Jlc291cmNlcy9pbmRleC5tanMnXG5pbXBvcnQgdHlwZSB7IFN0cnVjdHVyZWRQYXRjaEh1bmsgfSBmcm9tICdkaWZmJ1xuaW1wb3J0IHsgaXNBYnNvbHV0ZSwgcmVsYXRpdmUsIHJlc29sdmUgfSBmcm9tICdwYXRoJ1xuaW1wb3J0ICogYXMgUmVhY3QgZnJvbSAncmVhY3QnXG5pbXBvcnQgeyBTdXNwZW5zZSwgdXNlLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgTWVzc2FnZVJlc3BvbnNlIH0gZnJvbSAnc3JjL2NvbXBvbmVudHMvTWVzc2FnZVJlc3BvbnNlLmpzJ1xuaW1wb3J0IHsgZXh0cmFjdFRhZyB9IGZyb20gJ3NyYy91dGlscy9tZXNzYWdlcy5qcydcbmltcG9ydCB7IEN0cmxPVG9FeHBhbmQgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL0N0cmxPVG9FeHBhbmQuanMnXG5pbXBvcnQgeyBGYWxsYmFja1Rvb2xVc2VFcnJvck1lc3NhZ2UgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL0ZhbGxiYWNrVG9vbFVzZUVycm9yTWVzc2FnZS5qcydcbmltcG9ydCB7IEZpbGVFZGl0VG9vbFVwZGF0ZWRNZXNzYWdlIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9GaWxlRWRpdFRvb2xVcGRhdGVkTWVzc2FnZS5qcydcbmltcG9ydCB7IEZpbGVFZGl0VG9vbFVzZVJlamVjdGVkTWVzc2FnZSB9IGZyb20gJy4uLy4uL2NvbXBvbmVudHMvRmlsZUVkaXRUb29sVXNlUmVqZWN0ZWRNZXNzYWdlLmpzJ1xuaW1wb3J0IHsgRmlsZVBhdGhMaW5rIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9GaWxlUGF0aExpbmsuanMnXG5pbXBvcnQgeyBIaWdobGlnaHRlZENvZGUgfSBmcm9tICcuLi8uLi9jb21wb25lbnRzL0hpZ2hsaWdodGVkQ29kZS5qcydcbmltcG9ydCB7IHVzZVRlcm1pbmFsU2l6ZSB9IGZyb20gJy4uLy4uL2hvb2tzL3VzZVRlcm1pbmFsU2l6ZS5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB0eXBlIHsgVG9vbFByb2dyZXNzRGF0YSB9IGZyb20gJy4uLy4uL1Rvb2wuanMnXG5pbXBvcnQgdHlwZSB7IFByb2dyZXNzTWVzc2FnZSB9IGZyb20gJy4uLy4uL3R5cGVzL21lc3NhZ2UuanMnXG5pbXBvcnQgeyBnZXRDd2QgfSBmcm9tICcuLi8uLi91dGlscy9jd2QuanMnXG5pbXBvcnQgeyBnZXRQYXRjaEZvckRpc3BsYXkgfSBmcm9tICcuLi8uLi91dGlscy9kaWZmLmpzJ1xuaW1wb3J0IHsgZ2V0RGlzcGxheVBhdGggfSBmcm9tICcuLi8uLi91dGlscy9maWxlLmpzJ1xuaW1wb3J0IHsgbG9nRXJyb3IgfSBmcm9tICcuLi8uLi91dGlscy9sb2cuanMnXG5pbXBvcnQgeyBnZXRQbGFuc0RpcmVjdG9yeSB9IGZyb20gJy4uLy4uL3V0aWxzL3BsYW5zLmpzJ1xuaW1wb3J0IHsgb3BlbkZvclNjYW4sIHJlYWRDYXBwZWQgfSBmcm9tICcuLi8uLi91dGlscy9yZWFkRWRpdENvbnRleHQuanMnXG5pbXBvcnQgdHlwZSB7IE91dHB1dCB9IGZyb20gJy4vRmlsZVdyaXRlVG9vbC5qcydcblxuY29uc3QgTUFYX0xJTkVTX1RPX1JFTkRFUiA9IDEwXG4vLyBNb2RlbCBvdXRwdXQgdXNlcyBcXG4gcmVnYXJkbGVzcyBvZiBwbGF0Zm9ybSwgc28gYWx3YXlzIHNwbGl0IG9uIFxcbi5cbi8vIG9zLkVPTCBpcyBcXHJcXG4gb24gV2luZG93cywgd2hpY2ggd291bGQgZ2l2ZSBudW1MaW5lcz0xIGZvciBhbGwgZmlsZXMuXG5jb25zdCBFT0wgPSAnXFxuJ1xuXG4vKipcbiAqIENvdW50IHZpc2libGUgbGluZXMgaW4gZmlsZSBjb250ZW50LiBBIHRyYWlsaW5nIG5ld2xpbmUgaXMgdHJlYXRlZCBhcyBhXG4gKiBsaW5lIHRlcm1pbmF0b3IgKG5vdCBhIG5ldyBlbXB0eSBsaW5lKSwgbWF0Y2hpbmcgZWRpdG9yIGxpbmUgbnVtYmVyaW5nLlxuICovXG5leHBvcnQgZnVuY3Rpb24gY291bnRMaW5lcyhjb250ZW50OiBzdHJpbmcpOiBudW1iZXIge1xuICBjb25zdCBwYXJ0cyA9IGNvbnRlbnQuc3BsaXQoRU9MKVxuICByZXR1cm4gY29udGVudC5lbmRzV2l0aChFT0wpID8gcGFydHMubGVuZ3RoIC0gMSA6IHBhcnRzLmxlbmd0aFxufVxuXG5mdW5jdGlvbiBGaWxlV3JpdGVUb29sQ3JlYXRlZE1lc3NhZ2Uoe1xuICBmaWxlUGF0aCxcbiAgY29udGVudCxcbiAgdmVyYm9zZSxcbn06IHtcbiAgZmlsZVBhdGg6IHN0cmluZ1xuICBjb250ZW50OiBzdHJpbmdcbiAgdmVyYm9zZTogYm9vbGVhblxufSk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IHsgY29sdW1ucyB9ID0gdXNlVGVybWluYWxTaXplKClcbiAgY29uc3QgY29udGVudFdpdGhGYWxsYmFjayA9IGNvbnRlbnQgfHwgJyhObyBjb250ZW50KSdcbiAgY29uc3QgbnVtTGluZXMgPSBjb3VudExpbmVzKGNvbnRlbnQpXG4gIGNvbnN0IHBsdXNMaW5lcyA9IG51bUxpbmVzIC0gTUFYX0xJTkVTX1RPX1JFTkRFUlxuXG4gIHJldHVybiAoXG4gICAgPE1lc3NhZ2VSZXNwb25zZT5cbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgICA8VGV4dD5cbiAgICAgICAgICBXcm90ZSA8VGV4dCBib2xkPntudW1MaW5lc308L1RleHQ+IGxpbmVzIHRveycgJ31cbiAgICAgICAgICA8VGV4dCBib2xkPnt2ZXJib3NlID8gZmlsZVBhdGggOiByZWxhdGl2ZShnZXRDd2QoKSwgZmlsZVBhdGgpfTwvVGV4dD5cbiAgICAgICAgPC9UZXh0PlxuICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIj5cbiAgICAgICAgICA8SGlnaGxpZ2h0ZWRDb2RlXG4gICAgICAgICAgICBjb2RlPXtcbiAgICAgICAgICAgICAgdmVyYm9zZVxuICAgICAgICAgICAgICAgID8gY29udGVudFdpdGhGYWxsYmFja1xuICAgICAgICAgICAgICAgIDogY29udGVudFdpdGhGYWxsYmFja1xuICAgICAgICAgICAgICAgICAgICAuc3BsaXQoJ1xcbicpXG4gICAgICAgICAgICAgICAgICAgIC5zbGljZSgwLCBNQVhfTElORVNfVE9fUkVOREVSKVxuICAgICAgICAgICAgICAgICAgICAuam9pbignXFxuJylcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGZpbGVQYXRoPXtmaWxlUGF0aH1cbiAgICAgICAgICAgIHdpZHRoPXtjb2x1bW5zIC0gMTJ9XG4gICAgICAgICAgLz5cbiAgICAgICAgPC9Cb3g+XG4gICAgICAgIHshdmVyYm9zZSAmJiBwbHVzTGluZXMgPiAwICYmIChcbiAgICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICAgIOKApiAre3BsdXNMaW5lc30ge3BsdXNMaW5lcyA9PT0gMSA/ICdsaW5lJyA6ICdsaW5lcyd9eycgJ31cbiAgICAgICAgICAgIHtudW1MaW5lcyA+IDAgJiYgPEN0cmxPVG9FeHBhbmQgLz59XG4gICAgICAgICAgPC9UZXh0PlxuICAgICAgICApfVxuICAgICAgPC9Cb3g+XG4gICAgPC9NZXNzYWdlUmVzcG9uc2U+XG4gIClcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIHVzZXJGYWNpbmdOYW1lKFxuICBpbnB1dDogUGFydGlhbDx7IGZpbGVfcGF0aDogc3RyaW5nOyBjb250ZW50OiBzdHJpbmcgfT4gfCB1bmRlZmluZWQsXG4pOiBzdHJpbmcge1xuICBpZiAoaW5wdXQ/LmZpbGVfcGF0aD8uc3RhcnRzV2l0aChnZXRQbGFuc0RpcmVjdG9yeSgpKSkge1xuICAgIHJldHVybiAnVXBkYXRlZCBwbGFuJ1xuICB9XG4gIHJldHVybiAnV3JpdGUnXG59XG5cbi8qKiBHYXRlcyBmdWxsc2NyZWVuIGNsaWNrLXRvLWV4cGFuZC4gT25seSBgY3JlYXRlYCB0cnVuY2F0ZXMgKHRvXG4gKiAgTUFYX0xJTkVTX1RPX1JFTkRFUik7IGB1cGRhdGVgIHJlbmRlcnMgdGhlIGZ1bGwgZGlmZiByZWdhcmRsZXNzIG9mIHZlcmJvc2UuXG4gKiAgQ2FsbGVkIHBlciB2aXNpYmxlIG1lc3NhZ2Ugb24gaG92ZXIvc2Nyb2xsLCBzbyBlYXJseS1leGl0IGFmdGVyIGZpbmRpbmcgdGhlXG4gKiAgKE1BWCsxKXRoIGxpbmUgaW5zdGVhZCBvZiBzcGxpdHRpbmcgdGhlIHdob2xlIChwb3NzaWJseSBodWdlKSBjb250ZW50LiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGlzUmVzdWx0VHJ1bmNhdGVkKHsgdHlwZSwgY29udGVudCB9OiBPdXRwdXQpOiBib29sZWFuIHtcbiAgaWYgKHR5cGUgIT09ICdjcmVhdGUnKSByZXR1cm4gZmFsc2VcbiAgbGV0IHBvcyA9IDBcbiAgZm9yIChsZXQgaSA9IDA7IGkgPCBNQVhfTElORVNfVE9fUkVOREVSOyBpKyspIHtcbiAgICBwb3MgPSBjb250ZW50LmluZGV4T2YoRU9MLCBwb3MpXG4gICAgaWYgKHBvcyA9PT0gLTEpIHJldHVybiBmYWxzZVxuICAgIHBvcysrXG4gIH1cbiAgLy8gY291bnRMaW5lcyB0cmVhdHMgYSB0cmFpbGluZyBFT0wgYXMgYSB0ZXJtaW5hdG9yLCBub3QgYSBuZXcgbGluZVxuICByZXR1cm4gcG9zIDwgY29udGVudC5sZW5ndGhcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGdldFRvb2xVc2VTdW1tYXJ5KFxuICBpbnB1dDogUGFydGlhbDx7IGZpbGVfcGF0aDogc3RyaW5nOyBjb250ZW50OiBzdHJpbmcgfT4gfCB1bmRlZmluZWQsXG4pOiBzdHJpbmcgfCBudWxsIHtcbiAgaWYgKCFpbnB1dD8uZmlsZV9wYXRoKSB7XG4gICAgcmV0dXJuIG51bGxcbiAgfVxuICByZXR1cm4gZ2V0RGlzcGxheVBhdGgoaW5wdXQuZmlsZV9wYXRoKVxufVxuXG5leHBvcnQgZnVuY3Rpb24gcmVuZGVyVG9vbFVzZU1lc3NhZ2UoXG4gIGlucHV0OiBQYXJ0aWFsPHsgZmlsZV9wYXRoOiBzdHJpbmc7IGNvbnRlbnQ6IHN0cmluZyB9PixcbiAgeyB2ZXJib3NlIH06IHsgdmVyYm9zZTogYm9vbGVhbiB9LFxuKTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgaWYgKCFpbnB1dC5maWxlX3BhdGgpIHtcbiAgICByZXR1cm4gbnVsbFxuICB9XG4gIC8vIEZvciBwbGFuIGZpbGVzLCBwYXRoIGlzIGFscmVhZHkgaW4gdXNlckZhY2luZ05hbWVcbiAgaWYgKGlucHV0LmZpbGVfcGF0aC5zdGFydHNXaXRoKGdldFBsYW5zRGlyZWN0b3J5KCkpKSB7XG4gICAgcmV0dXJuICcnXG4gIH1cbiAgcmV0dXJuIChcbiAgICA8RmlsZVBhdGhMaW5rIGZpbGVQYXRoPXtpbnB1dC5maWxlX3BhdGh9PlxuICAgICAge3ZlcmJvc2UgPyBpbnB1dC5maWxlX3BhdGggOiBnZXREaXNwbGF5UGF0aChpbnB1dC5maWxlX3BhdGgpfVxuICAgIDwvRmlsZVBhdGhMaW5rPlxuICApXG59XG5cbmV4cG9ydCBmdW5jdGlvbiByZW5kZXJUb29sVXNlUmVqZWN0ZWRNZXNzYWdlKFxuICB7IGZpbGVfcGF0aCwgY29udGVudCB9OiB7IGZpbGVfcGF0aDogc3RyaW5nOyBjb250ZW50OiBzdHJpbmcgfSxcbiAgeyBzdHlsZSwgdmVyYm9zZSB9OiB7IHN0eWxlPzogJ2NvbmRlbnNlZCc7IHZlcmJvc2U6IGJvb2xlYW4gfSxcbik6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIHJldHVybiAoXG4gICAgPFdyaXRlUmVqZWN0aW9uRGlmZlxuICAgICAgZmlsZVBhdGg9e2ZpbGVfcGF0aH1cbiAgICAgIGNvbnRlbnQ9e2NvbnRlbnR9XG4gICAgICBzdHlsZT17c3R5bGV9XG4gICAgICB2ZXJib3NlPXt2ZXJib3NlfVxuICAgIC8+XG4gIClcbn1cblxudHlwZSBSZWplY3Rpb25EaWZmRGF0YSA9XG4gIHwgeyB0eXBlOiAnY3JlYXRlJyB9XG4gIHwgeyB0eXBlOiAndXBkYXRlJzsgcGF0Y2g6IFN0cnVjdHVyZWRQYXRjaEh1bmtbXTsgb2xkQ29udGVudDogc3RyaW5nIH1cbiAgfCB7IHR5cGU6ICdlcnJvcicgfVxuXG5mdW5jdGlvbiBXcml0ZVJlamVjdGlvbkRpZmYoe1xuICBmaWxlUGF0aCxcbiAgY29udGVudCxcbiAgc3R5bGUsXG4gIHZlcmJvc2UsXG59OiB7XG4gIGZpbGVQYXRoOiBzdHJpbmdcbiAgY29udGVudDogc3RyaW5nXG4gIHN0eWxlPzogJ2NvbmRlbnNlZCdcbiAgdmVyYm9zZTogYm9vbGVhblxufSk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IFtkYXRhUHJvbWlzZV0gPSB1c2VTdGF0ZSgoKSA9PiBsb2FkUmVqZWN0aW9uRGlmZihmaWxlUGF0aCwgY29udGVudCkpXG4gIGNvbnN0IGZpcnN0TGluZSA9IGNvbnRlbnQuc3BsaXQoJ1xcbicpWzBdID8/IG51bGxcbiAgY29uc3QgY3JlYXRlRmFsbGJhY2sgPSAoXG4gICAgPEZpbGVFZGl0VG9vbFVzZVJlamVjdGVkTWVzc2FnZVxuICAgICAgZmlsZV9wYXRoPXtmaWxlUGF0aH1cbiAgICAgIG9wZXJhdGlvbj1cIndyaXRlXCJcbiAgICAgIGNvbnRlbnQ9e2NvbnRlbnR9XG4gICAgICBmaXJzdExpbmU9e2ZpcnN0TGluZX1cbiAgICAgIHZlcmJvc2U9e3ZlcmJvc2V9XG4gICAgLz5cbiAgKVxuICByZXR1cm4gKFxuICAgIDxTdXNwZW5zZSBmYWxsYmFjaz17Y3JlYXRlRmFsbGJhY2t9PlxuICAgICAgPFdyaXRlUmVqZWN0aW9uQm9keVxuICAgICAgICBwcm9taXNlPXtkYXRhUHJvbWlzZX1cbiAgICAgICAgZmlsZVBhdGg9e2ZpbGVQYXRofVxuICAgICAgICBmaXJzdExpbmU9e2ZpcnN0TGluZX1cbiAgICAgICAgY3JlYXRlRmFsbGJhY2s9e2NyZWF0ZUZhbGxiYWNrfVxuICAgICAgICBzdHlsZT17c3R5bGV9XG4gICAgICAgIHZlcmJvc2U9e3ZlcmJvc2V9XG4gICAgICAvPlxuICAgIDwvU3VzcGVuc2U+XG4gIClcbn1cblxuZnVuY3Rpb24gV3JpdGVSZWplY3Rpb25Cb2R5KHtcbiAgcHJvbWlzZSxcbiAgZmlsZVBhdGgsXG4gIGZpcnN0TGluZSxcbiAgY3JlYXRlRmFsbGJhY2ssXG4gIHN0eWxlLFxuICB2ZXJib3NlLFxufToge1xuICBwcm9taXNlOiBQcm9taXNlPFJlamVjdGlvbkRpZmZEYXRhPlxuICBmaWxlUGF0aDogc3RyaW5nXG4gIGZpcnN0TGluZTogc3RyaW5nIHwgbnVsbFxuICBjcmVhdGVGYWxsYmFjazogUmVhY3QuUmVhY3ROb2RlXG4gIHN0eWxlPzogJ2NvbmRlbnNlZCdcbiAgdmVyYm9zZTogYm9vbGVhblxufSk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IGRhdGEgPSB1c2UocHJvbWlzZSlcbiAgaWYgKGRhdGEudHlwZSA9PT0gJ2NyZWF0ZScpIHJldHVybiBjcmVhdGVGYWxsYmFja1xuICBpZiAoZGF0YS50eXBlID09PSAnZXJyb3InKSB7XG4gICAgcmV0dXJuIChcbiAgICAgIDxNZXNzYWdlUmVzcG9uc2U+XG4gICAgICAgIDxUZXh0PihObyBjaGFuZ2VzKTwvVGV4dD5cbiAgICAgIDwvTWVzc2FnZVJlc3BvbnNlPlxuICAgIClcbiAgfVxuICByZXR1cm4gKFxuICAgIDxGaWxlRWRpdFRvb2xVc2VSZWplY3RlZE1lc3NhZ2VcbiAgICAgIGZpbGVfcGF0aD17ZmlsZVBhdGh9XG4gICAgICBvcGVyYXRpb249XCJ1cGRhdGVcIlxuICAgICAgcGF0Y2g9e2RhdGEucGF0Y2h9XG4gICAgICBmaXJzdExpbmU9e2ZpcnN0TGluZX1cbiAgICAgIGZpbGVDb250ZW50PXtkYXRhLm9sZENvbnRlbnR9XG4gICAgICBzdHlsZT17c3R5bGV9XG4gICAgICB2ZXJib3NlPXt2ZXJib3NlfVxuICAgIC8+XG4gIClcbn1cblxuYXN5bmMgZnVuY3Rpb24gbG9hZFJlamVjdGlvbkRpZmYoXG4gIGZpbGVQYXRoOiBzdHJpbmcsXG4gIGNvbnRlbnQ6IHN0cmluZyxcbik6IFByb21pc2U8UmVqZWN0aW9uRGlmZkRhdGE+IHtcbiAgdHJ5IHtcbiAgICBjb25zdCBmdWxsRmlsZVBhdGggPSBpc0Fic29sdXRlKGZpbGVQYXRoKVxuICAgICAgPyBmaWxlUGF0aFxuICAgICAgOiByZXNvbHZlKGdldEN3ZCgpLCBmaWxlUGF0aClcbiAgICBjb25zdCBoYW5kbGUgPSBhd2FpdCBvcGVuRm9yU2NhbihmdWxsRmlsZVBhdGgpXG4gICAgaWYgKGhhbmRsZSA9PT0gbnVsbCkgcmV0dXJuIHsgdHlwZTogJ2NyZWF0ZScgfVxuICAgIGxldCBvbGRDb250ZW50OiBzdHJpbmcgfCBudWxsXG4gICAgdHJ5IHtcbiAgICAgIG9sZENvbnRlbnQgPSBhd2FpdCByZWFkQ2FwcGVkKGhhbmRsZSlcbiAgICB9IGZpbmFsbHkge1xuICAgICAgYXdhaXQgaGFuZGxlLmNsb3NlKClcbiAgICB9XG4gICAgLy8gRmlsZSBleGNlZWRzIE1BWF9TQ0FOX0JZVEVTIOKAlCBmYWxsIGJhY2sgdG8gdGhlIGNyZWF0ZSB2aWV3IHJhdGhlciB0aGFuXG4gICAgLy8gT09NaW5nIG9uIGEgZGlmZiBvZiBhIG11bHRpLUdCIGZpbGUuXG4gICAgaWYgKG9sZENvbnRlbnQgPT09IG51bGwpIHJldHVybiB7IHR5cGU6ICdjcmVhdGUnIH1cbiAgICBjb25zdCBwYXRjaCA9IGdldFBhdGNoRm9yRGlzcGxheSh7XG4gICAgICBmaWxlUGF0aCxcbiAgICAgIGZpbGVDb250ZW50czogb2xkQ29udGVudCxcbiAgICAgIGVkaXRzOiBbXG4gICAgICAgIHsgb2xkX3N0cmluZzogb2xkQ29udGVudCwgbmV3X3N0cmluZzogY29udGVudCwgcmVwbGFjZV9hbGw6IGZhbHNlIH0sXG4gICAgICBdLFxuICAgIH0pXG4gICAgcmV0dXJuIHsgdHlwZTogJ3VwZGF0ZScsIHBhdGNoLCBvbGRDb250ZW50IH1cbiAgfSBjYXRjaCAoZSkge1xuICAgIC8vIFVzZXIgbWF5IGhhdmUgbWFudWFsbHkgYXBwbGllZCB0aGUgY2hhbmdlIHdoaWxlIHRoZSBkaWZmIHdhcyBzaG93bi5cbiAgICBsb2dFcnJvcihlIGFzIEVycm9yKVxuICAgIHJldHVybiB7IHR5cGU6ICdlcnJvcicgfVxuICB9XG59XG5cbmV4cG9ydCBmdW5jdGlvbiByZW5kZXJUb29sVXNlRXJyb3JNZXNzYWdlKFxuICByZXN1bHQ6IFRvb2xSZXN1bHRCbG9ja1BhcmFtWydjb250ZW50J10sXG4gIHsgdmVyYm9zZSB9OiB7IHZlcmJvc2U6IGJvb2xlYW4gfSxcbik6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGlmIChcbiAgICAhdmVyYm9zZSAmJlxuICAgIHR5cGVvZiByZXN1bHQgPT09ICdzdHJpbmcnICYmXG4gICAgZXh0cmFjdFRhZyhyZXN1bHQsICd0b29sX3VzZV9lcnJvcicpXG4gICkge1xuICAgIHJldHVybiAoXG4gICAgICA8TWVzc2FnZVJlc3BvbnNlPlxuICAgICAgICA8VGV4dCBjb2xvcj1cImVycm9yXCI+RXJyb3Igd3JpdGluZyBmaWxlPC9UZXh0PlxuICAgICAgPC9NZXNzYWdlUmVzcG9uc2U+XG4gICAgKVxuICB9XG4gIHJldHVybiA8RmFsbGJhY2tUb29sVXNlRXJyb3JNZXNzYWdlIHJlc3VsdD17cmVzdWx0fSB2ZXJib3NlPXt2ZXJib3NlfSAvPlxufVxuXG5leHBvcnQgZnVuY3Rpb24gcmVuZGVyVG9vbFJlc3VsdE1lc3NhZ2UoXG4gIHsgZmlsZVBhdGgsIGNvbnRlbnQsIHN0cnVjdHVyZWRQYXRjaCwgdHlwZSwgb3JpZ2luYWxGaWxlIH06IE91dHB1dCxcbiAgX3Byb2dyZXNzTWVzc2FnZXNGb3JNZXNzYWdlOiBQcm9ncmVzc01lc3NhZ2U8VG9vbFByb2dyZXNzRGF0YT5bXSxcbiAgeyBzdHlsZSwgdmVyYm9zZSB9OiB7IHN0eWxlPzogJ2NvbmRlbnNlZCc7IHZlcmJvc2U6IGJvb2xlYW4gfSxcbik6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIHN3aXRjaCAodHlwZSkge1xuICAgIGNhc2UgJ2NyZWF0ZSc6IHtcbiAgICAgIGNvbnN0IGlzUGxhbkZpbGUgPSBmaWxlUGF0aC5zdGFydHNXaXRoKGdldFBsYW5zRGlyZWN0b3J5KCkpXG5cbiAgICAgIC8vIFBsYW4gZmlsZXM6IGludmVydCBjb25kZW5zZWQgYmVoYXZpb3JcbiAgICAgIC8vIC0gUmVndWxhciBtb2RlOiBqdXN0IHNob3cgaGludCAodXNlciBjYW4gdHlwZSAvcGxhbiB0byBzZWUgZnVsbCBjb250ZW50KVxuICAgICAgLy8gLSBDb25kZW5zZWQgbW9kZSAoc3ViYWdlbnQgdmlldyk6IHNob3cgZnVsbCBjb250ZW50XG4gICAgICBpZiAoaXNQbGFuRmlsZSAmJiAhdmVyYm9zZSkge1xuICAgICAgICBpZiAoc3R5bGUgIT09ICdjb25kZW5zZWQnKSB7XG4gICAgICAgICAgcmV0dXJuIChcbiAgICAgICAgICAgIDxNZXNzYWdlUmVzcG9uc2U+XG4gICAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPi9wbGFuIHRvIHByZXZpZXc8L1RleHQ+XG4gICAgICAgICAgICA8L01lc3NhZ2VSZXNwb25zZT5cbiAgICAgICAgICApXG4gICAgICAgIH1cbiAgICAgIH0gZWxzZSBpZiAoc3R5bGUgPT09ICdjb25kZW5zZWQnICYmICF2ZXJib3NlKSB7XG4gICAgICAgIGNvbnN0IG51bUxpbmVzID0gY291bnRMaW5lcyhjb250ZW50KVxuICAgICAgICByZXR1cm4gKFxuICAgICAgICAgIDxUZXh0PlxuICAgICAgICAgICAgV3JvdGUgPFRleHQgYm9sZD57bnVtTGluZXN9PC9UZXh0PiBsaW5lcyB0b3snICd9XG4gICAgICAgICAgICA8VGV4dCBib2xkPntyZWxhdGl2ZShnZXRDd2QoKSwgZmlsZVBhdGgpfTwvVGV4dD5cbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgIClcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIChcbiAgICAgICAgPEZpbGVXcml0ZVRvb2xDcmVhdGVkTWVzc2FnZVxuICAgICAgICAgIGZpbGVQYXRoPXtmaWxlUGF0aH1cbiAgICAgICAgICBjb250ZW50PXtjb250ZW50fVxuICAgICAgICAgIHZlcmJvc2U9e3ZlcmJvc2V9XG4gICAgICAgIC8+XG4gICAgICApXG4gICAgfVxuICAgIGNhc2UgJ3VwZGF0ZSc6IHtcbiAgICAgIGNvbnN0IGlzUGxhbkZpbGUgPSBmaWxlUGF0aC5zdGFydHNXaXRoKGdldFBsYW5zRGlyZWN0b3J5KCkpXG4gICAgICByZXR1cm4gKFxuICAgICAgICA8RmlsZUVkaXRUb29sVXBkYXRlZE1lc3NhZ2VcbiAgICAgICAgICBmaWxlUGF0aD17ZmlsZVBhdGh9XG4gICAgICAgICAgc3RydWN0dXJlZFBhdGNoPXtzdHJ1Y3R1cmVkUGF0Y2h9XG4gICAgICAgICAgZmlyc3RMaW5lPXtjb250ZW50LnNwbGl0KCdcXG4nKVswXSA/PyBudWxsfVxuICAgICAgICAgIGZpbGVDb250ZW50PXtvcmlnaW5hbEZpbGUgPz8gdW5kZWZpbmVkfVxuICAgICAgICAgIHN0eWxlPXtzdHlsZX1cbiAgICAgICAgICB2ZXJib3NlPXt2ZXJib3NlfVxuICAgICAgICAgIHByZXZpZXdIaW50PXtpc1BsYW5GaWxlID8gJy9wbGFuIHRvIHByZXZpZXcnIDogdW5kZWZpbmVkfVxuICAgICAgICAvPlxuICAgICAgKVxuICAgIH1cbiAgfVxufVxuIl0sIm1hcHBpbmdzIjoiO0FBQUEsY0FBY0Esb0JBQW9CLFFBQVEsdUNBQXVDO0FBQ2pGLGNBQWNDLG1CQUFtQixRQUFRLE1BQU07QUFDL0MsU0FBU0MsVUFBVSxFQUFFQyxRQUFRLEVBQUVDLE9BQU8sUUFBUSxNQUFNO0FBQ3BELE9BQU8sS0FBS0MsS0FBSyxNQUFNLE9BQU87QUFDOUIsU0FBU0MsUUFBUSxFQUFFQyxHQUFHLEVBQUVDLFFBQVEsUUFBUSxPQUFPO0FBQy9DLFNBQVNDLGVBQWUsUUFBUSxtQ0FBbUM7QUFDbkUsU0FBU0MsVUFBVSxRQUFRLHVCQUF1QjtBQUNsRCxTQUFTQyxhQUFhLFFBQVEsbUNBQW1DO0FBQ2pFLFNBQVNDLDJCQUEyQixRQUFRLGlEQUFpRDtBQUM3RixTQUFTQywwQkFBMEIsUUFBUSxnREFBZ0Q7QUFDM0YsU0FBU0MsOEJBQThCLFFBQVEsb0RBQW9EO0FBQ25HLFNBQVNDLFlBQVksUUFBUSxrQ0FBa0M7QUFDL0QsU0FBU0MsZUFBZSxRQUFRLHFDQUFxQztBQUNyRSxTQUFTQyxlQUFlLFFBQVEsZ0NBQWdDO0FBQ2hFLFNBQVNDLEdBQUcsRUFBRUMsSUFBSSxRQUFRLGNBQWM7QUFDeEMsY0FBY0MsZ0JBQWdCLFFBQVEsZUFBZTtBQUNyRCxjQUFjQyxlQUFlLFFBQVEsd0JBQXdCO0FBQzdELFNBQVNDLE1BQU0sUUFBUSxvQkFBb0I7QUFDM0MsU0FBU0Msa0JBQWtCLFFBQVEscUJBQXFCO0FBQ3hELFNBQVNDLGNBQWMsUUFBUSxxQkFBcUI7QUFDcEQsU0FBU0MsUUFBUSxRQUFRLG9CQUFvQjtBQUM3QyxTQUFTQyxpQkFBaUIsUUFBUSxzQkFBc0I7QUFDeEQsU0FBU0MsV0FBVyxFQUFFQyxVQUFVLFFBQVEsZ0NBQWdDO0FBQ3hFLGNBQWNDLE1BQU0sUUFB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