source dump of claude code
at main 834 lines 148 kB view raw
1import { c as _c } from "react/compiler-runtime"; 2import { feature } from 'bun:bundle'; 3import chalk from 'chalk'; 4import type { UUID } from 'crypto'; 5import type { RefObject } from 'react'; 6import * as React from 'react'; 7import { useCallback, useEffect, useMemo, useRef, useState } from 'react'; 8import { every } from 'src/utils/set.js'; 9import { getIsRemoteMode } from '../bootstrap/state.js'; 10import type { Command } from '../commands.js'; 11import { BLACK_CIRCLE } from '../constants/figures.js'; 12import { useTerminalSize } from '../hooks/useTerminalSize.js'; 13import type { ScrollBoxHandle } from '../ink/components/ScrollBox.js'; 14import { useTerminalNotification } from '../ink/useTerminalNotification.js'; 15import { Box, Text } from '../ink.js'; 16import { useShortcutDisplay } from '../keybindings/useShortcutDisplay.js'; 17import type { Screen } from '../screens/REPL.js'; 18import type { Tools } from '../Tool.js'; 19import { findToolByName } from '../Tool.js'; 20import type { AgentDefinitionsResult } from '../tools/AgentTool/loadAgentsDir.js'; 21import type { Message as MessageType, NormalizedMessage, ProgressMessage as ProgressMessageType, RenderableMessage } from '../types/message.js'; 22import { type AdvisorBlock, isAdvisorBlock } from '../utils/advisor.js'; 23import { collapseBackgroundBashNotifications } from '../utils/collapseBackgroundBashNotifications.js'; 24import { collapseHookSummaries } from '../utils/collapseHookSummaries.js'; 25import { collapseReadSearchGroups } from '../utils/collapseReadSearch.js'; 26import { collapseTeammateShutdowns } from '../utils/collapseTeammateShutdowns.js'; 27import { getGlobalConfig } from '../utils/config.js'; 28import { isEnvTruthy } from '../utils/envUtils.js'; 29import { isFullscreenEnvEnabled } from '../utils/fullscreen.js'; 30import { applyGrouping } from '../utils/groupToolUses.js'; 31import { buildMessageLookups, createAssistantMessage, deriveUUID, getMessagesAfterCompactBoundary, getToolUseID, getToolUseIDs, hasUnresolvedHooksFromLookup, isNotEmptyMessage, normalizeMessages, reorderMessagesInUI, type StreamingThinking, type StreamingToolUse, shouldShowUserMessage } from '../utils/messages.js'; 32import { plural } from '../utils/stringUtils.js'; 33import { renderableSearchText } from '../utils/transcriptSearch.js'; 34import { Divider } from './design-system/Divider.js'; 35import type { UnseenDivider } from './FullscreenLayout.js'; 36import { LogoV2 } from './LogoV2/LogoV2.js'; 37import { StreamingMarkdown } from './Markdown.js'; 38import { hasContentAfterIndex, MessageRow } from './MessageRow.js'; 39import { InVirtualListContext, type MessageActionsNav, MessageActionsSelectedContext, type MessageActionsState } from './messageActions.js'; 40import { AssistantThinkingMessage } from './messages/AssistantThinkingMessage.js'; 41import { isNullRenderingAttachment } from './messages/nullRenderingAttachments.js'; 42import { OffscreenFreeze } from './OffscreenFreeze.js'; 43import type { ToolUseConfirm } from './permissions/PermissionRequest.js'; 44import { StatusNotices } from './StatusNotices.js'; 45import type { JumpHandle } from './VirtualMessageList.js'; 46 47// Memoed logo header: this box is the FIRST sibling before all MessageRows 48// in main-screen mode. If it becomes dirty on every Messages re-render, 49// renderChildren's seenDirtyChild cascade disables prevScreen (blit) for 50// ALL subsequent siblings — every MessageRow re-writes from scratch instead 51// of blitting. In long sessions (~2800 messages) this is 150K+ writes/frame 52// and pegs CPU at 100%. Memo on agentDefinitions so a new messages array 53// doesn't invalidate the logo subtree. LogoV2/StatusNotices internally 54// subscribe to useAppState/useSettings for their own updates. 55const LogoHeader = React.memo(function LogoHeader(t0) { 56 const $ = _c(3); 57 const { 58 agentDefinitions 59 } = t0; 60 let t1; 61 if ($[0] === Symbol.for("react.memo_cache_sentinel")) { 62 t1 = <LogoV2 />; 63 $[0] = t1; 64 } else { 65 t1 = $[0]; 66 } 67 let t2; 68 if ($[1] !== agentDefinitions) { 69 t2 = <OffscreenFreeze><Box flexDirection="column" gap={1}>{t1}<React.Suspense fallback={null}><StatusNotices agentDefinitions={agentDefinitions} /></React.Suspense></Box></OffscreenFreeze>; 70 $[1] = agentDefinitions; 71 $[2] = t2; 72 } else { 73 t2 = $[2]; 74 } 75 return t2; 76}); 77 78// Dead code elimination: conditional import for proactive mode 79/* eslint-disable @typescript-eslint/no-require-imports */ 80const proactiveModule = feature('PROACTIVE') || feature('KAIROS') ? require('../proactive/index.js') : null; 81const BRIEF_TOOL_NAME: string | null = feature('KAIROS') || feature('KAIROS_BRIEF') ? (require('../tools/BriefTool/prompt.js') as typeof import('../tools/BriefTool/prompt.js')).BRIEF_TOOL_NAME : null; 82const SEND_USER_FILE_TOOL_NAME: string | null = feature('KAIROS') ? (require('../tools/SendUserFileTool/prompt.js') as typeof import('../tools/SendUserFileTool/prompt.js')).SEND_USER_FILE_TOOL_NAME : null; 83 84/* eslint-enable @typescript-eslint/no-require-imports */ 85import { VirtualMessageList } from './VirtualMessageList.js'; 86 87/** 88 * In brief-only mode, filter messages to show ONLY Brief tool_use blocks, 89 * their tool_results, and real user input. All assistant text is dropped — 90 * if the model forgets to call Brief, the user sees nothing for that turn. 91 * That's on the model to get right; the filter does not second-guess it. 92 */ 93export function filterForBriefTool<T extends { 94 type: string; 95 subtype?: string; 96 isMeta?: boolean; 97 isApiErrorMessage?: boolean; 98 message?: { 99 content: Array<{ 100 type: string; 101 name?: string; 102 tool_use_id?: string; 103 }>; 104 }; 105 attachment?: { 106 type: string; 107 isMeta?: boolean; 108 origin?: unknown; 109 commandMode?: string; 110 }; 111}>(messages: T[], briefToolNames: string[]): T[] { 112 const nameSet = new Set(briefToolNames); 113 // tool_use always precedes its tool_result in the array, so we can collect 114 // IDs and match against them in a single pass. 115 const briefToolUseIDs = new Set<string>(); 116 return messages.filter(msg => { 117 // System messages (attach confirmation, remote errors, compact boundaries) 118 // must stay visible — dropping them leaves the viewer with no feedback. 119 // Exception: api_metrics is per-turn debug noise (TTFT, config writes, 120 // hook timing) that defeats the point of brief mode. Still visible in 121 // transcript mode (ctrl+o) which bypasses this filter. 122 if (msg.type === 'system') return msg.subtype !== 'api_metrics'; 123 const block = msg.message?.content[0]; 124 if (msg.type === 'assistant') { 125 // API error messages (auth failures, rate limits, etc.) must stay visible 126 if (msg.isApiErrorMessage) return true; 127 // Keep Brief tool_use blocks (renders with standard tool call chrome, 128 // and must be in the list so buildMessageLookups can resolve tool results) 129 if (block?.type === 'tool_use' && block.name && nameSet.has(block.name)) { 130 if ('id' in block) { 131 briefToolUseIDs.add((block as { 132 id: string; 133 }).id); 134 } 135 return true; 136 } 137 return false; 138 } 139 if (msg.type === 'user') { 140 if (block?.type === 'tool_result') { 141 return block.tool_use_id !== undefined && briefToolUseIDs.has(block.tool_use_id); 142 } 143 // Real user input only — drop meta/tick messages. 144 return !msg.isMeta; 145 } 146 if (msg.type === 'attachment') { 147 // Human input drained mid-turn arrives as a queued_command attachment 148 // (query.ts mid-chain drain → getQueuedCommandAttachments). Keep it — 149 // it's what the user typed. commandMode === 'prompt' positively 150 // identifies human-typed input; task-notification callers set 151 // mode: 'task-notification' but not origin/isMeta, so the positive 152 // commandMode check is required to exclude them. 153 const att = msg.attachment; 154 return att?.type === 'queued_command' && att.commandMode === 'prompt' && !att.isMeta && att.origin === undefined; 155 } 156 return false; 157 }); 158} 159 160/** 161 * Full-transcript companion to filterForBriefTool. When the Brief tool is 162 * in use, the model's text output is redundant with the SendUserMessage 163 * content it wrote right after — drop the text so only the SendUserMessage 164 * block shows. Tool calls and their results stay visible. 165 * 166 * Per-turn: only drops text in turns that actually called Brief. If the 167 * model forgets, text still shows — otherwise the user would see nothing. 168 */ 169export function dropTextInBriefTurns<T extends { 170 type: string; 171 isMeta?: boolean; 172 message?: { 173 content: Array<{ 174 type: string; 175 name?: string; 176 }>; 177 }; 178}>(messages: T[], briefToolNames: string[]): T[] { 179 const nameSet = new Set(briefToolNames); 180 // First pass: find which turns (bounded by non-meta user messages) contain 181 // a Brief tool_use. Tag each assistant text block with its turn index. 182 const turnsWithBrief = new Set<number>(); 183 const textIndexToTurn: number[] = []; 184 let turn = 0; 185 for (let i = 0; i < messages.length; i++) { 186 const msg = messages[i]!; 187 const block = msg.message?.content[0]; 188 if (msg.type === 'user' && block?.type !== 'tool_result' && !msg.isMeta) { 189 turn++; 190 continue; 191 } 192 if (msg.type === 'assistant') { 193 if (block?.type === 'text') { 194 textIndexToTurn[i] = turn; 195 } else if (block?.type === 'tool_use' && block.name && nameSet.has(block.name)) { 196 turnsWithBrief.add(turn); 197 } 198 } 199 } 200 if (turnsWithBrief.size === 0) return messages; 201 // Second pass: drop text blocks whose turn called Brief. 202 return messages.filter((_, i) => { 203 const t = textIndexToTurn[i]; 204 return t === undefined || !turnsWithBrief.has(t); 205 }); 206} 207type Props = { 208 messages: MessageType[]; 209 tools: Tools; 210 commands: Command[]; 211 verbose: boolean; 212 toolJSX: { 213 jsx: React.ReactNode | null; 214 shouldHidePromptInput: boolean; 215 shouldContinueAnimation?: true; 216 } | null; 217 toolUseConfirmQueue: ToolUseConfirm[]; 218 inProgressToolUseIDs: Set<string>; 219 isMessageSelectorVisible: boolean; 220 conversationId: string; 221 screen: Screen; 222 streamingToolUses: StreamingToolUse[]; 223 showAllInTranscript?: boolean; 224 agentDefinitions?: AgentDefinitionsResult; 225 onOpenRateLimitOptions?: () => void; 226 /** Hide the logo/header - used for subagent zoom view */ 227 hideLogo?: boolean; 228 isLoading: boolean; 229 /** In transcript mode, hide all thinking blocks except the last one */ 230 hidePastThinking?: boolean; 231 /** Streaming thinking content (live updates, not frozen) */ 232 streamingThinking?: StreamingThinking | null; 233 /** Streaming text preview (rendered as last item so transition to final message is positionally seamless) */ 234 streamingText?: string | null; 235 /** When true, only show Brief tool output (hide everything else) */ 236 isBriefOnly?: boolean; 237 /** Fullscreen-mode "─── N new ───" divider. Renders before the first 238 * renderableMessage derived from firstUnseenUuid (matched by the 24-char 239 * prefix that deriveUUID preserves). */ 240 unseenDivider?: UnseenDivider; 241 /** Fullscreen-mode ScrollBox handle. Enables React-level virtualization when present. */ 242 scrollRef?: RefObject<ScrollBoxHandle | null>; 243 /** Fullscreen-mode: enable sticky-prompt tracking (writes via ScrollChromeContext). */ 244 trackStickyPrompt?: boolean; 245 /** Transcript search: jump-to-index + setSearchQuery/nextMatch/prevMatch. */ 246 jumpRef?: RefObject<JumpHandle | null>; 247 /** Transcript search: fires when match count/position changes. */ 248 onSearchMatchesChange?: (count: number, current: number) => void; 249 /** Paint an existing DOM subtree to fresh Screen, scan. Element comes 250 * from the main tree (all real providers). Message-relative positions. */ 251 scanElement?: (el: import('../ink/dom.js').DOMElement) => import('../ink/render-to-screen.js').MatchPosition[]; 252 /** Position-based CURRENT highlight. positions stable (msg-relative), 253 * rowOffset tracks scroll. null clears. */ 254 setPositions?: (state: { 255 positions: import('../ink/render-to-screen.js').MatchPosition[]; 256 rowOffset: number; 257 currentIdx: number; 258 } | null) => void; 259 /** Bypass MAX_MESSAGES_WITHOUT_VIRTUALIZATION. For one-shot headless renders 260 * (e.g. /export via renderToString) where the memory concern doesn't apply 261 * and the "already in scrollback" justification doesn't hold. */ 262 disableRenderCap?: boolean; 263 /** In-transcript cursor; expanded overrides verbose for selected message. */ 264 cursor?: MessageActionsState | null; 265 setCursor?: (cursor: MessageActionsState | null) => void; 266 /** Passed through to VirtualMessageList (heightCache owns visibility). */ 267 cursorNavRef?: React.Ref<MessageActionsNav>; 268 /** Render only collapsed.slice(start, end). For chunked headless export 269 * (streamRenderedMessages in exportRenderer.tsx): prep runs on the FULL 270 * messages array so grouping/lookups are correct, but only this slice 271 * chunk instead of the full session. The logo renders only for chunk 0 272 * (start === 0); later chunks are mid-stream continuations. 273 * Measured Mar 2026: 538-msg session, 20 slices → −55% plateau RSS. */ 274 renderRange?: readonly [start: number, end: number]; 275}; 276const MAX_MESSAGES_TO_SHOW_IN_TRANSCRIPT_MODE = 30; 277 278// Safety cap for the non-virtualized render path (fullscreen off or 279// explicitly disabled). Ink mounts a full fiber tree per message (~250 KB 280// RSS each); yoga layout height grows unbounded; the screen buffer is sized 281// to fit every line. At ~2000 messages this is ~3000-line screens, ~500 MB 282// of fibers, and per-frame write costs that push the process into a GC 283// death spiral (observed: 59 GB RSS, 14k mmap/munmap/sec). Content dropped 284// from this slice has already been printed to terminal scrollback — users 285// can still scroll up natively. VirtualMessageList (the default ant path) 286// bypasses this cap entirely. Headless one-shot renders (e.g. /export) 287// pass disableRenderCap to opt out — they have no scrollback and the 288// memory concern doesn't apply to renderToString. 289// 290// The slice boundary is tracked as a UUID anchor, not a count-derived 291// index. Count-based slicing (slice(-200)) drops one message from the 292// front on every append, shifting scrollback content and forcing a full 293// terminal reset per turn (CC-941). Quantizing to 50-message steps 294// (CC-1154) helped but still shifted on compaction and collapse regrouping 295// since those change collapsed.length without adding messages. The UUID 296// anchor only advances when rendered count genuinely exceeds CAP+STEP — 297// immune to length churn from grouping/compaction (CC-1174). 298// 299// The anchor stores BOTH uuid and index. Some uuids are unstable between 300// renders: collapseHookSummaries derives the merged uuid from the first 301// summary in a group, but reorderMessagesInUI reshuffles hook adjacency 302// as tool results stream in, changing which summary is first. When the 303// uuid vanishes, falling back to the stored index (clamped) keeps the 304// slice roughly where it was instead of resetting to 0 — which would 305// jump from ~200 rendered messages to the full history, orphaning 306// in-progress badge snapshots in scrollback. 307const MAX_MESSAGES_WITHOUT_VIRTUALIZATION = 200; 308const MESSAGE_CAP_STEP = 50; 309export type SliceAnchor = { 310 uuid: string; 311 idx: number; 312} | null; 313 314/** Exported for testing. Mutates anchorRef when the window needs to advance. */ 315export function computeSliceStart(collapsed: ReadonlyArray<{ 316 uuid: string; 317}>, anchorRef: { 318 current: SliceAnchor; 319}, cap = MAX_MESSAGES_WITHOUT_VIRTUALIZATION, step = MESSAGE_CAP_STEP): number { 320 const anchor = anchorRef.current; 321 const anchorIdx = anchor ? collapsed.findIndex(m => m.uuid === anchor.uuid) : -1; 322 // Anchor found → use it. Anchor lost → fall back to stored index 323 // (clamped) so collapse-regrouping uuid churn doesn't reset to 0. 324 let start = anchorIdx >= 0 ? anchorIdx : anchor ? Math.min(anchor.idx, Math.max(0, collapsed.length - cap)) : 0; 325 if (collapsed.length - start > cap + step) { 326 start = collapsed.length - cap; 327 } 328 // Refresh anchor from whatever lives at the current start — heals a 329 // stale uuid after fallback and captures a new one after advancement. 330 const msgAtStart = collapsed[start]; 331 if (msgAtStart && (anchor?.uuid !== msgAtStart.uuid || anchor.idx !== start)) { 332 anchorRef.current = { 333 uuid: msgAtStart.uuid, 334 idx: start 335 }; 336 } else if (!msgAtStart && anchor) { 337 anchorRef.current = null; 338 } 339 return start; 340} 341const MessagesImpl = ({ 342 messages, 343 tools, 344 commands, 345 verbose, 346 toolJSX, 347 toolUseConfirmQueue, 348 inProgressToolUseIDs, 349 isMessageSelectorVisible, 350 conversationId, 351 screen, 352 streamingToolUses, 353 showAllInTranscript = false, 354 agentDefinitions, 355 onOpenRateLimitOptions, 356 hideLogo = false, 357 isLoading, 358 hidePastThinking = false, 359 streamingThinking, 360 streamingText, 361 isBriefOnly = false, 362 unseenDivider, 363 scrollRef, 364 trackStickyPrompt, 365 jumpRef, 366 onSearchMatchesChange, 367 scanElement, 368 setPositions, 369 disableRenderCap = false, 370 cursor = null, 371 setCursor, 372 cursorNavRef, 373 renderRange 374}: Props): React.ReactNode => { 375 const { 376 columns 377 } = useTerminalSize(); 378 const toggleShowAllShortcut = useShortcutDisplay('transcript:toggleShowAll', 'Transcript', 'Ctrl+E'); 379 const normalizedMessages = useMemo(() => normalizeMessages(messages).filter(isNotEmptyMessage), [messages]); 380 381 // Check if streaming thinking should be visible (streaming or within 30s timeout) 382 const isStreamingThinkingVisible = useMemo(() => { 383 if (!streamingThinking) return false; 384 if (streamingThinking.isStreaming) return true; 385 if (streamingThinking.streamingEndedAt) { 386 return Date.now() - streamingThinking.streamingEndedAt < 30000; 387 } 388 return false; 389 }, [streamingThinking]); 390 391 // Find the last thinking block (message UUID + content index) for hiding past thinking in transcript mode 392 // When streaming thinking is visible, use a special ID that won't match any completed thinking block 393 // With adaptive thinking, only consider thinking blocks from the current turn and stop searching once we 394 // hit the last user message. 395 const lastThinkingBlockId = useMemo(() => { 396 if (!hidePastThinking) return null; 397 // If streaming thinking is visible, hide all completed thinking blocks by using a non-matching ID 398 if (isStreamingThinkingVisible) return 'streaming'; 399 // Iterate backwards to find the last message with a thinking block 400 for (let i = normalizedMessages.length - 1; i >= 0; i--) { 401 const msg = normalizedMessages[i]; 402 if (msg?.type === 'assistant') { 403 const content = msg.message.content; 404 // Find the last thinking block in this message 405 for (let j = content.length - 1; j >= 0; j--) { 406 if (content[j]?.type === 'thinking') { 407 return `${msg.uuid}:${j}`; 408 } 409 } 410 } else if (msg?.type === 'user') { 411 const hasToolResult = msg.message.content.some(block => block.type === 'tool_result'); 412 if (!hasToolResult) { 413 // Reached a previous user turn so don't show stale thinking from before 414 return 'no-thinking'; 415 } 416 } 417 } 418 return null; 419 }, [normalizedMessages, hidePastThinking, isStreamingThinkingVisible]); 420 421 // Find the latest user bash output message (from ! commands) 422 // This allows us to show full output for the most recent bash command 423 const latestBashOutputUUID = useMemo(() => { 424 // Iterate backwards to find the last user message with bash output 425 for (let i_0 = normalizedMessages.length - 1; i_0 >= 0; i_0--) { 426 const msg_0 = normalizedMessages[i_0]; 427 if (msg_0?.type === 'user') { 428 const content_0 = msg_0.message.content; 429 // Check if any text content is bash output 430 for (const block_0 of content_0) { 431 if (block_0.type === 'text') { 432 const text = block_0.text; 433 if (text.startsWith('<bash-stdout') || text.startsWith('<bash-stderr')) { 434 return msg_0.uuid; 435 } 436 } 437 } 438 } 439 } 440 return null; 441 }, [normalizedMessages]); 442 443 // streamingToolUses updates on every input_json_delta while normalizedMessages 444 // stays stable — precompute the Set so the filter is O(k) not O(n×k) per chunk. 445 const normalizedToolUseIDs = useMemo(() => getToolUseIDs(normalizedMessages), [normalizedMessages]); 446 const streamingToolUsesWithoutInProgress = useMemo(() => streamingToolUses.filter(stu => !inProgressToolUseIDs.has(stu.contentBlock.id) && !normalizedToolUseIDs.has(stu.contentBlock.id)), [streamingToolUses, inProgressToolUseIDs, normalizedToolUseIDs]); 447 const syntheticStreamingToolUseMessages = useMemo(() => streamingToolUsesWithoutInProgress.flatMap(streamingToolUse => { 448 const msg_1 = createAssistantMessage({ 449 content: [streamingToolUse.contentBlock] 450 }); 451 // Override randomUUID with deterministic value derived from content 452 // block ID to prevent React key changes on every memo recomputation. 453 // Same class of bug fixed in normalizeMessages (commit 383326e613): 454 // fresh randomUUID → unstable React keys → component remounts → 455 // Ink rendering corruption (overlapping text from stale DOM nodes). 456 msg_1.uuid = deriveUUID(streamingToolUse.contentBlock.id as UUID, 0); 457 return normalizeMessages([msg_1]); 458 }), [streamingToolUsesWithoutInProgress]); 459 const isTranscriptMode = screen === 'transcript'; 460 // Hoisted to mount-time — this component re-renders on every scroll. 461 const disableVirtualScroll = useMemo(() => isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_VIRTUAL_SCROLL), []); 462 // Virtual scroll replaces the transcript cap: everything is scrollable and 463 // memory is bounded by the mounted-item count, not the total. scrollRef is 464 // only passed when isFullscreenEnvEnabled() is true (REPL.tsx gates it), 465 // so scrollRef's presence is the signal. 466 const virtualScrollRuntimeGate = scrollRef != null && !disableVirtualScroll; 467 const shouldTruncate = isTranscriptMode && !showAllInTranscript && !virtualScrollRuntimeGate; 468 469 // Anchor for the first rendered message in the non-virtualized cap slice. 470 // Monotonic advance only — mutation during render is idempotent (safe 471 // under StrictMode double-render). See MAX_MESSAGES_WITHOUT_VIRTUALIZATION 472 // comment above for why this replaced count-based slicing. 473 const sliceAnchorRef = useRef<SliceAnchor>(null); 474 475 // Expensive message transforms — filter, reorder, group, collapse, lookups. 476 // All O(n) over 27k messages. Split from the renderRange slice so scrolling 477 // (which only changes renderRange) doesn't re-run these. Previously this 478 // useMemo included renderRange → every scroll rebuilt 6 Maps over 27k 479 // messages + 4 filter/map passes = ~50ms alloc per scroll → GC pressure → 480 // 100-173ms stop-the-world pauses on the 1GB heap. 481 const { 482 collapsed: collapsed_0, 483 lookups: lookups_0, 484 hasTruncatedMessages: hasTruncatedMessages_0, 485 hiddenMessageCount: hiddenMessageCount_0 486 } = useMemo(() => { 487 // In fullscreen mode the alt buffer has no native scrollback, so the 488 // compact-boundary filter just hides history the ScrollBox could 489 // otherwise scroll to. Main-screen mode keeps the filter — pre-compact 490 // rows live above the viewport in native scrollback there, and 491 // re-rendering them triggers full resets. 492 // includeSnipped: UI rendering keeps snipped messages for scrollback 493 // (this PR's core goal — full history in UI, filter only for the model). 494 // Also avoids a UUID mismatch: normalizeMessages derives new UUIDs, so 495 // projectSnippedView's check against original removedUuids would fail. 496 const compactAwareMessages = verbose || isFullscreenEnvEnabled() ? normalizedMessages : getMessagesAfterCompactBoundary(normalizedMessages, { 497 includeSnipped: true 498 }); 499 const messagesToShowNotTruncated = reorderMessagesInUI(compactAwareMessages.filter((msg_2): msg_2 is Exclude<NormalizedMessage, ProgressMessageType> => msg_2.type !== 'progress') 500 // CC-724: drop attachment messages that AttachmentMessage renders as 501 // null (hook_success, hook_additional_context, hook_cancelled, etc.) 502 // BEFORE counting/slicing so they don't inflate the "N messages" 503 // count in ctrl-o or consume slots in the 200-message render cap. 504 .filter(msg_3 => !isNullRenderingAttachment(msg_3)).filter(_ => shouldShowUserMessage(_, isTranscriptMode)), syntheticStreamingToolUseMessages); 505 // Three-tier filtering. Transcript mode (ctrl+o screen) is truly unfiltered. 506 // Brief-only: SendUserMessage + user input only. Default: drop redundant 507 // assistant text in turns where SendUserMessage was called (the model's 508 // text is working-notes that duplicate the SendUserMessage content). 509 const briefToolNames = [BRIEF_TOOL_NAME, SEND_USER_FILE_TOOL_NAME].filter((n): n is string => n !== null); 510 // dropTextInBriefTurns should only trigger on SendUserMessage turns — 511 // SendUserFile delivers a file without replacement text, so dropping 512 // assistant text for file-only turns would leave the user with no context. 513 const dropTextToolNames = [BRIEF_TOOL_NAME].filter((n_0): n_0 is string => n_0 !== null); 514 const briefFiltered = briefToolNames.length > 0 && !isTranscriptMode ? isBriefOnly ? filterForBriefTool(messagesToShowNotTruncated, briefToolNames) : dropTextToolNames.length > 0 ? dropTextInBriefTurns(messagesToShowNotTruncated, dropTextToolNames) : messagesToShowNotTruncated : messagesToShowNotTruncated; 515 const messagesToShow = shouldTruncate ? briefFiltered.slice(-MAX_MESSAGES_TO_SHOW_IN_TRANSCRIPT_MODE) : briefFiltered; 516 const hasTruncatedMessages = shouldTruncate && briefFiltered.length > MAX_MESSAGES_TO_SHOW_IN_TRANSCRIPT_MODE; 517 const { 518 messages: groupedMessages 519 } = applyGrouping(messagesToShow, tools, verbose); 520 const collapsed = collapseBackgroundBashNotifications(collapseHookSummaries(collapseTeammateShutdowns(collapseReadSearchGroups(groupedMessages, tools))), verbose); 521 const lookups = buildMessageLookups(normalizedMessages, messagesToShow); 522 const hiddenMessageCount = messagesToShowNotTruncated.length - MAX_MESSAGES_TO_SHOW_IN_TRANSCRIPT_MODE; 523 return { 524 collapsed, 525 lookups, 526 hasTruncatedMessages, 527 hiddenMessageCount 528 }; 529 }, [verbose, normalizedMessages, isTranscriptMode, syntheticStreamingToolUseMessages, shouldTruncate, tools, isBriefOnly]); 530 531 // Cheap slice — only runs when scroll range or slice config changes. 532 const renderableMessages = useMemo(() => { 533 // Safety cap for the non-virtualized render path. Applied here (not at 534 // the JSX site) so renderMessageRow's index-based lookups and 535 // dividerBeforeIndex compute on the same array. VirtualMessageList 536 // never sees this slice — virtualScrollRuntimeGate is constant for the 537 // component's lifetime (scrollRef is either always passed or never). 538 // renderRange is first: the chunked export path slices the 539 // post-grouping array so each chunk gets correct tool-call grouping. 540 const capApplies = !virtualScrollRuntimeGate && !disableRenderCap; 541 const sliceStart = capApplies ? computeSliceStart(collapsed_0, sliceAnchorRef) : 0; 542 return renderRange ? collapsed_0.slice(renderRange[0], renderRange[1]) : sliceStart > 0 ? collapsed_0.slice(sliceStart) : collapsed_0; 543 }, [collapsed_0, renderRange, virtualScrollRuntimeGate, disableRenderCap]); 544 const streamingToolUseIDs = useMemo(() => new Set(streamingToolUses.map(__0 => __0.contentBlock.id)), [streamingToolUses]); 545 546 // Divider insertion point: first renderableMessage whose uuid shares the 547 // 24-char prefix with firstUnseenUuid (deriveUUID keeps the first 24 548 // chars of the source message uuid, so this matches any block from it). 549 const dividerBeforeIndex = useMemo(() => { 550 if (!unseenDivider) return -1; 551 const prefix = unseenDivider.firstUnseenUuid.slice(0, 24); 552 return renderableMessages.findIndex(m => m.uuid.slice(0, 24) === prefix); 553 }, [unseenDivider, renderableMessages]); 554 const selectedIdx = useMemo(() => { 555 if (!cursor) return -1; 556 return renderableMessages.findIndex(m_0 => m_0.uuid === cursor.uuid); 557 }, [cursor, renderableMessages]); 558 559 // Fullscreen: click a message to toggle verbose rendering for it. Keyed by 560 // tool_use_id where available so a tool_use and its tool_result (separate 561 // rows) expand together; falls back to uuid for groups/thinking. Stale keys 562 // are harmless — they never match anything in renderableMessages. 563 const [expandedKeys, setExpandedKeys] = useState<ReadonlySet<string>>(() => new Set()); 564 const onItemClick = useCallback((msg_4: RenderableMessage) => { 565 const k = expandKey(msg_4); 566 setExpandedKeys(prev => { 567 const next = new Set(prev); 568 if (next.has(k)) next.delete(k);else next.add(k); 569 return next; 570 }); 571 }, []); 572 const isItemExpanded = useCallback((msg_5: RenderableMessage) => expandedKeys.size > 0 && expandedKeys.has(expandKey(msg_5)), [expandedKeys]); 573 // Only hover/click messages where the verbose toggle reveals more: 574 // collapsed read/search groups, or tool results that self-report truncation 575 // via isResultTruncated. Callback must be stable across message updates: if 576 // its identity (or return value) flips during streaming, onMouseEnter 577 // attaches after the mouse is already inside → hover never fires. tools is 578 // session-stable; lookups is read via ref so the callback doesn't churn on 579 // every new message. 580 const lookupsRef = useRef(lookups_0); 581 lookupsRef.current = lookups_0; 582 const isItemClickable = useCallback((msg_6: RenderableMessage): boolean => { 583 if (msg_6.type === 'collapsed_read_search') return true; 584 if (msg_6.type === 'assistant') { 585 const b = msg_6.message.content[0] as unknown as AdvisorBlock | undefined; 586 return b != null && isAdvisorBlock(b) && b.type === 'advisor_tool_result' && b.content.type === 'advisor_result'; 587 } 588 if (msg_6.type !== 'user') return false; 589 const b_0 = msg_6.message.content[0]; 590 if (b_0?.type !== 'tool_result' || b_0.is_error || !msg_6.toolUseResult) return false; 591 const name = lookupsRef.current.toolUseByToolUseID.get(b_0.tool_use_id)?.name; 592 const tool = name ? findToolByName(tools, name) : undefined; 593 return tool?.isResultTruncated?.(msg_6.toolUseResult as never) ?? false; 594 }, [tools]); 595 const canAnimate = (!toolJSX || !!toolJSX.shouldContinueAnimation) && !toolUseConfirmQueue.length && !isMessageSelectorVisible; 596 const hasToolsInProgress = inProgressToolUseIDs.size > 0; 597 598 // Report progress to terminal (for terminals that support OSC 9;4) 599 const { 600 progress 601 } = useTerminalNotification(); 602 const prevProgressState = useRef<string | null>(null); 603 const progressEnabled = getGlobalConfig().terminalProgressBarEnabled && !getIsRemoteMode() && !(proactiveModule?.isProactiveActive() ?? false); 604 useEffect(() => { 605 const state = progressEnabled ? hasToolsInProgress ? 'indeterminate' : 'completed' : null; 606 if (prevProgressState.current === state) return; 607 prevProgressState.current = state; 608 progress(state); 609 }, [progress, progressEnabled, hasToolsInProgress]); 610 useEffect(() => { 611 return () => progress(null); 612 }, [progress]); 613 const messageKey = useCallback((msg_7: RenderableMessage) => `${msg_7.uuid}-${conversationId}`, [conversationId]); 614 const renderMessageRow = (msg_8: RenderableMessage, index: number) => { 615 const prevType = index > 0 ? renderableMessages[index - 1]?.type : undefined; 616 const isUserContinuation = msg_8.type === 'user' && prevType === 'user'; 617 // hasContentAfter is only consumed for collapsed_read_search groups; 618 // skip the scan for everything else. streamingText is rendered as a 619 // sibling after this map, so it's never in renderableMessages — OR it 620 // in explicitly so the group flips to past tense as soon as text starts 621 // streaming instead of waiting for the block to finalize. 622 const hasContentAfter = msg_8.type === 'collapsed_read_search' && (!!streamingText || hasContentAfterIndex(renderableMessages, index, tools, streamingToolUseIDs)); 623 const k_0 = messageKey(msg_8); 624 const row = <MessageRow key={k_0} message={msg_8} isUserContinuation={isUserContinuation} hasContentAfter={hasContentAfter} tools={tools} commands={commands} verbose={verbose || isItemExpanded(msg_8) || cursor?.expanded === true && index === selectedIdx} inProgressToolUseIDs={inProgressToolUseIDs} streamingToolUseIDs={streamingToolUseIDs} screen={screen} canAnimate={canAnimate} onOpenRateLimitOptions={onOpenRateLimitOptions} lastThinkingBlockId={lastThinkingBlockId} latestBashOutputUUID={latestBashOutputUUID} columns={columns} isLoading={isLoading} lookups={lookups_0} />; 625 626 // Per-row Provider — only 2 rows re-render on selection change. 627 // Wrapped BEFORE divider branch so both return paths get it. 628 const wrapped = <MessageActionsSelectedContext.Provider key={k_0} value={index === selectedIdx}> 629 {row} 630 </MessageActionsSelectedContext.Provider>; 631 if (unseenDivider && index === dividerBeforeIndex) { 632 return [<Box key="unseen-divider" marginTop={1}> 633 <Divider title={`${unseenDivider.count} new ${plural(unseenDivider.count, 'message')}`} width={columns} color="inactive" /> 634 </Box>, wrapped]; 635 } 636 return wrapped; 637 }; 638 639 // Search indexing: for tool_result messages, look up the Tool and use 640 // its extractSearchText — tool-owned, precise, matches what 641 // renderToolResultMessage shows. Falls back to renderableSearchText 642 // (duck-types toolUseResult) for tools that haven't implemented it, 643 // and for all non-tool-result message types. The drift-catcher test 644 // (toolSearchText.test.tsx) renders + compares to keep these in sync. 645 // 646 // A second-React-root reconcile approach was tried and ruled out 647 // (measured 3.1ms/msg, growing — flushSyncWork processes all roots; 648 // component hooks mutate shared state → main root accumulates updates). 649 const searchTextCache = useRef(new WeakMap<RenderableMessage, string>()); 650 const extractSearchText = useCallback((msg_9: RenderableMessage): string => { 651 const cached = searchTextCache.current.get(msg_9); 652 if (cached !== undefined) return cached; 653 let text_0 = renderableSearchText(msg_9); 654 // If this is a tool_result message and the tool implements 655 // extractSearchText, prefer that — it's precise (tool-owned) 656 // vs renderableSearchText's field-name heuristic. 657 if (msg_9.type === 'user' && msg_9.toolUseResult && Array.isArray(msg_9.message.content)) { 658 const tr = msg_9.message.content.find(b_1 => b_1.type === 'tool_result'); 659 if (tr && 'tool_use_id' in tr) { 660 const tu = lookups_0.toolUseByToolUseID.get(tr.tool_use_id); 661 const tool_0 = tu && findToolByName(tools, tu.name); 662 const extracted = tool_0?.extractSearchText?.(msg_9.toolUseResult as never); 663 // undefined = tool didn't implement → keep heuristic. Empty 664 // string = tool says "nothing to index" → respect that. 665 if (extracted !== undefined) text_0 = extracted; 666 } 667 } 668 // Cache LOWERED: setSearchQuery's hot loop indexOfs per keystroke. 669 // Lowering here (once, at warm) vs there (every keystroke) trades 670 // ~same steady-state memory for zero per-keystroke alloc. Cache 671 // GC's with messages on transcript exit. Tool methods return raw; 672 // renderableSearchText already lowercases (redundant but cheap). 673 const lowered = text_0.toLowerCase(); 674 searchTextCache.current.set(msg_9, lowered); 675 return lowered; 676 }, [tools, lookups_0]); 677 return <> 678 {/* Logo */} 679 {!hideLogo && !(renderRange && renderRange[0] > 0) && <LogoHeader agentDefinitions={agentDefinitions} />} 680 681 {/* Truncation indicator */} 682 {hasTruncatedMessages_0 && <Divider title={`${toggleShowAllShortcut} to show ${chalk.bold(hiddenMessageCount_0)} previous messages`} width={columns} />} 683 684 {/* Show all indicator */} 685 {isTranscriptMode && showAllInTranscript && hiddenMessageCount_0 > 0 && 686 // disableRenderCap (e.g. [ dump-to-scrollback) means we're uncapped 687 // as a one-shot escape hatch, not a toggle — ctrl+e is dead and 688 // nothing is actually "hidden" to restore. 689 !disableRenderCap && <Divider title={`${toggleShowAllShortcut} to hide ${chalk.bold(hiddenMessageCount_0)} previous messages`} width={columns} />} 690 691 {/* Messages - rendered as memoized MessageRow components. 692 flatMap inserts the unseen-divider as a separate keyed sibling so 693 (a) non-fullscreen renders pay no per-message Fragment wrap, and 694 (b) divider toggle in fullscreen preserves all MessageRows by key. 695 Pre-compute derived values instead of passing renderableMessages to 696 each row - React Compiler pins props in the fiber's memoCache, so 697 passing the array would accumulate every historical version 698 (~1-2MB over a 7-turn session). */} 699 {virtualScrollRuntimeGate ? <InVirtualListContext.Provider value={true}> 700 <VirtualMessageList messages={renderableMessages} scrollRef={scrollRef} columns={columns} itemKey={messageKey} renderItem={renderMessageRow} onItemClick={onItemClick} isItemClickable={isItemClickable} isItemExpanded={isItemExpanded} trackStickyPrompt={trackStickyPrompt} selectedIndex={selectedIdx >= 0 ? selectedIdx : undefined} cursorNavRef={cursorNavRef} setCursor={setCursor} jumpRef={jumpRef} onSearchMatchesChange={onSearchMatchesChange} scanElement={scanElement} setPositions={setPositions} extractSearchText={extractSearchText} /> 701 </InVirtualListContext.Provider> : renderableMessages.flatMap(renderMessageRow)} 702 703 {streamingText && !isBriefOnly && <Box alignItems="flex-start" flexDirection="row" marginTop={1} width="100%"> 704 <Box flexDirection="row"> 705 <Box minWidth={2}> 706 <Text color="text">{BLACK_CIRCLE}</Text> 707 </Box> 708 <Box flexDirection="column"> 709 <StreamingMarkdown>{streamingText}</StreamingMarkdown> 710 </Box> 711 </Box> 712 </Box>} 713 714 {isStreamingThinkingVisible && streamingThinking && !isBriefOnly && <Box marginTop={1}> 715 <AssistantThinkingMessage param={{ 716 type: 'thinking', 717 thinking: streamingThinking.thinking 718 }} addMargin={false} isTranscriptMode={true} verbose={verbose} hideInTranscript={false} /> 719 </Box>} 720 </>; 721}; 722 723/** Key for click-to-expand: tool_use_id where available (so tool_use + its 724 * tool_result expand together), else uuid for groups/thinking. */ 725function expandKey(msg: RenderableMessage): string { 726 return (msg.type === 'assistant' || msg.type === 'user' ? getToolUseID(msg) : null) ?? msg.uuid; 727} 728 729// Custom comparator to prevent unnecessary re-renders during streaming. 730// Default React.memo does shallow comparison which fails when: 731// 1. onOpenRateLimitOptions callback is recreated (doesn't affect render output) 732// 2. streamingToolUses array is recreated on every delta, but only contentBlock matters for rendering 733// 3. streamingThinking changes on every delta - we DO want to re-render for this 734function setsEqual<T>(a: Set<T>, b: Set<T>): boolean { 735 if (a.size !== b.size) return false; 736 for (const item of a) { 737 if (!b.has(item)) return false; 738 } 739 return true; 740} 741export const Messages = React.memo(MessagesImpl, (prev, next) => { 742 const keys = Object.keys(prev) as (keyof typeof prev)[]; 743 for (const key of keys) { 744 if (key === 'onOpenRateLimitOptions' || key === 'scrollRef' || key === 'trackStickyPrompt' || key === 'setCursor' || key === 'cursorNavRef' || key === 'jumpRef' || key === 'onSearchMatchesChange' || key === 'scanElement' || key === 'setPositions') continue; 745 if (prev[key] !== next[key]) { 746 if (key === 'streamingToolUses') { 747 const p = prev.streamingToolUses; 748 const n = next.streamingToolUses; 749 if (p.length === n.length && p.every((item, i) => item.contentBlock === n[i]?.contentBlock)) { 750 continue; 751 } 752 } 753 if (key === 'inProgressToolUseIDs') { 754 if (setsEqual(prev.inProgressToolUseIDs, next.inProgressToolUseIDs)) { 755 continue; 756 } 757 } 758 if (key === 'unseenDivider') { 759 const p = prev.unseenDivider; 760 const n = next.unseenDivider; 761 if (p?.firstUnseenUuid === n?.firstUnseenUuid && p?.count === n?.count) { 762 continue; 763 } 764 } 765 if (key === 'tools') { 766 const p = prev.tools; 767 const n = next.tools; 768 if (p.length === n.length && p.every((tool, i) => tool.name === n[i]?.name)) { 769 continue; 770 } 771 } 772 // streamingThinking changes frequently - always re-render when it changes 773 // (no special handling needed, default behavior is correct) 774 return false; 775 } 776 } 777 return true; 778}); 779export function shouldRenderStatically(message: RenderableMessage, streamingToolUseIDs: Set<string>, inProgressToolUseIDs: Set<string>, siblingToolUseIDs: ReadonlySet<string>, screen: Screen, lookups: ReturnType<typeof buildMessageLookups>): boolean { 780 if (screen === 'transcript') { 781 return true; 782 } 783 switch (message.type) { 784 case 'attachment': 785 case 'user': 786 case 'assistant': 787 { 788 if (message.type === 'assistant') { 789 const block = message.message.content[0]; 790 if (block?.type === 'server_tool_use') { 791 return lookups.resolvedToolUseIDs.has(block.id); 792 } 793 } 794 const toolUseID = getToolUseID(message); 795 if (!toolUseID) { 796 return true; 797 } 798 if (streamingToolUseIDs.has(toolUseID)) { 799 return false; 800 } 801 if (inProgressToolUseIDs.has(toolUseID)) { 802 return false; 803 } 804 805 // Check if there are any unresolved PostToolUse hooks for this tool use 806 // If so, keep the message transient so the HookProgressMessage can update 807 if (hasUnresolvedHooksFromLookup(toolUseID, 'PostToolUse', lookups)) { 808 return false; 809 } 810 return every(siblingToolUseIDs, lookups.resolvedToolUseIDs); 811 } 812 case 'system': 813 { 814 // api errors always render dynamically, since we hide 815 // them as soon as we see another non-error message. 816 return message.subtype !== 'api_error'; 817 } 818 case 'grouped_tool_use': 819 { 820 const allResolved = message.messages.every(msg => { 821 const content = msg.message.content[0]; 822 return content?.type === 'tool_use' && lookups.resolvedToolUseIDs.has(content.id); 823 }); 824 return allResolved; 825 } 826 case 'collapsed_read_search': 827 { 828 // In prompt mode, never mark as static to prevent flicker between API turns 829 // (In transcript mode, we already returned true at the top of this function) 830 return false; 831 } 832 } 833} 834//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmZWF0dXJlIiwiY2hhbGsiLCJVVUlEIiwiUmVmT2JqZWN0IiwiUmVhY3QiLCJ1c2VDYWxsYmFjayIsInVzZUVmZmVjdCIsInVzZU1lbW8iLCJ1c2VSZWYiLCJ1c2VTdGF0ZSIsImV2ZXJ5IiwiZ2V0SXNSZW1vdGVNb2RlIiwiQ29tbWFuZCIsIkJMQUNLX0NJUkNMRSIsInVzZVRlcm1pbmFsU2l6ZSIsIlNjcm9sbEJveEhhbmRsZSIsInVzZVRlcm1pbmFsTm90aWZpY2F0aW9uIiwiQm94IiwiVGV4dCIsInVzZVNob3J0Y3V0RGlzcGxheSIsIlNjcmVlbiIsIlRvb2xzIiwiZmluZFRvb2xCeU5hbWUiLCJBZ2VudERlZmluaXRpb25zUmVzdWx0IiwiTWVzc2FnZSIsIk1lc3NhZ2VUeXBlIiwiTm9ybWFsaXplZE1lc3NhZ2UiLCJQcm9ncmVzc01lc3NhZ2UiLCJQcm9ncmVzc01lc3NhZ2VUeXBlIiwiUmVuZGVyYWJsZU1lc3NhZ2UiLCJBZHZpc29yQmxvY2siLCJpc0Fkdmlzb3JCbG9jayIsImNvbGxhcHNlQmFja2dyb3VuZEJhc2hOb3RpZmljYXRpb25zIiwiY29sbGFwc2VIb29rU3VtbWFyaWVzIiwiY29sbGFwc2VSZWFkU2VhcmNoR3JvdXBzIiwiY29sbGFwc2VUZWFtbWF0ZVNodXRkb3ducyIsImdldEdsb2JhbENvbmZpZyIsImlzRW52VHJ1dGh5IiwiaXNGdWxsc2NyZWVuRW52RW5hYmxlZCIsImFwcGx5R3JvdXBpbmciLCJidWlsZE1lc3NhZ2VMb29rdXBzIiwiY3JlYXRlQXNzaXN0YW50TWVzc2FnZSIsImRlcml2ZVVVSUQiLCJnZXRNZXNzYWdlc0FmdGVyQ29tcGFjdEJvdW5kYXJ5IiwiZ2V0VG9vbFVzZUlEIiwiZ2V0VG9vbFVzZUlEcyIsImhhc1VucmVzb2x2ZWRIb29rc0Zyb21Mb29rdXAiLCJpc05vdEVtcHR5TWVzc2FnZSIsIm5vcm1hbGl6ZU1lc3NhZ2VzIiwicmVvcmRlck1lc3NhZ2VzSW5VSSIsIlN0cmVhbWluZ1RoaW5raW5nIiwiU3RyZWFtaW5nVG9vbFVzZSIsInNob3VsZFNob3dVc2VyTWVzc2FnZSIsInBsdXJhbCIsInJlbmRlcmFibGVTZWFyY2hUZXh0IiwiRGl2aWRlciIsIlVuc2VlbkRpdmlkZXIiLCJMb2dvVjIiLCJTdHJlYW1pbmdNYXJrZG93biIsImhhc0NvbnRlbnRBZnRlckluZGV4IiwiTWVzc2FnZVJvdyIsIkluVmlydHVhbExpc3RDb250ZXh0IiwiTWVzc2FnZUFjdGlvbnNOYXYiLCJNZXNzYWdlQWN0aW9uc1NlbGVjdGVkQ29udGV4dCIsIk1lc3NhZ2VBY3Rpb25zU3RhdGUiLCJBc3Npc3RhbnRUaGlua2luZ01lc3NhZ2UiLCJpc051bGxSZW5kZXJpbmdBdHRhY2htZW50IiwiT2Zmc2NyZWVuRnJlZXplIiwiVG9vbFVzZUNvbmZpcm0iLCJTdGF0dXNOb3RpY2VzIiwiSnVtcEhhbmRsZSIsIkxvZ29IZWFkZXIiLCJtZW1vIiwidDAiLCIkIiwiX2MiLCJhZ2VudERlZmluaXRpb25zIiwidDEiLCJTeW1ib2wiLCJmb3IiLCJ0MiIsInByb2FjdGl2ZU1vZHVsZSIsInJlcXVpcmUiLCJCUklFRl9UT09MX05BTUUiLCJTRU5EX1VTRVJfRklMRV9UT09MX05BTUUiLCJWaXJ0dWFsTWVzc2FnZUxpc3QiLCJmaWx0ZXJGb3JCcmllZlRvb2wiLCJ0eXBlIiwic3VidHlwZSIsImlzTWV0YSIsImlzQXBpRXJyb3JNZXNzYWdlIiwibWVzc2FnZSIsImNvbnRlbnQiLCJBcnJheSIsIm5hbWUiLCJ0b29sX3VzZV9pZCIsImF0dGFjaG1lbnQiLCJvcmlnaW4iLCJjb21tYW5kTW9kZSIsIm1lc3NhZ2VzIiwiVCIsImJyaWVmVG9vbE5hbWVzIiwibmFtZVNldCIsIlNldCIsImJyaWVmVG9vbFVzZUlEcyIsImZpbHRlciIsIm1zZyIsImJsb2NrIiwiaGFzIiwiYWRkIiwiaWQiLCJ1bmRlZmluZWQiLCJhdHQiLCJkcm9wVGV4dEluQnJpZWZUdXJucyIsInR1cm5zV2l0aEJyaWVmIiwidGV4dEluZGV4VG9UdXJuIiwidHVybiIsImkiLCJsZW5ndGgiLCJzaXplIiwiXyIsInQiLCJQcm9wcyIsInRvb2xzIiwiY29tbWFuZHMiLCJ2ZXJib3NlIiwidG9vbEpTWCIsImpzeCIsIlJlYWN0Tm9kZSIsInNob3VsZEhpZGVQcm9tcHRJbnB1dCIsInNob3VsZENvbnRpbnVlQW5pbWF0aW9uIiwidG9vbFVzZUNvbmZpcm1RdWV1ZSIsImluUHJvZ3Jlc3NUb29sVXNlSURzIiwiaXNNZXNzYWdlU2VsZWN0b3JWaXNpYmxlIiwiY29udmVyc2F0aW9uSWQiLCJzY3JlZW4iLCJzdHJlYW1pbmdUb29sVXNlcyIsInNob3dBbGxJblRyYW5zY3JpcHQiLCJvbk9wZW5SYXRlTGltaXRPcHRpb25zIiwiaGlkZUxvZ28iLCJpc0xvYWRpbmciLCJoaWRlUGFzdFRoaW5raW5nIiwic3RyZWFtaW5nVGhpbmtpbmciLCJzdHJlYW1pbmdUZXh0IiwiaXNCcmllZk9ubHkiLCJ1bnNlZW5EaXZpZGVyIiwic2Nyb2xsUmVmIiwidHJhY2tTdGlja3lQcm9tcHQiLCJqdW1wUmVmIiwib25TZWFyY2hNYXRjaGVzQ2hhbmdlIiwiY291bnQiLCJjdXJyZW50Iiwic2NhbkVsZW1lbnQiLCJlbCIsIkRPTUVsZW1lbnQiLCJNYXRjaFBvc2l0aW9uIiwic2V0UG9zaXRpb25zIiwic3RhdGUiLCJwb3NpdGlvbnMiLCJyb3dPZmZzZXQiLCJjdXJyZW50SWR4IiwiZGlzYWJsZVJlbmRlckNhcCIsImN1cnNvciIsInNldEN1cnNvciIsImN1cnNvck5hdlJlZiIsIlJlZiIsInJlbmRlclJhbmdlIiwic3RhcnQiLCJlbmQiLCJNQVhfTUVTU0FHRVNfVE9fU0hPV19JTl9UUkFOU0NSSVBUX01PREUiLCJNQVhfTUVTU0FHRVNfV0lUSE9VVF9WSVJUVUFMSVpBVElPTiIsIk1FU1NBR0VfQ0FQX1NURVAiLCJTbGljZUFuY2hvciIsInV1aWQiLCJpZHgiLCJjb21wdXRlU2xpY2VTdGFydCIsImNvbGxhcHNlZCIsIlJlYWRvbmx5QXJyYXkiLCJhbmNob3JSZWYiLCJjYXAiLCJzdGVwIiwiYW5jaG9yIiwiYW5jaG9ySWR4IiwiZmluZEluZGV4IiwibSIsIk1hdGgiLCJtaW4iLCJtYXgiLCJtc2dBdFN0YXJ0IiwiTWVzc2FnZXNJbXBsIiwiY29sdW1ucyIsInRvZ2dsZVNob3dBbGxTaG9ydGN1dCIsIm5vcm1hbGl6ZWRNZXNzYWdlcyIsImlzU3RyZWFtaW5nVGhpbmtpbmdWaXNpYmxlIiwiaXNTdHJlYW1pbmciLCJzdHJlYW1pbmdFbmRlZEF0IiwiRGF0ZSIsIm5vdyIsImxhc3RUaGlua2luZ0Jsb2NrSWQiLCJqIiwiaGFzVG9vbFJlc3VsdCIsInNvbWUiLCJsYXRlc3RCYXNoT3V0cHV0VVVJRCIsInRleHQiLCJzdGFydHNXaXRoIiwibm9ybWFsaXplZFRvb2xVc2VJRHMiLCJzdHJlYW1pbmdUb29sVXNlc1dpdGhvdXRJblByb2dyZXNzIiwic3R1IiwiY29udGVudEJsb2NrIiwic3ludGhldGljU3RyZWFtaW5nVG9vbFVzZU1lc3NhZ2VzIiwiZmxhdE1hcCIsInN0cmVhbWluZ1Rvb2xVc2UiLCJpc1RyYW5zY3JpcHRNb2RlIiwiZGlzYWJsZVZpcnR1YWxTY3JvbGwiLCJwcm9jZXNzIiwiZW52IiwiQ0xBVURFX0NPREVfRElTQUJMRV9WSVJUVUFMX1NDUk9MTCIsInZpcnR1YWxTY3JvbGxSdW50aW1lR2F0ZSIsInNob3VsZFRydW5jYXRlIiwic2xpY2VBbmNob3JSZWYiLCJsb29rdXBzIiwiaGFzVHJ1bmNhdGVkTWVzc2FnZXMiLCJoaWRkZW5NZXNzYWdlQ291bnQiLCJjb21wYWN0QXdhcmVNZXNzYWdlcyIsImluY2x1ZGVTbmlwcGVkIiwibWVzc2FnZXNUb1Nob3dOb3RUcnVuY2F0ZWQiLCJFeGNsdWRlIiwibiIsImRyb3BUZXh0VG9vbE5hbWVzIiwiYnJpZWZGaWx0ZXJlZCIsIm1lc3NhZ2VzVG9TaG93Iiwic2xpY2UiLCJncm91cGVkTWVzc2FnZXMiLCJyZW5kZXJhYmxlTWVzc2FnZXMiLCJjYXBBcHBsaWVzIiwic2xpY2VTdGFydCIsInN0cmVhbWluZ1Rvb2xVc2VJRHMiLCJtYXAiLCJkaXZpZGVyQmVmb3JlSW5kZXgiLCJwcmVmaXgiLCJmaXJzdFVuc2VlblV1aWQiLCJzZWxlY3RlZElkeCIsImV4cGFuZGVkS2V5cyIsInNldEV4cGFuZGVkS2V5cyIsIlJlYWRvbmx5U2V0Iiwib25JdGVtQ2xpY2siLCJrIiwiZXhwYW5kS2V5IiwicHJldiIsIm5leHQiLCJkZWxldGUiLCJpc0l0ZW1FeHBhbmRlZCIsImxvb2t1cHNSZWYiLCJpc0l0ZW1DbGlja2FibGUiLCJiIiwiaXNfZXJyb3IiLCJ0b29sVXNlUmVzdWx0IiwidG9vbFVzZUJ5VG9vbFVzZUlEIiwiZ2V0IiwidG9vbCIsImlzUmVzdWx0VHJ1bmNhdGVkIiwiY2FuQW5pbWF0ZSIsImhhc1Rvb2xzSW5Qcm9ncmVzcyIsInByb2dyZXNzIiwicHJldlByb2dyZXNzU3RhdGUiLCJwcm9ncmVzc0VuYWJsZWQiLCJ0ZXJtaW5hbFByb2dyZXNzQmFyRW5hYmxlZCIsImlzUHJvYWN0aXZlQWN0aXZlIiwibWVzc2FnZUtleSIsInJlbmRlck1lc3NhZ2VSb3ciLCJpbmRleCIsInByZXZUeXBlIiwiaXNVc2VyQ29udGludWF0aW9uIiwiaGFzQ29udGVudEFmdGVyIiwicm93IiwiZXhwYW5kZWQiLCJ3cmFwcGVkIiwic2VhcmNoVGV4dENhY2hlIiwiV2Vha01hcCIsImV4dHJhY3RTZWFyY2hUZXh0IiwiY2FjaGVkIiwiaXNBcnJheSIsInRyIiwiZmluZCIsInR1IiwiZXh0cmFjdGVkIiwibG93ZXJlZCIsInRvTG93ZXJDYXNlIiwic2V0IiwiYm9sZCIsInRoaW5raW5nIiwic2V0c0VxdWFsIiwiYSIsIml0ZW0iLCJNZXNzYWdlcyIsImtleXMiLCJPYmplY3QiLCJrZXkiLCJwIiwic2hvdWxkUmVuZGVyU3RhdGljYWxseSIsInNpYmxpbmdUb29sVXNlSURzIiwiUmV0dXJuVHlwZSIsInJlc29sdmVkVG9vbFVzZUlEcyIsInRvb2xVc2VJRCIsImFsbFJlc29sdmVkIl0sInNvdXJjZXMiOlsiTWVzc2FnZXMudHN4Il0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGZlYXR1cmUgfSBmcm9tICdidW46YnVuZGxlJ1xuaW1wb3J0IGNoYWxrIGZyb20gJ2NoYWxrJ1xuaW1wb3J0IHR5cGUgeyBVVUlEIH0gZnJvbSAnY3J5cHRvJ1xuaW1wb3J0IHR5cGUgeyBSZWZPYmplY3QgfSBmcm9tICdyZWFjdCdcbmltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlQ2FsbGJhY2ssIHVzZUVmZmVjdCwgdXNlTWVtbywgdXNlUmVmLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgZXZlcnkgfSBmcm9tICdzcmMvdXRpbHMvc2V0LmpzJ1xuaW1wb3J0IHsgZ2V0SXNSZW1vdGVNb2RlIH0gZnJvbSAnLi4vYm9vdHN0cmFwL3N0YXRlLmpzJ1xuaW1wb3J0IHR5cGUgeyBDb21tYW5kIH0gZnJvbSAnLi4vY29tbWFuZHMuanMnXG5pbXBvcnQgeyBCTEFDS19DSVJDTEUgfSBmcm9tICcuLi9jb25zdGFudHMvZmlndXJlcy5qcydcbmltcG9ydCB7IHVzZVRlcm1pbmFsU2l6ZSB9IGZyb20gJy4uL2hvb2tzL3VzZVRlcm1pbmFsU2l6ZS5qcydcbmltcG9ydCB0eXBlIHsgU2Nyb2xsQm94SGFuZGxlIH0gZnJvbSAnLi4vaW5rL2NvbXBvbmVudHMvU2Nyb2xsQm94LmpzJ1xuaW1wb3J0IHsgdXNlVGVybWluYWxOb3RpZmljYXRpb24gfSBmcm9tICcuLi9pbmsvdXNlVGVybWluYWxOb3RpZmljYXRpb24uanMnXG5pbXBvcnQgeyBCb3gsIFRleHQgfSBmcm9tICcuLi9pbmsuanMnXG5pbXBvcnQgeyB1c2VTaG9ydGN1dERpc3BsYXkgfSBmcm9tICcuLi9rZXliaW5kaW5ncy91c2VTaG9ydGN1dERpc3BsYXkuanMnXG5pbXBvcnQgdHlwZSB7IFNjcmVlbiB9IGZyb20gJy4uL3NjcmVlbnMvUkVQTC5qcydcbmltcG9ydCB0eXBlIHsgVG9vbHMgfSBmcm9tICcuLi9Ub29sLmpzJ1xuaW1wb3J0IHsgZmluZFRvb2xCeU5hbWUgfSBmcm9tICcuLi9Ub29sLmpzJ1xuaW1wb3J0IHR5cGUgeyBBZ2VudERlZmluaXRpb25zUmVzdWx0IH0gZnJvbSAnLi4vdG9vbHMvQWdlbnRUb29sL2xvYWRBZ2VudHNEaXIuanMnXG5pbXBvcnQgdHlwZSB7XG4gIE1lc3NhZ2UgYXMgTWVzc2FnZVR5cGUsXG4gIE5vcm1hbGl6ZWRNZXNzYWdlLFxuICBQcm9ncmVzc01lc3NhZ2UgYXMgUHJvZ3Jlc3NNZXNzYWdlVHlwZSxcbiAgUmVuZGVyYWJsZU1lc3NhZ2UsXG59IGZyb20gJy4uL3R5cGVzL21lc3NhZ2UuanMnXG5pbXBvcnQgeyB0eXBlIEFkdmlzb3JCbG9jaywgaXNBZHZpc29yQmxvY2sgfSBmcm9tICcuLi91dGlscy9hZHZpc29yLmpzJ1xuaW1wb3J0IHsgY29sbGFwc2VCYWNrZ3JvdW5kQmFzaE5vdGlmaWNhdGlvbnMgfSBmcm9tICcuLi91dGlscy9jb2xsYXBzZUJhY2tncm91bmRCYXNoTm90aWZpY2F0aW9ucy5qcydcbmltcG9ydCB7IGNvbGxhcHNlSG9va1N1bW1hcmllcyB9IGZyb20gJy4uL3V0aWxzL2NvbGxhcHNlSG9va1N1bW1hcmllcy5qcydcbmltcG9ydCB7IGNvbGxhcHNlUmVhZFNlYXJjaEdyb3VwcyB9IGZyb20gJy4uL3V0aWxzL2NvbGxhcHNlUmVhZFNlYXJjaC5qcydcbmltcG9ydCB7IGNvbGxhcHNlVGVhbW1hdGVTaHV0ZG93bnMgfSBmcm9tICcuLi91dGlscy9jb2xsYXBzZVRlYW1tYXRlU2h1dGRvd25zLmpzJ1xuaW1wb3J0IHsgZ2V0R2xvYmFsQ29uZmlnIH0gZnJvbSAnLi4vdXRpbHMvY29uZmlnLmpzJ1xuaW1wb3J0IHsgaXNFbnZUcnV0aHkgfSBmcm9tICcuLi91dGlscy9lbnZVdGlscy5qcydcbmltcG9ydCB7IGlzRnVsbHNjcmVlbkVudkVuYWJsZWQgfSBmcm9tICcuLi91dGlscy9mdWxsc2NyZWVuLmpzJ1xuaW1wb3J0IHsgYXBwbHlHcm91cGluZyB9IGZyb20gJy4uL3V0aWxzL2dyb3VwVG9vbFVzZXMuanMnXG5pbXBvcnQge1xuICBidWlsZE1lc3NhZ2VMb29rdXBzLFxuICBjcmVhdGVBc3Npc3RhbnRNZXNzYWdlLFxuICBkZXJpdmVVVUlELFxuICBnZXRNZXNzYWdlc0FmdGVyQ29tcGFjdEJvdW5kYXJ5LFxuICBnZXRUb29sVXNlSUQsXG4gIGdldFRvb2xVc2VJRHMsXG4gIGhhc1VucmVzb2x2ZWRIb29rc0Zyb21Mb29rdXAsXG4gIGlzTm90RW1wdHlNZXNzYWdlLFxuICBub3JtYWxpemVNZXNzYWdlcyxcbiAgcmVvcmRlck1lc3NhZ2VzSW5VSSxcbiAgdHlwZSBTdHJlYW1pbmdUaGlua2luZyxcbiAgdHlwZSBTdHJlYW1pbmdUb29sVXNlLFxuICBzaG91bGRTaG93VXNlck1lc3NhZ2UsXG59IGZyb20gJy4uL3V0aWxzL21lc3NhZ2VzLmpzJ1xuaW1wb3J0IHsgcGx1cmFsIH0gZnJvbSAnLi4vdXRpbHMvc3RyaW5nVXRpbHMuanMnXG5pbXBvcnQgeyByZW5kZXJhYmxlU2VhcmNoVGV4dCB9IGZyb20gJy4uL3V0aWxzL3RyYW5zY3JpcHRTZWFyY2guanMnXG5pbXBvcnQgeyBEaXZpZGVyIH0gZnJvbSAnLi9kZXNpZ24tc3lzdGVtL0RpdmlkZXIuanMnXG5pbXBvcnQgdHlwZSB7IFVuc2VlbkRpdmlkZXIgfSBmcm9tICcuL0Z1bGxzY3JlZW5MYXlvdXQuanMnXG5pbXBvcnQgeyBMb2dvVjIgfSBmcm9tICcuL0xvZ29WMi9Mb2dvVjIuanMnXG5pbXBvcnQgeyBTdHJlYW1pbmdNYXJrZG93biB9IGZyb20gJy4vTWFya2Rvd24uanMnXG5pbXBvcnQgeyBoYXNDb250ZW50QWZ0ZXJJbmRleCwgTWVzc2FnZVJvdyB9IGZyb20gJy4vTWVzc2FnZVJvdy5qcydcbmltcG9ydCB7XG4gIEluVmlydHVhbExpc3RDb250ZXh0LFxuICB0eXBlIE1lc3NhZ2VBY3Rpb25zTmF2LFxuICBNZXNzYWdlQWN0aW9uc1NlbGVjdGVkQ29udGV4dCxcbiAgdHlwZSBNZXNzYWdlQWN0aW9uc1N0YXRlLFxufSBmcm9tICcuL21lc3NhZ2VBY3Rpb25zLmpzJ1xuaW1wb3J0IHsgQXNzaXN0YW50VGhpbmtpbmdNZXNzYWdlIH0gZnJvbSAnLi9tZXNzYWdlcy9Bc3Npc3RhbnRUaGlua2luZ01lc3NhZ2UuanMnXG5pbXBvcnQgeyBpc051bGxSZW5kZXJpbmdBdHRhY2htZW50IH0gZnJvbSAnLi9tZXNzYWdlcy9udWxsUmVuZGVyaW5nQXR0YWNobWVudHMuanMnXG5pbXBvcnQgeyBPZmZzY3JlZW5GcmVlemUgfSBmcm9tICcuL09mZnNjcmVlbkZyZWV6ZS5qcydcbmltcG9ydCB0eXBlIHsgVG9vbFVzZUNvbmZpcm0gfSBmcm9tICcuL3Blcm1pc3Npb25zL1Blcm1pc3Npb25SZXF1ZXN0LmpzJ1xuaW1wb3J0IHsgU3RhdHVzTm90aWNlcyB9IGZyb20gJy4vU3RhdHVzTm90aWNlcy5qcydcbmltcG9ydCB0eXBlIHsgSnVtcEhhbmRsZSB9IGZyb20gJy4vVmlydHVhbE1lc3NhZ2VMaXN0LmpzJ1xuXG4vLyBNZW1vZWQgbG9nbyBoZWFkZXI6IHRoaXMgYm94IGlzIHRoZSBGSVJTVCBzaWJsaW5nIGJlZm9yZSBhbGwgTWVzc2FnZVJvd3Ncbi8vIGluIG1haW4tc2NyZWVuIG1vZGUuIElmIGl0IGJlY29tZXMgZGlydHkgb24gZXZlcnkgTWVzc2FnZXMgcmUtcmVuZGVyLFxuLy8gcmVuZGVyQ2hpbGRyZW4ncyBzZWVuRGlydHlDaGlsZCBjYXNjYWRlIGRpc2FibGVzIHByZXZTY3JlZW4gKGJsaXQpIGZvclxuLy8gQUxMIHN1YnNlcXVlbnQgc2libGluZ3Mg4oCUIGV2ZXJ5IE1lc3NhZ2VSb3cgcmUtd3JpdGVzIGZyb20gc2NyYXRjaCBpbnN0ZWFkXG4vLyBvZiBibGl0dGluZy4gSW4gbG9uZyBzZXNzaW9ucyAofjI4MDAgbWVzc2FnZXMpIHRoaXMgaXMgMTUwSysgd3JpdGVzL2ZyYW1lXG4vLyBhbmQgcGVncyBDUFUgYXQgMTAwJS4gTWVtbyBvbiBhZ2VudERlZmluaXRpb25zIHNvIGEgbmV3IG1lc3NhZ2VzIGFycmF5XG4vLyBkb2Vzbid0IGludmFsaWRhdGUgdGhlIGxvZ28gc3VidHJlZS4gTG9nb1YyL1N0YXR1c05vdGljZXMgaW50ZXJuYWxseVxuLy8gc3Vic2NyaWJlIHRvIHVzZUFwcFN0YXRlL3VzZVNldHRpbmdzIGZvciB0aGVpciBvd24gdXBkYXRlcy5cbmNvbnN0IExvZ29IZWFkZXIgPSBSZWFjdC5tZW1vKGZ1bmN0aW9uIExvZ29IZWFkZXIoe1xuICBhZ2VudERlZmluaXRpb25zLFxufToge1xuICBhZ2VudERlZmluaXRpb25zOiBBZ2VudERlZmluaXRpb25zUmVzdWx0IHwgdW5kZWZpbmVkXG59KTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgLy8gTG9nb1YyIGhhcyBpdHMgb3duIGludGVybmFsIE9mZnNjcmVlbkZyZWV6ZSAoY2F0Y2hlcyBpdHMgdXNlQXBwU3RhdGVcbiAgLy8gcmUtcmVuZGVycykuIFRoaXMgb3V0ZXIgZnJlZXplIGNhdGNoZXMgYWdlbnREZWZpbml0aW9ucyBjaGFuZ2VzIGFuZCBhbnlcbiAgLy8gZnV0dXJlIFN0YXR1c05vdGljZXMgc3Vic2NyaXB0aW9ucyB3aGlsZSB0aGUgaGVhZGVyIGlzIGluIHNjcm9sbGJhY2suXG4gIHJldHVybiAoXG4gICAgPE9mZnNjcmVlbkZyZWV6ZT5cbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIGdhcD17MX0+XG4gICAgICAgIDxMb2dvVjIgLz5cbiAgICAgICAgPFJlYWN0LlN1c3BlbnNlIGZhbGxiYWNrPXtudWxsfT5cbiAgICAgICAgICA8U3RhdHVzTm90aWNlcyBhZ2VudERlZmluaXRpb25zPXthZ2VudERlZmluaXRpb25zfSAvPlxuICAgICAgICA8L1JlYWN0LlN1c3BlbnNlPlxuICAgICAgPC9Cb3g+XG4gICAgPC9PZmZzY3JlZW5GcmVlemU+XG4gIClcbn0pXG5cbi8vIERlYWQgY29kZSBlbGltaW5hdGlvbjogY29uZGl0aW9uYWwgaW1wb3J0IGZvciBwcm9hY3RpdmUgbW9kZVxuLyogZXNsaW50LWRpc2FibGUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXJlcXVpcmUtaW1wb3J0cyAqL1xuY29uc3QgcHJvYWN0aXZlTW9kdWxlID1cbiAgZmVhdHVyZSgnUFJPQUNUSVZFJykgfHwgZmVhdHVyZSgnS0FJUk9TJylcbiAgICA/IHJlcXVpcmUoJy4uL3Byb2FjdGl2ZS9pbmRleC5qcycpXG4gICAgOiBudWxsXG5jb25zdCBCUklFRl9UT09MX05BTUU6IHN0cmluZyB8IG51bGwgPVxuICBmZWF0dXJlKCdLQUlST1MnKSB8fCBmZWF0dXJlKCdLQUlST1NfQlJJRUYnKVxuICAgID8gKFxuICAgICAgICByZXF1aXJlKCcuLi90b29scy9CcmllZlRvb2wvcHJvbXB0LmpzJykgYXMgdHlwZW9mIGltcG9ydCgnLi4vdG9vbHMvQnJpZWZUb29sL3Byb21wdC5qcycpXG4gICAgICApLkJSSUVGX1RPT0xfTkFNRVxuICAgIDogbnVsbFxuY29uc3QgU0VORF9VU0VSX0ZJTEVfVE9PTF9OQU1FOiBzdHJpbmcgfCBudWxsID0gZmVhdHVyZSgnS0FJUk9TJylcbiAgPyAoXG4gICAgICByZXF1aXJlKCcuLi90b29scy9TZW5kVXNlckZpbGVUb29sL3Byb21wdC5qcycpIGFzIHR5cGVvZiBpbXBvcnQoJy4uL3Rvb2xzL1NlbmRVc2VyRmlsZVRvb2wvcHJvbXB0LmpzJylcbiAgICApLlNFTkRfVVNFUl9GSUxFX1RPT0xfTkFNRVxuICA6IG51bGxcblxuLyogZXNsaW50LWVuYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tcmVxdWlyZS1pbXBvcnRzICovXG5pbXBvcnQgeyBWaXJ0dWFsTWVzc2FnZUxpc3QgfSBmcm9tICcuL1ZpcnR1YWxNZXNzYWdlTGlzdC5qcydcblxuLyoqXG4gKiBJbiBicmllZi1vbmx5IG1vZGUsIGZpbHRlciBtZXNzYWdlcyB0byBzaG93IE9OTFkgQnJpZWYgdG9vbF91c2UgYmxvY2tzLFxuICogdGhlaXIgdG9vbF9yZXN1bHRzLCBhbmQgcmVhbCB1c2VyIGlucHV0LiBBbGwgYXNzaXN0YW50IHRleHQgaXMgZHJvcHBlZCDigJRcbiAqIGlmIHRoZSBtb2RlbCBmb3JnZXRzIHRvIGNhbGwgQnJpZWYsIHRoZSB1c2VyIHNlZXMgbm90aGluZyBmb3IgdGhhdCB0dXJuLlxuICogVGhhdCdzIG9uIHRoZSBtb2RlbCB0byBnZXQgcmlnaHQ7IHRoZSBmaWx0ZXIgZG9lcyBub3Qgc2Vjb25kLWd1ZXNzIGl0LlxuICovXG5leHBvcnQgZnVuY3Rpb24gZmlsdGVyRm9yQnJpZWZUb29sPFxuICBUIGV4dGVuZHMge1xuICAgIHR5cGU6IHN0cmluZ1xuICAgIHN1YnR5cGU/OiBzdHJpbmdcbiAgICBpc01ldGE/OiBib29sZWFuXG4gICAgaXNBcGlFcnJvck1lc3NhZ2U/OiBib29sZWFuXG4gICAgbWVzc2FnZT86IHtcbiAgICAgIGNvbnRlbnQ6IEFycmF5PHtcbiAgICAgICAgdHlwZTogc3RyaW5nXG4gICAgICAgIG5hbWU/OiBzdHJpbmdcbiAgICAgICAgdG9vbF91c2VfaWQ/OiBzdHJpbmdcbiAgICAgIH0+XG4gICAgfVxuICAgIGF0dGFjaG1lbnQ/OiB7XG4gICAgICB0eXBlOiBzdHJpbmdcbiAgICAgIGlzTWV0YT86IGJvb2xlYW5cbiAgICAgIG9yaWdpbj86IHVua25vd25cbiAgICAgIGNvbW1hbmRNb2RlPzogc3RyaW5nXG4gICAgfVxuICB9LFxuPihtZXNzYWdlczogVFtdLCBicmllZlRvb2xOYW1lczogc3RyaW5nW10pOiBUW10ge1xuICBjb25zdCBuYW1lU2V0ID0gbmV3IFNldChicmllZlRvb2xOYW1lcylcbiAgLy8gdG9vbF91c2UgYWx3YXlzIHByZWNlZGVzIGl0cyB0b29sX3Jlc3VsdCBpbiB0aGUgYXJyYXksIHNvIHdlIGNhbiBjb2xsZWN0XG4gIC8vIElEcyBhbmQgbWF0Y2ggYWdhaW5zdCB0aGVtIGluIGEgc2luZ2xlIHBhc3MuXG4gIGNvbnN0IGJyaWVmVG9vbFVzZUlEcyA9IG5ldyBTZXQ8c3RyaW5nPigpXG4gIHJldHVybiBtZXNzYWdlcy5maWx0ZXIobXNnID0+IHtcbiAgICAvLyBTeXN0ZW0gbWVzc2FnZXMgKGF0dGFjaCBjb25maXJtYXRpb24sIHJlbW90ZSBlcnJvcnMsIGNvbXBhY3QgYm91bmRhcmllcylcbiAgICAvLyBtdXN0IHN0YXkgdmlzaWJsZSDigJQgZHJvcHBpbmcgdGhlbSBsZWF2ZXMgdGhlIHZpZXdlciB3aXRoIG5vIGZlZWRiYWNrLlxuICAgIC8vIEV4Y2VwdGlvbjogYXBpX21ldHJpY3MgaXMgcGVyLXR1cm4gZGVidWcgbm9pc2UgKFRURlQsIGNvbmZpZyB3cml0ZXMsXG4gICAgLy8gaG9vayB0aW1pbmcpIHRoYXQgZGVmZWF0cyB0aGUgcG9pbnQgb2YgYnJpZWYgbW9kZS4gU3RpbGwgdmlzaWJsZSBpblxuICAgIC8vIHRyYW5zY3JpcHQgbW9kZSAoY3RybCtvKSB3aGljaCBieXBhc3NlcyB0aGlzIGZpbHRlci5cbiAgICBpZiAobXNnLnR5cGUgPT09ICdzeXN0ZW0nKSByZXR1cm4gbXNnLnN1YnR5cGUgIT09ICdhcGlfbWV0cmljcydcbiAgICBjb25zdCBibG9jayA9IG1zZy5tZXNzYWdlPy5jb250ZW50WzBdXG4gICAgaWYgKG1zZy50eXBlID09PSAnYXNzaXN0YW50Jykge1xuICAgICAgLy8gQVBJIGVycm9yIG1lc3NhZ2VzIChhdXRoIGZhaWx1cmVzLCByYXRlIGxpbWl0cywgZXRjLikgbXVzdCBzdGF5IHZpc2libGVcbiAgICAgIGlmIChtc2cuaXNBcGlFcnJvck1lc3NhZ2UpIHJldHVybiB0cnVlXG4gICAgICAvLyBLZWVwIEJyaWVmIHRvb2xfdXNlIGJsb2NrcyAocmVuZGVycyB3aXRoIHN0YW5kYXJkIHRvb2wgY2FsbCBjaHJvbWUsXG4gICAgICAvLyBhbmQgbXVzdCBiZSBpbiB0aGUgbGlzdCBzbyBidWlsZE1lc3NhZ2VMb29rdXBzIGNhbiByZXNvbHZlIHRvb2wgcmVzdWx0cylcbiAgICAgIGlmIChibG9jaz8udHlwZSA9PT0gJ3Rvb2xfdXNlJyAmJiBibG9jay5uYW1lICYmIG5hbWVTZXQuaGFzKGJsb2NrLm5hbWUpKSB7XG4gICAgICAgIGlmICgnaWQnIGluIGJsb2NrKSB7XG4gICAgICAgICAgYnJpZWZUb29sVXNlSURzLmFkZCgoYmxvY2sgYXMgeyBpZDogc3RyaW5nIH0pLmlkKVxuICAgICAgICB9XG4gICAgICAgIHJldHVybiB0cnVlXG4gICAgICB9XG4gICAgICByZXR1cm4gZmFsc2VcbiAgICB9XG4gICAgaWYgKG1zZy50eXBlID09PSAndXNlcicpIHtcbiAgICAgIGlmIChibG9jaz8udHlwZSA9PT0gJ3Rvb2xfcmVzdWx0Jykge1xuICAgICAgICByZXR1cm4gKFxuICAgICAgICAgIGJsb2NrLnRvb2xfdXNlX2lkICE9PSB1bmRlZmluZWQgJiZcbiAgICAgICAgICBicmllZlRvb2xVc2VJRHMuaGFzKGJsb2NrLnRvb2xfdXNlX2lkKVxuICAgICAgICApXG4gICAgICB9XG4gICAgICAvLyBSZWFsIHVzZXIgaW5wdXQgb25seSDigJQgZHJvcCBtZXRhL3RpY2sgbWVzc2FnZXMuXG4gICAgICByZXR1cm4gIW1zZy5pc01ldGFcbiAgICB9XG4gICAgaWYgKG1zZy50eXBlID09PSAnYXR0YWNobWVudCcpIHtcbiAgICAgIC8vIEh1bWFuIGlucHV0IGRyYWluZWQgbWlkLXR1cm4gYXJyaXZlcyBhcyBhIHF1ZXVlZF9jb21tYW5kIGF0dGFjaG1lbnRcbiAgICAgIC8vIChxdWVyeS50cyBtaWQtY2hhaW4gZHJhaW4g4oaSIGdldFF1ZXVlZENvbW1hbmRBdHRhY2htZW50cykuIEtlZXAgaXQg4oCUXG4gICAgICAvLyBpdCdzIHdoYXQgdGhlIHVzZXIgdHlwZWQuIGNvbW1hbmRNb2RlID09PSAncHJvbXB0JyBwb3NpdGl2ZWx5XG4gICAgICAvLyBpZGVudGlmaWVzIGh1bWFuLXR5cGVkIGlucHV0OyB0YXNrLW5vdGlmaWNhdGlvbiBjYWxsZXJzIHNldFxuICAgICAgLy8gbW9kZTogJ3Rhc2stbm90aWZpY2F0aW9uJyBidXQgbm90IG9yaWdpbi9pc01ldGEsIHNvIHRoZSBwb3NpdGl2ZVxuICAgICAgLy8gY29tbWFuZE1vZGUgY2hlY2sgaXMgcmVxdWlyZWQgdG8gZXhjbHVkZSB0aGVtLlxuICAgICAgY29uc3QgYXR0ID0gbXNnLmF0dGFjaG1lbnRcbiAgICAgIHJldHVybiAoXG4gICAgICAgIGF0dD8udHlwZSA9PT0gJ3F1ZXVlZF9jb21tYW5kJyAmJlxuICAgICAgICBhdHQuY29tbWFuZE1vZGUgPT09ICdwcm9tcHQnICYmXG4gICAgICAgICFhdHQuaXNNZXRhICYmXG4gICAgICAgIGF0dC5vcmlnaW4gPT09IHVuZGVmaW5lZFxuICAgICAgKVxuICAgIH1cbiAgICByZXR1cm4gZmFsc2VcbiAgfSlcbn1cblxuLyoqXG4gKiBGdWxsLXRyYW5zY3JpcHQgY29tcGFuaW9uIHRvIGZpbHRlckZvckJyaWVmVG9vbC4gV2hlbiB0aGUgQnJpZWYgdG9vbCBpc1xuICogaW4gdXNlLCB0aGUgbW9kZWwncyB0ZXh0IG91dHB1dCBpcyByZWR1bmRhbnQgd2l0aCB0aGUgU2VuZFVzZXJNZXNzYWdlXG4gKiBjb250ZW50IGl0IHdyb3RlIHJpZ2h0IGFmdGVyIOKAlCBkcm9wIHRoZSB0ZXh0IHNvIG9ubHkgdGhlIFNlbmRVc2VyTWVzc2FnZVxuICogYmxvY2sgc2hvd3MuIFRvb2wgY2FsbHMgYW5kIHRoZWlyIHJlc3VsdHMgc3RheSB2aXNpYmxlLlxuICpcbiAqIFBlci10dXJuOiBvbmx5IGRyb3BzIHRleHQgaW4gdHVybnMgdGhhdCBhY3R1YWxseSBjYWxsZWQgQnJpZWYuIElmIHRoZVxuICogbW9kZWwgZm9yZ2V0cywgdGV4dCBzdGlsbCBzaG93cyDigJQgb3RoZXJ3aXNlIHRoZSB1c2VyIHdvdWxkIHNlZSBub3RoaW5nLlxuICovXG5leHBvcnQgZnVuY3Rpb24gZHJvcFRleHRJbkJyaWVmVHVybnM8XG4gIFQgZXh0ZW5kcyB7XG4gICAgdHlwZTogc3RyaW5nXG4gICAgaXNNZXRhPzogYm9vbGVhblxuICAgIG1lc3NhZ2U/OiB7IGNvbnRlbnQ6IEFycmF5PHsgdHlwZTogc3RyaW5nOyBuYW1lPzogc3RyaW5nIH0+IH1cbiAgfSxcbj4obWVzc2FnZXM6IFRbXSwgYnJpZWZUb29sTmFtZXM6IHN0cmluZ1tdKTogVFtdIHtcbiAgY29uc3QgbmFtZVNldCA9IG5ldyBTZXQoYnJpZWZUb29sTmFtZXMpXG4gIC8vIEZpcnN0IHBhc3M6IGZpbmQgd2hpY2ggdHVybnMgKGJvdW5kZWQgYnkgbm9uLW1ldGEgdXNlciBtZXNzYWdlcykgY29udGFpblxuICAvLyBhIEJyaWVmIHRvb2xfdXNlLiBUYWcgZWFjaCBhc3Npc3RhbnQgdGV4dCBibG9jayB3aXRoIGl0cyB0dXJuIGluZGV4LlxuICBjb25zdCB0dXJuc1dpdGhCcmllZiA9IG5ldyBTZXQ8bnVtYmVyPigpXG4gIGNvbnN0IHRleHRJbmRleFRvVHVybjogbnVtYmVyW10gPSBbXVxuICBsZXQgdHVybiA9IDBcbiAgZm9yIChsZXQgaSA9IDA7IGkgPCBtZXNzYWdlcy5sZW5ndGg7IGkrKykge1xuICAgIGNvbnN0IG1zZyA9IG1lc3NhZ2VzW2ldIVxuICAgIGNvbnN0IGJsb2NrID0gbXNnLm1lc3NhZ2U/LmNvbnRlbnRbMF1cbiAgICBpZiAobXNnLnR5cGUgPT09ICd1c2VyJyAmJiBibG9jaz8udHlwZSAhPT0gJ3Rvb2xfcmVzdWx0JyAmJiAhbXNnLmlzTWV0YSkge1xuICAgICAgdHVybisrXG4gICAgICBjb250aW51ZVxuICAgIH1cbiAgICBpZiAobXNnLnR5cGUgPT09ICdhc3Npc3RhbnQnKSB7XG4gICAgICBpZiAoYmxvY2s/LnR5cGUgPT09ICd0ZXh0Jykge1xuICAgICAgICB0ZXh0SW5kZXhUb1R1cm5baV0gPSB0dXJuXG4gICAgICB9IGVsc2UgaWYgKFxuICAgICAgICBibG9jaz8udHlwZSA9PT0gJ3Rvb2xfdXNlJyAmJlxuICAgICAgICBibG9jay5uYW1lICYmXG4gICAgICAgIG5hbWVTZXQuaGFzKGJsb2NrLm5hbWUpXG4gICAgICApIHtcbiAgICAgICAgdHVybnNXaXRoQnJpZWYuYWRkKHR1cm4pXG4gICAgICB9XG4gICAgfVxuICB9XG4gIGlmICh0dXJuc1dpdGhCcmllZi5zaXplID09PSAwKSByZXR1cm4gbWVzc2FnZXNcbiAgLy8gU2Vjb25kIHBhc3M6IGRyb3AgdGV4dCBibG9ja3Mgd2hvc2UgdHVybiBjYWxsZWQgQnJpZWYuXG4gIHJldHVybiBtZXNzYWdlcy5maWx0ZXIoKF8sIGkpID0+IHtcbiAgICBjb25zdCB0ID0gdGV4dEluZGV4VG9UdXJuW2ldXG4gICAgcmV0dXJuIHQgPT09IHVuZGVmaW5lZCB8fCAhdHVybnNXaXRoQnJpZWYuaGFzKHQpXG4gIH0pXG59XG5cbnR5cGUgUHJvcHMgPSB7XG4gIG1lc3NhZ2VzOiBNZXNzYWdlVHlwZVtdXG4gIHRvb2xzOiBUb29sc1xuICBjb21tYW5kczogQ29tbWFuZFtdXG4gIHZlcmJvc2U6IGJvb2xlYW5cbiAgdG9vbEpTWDoge1xuICAgIGpzeDogUmVhY3QuUmVhY3ROb2RlIHwgbnVsbFxuICAgIHNob3VsZEhpZGVQcm9tcHRJbnB1dDogYm9vbGVhblxuICAgIHNob3VsZENvbnRpbnVlQW5pbWF0aW9uPzogdHJ1ZVxuICB9IHwgbnVsbFxuICB0b29sVXNlQ29uZmlybVF1ZXVlOiBUb29sVXNlQ29uZmlybVtdXG4gIGluUHJvZ3Jlc3NUb29sVXNlSURzOiBTZXQ8c3RyaW5nPlxuICBpc01lc3NhZ2VTZWxlY3RvclZpc2libGU6IGJvb2xlYW5cbiAgY29udmVyc2F0aW9uSWQ6IHN0cmluZ1xuICBzY3JlZW46IFNjcmVlblxuICBzdHJlYW1pbmdUb29sVXNlczogU3RyZWFtaW5nVG9vbFVzZVtdXG4gIHNob3dBbGxJblRyYW5zY3JpcHQ/OiBib29sZWFuXG4gIGFnZW50RGVmaW5pdGlvbnM/OiBBZ2VudERlZmluaXRpb25zUmVzdWx0XG4gIG9uT3BlblJhdGVMaW1pdE9wdGlvbnM/OiAoKSA9PiB2b2lkXG4gIC8qKiBIaWRlIHRoZSBsb2dvL2hlYWRlciAtIHVzZWQgZm9yIHN1YmFnZW50IHpvb20gdmlldyAqL1xuICBoaWRlTG9nbz86IGJvb2xlYW5cbiAgaXNMb2FkaW5nOiBib29sZWFuXG4gIC8qKiBJbiB0cmFuc2NyaXB0IG1vZGUsIGhpZGUgYWxsIHRoaW5raW5nIGJsb2NrcyBleGNlcHQgdGhlIGxhc3Qgb25lICovXG4gIGhpZGVQYXN0VGhpbmtpbmc/OiBib29sZWFuXG4gIC8qKiBTdHJlYW1pbmcgdGhpbmtpbmcgY29udGVudCAobGl2ZSB1cGRhdGVzLCBub3QgZnJvemVuKSAqL1xuICBzdHJlYW1pbmdUaGlua2luZz86IFN0cmVhbWluZ1RoaW5raW5nIHwgbnVsbFxuICAvKiogU3RyZWFtaW5nIHRleHQgcHJldmlldyAocmVuZGVyZWQgYXMgbGFzdCBpdGVtIHNvIHRyYW5zaXRpb24gdG8gZmluYWwgbWVzc2FnZSBpcyBwb3NpdGlvbmFsbHkgc2VhbWxlc3MpICovXG4gIHN0cmVhbWluZ1RleHQ/OiBzdHJpbmcgfCBudWxsXG4gIC8qKiBXaGVuIHRydWUsIG9ubHkgc2hvdyBCcmllZiB0b29sIG91dHB1dCAoaGlkZSBldmVyeXRoaW5nIGVsc2UpICovXG4gIGlzQnJpZWZPbmx5PzogYm9vbGVhblxuICAvKiogRnVsbHNjcmVlbi1tb2RlIFwi4pSA4pSA4pSAIE4gbmV3IOKUgOKUgOKUgFwiIGRpdmlkZXIuIFJlbmRlcnMgYmVmb3JlIHRoZSBmaXJzdFxuICAgKiAgcmVuZGVyYWJsZU1lc3NhZ2UgZGVyaXZlZCBmcm9tIGZpcnN0VW5zZWVuVXVpZCAobWF0Y2hlZCBieSB0aGUgMjQtY2hhclxuICAgKiAgcHJlZml4IHRoYXQgZGVyaXZlVVVJRCBwcmVzZXJ2ZXMpLiAqL1xuICB1bnNlZW5EaXZpZGVyPzogVW5zZWVuRGl2aWRlclxuICAvKiogRnVsbHNjcmVlbi1tb2RlIFNjcm9sbEJveCBoYW5kbGUuIEVuYWJsZXMgUmVhY3QtbGV2ZWwgdmlydHVhbGl6YXRpb24gd2hlbiBwcmVzZW50LiAqL1xuICBzY3JvbGxSZWY/OiBSZWZPYmplY3Q8U2Nyb2xsQm94SGFuZGxlIHwgbnVsbD5cbiAgLyoqIEZ1bGxzY3JlZW4tbW9kZTogZW5hYmxlIHN0aWNreS1wcm9tcHQgdHJhY2tpbmcgKHdyaXRlcyB2aWEgU2Nyb2xsQ2hyb21lQ29udGV4dCkuICovXG4gIHRyYWNrU3RpY2t5UHJvbXB0PzogYm9vbGVhblxuICAvKiogVHJhbnNjcmlwdCBzZWFyY2g6IGp1bXAtdG8taW5kZXggKyBzZXRTZWFyY2hRdWVyeS9uZXh0TWF0Y2gvcHJldk1hdGNoLiAqL1xuICBqdW1wUmVmPzogUmVmT2JqZWN0PEp1bXBIYW5kbGUgfCBudWxsPlxuICAvKiogVHJhbnNjcmlwdCBzZWFyY2g6IGZpcmVzIHdoZW4gbWF0Y2ggY291bnQvcG9zaXRpb24gY2hhbmdlcy4gKi9cbiAgb25TZWFyY2hNYXRjaGVzQ2hhbmdlPzogKGNvdW50OiBudW1iZXIsIGN1cnJlbnQ6IG51bWJlcikgPT4gdm9pZFxuICAvKiogUGFpbnQgYW4gZXhpc3RpbmcgRE9NIHN1YnRyZWUgdG8gZnJlc2ggU2NyZWVuLCBzY2FuLiBFbGVtZW50IGNvbWVzXG4gICAqICBmcm9tIHRoZSBtYWluIHRyZWUgKGFsbCByZWFsIHByb3ZpZGVycykuIE1lc3NhZ2UtcmVsYXRpdmUgcG9zaXRpb25zLiAqL1xuICBzY2FuRWxlbWVudD86IChcbiAgICBlbDogaW1wb3J0KCcuLi9pbmsvZG9tLmpzJykuRE9NRWxlbWVudCxcbiAgKSA9PiBpbXBvcnQoJy4uL2luay9yZW5kZXItdG8tc2NyZWVuLmpzJykuTWF0Y2hQb3NpdGlvbltdXG4gIC8qKiBQb3NpdGlvbi1iYXNlZCBDVVJSRU5UIGhpZ2hsaWdodC4gcG9zaXRpb25zIHN0YWJsZSAobXNnLXJlbGF0aXZlKSxcbiAgICogIHJvd09mZnNldCB0cmFja3Mgc2Nyb2xsLiBudWxsIGNsZWFycy4gKi9cbiAgc2V0UG9zaXRpb25zPzogKFxuICAgIHN0YXRlOiB7XG4gICAgICBwb3NpdGlvbnM6IGltcG9ydCgnLi4vaW5rL3JlbmRlci10by1zY3JlZW4uanMnKS5NYXRjaFBvc2l0aW9uW11cbiAgICAgIHJvd09mZnNldDogbnVtYmVyXG4gICAgICBjdXJyZW50SWR4OiBudW1iZXJcbiAgICB9IHwgbnVsbCxcbiAgKSA9PiB2b2lkXG4gIC8qKiBCeXBhc3MgTUFYX01FU1NBR0VTX1dJVEhPVVRfVklSVFVBTElaQVRJT04uIEZvciBvbmUtc2hvdCBoZWFkbGVzcyByZW5kZXJzXG4gICAqICAoZS5nLiAvZXhwb3J0IHZpYSByZW5kZXJUb1N0cmluZykgd2hlcmUgdGhlIG1lbW9yeSBjb25jZXJuIGRvZXNuJ3QgYXBwbHlcbiAgICogIGFuZCB0aGUgXCJhbHJlYWR5IGluIHNjcm9sbGJhY2tcIiBqdXN0aWZpY2F0aW9uIGRvZXNuJ3QgaG9sZC4gKi9cbiAgZGlzYWJsZVJlbmRlckNhcD86IGJvb2xlYW5cbiAgLyoqIEluLXRyYW5zY3JpcHQgY3Vyc29yOyBleHBhbmRlZCBvdmVycmlkZXMgdmVyYm9zZSBmb3Igc2VsZWN0ZWQgbWVzc2FnZS4gKi9cbiAgY3Vyc29yPzogTWVzc2FnZUFjdGlvbnNTdGF0ZSB8IG51bGxcbiAgc2V0Q3Vyc29yPzogKGN1cnNvcjogTWVzc2FnZUFjdGlvbnNTdGF0ZSB8IG51bGwpID0+IHZvaWRcbiAgLyoqIFBhc3NlZCB0aHJvdWdoIHRvIFZpcnR1YWxNZXNzYWdlTGlzdCAoaGVpZ2h0Q2FjaGUgb3ducyB2aXNpYmlsaXR5KS4gKi9cbiAgY3Vyc29yTmF2UmVmPzogUmVhY3QuUmVmPE1lc3NhZ2VBY3Rpb25zTmF2PlxuICAvKiogUmVuZGVyIG9ubHkgY29sbGFwc2VkLnNsaWNlKHN0YXJ0LCBlbmQpLiBGb3IgY2h1bmtlZCBoZWFkbGVzcyBleHBvcnRcbiAgICogIChzdHJlYW1SZW5kZXJlZE1lc3NhZ2VzIGluIGV4cG9ydFJlbmRlcmVyLnRzeCk6IHByZXAgcnVucyBvbiB0aGUgRlVMTFxuICAgKiAgbWVzc2FnZXMgYXJyYXkgc28gZ3JvdXBpbmcvbG9va3VwcyBhcmUgY29ycmVjdCwgYnV0IG9ubHkgdGhpcyBzbGljZVxuICAgKiAgY2h1bmsgaW5zdGVhZCBvZiB0aGUgZnVsbCBzZXNzaW9uLiBUaGUgbG9nbyByZW5kZXJzIG9ubHkgZm9yIGNodW5rIDBcbiAgICogIChzdGFydCA9PT0gMCk7IGxhdGVyIGNodW5rcyBhcmUgbWlkLXN0cmVhbSBjb250aW51YXRpb25zLlxuICAgKiAgTWVhc3VyZWQgTWFyIDIwMjY6IDUzOC1tc2cgc2Vzc2lvbiwgMjAgc2xpY2VzIOKGkiDiiJI1NSUgcGxhdGVhdSBSU1MuICovXG4gIHJlbmRlclJhbmdlPzogcmVhZG9ubHkgW3N0YXJ0OiBudW1iZXIsIGVuZDogbnVtYmVyXVxufVxuXG5jb25zdCBNQVhfTUVTU0FHRVNfVE9fU0hPV19JTl9UUkFOU0NSSVBUX01PREUgPSAzMFxuXG4vLyBTYWZldHkgY2FwIGZvciB0aGUgbm9uLXZpcnR1YWxpemVkIHJlbmRlciBwYXRoIChmdWxsc2NyZWVuIG9mZiBvclxuLy8gZXhwbGljaXRseSBkaXNhYmxlZCkuIEluayBtb3VudHMgYSBmdWxsIGZpYmVyIHRyZWUgcGVyIG1lc3NhZ2UgKH4yNTAgS0Jcbi8vIFJTUyBlYWNoKTsgeW9nYSBsYXlvdXQgaGVpZ2h0IGdyb3dzIHVuYm91bmRlZDsgdGhlIHNjcmVlbiBidWZmZXIgaXMgc2l6ZWRcbi8vIHRvIGZpdCBldmVyeSBsaW5lLiBBdCB+MjAwMCBtZXNzYWdlcyB0aGlzIGlzIH4zMDAwLWxpbmUgc2NyZWVucywgfjUwMCBNQlxuLy8gb2YgZmliZXJzLCBhbmQgcGVyLWZyYW1lIHdyaXRlIGNvc3RzIHRoYXQgcHVzaCB0aGUgcHJvY2VzcyBpbnRvIGEgR0Ncbi8vIGRlYXRoIHNwaXJhbCAob2JzZXJ2ZWQ6IDU5IEdCIFJTUywgMTRrIG1tYXAvbXVubWFwL3NlYykuIENvbnRlbnQgZHJvcHBlZFxuLy8gZnJvbSB0aGlzIHNsaWNlIGhhcyBhbHJlYWR5IGJlZW4gcHJpbnRlZCB0byB0ZXJtaW5hbCBzY3JvbGxiYWNrIOKAlCB1c2Vyc1xuLy8gY2FuIHN0aWxsIHNjcm9sbCB1cCBuYXRpdmVseS4gVmlydHVhbE1lc3NhZ2VMaXN0ICh0aGUgZGVmYXVsdCBhbnQgcGF0aClcbi8vIGJ5cGFzc2VzIHRoaXMgY2FwIGVudGlyZWx5LiBIZWFkbGVzcyBvbmUtc2hvdCByZW5kZXJzIChlLmcuIC9leHBvcnQpXG4vLyBwYXNzIGRpc2FibGVSZW5kZXJDYXAgdG8gb3B0IG91dCDigJQgdGhleSBoYXZlIG5vIHNjcm9sbGJhY2sgYW5kIHRoZVxuLy8gbWVtb3J5IGNvbmNlcm4gZG9lc24ndCBhcHBseSB0byByZW5kZXJUb1N0cmluZy5cbi8vXG4vLyBUaGUgc2xpY2UgYm91bmRhcnkgaXMgdHJhY2tlZCBhcyBhIFVVSUQgYW5jaG9yLCBub3QgYSBjb3VudC1kZXJpdmVkXG4vLyBpbmRleC4gQ291bnQtYmFzZWQgc2xpY2luZyAoc2xpY2UoLTIwMCkpIGRyb3BzIG9uZSBtZXNzYWdlIGZyb20gdGhlXG4vLyBmcm9udCBvbiBldmVyeSBhcHBlbmQsIHNoaWZ0aW5nIHNjcm9sbGJhY2sgY29udGVudCBhbmQgZm9yY2luZyBhIGZ1bGxcbi8vIHRlcm1pbmFsIHJlc2V0IHBlciB0dXJuIChDQy05NDEpLiBRdWFudGl6aW5nIHRvIDUwLW1lc3NhZ2Ugc3RlcHNcbi8vIChDQy0xMTU0KSBoZWxwZWQgYnV0IHN0aWxsIHNoaWZ0ZWQgb24gY29tcGFjdGlvbiBhbmQgY29sbGFwc2UgcmVncm91cGluZ1xuLy8gc2luY2UgdGhvc2UgY2hhbmdlIGNvbGxhcHNlZC5sZW5ndGggd2l0aG91dCBhZGRpbmcgbWVzc2FnZXMuIFRoZSBVVUlEXG4vLyBhbmNob3Igb25seSBhZHZhbmNlcyB3aGVuIHJlbmRlcmVkIGNvdW50IGdlbnVpbmVseSBleGNlZWRzIENBUCtTVEVQIOKAlFxuLy8gaW1tdW5lIHRvIGxlbmd0aCBjaHVybiBmcm9tIGdyb3VwaW5nL2NvbXBhY3Rpb24gKENDLTExNzQpLlxuLy9cbi8vIFRoZSBhbmNob3Igc3RvcmVzIEJPVEggdXVpZCBhbmQgaW5kZXguIFNvbWUgdXVpZHMgYXJlIHVuc3RhYmxlIGJldHdlZW5cbi8vIHJlbmRlcnM6IGNvbGxhcHNlSG9va1N1bW1hcmllcyBkZXJpdmVzIHRoZSBtZXJnZWQgdXVpZCBmcm9tIHRoZSBmaXJzdFxuLy8gc3VtbWFyeSBpbiBhIGdyb3VwLCBidXQgcmVvcmRlck1lc3NhZ2VzSW5VSSByZXNodWZmbGVzIGhvb2sgYWRqYWNlbmN5XG4vLyBhcyB0b29sIHJlc3VsdHMgc3RyZWFtIGluLCBjaGFuZ2luZyB3aGljaCBzdW1tYXJ5IGlzIGZpcnN0LiBXaGVuIHRoZVxuLy8gdXVpZCB2YW5pc2hlcywgZmFsbGluZyBiYWNrIHRvIHRoZSBzdG9yZWQgaW5kZXggKGNsYW1wZWQpIGtlZXBzIHRoZVxuLy8gc2xpY2Ugcm91Z2hseSB3aGVyZSBpdCB3YXMgaW5zdGVhZCBvZiByZXNldHRpbmcgdG8gMCDigJQgd2hpY2ggd291bGRcbi8vIGp1bXAgZnJvbSB+MjAwIHJlbmRlcmVkIG1lc3NhZ2VzIHRvIHRoZSBmdWxsIGhpc3RvcnksIG9ycGhhbmluZ1xuLy8gaW4tcHJvZ3Jlc3MgYmFkZ2Ugc25hcHNob3RzIGluIHNjcm9sbGJhY2suXG5jb25zdCBNQVhfTUVTU0FHRVNfV0lUSE9VVF9WSVJUVUFMSVpBVElPTiA9IDIwMFxuY29uc3QgTUVTU0FHRV9DQVBfU1RFUCA9IDUwXG5cbmV4cG9ydCB0eXBlIFNsaWNlQW5jaG9yID0geyB1dWlkOiBzdHJpbmc7IGlkeDogbnVtYmVyIH0gfCBudWxsXG5cbi8qKiBFeHBvcnRlZCBmb3IgdGVzdGluZy4gTXV0YXRlcyBhbmNob3JSZWYgd2hlbiB0aGUgd2luZG93IG5lZWRzIHRvIGFkdmFuY2UuICovXG5leHBvcnQgZnVuY3Rpb24gY29tcHV0ZVNsaWNlU3RhcnQoXG4gIGNvbGxhcHNlZDogUmVhZG9ubHlBcnJheTx7IHV1aWQ6IHN0cmluZyB9PixcbiAgYW5jaG9yUmVmOiB7IGN1cnJlbnQ6IFNsaWNlQW5jaG9yIH0sXG4gIGNhcCA9IE1BWF9NRVNTQUdFU19XSVRIT1VUX1ZJUlRVQUxJWkFUSU9OLFxuICBzdGVwID0gTUVTU0FHRV9DQVBfU1RFUCxcbik6IG51bWJlciB7XG4gIGNvbnN0IGFuY2hvciA9IGFuY2hvclJlZi5jdXJyZW50XG4gIGNvbnN0IGFuY2hvcklkeCA9IGFuY2hvclxuICAgID8gY29sbGFwc2VkLmZpbmRJbmRleChtID0+IG0udXVpZCA9PT0gYW5jaG9yLnV1aWQpXG4gICAgOiAtMVxuICAvLyBBbmNob3IgZm91bmQg4oaSIHVzZSBpdC4gQW5jaG9yIGxvc3Qg4oaSIGZhbGwgYmFjayB0byBzdG9yZWQgaW5kZXhcbiAgLy8gKGNsYW1wZWQpIHNvIGNvbGxhcHNlLXJlZ3JvdXBpbmcgdXVpZCBjaHVybiBkb2Vzbid0IHJlc2V0IHRvIDAuXG4gIGxldCBzdGFydCA9XG4gICAgYW5jaG9ySWR4ID49IDBcbiAgICAgID8gYW5jaG9ySWR4XG4gICAgICA6IGFuY2hvclxuICAgICAgICA/IE1hdGgubWluKGFuY2hvci5pZHgsIE1hdGgubWF4KDAsIGNvbGxhcHNlZC5sZW5ndGggLSBjYXApKVxuICAgICAgICA6IDBcbiAgaWYgKGNvbGxhcHNlZC5sZW5ndGggLSBzdGFydCA+IGNhcCArIHN0ZXApIHtcbiAgICBzdGFydCA9IGNvbGxhcHNlZC5sZW5ndGggLSBjYXBcbiAgfVxuICAvLyBSZWZyZXNoIGFuY2hvciBmcm9tIHdoYXRldmVyIGxpdmVzIGF0IHRoZSBjdXJyZW50IHN0YXJ0IOKAlCBoZWFscyBhXG4gIC8vIHN0YWxlIHV1aWQgYWZ0ZXIgZmFsbGJhY2sgYW5kIGNhcHR1cmVzIGEgbmV3IG9uZSBhZnRlciBhZHZhbmNlbWVudC5cbiAgY29uc3QgbXNnQXRTdGFydCA9IGNvbGxhcHNlZFtzdGFydF1cbiAgaWYgKFxuICAgIG1zZ0F0U3RhcnQgJiZcbiAgICAoYW5jaG9yPy51dWlkICE9PSBtc2dBdFN0YXJ0LnV1aWQgfHwgYW5jaG9yLmlkeCAhPT0gc3RhcnQpXG4gICkge1xuICAgIGFuY2hvclJlZi5jdXJyZW50ID0geyB1dWlkOiBtc2dBdFN0YXJ0LnV1aWQsIGlkeDogc3RhcnQgfVxuICB9IGVsc2UgaWYgKCFtc2dBdFN0YXJ0ICYmIGFuY2hvcikge1xuICAgIGFuY2hvclJlZi5jdXJyZW50ID0gbnVsbFxuICB9XG4gIHJldHVybiBzdGFydFxufVxuXG5jb25zdCBNZXNzYWdlc0ltcGwgPSAoe1xuICBtZXNzYWdlcyxcbiAgdG9vbHMsXG4gIGNvbW1hbmRzLFxuICB2ZXJib3NlLFxuICB0b29sSlNYLFxuICB0b29sVXNlQ29uZmlybVF1ZXVlLFxuICBpblByb2dyZXNzVG9vbFVzZUlEcyxcbiAgaXNNZXNzYWdlU2VsZWN0b3JWaXNpYmxlLFxuICBjb252ZXJzYXRpb25JZCxcbiAgc2NyZWVuLFxuICBzdHJlYW1pbmdUb29sVXNlcyxcbiAgc2hvd0FsbEluVHJhbnNjcmlwdCA9IGZhbHNlLFxuICBhZ2VudERlZmluaXRpb25zLFxuICBvbk9wZW5SYXRlTGltaXRPcHRpb25zLFxuICBoaWRlTG9nbyA9IGZhbHNlLFxuICBpc0xvYWRpbmcsXG4gIGhpZGVQYXN0VGhpbmtpbmcgPSBmYWxzZSxcbiAgc3RyZWFtaW5nVGhpbmtpbmcsXG4gIHN0cmVhbWluZ1RleHQsXG4gIGlzQnJpZWZPbmx5ID0gZmFsc2UsXG4gIHVuc2VlbkRpdmlkZXIsXG4gIHNjcm9sbFJlZixcbiAgdHJhY2tTdGlja3lQcm9tcHQsXG4gIGp1bXBSZWYsXG4gIG9uU2VhcmNoTWF0Y2hlc0NoYW5nZSxcbiAgc2NhbkVsZW1lbnQsXG4gIHNldFBvc2l0aW9ucyxcbiAgZGlzYWJsZVJlbmRlckNhcCA9IGZhbHNlLFxuICBjdXJzb3IgPSBudWxsLFxuICBzZXRDdXJzb3IsXG4gIGN1cnNvck5hdlJlZixcbiAgcmVuZGVyUmFuZ2UsXG59OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSA9PiB7XG4gIGNvbnN0IHsgY29sdW1ucyB9ID0gdXNlVGVybWluYWxTaXplKClcbiAgY29uc3QgdG9nZ2xlU2hvd0FsbFNob3J0Y3V0ID0gdXNlU2hvcnRjdXREaXNwbGF5KFxuICAgICd0cmFuc2NyaXB0OnRvZ2dsZVNob3dBbGwnLFxuICAgICdUcmFuc2NyaXB0JyxcbiAgICAnQ3RybCtFJyxcbiAgKVxuXG4gIGNvbnN0IG5vcm1hbGl6ZWRNZXNzYWdlcyA9IHVzZU1lbW8oXG4gICAgKCkgPT4gbm9ybWFsaXplTWVzc2FnZXMobWVzc2FnZXMpLmZpbHRlcihpc05vdEVtcHR5TWVzc2FnZSksXG4gICAgW21lc3NhZ2VzXSxcbiAgKVxuXG4gIC8vIENoZWNrIGlmIHN0cmVhbWluZyB0aGlua2luZyBzaG91bGQgYmUgdmlzaWJsZSAoc3RyZWFtaW5nIG9yIHdpdGhpbiAzMHMgdGltZW91dClcbiAgY29uc3QgaXNTdHJlYW1pbmdUaGlua2luZ1Zpc2libGUgPSB1c2VNZW1vKCgpID0+IHtcbiAgICBpZiAoIXN0cmVhbWluZ1RoaW5raW5nKSByZXR1cm4gZmFsc2VcbiAgICBpZiAoc3RyZWFtaW5nVGhpbmtpbmcuaXNTdHJlYW1pbmcpIHJldHVybiB0cnVlXG4gICAgaWYgKHN0cmVhbWluZ1RoaW5raW5nLnN0cmVhbWluZ0VuZGVkQXQpIHtcbiAgICAgIHJldHVybiBEYXRlLm5vdygpIC0gc3RyZWFtaW5nVGhpbmtpbmcuc3RyZWFtaW5nRW5kZWRBdCA8IDMwMDAwXG4gICAgfVxuICAgIHJldHVybiBmYWxzZVxuICB9LCBbc3RyZWFtaW5nVGhpbmtpbmddKVxuXG4gIC8vIEZpbmQgdGhlIGxhc3QgdGhpbmtpbmcgYmxvY2sgKG1lc3NhZ2UgVVVJRCArIGNvbnRlbnQgaW5kZXgpIGZvciBoaWRpbmcgcGFzdCB0aGlua2luZyBpbiB0cmFuc2NyaXB0IG1vZGVcbiAgLy8gV2hlbiBzdHJlYW1pbmcgdGhpbmtpbmcgaXMgdmlzaWJsZSwgdXNlIGEgc3BlY2lhbCBJRCB0aGF0IHdvbid0IG1hdGNoIGFueSBjb21wbGV0ZWQgdGhpbmtpbmcgYmxvY2tcbiAgLy8gV2l0aCBhZGFwdGl2ZSB0aGlua2luZywgb25seSBjb25zaWRlciB0aGlua2luZyBibG9ja3MgZnJvbSB0aGUgY3VycmVudCB0dXJuIGFuZCBzdG9wIHNlYXJjaGluZyBvbmNlIHdlXG4gIC8vIGhpdCB0aGUgbGFzdCB1c2VyIG1lc3NhZ2UuXG4gIGNvbnN0IGxhc3RUaGlua2luZ0Jsb2NrSWQgPSB1c2VNZW1vKCgpID0+IHtcbiAgICBpZiAoIWhpZGVQYXN0VGhpbmtpbmcpIHJldHVybiBudWxsXG4gICAgLy8gSWYgc3RyZWFtaW5nIHRoaW5raW5nIGlzIHZpc2libGUsIGhpZGUgYWxsIGNvbXBsZXRlZCB0aGlua2luZyBibG9ja3MgYnkgdXNpbmcgYSBub24tbWF0Y2hpbmcgSURcbiAgICBpZiAoaXNTdHJlYW1pbmdUaGlua2luZ1Zpc2libGUpIHJldHVybiAnc3RyZWFtaW5nJ1xuICAgIC8vIEl0ZXJhdGUgYmFja3dhcmRzIHRvIGZpbmQgdGhlIGxhc3QgbWVzc2FnZSB3aXRoIGEgdGhpbmtpbmcgYmxvY2tcbiAgICBmb3IgKGxldCBpID0gbm9ybWFsaXplZE1lc3NhZ2VzLmxlbmd0aCAtIDE7IGkgPj0gMDsgaS0tKSB7XG4gICAgICBjb25zdCBtc2cgPSBub3JtYWxpemVkTWVzc2FnZXNbaV1cbiAgICAgIGlmIChtc2c/LnR5cGUgPT09ICdhc3Npc3RhbnQnKSB7XG4gICAgICAgIGNvbnN0IGNvbnRlbnQgPSBtc2cubWVzc2FnZS5jb250ZW50XG4gICAgICAgIC8vIEZpbmQgdGhlIGxhc3QgdGhpbmtpbmcgYmxvY2sgaW4gdGhpcyBtZXNzYWdlXG4gICAgICAgIGZvciAobGV0IGogPSBjb250ZW50Lmxlbmd0aCAtIDE7IGogPj0gMDsgai0tKSB7XG4gICAgICAgICAgaWYgKGNvbnRlbnRbal0/LnR5cGUgPT09ICd0aGlua2luZycpIHtcbiAgICAgICAgICAgIHJldHVybiBgJHttc2cudXVpZH06JHtqfWBcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH0gZWxzZSBpZiAobXNnPy50eXBlID09PSAndXNlcicpIHtcbiAgICAgICAgY29uc3QgaGFzVG9vbFJlc3VsdCA9IG1zZy5tZXNzYWdlLmNvbnRlbnQuc29tZShcbiAgICAgICAgICBibG9jayA9PiBibG9jay50eXBlID09PSAndG9vbF9yZXN1bHQnLFxuICAgICAgICApXG4gICAgICAgIGlmICghaGFzVG9vbFJlc3VsdCkge1xuICAgICAgICAgIC8vIFJlYWNoZWQgYSBwcmV2aW91cyB1c2VyIHR1cm4gc28gZG9uJ3Qgc2hvdyBzdGFsZSB0aGlua2luZyBmcm9tIGJlZm9yZVxuICAgICAgICAgIHJldHVybiAnbm8tdGhpbmtpbmcnXG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG4gICAgcmV0dXJuIG51bGxcbiAgfSwgW25vcm1hbGl6ZWRNZXNzYWdlcywgaGlkZVBhc3RUaGlua2luZywgaXNTdHJlYW1pbmdUaGlua2luZ1Zpc2libGVdKVxuXG4gIC8vIEZpbmQgdGhlIGxhdGVzdCB1c2VyIGJhc2ggb3V0cHV0IG1lc3NhZ2UgKGZyb20gISBjb21tYW5kcylcbiAgLy8gVGhpcyBhbGxvd3MgdXMgdG8gc2hvdyBmdWxsIG91dHB1dCBmb3IgdGhlIG1vc3QgcmVjZW50IGJhc2ggY29tbWFuZFxuICBjb25zdCBsYXRlc3RCYXNoT3V0cHV0VVVJRCA9IHVzZU1lbW8oKCkgPT4ge1xuICAgIC8vIEl0ZXJhdGUgYmFja3dhcmRzIHRvIGZpbmQgdGhlIGxhc3QgdXNlciBtZXNzYWdlIHdpdGggYmFzaCBvdXRwdXRcbiAgICBmb3IgKGxldCBpID0gbm9ybWFsaXplZE1lc3NhZ2VzLmxlbmd0aCAtIDE7IGkgPj0gMDsgaS0tKSB7XG4gICAgICBjb25zdCBtc2cgPSBub3JtYWxpemVkTWVzc2FnZXNbaV1cbiAgICAgIGlmIChtc2c/LnR5cGUgPT09ICd1c2VyJykge1xuICAgICAgICBjb25zdCBjb250ZW50ID0gbXNnLm1lc3NhZ2UuY29udGVudFxuICAgICAgICAvLyBDaGVjayBpZiBhbnkgdGV4dCBjb250ZW50IGlzIGJhc2ggb3V0cHV0XG4gICAgICAgIGZvciAoY29uc3QgYmxvY2sgb2YgY29udGVudCkge1xuICAgICAgICAgIGlmIChibG9jay50eXBlID09PSAndGV4dCcpIHtcbiAgICAgICAgICAgIGNvbnN0IHRleHQgPSBibG9jay50ZXh0XG4gICAgICAgICAgICBpZiAoXG4gICAgICAgICAgICAgIHRleHQuc3RhcnRzV2l0aCgnPGJhc2gtc3Rkb3V0JykgfHxcbiAgICAgICAgICAgICAgdGV4dC5zdGFydHNXaXRoKCc8YmFzaC1zdGRlcnInKVxuICAgICAgICAgICAgKSB7XG4gICAgICAgICAgICAgIHJldHVybiBtc2cudXVpZFxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgICByZXR1cm4gbnVsbFxuICB9LCBbbm9ybWFsaXplZE1lc3NhZ2VzXSlcblxuICAvLyBzdHJlYW1pbmdUb29sVXNlcyB1cGRhdGVzIG9uIGV2ZXJ5IGlucHV0X2pzb25fZGVsdGEgd2hpbGUgbm9ybWFsaXplZE1lc3NhZ2VzXG4gIC8vIHN0YXlzIHN0YWJsZSDigJQgcHJlY29tcHV0ZSB0aGUgU2V0IHNvIHRoZSBmaWx0ZXIgaXMgTyhrKSBub3QgTyhuw5drKSBwZXIgY2h1bmsuXG4gIGNvbnN0IG5vcm1hbGl6ZWRUb29sVXNlSURzID0gdXNlTWVtbyhcbiAgICAoKSA9PiBnZXRUb29sVXNlSURzKG5vcm1hbGl6ZWRNZXNzYWdlcyksXG4gICAgW25vcm1hbGl6ZWRNZXNzYWdlc10sXG4gIClcblxuICBjb25zdCBzdHJlYW1pbmdUb29sVXNlc1dpdGhvdXRJblByb2dyZXNzID0gdXNlTWVtbyhcbiAgICAoKSA9PlxuICAgICAgc3RyZWFtaW5nVG9vbFVzZXMuZmlsdGVyKFxuICAgICAgICBzdHUgPT5cbiAgICAgICAgICAhaW5Qcm9ncmVzc1Rvb2xVc2VJRHMuaGFzKHN0dS5jb250ZW50QmxvY2suaWQpICYmXG4gICAgICAgICAgIW5vcm1hbGl6ZWRUb29sVXNlSURzLmhhcyhzdHUuY29udGVudEJsb2NrLmlkKSxcbiAgICAgICksXG4gICAgW3N0cmVhbWluZ1Rvb2xVc2VzLCBpblByb2dyZXNzVG9vbFVzZUlEcywgbm9ybWFsaXplZFRvb2xVc2VJRHNdLFxuICApXG5cbiAgY29uc3Qgc3ludGhldGljU3RyZWFtaW5nVG9vbFVzZU1lc3NhZ2VzID0gdXNlTWVtbyhcbiAgICAoKSA9PlxuICAgICAgc3RyZWFtaW5nVG9vbFVzZXNXaXRob3V0SW5Qcm9ncmVzcy5mbGF0TWFwKHN0cmVhbWluZ1Rvb2xVc2UgPT4ge1xuICAgICAgICBjb25zdCBtc2cgPSBjcmVhdGVBc3Npc3RhbnRNZXNzYWdlKHtcbiAgICAgICAgICBjb250ZW50OiBbc3RyZWFtaW5nVG9vbFVzZS5jb250ZW50QmxvY2tdLFxuICAgICAgICB9KVxuICAgICAgICAvLyBPdmVycmlkZSByYW5kb21VVUlEIHdpdGggZGV0ZXJtaW5pc3RpYyB2YWx1ZSBkZXJpdmVkIGZyb20gY29udGVudFxuICAgICAgICAvLyBibG9jayBJRCB0byBwcmV2ZW50IFJlYWN0IGtleSBjaGFuZ2VzIG9uIGV2ZXJ5IG1lbW8gcmVjb21wdXRhdGlvbi5cbiAgICAgICAgLy8gU2FtZSBjbGFzcyBvZiBidWcgZml4ZWQgaW4gbm9ybWFsaXplTWVzc2FnZXMgKGNvbW1pdCAzODMzMjZlNjEzKTpcbiAgICAgICAgLy8gZnJlc2ggcmFuZG9tVVVJRCDihpIgdW5zdGFibGUgUmVhY3Qga2V5cyDihpIgY29tcG9uZW50IHJlbW91bnRzIOKGklxuICAgICAgICAvLyBJbmsgcmVuZGVyaW5nIGNvcnJ1cHRpb24gKG92ZXJsYXBwaW5nIHRleHQgZnJvbSBzdGFsZSBET00gbm9kZXMpLlxuICAgICAgICBtc2cudXVpZCA9IGRlcml2ZVVVSUQoc3RyZWFtaW5nVG9vbFVzZS5jb250ZW50QmxvY2suaWQgYXMgVVVJRCwgMClcbiAgICAgICAgcmV0dXJuIG5vcm1hbGl6ZU1lc3NhZ2VzKFttc2ddKVxuICAgICAgfSksXG4gICAgW3N0cmVhbWluZ1Rvb2xVc2VzV2l0aG91dEluUHJvZ3Jlc3NdLFxuICApXG5cbiAgY29uc3QgaXNUcmFuc2NyaXB0TW9kZSA9IHNjcmVlbiA9PT0gJ3RyYW5zY3JpcHQnXG4gIC8vIEhvaXN0ZWQgdG8gbW91bnQtdGltZSDigJQgdGhpcyBjb21wb25lbnQgcmUtcmVuZGVycyBvbiBldmVyeSBzY3JvbGwuXG4gIGNvbnN0IGRpc2FibGVWaXJ0dWFsU2Nyb2xsID0gdXNlTWVtbyhcbiAgICAoKSA9PiBpc0VudlRydXRoeShwcm9jZXNzLmVudi5DTEFVREVfQ09ERV9ESVNBQkxFX1ZJUlRVQUxfU0NST0xMKSxcbiAgICBbXSxcbiAgKVxuICAvLyBWaXJ0dWFsIHNjcm9sbCByZXBsYWNlcyB0aGUgdHJhbnNjcmlwdCBjYXA6IGV2ZXJ5dGhpbmcgaXMgc2Nyb2xsYWJsZSBhbmRcbiAgLy8gbWVtb3J5IGlzIGJvdW5kZWQgYnkgdGhlIG1vdW50ZWQtaXRlbSBjb3VudCwgbm90IHRoZSB0b3RhbC4gc2Nyb2xsUmVmIGlzXG4gIC8vIG9ubHkgcGFzc2VkIHdoZW4gaXNGdWxsc2NyZWVuRW52RW5hYmxlZCgpIGlzIHRydWUgKFJFUEwudHN4IGdhdGVzIGl0KSxcbiAgLy8gc28gc2Nyb2xsUmVmJ3MgcHJlc2VuY2UgaXMgdGhlIHNpZ25hbC5cbiAgY29uc3QgdmlydHVhbFNjcm9sbFJ1bnRpbWVHYXRlID0gc2Nyb2xsUmVmICE9IG51bGwgJiYgIWRpc2FibGVWaXJ0dWFsU2Nyb2xsXG4gIGNvbnN0IHNob3VsZFRydW5jYXRlID1cbiAgICBpc1RyYW5zY3JpcHRNb2RlICYmICFzaG93QWxsSW5UcmFuc2NyaXB0ICYmICF2aXJ0dWFsU2Nyb2xsUnVudGltZUdhdGVcblxuICAvLyBBbmNob3IgZm9yIHRoZSBmaXJzdCByZW5kZXJlZCBtZXNzYWdlIGluIHRoZSBub24tdmlydHVhbGl6ZWQgY2FwIHNsaWNlLlxuICAvLyBNb25vdG9uaWMgYWR2YW5jZSBvbmx5IOKAlCBtdXRhdGlvbiBkdXJpbmcgcmVuZGVyIGlzIGlkZW1wb3RlbnQgKHNhZmVcbiAgLy8gdW5kZXIgU3RyaWN0TW9kZSBkb3VibGUtcmVuZGVyKS4gU2VlIE1BWF9NRVNTQUdFU19XSVRIT1VUX1ZJUlRVQUxJWkFUSU9OXG4gIC8vIGNvbW1lbnQgYWJvdmUgZm9yIHdoeSB0aGlzIHJlcGxhY2VkIGNvdW50LWJhc2VkIHNsaWNpbmcuXG4gIGNvbnN0IHNsaWNlQW5jaG9yUmVmID0gdXNlUmVmPFNsaWNlQW5jaG9yPihudWxsKVxuXG4gIC8vIEV4cGVuc2l2ZSBtZXNzYWdlIHRyYW5zZm9ybXMg4oCUIGZpbHRlciwgcmVvcmRlciwgZ3JvdXAsIGNvbGxhcHNlLCBsb29rdXBzLlxuICAvLyBBbGwgTyhuKSBvdmVyIDI3ayBtZXNzYWdlcy4gU3BsaXQgZnJvbSB0aGUgcmVuZGVyUmFuZ2Ugc2xpY2Ugc28gc2Nyb2xsaW5nXG4gIC8vICh3aGljaCBvbmx5IGNoYW5nZXMgcmVuZGVyUmFuZ2UpIGRvZXNuJ3QgcmUtcnVuIHRoZXNlLiBQcmV2aW91c2x5IHRoaXNcbiAgLy8gdXNlTWVtbyBpbmNsdWRlZCByZW5kZXJSYW5nZSDihpIgZXZlcnkgc2Nyb2xsIHJlYnVpbHQgNiBNYXBzIG92ZXIgMjdrXG4gIC8vIG1lc3NhZ2VzICsgNCBmaWx0ZXIvbWFwIHBhc3NlcyA9IH41MG1zIGFsbG9jIHBlciBzY3JvbGwg4oaSIEdDIHByZXNzdXJlIOKGklxuICAvLyAxMDAtMTczbXMgc3RvcC10aGUtd29ybGQgcGF1c2VzIG9uIHRoZSAxR0IgaGVhcC5cbiAgY29uc3QgeyBjb2xsYXBzZWQsIGxvb2t1cHMsIGhhc1RydW5jYXRlZE1lc3NhZ2VzLCBoaWRkZW5NZXNzYWdlQ291bnQgfSA9XG4gICAgdXNlTWVtbygoKSA9PiB7XG4gICAgICAvLyBJbiBmdWxsc2NyZWVuIG1vZGUgdGhlIGFsdCBidWZmZXIgaGFzIG5vIG5hdGl2ZSBzY3JvbGxiYWNrLCBzbyB0aGVcbiAgICAgIC8vIGNvbXBhY3QtYm91bmRhcnkgZmlsdGVyIGp1c3QgaGlkZXMgaGlzdG9yeSB0aGUgU2Nyb2xsQm94IGNvdWxkXG4gICAgICAvLyBvdGhlcndpc2Ugc2Nyb2xsIHRvLiBNYWluLXNjcmVlbiBtb2RlIGtlZXBzIHRoZSBmaWx0ZXIg4oCUIHByZS1jb21wYWN0XG4gICAgICAvLyByb3dzIGxpdmUgYWJvdmUgdGhlIHZpZXdwb3J0IGluIG5hdGl2ZSBzY3JvbGxiYWNrIHRoZXJlLCBhbmRcbiAgICAgIC8vIHJlLXJlbmRlcmluZyB0aGVtIHRyaWdnZXJzIGZ1bGwgcmVzZXRzLlxuICAgICAgLy8gaW5jbHVkZVNuaXBwZWQ6IFVJIHJlbmRlcmluZyBrZWVwcyBzbmlwcGVkIG1lc3NhZ2VzIGZvciBzY3JvbGxiYWNrXG4gICAgICAvLyAodGhpcyBQUidzIGNvcmUgZ29hbCDigJQgZnVsbCBoaXN0b3J5IGluIFVJLCBmaWx0ZXIgb25seSBmb3IgdGhlIG1vZGVsKS5cbiAgICAgIC8vIEFsc28gYXZvaWRzIGEgVVVJRCBtaXNtYXRjaDogbm9ybWFsaXplTWVzc2FnZXMgZGVyaXZlcyBuZXcgVVVJRHMsIHNvXG4gICAgICAvLyBwcm9qZWN0U25pcHBlZFZpZXcncyBjaGVjayBhZ2FpbnN0IG9yaWdpbmFsIHJlbW92ZWRVdWlkcyB3b3VsZCBmYWlsLlxuICAgICAgY29uc3QgY29tcGFjdEF3YXJlTWVzc2FnZXMgPVxuICAgICAgICB2ZXJib3NlIHx8IGlzRnVsbHNjcmVlbkVudkVuYWJsZWQoKVxuICAgICAgICAgID8gbm9ybWFsaXplZE1lc3NhZ2VzXG4gICAgICAgICAgOiBnZXRNZXNzYWdlc0FmdGVyQ29tcGFjdEJvdW5kYXJ5KG5vcm1hbGl6ZWRNZXNzYWdlcywge1xuICAgICAgICAgICAgICBpbmNsdWRlU25pcHBlZDogdHJ1ZSxcbiAgICAgICAgICAgIH0pXG5cbiAgICAgIGNvbnN0IG1lc3NhZ2VzVG9TaG93Tm90VHJ1bmNhdGVkID0gcmVvcmRlck1lc3NhZ2VzSW5VSShcbiAgICAgICAgY29tcGFjdEF3YXJlTWVzc2FnZXNcbiAgICAgICAgICAuZmlsdGVyKFxuICAgICAgICAgICAgKG1zZyk6IG1zZyBpcyBFeGNsdWRlPE5vcm1hbGl6ZWRNZXNzYWdlLCBQcm9ncmVzc01lc3NhZ2VUeXBlPiA9PlxuICAgICAgICAgICAgICBtc2cudHlwZSAhPT0gJ3Byb2dyZXNzJyxcbiAgICAgICAgICApXG4gICAgICAgICAgLy8gQ0MtNzI0OiBkcm9wIGF0dGFjaG1lbnQgbWVzc2FnZXMgdGhhdCBBdHRhY2htZW50TWVzc2FnZSByZW5kZXJzIGFzXG4gICAgICAgICAgLy8gbnVsbCAoaG9va19zdWNjZXNzLCBob29rX2FkZGl0aW9uYWxfY29udGV4dCwgaG9va19jYW5jZWxsZWQsIGV0Yy4pXG4gICAgICAgICAgLy8gQkVGT1JFIGNvdW50aW5nL3NsaWNpbmcgc28gdGhleSBkb24ndCBpbmZsYXRlIHRoZSBcIk4gbWVzc2FnZXNcIlxuICAgICAgICAgIC8vIGNvdW50IGluIGN0cmwtbyBvciBjb25zdW1lIHNsb3RzIGluIHRoZSAyMDAtbWVzc2FnZSByZW5kZXIgY2FwLlxuICAgICAgICAgIC5maWx0ZXIobXNnID0+ICFpc051bGxSZW5kZXJpbmdBdHRhY2htZW50KG1zZykpXG4gICAgICAgICAgLmZpbHRlcihfID0+IHNob3VsZFNob3dVc2VyTWVzc2FnZShfLCBpc1RyYW5zY3JpcHRNb2RlKSksXG4gICAgICAgIHN5bnRoZXRpY1N0cmVhbWluZ1Rvb2xVc2VNZXNzYWdlcyxcbiAgICAgIClcbiAgICAgIC8vIFRocmVlLXRpZXIgZmlsdGVyaW5nLiBUcmFuc2NyaXB0IG1vZGUgKGN0cmwrbyBzY3JlZW4pIGlzIHRydWx5IHVuZmlsdGVyZWQuXG4gICAgICAvLyBCcmllZi1vbmx5OiBTZW5kVXNlck1lc3NhZ2UgKyB1c2VyIGlucHV0IG9ubHkuIERlZmF1bHQ6IGRyb3AgcmVkdW5kYW50XG4gICAgICAvLyBhc3Npc3RhbnQgdGV4dCBpbiB0dXJucyB3aGVyZSBTZW5kVXNlck1lc3NhZ2Ugd2FzIGNhbGxlZCAodGhlIG1vZGVsJ3NcbiAgICAgIC8vIHRleHQgaXMgd29ya2luZy1ub3RlcyB0aGF0IGR1cGxpY2F0ZSB0aGUgU2VuZFVzZXJNZXNzYWdlIGNvbnRlbnQpLlxuICAgICAgY29uc3QgYnJpZWZUb29sTmFtZXMgPSBbQlJJRUZfVE9PTF9OQU1FLCBTRU5EX1VTRVJfRklMRV9UT09MX05BTUVdLmZpbHRlcihcbiAgICAgICAgKG4pOiBuIGlzIHN0cmluZyA9PiBuICE9PSBudWxsLFxuICAgICAgKVxuICAgICAgLy8gZHJvcFRleHRJbkJyaWVmVHVybnMgc2hvdWxkIG9ubHkgdHJpZ2dlciBvbiBTZW5kVXNlck1lc3NhZ2UgdHVybnMg4oCUXG4gICAgICAvLyBTZW5kVXNlckZpbGUgZGVsaXZlcnMgYSBmaWxlIHdpdGhvdXQgcmVwbGFjZW1lbnQgdGV4dCwgc28gZHJvcHBpbmdcbiAgICAgIC8vIGFzc2lzdGFudCB0ZXh0IGZvciBmaWxlLW9ubHkgdHVybnMgd291bGQgbGVhdmUgdGhlIHVzZXIgd2l0aCBubyBjb250ZXh0LlxuICAgICAgY29uc3QgZHJvcFRleHRUb29sTmFtZXMgPSBbQlJJRUZfVE9PTF9OQU1FXS5maWx0ZXIoXG4gICAgICAgIChuKTogbiBpcyBzdHJpbmcgPT4gbiAhPT0gbnVsbCxcbiAgICAgIClcbiAgICAgIGNvbnN0IGJyaWVmRmlsdGVyZWQgPVxuICAgICAgICBicmllZlRvb2xOYW1lcy5sZW5ndGggPiAwICYmICFpc1RyYW5zY3JpcHRNb2RlXG4gICAgICAgICAgPyBpc0JyaWVmT25seVxuICAgICAgICAgICAgPyBmaWx0ZXJGb3JCcmllZlRvb2wobWVzc2FnZXNUb1Nob3dOb3RUcnVuY2F0ZWQsIGJyaWVmVG9vbE5hbWVzKVxuICAgICAgICAgICAgOiBkcm9wVGV4dFRvb2xOYW1lcy5sZW5ndGggPiAwXG4gICAgICAgICAgICAgID8gZHJvcFRleHRJbkJyaWVmVHVybnMoXG4gICAgICAgICAgICAgICAgICBtZXNzYWdlc1RvU2hvd05vdFRydW5jYXRlZCxcbiAgICAgICAgICAgICAgICAgIGRyb3BUZXh0VG9vbE5hbWVzLFxuICAgICAgICAgICAgICAgIClcbiAgICAgICAgICAgICAgOiBtZXNzYWdlc1RvU2hvd05vdFRydW5jYXRlZFxuICAgICAgICAgIDogbWVzc2FnZXNUb1Nob3dOb3RUcnVuY2F0ZWRcblxuICAgICAgY29uc3QgbWVzc2FnZXNUb1Nob3cgPSBzaG91bGRUcnVuY2F0ZVxuICAgICAgICA/IGJyaWVmRmlsdGVyZWQuc2xpY2UoLU1BWF9NRVNTQUdFU19UT19TSE9XX0lOX1RSQU5TQ1JJUFRfTU9ERSlcbiAgICAgICAgOiBicmllZkZpbHRlcmVkXG5cbiAgICAgIGNvbnN0IGhhc1RydW5jYXRlZE1lc3NhZ2VzID1cbiAgICAgICAgc2hvdWxkVHJ1bmNhdGUgJiZcbiAgICAgICAgYnJpZWZGaWx0ZXJlZC5sZW5ndGggPiBNQVhfTUVTU0FHRVNfVE9fU0hPV19JTl9UUkFOU0NSSVBUX01PREVcblxuICAgICAgY29uc3QgeyBtZXNzYWdlczogZ3JvdXBlZE1lc3NhZ2VzIH0gPSBhcHBseUdyb3VwaW5nKFxuICAgICAgICBtZXNzYWdlc1RvU2hvdyxcbiAgICAgICAgdG9vbHMsXG4gICAgICAgIHZlcmJvc2UsXG4gICAgICApXG5cbiAgICAgIGNvbnN0IGNvbGxhcHNlZCA9IGNvbGxhcHNlQmFja2dyb3VuZEJhc2hOb3RpZmljYXRpb25zKFxuICAgICAgICBjb2xsYXBzZUhvb2tTdW1tYXJpZXMoXG4gICAgICAgICAgY29sbGFwc2VUZWFtbWF0ZVNodXRkb3ducyhcbiAgICAgICAgICAgIGNvbGxhcHNlUmVhZFNlYXJjaEdyb3Vwcyhncm91cGVkTWVzc2FnZXMsIHRvb2xzKSxcbiAgICAgICAgICApLFxuICAgICAgICApLFxuICAgICAgICB2ZXJib3NlLFxuICAgICAgKVxuXG4gICAgICBjb25zdCBsb29rdXBzID0gYnVpbGRNZXNzYWdlTG9va3Vwcyhub3JtYWxpemVkTWVzc2FnZXMsIG1lc3NhZ2VzVG9TaG93KVxuXG4gICAgICBjb25zdCBoaWRkZW5NZXNzYWdlQ291bnQgPVxuICAgICAgICBtZXNzYWdlc1RvU2hvd05vdFRydW5jYXRlZC5sZW5ndGggLVxuICAgICAgICBNQVhfTUVTU0FHRVNfVE9fU0hPV19JTl9UUkFOU0NSSVBUX01PREVcblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgY29sbGFwc2VkLFxuICAgICAgICBsb29rdXBzLFxuICAgICAgICBoYXNUcnVuY2F0ZWRNZXNzYWdlcyxcbiAgICAgICAgaGlkZGVuTWVzc2FnZUNvdW50LFxuICAgICAgfVxuICAgIH0sIFtcbiAgICAgIHZlcmJvc2UsXG4gICAgICBub3JtYWxpemVkTWVzc2FnZXMsXG4gICAgICBpc1RyYW5zY3JpcHRNb2RlLFxuICAgICAgc3ludGhldGljU3RyZWFtaW5nVG9vbFVzZU1lc3NhZ2VzLFxuICAgICAgc2hvdWxkVHJ1bmNhdGUsXG4gICAgICB0b29scyxcbiAgICAgIGlzQnJpZWZPbmx5LFxuICAgIF0pXG5cbiAgLy8gQ2hlYXAgc2xpY2Ug4oCUIG9ubHkgcnVucyB3aGVuIHNjcm9sbCByYW5nZSBvciBzbGljZSBjb25maWcgY2hhbmdlcy5cbiAgY29uc3QgcmVuZGVyYWJsZU1lc3NhZ2VzID0gdXNlTWVtbygoKSA9PiB7XG4gICAgLy8gU2FmZXR5IGNhcCBmb3IgdGhlIG5vbi12aXJ0dWFsaXplZCByZW5kZXIgcGF0aC4gQXBwbGllZCBoZXJlIChub3QgYXRcbiAgICAvLyB0aGUgSlNYIHNpdGUpIHNvIHJlbmRlck1lc3NhZ2VSb3cncyBpbmRleC1iYXNlZCBsb29rdXBzIGFuZFxuICAgIC8vIGRpdmlkZXJCZWZvcmVJbmRleCBjb21wdXRlIG9uIHRoZSBzYW1lIGFycmF5LiBWaXJ0dWFsTWVzc2FnZUxpc3RcbiAgICAvLyBuZXZlciBzZWVzIHRoaXMgc2xpY2Ug4oCUIHZpcnR1YWxTY3JvbGxSdW50aW1lR2F0ZSBpcyBjb25zdGFudCBmb3IgdGhlXG4gICAgLy8gY29tcG9uZW50J3MgbGlmZXRpbWUgKHNjcm9sbFJlZiBpcyBlaXRoZXIgYWx3YXlzIHBhc3NlZCBvciBuZXZlcikuXG4gICAgLy8gcmVuZGVyUmFuZ2UgaXMgZmlyc3Q6IHRoZSBjaHVua2VkIGV4cG9ydCBwYXRoIHNsaWNlcyB0aGVcbiAgICAvLyBwb3N0LWdyb3VwaW5nIGFycmF5IHNvIGVhY2ggY2h1bmsgZ2V0cyBjb3JyZWN0IHRvb2wtY2FsbCBncm91cGluZy5cbiAgICBjb25zdCBjYXBBcHBsaWVzID0gIXZpcnR1YWxTY3JvbGxSdW50aW1lR2F0ZSAmJiAhZGlzYWJsZVJlbmRlckNhcFxuICAgIGNvbnN0IHNsaWNlU3RhcnQgPSBjYXBBcHBsaWVzXG4gICAgICA/IGNvbXB1dGVTbGljZVN0YXJ0KGNvbGxhcHNlZCwgc2xpY2VBbmNob3JSZWYpXG4gICAgICA6IDBcbiAgICByZXR1cm4gcmVuZGVyUmFuZ2VcbiAgICAgID8gY29sbGFwc2VkLnNsaWNlKHJlbmRlclJhbmdlWzBdLCByZW5kZXJSYW5nZVsxXSlcbiAgICAgIDogc2xpY2VTdGFydCA+IDBcbiAgICAgICAgPyBjb2xsYXBzZWQuc2xpY2Uoc2xpY2VTdGFydClcbiAgICAgICAgOiBjb2xsYXBzZWRcbiAgfSwgW2NvbGxhcHNlZCwgcmVuZGVyUmFuZ2UsIHZpcnR1YWxTY3JvbGxSdW50aW1lR2F0ZSwgZGlzYWJsZVJlbmRlckNhcF0pXG5cbiAgY29uc3Qgc3RyZWFtaW5nVG9vbFVzZUlEcyA9IHVzZU1lbW8oXG4gICAgKCkgPT4gbmV3IFNldChzdHJlYW1pbmdUb29sVXNlcy5tYXAoXyA9PiBfLmNvbnRlbnRCbG9jay5pZCkpLFxuICAgIFtzdHJlYW1pbmdUb29sVXNlc10sXG4gIClcblxuICAvLyBEaXZpZGVyIGluc2VydGlvbiBwb2ludDogZmlyc3QgcmVuZGVyYWJsZU1lc3NhZ2Ugd2hvc2UgdXVpZCBzaGFyZXMgdGhlXG4gIC8vIDI0LWNoYXIgcHJlZml4IHdpdGggZmlyc3RVbnNlZW5VdWlkIChkZXJpdmVVVUlEIGtlZXBzIHRoZSBmaXJzdCAyNFxuICAvLyBjaGFycyBvZiB0aGUgc291cmNlIG1lc3NhZ2UgdXVpZCwgc28gdGhpcyBtYXRjaGVzIGFueSBibG9jayBmcm9tIGl0KS5cbiAgY29uc3QgZGl2aWRlckJlZm9yZUluZGV4ID0gdXNlTWVtbygoKSA9PiB7XG4gICAgaWYgKCF1bnNlZW5EaXZpZGVyKSByZXR1cm4gLTFcbiAgICBjb25zdCBwcmVmaXggPSB1bnNlZW5EaXZpZGVyLmZpcnN0VW5zZWVuVXVpZC5zbGljZSgwLCAyNClcbiAgICByZXR1cm4gcmVuZGVyYWJsZU1lc3NhZ2VzLmZpbmRJbmRleChtID0+IG0udXVpZC5zbGljZSgwLCAyNCkgPT09IHByZWZpeClcbiAgfSwgW3Vuc2VlbkRpdmlkZXIsIHJlbmRlcmFibGVNZXNzYWdlc10pXG5cbiAgY29uc3Qgc2VsZWN0ZWRJZHggPSB1c2VNZW1vKCgpID0+IHtcbiAgICBpZiAoIWN1cnNvcikgcmV0dXJuIC0xXG4gICAgcmV0dXJuIHJlbmRlcmFibGVNZXNzYWdlcy5maW5kSW5kZXgobSA9PiBtLnV1aWQgPT09IGN1cnNvci51dWlkKVxuICB9LCBbY3Vyc29yLCByZW5kZXJhYmxlTWVzc2FnZXNdKVxuXG4gIC8vIEZ1bGxzY3JlZW46IGNsaWNrIGEgbWVzc2FnZSB0byB0b2dnbGUgdmVyYm9zZSByZW5kZXJpbmcgZm9yIGl0LiBLZXllZCBieVxuICAvLyB0b29sX3VzZV9pZCB3aGVyZSBhdmFpbGFibGUgc28gYSB0b29sX3VzZSBhbmQgaXRzIHRvb2xfcmVzdWx0IChzZXBhcmF0ZVxuICAvLyByb3dzKSBleHBhbmQgdG9nZXRoZXI7IGZhbGxzIGJhY2sgdG8gdXVpZCBmb3IgZ3JvdXBzL3RoaW5raW5nLiBTdGFsZSBrZXlzXG4gIC8vIGFyZSBoYXJtbGVzcyDigJQgdGhleSBuZXZlciBtYXRjaCBhbnl0aGluZyBpbiByZW5kZXJhYmxlTWVzc2FnZXMuXG4gIGNvbnN0IFtleHBhbmRlZEtleXMsIHNldEV4cGFuZGVkS2V5c10gPSB1c2VTdGF0ZTxSZWFkb25seVNldDxzdHJpbmc+PihcbiAgICAoKSA9PiBuZXcgU2V0KCksXG4gIClcbiAgY29uc3Qgb25JdGVtQ2xpY2sgPSB1c2VDYWxsYmFjaygobXNnOiBSZW5kZXJhYmxlTWVzc2FnZSkgPT4ge1xuICAgIGNvbnN0IGsgPSBleHBhbmRLZXkobXNnKVxuICAgIHNldEV4cGFuZGVkS2V5cyhwcmV2ID0+IHtcbiAgICAgIGNvbnN0IG5leHQgPSBuZXcgU2V0KHByZXYpXG4gICAgICBpZiAobmV4dC5oYXMoaykpIG5leHQuZGVsZXRlKGspXG4gICAgICBlbHNlIG5leHQuYWRkKGspXG4gICAgICByZXR1cm4gbmV4dFxuICAgIH0pXG4gIH0sIFtdKVxuICBjb25zdCBpc0l0ZW1FeHBhbmRlZCA9IHVzZUNhbGxiYWNrKFxuICAgIChtc2c6IFJlbmRlcmFibGVNZXNzYWdlKSA9PlxuICAgICAgZXhwYW5kZWRLZXlzLnNpemUgPiAwICYmIGV4cGFuZGVkS2V5cy5oYXMoZXhwYW5kS2V5KG1zZykpLFxuICAgIFtleHBhbmRlZEtleXNdLFxuICApXG4gIC8vIE9ubHkgaG92ZXIvY2xpY2sgbWVzc2FnZXMgd2hlcmUgdGhlIHZlcmJvc2UgdG9nZ2xlIHJldmVhbHMgbW9yZTpcbiAgLy8gY29sbGFwc2VkIHJlYWQvc2VhcmNoIGdyb3Vwcywgb3IgdG9vbCByZXN1bHRzIHRoYXQgc2VsZi1yZXBvcnQgdHJ1bmNhdGlvblxuICAvLyB2aWEgaXNSZXN1bHRUcnVuY2F0ZWQuIENhbGxiYWNrIG11c3QgYmUgc3RhYmxlIGFjcm9zcyBtZXNzYWdlIHVwZGF0ZXM6IGlmXG4gIC8vIGl0cyBpZGVudGl0eSAob3IgcmV0dXJuIHZhbHVlKSBmbGlwcyBkdXJpbmcgc3RyZWFtaW5nLCBvbk1vdXNlRW50ZXJcbiAgLy8gYXR0YWNoZXMgYWZ0ZXIgdGhlIG1vdXNlIGlzIGFscmVhZHkgaW5zaWRlIOKGkiBob3ZlciBuZXZlciBmaXJlcy4gdG9vbHMgaXNcbiAgLy8gc2Vzc2lvbi1zdGFibGU7IGxvb2t1cHMgaXMgcmVhZCB2aWEgcmVmIHNvIHRoZSBjYWxsYmFjayBkb2Vzbid0IGNodXJuIG9uXG4gIC8vIGV2ZXJ5IG5ldyBtZXNzYWdlLlxuICBjb25zdCBsb29rdXBzUmVmID0gdXNlUmVmKGxvb2t1cHMpXG4gIGxvb2t1cHNSZWYuY3VycmVudCA9IGxvb2t1cHNcbiAgY29uc3QgaXNJdGVtQ2xpY2thYmxlID0gdXNlQ2FsbGJhY2soXG4gICAgKG1zZzogUmVuZGVyYWJsZU1lc3NhZ2UpOiBib29sZWFuID0+IHtcbiAgICAgIGlmIChtc2cudHlwZSA9PT0gJ2NvbGxhcHNlZF9yZWFkX3NlYXJjaCcpIHJldHVybiB0cnVlXG4gICAgICBpZiAobXNnLnR5cGUgPT09ICdhc3Npc3RhbnQnKSB7XG4gICAgICAgIGNvbnN0IGIgPSBtc2cubWVzc2FnZS5jb250ZW50WzBdIGFzIHVua25vd24gYXMgQWR2aXNvckJsb2NrIHwgdW5kZWZpbmVkXG4gICAgICAgIHJldHVybiAoXG4gICAgICAgICAgYiAhPSBudWxsICYmXG4gICAgICAgICAgaXNBZHZpc29yQmxvY2soYikgJiZcbiAgICAgICAgICBiLnR5cGUgPT09ICdhZHZpc29yX3Rvb2xfcmVzdWx0JyAmJlxuICAgICAgICAgIGIuY29udGVudC50eXBlID09PSAnYWR2aXNvcl9yZXN1bHQnXG4gICAgICAgIClcbiAgICAgIH1cbiAgICAgIGlmIChtc2cudHlwZSAhPT0gJ3VzZXInKSByZXR1cm4gZmFsc2VcbiAgICAgIGNvbnN0IGIgPSBtc2cubWVzc2FnZS5jb250ZW50WzBdXG4gICAgICBpZiAoYj8udHlwZSAhPT0gJ3Rvb2xfcmVzdWx0JyB8fCBiLmlzX2Vycm9yIHx8ICFtc2cudG9vbFVzZVJlc3VsdClcbiAgICAgICAgcmV0dXJuIGZhbHNlXG4gICAgICBjb25zdCBuYW1lID0gbG9va3Vwc1JlZi5jdXJyZW50LnRvb2xVc2VCeVRvb2xVc2VJRC5nZXQoXG4gICAgICAgIGIudG9vbF91c2VfaWQsXG4gICAgICApPy5uYW1lXG4gICAgICBjb25zdCB0b29sID0gbmFtZSA/IGZpbmRUb29sQnlOYW1lKHRvb2xzLCBuYW1lKSA6IHVuZGVmaW5lZFxuICAgICAgcmV0dXJuIHRvb2w/LmlzUmVzdWx0VHJ1bmNhdGVkPy4obXNnLnRvb2xVc2VSZXN1bHQgYXMgbmV2ZXIpID8/IGZhbHNlXG4gICAgfSxcbiAgICBbdG9vbHNdLFxuICApXG5cbiAgY29uc3QgY2FuQW5pbWF0ZSA9XG4gICAgKCF0b29sSlNYIHx8ICEhdG9vbEpTWC5zaG91bGRDb250aW51ZUFuaW1hdGlvbikgJiZcbiAgICAhdG9vbFVzZUNvbmZpcm1RdWV1ZS5sZW5ndGggJiZcbiAgICAhaXNNZXNzYWdlU2VsZWN0b3JWaXNpYmxlXG5cbiAgY29uc3QgaGFzVG9vbHNJblByb2dyZXNzID0gaW5Qcm9ncmVzc1Rvb2xVc2VJRHMuc2l6ZSA+IDBcblxuICAvLyBSZXBvcnQgcHJvZ3Jlc3MgdG8gdGVybWluYWwgKGZvciB0ZXJtaW5hbHMgdGhhdCBzdXBwb3J0IE9TQyA5OzQpXG4gIGNvbnN0IHsgcHJvZ3Jlc3MgfSA9IHVzZVRlcm1pbmFsTm90aWZpY2F0aW9uKClcbiAgY29uc3QgcHJldlByb2dyZXNzU3RhdGUgPSB1c2VSZWY8c3RyaW5nIHwgbnVsbD4obnVsbClcbiAgY29uc3QgcHJvZ3Jlc3NFbmFibGVkID1cbiAgICBnZXRHbG9iYWxDb25maWcoKS50ZXJtaW5hbFByb2dyZXNzQmFyRW5hYmxlZCAmJlxuICAgICFnZXRJc1JlbW90ZU1vZGUoKSAmJlxuICAgICEocHJvYWN0aXZlTW9kdWxlPy5pc1Byb2FjdGl2ZUFjdGl2ZSgpID8/IGZhbHNlKVxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGNvbnN0IHN0YXRlID0gcHJvZ3Jlc3NFbmFibGVkXG4gICAgICA/IGhhc1Rvb2xzSW5Qcm9ncmVzc1xuICAgICAgICA/ICdpbmRldGVybWluYXRlJ1xuICAgICAgICA6ICdjb21wbGV0ZWQnXG4gICAgICA6IG51bGxcbiAgICBpZiAocHJldlByb2dyZXNzU3RhdGUuY3VycmVudCA9PT0gc3RhdGUpIHJldHVyblxuICAgIHByZXZQcm9ncmVzc1N0YXRlLmN1cnJlbnQgPSBzdGF0ZVxuICAgIHByb2dyZXNzKHN0YXRlKVxuICB9LCBbcHJvZ3Jlc3MsIHByb2dyZXNzRW5hYmxlZCwgaGFzVG9vbHNJblByb2dyZXNzXSlcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICByZXR1cm4gKCkgPT4gcHJvZ3Jlc3MobnVsbClcbiAgfSwgW3Byb2dyZXNzXSlcblxuICBjb25zdCBtZXNzYWdlS2V5ID0gdXNlQ2FsbGJhY2soXG4gICAgKG1zZzogUmVuZGVyYWJsZU1lc3NhZ2UpID0+IGAke21zZy51dWlkfS0ke2NvbnZlcnNhdGlvbklkfWAsXG4gICAgW2NvbnZlcnNhdGlvbklkXSxcbiAgKVxuXG4gIGNvbnN0IHJlbmRlck1lc3NhZ2VSb3cgPSAobXNnOiBSZW5kZXJhYmxlTWVzc2FnZSwgaW5kZXg6IG51bWJlcikgPT4ge1xuICAgIGNvbnN0IHByZXZUeXBlID0gaW5kZXggPiAwID8gcmVuZGVyYWJsZU1lc3NhZ2VzW2luZGV4IC0gMV0/LnR5cGUgOiB1bmRlZmluZWRcbiAgICBjb25zdCBpc1VzZXJDb250aW51YXRpb24gPSBtc2cudHlwZSA9PT0gJ3VzZXInICYmIHByZXZUeXBlID09PSAndXNlcidcbiAgICAvLyBoYXNDb250ZW50QWZ0ZXIgaXMgb25seSBjb25zdW1lZCBmb3IgY29sbGFwc2VkX3JlYWRfc2VhcmNoIGdyb3VwcztcbiAgICAvLyBza2lwIHRoZSBzY2FuIGZvciBldmVyeXRoaW5nIGVsc2UuIHN0cmVhbWluZ1RleHQgaXMgcmVuZGVyZWQgYXMgYVxuICAgIC8vIHNpYmxpbmcgYWZ0ZXIgdGhpcyBtYXAsIHNvIGl0J3MgbmV2ZXIgaW4gcmVuZGVyYWJsZU1lc3NhZ2VzIOKAlCBPUiBpdFxuICAgIC8vIGluIGV4cGxpY2l0bHkgc28gdGhlIGdyb3VwIGZsaXBzIHRvIHBhc3QgdGVuc2UgYXMgc29vbiBhcyB0ZXh0IHN0YXJ0c1xuICAgIC8vIHN0cmVhbWluZyBpbnN0ZWFkIG9mIHdhaXRpbmcgZm9yIHRoZSBibG9jayB0byBmaW5hbGl6ZS5cbiAgICBjb25zdCBoYXNDb250ZW50QWZ0ZXIgPVxuICAgICAgbXNnLnR5cGUgPT09ICdjb2xsYXBzZWRfcmVhZF9zZWFyY2gnICYmXG4gICAgICAoISFzdHJlYW1pbmdUZXh0IHx8XG4gICAgICAgIGhhc0NvbnRlbnRBZnRlckluZGV4KFxuICAgICAgICAgIHJlbmRlcmFibGVNZXNzYWdlcyxcbiAgICAgICAgICBpbmRleCxcbiAgICAgICAgICB0b29scyxcbiAgICAgICAgICBzdHJlYW1pbmdUb29sVXNlSURzLFxuICAgICAgICApKVxuXG4gICAgY29uc3QgayA9IG1lc3NhZ2VLZXkobXNnKVxuICAgIGNvbnN0IHJvdyA9IChcbiAgICAgIDxNZXNzYWdlUm93XG4gICAgICAgIGtleT17a31cbiAgICAgICAgbWVzc2FnZT17bXNnfVxuICAgICAgICBpc1VzZXJDb250aW51YXRpb249e2lzVXNlckNvbnRpbnVhdGlvbn1cbiAgICAgICAgaGFzQ29udGVudEFmdGVyPXtoYXNDb250ZW50QWZ0ZXJ9XG4gICAgICAgIHRvb2xzPXt0b29sc31cbiAgICAgICAgY29tbWFuZHM9e2NvbW1hbmRzfVxuICAgICAgICB2ZXJib3NlPXtcbiAgICAgICAgICB2ZXJib3NlIHx8XG4gICAgICAgICAgaXNJdGVtRXhwYW5kZWQobXNnKSB8fFxuICAgICAgICAgIChjdXJzb3I/LmV4cGFuZGVkID09PSB0cnVlICYmIGluZGV4ID09PSBzZWxlY3RlZElkeClcbiAgICAgICAgfVxuICAgICAgICBpblByb2dyZXNzVG9vbFVzZUlEcz17aW5Qcm9ncmVzc1Rvb2xVc2VJRHN9XG4gICAgICAgIHN0cmVhbWluZ1Rvb2xVc2VJRHM9e3N0cmVhbWluZ1Rvb2xVc2VJRHN9XG4gICAgICAgIHNjcmVlbj17c2NyZWVufVxuICAgICAgICBjYW5BbmltYXRlPXtjYW5BbmltYXRlfVxuICAgICAgICBvbk9wZW5SYXRlTGltaXRPcHRpb25zPXtvbk9wZW5SYXRlTGltaXRPcHRpb25zfVxuICAgICAgICBsYXN0VGhpbmtpbmdCbG9ja0lkPXtsYXN0VGhpbmtpbmdCbG9ja0lkfVxuICAgICAgICBsYXRlc3RCYXNoT3V0cHV0VVVJRD17bGF0ZXN0QmFzaE91dHB1dFVVSUR9XG4gICAgICAgIGNvbHVtbnM9e2NvbHVtbnN9XG4gICAgICAgIGlzTG9hZGluZz17aXNMb2FkaW5nfVxuICAgICAgICBsb29rdXBzPXtsb29rdXBzfVxuICAgICAgLz5cbiAgICApXG5cbiAgICAvLyBQZXItcm93IFByb3ZpZGVyIOKAlCBvbmx5IDIgcm93cyByZS1yZW5kZXIgb24gc2VsZWN0aW9uIGNoYW5nZS5cbiAgICAvLyBXcmFwcGVkIEJFRk9SRSBkaXZpZGVyIGJyYW5jaCBzbyBib3RoIHJldHVybiBwYXRocyBnZXQgaXQuXG4gICAgY29uc3Qgd3JhcHBlZCA9IChcbiAgICAgIDxNZXNzYWdlQWN0aW9uc1NlbGVjdGVkQ29udGV4dC5Qcm92aWRlclxuICAgICAgICBrZXk9e2t9XG4gICAgICAgIHZhbHVlPXtpbmRleCA9PT0gc2VsZWN0ZWRJZHh9XG4gICAgICA+XG4gICAgICAgIHtyb3d9XG4gICAgICA8L01lc3NhZ2VBY3Rpb25zU2VsZWN0ZWRDb250ZXh0LlByb3ZpZGVyPlxuICAgIClcblxuICAgIGlmICh1bnNlZW5EaXZpZGVyICYmIGluZGV4ID09PSBkaXZpZGVyQmVmb3JlSW5kZXgpIHtcbiAgICAgIHJldHVybiBbXG4gICAgICAgIDxCb3gga2V5PVwidW5zZWVuLWRpdmlkZXJcIiBtYXJnaW5Ub3A9ezF9PlxuICAgICAgICAgIDxEaXZpZGVyXG4gICAgICAgICAgICB0aXRsZT17YCR7dW5zZWVuRGl2aWRlci5jb3VudH0gbmV3ICR7cGx1cmFsKHVuc2VlbkRpdmlkZXIuY291bnQsICdtZXNzYWdlJyl9YH1cbiAgICAgICAgICAgIHdpZHRoPXtjb2x1bW5zfVxuICAgICAgICAgICAgY29sb3I9XCJpbmFjdGl2ZVwiXG4gICAgICAgICAgLz5cbiAgICAgICAgPC9Cb3g+LFxuICAgICAgICB3cmFwcGVkLFxuICAgICAgXVxuICAgIH1cbiAgICByZXR1cm4gd3JhcHBlZFxuICB9XG5cbiAgLy8gU2VhcmNoIGluZGV4aW5nOiBmb3IgdG9vbF9yZXN1bHQgbWVzc2FnZXMsIGxvb2sgdXAgdGhlIFRvb2wgYW5kIHVzZVxuICAvLyBpdHMgZXh0cmFjdFNlYXJjaFRleHQg4oCUIHRvb2wtb3duZWQsIHByZWNpc2UsIG1hdGNoZXMgd2hhdFxuICAvLyByZW5kZXJUb29sUmVzdWx0TWVzc2FnZSBzaG93cy4gRmFsbHMgYmFjayB0byByZW5kZXJhYmxlU2VhcmNoVGV4dFxuICAvLyAoZHVjay10eXBlcyB0b29sVXNlUmVzdWx0KSBmb3IgdG9vbHMgdGhhdCBoYXZlbid0IGltcGxlbWVudGVkIGl0LFxuICAvLyBhbmQgZm9yIGFsbCBub24tdG9vbC1yZXN1bHQgbWVzc2FnZSB0eXBlcy4gVGhlIGRyaWZ0LWNhdGNoZXIgdGVzdFxuICAvLyAodG9vbFNlYXJjaFRleHQudGVzdC50c3gpIHJlbmRlcnMgKyBjb21wYXJlcyB0byBrZWVwIHRoZXNlIGluIHN5bmMuXG4gIC8vXG4gIC8vIEEgc2Vjb25kLVJlYWN0LXJvb3QgcmVjb25jaWxlIGFwcHJvYWNoIHdhcyB0cmllZCBhbmQgcnVsZWQgb3V0XG4gIC8vIChtZWFzdXJlZCAzLjFtcy9tc2csIGdyb3dpbmcg4oCUIGZsdXNoU3luY1dvcmsgcHJvY2Vzc2VzIGFsbCByb290cztcbiAgLy8gY29tcG9uZW50IGhvb2tzIG11dGF0ZSBzaGFyZWQgc3RhdGUg4oaSIG1haW4gcm9vdCBhY2N1bXVsYXRlcyB1cGRhdGVzKS5cbiAgY29uc3Qgc2VhcmNoVGV4dENhY2hlID0gdXNlUmVmKG5ldyBXZWFrTWFwPFJlbmRlcmFibGVNZXNzYWdlLCBzdHJpbmc+KCkpXG4gIGNvbnN0IGV4dHJhY3RTZWFyY2hUZXh0ID0gdXNlQ2FsbGJhY2soXG4gICAgKG1zZzogUmVuZGVyYWJsZU1lc3NhZ2UpOiBzdHJpbmcgPT4ge1xuICAgICAgY29uc3QgY2FjaGVkID0gc2VhcmNoVGV4dENhY2hlLmN1cnJlbnQuZ2V0KG1zZylcbiAgICAgIGlmIChjYWNoZWQgIT09IHVuZGVmaW5lZCkgcmV0dXJuIGNhY2hlZFxuICAgICAgbGV0IHRleHQgPSByZW5kZXJhYmxlU2VhcmNoVGV4dChtc2cpXG4gICAgICAvLyBJZiB0aGlzIGlzIGEgdG9vbF9yZXN1bHQgbWVzc2FnZSBhbmQgdGhlIHRvb2wgaW1wbGVtZW50c1xuICAgICAgLy8gZXh0cmFjdFNlYXJjaFRleHQsIHByZWZlciB0aGF0IOKAlCBpdCdzIHByZWNpc2UgKHRvb2wtb3duZWQpXG4gICAgICAvLyB2cyByZW5kZXJhYmxlU2VhcmNoVGV4dCdzIGZpZWxkLW5hbWUgaGV1cmlzdGljLlxuICAgICAgaWYgKFxuICAgICAgICBtc2cudHlwZSA9PT0gJ3VzZXInICYmXG4gICAgICAgIG1zZy50b29sVXNlUmVzdWx0ICYmXG4gICAgICAgIEFycmF5LmlzQXJyYXkobXNnLm1lc3NhZ2UuY29udGVudClcbiAgICAgICkge1xuICAgICAgICBjb25zdCB0ciA9IG1zZy5tZXNzYWdlLmNvbnRlbnQuZmluZChiID0+IGIudHlwZSA9PT0gJ3Rvb2xfcmVzdWx0JylcbiAgICAgICAgaWYgKHRyICYmICd0b29sX3VzZV9pZCcgaW4gdHIpIHtcbiAgICAgICAgICBjb25zdCB0dSA9IGxvb2t1cHMudG9vbFVzZUJ5VG9vbFVzZUlELmdldCh0ci50b29sX3VzZV9pZClcbiAgICAgICAgICBjb25zdCB0b29sID0gdHUgJiYgZmluZFRvb2xCeU5hbWUodG9vbHMsIHR1Lm5hbWUpXG4gICAgICAgICAgY29uc3QgZXh0cmFjdGVkID0gdG9vbD8uZXh0cmFjdFNlYXJjaFRleHQ/LihcbiAgICAgICAgICAgIG1zZy50b29sVXNlUmVzdWx0IGFzIG5ldmVyLFxuICAgICAgICAgIClcbiAgICAgICAgICAvLyB1bmRlZmluZWQgPSB0b29sIGRpZG4ndCBpbXBsZW1lbnQg4oaSIGtlZXAgaGV1cmlzdGljLiBFbXB0eVxuICAgICAgICAgIC8vIHN0cmluZyA9IHRvb2wgc2F5cyBcIm5vdGhpbmcgdG8gaW5kZXhcIiDihpIgcmVzcGVjdCB0aGF0LlxuICAgICAgICAgIGlmIChleHRyYWN0ZWQgIT09IHVuZGVmaW5lZCkgdGV4dCA9IGV4dHJhY3RlZFxuICAgICAgICB9XG4gICAgICB9XG4gICAgICAvLyBDYWNoZSBMT1dFUkVEOiBzZXRTZWFyY2hRdWVyeSdzIGhvdCBsb29wIGluZGV4T2ZzIHBlciBrZXlzdHJva2UuXG4gICAgICAvLyBMb3dlcmluZyBoZXJlIChvbmNlLCBhdCB3YXJtKSB2cyB0aGVyZSAoZXZlcnkga2V5c3Ryb2tlKSB0cmFkZXNcbiAgICAgIC8vIH5zYW1lIHN0ZWFkeS1zdGF0ZSBtZW1vcnkgZm9yIHplcm8gcGVyLWtleXN0cm9rZSBhbGxvYy4gQ2FjaGVcbiAgICAgIC8vIEdDJ3Mgd2l0aCBtZXNzYWdlcyBvbiB0cmFuc2NyaXB0IGV4aXQuIFRvb2wgbWV0aG9kcyByZXR1cm4gcmF3O1xuICAgICAgLy8gcmVuZGVyYWJsZVNlYXJjaFRleHQgYWxyZWFkeSBsb3dlcmNhc2VzIChyZWR1bmRhbnQgYnV0IGNoZWFwKS5cbiAgICAgIGNvbnN0IGxvd2VyZWQgPSB0ZXh0LnRvTG93ZXJDYXNlKClcbiAgICAgIHNlYXJjaFRleHRDYWNoZS5jdXJyZW50LnNldChtc2csIGxvd2VyZWQpXG4gICAgICByZXR1cm4gbG93ZXJlZFxuICAgIH0sXG4gICAgW3Rvb2xzLCBsb29rdXBzXSxcbiAgKVxuXG4gIHJldHVybiAoXG4gICAgPD5cbiAgICAgIHsvKiBMb2dvICovfVxuICAgICAgeyFoaWRlTG9nbyAmJiAhKHJlbmRlclJhbmdlICYmIHJlbmRlclJhbmdlWzBdID4gMCkgJiYgKFxuICAgICAgICA8TG9nb0hlYWRlciBhZ2VudERlZmluaXRpb25zPXthZ2VudERlZmluaXRpb25zfSAvPlxuICAgICAgKX1cblxuICAgICAgey8qIFRydW5jYXRpb24gaW5kaWNhdG9yICovfVxuICAgICAge2hhc1RydW5jYXRlZE1lc3NhZ2VzICYmIChcbiAgICAgICAgPERpdmlkZXJcbiAgICAgICAgICB0aXRsZT17YCR7dG9nZ2xlU2hvd0FsbFNob3J0Y3V0fSB0byBzaG93ICR7Y2hhbGsuYm9sZChoaWRkZW5NZXNzYWdlQ291bnQpfSBwcmV2aW91cyBtZXNzYWdlc2B9XG4gICAgICAgICAgd2lkdGg9e2NvbHVtbnN9XG4gICAgICAgIC8+XG4gICAgICApfVxuXG4gICAgICB7LyogU2hvdyBhbGwgaW5kaWNhdG9yICovfVxuICAgICAge2lzVHJhbnNjcmlwdE1vZGUgJiZcbiAgICAgICAgc2hvd0FsbEluVHJhbnNjcmlwdCAmJlxuICAgICAgICBoaWRkZW5NZXNzYWdlQ291bnQgPiAwICYmXG4gICAgICAgIC8vIGRpc2FibGVSZW5kZXJDYXAgKGUuZy4gWyBkdW1wLXRvLXNjcm9sbGJhY2spIG1lYW5zIHdlJ3JlIHVuY2FwcGVkXG4gICAgICAgIC8vIGFzIGEgb25lLXNob3QgZXNjYXBlIGhhdGNoLCBub3QgYSB0b2dnbGUg4oCUIGN0cmwrZSBpcyBkZWFkIGFuZFxuICAgICAgICAvLyBub3RoaW5nIGlzIGFjdHVhbGx5IFwiaGlkZGVuXCIgdG8gcmVzdG9yZS5cbiAgICAgICAgIWRpc2FibGVSZW5kZXJDYXAgJiYgKFxuICAgICAgICAgIDxEaXZpZGVyXG4gICAgICAgICAgICB0aXRsZT17YCR7dG9nZ2xlU2hvd0FsbFNob3J0Y3V0fSB0byBoaWRlICR7Y2hhbGsuYm9sZChoaWRkZW5NZXNzYWdlQ291bnQpfSBwcmV2aW91cyBtZXNzYWdlc2B9XG4gICAgICAgICAgICB3aWR0aD17Y29sdW1uc31cbiAgICAgICAgICAvPlxuICAgICAgICApfVxuXG4gICAgICB7LyogTWVzc2FnZXMgLSByZW5kZXJlZCBhcyBtZW1vaXplZCBNZXNzYWdlUm93IGNvbXBvbmVudHMuXG4gICAgICAgICAgZmxhdE1hcCBpbnNlcnRzIHRoZSB1bnNlZW4tZGl2aWRlciBhcyBhIHNlcGFyYXRlIGtleWVkIHNpYmxpbmcgc29cbiAgICAgICAgICAoYSkgbm9uLWZ1bGxzY3JlZW4gcmVuZGVycyBwYXkgbm8gcGVyLW1lc3NhZ2UgRnJhZ21lbnQgd3JhcCwgYW5kXG4gICAgICAgICAgKGIpIGRpdmlkZXIgdG9nZ2xlIGluIGZ1bGxzY3JlZW4gcHJlc2VydmVzIGFsbCBNZXNzYWdlUm93cyBieSBrZXkuXG4gICAgICAgICAgUHJlLWNvbXB1dGUgZGVyaXZlZCB2YWx1ZXMgaW5zdGVhZCBvZiBwYXNzaW5nIHJlbmRlcmFibGVNZXNzYWdlcyB0b1xuICAgICAgICAgIGVhY2ggcm93IC0gUmVhY3QgQ29tcGlsZXIgcGlucyBwcm9wcyBpbiB0aGUgZmliZXIncyBtZW1vQ2FjaGUsIHNvXG4gICAgICAgICAgcGFzc2luZyB0aGUgYXJyYXkgd291bGQgYWNjdW11bGF0ZSBldmVyeSBoaXN0b3JpY2FsIHZlcnNpb25cbiAgICAgICAgICAofjEtMk1CIG92ZXIgYSA3LXR1cm4gc2Vzc2lvbikuICovfVxuICAgICAge3ZpcnR1YWxTY3JvbGxSdW50aW1lR2F0ZSA/IChcbiAgICAgICAgPEluVmlydHVhbExpc3RDb250ZXh0LlByb3ZpZGVyIHZhbHVlPXt0cnVlfT5cbiAgICAgICAgICA8VmlydHVhbE1lc3NhZ2VMaXN0XG4gICAgICAgICAgICBtZXNzYWdlcz17cmVuZGVyYWJsZU1lc3NhZ2VzfVxuICAgICAgICAgICAgc2Nyb2xsUmVmPXtzY3JvbGxSZWZ9XG4gICAgICAgICAgICBjb2x1bW5zPXtjb2x1bW5zfVxuICAgICAgICAgICAgaXRlbUtleT17bWVzc2FnZUtleX1cbiAgICAgICAgICAgIHJlbmRlckl0ZW09e3JlbmRlck1lc3NhZ2VSb3d9XG4gICAgICAgICAgICBvbkl0ZW1DbGljaz17b25JdGVtQ2xpY2t9XG4gICAgICAgICAgICBpc0l0ZW1DbGlja2FibGU9e2lzSXRlbUNsaWNrYWJsZX1cbiAgICAgICAgICAgIGlzSXRlbUV4cGFuZGVkPXtpc0l0ZW1FeHBhbmRlZH1cbiAgICAgICAgICAgIHRyYWNrU3RpY2t5UHJvbXB0PXt0cmFja1N0aWNreVByb21wdH1cbiAgICAgICAgICAgIHNlbGVjdGVkSW5kZXg9e3NlbGVjdGVkSWR4ID49IDAgPyBzZWxlY3RlZElkeCA6IHVuZGVmaW5lZH1cbiAgICAgICAgICAgIGN1cnNvck5hdlJlZj17Y3Vyc29yTmF2UmVmfVxuICAgICAgICAgICAgc2V0Q3Vyc29yPXtzZXRDdXJzb3J9XG4gICAgICAgICAgICBqdW1wUmVmPXtqdW1wUmVmfVxuICAgICAgICAgICAgb25TZWFyY2hNYXRjaGVzQ2hhbmdlPXtvblNlYXJjaE1hdGNoZXNDaGFuZ2V9XG4gICAgICAgICAgICBzY2FuRWxlbWVudD17c2NhbkVsZW1lbnR9XG4gICAgICAgICAgICBzZXRQb3NpdGlvbnM9e3NldFBvc2l0aW9uc31cbiAgICAgICAgICAgIGV4dHJhY3RTZWFyY2hUZXh0PXtleHRyYWN0U2VhcmNoVGV4dH1cbiAgICAgICAgICAvPlxuICAgICAgICA8L0luVmlydHVhbExpc3RDb250ZXh0LlByb3ZpZGVyPlxuICAgICAgKSA6IChcbiAgICAgICAgcmVuZGVyYWJsZU1lc3NhZ2VzLmZsYXRNYXAocmVuZGVyTWVzc2FnZVJvdylcbiAgICAgICl9XG5cbiAgICAgIHtzdHJlYW1pbmdUZXh0ICYmICFpc0JyaWVmT25seSAmJiAoXG4gICAgICAgIDxCb3hcbiAgICAgICAgICBhbGlnbkl0ZW1zPVwiZmxleC1zdGFydFwiXG4gICAgICAgICAgZmxleERpcmVjdGlvbj1cInJvd1wiXG4gICAgICAgICAgbWFyZ2luVG9wPXsxfVxuICAgICAgICAgIHdpZHRoPVwiMTAwJVwiXG4gICAgICAgID5cbiAgICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJyb3dcIj5cbiAgICAgICAgICAgIDxCb3ggbWluV2lkdGg9ezJ9PlxuICAgICAgICAgICAgICA8VGV4dCBjb2xvcj1cInRleHRcIj57QkxBQ0tfQ0lSQ0xFfTwvVGV4dD5cbiAgICAgICAgICAgIDwvQm94PlxuICAgICAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgICAgICAgIDxTdHJlYW1pbmdNYXJrZG93bj57c3RyZWFtaW5nVGV4dH08L1N0cmVhbWluZ01hcmtkb3duPlxuICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgIDwvQm94PlxuICAgICAgKX1cblxuICAgICAge2lzU3RyZWFtaW5nVGhpbmtpbmdWaXNpYmxlICYmIHN0cmVhbWluZ1RoaW5raW5nICYmICFpc0JyaWVmT25seSAmJiAoXG4gICAgICAgIDxCb3ggbWFyZ2luVG9wPXsxfT5cbiAgICAgICAgICA8QXNzaXN0YW50VGhpbmtpbmdNZXNzYWdlXG4gICAgICAgICAgICBwYXJhbT17e1xuICAgICAgICAgICAgICB0eXBlOiAndGhpbmtpbmcnLFxuICAgICAgICAgICAgICB0aGlua2luZzogc3RyZWFtaW5nVGhpbmtpbmcudGhpbmtpbmcsXG4gICAgICAgICAgICB9fVxuICAgICAgICAgICAgYWRkTWFyZ2luPXtmYWxzZX1cbiAgICAgICAgICAgIGlzVHJhbnNjcmlwdE1vZGU9e3RydWV9XG4gICAgICAgICAgICB2ZXJib3NlPXt2ZXJib3NlfVxuICAgICAgICAgICAgaGlkZUluVHJhbnNjcmlwdD17ZmFsc2V9XG4gICAgICAgICAgLz5cbiAgICAgICAgPC9Cb3g+XG4gICAgICApfVxuICAgIDwvPlxuICApXG59XG5cbi8qKiBLZXkgZm9yIGNsaWNrLXRvLWV4cGFuZDogdG9vbF91c2VfaWQgd2hlcmUgYXZhaWxhYmxlIChzbyB0b29sX3VzZSArIGl0c1xuICogIHRvb2xfcmVzdWx0IGV4cGFuZCB0b2dldGhlciksIGVsc2UgdXVpZCBmb3IgZ3JvdXBzL3RoaW5raW5nLiAqL1xuZnVuY3Rpb24gZXhwYW5kS2V5KG1zZzogUmVuZGVyYWJsZU1lc3NhZ2UpOiBzdHJpbmcge1xuICByZXR1cm4gKFxuICAgIChtc2cudHlwZSA9PT0gJ2Fzc2lzdGFudCcgfHwgbXNnLnR5cGUgPT09ICd1c2VyJ1xuICAgICAgPyBnZXRUb29sVXNlSUQobXNnKVxuICAgICAgOiBudWxsKSA/PyBtc2cudXVpZFxuICApXG59XG5cbi8vIEN1c3RvbSBjb21wYXJhdG9yIHRvIHByZXZlbnQgdW5uZWNlc3NhcnkgcmUtcmVuZGVycyBkdXJpbmcgc3RyZWFtaW5nLlxuLy8gRGVmYXVsdCBSZWFjdC5tZW1vIGRvZXMgc2hhbGxvdyBjb21wYXJpc29uIHdoaWNoIGZhaWxzIHdoZW46XG4vLyAxLiBvbk9wZW5SYXRlTGltaXRPcHRpb25zIGNhbGxiYWNrIGlzIHJlY3JlYXRlZCAoZG9lc24ndCBhZmZlY3QgcmVuZGVyIG91dHB1dClcbi8vIDIuIHN0cmVhbWluZ1Rvb2xVc2VzIGFycmF5IGlzIHJlY3JlYXRlZCBvbiBldmVyeSBkZWx0YSwgYnV0IG9ubHkgY29udGVudEJsb2NrIG1hdHRlcnMgZm9yIHJlbmRlcmluZ1xuLy8gMy4gc3RyZWFtaW5nVGhpbmtpbmcgY2hhbmdlcyBvbiBldmVyeSBkZWx0YSAtIHdlIERPIHdhbnQgdG8gcmUtcmVuZGVyIGZvciB0aGlzXG5mdW5jdGlvbiBzZXRzRXF1YWw8VD4oYTogU2V0PFQ+LCBiOiBTZXQ8VD4pOiBib29sZWFuIHtcbiAgaWYgKGEuc2l6ZSAhPT0gYi5zaXplKSByZXR1cm4gZmFsc2VcbiAgZm9yIChjb25zdCBpdGVtIG9mIGEpIHtcbiAgICBpZiAoIWIuaGFzKGl0ZW0pKSByZXR1cm4gZmFsc2VcbiAgfVxuICByZXR1cm4gdHJ1ZVxufVxuXG5leHBvcnQgY29uc3QgTWVzc2FnZXMgPSBSZWFjdC5tZW1vKE1lc3NhZ2VzSW1wbCwgKHByZXYsIG5leHQpID0+IHtcbiAgY29uc3Qga2V5cyA9IE9iamVjdC5rZXlzKHByZXYpIGFzIChrZXlvZiB0eXBlb2YgcHJldilbXVxuICBmb3IgKGNvbnN0IGtleSBvZiBrZXlzKSB7XG4gICAgaWYgKFxuICAgICAga2V5ID09PSAnb25PcGVuUmF0ZUxpbWl0T3B0aW9ucycgfHxcbiAgICAgIGtleSA9PT0gJ3Njcm9sbFJlZicgfHxcbiAgICAgIGtleSA9PT0gJ3RyYWNrU3RpY2t5UHJvbXB0JyB8fFxuICAgICAga2V5ID09PSAnc2V0Q3Vyc29yJyB8fFxuICAgICAga2V5ID09PSAnY3Vyc29yTmF2UmVmJyB8fFxuICAgICAga2V5ID09PSAnanVtcFJlZicgfHxcbiAgICAgIGtleSA9PT0gJ29uU2VhcmNoTWF0Y2hlc0NoYW5nZScgfHxcbiAgICAgIGtleSA9PT0gJ3NjYW5FbGVtZW50JyB8fFxuICAgICAga2V5ID09PSAnc2V0UG9zaXRpb25zJ1xuICAgIClcbiAgICAgIGNvbnRpbnVlXG4gICAgaWYgKHByZXZba2V5XSAhPT0gbmV4dFtrZXldKSB7XG4gICAgICBpZiAoa2V5ID09PSAnc3RyZWFtaW5nVG9vbFVzZXMnKSB7XG4gICAgICAgIGNvbnN0IHAgPSBwcmV2LnN0cmVhbWluZ1Rvb2xVc2VzXG4gICAgICAgIGNvbnN0IG4gPSBuZXh0LnN0cmVhbWluZ1Rvb2xVc2VzXG4gICAgICAgIGlmIChcbiAgICAgICAgICBwLmxlbmd0aCA9PT0gbi5sZW5ndGggJiZcbiAgICAgICAgICBwLmV2ZXJ5KChpdGVtLCBpKSA9PiBpdGVtLmNvbnRlbnRCbG9jayA9PT0gbltpXT8uY29udGVudEJsb2NrKVxuICAgICAgICApIHtcbiAgICAgICAgICBjb250aW51ZVxuICAgICAgICB9XG4gICAgICB9XG4gICAgICBpZiAoa2V5ID09PSAnaW5Qcm9ncmVzc1Rvb2xVc2VJRHMnKSB7XG4gICAgICAgIGlmIChzZXRzRXF1YWwocHJldi5pblByb2dyZXNzVG9vbFVzZUlEcywgbmV4dC5pblByb2dyZXNzVG9vbFVzZUlEcykpIHtcbiAgICAgICAgICBjb250aW51ZVxuICAgICAgICB9XG4gICAgICB9XG4gICAgICBpZiAoa2V5ID09PSAndW5zZWVuRGl2aWRlcicpIHtcbiAgICAgICAgY29uc3QgcCA9IHByZXYudW5zZWVuRGl2aWRlclxuICAgICAgICBjb25zdCBuID0gbmV4dC51bnNlZW5EaXZpZGVyXG4gICAgICAgIGlmIChcbiAgICAgICAgICBwPy5maXJzdFVuc2VlblV1aWQgPT09IG4/LmZpcnN0VW5zZWVuVXVpZCAmJlxuICAgICAgICAgIHA/LmNvdW50ID09PSBuPy5jb3VudFxuICAgICAgICApIHtcbiAgICAgICAgICBjb250aW51ZVxuICAgICAgICB9XG4gICAgICB9XG4gICAgICBpZiAoa2V5ID09PSAndG9vbHMnKSB7XG4gICAgICAgIGNvbnN0IHAgPSBwcmV2LnRvb2xzXG4gICAgICAgIGNvbnN0IG4gPSBuZXh0LnRvb2xzXG4gICAgICAgIGlmIChcbiAgICAgICAgICBwLmxlbmd0aCA9PT0gbi5sZW5ndGggJiZcbiAgICAgICAgICBwLmV2ZXJ5KCh0b29sLCBpKSA9PiB0b29sLm5hbWUgPT09IG5baV0/Lm5hbWUpXG4gICAgICAgICkge1xuICAgICAgICAgIGNvbnRpbnVlXG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIC8vIHN0cmVhbWluZ1RoaW5raW5nIGNoYW5nZXMgZnJlcXVlbnRseSAtIGFsd2F5cyByZS1yZW5kZXIgd2hlbiBpdCBjaGFuZ2VzXG4gICAgICAvLyAobm8gc3BlY2lhbCBoYW5kbGluZyBuZWVkZWQsIGRlZmF1bHQgYmVoYXZpb3IgaXMgY29ycmVjdClcbiAgICAgIHJldHVybiBmYWxzZVxuICAgIH1cbiAgfVxuICByZXR1cm4gdHJ1ZVxufSlcblxuZXhwb3J0IGZ1bmN0aW9uIHNob3VsZFJlbmRlclN0YXRpY2FsbHkoXG4gIG1lc3NhZ2U6IFJlbmRlcmFibGVNZXNzYWdlLFxuICBzdHJlYW1pbmdUb29sVXNlSURzOiBTZXQ8c3RyaW5nPixcbiAgaW5Qcm9ncmVzc1Rvb2xVc2VJRHM6IFNldDxzdHJpbmc+LFxuICBzaWJsaW5nVG9vbFVzZUlEczogUmVhZG9ubHlTZXQ8c3RyaW5nPixcbiAgc2NyZWVuOiBTY3JlZW4sXG4gIGxvb2t1cHM6IFJldHVyblR5cGU8dHlwZW9mIGJ1aWxkTWVzc2FnZUxvb2t1cHM+LFxuKTogYm9vbGVhbiB7XG4gIGlmIChzY3JlZW4gPT09ICd0cmFuc2NyaXB0Jykge1xuICAgIHJldHVybiB0cnVlXG4gIH1cbiAgc3dpdGNoIChtZXNzYWdlLnR5cGUpIHtcbiAgICBjYXNlICdhdHRhY2htZW50JzpcbiAgICBjYXNlICd1c2VyJzpcbiAgICBjYXNlICdhc3Npc3RhbnQnOiB7XG4gICAgICBpZiAobWVzc2FnZS50eXBlID09PSAnYXNzaXN0YW50Jykge1xuICAgICAgICBjb25zdCBibG9jayA9IG1lc3NhZ2UubWVzc2FnZS5jb250ZW50WzBdXG4gICAgICAgIGlmIChibG9jaz8udHlwZSA9PT0gJ3NlcnZlcl90b29sX3VzZScpIHtcbiAgICAgICAgICByZXR1cm4gbG9va3Vwcy5yZXNvbHZlZFRvb2xVc2VJRHMuaGFzKGJsb2NrLmlkKVxuICAgICAgICB9XG4gICAgICB9XG4gICAgICBjb25zdCB0b29sVXNlSUQgPSBnZXRUb29sVXNlSUQobWVzc2FnZSlcbiAgICAgIGlmICghdG9vbFVzZUlEKSB7XG4gICAgICAgIHJldHVybiB0cnVlXG4gICAgICB9XG4gICAgICBpZiAoc3RyZWFtaW5nVG9vbFVzZUlEcy5oYXModG9vbFVzZUlEKSkge1xuICAgICAgICByZXR1cm4gZmFsc2VcbiAgICAgIH1cbiAgICAgIGlmIChpblByb2dyZXNzVG9vbFVzZUlEcy5oYXModG9vbFVzZUlEKSkge1xuICAgICAgICByZXR1cm4gZmFsc2VcbiAgICAgIH1cblxuICAgICAgLy8gQ2hlY2sgaWYgdGhlcmUgYXJlIGFueSB1bnJlc29sdmVkIFBvc3RUb29sVXNlIGhvb2tzIGZvciB0aGlzIHRvb2wgdXNlXG4gICAgICAvLyBJZiBzbywga2VlcCB0aGUgbWVzc2FnZSB0cmFuc2llbnQgc28gdGhlIEhvb2tQcm9ncmVzc01lc3NhZ2UgY2FuIHVwZGF0ZVxuICAgICAgaWYgKGhhc1VucmVzb2x2ZWRIb29rc0Zyb21Mb29rdXAodG9vbFVzZUlELCAnUG9zdFRvb2xVc2UnLCBsb29rdXBzKSkge1xuICAgICAgICByZXR1cm4gZmFsc2VcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIGV2ZXJ5KHNpYmxpbmdUb29sVXNlSURzLCBsb29rdXBzLnJlc29sdmVkVG9vbFVzZUlEcylcbiAgICB9XG4gICAgY2FzZSAnc3lzdGVtJzoge1xuICAgICAgLy8gYXBpIGVycm9ycyBhbHdheXMgcmVuZGVyIGR5bmFtaWNhbGx5LCBzaW5jZSB3ZSBoaWRlXG4gICAgICAvLyB0aGVtIGFzIHNvb24gYXMgd2Ugc2VlIGFub3RoZXIgbm9uLWVycm9yIG1lc3NhZ2UuXG4gICAgICByZXR1cm4gbWVzc2FnZS5zdWJ0eXBlICE9PSAnYXBpX2Vycm9yJ1xuICAgIH1cbiAgICBjYXNlICdncm91cGVkX3Rvb2xfdXNlJzoge1xuICAgICAgY29uc3QgYWxsUmVzb2x2ZWQgPSBtZXNzYWdlLm1lc3NhZ2VzLmV2ZXJ5KG1zZyA9PiB7XG4gICAgICAgIGNvbnN0IGNvbnRlbnQgPSBtc2cubWVzc2FnZS5jb250ZW50WzBdXG4gICAgICAgIHJldHVybiAoXG4gICAgICAgICAgY29udGVudD8udHlwZSA9PT0gJ3Rvb2xfdXNlJyAmJlxuICAgICAgICAgIGxvb2t1cHMucmVzb2x2ZWRUb29sVXNlSURzLmhhcyhjb250ZW50LmlkKVxuICAgICAgICApXG4gICAgICB9KVxuICAgICAgcmV0dXJuIGFsbFJlc29sdmVkXG4gICAgfVxuICAgIGNhc2UgJ2NvbGxhcHNlZF9yZWFkX3NlYXJjaCc6IHtcbiAgICAgIC8vIEluIHByb21wdCBtb2RlLCBuZXZlciBtYXJrIGFzIHN0YXRpYyB0byBwcmV2ZW50IGZsaWNrZXIgYmV0d2VlbiBBUEkgdHVybnNcbiAgICAgIC8vIChJbiB0cmFuc2NyaXB0IG1vZGUsIHdlIGFscmVhZHkgcmV0dXJuZWQgdHJ1ZSBhdCB0aGUgdG9wIG9mIHRoaXMgZnVuY3Rpb24pXG4gICAgICByZXR1cm4gZmFsc2VcbiAgICB9XG4gIH1cbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLFNBQVNBLE9BQU8sUUFBUSxZQUFZO0FBQ3BDLE9BQU9DLEtBQUssTUFBTSxPQUFPO0FBQ3pCLGNBQWNDLElBQUksUUFBUSxRQUFRO0FBQ2xDLGNBQWNDLFNBQVMsUUFBUSxPQUFPO0FBQ3RDLE9BQU8sS0FBS0MsS0FBSyxNQUFNLE9BQU87QUFDOUIsU0FBU0MsV0FBVyxFQUFFQyxTQUFTLEVBQUVDLE9BQU8sRUFBRUMsTUFBTSxFQUFFQyxRQUFRLFFBQVEsT0FBTztBQUN6RSxTQUFTQyxLQUFLLFFBQVEsa0JBQWtCO0FBQ3hDLFNBQVNDLGVBQWUsUUFBUSx1QkFBdUI7QUFDdkQsY0FBY0MsT0FBTyxRQUFRLGdCQUFnQjtBQUM3QyxTQUFTQyxZQUFZLFFBQVEseUJBQXlCO0FBQ3RELFNBQVNDLGVBQWUsUUFBUSw2QkFBNkI7QUFDN0QsY0FBY0MsZUFBZSxRQUFRLGdDQUFnQztBQUNyRSxTQUFTQyx1QkFBdUIsUUFBUSxtQ0FBbUM7QUFDM0UsU0FBU0MsR0FBRyxFQUFFQyxJQUFJLFFBQVEsV0FBVztBQUNyQyxTQUFTQyxrQkFBa0IsUUFBUSxzQ0FBc0M7QUFDekUsY0FBY0MsTUFBTSxRQUFRLG9CQUFvQjtBQUNoRCxjQUFjQyxLQUFLLFFBQVEsWUFBWTtBQUN2QyxTQUFTQyxjQUFjLFFBQVEsWUFBWTtBQUMzQyxjQUFjQyxzQkFBc0IsUUFBUSxxQ0FBcUM7QUFDakYsY0FDRUMsT0FBTyxJQUFJQyxXQUFXLEVBQ3RCQyxpQkFBaUIsRUFDakJDLGVBQWUsSUFBSUMsbUJBQW1CLEVBQ3RDQyxpQkFBaUIsUUFDWixxQkFBcUI7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