/ screens / REPL.tsx
REPL.tsx
   1  import { c as _c } from "react/compiler-runtime";
   2  // biome-ignore-all assist/source/organizeImports: ANT-ONLY import markers must not be reordered
   3  import { feature } from 'bun:bundle';
   4  import { spawnSync } from 'child_process';
   5  import { snapshotOutputTokensForTurn, getCurrentTurnTokenBudget, getTurnOutputTokens, getBudgetContinuationCount, getTotalInputTokens } from '../bootstrap/state.js';
   6  import { parseTokenBudget } from '../utils/tokenBudget.js';
   7  import { count } from '../utils/array.js';
   8  import { dirname, join } from 'path';
   9  import { tmpdir } from 'os';
  10  import figures from 'figures';
  11  // eslint-disable-next-line custom-rules/prefer-use-keybindings -- / n N Esc [ v are bare letters in transcript modal context, same class as g/G/j/k in ScrollKeybindingHandler
  12  import { useInput } from '../ink.js';
  13  import { useSearchInput } from '../hooks/useSearchInput.js';
  14  import { useTerminalSize } from '../hooks/useTerminalSize.js';
  15  import { useSearchHighlight } from '../ink/hooks/use-search-highlight.js';
  16  import type { JumpHandle } from '../components/VirtualMessageList.js';
  17  import { renderMessagesToPlainText } from '../utils/exportRenderer.js';
  18  import { openFileInExternalEditor } from '../utils/editor.js';
  19  import { writeFile } from 'fs/promises';
  20  import { Box, Text, useStdin, useTheme, useTerminalFocus, useTerminalTitle, useTabStatus } from '../ink.js';
  21  import type { TabStatusKind } from '../ink/hooks/use-tab-status.js';
  22  import { CostThresholdDialog } from '../components/CostThresholdDialog.js';
  23  import { IdleReturnDialog } from '../components/IdleReturnDialog.js';
  24  import * as React from 'react';
  25  import { useEffect, useMemo, useRef, useState, useCallback, useDeferredValue, useLayoutEffect, type RefObject } from 'react';
  26  import { useNotifications } from '../context/notifications.js';
  27  import { sendNotification } from '../services/notifier.js';
  28  import { startPreventSleep, stopPreventSleep } from '../services/preventSleep.js';
  29  import { useTerminalNotification } from '../ink/useTerminalNotification.js';
  30  import { hasCursorUpViewportYankBug } from '../ink/terminal.js';
  31  import { createFileStateCacheWithSizeLimit, mergeFileStateCaches, READ_FILE_STATE_CACHE_SIZE } from '../utils/fileStateCache.js';
  32  import { updateLastInteractionTime, getLastInteractionTime, getOriginalCwd, getProjectRoot, getSessionId, switchSession, setCostStateForRestore, getTurnHookDurationMs, getTurnHookCount, resetTurnHookDuration, getTurnToolDurationMs, getTurnToolCount, resetTurnToolDuration, getTurnClassifierDurationMs, getTurnClassifierCount, resetTurnClassifierDuration } from '../bootstrap/state.js';
  33  import { asSessionId, asAgentId } from '../types/ids.js';
  34  import { logForDebugging } from '../utils/debug.js';
  35  import { QueryGuard } from '../utils/QueryGuard.js';
  36  import { isEnvTruthy } from '../utils/envUtils.js';
  37  import { formatTokens, truncateToWidth } from '../utils/format.js';
  38  import { consumeEarlyInput } from '../utils/earlyInput.js';
  39  import { setMemberActive } from '../utils/swarm/teamHelpers.js';
  40  import { isSwarmWorker, generateSandboxRequestId, sendSandboxPermissionRequestViaMailbox, sendSandboxPermissionResponseViaMailbox } from '../utils/swarm/permissionSync.js';
  41  import { registerSandboxPermissionCallback } from '../hooks/useSwarmPermissionPoller.js';
  42  import { getTeamName, getAgentName } from '../utils/teammate.js';
  43  import { WorkerPendingPermission } from '../components/permissions/WorkerPendingPermission.js';
  44  import { injectUserMessageToTeammate, getAllInProcessTeammateTasks } from '../tasks/InProcessTeammateTask/InProcessTeammateTask.js';
  45  import { isLocalAgentTask, queuePendingMessage, appendMessageToLocalAgent, type LocalAgentTaskState } from '../tasks/LocalAgentTask/LocalAgentTask.js';
  46  import { registerLeaderToolUseConfirmQueue, unregisterLeaderToolUseConfirmQueue, registerLeaderSetToolPermissionContext, unregisterLeaderSetToolPermissionContext } from '../utils/swarm/leaderPermissionBridge.js';
  47  import { endInteractionSpan } from '../utils/telemetry/sessionTracing.js';
  48  import { useLogMessages } from '../hooks/useLogMessages.js';
  49  import { useReplBridge } from '../hooks/useReplBridge.js';
  50  import { type Command, type CommandResultDisplay, type ResumeEntrypoint, getCommandName, isCommandEnabled } from '../commands.js';
  51  import type { PromptInputMode, QueuedCommand, VimMode } from '../types/textInputTypes.js';
  52  import { MessageSelector, selectableUserMessagesFilter, messagesAfterAreOnlySynthetic } from '../components/MessageSelector.js';
  53  import { useIdeLogging } from '../hooks/useIdeLogging.js';
  54  import { PermissionRequest, type ToolUseConfirm } from '../components/permissions/PermissionRequest.js';
  55  import { ElicitationDialog } from '../components/mcp/ElicitationDialog.js';
  56  import { PromptDialog } from '../components/hooks/PromptDialog.js';
  57  import type { PromptRequest, PromptResponse } from '../types/hooks.js';
  58  import PromptInput from '../components/PromptInput/PromptInput.js';
  59  import { PromptInputQueuedCommands } from '../components/PromptInput/PromptInputQueuedCommands.js';
  60  import { useRemoteSession } from '../hooks/useRemoteSession.js';
  61  import { useDirectConnect } from '../hooks/useDirectConnect.js';
  62  import type { DirectConnectConfig } from '../server/directConnectManager.js';
  63  import { useSSHSession } from '../hooks/useSSHSession.js';
  64  import { useAssistantHistory } from '../hooks/useAssistantHistory.js';
  65  import type { SSHSession } from '../ssh/createSSHSession.js';
  66  import { SkillImprovementSurvey } from '../components/SkillImprovementSurvey.js';
  67  import { useSkillImprovementSurvey } from '../hooks/useSkillImprovementSurvey.js';
  68  import { useMoreRight } from '../moreright/useMoreRight.js';
  69  import { SpinnerWithVerb, BriefIdleStatus, type SpinnerMode } from '../components/Spinner.js';
  70  import { getSystemPrompt } from '../constants/prompts.js';
  71  import { buildEffectiveSystemPrompt } from '../utils/systemPrompt.js';
  72  import { getSystemContext, getUserContext } from '../context.js';
  73  import { getMemoryFiles } from '../utils/claudemd.js';
  74  import { startBackgroundHousekeeping } from '../utils/backgroundHousekeeping.js';
  75  import { getTotalCost, saveCurrentSessionCosts, resetCostState, getStoredSessionCosts } from '../cost-tracker.js';
  76  import { useCostSummary } from '../costHook.js';
  77  import { useFpsMetrics } from '../context/fpsMetrics.js';
  78  import { useAfterFirstRender } from '../hooks/useAfterFirstRender.js';
  79  import { useDeferredHookMessages } from '../hooks/useDeferredHookMessages.js';
  80  import { addToHistory, removeLastFromHistory, expandPastedTextRefs, parseReferences } from '../history.js';
  81  import { prependModeCharacterToInput } from '../components/PromptInput/inputModes.js';
  82  import { prependToShellHistoryCache } from '../utils/suggestions/shellHistoryCompletion.js';
  83  import { useApiKeyVerification } from '../hooks/useApiKeyVerification.js';
  84  import { GlobalKeybindingHandlers } from '../hooks/useGlobalKeybindings.js';
  85  import { CommandKeybindingHandlers } from '../hooks/useCommandKeybindings.js';
  86  import { KeybindingSetup } from '../keybindings/KeybindingProviderSetup.js';
  87  import { useShortcutDisplay } from '../keybindings/useShortcutDisplay.js';
  88  import { getShortcutDisplay } from '../keybindings/shortcutFormat.js';
  89  import { CancelRequestHandler } from '../hooks/useCancelRequest.js';
  90  import { useBackgroundTaskNavigation } from '../hooks/useBackgroundTaskNavigation.js';
  91  import { useSwarmInitialization } from '../hooks/useSwarmInitialization.js';
  92  import { useTeammateViewAutoExit } from '../hooks/useTeammateViewAutoExit.js';
  93  import { errorMessage } from '../utils/errors.js';
  94  import { isHumanTurn } from '../utils/messagePredicates.js';
  95  import { logError } from '../utils/log.js';
  96  // Dead code elimination: conditional imports
  97  /* eslint-disable custom-rules/no-process-env-top-level, @typescript-eslint/no-require-imports */
  98  const useVoiceIntegration: typeof import('../hooks/useVoiceIntegration.js').useVoiceIntegration = feature('VOICE_MODE') ? require('../hooks/useVoiceIntegration.js').useVoiceIntegration : () => ({
  99    stripTrailing: () => 0,
 100    handleKeyEvent: () => {},
 101    resetAnchor: () => {}
 102  });
 103  const VoiceKeybindingHandler: typeof import('../hooks/useVoiceIntegration.js').VoiceKeybindingHandler = feature('VOICE_MODE') ? require('../hooks/useVoiceIntegration.js').VoiceKeybindingHandler : () => null;
 104  // Frustration detection is ant-only (dogfooding). Conditional require so external
 105  // builds eliminate the module entirely (including its two O(n) useMemos that run
 106  // on every messages change, plus the GrowthBook fetch).
 107  const useFrustrationDetection: typeof import('../components/FeedbackSurvey/useFrustrationDetection.js').useFrustrationDetection = "external" === 'ant' ? require('../components/FeedbackSurvey/useFrustrationDetection.js').useFrustrationDetection : () => ({
 108    state: 'closed',
 109    handleTranscriptSelect: () => {}
 110  });
 111  // Ant-only org warning. Conditional require so the org UUID list is
 112  // eliminated from external builds (one UUID is on excluded-strings).
 113  const useAntOrgWarningNotification: typeof import('../hooks/notifs/useAntOrgWarningNotification.js').useAntOrgWarningNotification = "external" === 'ant' ? require('../hooks/notifs/useAntOrgWarningNotification.js').useAntOrgWarningNotification : () => {};
 114  // Dead code elimination: conditional import for coordinator mode
 115  const getCoordinatorUserContext: (mcpClients: ReadonlyArray<{
 116    name: string;
 117  }>, scratchpadDir?: string) => {
 118    [k: string]: string;
 119  } = feature('COORDINATOR_MODE') ? require('../coordinator/coordinatorMode.js').getCoordinatorUserContext : () => ({});
 120  /* eslint-enable custom-rules/no-process-env-top-level, @typescript-eslint/no-require-imports */
 121  import useCanUseTool from '../hooks/useCanUseTool.js';
 122  import type { ToolPermissionContext, Tool } from '../Tool.js';
 123  import { applyPermissionUpdate, applyPermissionUpdates, persistPermissionUpdate } from '../utils/permissions/PermissionUpdate.js';
 124  import { buildPermissionUpdates } from '../components/permissions/ExitPlanModePermissionRequest/ExitPlanModePermissionRequest.js';
 125  import { stripDangerousPermissionsForAutoMode } from '../utils/permissions/permissionSetup.js';
 126  import { getScratchpadDir, isScratchpadEnabled } from '../utils/permissions/filesystem.js';
 127  import { WEB_FETCH_TOOL_NAME } from '../tools/WebFetchTool/prompt.js';
 128  import { SLEEP_TOOL_NAME } from '../tools/SleepTool/prompt.js';
 129  import { clearSpeculativeChecks } from '../tools/BashTool/bashPermissions.js';
 130  import type { AutoUpdaterResult } from '../utils/autoUpdater.js';
 131  import { getGlobalConfig, saveGlobalConfig, getGlobalConfigWriteCount } from '../utils/config.js';
 132  import { hasConsoleBillingAccess } from '../utils/billing.js';
 133  import { logEvent, type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS } from 'src/services/analytics/index.js';
 134  import { getFeatureValue_CACHED_MAY_BE_STALE } from 'src/services/analytics/growthbook.js';
 135  import { textForResubmit, handleMessageFromStream, type StreamingToolUse, type StreamingThinking, isCompactBoundaryMessage, getMessagesAfterCompactBoundary, getContentText, createUserMessage, createAssistantMessage, createTurnDurationMessage, createAgentsKilledMessage, createApiMetricsMessage, createSystemMessage, createCommandInputMessage, formatCommandInputTags } from '../utils/messages.js';
 136  import { generateSessionTitle } from '../utils/sessionTitle.js';
 137  import { BASH_INPUT_TAG, COMMAND_MESSAGE_TAG, COMMAND_NAME_TAG, LOCAL_COMMAND_STDOUT_TAG } from '../constants/xml.js';
 138  import { escapeXml } from '../utils/xml.js';
 139  import type { ThinkingConfig } from '../utils/thinking.js';
 140  import { gracefulShutdownSync } from '../utils/gracefulShutdown.js';
 141  import { handlePromptSubmit, type PromptInputHelpers } from '../utils/handlePromptSubmit.js';
 142  import { useQueueProcessor } from '../hooks/useQueueProcessor.js';
 143  import { useMailboxBridge } from '../hooks/useMailboxBridge.js';
 144  import { queryCheckpoint, logQueryProfileReport } from '../utils/queryProfiler.js';
 145  import type { Message as MessageType, UserMessage, ProgressMessage, HookResultMessage, PartialCompactDirection } from '../types/message.js';
 146  import { query } from '../query.js';
 147  import { mergeClients, useMergedClients } from '../hooks/useMergedClients.js';
 148  import { getQuerySourceForREPL } from '../utils/promptCategory.js';
 149  import { useMergedTools } from '../hooks/useMergedTools.js';
 150  import { mergeAndFilterTools } from '../utils/toolPool.js';
 151  import { useMergedCommands } from '../hooks/useMergedCommands.js';
 152  import { useSkillsChange } from '../hooks/useSkillsChange.js';
 153  import { useManagePlugins } from '../hooks/useManagePlugins.js';
 154  import { Messages } from '../components/Messages.js';
 155  import { TaskListV2 } from '../components/TaskListV2.js';
 156  import { TeammateViewHeader } from '../components/TeammateViewHeader.js';
 157  import { useTasksV2WithCollapseEffect } from '../hooks/useTasksV2.js';
 158  import { maybeMarkProjectOnboardingComplete } from '../projectOnboardingState.js';
 159  import type { MCPServerConnection } from '../services/mcp/types.js';
 160  import type { ScopedMcpServerConfig } from '../services/mcp/types.js';
 161  import { randomUUID, type UUID } from 'crypto';
 162  import { processSessionStartHooks } from '../utils/sessionStart.js';
 163  import { executeSessionEndHooks, getSessionEndHookTimeoutMs } from '../utils/hooks.js';
 164  import { type IDESelection, useIdeSelection } from '../hooks/useIdeSelection.js';
 165  import { getTools, assembleToolPool } from '../tools.js';
 166  import type { AgentDefinition } from '../tools/AgentTool/loadAgentsDir.js';
 167  import { resolveAgentTools } from '../tools/AgentTool/agentToolUtils.js';
 168  import { resumeAgentBackground } from '../tools/AgentTool/resumeAgent.js';
 169  import { useMainLoopModel } from '../hooks/useMainLoopModel.js';
 170  import { useAppState, useSetAppState, useAppStateStore } from '../state/AppState.js';
 171  import type { ContentBlockParam, ImageBlockParam } from '@anthropic-ai/sdk/resources/messages.mjs';
 172  import type { ProcessUserInputContext } from '../utils/processUserInput/processUserInput.js';
 173  import type { PastedContent } from '../utils/config.js';
 174  import { copyPlanForFork, copyPlanForResume, getPlanSlug, setPlanSlug } from '../utils/plans.js';
 175  import { clearSessionMetadata, resetSessionFilePointer, adoptResumedSessionFile, removeTranscriptMessage, restoreSessionMetadata, getCurrentSessionTitle, isEphemeralToolProgress, isLoggableMessage, saveWorktreeState, getAgentTranscript } from '../utils/sessionStorage.js';
 176  import { deserializeMessages } from '../utils/conversationRecovery.js';
 177  import { extractReadFilesFromMessages, extractBashToolsFromMessages } from '../utils/queryHelpers.js';
 178  import { resetMicrocompactState } from '../services/compact/microCompact.js';
 179  import { runPostCompactCleanup } from '../services/compact/postCompactCleanup.js';
 180  import { provisionContentReplacementState, reconstructContentReplacementState, type ContentReplacementRecord } from '../utils/toolResultStorage.js';
 181  import { partialCompactConversation } from '../services/compact/compact.js';
 182  import type { LogOption } from '../types/logs.js';
 183  import type { AgentColorName } from '../tools/AgentTool/agentColorManager.js';
 184  import { fileHistoryMakeSnapshot, type FileHistoryState, fileHistoryRewind, type FileHistorySnapshot, copyFileHistoryForResume, fileHistoryEnabled, fileHistoryHasAnyChanges } from '../utils/fileHistory.js';
 185  import { type AttributionState, incrementPromptCount } from '../utils/commitAttribution.js';
 186  import { recordAttributionSnapshot } from '../utils/sessionStorage.js';
 187  import { computeStandaloneAgentContext, restoreAgentFromSession, restoreSessionStateFromLog, restoreWorktreeForResume, exitRestoredWorktree } from '../utils/sessionRestore.js';
 188  import { isBgSession, updateSessionName, updateSessionActivity } from '../utils/concurrentSessions.js';
 189  import { isInProcessTeammateTask, type InProcessTeammateTaskState } from '../tasks/InProcessTeammateTask/types.js';
 190  import { restoreRemoteAgentTasks } from '../tasks/RemoteAgentTask/RemoteAgentTask.js';
 191  import { useInboxPoller } from '../hooks/useInboxPoller.js';
 192  // Dead code elimination: conditional import for loop mode
 193  /* eslint-disable @typescript-eslint/no-require-imports */
 194  const proactiveModule = feature('PROACTIVE') || feature('KAIROS') ? require('../proactive/index.js') : null;
 195  const PROACTIVE_NO_OP_SUBSCRIBE = (_cb: () => void) => () => {};
 196  const PROACTIVE_FALSE = () => false;
 197  const SUGGEST_BG_PR_NOOP = (_p: string, _n: string): boolean => false;
 198  const useProactive = feature('PROACTIVE') || feature('KAIROS') ? require('../proactive/useProactive.js').useProactive : null;
 199  const useScheduledTasks = feature('AGENT_TRIGGERS') ? require('../hooks/useScheduledTasks.js').useScheduledTasks : null;
 200  /* eslint-enable @typescript-eslint/no-require-imports */
 201  import { isAgentSwarmsEnabled } from '../utils/agentSwarmsEnabled.js';
 202  import { useTaskListWatcher } from '../hooks/useTaskListWatcher.js';
 203  import type { SandboxAskCallback, NetworkHostPattern } from '../utils/sandbox/sandbox-adapter.js';
 204  import { type IDEExtensionInstallationStatus, closeOpenDiffs, getConnectedIdeClient, type IdeType } from '../utils/ide.js';
 205  import { useIDEIntegration } from '../hooks/useIDEIntegration.js';
 206  import exit from '../commands/exit/index.js';
 207  import { ExitFlow } from '../components/ExitFlow.js';
 208  import { getCurrentWorktreeSession } from '../utils/worktree.js';
 209  import { popAllEditable, enqueue, type SetAppState, getCommandQueue, getCommandQueueLength, removeByFilter } from '../utils/messageQueueManager.js';
 210  import { useCommandQueue } from '../hooks/useCommandQueue.js';
 211  import { SessionBackgroundHint } from '../components/SessionBackgroundHint.js';
 212  import { startBackgroundSession } from '../tasks/LocalMainSessionTask.js';
 213  import { useSessionBackgrounding } from '../hooks/useSessionBackgrounding.js';
 214  import { diagnosticTracker } from '../services/diagnosticTracking.js';
 215  import { handleSpeculationAccept, type ActiveSpeculationState } from '../services/PromptSuggestion/speculation.js';
 216  import { IdeOnboardingDialog } from '../components/IdeOnboardingDialog.js';
 217  import { EffortCallout, shouldShowEffortCallout } from '../components/EffortCallout.js';
 218  import type { EffortValue } from '../utils/effort.js';
 219  import { RemoteCallout } from '../components/RemoteCallout.js';
 220  /* eslint-disable custom-rules/no-process-env-top-level, @typescript-eslint/no-require-imports */
 221  const AntModelSwitchCallout = "external" === 'ant' ? require('../components/AntModelSwitchCallout.js').AntModelSwitchCallout : null;
 222  const shouldShowAntModelSwitch = "external" === 'ant' ? require('../components/AntModelSwitchCallout.js').shouldShowModelSwitchCallout : (): boolean => false;
 223  const UndercoverAutoCallout = "external" === 'ant' ? require('../components/UndercoverAutoCallout.js').UndercoverAutoCallout : null;
 224  /* eslint-enable custom-rules/no-process-env-top-level, @typescript-eslint/no-require-imports */
 225  import { activityManager } from '../utils/activityManager.js';
 226  import { createAbortController } from '../utils/abortController.js';
 227  import { MCPConnectionManager } from 'src/services/mcp/MCPConnectionManager.js';
 228  import { useFeedbackSurvey } from 'src/components/FeedbackSurvey/useFeedbackSurvey.js';
 229  import { useMemorySurvey } from 'src/components/FeedbackSurvey/useMemorySurvey.js';
 230  import { usePostCompactSurvey } from 'src/components/FeedbackSurvey/usePostCompactSurvey.js';
 231  import { FeedbackSurvey } from 'src/components/FeedbackSurvey/FeedbackSurvey.js';
 232  import { useInstallMessages } from 'src/hooks/notifs/useInstallMessages.js';
 233  import { useAwaySummary } from 'src/hooks/useAwaySummary.js';
 234  import { useChromeExtensionNotification } from 'src/hooks/useChromeExtensionNotification.js';
 235  import { useOfficialMarketplaceNotification } from 'src/hooks/useOfficialMarketplaceNotification.js';
 236  import { usePromptsFromClaudeInChrome } from 'src/hooks/usePromptsFromClaudeInChrome.js';
 237  import { getTipToShowOnSpinner, recordShownTip } from 'src/services/tips/tipScheduler.js';
 238  import type { Theme } from 'src/utils/theme.js';
 239  import { checkAndDisableBypassPermissionsIfNeeded, checkAndDisableAutoModeIfNeeded, useKickOffCheckAndDisableBypassPermissionsIfNeeded, useKickOffCheckAndDisableAutoModeIfNeeded } from 'src/utils/permissions/bypassPermissionsKillswitch.js';
 240  import { SandboxManager } from 'src/utils/sandbox/sandbox-adapter.js';
 241  import { SANDBOX_NETWORK_ACCESS_TOOL_NAME } from 'src/cli/structuredIO.js';
 242  import { useFileHistorySnapshotInit } from 'src/hooks/useFileHistorySnapshotInit.js';
 243  import { SandboxPermissionRequest } from 'src/components/permissions/SandboxPermissionRequest.js';
 244  import { SandboxViolationExpandedView } from 'src/components/SandboxViolationExpandedView.js';
 245  import { useSettingsErrors } from 'src/hooks/notifs/useSettingsErrors.js';
 246  import { useMcpConnectivityStatus } from 'src/hooks/notifs/useMcpConnectivityStatus.js';
 247  import { useAutoModeUnavailableNotification } from 'src/hooks/notifs/useAutoModeUnavailableNotification.js';
 248  import { AUTO_MODE_DESCRIPTION } from 'src/components/AutoModeOptInDialog.js';
 249  import { useLspInitializationNotification } from 'src/hooks/notifs/useLspInitializationNotification.js';
 250  import { useLspPluginRecommendation } from 'src/hooks/useLspPluginRecommendation.js';
 251  import { LspRecommendationMenu } from 'src/components/LspRecommendation/LspRecommendationMenu.js';
 252  import { useClaudeCodeHintRecommendation } from 'src/hooks/useClaudeCodeHintRecommendation.js';
 253  import { PluginHintMenu } from 'src/components/ClaudeCodeHint/PluginHintMenu.js';
 254  import { DesktopUpsellStartup, shouldShowDesktopUpsellStartup } from 'src/components/DesktopUpsell/DesktopUpsellStartup.js';
 255  import { usePluginInstallationStatus } from 'src/hooks/notifs/usePluginInstallationStatus.js';
 256  import { usePluginAutoupdateNotification } from 'src/hooks/notifs/usePluginAutoupdateNotification.js';
 257  import { performStartupChecks } from 'src/utils/plugins/performStartupChecks.js';
 258  import { UserTextMessage } from 'src/components/messages/UserTextMessage.js';
 259  import { AwsAuthStatusBox } from '../components/AwsAuthStatusBox.js';
 260  import { useRateLimitWarningNotification } from 'src/hooks/notifs/useRateLimitWarningNotification.js';
 261  import { useDeprecationWarningNotification } from 'src/hooks/notifs/useDeprecationWarningNotification.js';
 262  import { useNpmDeprecationNotification } from 'src/hooks/notifs/useNpmDeprecationNotification.js';
 263  import { useIDEStatusIndicator } from 'src/hooks/notifs/useIDEStatusIndicator.js';
 264  import { useModelMigrationNotifications } from 'src/hooks/notifs/useModelMigrationNotifications.js';
 265  import { useCanSwitchToExistingSubscription } from 'src/hooks/notifs/useCanSwitchToExistingSubscription.js';
 266  import { useTeammateLifecycleNotification } from 'src/hooks/notifs/useTeammateShutdownNotification.js';
 267  import { useFastModeNotification } from 'src/hooks/notifs/useFastModeNotification.js';
 268  import { AutoRunIssueNotification, shouldAutoRunIssue, getAutoRunIssueReasonText, getAutoRunCommand, type AutoRunIssueReason } from '../utils/autoRunIssue.js';
 269  import type { HookProgress } from '../types/hooks.js';
 270  import { TungstenLiveMonitor } from '../tools/TungstenTool/TungstenLiveMonitor.js';
 271  /* eslint-disable @typescript-eslint/no-require-imports */
 272  const WebBrowserPanelModule = feature('WEB_BROWSER_TOOL') ? require('../tools/WebBrowserTool/WebBrowserPanel.js') as typeof import('../tools/WebBrowserTool/WebBrowserPanel.js') : null;
 273  /* eslint-enable @typescript-eslint/no-require-imports */
 274  import { IssueFlagBanner } from '../components/PromptInput/IssueFlagBanner.js';
 275  import { useIssueFlagBanner } from '../hooks/useIssueFlagBanner.js';
 276  import { CompanionSprite, CompanionFloatingBubble, MIN_COLS_FOR_FULL_SPRITE } from '../buddy/CompanionSprite.js';
 277  import { DevBar } from '../components/DevBar.js';
 278  // Session manager removed - using AppState now
 279  import type { RemoteSessionConfig } from '../remote/RemoteSessionManager.js';
 280  import { REMOTE_SAFE_COMMANDS } from '../commands.js';
 281  import type { RemoteMessageContent } from '../utils/teleport/api.js';
 282  import { FullscreenLayout, useUnseenDivider, computeUnseenDivider } from '../components/FullscreenLayout.js';
 283  import { isFullscreenEnvEnabled, maybeGetTmuxMouseHint, isMouseTrackingEnabled } from '../utils/fullscreen.js';
 284  import { AlternateScreen } from '../ink/components/AlternateScreen.js';
 285  import { ScrollKeybindingHandler } from '../components/ScrollKeybindingHandler.js';
 286  import { useMessageActions, MessageActionsKeybindings, MessageActionsBar, type MessageActionsState, type MessageActionsNav, type MessageActionCaps } from '../components/messageActions.js';
 287  import { setClipboard } from '../ink/termio/osc.js';
 288  import type { ScrollBoxHandle } from '../ink/components/ScrollBox.js';
 289  import { createAttachmentMessage, getQueuedCommandAttachments } from '../utils/attachments.js';
 290  
 291  // Stable empty array for hooks that accept MCPServerConnection[] — avoids
 292  // creating a new [] literal on every render in remote mode, which would
 293  // cause useEffect dependency changes and infinite re-render loops.
 294  const EMPTY_MCP_CLIENTS: MCPServerConnection[] = [];
 295  
 296  // Stable stub for useAssistantHistory's non-KAIROS branch — avoids a new
 297  // function identity each render, which would break composedOnScroll's memo.
 298  const HISTORY_STUB = {
 299    maybeLoadOlder: (_: ScrollBoxHandle) => {}
 300  };
 301  // Window after a user-initiated scroll during which type-into-empty does NOT
 302  // repin to bottom. Josh Rosen's workflow: Claude emits long output → scroll
 303  // up to read the start → start typing → before this fix, snapped to bottom.
 304  // https://anthropic.slack.com/archives/C07VBSHV7EV/p1773545449871739
 305  const RECENT_SCROLL_REPIN_WINDOW_MS = 3000;
 306  
 307  // Use LRU cache to prevent unbounded memory growth
 308  // 100 files should be sufficient for most coding sessions while preventing
 309  // memory issues when working across many files in large projects
 310  
 311  function median(values: number[]): number {
 312    const sorted = [...values].sort((a, b) => a - b);
 313    const mid = Math.floor(sorted.length / 2);
 314    return sorted.length % 2 === 0 ? Math.round((sorted[mid - 1]! + sorted[mid]!) / 2) : sorted[mid]!;
 315  }
 316  
 317  /**
 318   * Small component to display transcript mode footer with dynamic keybinding.
 319   * Must be rendered inside KeybindingSetup to access keybinding context.
 320   */
 321  function TranscriptModeFooter(t0) {
 322    const $ = _c(9);
 323    const {
 324      showAllInTranscript,
 325      virtualScroll,
 326      searchBadge,
 327      suppressShowAll: t1,
 328      status
 329    } = t0;
 330    const suppressShowAll = t1 === undefined ? false : t1;
 331    const toggleShortcut = useShortcutDisplay("app:toggleTranscript", "Global", "ctrl+o");
 332    const showAllShortcut = useShortcutDisplay("transcript:toggleShowAll", "Transcript", "ctrl+e");
 333    const t2 = searchBadge ? " \xB7 n/N to navigate" : virtualScroll ? ` · ${figures.arrowUp}${figures.arrowDown} scroll · home/end top/bottom` : suppressShowAll ? "" : ` · ${showAllShortcut} to ${showAllInTranscript ? "collapse" : "show all"}`;
 334    let t3;
 335    if ($[0] !== t2 || $[1] !== toggleShortcut) {
 336      t3 = <Text dimColor={true}>Showing detailed transcript · {toggleShortcut} to toggle{t2}</Text>;
 337      $[0] = t2;
 338      $[1] = toggleShortcut;
 339      $[2] = t3;
 340    } else {
 341      t3 = $[2];
 342    }
 343    let t4;
 344    if ($[3] !== searchBadge || $[4] !== status) {
 345      t4 = status ? <><Box flexGrow={1} /><Text>{status} </Text></> : searchBadge ? <><Box flexGrow={1} /><Text dimColor={true}>{searchBadge.current}/{searchBadge.count}{"  "}</Text></> : null;
 346      $[3] = searchBadge;
 347      $[4] = status;
 348      $[5] = t4;
 349    } else {
 350      t4 = $[5];
 351    }
 352    let t5;
 353    if ($[6] !== t3 || $[7] !== t4) {
 354      t5 = <Box noSelect={true} alignItems="center" alignSelf="center" borderTopDimColor={true} borderBottom={false} borderLeft={false} borderRight={false} borderStyle="single" marginTop={1} paddingLeft={2} width="100%">{t3}{t4}</Box>;
 355      $[6] = t3;
 356      $[7] = t4;
 357      $[8] = t5;
 358    } else {
 359      t5 = $[8];
 360    }
 361    return t5;
 362  }
 363  
 364  /** less-style / bar. 1-row, same border-top styling as TranscriptModeFooter
 365   *  so swapping them in the bottom slot doesn't shift ScrollBox height.
 366   *  useSearchInput handles readline editing; we report query changes and
 367   *  render the counter. Incremental — re-search + highlight per keystroke. */
 368  function TranscriptSearchBar({
 369    jumpRef,
 370    count,
 371    current,
 372    onClose,
 373    onCancel,
 374    setHighlight,
 375    initialQuery
 376  }: {
 377    jumpRef: RefObject<JumpHandle | null>;
 378    count: number;
 379    current: number;
 380    /** Enter — commit. Query persists for n/N. */
 381    onClose: (lastQuery: string) => void;
 382    /** Esc/ctrl+c/ctrl+g — undo to pre-/ state. */
 383    onCancel: () => void;
 384    setHighlight: (query: string) => void;
 385    // Seed with the previous query (less: / shows last pattern). Mount-fire
 386    // of the effect re-scans with the same query — idempotent (same matches,
 387    // nearest-ptr, same highlights). User can edit or clear.
 388    initialQuery: string;
 389  }): React.ReactNode {
 390    const {
 391      query,
 392      cursorOffset
 393    } = useSearchInput({
 394      isActive: true,
 395      initialQuery,
 396      onExit: () => onClose(query),
 397      onCancel
 398    });
 399    // Index warm-up runs before the query effect so it measures the real
 400    // cost — otherwise setSearchQuery fills the cache first and warm
 401    // reports ~0ms while the user felt the actual lag.
 402    // First / in a transcript session pays the extractSearchText cost.
 403    // Subsequent / return 0 immediately (indexWarmed ref in VML).
 404    // Transcript is frozen at ctrl+o so the cache stays valid.
 405    // Initial 'building' so warmDone is false on mount — the [query] effect
 406    // waits for the warm effect's first resolve instead of racing it. With
 407    // null initial, warmDone would be true on mount → [query] fires →
 408    // setSearchQuery fills cache → warm reports ~0ms while the user felt
 409    // the real lag.
 410    const [indexStatus, setIndexStatus] = React.useState<'building' | {
 411      ms: number;
 412    } | null>('building');
 413    React.useEffect(() => {
 414      let alive = true;
 415      const warm = jumpRef.current?.warmSearchIndex;
 416      if (!warm) {
 417        setIndexStatus(null); // VML not mounted yet — rare, skip indicator
 418        return;
 419      }
 420      setIndexStatus('building');
 421      warm().then(ms => {
 422        if (!alive) return;
 423        // <20ms = imperceptible. No point showing "indexed in 3ms".
 424        if (ms < 20) {
 425          setIndexStatus(null);
 426        } else {
 427          setIndexStatus({
 428            ms
 429          });
 430          setTimeout(() => alive && setIndexStatus(null), 2000);
 431        }
 432      });
 433      return () => {
 434        alive = false;
 435      };
 436      // eslint-disable-next-line react-hooks/exhaustive-deps
 437    }, []); // mount-only: bar opens once per /
 438    // Gate the query effect on warm completion. setHighlight stays instant
 439    // (screen-space overlay, no indexing). setSearchQuery (the scan) waits.
 440    const warmDone = indexStatus !== 'building';
 441    useEffect(() => {
 442      if (!warmDone) return;
 443      jumpRef.current?.setSearchQuery(query);
 444      setHighlight(query);
 445      // eslint-disable-next-line react-hooks/exhaustive-deps
 446    }, [query, warmDone]);
 447    const off = cursorOffset;
 448    const cursorChar = off < query.length ? query[off] : ' ';
 449    return <Box borderTopDimColor borderBottom={false} borderLeft={false} borderRight={false} borderStyle="single" marginTop={1} paddingLeft={2} width="100%"
 450    // applySearchHighlight scans the whole screen buffer. The query
 451    // text rendered here IS on screen — /foo matches its own 'foo' in
 452    // the bar. With no content matches that's the ONLY visible match →
 453    // gets CURRENT → underlined. noSelect makes searchHighlight.ts:76
 454    // skip these cells (same exclusion as gutters). You can't text-
 455    // select the bar either; it's transient chrome, fine.
 456    noSelect>
 457        <Text>/</Text>
 458        <Text>{query.slice(0, off)}</Text>
 459        <Text inverse>{cursorChar}</Text>
 460        {off < query.length && <Text>{query.slice(off + 1)}</Text>}
 461        <Box flexGrow={1} />
 462        {indexStatus === 'building' ? <Text dimColor>indexing… </Text> : indexStatus ? <Text dimColor>indexed in {indexStatus.ms}ms </Text> : count === 0 && query ? <Text color="error">no matches </Text> : count > 0 ?
 463      // Engine-counted (indexOf on extractSearchText). May drift from
 464      // render-count for ghost/phantom messages — badge is a rough
 465      // location hint. scanElement gives exact per-message positions
 466      // but counting ALL would cost ~1-3ms × matched-messages.
 467      <Text dimColor>
 468            {current}/{count}
 469            {'  '}
 470          </Text> : null}
 471      </Box>;
 472  }
 473  const TITLE_ANIMATION_FRAMES = ['⠂', '⠐'];
 474  const TITLE_STATIC_PREFIX = '✳';
 475  const TITLE_ANIMATION_INTERVAL_MS = 960;
 476  
 477  /**
 478   * Sets the terminal tab title, with an animated prefix glyph while a query
 479   * is running. Isolated from REPL so the 960ms animation tick re-renders only
 480   * this leaf component (which returns null — pure side-effect) instead of the
 481   * entire REPL tree. Before extraction, the tick was ~1 REPL render/sec for
 482   * the duration of every turn, dragging PromptInput and friends along.
 483   */
 484  function AnimatedTerminalTitle(t0) {
 485    const $ = _c(6);
 486    const {
 487      isAnimating,
 488      title,
 489      disabled,
 490      noPrefix
 491    } = t0;
 492    const terminalFocused = useTerminalFocus();
 493    const [frame, setFrame] = useState(0);
 494    let t1;
 495    let t2;
 496    if ($[0] !== disabled || $[1] !== isAnimating || $[2] !== noPrefix || $[3] !== terminalFocused) {
 497      t1 = () => {
 498        if (disabled || noPrefix || !isAnimating || !terminalFocused) {
 499          return;
 500        }
 501        const interval = setInterval(_temp2, TITLE_ANIMATION_INTERVAL_MS, setFrame);
 502        return () => clearInterval(interval);
 503      };
 504      t2 = [disabled, noPrefix, isAnimating, terminalFocused];
 505      $[0] = disabled;
 506      $[1] = isAnimating;
 507      $[2] = noPrefix;
 508      $[3] = terminalFocused;
 509      $[4] = t1;
 510      $[5] = t2;
 511    } else {
 512      t1 = $[4];
 513      t2 = $[5];
 514    }
 515    useEffect(t1, t2);
 516    const prefix = isAnimating ? TITLE_ANIMATION_FRAMES[frame] ?? TITLE_STATIC_PREFIX : TITLE_STATIC_PREFIX;
 517    useTerminalTitle(disabled ? null : noPrefix ? title : `${prefix} ${title}`);
 518    return null;
 519  }
 520  function _temp2(setFrame_0) {
 521    return setFrame_0(_temp);
 522  }
 523  function _temp(f) {
 524    return (f + 1) % TITLE_ANIMATION_FRAMES.length;
 525  }
 526  export type Props = {
 527    commands: Command[];
 528    debug: boolean;
 529    initialTools: Tool[];
 530    // Initial messages to populate the REPL with
 531    initialMessages?: MessageType[];
 532    // Deferred hook messages promise — REPL renders immediately and injects
 533    // hook messages when they resolve. Awaited before the first API call.
 534    pendingHookMessages?: Promise<HookResultMessage[]>;
 535    initialFileHistorySnapshots?: FileHistorySnapshot[];
 536    // Content-replacement records from a resumed session's transcript — used to
 537    // reconstruct contentReplacementState so the same results are re-replaced
 538    initialContentReplacements?: ContentReplacementRecord[];
 539    // Initial agent context for session resume (name/color set via /rename or /color)
 540    initialAgentName?: string;
 541    initialAgentColor?: AgentColorName;
 542    mcpClients?: MCPServerConnection[];
 543    dynamicMcpConfig?: Record<string, ScopedMcpServerConfig>;
 544    autoConnectIdeFlag?: boolean;
 545    strictMcpConfig?: boolean;
 546    systemPrompt?: string;
 547    appendSystemPrompt?: string;
 548    // Optional callback invoked before query execution
 549    // Called after user message is added to conversation but before API call
 550    // Return false to prevent query execution
 551    onBeforeQuery?: (input: string, newMessages: MessageType[]) => Promise<boolean>;
 552    // Optional callback when a turn completes (model finishes responding)
 553    onTurnComplete?: (messages: MessageType[]) => void | Promise<void>;
 554    // When true, disables REPL input (hides prompt and prevents message selector)
 555    disabled?: boolean;
 556    // Optional agent definition to use for the main thread
 557    mainThreadAgentDefinition?: AgentDefinition;
 558    // When true, disables all slash commands
 559    disableSlashCommands?: boolean;
 560    // Task list id: when set, enables tasks mode that watches a task list and auto-processes tasks.
 561    taskListId?: string;
 562    // Remote session config for --remote mode (uses CCR as execution engine)
 563    remoteSessionConfig?: RemoteSessionConfig;
 564    // Direct connect config for `claude connect` mode (connects to a claude server)
 565    directConnectConfig?: DirectConnectConfig;
 566    // SSH session for `claude ssh` mode (local REPL, remote tools over ssh)
 567    sshSession?: SSHSession;
 568    // Thinking configuration to use when thinking is enabled
 569    thinkingConfig: ThinkingConfig;
 570  };
 571  export type Screen = 'prompt' | 'transcript';
 572  export function REPL({
 573    commands: initialCommands,
 574    debug,
 575    initialTools,
 576    initialMessages,
 577    pendingHookMessages,
 578    initialFileHistorySnapshots,
 579    initialContentReplacements,
 580    initialAgentName,
 581    initialAgentColor,
 582    mcpClients: initialMcpClients,
 583    dynamicMcpConfig: initialDynamicMcpConfig,
 584    autoConnectIdeFlag,
 585    strictMcpConfig = false,
 586    systemPrompt: customSystemPrompt,
 587    appendSystemPrompt,
 588    onBeforeQuery,
 589    onTurnComplete,
 590    disabled = false,
 591    mainThreadAgentDefinition: initialMainThreadAgentDefinition,
 592    disableSlashCommands = false,
 593    taskListId,
 594    remoteSessionConfig,
 595    directConnectConfig,
 596    sshSession,
 597    thinkingConfig
 598  }: Props): React.ReactNode {
 599    const isRemoteSession = !!remoteSessionConfig;
 600  
 601    // Env-var gates hoisted to mount-time — isEnvTruthy does toLowerCase+trim+
 602    // includes, and these were on the render path (hot during PageUp spam).
 603    const titleDisabled = useMemo(() => isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_TERMINAL_TITLE), []);
 604    const moreRightEnabled = useMemo(() => "external" === 'ant' && isEnvTruthy(process.env.CLAUDE_MORERIGHT), []);
 605    const disableVirtualScroll = useMemo(() => isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_VIRTUAL_SCROLL), []);
 606    const disableMessageActions = feature('MESSAGE_ACTIONS') ?
 607    // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
 608    useMemo(() => isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_MESSAGE_ACTIONS), []) : false;
 609  
 610    // Log REPL mount/unmount lifecycle
 611    useEffect(() => {
 612      logForDebugging(`[REPL:mount] REPL mounted, disabled=${disabled}`);
 613      return () => logForDebugging(`[REPL:unmount] REPL unmounting`);
 614    }, [disabled]);
 615  
 616    // Agent definition is state so /resume can update it mid-session
 617    const [mainThreadAgentDefinition, setMainThreadAgentDefinition] = useState(initialMainThreadAgentDefinition);
 618    const toolPermissionContext = useAppState(s => s.toolPermissionContext);
 619    const verbose = useAppState(s => s.verbose);
 620    const mcp = useAppState(s => s.mcp);
 621    const plugins = useAppState(s => s.plugins);
 622    const agentDefinitions = useAppState(s => s.agentDefinitions);
 623    const fileHistory = useAppState(s => s.fileHistory);
 624    const initialMessage = useAppState(s => s.initialMessage);
 625    const queuedCommands = useCommandQueue();
 626    // feature() is a build-time constant — dead code elimination removes the hook
 627    // call entirely in external builds, so this is safe despite looking conditional.
 628    // These fields contain excluded strings that must not appear in external builds.
 629    const spinnerTip = useAppState(s => s.spinnerTip);
 630    const showExpandedTodos = useAppState(s => s.expandedView) === 'tasks';
 631    const pendingWorkerRequest = useAppState(s => s.pendingWorkerRequest);
 632    const pendingSandboxRequest = useAppState(s => s.pendingSandboxRequest);
 633    const teamContext = useAppState(s => s.teamContext);
 634    const tasks = useAppState(s => s.tasks);
 635    const workerSandboxPermissions = useAppState(s => s.workerSandboxPermissions);
 636    const elicitation = useAppState(s => s.elicitation);
 637    const ultraplanPendingChoice = useAppState(s => s.ultraplanPendingChoice);
 638    const ultraplanLaunchPending = useAppState(s => s.ultraplanLaunchPending);
 639    const viewingAgentTaskId = useAppState(s => s.viewingAgentTaskId);
 640    const setAppState = useSetAppState();
 641  
 642    // Bootstrap: retained local_agent that hasn't loaded disk yet → read
 643    // sidechain JSONL and UUID-merge with whatever stream has appended so far.
 644    // Stream appends immediately on retain (no defer); bootstrap fills the
 645    // prefix. Disk-write-before-yield means live is always a suffix of disk.
 646    const viewedLocalAgent = viewingAgentTaskId ? tasks[viewingAgentTaskId] : undefined;
 647    const needsBootstrap = isLocalAgentTask(viewedLocalAgent) && viewedLocalAgent.retain && !viewedLocalAgent.diskLoaded;
 648    useEffect(() => {
 649      if (!viewingAgentTaskId || !needsBootstrap) return;
 650      const taskId = viewingAgentTaskId;
 651      void getAgentTranscript(asAgentId(taskId)).then(result => {
 652        setAppState(prev => {
 653          const t = prev.tasks[taskId];
 654          if (!isLocalAgentTask(t) || t.diskLoaded || !t.retain) return prev;
 655          const live = t.messages ?? [];
 656          const liveUuids = new Set(live.map(m => m.uuid));
 657          const diskOnly = result ? result.messages.filter(m => !liveUuids.has(m.uuid)) : [];
 658          return {
 659            ...prev,
 660            tasks: {
 661              ...prev.tasks,
 662              [taskId]: {
 663                ...t,
 664                messages: [...diskOnly, ...live],
 665                diskLoaded: true
 666              }
 667            }
 668          };
 669        });
 670      });
 671    }, [viewingAgentTaskId, needsBootstrap, setAppState]);
 672    const store = useAppStateStore();
 673    const terminal = useTerminalNotification();
 674    const mainLoopModel = useMainLoopModel();
 675  
 676    // Note: standaloneAgentContext is initialized in main.tsx (via initialState) or
 677    // ResumeConversation.tsx (via setAppState before rendering REPL) to avoid
 678    // useEffect-based state initialization on mount (per CLAUDE.md guidelines)
 679  
 680    // Local state for commands (hot-reloadable when skill files change)
 681    const [localCommands, setLocalCommands] = useState(initialCommands);
 682  
 683    // Watch for skill file changes and reload all commands
 684    useSkillsChange(isRemoteSession ? undefined : getProjectRoot(), setLocalCommands);
 685  
 686    // Track proactive mode for tools dependency - SleepTool filters by proactive state
 687    const proactiveActive = React.useSyncExternalStore(proactiveModule?.subscribeToProactiveChanges ?? PROACTIVE_NO_OP_SUBSCRIBE, proactiveModule?.isProactiveActive ?? PROACTIVE_FALSE);
 688  
 689    // BriefTool.isEnabled() reads getUserMsgOptIn() from bootstrap state, which
 690    // /brief flips mid-session alongside isBriefOnly. The memo below needs a
 691    // React-visible dep to re-run getTools() when that happens; isBriefOnly is
 692    // the AppState mirror that triggers the re-render. Without this, toggling
 693    // /brief mid-session leaves the stale tool list (no SendUserMessage) and
 694    // the model emits plain text the brief filter hides.
 695    const isBriefOnly = useAppState(s => s.isBriefOnly);
 696    const localTools = useMemo(() => getTools(toolPermissionContext), [toolPermissionContext, proactiveActive, isBriefOnly]);
 697    useKickOffCheckAndDisableBypassPermissionsIfNeeded();
 698    useKickOffCheckAndDisableAutoModeIfNeeded();
 699    const [dynamicMcpConfig, setDynamicMcpConfig] = useState<Record<string, ScopedMcpServerConfig> | undefined>(initialDynamicMcpConfig);
 700    const onChangeDynamicMcpConfig = useCallback((config: Record<string, ScopedMcpServerConfig>) => {
 701      setDynamicMcpConfig(config);
 702    }, [setDynamicMcpConfig]);
 703    const [screen, setScreen] = useState<Screen>('prompt');
 704    const [showAllInTranscript, setShowAllInTranscript] = useState(false);
 705    // [ forces the dump-to-scrollback path inside transcript mode. Separate
 706    // from CLAUDE_CODE_NO_FLICKER=0 (which is process-lifetime) — this is
 707    // ephemeral, reset on transcript exit. Diagnostic escape hatch so
 708    // terminal/tmux native cmd-F can search the full flat render.
 709    const [dumpMode, setDumpMode] = useState(false);
 710    // v-for-editor render progress. Inline in the footer — notifications
 711    // render inside PromptInput which isn't mounted in transcript.
 712    const [editorStatus, setEditorStatus] = useState('');
 713    // Incremented on transcript exit. Async v-render captures this at start;
 714    // each status write no-ops if stale (user left transcript mid-render —
 715    // the stable setState would otherwise stamp a ghost toast into the next
 716    // session). Also clears any pending 4s auto-clear.
 717    const editorGenRef = useRef(0);
 718    const editorTimerRef = useRef<ReturnType<typeof setTimeout> | undefined>(undefined);
 719    const editorRenderingRef = useRef(false);
 720    const {
 721      addNotification,
 722      removeNotification
 723    } = useNotifications();
 724  
 725    // eslint-disable-next-line prefer-const
 726    let trySuggestBgPRIntercept = SUGGEST_BG_PR_NOOP;
 727    const mcpClients = useMergedClients(initialMcpClients, mcp.clients);
 728  
 729    // IDE integration
 730    const [ideSelection, setIDESelection] = useState<IDESelection | undefined>(undefined);
 731    const [ideToInstallExtension, setIDEToInstallExtension] = useState<IdeType | null>(null);
 732    const [ideInstallationStatus, setIDEInstallationStatus] = useState<IDEExtensionInstallationStatus | null>(null);
 733    const [showIdeOnboarding, setShowIdeOnboarding] = useState(false);
 734    // Dead code elimination: model switch callout state (ant-only)
 735    const [showModelSwitchCallout, setShowModelSwitchCallout] = useState(() => {
 736      if ("external" === 'ant') {
 737        return shouldShowAntModelSwitch();
 738      }
 739      return false;
 740    });
 741    const [showEffortCallout, setShowEffortCallout] = useState(() => shouldShowEffortCallout(mainLoopModel));
 742    const showRemoteCallout = useAppState(s => s.showRemoteCallout);
 743    const [showDesktopUpsellStartup, setShowDesktopUpsellStartup] = useState(() => shouldShowDesktopUpsellStartup());
 744    // notifications
 745    useModelMigrationNotifications();
 746    useCanSwitchToExistingSubscription();
 747    useIDEStatusIndicator({
 748      ideSelection,
 749      mcpClients,
 750      ideInstallationStatus
 751    });
 752    useMcpConnectivityStatus({
 753      mcpClients
 754    });
 755    useAutoModeUnavailableNotification();
 756    usePluginInstallationStatus();
 757    usePluginAutoupdateNotification();
 758    useSettingsErrors();
 759    useRateLimitWarningNotification(mainLoopModel);
 760    useFastModeNotification();
 761    useDeprecationWarningNotification(mainLoopModel);
 762    useNpmDeprecationNotification();
 763    useAntOrgWarningNotification();
 764    useInstallMessages();
 765    useChromeExtensionNotification();
 766    useOfficialMarketplaceNotification();
 767    useLspInitializationNotification();
 768    useTeammateLifecycleNotification();
 769    const {
 770      recommendation: lspRecommendation,
 771      handleResponse: handleLspResponse
 772    } = useLspPluginRecommendation();
 773    const {
 774      recommendation: hintRecommendation,
 775      handleResponse: handleHintResponse
 776    } = useClaudeCodeHintRecommendation();
 777  
 778    // Memoize the combined initial tools array to prevent reference changes
 779    const combinedInitialTools = useMemo(() => {
 780      return [...localTools, ...initialTools];
 781    }, [localTools, initialTools]);
 782  
 783    // Initialize plugin management
 784    useManagePlugins({
 785      enabled: !isRemoteSession
 786    });
 787    const tasksV2 = useTasksV2WithCollapseEffect();
 788  
 789    // Start background plugin installations
 790  
 791    // SECURITY: This code is guaranteed to run ONLY after the "trust this folder" dialog
 792    // has been confirmed by the user. The trust dialog is shown in cli.tsx (line ~387)
 793    // before the REPL component is rendered. The dialog blocks execution until the user
 794    // accepts, and only then is the REPL component mounted and this effect runs.
 795    // This ensures that plugin installations from repository and user settings only
 796    // happen after explicit user consent to trust the current working directory.
 797    useEffect(() => {
 798      if (isRemoteSession) return;
 799      void performStartupChecks(setAppState);
 800    }, [setAppState, isRemoteSession]);
 801  
 802    // Allow Claude in Chrome MCP to send prompts through MCP notifications
 803    // and sync permission mode changes to the Chrome extension
 804    usePromptsFromClaudeInChrome(isRemoteSession ? EMPTY_MCP_CLIENTS : mcpClients, toolPermissionContext.mode);
 805  
 806    // Initialize swarm features: teammate hooks and context
 807    // Handles both fresh spawns and resumed teammate sessions
 808    useSwarmInitialization(setAppState, initialMessages, {
 809      enabled: !isRemoteSession
 810    });
 811    const mergedTools = useMergedTools(combinedInitialTools, mcp.tools, toolPermissionContext);
 812  
 813    // Apply agent tool restrictions if mainThreadAgentDefinition is set
 814    const {
 815      tools,
 816      allowedAgentTypes
 817    } = useMemo(() => {
 818      if (!mainThreadAgentDefinition) {
 819        return {
 820          tools: mergedTools,
 821          allowedAgentTypes: undefined as string[] | undefined
 822        };
 823      }
 824      const resolved = resolveAgentTools(mainThreadAgentDefinition, mergedTools, false, true);
 825      return {
 826        tools: resolved.resolvedTools,
 827        allowedAgentTypes: resolved.allowedAgentTypes
 828      };
 829    }, [mainThreadAgentDefinition, mergedTools]);
 830  
 831    // Merge commands from local state, plugins, and MCP
 832    const commandsWithPlugins = useMergedCommands(localCommands, plugins.commands as Command[]);
 833    const mergedCommands = useMergedCommands(commandsWithPlugins, mcp.commands as Command[]);
 834    // Filter out all commands if disableSlashCommands is true
 835    const commands = useMemo(() => disableSlashCommands ? [] : mergedCommands, [disableSlashCommands, mergedCommands]);
 836    useIdeLogging(isRemoteSession ? EMPTY_MCP_CLIENTS : mcp.clients);
 837    useIdeSelection(isRemoteSession ? EMPTY_MCP_CLIENTS : mcp.clients, setIDESelection);
 838    const [streamMode, setStreamMode] = useState<SpinnerMode>('responding');
 839    // Ref mirror so onSubmit can read the latest value without adding
 840    // streamMode to its deps. streamMode flips between
 841    // requesting/responding/tool-use ~10x per turn during streaming; having it
 842    // in onSubmit's deps was recreating onSubmit on every flip, which
 843    // cascaded into PromptInput prop churn and downstream useCallback/useMemo
 844    // invalidation. The only consumers inside callbacks are debug logging and
 845    // telemetry (handlePromptSubmit.ts), so a stale-by-one-render value is
 846    // harmless — but ref mirrors sync on every render anyway so it's fresh.
 847    const streamModeRef = useRef(streamMode);
 848    streamModeRef.current = streamMode;
 849    const [streamingToolUses, setStreamingToolUses] = useState<StreamingToolUse[]>([]);
 850    const [streamingThinking, setStreamingThinking] = useState<StreamingThinking | null>(null);
 851  
 852    // Auto-hide streaming thinking after 30 seconds of being completed
 853    useEffect(() => {
 854      if (streamingThinking && !streamingThinking.isStreaming && streamingThinking.streamingEndedAt) {
 855        const elapsed = Date.now() - streamingThinking.streamingEndedAt;
 856        const remaining = 30000 - elapsed;
 857        if (remaining > 0) {
 858          const timer = setTimeout(setStreamingThinking, remaining, null);
 859          return () => clearTimeout(timer);
 860        } else {
 861          setStreamingThinking(null);
 862        }
 863      }
 864    }, [streamingThinking]);
 865    const [abortController, setAbortController] = useState<AbortController | null>(null);
 866    // Ref that always points to the current abort controller, used by the
 867    // REPL bridge to abort the active query when a remote interrupt arrives.
 868    const abortControllerRef = useRef<AbortController | null>(null);
 869    abortControllerRef.current = abortController;
 870  
 871    // Ref for the bridge result callback — set after useReplBridge initializes,
 872    // read in the onQuery finally block to notify mobile clients that a turn ended.
 873    const sendBridgeResultRef = useRef<() => void>(() => {});
 874  
 875    // Ref for the synchronous restore callback — set after restoreMessageSync is
 876    // defined, read in the onQuery finally block for auto-restore on interrupt.
 877    const restoreMessageSyncRef = useRef<(m: UserMessage) => void>(() => {});
 878  
 879    // Ref to the fullscreen layout's scroll box for keyboard scrolling.
 880    // Null when fullscreen mode is disabled (ref never attached).
 881    const scrollRef = useRef<ScrollBoxHandle>(null);
 882    // Separate ref for the modal slot's inner ScrollBox — passed through
 883    // FullscreenLayout → ModalContext so Tabs can attach it to its own
 884    // ScrollBox for tall content (e.g. /status's MCP-server list). NOT
 885    // keyboard-driven — ScrollKeybindingHandler stays on the outer ref so
 886    // PgUp/PgDn/wheel always scroll the transcript behind the modal.
 887    // Plumbing kept for future modal-scroll wiring.
 888    const modalScrollRef = useRef<ScrollBoxHandle>(null);
 889    // Timestamp of the last user-initiated scroll (wheel, PgUp/PgDn, ctrl+u,
 890    // End/Home, G, drag-to-scroll). Stamped in composedOnScroll — the single
 891    // chokepoint ScrollKeybindingHandler calls for every user scroll action.
 892    // Programmatic scrolls (repinScroll's scrollToBottom, sticky auto-follow)
 893    // do NOT go through composedOnScroll, so they don't stamp this. Ref not
 894    // state: no re-render on every wheel tick.
 895    const lastUserScrollTsRef = useRef(0);
 896  
 897    // Synchronous state machine for the query lifecycle. Replaces the
 898    // error-prone dual-state pattern where isLoading (React state, async
 899    // batched) and isQueryRunning (ref, sync) could desync. See QueryGuard.ts.
 900    const queryGuard = React.useRef(new QueryGuard()).current;
 901  
 902    // Subscribe to the guard — true during dispatching or running.
 903    // This is the single source of truth for "is a local query in flight".
 904    const isQueryActive = React.useSyncExternalStore(queryGuard.subscribe, queryGuard.getSnapshot);
 905  
 906    // Separate loading flag for operations outside the local query guard:
 907    // remote sessions (useRemoteSession / useDirectConnect) and foregrounded
 908    // background tasks (useSessionBackgrounding). These don't route through
 909    // onQuery / queryGuard, so they need their own spinner-visibility state.
 910    // Initialize true if remote mode with initial prompt (CCR processing it).
 911    const [isExternalLoading, setIsExternalLoadingRaw] = React.useState(remoteSessionConfig?.hasInitialPrompt ?? false);
 912  
 913    // Derived: any loading source active. Read-only — no setter. Local query
 914    // loading is driven by queryGuard (reserve/tryStart/end/cancelReservation),
 915    // external loading by setIsExternalLoading.
 916    const isLoading = isQueryActive || isExternalLoading;
 917  
 918    // Elapsed time is computed by SpinnerWithVerb from these refs on each
 919    // animation frame, avoiding a useInterval that re-renders the entire REPL.
 920    const [userInputOnProcessing, setUserInputOnProcessingRaw] = React.useState<string | undefined>(undefined);
 921    // messagesRef.current.length at the moment userInputOnProcessing was set.
 922    // The placeholder hides once displayedMessages grows past this — i.e. the
 923    // real user message has landed in the visible transcript.
 924    const userInputBaselineRef = React.useRef(0);
 925    // True while the submitted prompt is being processed but its user message
 926    // hasn't reached setMessages yet. setMessages uses this to keep the
 927    // baseline in sync when unrelated async messages (bridge status, hook
 928    // results, scheduled tasks) land during that window.
 929    const userMessagePendingRef = React.useRef(false);
 930  
 931    // Wall-clock time tracking refs for accurate elapsed time calculation
 932    const loadingStartTimeRef = React.useRef<number>(0);
 933    const totalPausedMsRef = React.useRef(0);
 934    const pauseStartTimeRef = React.useRef<number | null>(null);
 935    const resetTimingRefs = React.useCallback(() => {
 936      loadingStartTimeRef.current = Date.now();
 937      totalPausedMsRef.current = 0;
 938      pauseStartTimeRef.current = null;
 939    }, []);
 940  
 941    // Reset timing refs inline when isQueryActive transitions false→true.
 942    // queryGuard.reserve() (in executeUserInput) fires BEFORE processUserInput's
 943    // first await, but the ref reset in onQuery's try block runs AFTER. During
 944    // that gap, React renders the spinner with loadingStartTimeRef=0, computing
 945    // elapsedTimeMs = Date.now() - 0 ≈ 56 years. This inline reset runs on the
 946    // first render where isQueryActive is observed true — the same render that
 947    // first shows the spinner — so the ref is correct by the time the spinner
 948    // reads it. See INC-4549.
 949    const wasQueryActiveRef = React.useRef(false);
 950    if (isQueryActive && !wasQueryActiveRef.current) {
 951      resetTimingRefs();
 952    }
 953    wasQueryActiveRef.current = isQueryActive;
 954  
 955    // Wrapper for setIsExternalLoading that resets timing refs on transition
 956    // to true — SpinnerWithVerb reads these for elapsed time, so they must be
 957    // reset for remote sessions / foregrounded tasks too (not just local
 958    // queries, which reset them in onQuery). Without this, a remote-only
 959    // session would show ~56 years elapsed (Date.now() - 0).
 960    const setIsExternalLoading = React.useCallback((value: boolean) => {
 961      setIsExternalLoadingRaw(value);
 962      if (value) resetTimingRefs();
 963    }, [resetTimingRefs]);
 964  
 965    // Start time of the first turn that had swarm teammates running
 966    // Used to compute total elapsed time (including teammate execution) for the deferred message
 967    const swarmStartTimeRef = React.useRef<number | null>(null);
 968    const swarmBudgetInfoRef = React.useRef<{
 969      tokens: number;
 970      limit: number;
 971      nudges: number;
 972    } | undefined>(undefined);
 973  
 974    // Ref to track current focusedInputDialog for use in callbacks
 975    // This avoids stale closures when checking dialog state in timer callbacks
 976    const focusedInputDialogRef = React.useRef<ReturnType<typeof getFocusedInputDialog>>(undefined);
 977  
 978    // How long after the last keystroke before deferred dialogs are shown
 979    const PROMPT_SUPPRESSION_MS = 1500;
 980    // True when user is actively typing — defers interrupt dialogs so keystrokes
 981    // don't accidentally dismiss or answer a permission prompt the user hasn't read yet.
 982    const [isPromptInputActive, setIsPromptInputActive] = React.useState(false);
 983    const [autoUpdaterResult, setAutoUpdaterResult] = useState<AutoUpdaterResult | null>(null);
 984    useEffect(() => {
 985      if (autoUpdaterResult?.notifications) {
 986        autoUpdaterResult.notifications.forEach(notification => {
 987          addNotification({
 988            key: 'auto-updater-notification',
 989            text: notification,
 990            priority: 'low'
 991          });
 992        });
 993      }
 994    }, [autoUpdaterResult, addNotification]);
 995  
 996    // tmux + fullscreen + `mouse off`: one-time hint that wheel won't scroll.
 997    // We no longer mutate tmux's session-scoped mouse option (it poisoned
 998    // sibling panes); tmux users already know this tradeoff from vim/less.
 999    useEffect(() => {
1000      if (isFullscreenEnvEnabled()) {
1001        void maybeGetTmuxMouseHint().then(hint => {
1002          if (hint) {
1003            addNotification({
1004              key: 'tmux-mouse-hint',
1005              text: hint,
1006              priority: 'low'
1007            });
1008          }
1009        });
1010      }
1011      // eslint-disable-next-line react-hooks/exhaustive-deps
1012    }, []);
1013    const [showUndercoverCallout, setShowUndercoverCallout] = useState(false);
1014    useEffect(() => {
1015      if ("external" === 'ant') {
1016        void (async () => {
1017          // Wait for repo classification to settle (memoized, no-op if primed).
1018          const {
1019            isInternalModelRepo
1020          } = await import('../utils/commitAttribution.js');
1021          await isInternalModelRepo();
1022          const {
1023            shouldShowUndercoverAutoNotice
1024          } = await import('../utils/undercover.js');
1025          if (shouldShowUndercoverAutoNotice()) {
1026            setShowUndercoverCallout(true);
1027          }
1028        })();
1029      }
1030      // eslint-disable-next-line react-hooks/exhaustive-deps
1031    }, []);
1032    const [toolJSX, setToolJSXInternal] = useState<{
1033      jsx: React.ReactNode | null;
1034      shouldHidePromptInput: boolean;
1035      shouldContinueAnimation?: true;
1036      showSpinner?: boolean;
1037      isLocalJSXCommand?: boolean;
1038      isImmediate?: boolean;
1039    } | null>(null);
1040  
1041    // Track local JSX commands separately so tools can't overwrite them.
1042    // This enables "immediate" commands (like /btw) to persist while Claude is processing.
1043    const localJSXCommandRef = useRef<{
1044      jsx: React.ReactNode | null;
1045      shouldHidePromptInput: boolean;
1046      shouldContinueAnimation?: true;
1047      showSpinner?: boolean;
1048      isLocalJSXCommand: true;
1049    } | null>(null);
1050  
1051    // Wrapper for setToolJSX that preserves local JSX commands (like /btw).
1052    // When a local JSX command is active, we ignore updates from tools
1053    // unless they explicitly set clearLocalJSX: true (from onDone callbacks).
1054    //
1055    // TO ADD A NEW IMMEDIATE COMMAND:
1056    // 1. Set `immediate: true` in the command definition
1057    // 2. Set `isLocalJSXCommand: true` when calling setToolJSX in the command's JSX
1058    // 3. In the onDone callback, use `setToolJSX({ jsx: null, shouldHidePromptInput: false, clearLocalJSX: true })`
1059    //    to explicitly clear the overlay when the user dismisses it
1060    const setToolJSX = useCallback((args: {
1061      jsx: React.ReactNode | null;
1062      shouldHidePromptInput: boolean;
1063      shouldContinueAnimation?: true;
1064      showSpinner?: boolean;
1065      isLocalJSXCommand?: boolean;
1066      clearLocalJSX?: boolean;
1067    } | null) => {
1068      // If setting a local JSX command, store it in the ref
1069      if (args?.isLocalJSXCommand) {
1070        const {
1071          clearLocalJSX: _,
1072          ...rest
1073        } = args;
1074        localJSXCommandRef.current = {
1075          ...rest,
1076          isLocalJSXCommand: true
1077        };
1078        setToolJSXInternal(rest);
1079        return;
1080      }
1081  
1082      // If there's an active local JSX command in the ref
1083      if (localJSXCommandRef.current) {
1084        // Allow clearing only if explicitly requested (from onDone callbacks)
1085        if (args?.clearLocalJSX) {
1086          localJSXCommandRef.current = null;
1087          setToolJSXInternal(null);
1088          return;
1089        }
1090        // Otherwise, keep the local JSX command visible - ignore tool updates
1091        return;
1092      }
1093  
1094      // No active local JSX command, allow any update
1095      if (args?.clearLocalJSX) {
1096        setToolJSXInternal(null);
1097        return;
1098      }
1099      setToolJSXInternal(args);
1100    }, []);
1101    const [toolUseConfirmQueue, setToolUseConfirmQueue] = useState<ToolUseConfirm[]>([]);
1102    // Sticky footer JSX registered by permission request components (currently
1103    // only ExitPlanModePermissionRequest). Renders in FullscreenLayout's `bottom`
1104    // slot so response options stay visible while the user scrolls a long plan.
1105    const [permissionStickyFooter, setPermissionStickyFooter] = useState<React.ReactNode | null>(null);
1106    const [sandboxPermissionRequestQueue, setSandboxPermissionRequestQueue] = useState<Array<{
1107      hostPattern: NetworkHostPattern;
1108      resolvePromise: (allowConnection: boolean) => void;
1109    }>>([]);
1110    const [promptQueue, setPromptQueue] = useState<Array<{
1111      request: PromptRequest;
1112      title: string;
1113      toolInputSummary?: string | null;
1114      resolve: (response: PromptResponse) => void;
1115      reject: (error: Error) => void;
1116    }>>([]);
1117  
1118    // Track bridge cleanup functions for sandbox permission requests so the
1119    // local dialog handler can cancel the remote prompt when the local user
1120    // responds first. Keyed by host to support concurrent same-host requests.
1121    const sandboxBridgeCleanupRef = useRef<Map<string, Array<() => void>>>(new Map());
1122  
1123    // -- Terminal title management
1124    // Session title (set via /rename or restored on resume) wins over
1125    // the agent name, which wins over the Haiku-extracted topic;
1126    // all fall back to the product name.
1127    const terminalTitleFromRename = useAppState(s => s.settings.terminalTitleFromRename) !== false;
1128    const sessionTitle = terminalTitleFromRename ? getCurrentSessionTitle(getSessionId()) : undefined;
1129    const [haikuTitle, setHaikuTitle] = useState<string>();
1130    // Gates the one-shot Haiku call that generates the tab title. Seeded true
1131    // on resume (initialMessages present) so we don't re-title a resumed
1132    // session from mid-conversation context.
1133    const haikuTitleAttemptedRef = useRef((initialMessages?.length ?? 0) > 0);
1134    const agentTitle = mainThreadAgentDefinition?.agentType;
1135    const terminalTitle = sessionTitle ?? agentTitle ?? haikuTitle ?? 'Claude Code';
1136    const isWaitingForApproval = toolUseConfirmQueue.length > 0 || promptQueue.length > 0 || pendingWorkerRequest || pendingSandboxRequest;
1137    // Local-jsx commands (like /plugin, /config) show user-facing dialogs that
1138    // wait for input. Require jsx != null — if the flag is stuck true but jsx
1139    // is null, treat as not-showing so TextInput focus and queue processor
1140    // aren't deadlocked by a phantom overlay.
1141    const isShowingLocalJSXCommand = toolJSX?.isLocalJSXCommand === true && toolJSX?.jsx != null;
1142    const titleIsAnimating = isLoading && !isWaitingForApproval && !isShowingLocalJSXCommand;
1143    // Title animation state lives in <AnimatedTerminalTitle> so the 960ms tick
1144    // doesn't re-render REPL. titleDisabled/terminalTitle are still computed
1145    // here because onQueryImpl reads them (background session description,
1146    // haiku title extraction gate).
1147  
1148    // Prevent macOS from sleeping while Claude is working
1149    useEffect(() => {
1150      if (isLoading && !isWaitingForApproval && !isShowingLocalJSXCommand) {
1151        startPreventSleep();
1152        return () => stopPreventSleep();
1153      }
1154    }, [isLoading, isWaitingForApproval, isShowingLocalJSXCommand]);
1155    const sessionStatus: TabStatusKind = isWaitingForApproval || isShowingLocalJSXCommand ? 'waiting' : isLoading ? 'busy' : 'idle';
1156    const waitingFor = sessionStatus !== 'waiting' ? undefined : toolUseConfirmQueue.length > 0 ? `approve ${toolUseConfirmQueue[0]!.tool.name}` : pendingWorkerRequest ? 'worker request' : pendingSandboxRequest ? 'sandbox request' : isShowingLocalJSXCommand ? 'dialog open' : 'input needed';
1157  
1158    // Push status to the PID file for `claude ps`. Fire-and-forget; ps falls
1159    // back to transcript-tail derivation when this is missing/stale.
1160    useEffect(() => {
1161      if (feature('BG_SESSIONS')) {
1162        void updateSessionActivity({
1163          status: sessionStatus,
1164          waitingFor
1165        });
1166      }
1167    }, [sessionStatus, waitingFor]);
1168  
1169    // 3P default: off — OSC 21337 is ant-only while the spec stabilizes.
1170    // Gated so we can roll back if the sidebar indicator conflicts with
1171    // the title spinner in terminals that render both. When the flag is
1172    // on, the user-facing config setting controls whether it's active.
1173    const tabStatusGateEnabled = getFeatureValue_CACHED_MAY_BE_STALE('tengu_terminal_sidebar', false);
1174    const showStatusInTerminalTab = tabStatusGateEnabled && (getGlobalConfig().showStatusInTerminalTab ?? false);
1175    useTabStatus(titleDisabled || !showStatusInTerminalTab ? null : sessionStatus);
1176  
1177    // Register the leader's setToolUseConfirmQueue for in-process teammates
1178    useEffect(() => {
1179      registerLeaderToolUseConfirmQueue(setToolUseConfirmQueue);
1180      return () => unregisterLeaderToolUseConfirmQueue();
1181    }, [setToolUseConfirmQueue]);
1182    const [messages, rawSetMessages] = useState<MessageType[]>(initialMessages ?? []);
1183    const messagesRef = useRef(messages);
1184    // Stores the willowMode variant that was shown (or false if no hint shown).
1185    // Captured at hint_shown time so hint_converted telemetry reports the same
1186    // variant — the GrowthBook value shouldn't change mid-session, but reading
1187    // it once guarantees consistency between the paired events.
1188    const idleHintShownRef = useRef<string | false>(false);
1189    // Wrap setMessages so messagesRef is always current the instant the
1190    // call returns — not when React later processes the batch.  Apply the
1191    // updater eagerly against the ref, then hand React the computed value
1192    // (not the function).  rawSetMessages batching becomes last-write-wins,
1193    // and the last write is correct because each call composes against the
1194    // already-updated ref.  This is the Zustand pattern: ref is source of
1195    // truth, React state is the render projection.  Without this, paths
1196    // that queue functional updaters then synchronously read the ref
1197    // (e.g. handleSpeculationAccept → onQuery) see stale data.
1198    const setMessages = useCallback((action: React.SetStateAction<MessageType[]>) => {
1199      const prev = messagesRef.current;
1200      const next = typeof action === 'function' ? action(messagesRef.current) : action;
1201      messagesRef.current = next;
1202      if (next.length < userInputBaselineRef.current) {
1203        // Shrank (compact/rewind/clear) — clamp so placeholderText's length
1204        // check can't go stale.
1205        userInputBaselineRef.current = 0;
1206      } else if (next.length > prev.length && userMessagePendingRef.current) {
1207        // Grew while the submitted user message hasn't landed yet. If the
1208        // added messages don't include it (bridge status, hook results,
1209        // scheduled tasks landing async during processUserInputBase), bump
1210        // baseline so the placeholder stays visible. Once the user message
1211        // lands, stop tracking — later additions (assistant stream) should
1212        // not re-show the placeholder.
1213        const delta = next.length - prev.length;
1214        const added = prev.length === 0 || next[0] === prev[0] ? next.slice(-delta) : next.slice(0, delta);
1215        if (added.some(isHumanTurn)) {
1216          userMessagePendingRef.current = false;
1217        } else {
1218          userInputBaselineRef.current = next.length;
1219        }
1220      }
1221      rawSetMessages(next);
1222    }, []);
1223    // Capture the baseline message count alongside the placeholder text so
1224    // the render can hide it once displayedMessages grows past the baseline.
1225    const setUserInputOnProcessing = useCallback((input: string | undefined) => {
1226      if (input !== undefined) {
1227        userInputBaselineRef.current = messagesRef.current.length;
1228        userMessagePendingRef.current = true;
1229      } else {
1230        userMessagePendingRef.current = false;
1231      }
1232      setUserInputOnProcessingRaw(input);
1233    }, []);
1234    // Fullscreen: track the unseen-divider position. dividerIndex changes
1235    // only ~twice/scroll-session (first scroll-away + repin). pillVisible
1236    // and stickyPrompt now live in FullscreenLayout — they subscribe to
1237    // ScrollBox directly so per-frame scroll never re-renders REPL.
1238    const {
1239      dividerIndex,
1240      dividerYRef,
1241      onScrollAway,
1242      onRepin,
1243      jumpToNew,
1244      shiftDivider
1245    } = useUnseenDivider(messages.length);
1246    if (feature('AWAY_SUMMARY')) {
1247      // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
1248      useAwaySummary(messages, setMessages, isLoading);
1249    }
1250    const [cursor, setCursor] = useState<MessageActionsState | null>(null);
1251    const cursorNavRef = useRef<MessageActionsNav | null>(null);
1252    // Memoized so Messages' React.memo holds.
1253    const unseenDivider = useMemo(() => computeUnseenDivider(messages, dividerIndex),
1254    // eslint-disable-next-line react-hooks/exhaustive-deps -- length change covers appends; useUnseenDivider's count-drop guard clears dividerIndex on replace/rewind
1255    [dividerIndex, messages.length]);
1256    // Re-pin scroll to bottom and clear the unseen-messages baseline. Called
1257    // on any user-driven return-to-live action (submit, type-into-empty,
1258    // overlay appear/dismiss).
1259    const repinScroll = useCallback(() => {
1260      scrollRef.current?.scrollToBottom();
1261      onRepin();
1262      setCursor(null);
1263    }, [onRepin, setCursor]);
1264    // Backstop for the submit-handler repin at onSubmit. If a buffered stdin
1265    // event (wheel/drag) races between handler-fire and state-commit, the
1266    // handler's scrollToBottom can be undone. This effect fires on the render
1267    // where the user's message actually lands — tied to React's commit cycle,
1268    // so it can't race with stdin. Keyed on lastMsg identity (not messages.length)
1269    // so useAssistantHistory's prepends don't spuriously repin.
1270    const lastMsg = messages.at(-1);
1271    const lastMsgIsHuman = lastMsg != null && isHumanTurn(lastMsg);
1272    useEffect(() => {
1273      if (lastMsgIsHuman) {
1274        repinScroll();
1275      }
1276    }, [lastMsgIsHuman, lastMsg, repinScroll]);
1277    // Assistant-chat: lazy-load remote history on scroll-up. No-op unless
1278    // KAIROS build + config.viewerOnly. feature() is build-time constant so
1279    // the branch is dead-code-eliminated in non-KAIROS builds (same pattern
1280    // as useUnseenDivider above).
1281    const {
1282      maybeLoadOlder
1283    } = feature('KAIROS') ?
1284    // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
1285    useAssistantHistory({
1286      config: remoteSessionConfig,
1287      setMessages,
1288      scrollRef,
1289      onPrepend: shiftDivider
1290    }) : HISTORY_STUB;
1291    // Compose useUnseenDivider's callbacks with the lazy-load trigger.
1292    const composedOnScroll = useCallback((sticky: boolean, handle: ScrollBoxHandle) => {
1293      lastUserScrollTsRef.current = Date.now();
1294      if (sticky) {
1295        onRepin();
1296      } else {
1297        onScrollAway(handle);
1298        if (feature('KAIROS')) maybeLoadOlder(handle);
1299        // Dismiss the companion bubble on scroll — it's absolute-positioned
1300        // at bottom-right and covers transcript content. Scrolling = user is
1301        // trying to read something under it.
1302        if (feature('BUDDY')) {
1303          setAppState(prev => prev.companionReaction === undefined ? prev : {
1304            ...prev,
1305            companionReaction: undefined
1306          });
1307        }
1308      }
1309    }, [onRepin, onScrollAway, maybeLoadOlder, setAppState]);
1310    // Deferred SessionStart hook messages — REPL renders immediately and
1311    // hook messages are injected when they resolve. awaitPendingHooks()
1312    // must be called before the first API call so the model sees hook context.
1313    const awaitPendingHooks = useDeferredHookMessages(pendingHookMessages, setMessages);
1314  
1315    // Deferred messages for the Messages component — renders at transition
1316    // priority so the reconciler yields every 5ms, keeping input responsive
1317    // while the expensive message processing pipeline runs.
1318    const deferredMessages = useDeferredValue(messages);
1319    const deferredBehind = messages.length - deferredMessages.length;
1320    if (deferredBehind > 0) {
1321      logForDebugging(`[useDeferredValue] Messages deferred by ${deferredBehind} (${deferredMessages.length}→${messages.length})`);
1322    }
1323  
1324    // Frozen state for transcript mode - stores lengths instead of cloning arrays for memory efficiency
1325    const [frozenTranscriptState, setFrozenTranscriptState] = useState<{
1326      messagesLength: number;
1327      streamingToolUsesLength: number;
1328    } | null>(null);
1329    // Initialize input with any early input that was captured before REPL was ready.
1330    // Using lazy initialization ensures cursor offset is set correctly in PromptInput.
1331    const [inputValue, setInputValueRaw] = useState(() => consumeEarlyInput());
1332    const inputValueRef = useRef(inputValue);
1333    inputValueRef.current = inputValue;
1334    const insertTextRef = useRef<{
1335      insert: (text: string) => void;
1336      setInputWithCursor: (value: string, cursor: number) => void;
1337      cursorOffset: number;
1338    } | null>(null);
1339  
1340    // Wrap setInputValue to co-locate suppression state updates.
1341    // Both setState calls happen in the same synchronous context so React
1342    // batches them into a single render, eliminating the extra render that
1343    // the previous useEffect → setState pattern caused.
1344    const setInputValue = useCallback((value: string) => {
1345      if (trySuggestBgPRIntercept(inputValueRef.current, value)) return;
1346      // In fullscreen mode, typing into an empty prompt re-pins scroll to
1347      // bottom. Only fires on empty→non-empty so scrolling up to reference
1348      // something while composing a message doesn't yank the view back on
1349      // every keystroke. Restores the pre-fullscreen muscle memory of
1350      // typing to snap back to the end of the conversation.
1351      // Skipped if the user scrolled within the last 3s — they're actively
1352      // reading, not lost. lastUserScrollTsRef starts at 0 so the first-
1353      // ever keypress (no scroll yet) always repins.
1354      if (inputValueRef.current === '' && value !== '' && Date.now() - lastUserScrollTsRef.current >= RECENT_SCROLL_REPIN_WINDOW_MS) {
1355        repinScroll();
1356      }
1357      // Sync ref immediately (like setMessages) so callers that read
1358      // inputValueRef before React commits — e.g. the auto-restore finally
1359      // block's `=== ''` guard — see the fresh value, not the stale render.
1360      inputValueRef.current = value;
1361      setInputValueRaw(value);
1362      setIsPromptInputActive(value.trim().length > 0);
1363    }, [setIsPromptInputActive, repinScroll, trySuggestBgPRIntercept]);
1364  
1365    // Schedule a timeout to stop suppressing dialogs after the user stops typing.
1366    // Only manages the timeout — the immediate activation is handled by setInputValue above.
1367    useEffect(() => {
1368      if (inputValue.trim().length === 0) return;
1369      const timer = setTimeout(setIsPromptInputActive, PROMPT_SUPPRESSION_MS, false);
1370      return () => clearTimeout(timer);
1371    }, [inputValue]);
1372    const [inputMode, setInputMode] = useState<PromptInputMode>('prompt');
1373    const [stashedPrompt, setStashedPrompt] = useState<{
1374      text: string;
1375      cursorOffset: number;
1376      pastedContents: Record<number, PastedContent>;
1377    } | undefined>();
1378  
1379    // Callback to filter commands based on CCR's available slash commands
1380    const handleRemoteInit = useCallback((remoteSlashCommands: string[]) => {
1381      const remoteCommandSet = new Set(remoteSlashCommands);
1382      // Keep commands that CCR lists OR that are in the local-safe set
1383      setLocalCommands(prev => prev.filter(cmd => remoteCommandSet.has(cmd.name) || REMOTE_SAFE_COMMANDS.has(cmd)));
1384    }, [setLocalCommands]);
1385    const [inProgressToolUseIDs, setInProgressToolUseIDs] = useState<Set<string>>(new Set());
1386    const hasInterruptibleToolInProgressRef = useRef(false);
1387  
1388    // Remote session hook - manages WebSocket connection and message handling for --remote mode
1389    const remoteSession = useRemoteSession({
1390      config: remoteSessionConfig,
1391      setMessages,
1392      setIsLoading: setIsExternalLoading,
1393      onInit: handleRemoteInit,
1394      setToolUseConfirmQueue,
1395      tools: combinedInitialTools,
1396      setStreamingToolUses,
1397      setStreamMode,
1398      setInProgressToolUseIDs
1399    });
1400  
1401    // Direct connect hook - manages WebSocket to a claude server for `claude connect` mode
1402    const directConnect = useDirectConnect({
1403      config: directConnectConfig,
1404      setMessages,
1405      setIsLoading: setIsExternalLoading,
1406      setToolUseConfirmQueue,
1407      tools: combinedInitialTools
1408    });
1409  
1410    // SSH session hook - manages ssh child process for `claude ssh` mode.
1411    // Same callback shape as useDirectConnect; only the transport under the
1412    // hood differs (ChildProcess stdin/stdout vs WebSocket).
1413    const sshRemote = useSSHSession({
1414      session: sshSession,
1415      setMessages,
1416      setIsLoading: setIsExternalLoading,
1417      setToolUseConfirmQueue,
1418      tools: combinedInitialTools
1419    });
1420  
1421    // Use whichever remote mode is active
1422    const activeRemote = sshRemote.isRemoteMode ? sshRemote : directConnect.isRemoteMode ? directConnect : remoteSession;
1423    const [pastedContents, setPastedContents] = useState<Record<number, PastedContent>>({});
1424    const [submitCount, setSubmitCount] = useState(0);
1425    // Ref instead of state to avoid triggering React re-renders on every
1426    // streaming text_delta. The spinner reads this via its animation timer.
1427    const responseLengthRef = useRef(0);
1428    // API performance metrics ref for ant-only spinner display (TTFT/OTPS).
1429    // Accumulates metrics from all API requests in a turn for P50 aggregation.
1430    const apiMetricsRef = useRef<Array<{
1431      ttftMs: number;
1432      firstTokenTime: number;
1433      lastTokenTime: number;
1434      responseLengthBaseline: number;
1435      // Tracks responseLengthRef at the time of the last content addition.
1436      // Updated by both streaming deltas and subagent message content.
1437      // lastTokenTime is also updated at the same time, so the OTPS
1438      // denominator correctly includes subagent processing time.
1439      endResponseLength: number;
1440    }>>([]);
1441    const setResponseLength = useCallback((f: (prev: number) => number) => {
1442      const prev = responseLengthRef.current;
1443      responseLengthRef.current = f(prev);
1444      // When content is added (not a compaction reset), update the latest
1445      // metrics entry so OTPS reflects all content generation activity.
1446      // Updating lastTokenTime here ensures the denominator includes both
1447      // streaming time AND subagent execution time, preventing inflation.
1448      if (responseLengthRef.current > prev) {
1449        const entries = apiMetricsRef.current;
1450        if (entries.length > 0) {
1451          const lastEntry = entries.at(-1)!;
1452          lastEntry.lastTokenTime = Date.now();
1453          lastEntry.endResponseLength = responseLengthRef.current;
1454        }
1455      }
1456    }, []);
1457  
1458    // Streaming text display: set state directly per delta (Ink's 16ms render
1459    // throttle batches rapid updates). Cleared on message arrival (messages.ts)
1460    // so displayedMessages switches from deferredMessages to messages atomically.
1461    const [streamingText, setStreamingText] = useState<string | null>(null);
1462    const reducedMotion = useAppState(s => s.settings.prefersReducedMotion) ?? false;
1463    const showStreamingText = !reducedMotion && !hasCursorUpViewportYankBug();
1464    const onStreamingText = useCallback((f: (current: string | null) => string | null) => {
1465      if (!showStreamingText) return;
1466      setStreamingText(f);
1467    }, [showStreamingText]);
1468  
1469    // Hide the in-progress source line so text streams line-by-line, not
1470    // char-by-char. lastIndexOf returns -1 when no newline, giving '' → null.
1471    // Guard on showStreamingText so toggling reducedMotion mid-stream
1472    // immediately hides the streaming preview.
1473    const visibleStreamingText = streamingText && showStreamingText ? streamingText.substring(0, streamingText.lastIndexOf('\n') + 1) || null : null;
1474    const [lastQueryCompletionTime, setLastQueryCompletionTime] = useState(0);
1475    const [spinnerMessage, setSpinnerMessage] = useState<string | null>(null);
1476    const [spinnerColor, setSpinnerColor] = useState<keyof Theme | null>(null);
1477    const [spinnerShimmerColor, setSpinnerShimmerColor] = useState<keyof Theme | null>(null);
1478    const [isMessageSelectorVisible, setIsMessageSelectorVisible] = useState(false);
1479    const [messageSelectorPreselect, setMessageSelectorPreselect] = useState<UserMessage | undefined>(undefined);
1480    const [showCostDialog, setShowCostDialog] = useState(false);
1481    const [conversationId, setConversationId] = useState(randomUUID());
1482  
1483    // Idle-return dialog: shown when user submits after a long idle gap
1484    const [idleReturnPending, setIdleReturnPending] = useState<{
1485      input: string;
1486      idleMinutes: number;
1487    } | null>(null);
1488    const skipIdleCheckRef = useRef(false);
1489    const lastQueryCompletionTimeRef = useRef(lastQueryCompletionTime);
1490    lastQueryCompletionTimeRef.current = lastQueryCompletionTime;
1491  
1492    // Aggregate tool result budget: per-conversation decision tracking.
1493    // When the GrowthBook flag is on, query.ts enforces the budget; when
1494    // off (undefined), enforcement is skipped entirely. Stale entries after
1495    // /clear, rewind, or compact are harmless (tool_use_ids are UUIDs, stale
1496    // keys are never looked up). Memory is bounded by total replacement count
1497    // × ~2KB preview over the REPL lifetime — negligible.
1498    //
1499    // Lazy init via useState initializer — useRef(expr) evaluates expr on every
1500    // render (React ignores it after first, but the computation still runs).
1501    // For large resumed sessions, reconstruction does O(messages × blocks)
1502    // work; we only want that once.
1503    const [contentReplacementStateRef] = useState(() => ({
1504      current: provisionContentReplacementState(initialMessages, initialContentReplacements)
1505    }));
1506    const [haveShownCostDialog, setHaveShownCostDialog] = useState(getGlobalConfig().hasAcknowledgedCostThreshold);
1507    const [vimMode, setVimMode] = useState<VimMode>('INSERT');
1508    const [showBashesDialog, setShowBashesDialog] = useState<string | boolean>(false);
1509    const [isSearchingHistory, setIsSearchingHistory] = useState(false);
1510    const [isHelpOpen, setIsHelpOpen] = useState(false);
1511  
1512    // showBashesDialog is REPL-level so it survives PromptInput unmounting.
1513    // When ultraplan approval fires while the pill dialog is open, PromptInput
1514    // unmounts (focusedInputDialog → 'ultraplan-choice') but this stays true;
1515    // after accepting, PromptInput remounts into an empty "No tasks" dialog
1516    // (the completed ultraplan task has been filtered out). Close it here.
1517    useEffect(() => {
1518      if (ultraplanPendingChoice && showBashesDialog) {
1519        setShowBashesDialog(false);
1520      }
1521    }, [ultraplanPendingChoice, showBashesDialog]);
1522    const isTerminalFocused = useTerminalFocus();
1523    const terminalFocusRef = useRef(isTerminalFocused);
1524    terminalFocusRef.current = isTerminalFocused;
1525    const [theme] = useTheme();
1526  
1527    // resetLoadingState runs twice per turn (onQueryImpl tail + onQuery finally).
1528    // Without this guard, both calls pick a tip → two recordShownTip → two
1529    // saveGlobalConfig writes back-to-back. Reset at submit in onSubmit.
1530    const tipPickedThisTurnRef = React.useRef(false);
1531    const pickNewSpinnerTip = useCallback(() => {
1532      if (tipPickedThisTurnRef.current) return;
1533      tipPickedThisTurnRef.current = true;
1534      const newMessages = messagesRef.current.slice(bashToolsProcessedIdx.current);
1535      for (const tool of extractBashToolsFromMessages(newMessages)) {
1536        bashTools.current.add(tool);
1537      }
1538      bashToolsProcessedIdx.current = messagesRef.current.length;
1539      void getTipToShowOnSpinner({
1540        theme,
1541        readFileState: readFileState.current,
1542        bashTools: bashTools.current
1543      }).then(async tip => {
1544        if (tip) {
1545          const content = await tip.content({
1546            theme
1547          });
1548          setAppState(prev => ({
1549            ...prev,
1550            spinnerTip: content
1551          }));
1552          recordShownTip(tip);
1553        } else {
1554          setAppState(prev => {
1555            if (prev.spinnerTip === undefined) return prev;
1556            return {
1557              ...prev,
1558              spinnerTip: undefined
1559            };
1560          });
1561        }
1562      });
1563    }, [setAppState, theme]);
1564  
1565    // Resets UI loading state. Does NOT call onTurnComplete - that should be
1566    // called explicitly only when a query turn actually completes.
1567    const resetLoadingState = useCallback(() => {
1568      // isLoading is now derived from queryGuard — no setter call needed.
1569      // queryGuard.end() (onQuery finally) or cancelReservation() (executeUserInput
1570      // finally) have already transitioned the guard to idle by the time this runs.
1571      // External loading (remote/backgrounding) is reset separately by those hooks.
1572      setIsExternalLoading(false);
1573      setUserInputOnProcessing(undefined);
1574      responseLengthRef.current = 0;
1575      apiMetricsRef.current = [];
1576      setStreamingText(null);
1577      setStreamingToolUses([]);
1578      setSpinnerMessage(null);
1579      setSpinnerColor(null);
1580      setSpinnerShimmerColor(null);
1581      pickNewSpinnerTip();
1582      endInteractionSpan();
1583      // Speculative bash classifier checks are only valid for the current
1584      // turn's commands — clear after each turn to avoid accumulating
1585      // Promise chains for unconsumed checks (denied/aborted paths).
1586      clearSpeculativeChecks();
1587    }, [pickNewSpinnerTip]);
1588  
1589    // Session backgrounding — hook is below, after getToolUseContext
1590  
1591    const hasRunningTeammates = useMemo(() => getAllInProcessTeammateTasks(tasks).some(t => t.status === 'running'), [tasks]);
1592  
1593    // Show deferred turn duration message once all swarm teammates finish
1594    useEffect(() => {
1595      if (!hasRunningTeammates && swarmStartTimeRef.current !== null) {
1596        const totalMs = Date.now() - swarmStartTimeRef.current;
1597        const deferredBudget = swarmBudgetInfoRef.current;
1598        swarmStartTimeRef.current = null;
1599        swarmBudgetInfoRef.current = undefined;
1600        setMessages(prev => [...prev, createTurnDurationMessage(totalMs, deferredBudget,
1601        // Count only what recordTranscript will persist — ephemeral
1602        // progress ticks and non-ant attachments are filtered by
1603        // isLoggableMessage and never reach disk. Using raw prev.length
1604        // would make checkResumeConsistency report false delta<0 for
1605        // every turn that ran a progress-emitting tool.
1606        count(prev, isLoggableMessage))]);
1607      }
1608    }, [hasRunningTeammates, setMessages]);
1609  
1610    // Show auto permissions warning when entering auto mode
1611    // (either via Shift+Tab toggle or on startup). Debounced to avoid
1612    // flashing when the user is cycling through modes quickly.
1613    // Only shown 3 times total across sessions.
1614    const safeYoloMessageShownRef = useRef(false);
1615    useEffect(() => {
1616      if (feature('TRANSCRIPT_CLASSIFIER')) {
1617        if (toolPermissionContext.mode !== 'auto') {
1618          safeYoloMessageShownRef.current = false;
1619          return;
1620        }
1621        if (safeYoloMessageShownRef.current) return;
1622        const config = getGlobalConfig();
1623        const count = config.autoPermissionsNotificationCount ?? 0;
1624        if (count >= 3) return;
1625        const timer = setTimeout((ref, setMessages) => {
1626          ref.current = true;
1627          saveGlobalConfig(prev => {
1628            const prevCount = prev.autoPermissionsNotificationCount ?? 0;
1629            if (prevCount >= 3) return prev;
1630            return {
1631              ...prev,
1632              autoPermissionsNotificationCount: prevCount + 1
1633            };
1634          });
1635          setMessages(prev => [...prev, createSystemMessage(AUTO_MODE_DESCRIPTION, 'warning')]);
1636        }, 800, safeYoloMessageShownRef, setMessages);
1637        return () => clearTimeout(timer);
1638      }
1639    }, [toolPermissionContext.mode, setMessages]);
1640  
1641    // If worktree creation was slow and sparse-checkout isn't configured,
1642    // nudge the user toward settings.worktree.sparsePaths.
1643    const worktreeTipShownRef = useRef(false);
1644    useEffect(() => {
1645      if (worktreeTipShownRef.current) return;
1646      const wt = getCurrentWorktreeSession();
1647      if (!wt?.creationDurationMs || wt.usedSparsePaths) return;
1648      if (wt.creationDurationMs < 15_000) return;
1649      worktreeTipShownRef.current = true;
1650      const secs = Math.round(wt.creationDurationMs / 1000);
1651      setMessages(prev => [...prev, createSystemMessage(`Worktree creation took ${secs}s. For large repos, set \`worktree.sparsePaths\` in .claude/settings.json to check out only the directories you need — e.g. \`{"worktree": {"sparsePaths": ["src", "packages/foo"]}}\`.`, 'info')]);
1652    }, [setMessages]);
1653  
1654    // Hide spinner when the only in-progress tool is Sleep
1655    const onlySleepToolActive = useMemo(() => {
1656      const lastAssistant = messages.findLast(m => m.type === 'assistant');
1657      if (lastAssistant?.type !== 'assistant') return false;
1658      const inProgressToolUses = lastAssistant.message.content.filter(b => b.type === 'tool_use' && inProgressToolUseIDs.has(b.id));
1659      return inProgressToolUses.length > 0 && inProgressToolUses.every(b => b.type === 'tool_use' && b.name === SLEEP_TOOL_NAME);
1660    }, [messages, inProgressToolUseIDs]);
1661    const {
1662      onBeforeQuery: mrOnBeforeQuery,
1663      onTurnComplete: mrOnTurnComplete,
1664      render: mrRender
1665    } = useMoreRight({
1666      enabled: moreRightEnabled,
1667      setMessages,
1668      inputValue,
1669      setInputValue,
1670      setToolJSX
1671    });
1672    const showSpinner = (!toolJSX || toolJSX.showSpinner === true) && toolUseConfirmQueue.length === 0 && promptQueue.length === 0 && (
1673    // Show spinner during input processing, API call, while teammates are running,
1674    // or while pending task notifications are queued (prevents spinner bounce between consecutive notifications)
1675    isLoading || userInputOnProcessing || hasRunningTeammates ||
1676    // Keep spinner visible while task notifications are queued for processing.
1677    // Without this, the spinner briefly disappears between consecutive notifications
1678    // (e.g., multiple background agents completing in rapid succession) because
1679    // isLoading goes false momentarily between processing each one.
1680    getCommandQueueLength() > 0) &&
1681    // Hide spinner when waiting for leader to approve permission request
1682    !pendingWorkerRequest && !onlySleepToolActive && (
1683    // Hide spinner when streaming text is visible (the text IS the feedback),
1684    // but keep it when isBriefOnly suppresses the streaming text display
1685    !visibleStreamingText || isBriefOnly);
1686  
1687    // Check if any permission or ask question prompt is currently visible
1688    // This is used to prevent the survey from opening while prompts are active
1689    const hasActivePrompt = toolUseConfirmQueue.length > 0 || promptQueue.length > 0 || sandboxPermissionRequestQueue.length > 0 || elicitation.queue.length > 0 || workerSandboxPermissions.queue.length > 0;
1690    const feedbackSurveyOriginal = useFeedbackSurvey(messages, isLoading, submitCount, 'session', hasActivePrompt);
1691    const skillImprovementSurvey = useSkillImprovementSurvey(setMessages);
1692    const showIssueFlagBanner = useIssueFlagBanner(messages, submitCount);
1693  
1694    // Wrap feedback survey handler to trigger auto-run /issue
1695    const feedbackSurvey = useMemo(() => ({
1696      ...feedbackSurveyOriginal,
1697      handleSelect: (selected: 'dismissed' | 'bad' | 'fine' | 'good') => {
1698        // Reset the ref when a new survey response comes in
1699        didAutoRunIssueRef.current = false;
1700        const showedTranscriptPrompt = feedbackSurveyOriginal.handleSelect(selected);
1701        // Auto-run /issue for "bad" if transcript prompt wasn't shown
1702        if (selected === 'bad' && !showedTranscriptPrompt && shouldAutoRunIssue('feedback_survey_bad')) {
1703          setAutoRunIssueReason('feedback_survey_bad');
1704          didAutoRunIssueRef.current = true;
1705        }
1706      }
1707    }), [feedbackSurveyOriginal]);
1708  
1709    // Post-compact survey: shown after compaction if feature gate is enabled
1710    const postCompactSurvey = usePostCompactSurvey(messages, isLoading, hasActivePrompt, {
1711      enabled: !isRemoteSession
1712    });
1713  
1714    // Memory survey: shown when the assistant mentions memory and a memory file
1715    // was read this conversation
1716    const memorySurvey = useMemorySurvey(messages, isLoading, hasActivePrompt, {
1717      enabled: !isRemoteSession
1718    });
1719  
1720    // Frustration detection: show transcript sharing prompt after detecting frustrated messages
1721    const frustrationDetection = useFrustrationDetection(messages, isLoading, hasActivePrompt, feedbackSurvey.state !== 'closed' || postCompactSurvey.state !== 'closed' || memorySurvey.state !== 'closed');
1722  
1723    // Initialize IDE integration
1724    useIDEIntegration({
1725      autoConnectIdeFlag,
1726      ideToInstallExtension,
1727      setDynamicMcpConfig,
1728      setShowIdeOnboarding,
1729      setIDEInstallationState: setIDEInstallationStatus
1730    });
1731    useFileHistorySnapshotInit(initialFileHistorySnapshots, fileHistory, fileHistoryState => setAppState(prev => ({
1732      ...prev,
1733      fileHistory: fileHistoryState
1734    })));
1735    const resume = useCallback(async (sessionId: UUID, log: LogOption, entrypoint: ResumeEntrypoint) => {
1736      const resumeStart = performance.now();
1737      try {
1738        // Deserialize messages to properly clean up the conversation
1739        // This filters unresolved tool uses and adds a synthetic assistant message if needed
1740        const messages = deserializeMessages(log.messages);
1741  
1742        // Match coordinator/normal mode to the resumed session
1743        if (feature('COORDINATOR_MODE')) {
1744          /* eslint-disable @typescript-eslint/no-require-imports */
1745          const coordinatorModule = require('../coordinator/coordinatorMode.js') as typeof import('../coordinator/coordinatorMode.js');
1746          /* eslint-enable @typescript-eslint/no-require-imports */
1747          const warning = coordinatorModule.matchSessionMode(log.mode);
1748          if (warning) {
1749            // Re-derive agent definitions after mode switch so built-in agents
1750            // reflect the new coordinator/normal mode
1751            /* eslint-disable @typescript-eslint/no-require-imports */
1752            const {
1753              getAgentDefinitionsWithOverrides,
1754              getActiveAgentsFromList
1755            } = require('../tools/AgentTool/loadAgentsDir.js') as typeof import('../tools/AgentTool/loadAgentsDir.js');
1756            /* eslint-enable @typescript-eslint/no-require-imports */
1757            getAgentDefinitionsWithOverrides.cache.clear?.();
1758            const freshAgentDefs = await getAgentDefinitionsWithOverrides(getOriginalCwd());
1759            setAppState(prev => ({
1760              ...prev,
1761              agentDefinitions: {
1762                ...freshAgentDefs,
1763                allAgents: freshAgentDefs.allAgents,
1764                activeAgents: getActiveAgentsFromList(freshAgentDefs.allAgents)
1765              }
1766            }));
1767            messages.push(createSystemMessage(warning, 'warning'));
1768          }
1769        }
1770  
1771        // Fire SessionEnd hooks for the current session before starting the
1772        // resumed one, mirroring the /clear flow in conversation.ts.
1773        const sessionEndTimeoutMs = getSessionEndHookTimeoutMs();
1774        await executeSessionEndHooks('resume', {
1775          getAppState: () => store.getState(),
1776          setAppState,
1777          signal: AbortSignal.timeout(sessionEndTimeoutMs),
1778          timeoutMs: sessionEndTimeoutMs
1779        });
1780  
1781        // Process session start hooks for resume
1782        const hookMessages = await processSessionStartHooks('resume', {
1783          sessionId,
1784          agentType: mainThreadAgentDefinition?.agentType,
1785          model: mainLoopModel
1786        });
1787  
1788        // Append hook messages to the conversation
1789        messages.push(...hookMessages);
1790        // For forks, generate a new plan slug and copy the plan content so the
1791        // original and forked sessions don't clobber each other's plan files.
1792        // For regular resumes, reuse the original session's plan slug.
1793        if (entrypoint === 'fork') {
1794          void copyPlanForFork(log, asSessionId(sessionId));
1795        } else {
1796          void copyPlanForResume(log, asSessionId(sessionId));
1797        }
1798  
1799        // Restore file history and attribution state from the resumed conversation
1800        restoreSessionStateFromLog(log, setAppState);
1801        if (log.fileHistorySnapshots) {
1802          void copyFileHistoryForResume(log);
1803        }
1804  
1805        // Restore agent setting from the resumed conversation
1806        // Always reset to the new session's values (or clear if none),
1807        // matching the standaloneAgentContext pattern below
1808        const {
1809          agentDefinition: restoredAgent
1810        } = restoreAgentFromSession(log.agentSetting, initialMainThreadAgentDefinition, agentDefinitions);
1811        setMainThreadAgentDefinition(restoredAgent);
1812        setAppState(prev => ({
1813          ...prev,
1814          agent: restoredAgent?.agentType
1815        }));
1816  
1817        // Restore standalone agent context from the resumed conversation
1818        // Always reset to the new session's values (or clear if none)
1819        setAppState(prev => ({
1820          ...prev,
1821          standaloneAgentContext: computeStandaloneAgentContext(log.agentName, log.agentColor)
1822        }));
1823        void updateSessionName(log.agentName);
1824  
1825        // Restore read file state from the message history
1826        restoreReadFileState(messages, log.projectPath ?? getOriginalCwd());
1827  
1828        // Clear any active loading state (no queryId since we're not in a query)
1829        resetLoadingState();
1830        setAbortController(null);
1831        setConversationId(sessionId);
1832  
1833        // Get target session's costs BEFORE saving current session
1834        // (saveCurrentSessionCosts overwrites the config, so we need to read first)
1835        const targetSessionCosts = getStoredSessionCosts(sessionId);
1836  
1837        // Save current session's costs before switching to avoid losing accumulated costs
1838        saveCurrentSessionCosts();
1839  
1840        // Reset cost state for clean slate before restoring target session
1841        resetCostState();
1842  
1843        // Switch session (id + project dir atomically). fullPath may point to
1844        // a different project (cross-worktree, /branch); null derives from
1845        // current originalCwd.
1846        switchSession(asSessionId(sessionId), log.fullPath ? dirname(log.fullPath) : null);
1847        // Rename asciicast recording to match the resumed session ID
1848        const {
1849          renameRecordingForSession
1850        } = await import('../utils/asciicast.js');
1851        await renameRecordingForSession();
1852        await resetSessionFilePointer();
1853  
1854        // Clear then restore session metadata so it's re-appended on exit via
1855        // reAppendSessionMetadata. clearSessionMetadata must be called first:
1856        // restoreSessionMetadata only sets-if-truthy, so without the clear,
1857        // a session without an agent name would inherit the previous session's
1858        // cached name and write it to the wrong transcript on first message.
1859        clearSessionMetadata();
1860        restoreSessionMetadata(log);
1861        // Resumed sessions shouldn't re-title from mid-conversation context
1862        // (same reasoning as the useRef seed), and the previous session's
1863        // Haiku title shouldn't carry over.
1864        haikuTitleAttemptedRef.current = true;
1865        setHaikuTitle(undefined);
1866  
1867        // Exit any worktree a prior /resume entered, then cd into the one
1868        // this session was in. Without the exit, resuming from worktree B
1869        // to non-worktree C leaves cwd/currentWorktreeSession stale;
1870        // resuming B→C where C is also a worktree fails entirely
1871        // (getCurrentWorktreeSession guard blocks the switch).
1872        //
1873        // Skipped for /branch: forkLog doesn't carry worktreeSession, so
1874        // this would kick the user out of a worktree they're still working
1875        // in. Same fork skip as processResumedConversation for the adopt —
1876        // fork materializes its own file via recordTranscript on REPL mount.
1877        if (entrypoint !== 'fork') {
1878          exitRestoredWorktree();
1879          restoreWorktreeForResume(log.worktreeSession);
1880          adoptResumedSessionFile();
1881          void restoreRemoteAgentTasks({
1882            abortController: new AbortController(),
1883            getAppState: () => store.getState(),
1884            setAppState
1885          });
1886        } else {
1887          // Fork: same re-persist as /clear (conversation.ts). The clear
1888          // above wiped currentSessionWorktree, forkLog doesn't carry it,
1889          // and the process is still in the same worktree.
1890          const ws = getCurrentWorktreeSession();
1891          if (ws) saveWorktreeState(ws);
1892        }
1893  
1894        // Persist the current mode so future resumes know what mode this session was in
1895        if (feature('COORDINATOR_MODE')) {
1896          /* eslint-disable @typescript-eslint/no-require-imports */
1897          const {
1898            saveMode
1899          } = require('../utils/sessionStorage.js');
1900          const {
1901            isCoordinatorMode
1902          } = require('../coordinator/coordinatorMode.js') as typeof import('../coordinator/coordinatorMode.js');
1903          /* eslint-enable @typescript-eslint/no-require-imports */
1904          saveMode(isCoordinatorMode() ? 'coordinator' : 'normal');
1905        }
1906  
1907        // Restore target session's costs from the data we read earlier
1908        if (targetSessionCosts) {
1909          setCostStateForRestore(targetSessionCosts);
1910        }
1911  
1912        // Reconstruct replacement state for the resumed session. Runs after
1913        // setSessionId so any NEW replacements post-resume write to the
1914        // resumed session's tool-results dir. Gated on ref.current: the
1915        // initial mount already read the feature flag, so we don't re-read
1916        // it here (mid-session flag flips stay unobservable in both
1917        // directions).
1918        //
1919        // Skipped for in-session /branch: the existing ref is already correct
1920        // (branch preserves tool_use_ids), so there's no need to reconstruct.
1921        // createFork() does write content-replacement entries to the forked
1922        // JSONL with the fork's sessionId, so `claude -r {forkId}` also works.
1923        if (contentReplacementStateRef.current && entrypoint !== 'fork') {
1924          contentReplacementStateRef.current = reconstructContentReplacementState(messages, log.contentReplacements ?? []);
1925        }
1926  
1927        // Reset messages to the provided initial messages
1928        // Use a callback to ensure we're not dependent on stale state
1929        setMessages(() => messages);
1930  
1931        // Clear any active tool JSX
1932        setToolJSX(null);
1933  
1934        // Clear input to ensure no residual state
1935        setInputValue('');
1936        logEvent('tengu_session_resumed', {
1937          entrypoint: entrypoint as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1938          success: true,
1939          resume_duration_ms: Math.round(performance.now() - resumeStart)
1940        });
1941      } catch (error) {
1942        logEvent('tengu_session_resumed', {
1943          entrypoint: entrypoint as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1944          success: false
1945        });
1946        throw error;
1947      }
1948    }, [resetLoadingState, setAppState]);
1949  
1950    // Lazy init: useRef(createX()) would call createX on every render and
1951    // discard the result. LRUCache construction inside FileStateCache is
1952    // expensive (~170ms), so we use useState's lazy initializer to create
1953    // it exactly once, then feed that stable reference into useRef.
1954    const [initialReadFileState] = useState(() => createFileStateCacheWithSizeLimit(READ_FILE_STATE_CACHE_SIZE));
1955    const readFileState = useRef(initialReadFileState);
1956    const bashTools = useRef(new Set<string>());
1957    const bashToolsProcessedIdx = useRef(0);
1958    // Session-scoped skill discovery tracking (feeds was_discovered on
1959    // tengu_skill_tool_invocation). Must persist across getToolUseContext
1960    // rebuilds within a session: turn-0 discovery writes via processUserInput
1961    // before onQuery builds its own context, and discovery on turn N must
1962    // still attribute a SkillTool call on turn N+k. Cleared in clearConversation.
1963    const discoveredSkillNamesRef = useRef(new Set<string>());
1964    // Session-level dedup for nested_memory CLAUDE.md attachments.
1965    // readFileState is a 100-entry LRU; once it evicts a CLAUDE.md path,
1966    // the next discovery cycle re-injects it. Cleared in clearConversation.
1967    const loadedNestedMemoryPathsRef = useRef(new Set<string>());
1968  
1969    // Helper to restore read file state from messages (used for resume flows)
1970    // This allows Claude to edit files that were read in previous sessions
1971    const restoreReadFileState = useCallback((messages: MessageType[], cwd: string) => {
1972      const extracted = extractReadFilesFromMessages(messages, cwd, READ_FILE_STATE_CACHE_SIZE);
1973      readFileState.current = mergeFileStateCaches(readFileState.current, extracted);
1974      for (const tool of extractBashToolsFromMessages(messages)) {
1975        bashTools.current.add(tool);
1976      }
1977    }, []);
1978  
1979    // Extract read file state from initialMessages on mount
1980    // This handles CLI flag resume (--resume-session) and ResumeConversation screen
1981    // where messages are passed as props rather than through the resume callback
1982    useEffect(() => {
1983      if (initialMessages && initialMessages.length > 0) {
1984        restoreReadFileState(initialMessages, getOriginalCwd());
1985        void restoreRemoteAgentTasks({
1986          abortController: new AbortController(),
1987          getAppState: () => store.getState(),
1988          setAppState
1989        });
1990      }
1991      // Only run on mount - initialMessages shouldn't change during component lifetime
1992      // eslint-disable-next-line react-hooks/exhaustive-deps
1993    }, []);
1994    const {
1995      status: apiKeyStatus,
1996      reverify
1997    } = useApiKeyVerification();
1998  
1999    // Auto-run /issue state
2000    const [autoRunIssueReason, setAutoRunIssueReason] = useState<AutoRunIssueReason | null>(null);
2001    // Ref to track if autoRunIssue was triggered this survey cycle,
2002    // so we can suppress the [1] follow-up prompt even after
2003    // autoRunIssueReason is cleared.
2004    const didAutoRunIssueRef = useRef(false);
2005  
2006    // State for exit feedback flow
2007    const [exitFlow, setExitFlow] = useState<React.ReactNode>(null);
2008    const [isExiting, setIsExiting] = useState(false);
2009  
2010    // Calculate if cost dialog should be shown
2011    const showingCostDialog = !isLoading && showCostDialog;
2012  
2013    // Determine which dialog should have focus (if any)
2014    // Permission and interactive dialogs can show even when toolJSX is set,
2015    // as long as shouldContinueAnimation is true. This prevents deadlocks when
2016    // agents set background hints while waiting for user interaction.
2017    function getFocusedInputDialog(): 'message-selector' | 'sandbox-permission' | 'tool-permission' | 'prompt' | 'worker-sandbox-permission' | 'elicitation' | 'cost' | 'idle-return' | 'init-onboarding' | 'ide-onboarding' | 'model-switch' | 'undercover-callout' | 'effort-callout' | 'remote-callout' | 'lsp-recommendation' | 'plugin-hint' | 'desktop-upsell' | 'ultraplan-choice' | 'ultraplan-launch' | undefined {
2018      // Exit states always take precedence
2019      if (isExiting || exitFlow) return undefined;
2020  
2021      // High priority dialogs (always show regardless of typing)
2022      if (isMessageSelectorVisible) return 'message-selector';
2023  
2024      // Suppress interrupt dialogs while user is actively typing
2025      if (isPromptInputActive) return undefined;
2026      if (sandboxPermissionRequestQueue[0]) return 'sandbox-permission';
2027  
2028      // Permission/interactive dialogs (show unless blocked by toolJSX)
2029      const allowDialogsWithAnimation = !toolJSX || toolJSX.shouldContinueAnimation;
2030      if (allowDialogsWithAnimation && toolUseConfirmQueue[0]) return 'tool-permission';
2031      if (allowDialogsWithAnimation && promptQueue[0]) return 'prompt';
2032      // Worker sandbox permission prompts (network access) from swarm workers
2033      if (allowDialogsWithAnimation && workerSandboxPermissions.queue[0]) return 'worker-sandbox-permission';
2034      if (allowDialogsWithAnimation && elicitation.queue[0]) return 'elicitation';
2035      if (allowDialogsWithAnimation && showingCostDialog) return 'cost';
2036      if (allowDialogsWithAnimation && idleReturnPending) return 'idle-return';
2037      if (feature('ULTRAPLAN') && allowDialogsWithAnimation && !isLoading && ultraplanPendingChoice) return 'ultraplan-choice';
2038      if (feature('ULTRAPLAN') && allowDialogsWithAnimation && !isLoading && ultraplanLaunchPending) return 'ultraplan-launch';
2039  
2040      // Onboarding dialogs (special conditions)
2041      if (allowDialogsWithAnimation && showIdeOnboarding) return 'ide-onboarding';
2042  
2043      // Model switch callout (ant-only, eliminated from external builds)
2044      if ("external" === 'ant' && allowDialogsWithAnimation && showModelSwitchCallout) return 'model-switch';
2045  
2046      // Undercover auto-enable explainer (ant-only, eliminated from external builds)
2047      if ("external" === 'ant' && allowDialogsWithAnimation && showUndercoverCallout) return 'undercover-callout';
2048  
2049      // Effort callout (shown once for Opus 4.6 users when effort is enabled)
2050      if (allowDialogsWithAnimation && showEffortCallout) return 'effort-callout';
2051  
2052      // Remote callout (shown once before first bridge enable)
2053      if (allowDialogsWithAnimation && showRemoteCallout) return 'remote-callout';
2054  
2055      // LSP plugin recommendation (lowest priority - non-blocking suggestion)
2056      if (allowDialogsWithAnimation && lspRecommendation) return 'lsp-recommendation';
2057  
2058      // Plugin hint from CLI/SDK stderr (same priority band as LSP rec)
2059      if (allowDialogsWithAnimation && hintRecommendation) return 'plugin-hint';
2060  
2061      // Desktop app upsell (max 3 launches, lowest priority)
2062      if (allowDialogsWithAnimation && showDesktopUpsellStartup) return 'desktop-upsell';
2063      return undefined;
2064    }
2065    const focusedInputDialog = getFocusedInputDialog();
2066  
2067    // True when permission prompts exist but are hidden because the user is typing
2068    const hasSuppressedDialogs = isPromptInputActive && (sandboxPermissionRequestQueue[0] || toolUseConfirmQueue[0] || promptQueue[0] || workerSandboxPermissions.queue[0] || elicitation.queue[0] || showingCostDialog);
2069  
2070    // Keep ref in sync so timer callbacks can read the current value
2071    focusedInputDialogRef.current = focusedInputDialog;
2072  
2073    // Immediately capture pause/resume when focusedInputDialog changes
2074    // This ensures accurate timing even under high system load, rather than
2075    // relying on the 100ms polling interval to detect state changes
2076    useEffect(() => {
2077      if (!isLoading) return;
2078      const isPaused = focusedInputDialog === 'tool-permission';
2079      const now = Date.now();
2080      if (isPaused && pauseStartTimeRef.current === null) {
2081        // Just entered pause state - record the exact moment
2082        pauseStartTimeRef.current = now;
2083      } else if (!isPaused && pauseStartTimeRef.current !== null) {
2084        // Just exited pause state - accumulate paused time immediately
2085        totalPausedMsRef.current += now - pauseStartTimeRef.current;
2086        pauseStartTimeRef.current = null;
2087      }
2088    }, [focusedInputDialog, isLoading]);
2089  
2090    // Re-pin scroll to bottom whenever the permission overlay appears or
2091    // dismisses. Overlay now renders below messages inside the same
2092    // ScrollBox (no remount), so we need an explicit scrollToBottom for:
2093    //  - appear: user may have been scrolled up (sticky broken) — the
2094    //    dialog is blocking and must be visible
2095    //  - dismiss: user may have scrolled up to read context during the
2096    //    overlay, and onScroll was suppressed so the pill state is stale
2097    // useLayoutEffect so the re-pin commits before the Ink frame renders —
2098    // no 1-frame flash of the wrong scroll position.
2099    const prevDialogRef = useRef(focusedInputDialog);
2100    useLayoutEffect(() => {
2101      const was = prevDialogRef.current === 'tool-permission';
2102      const now = focusedInputDialog === 'tool-permission';
2103      if (was !== now) repinScroll();
2104      prevDialogRef.current = focusedInputDialog;
2105    }, [focusedInputDialog, repinScroll]);
2106    function onCancel() {
2107      if (focusedInputDialog === 'elicitation') {
2108        // Elicitation dialog handles its own Escape, and closing it shouldn't affect any loading state.
2109        return;
2110      }
2111      logForDebugging(`[onCancel] focusedInputDialog=${focusedInputDialog} streamMode=${streamMode}`);
2112  
2113      // Pause proactive mode so the user gets control back.
2114      // It will resume when they submit their next input (see onSubmit).
2115      if (feature('PROACTIVE') || feature('KAIROS')) {
2116        proactiveModule?.pauseProactive();
2117      }
2118      queryGuard.forceEnd();
2119      skipIdleCheckRef.current = false;
2120  
2121      // Preserve partially-streamed text so the user can read what was
2122      // generated before pressing Esc. Pushed before resetLoadingState clears
2123      // streamingText, and before query.ts yields the async interrupt marker,
2124      // giving final order [user, partial-assistant, [Request interrupted by user]].
2125      if (streamingText?.trim()) {
2126        setMessages(prev => [...prev, createAssistantMessage({
2127          content: streamingText
2128        })]);
2129      }
2130      resetLoadingState();
2131  
2132      // Clear any active token budget so the backstop doesn't fire on
2133      // a stale budget if the query generator hasn't exited yet.
2134      if (feature('TOKEN_BUDGET')) {
2135        snapshotOutputTokensForTurn(null);
2136      }
2137      if (focusedInputDialog === 'tool-permission') {
2138        // Tool use confirm handles the abort signal itself
2139        toolUseConfirmQueue[0]?.onAbort();
2140        setToolUseConfirmQueue([]);
2141      } else if (focusedInputDialog === 'prompt') {
2142        // Reject all pending prompts and clear the queue
2143        for (const item of promptQueue) {
2144          item.reject(new Error('Prompt cancelled by user'));
2145        }
2146        setPromptQueue([]);
2147        abortController?.abort('user-cancel');
2148      } else if (activeRemote.isRemoteMode) {
2149        // Remote mode: send interrupt signal to CCR
2150        activeRemote.cancelRequest();
2151      } else {
2152        abortController?.abort('user-cancel');
2153      }
2154  
2155      // Clear the controller so subsequent Escape presses don't see a stale
2156      // aborted signal. Without this, canCancelRunningTask is false (signal
2157      // defined but .aborted === true), so isActive becomes false if no other
2158      // activating conditions hold — leaving the Escape keybinding inactive.
2159      setAbortController(null);
2160  
2161      // forceEnd() skips the finally path — fire directly (aborted=true).
2162      void mrOnTurnComplete(messagesRef.current, true);
2163    }
2164  
2165    // Function to handle queued command when canceling a permission request
2166    const handleQueuedCommandOnCancel = useCallback(() => {
2167      const result = popAllEditable(inputValue, 0);
2168      if (!result) return;
2169      setInputValue(result.text);
2170      setInputMode('prompt');
2171  
2172      // Restore images from queued commands to pastedContents
2173      if (result.images.length > 0) {
2174        setPastedContents(prev => {
2175          const newContents = {
2176            ...prev
2177          };
2178          for (const image of result.images) {
2179            newContents[image.id] = image;
2180          }
2181          return newContents;
2182        });
2183      }
2184    }, [setInputValue, setInputMode, inputValue, setPastedContents]);
2185  
2186    // CancelRequestHandler props - rendered inside KeybindingSetup
2187    const cancelRequestProps = {
2188      setToolUseConfirmQueue,
2189      onCancel,
2190      onAgentsKilled: () => setMessages(prev => [...prev, createAgentsKilledMessage()]),
2191      isMessageSelectorVisible: isMessageSelectorVisible || !!showBashesDialog,
2192      screen,
2193      abortSignal: abortController?.signal,
2194      popCommandFromQueue: handleQueuedCommandOnCancel,
2195      vimMode,
2196      isLocalJSXCommand: toolJSX?.isLocalJSXCommand,
2197      isSearchingHistory,
2198      isHelpOpen,
2199      inputMode,
2200      inputValue,
2201      streamMode
2202    };
2203    useEffect(() => {
2204      const totalCost = getTotalCost();
2205      if (totalCost >= 5 /* $5 */ && !showCostDialog && !haveShownCostDialog) {
2206        logEvent('tengu_cost_threshold_reached', {});
2207        // Mark as shown even if the dialog won't render (no console billing
2208        // access). Otherwise this effect re-fires on every message change for
2209        // the rest of the session — 200k+ spurious events observed.
2210        setHaveShownCostDialog(true);
2211        if (hasConsoleBillingAccess()) {
2212          setShowCostDialog(true);
2213        }
2214      }
2215    }, [messages, showCostDialog, haveShownCostDialog]);
2216    const sandboxAskCallback: SandboxAskCallback = useCallback(async (hostPattern: NetworkHostPattern) => {
2217      // If running as a swarm worker, forward the request to the leader via mailbox
2218      if (isAgentSwarmsEnabled() && isSwarmWorker()) {
2219        const requestId = generateSandboxRequestId();
2220  
2221        // Send the request to the leader via mailbox
2222        const sent = await sendSandboxPermissionRequestViaMailbox(hostPattern.host, requestId);
2223        return new Promise(resolveShouldAllowHost => {
2224          if (!sent) {
2225            // If we couldn't send via mailbox, fall back to local handling
2226            setSandboxPermissionRequestQueue(prev => [...prev, {
2227              hostPattern,
2228              resolvePromise: resolveShouldAllowHost
2229            }]);
2230            return;
2231          }
2232  
2233          // Register the callback for when the leader responds
2234          registerSandboxPermissionCallback({
2235            requestId,
2236            host: hostPattern.host,
2237            resolve: resolveShouldAllowHost
2238          });
2239  
2240          // Update AppState to show pending indicator
2241          setAppState(prev => ({
2242            ...prev,
2243            pendingSandboxRequest: {
2244              requestId,
2245              host: hostPattern.host
2246            }
2247          }));
2248        });
2249      }
2250  
2251      // Normal flow for non-workers: show local UI and optionally race
2252      // against the REPL bridge (Remote Control) if connected.
2253      return new Promise(resolveShouldAllowHost => {
2254        let resolved = false;
2255        function resolveOnce(allow: boolean): void {
2256          if (resolved) return;
2257          resolved = true;
2258          resolveShouldAllowHost(allow);
2259        }
2260  
2261        // Queue the local sandbox permission dialog
2262        setSandboxPermissionRequestQueue(prev => [...prev, {
2263          hostPattern,
2264          resolvePromise: resolveOnce
2265        }]);
2266  
2267        // When the REPL bridge is connected, also forward the sandbox
2268        // permission request as a can_use_tool control_request so the
2269        // remote user (e.g. on claude.ai) can approve it too.
2270        if (feature('BRIDGE_MODE')) {
2271          const bridgeCallbacks = store.getState().replBridgePermissionCallbacks;
2272          if (bridgeCallbacks) {
2273            const bridgeRequestId = randomUUID();
2274            bridgeCallbacks.sendRequest(bridgeRequestId, SANDBOX_NETWORK_ACCESS_TOOL_NAME, {
2275              host: hostPattern.host
2276            }, randomUUID(), `Allow network connection to ${hostPattern.host}?`);
2277            const unsubscribe = bridgeCallbacks.onResponse(bridgeRequestId, response => {
2278              unsubscribe();
2279              const allow = response.behavior === 'allow';
2280              // Resolve ALL pending requests for the same host, not just
2281              // this one — mirrors the local dialog handler pattern.
2282              setSandboxPermissionRequestQueue(queue => {
2283                queue.filter(item => item.hostPattern.host === hostPattern.host).forEach(item => item.resolvePromise(allow));
2284                return queue.filter(item => item.hostPattern.host !== hostPattern.host);
2285              });
2286              // Clean up all sibling bridge subscriptions for this host
2287              // (other concurrent same-host requests) before deleting.
2288              const siblingCleanups = sandboxBridgeCleanupRef.current.get(hostPattern.host);
2289              if (siblingCleanups) {
2290                for (const fn of siblingCleanups) {
2291                  fn();
2292                }
2293                sandboxBridgeCleanupRef.current.delete(hostPattern.host);
2294              }
2295            });
2296  
2297            // Register cleanup so the local dialog handler can cancel
2298            // the remote prompt and unsubscribe when the local user
2299            // responds first.
2300            const cleanup = () => {
2301              unsubscribe();
2302              bridgeCallbacks.cancelRequest(bridgeRequestId);
2303            };
2304            const existing = sandboxBridgeCleanupRef.current.get(hostPattern.host) ?? [];
2305            existing.push(cleanup);
2306            sandboxBridgeCleanupRef.current.set(hostPattern.host, existing);
2307          }
2308        }
2309      });
2310    }, [setAppState, store]);
2311  
2312    // #34044: if user explicitly set sandbox.enabled=true but deps are missing,
2313    // isSandboxingEnabled() returns false silently. Surface the reason once at
2314    // mount so users know their security config isn't being enforced. Full
2315    // reason goes to debug log; notification points to /sandbox for details.
2316    // addNotification is stable (useCallback) so the effect fires once.
2317    useEffect(() => {
2318      const reason = SandboxManager.getSandboxUnavailableReason();
2319      if (!reason) return;
2320      if (SandboxManager.isSandboxRequired()) {
2321        process.stderr.write(`\nError: sandbox required but unavailable: ${reason}\n` + `  sandbox.failIfUnavailable is set — refusing to start without a working sandbox.\n\n`);
2322        gracefulShutdownSync(1, 'other');
2323        return;
2324      }
2325      logForDebugging(`sandbox disabled: ${reason}`, {
2326        level: 'warn'
2327      });
2328      addNotification({
2329        key: 'sandbox-unavailable',
2330        jsx: <>
2331            <Text color="warning">sandbox disabled</Text>
2332            <Text dimColor> · /sandbox</Text>
2333          </>,
2334        priority: 'medium'
2335      });
2336    }, [addNotification]);
2337    if (SandboxManager.isSandboxingEnabled()) {
2338      // If sandboxing is enabled (setting.sandbox is defined, initialise the manager)
2339      SandboxManager.initialize(sandboxAskCallback).catch(err => {
2340        // Initialization/validation failed - display error and exit
2341        process.stderr.write(`\n❌ Sandbox Error: ${errorMessage(err)}\n`);
2342        gracefulShutdownSync(1, 'other');
2343      });
2344    }
2345    const setToolPermissionContext = useCallback((context: ToolPermissionContext, options?: {
2346      preserveMode?: boolean;
2347    }) => {
2348      setAppState(prev => ({
2349        ...prev,
2350        toolPermissionContext: {
2351          ...context,
2352          // Preserve the coordinator's mode only when explicitly requested.
2353          // Workers' getAppState() returns a transformed context with mode
2354          // 'acceptEdits' that must not leak into the coordinator's actual
2355          // state via permission-rule updates — those call sites pass
2356          // { preserveMode: true }. User-initiated mode changes (e.g.,
2357          // selecting "allow all edits") must NOT be overridden.
2358          mode: options?.preserveMode ? prev.toolPermissionContext.mode : context.mode
2359        }
2360      }));
2361  
2362      // When permission context changes, recheck all queued items
2363      // This handles the case where approving item1 with "don't ask again"
2364      // should auto-approve other queued items that now match the updated rules
2365      setImmediate(setToolUseConfirmQueue => {
2366        // Use setToolUseConfirmQueue callback to get current queue state
2367        // instead of capturing it in the closure, to avoid stale closure issues
2368        setToolUseConfirmQueue(currentQueue => {
2369          currentQueue.forEach(item => {
2370            void item.recheckPermission();
2371          });
2372          return currentQueue;
2373        });
2374      }, setToolUseConfirmQueue);
2375    }, [setAppState, setToolUseConfirmQueue]);
2376  
2377    // Register the leader's setToolPermissionContext for in-process teammates
2378    useEffect(() => {
2379      registerLeaderSetToolPermissionContext(setToolPermissionContext);
2380      return () => unregisterLeaderSetToolPermissionContext();
2381    }, [setToolPermissionContext]);
2382    const canUseTool = useCanUseTool(setToolUseConfirmQueue, setToolPermissionContext);
2383    const requestPrompt = useCallback((title: string, toolInputSummary?: string | null) => (request: PromptRequest): Promise<PromptResponse> => new Promise<PromptResponse>((resolve, reject) => {
2384      setPromptQueue(prev => [...prev, {
2385        request,
2386        title,
2387        toolInputSummary,
2388        resolve,
2389        reject
2390      }]);
2391    }), []);
2392    const getToolUseContext = useCallback((messages: MessageType[], newMessages: MessageType[], abortController: AbortController, mainLoopModel: string): ProcessUserInputContext => {
2393      // Read mutable values fresh from the store rather than closure-capturing
2394      // useAppState() snapshots. Same values today (closure is refreshed by the
2395      // render between turns); decouples freshness from React's render cycle for
2396      // a future headless conversation loop. Same pattern refreshTools() uses.
2397      const s = store.getState();
2398  
2399      // Compute tools fresh from store.getState() rather than the closure-
2400      // captured `tools`. useManageMCPConnections populates appState.mcp
2401      // async as servers connect — the store may have newer MCP state than
2402      // the closure captured at render time. Also doubles as refreshTools()
2403      // for mid-query tool list updates.
2404      const computeTools = () => {
2405        const state = store.getState();
2406        const assembled = assembleToolPool(state.toolPermissionContext, state.mcp.tools);
2407        const merged = mergeAndFilterTools(combinedInitialTools, assembled, state.toolPermissionContext.mode);
2408        if (!mainThreadAgentDefinition) return merged;
2409        return resolveAgentTools(mainThreadAgentDefinition, merged, false, true).resolvedTools;
2410      };
2411      return {
2412        abortController,
2413        options: {
2414          commands,
2415          tools: computeTools(),
2416          debug,
2417          verbose: s.verbose,
2418          mainLoopModel,
2419          thinkingConfig: s.thinkingEnabled !== false ? thinkingConfig : {
2420            type: 'disabled'
2421          },
2422          // Merge fresh from store rather than closing over useMergedClients'
2423          // memoized output. initialMcpClients is a prop (session-constant).
2424          mcpClients: mergeClients(initialMcpClients, s.mcp.clients),
2425          mcpResources: s.mcp.resources,
2426          ideInstallationStatus: ideInstallationStatus,
2427          isNonInteractiveSession: false,
2428          dynamicMcpConfig,
2429          theme,
2430          agentDefinitions: allowedAgentTypes ? {
2431            ...s.agentDefinitions,
2432            allowedAgentTypes
2433          } : s.agentDefinitions,
2434          customSystemPrompt,
2435          appendSystemPrompt,
2436          refreshTools: computeTools
2437        },
2438        getAppState: () => store.getState(),
2439        setAppState,
2440        messages,
2441        setMessages,
2442        updateFileHistoryState(updater: (prev: FileHistoryState) => FileHistoryState) {
2443          // Perf: skip the setState when the updater returns the same reference
2444          // (e.g. fileHistoryTrackEdit returns `state` when the file is already
2445          // tracked). Otherwise every no-op call would notify all store listeners.
2446          setAppState(prev => {
2447            const updated = updater(prev.fileHistory);
2448            if (updated === prev.fileHistory) return prev;
2449            return {
2450              ...prev,
2451              fileHistory: updated
2452            };
2453          });
2454        },
2455        updateAttributionState(updater: (prev: AttributionState) => AttributionState) {
2456          setAppState(prev => {
2457            const updated = updater(prev.attribution);
2458            if (updated === prev.attribution) return prev;
2459            return {
2460              ...prev,
2461              attribution: updated
2462            };
2463          });
2464        },
2465        openMessageSelector: () => {
2466          if (!disabled) {
2467            setIsMessageSelectorVisible(true);
2468          }
2469        },
2470        onChangeAPIKey: reverify,
2471        readFileState: readFileState.current,
2472        setToolJSX,
2473        addNotification,
2474        appendSystemMessage: msg => setMessages(prev => [...prev, msg]),
2475        sendOSNotification: opts => {
2476          void sendNotification(opts, terminal);
2477        },
2478        onChangeDynamicMcpConfig,
2479        onInstallIDEExtension: setIDEToInstallExtension,
2480        nestedMemoryAttachmentTriggers: new Set<string>(),
2481        loadedNestedMemoryPaths: loadedNestedMemoryPathsRef.current,
2482        dynamicSkillDirTriggers: new Set<string>(),
2483        discoveredSkillNames: discoveredSkillNamesRef.current,
2484        setResponseLength,
2485        pushApiMetricsEntry: "external" === 'ant' ? (ttftMs: number) => {
2486          const now = Date.now();
2487          const baseline = responseLengthRef.current;
2488          apiMetricsRef.current.push({
2489            ttftMs,
2490            firstTokenTime: now,
2491            lastTokenTime: now,
2492            responseLengthBaseline: baseline,
2493            endResponseLength: baseline
2494          });
2495        } : undefined,
2496        setStreamMode,
2497        onCompactProgress: event => {
2498          switch (event.type) {
2499            case 'hooks_start':
2500              setSpinnerColor('claudeBlue_FOR_SYSTEM_SPINNER');
2501              setSpinnerShimmerColor('claudeBlueShimmer_FOR_SYSTEM_SPINNER');
2502              setSpinnerMessage(event.hookType === 'pre_compact' ? 'Running PreCompact hooks\u2026' : event.hookType === 'post_compact' ? 'Running PostCompact hooks\u2026' : 'Running SessionStart hooks\u2026');
2503              break;
2504            case 'compact_start':
2505              setSpinnerMessage('Compacting conversation');
2506              break;
2507            case 'compact_end':
2508              setSpinnerMessage(null);
2509              setSpinnerColor(null);
2510              setSpinnerShimmerColor(null);
2511              break;
2512          }
2513        },
2514        setInProgressToolUseIDs,
2515        setHasInterruptibleToolInProgress: (v: boolean) => {
2516          hasInterruptibleToolInProgressRef.current = v;
2517        },
2518        resume,
2519        setConversationId,
2520        requestPrompt: feature('HOOK_PROMPTS') ? requestPrompt : undefined,
2521        contentReplacementState: contentReplacementStateRef.current
2522      };
2523    }, [commands, combinedInitialTools, mainThreadAgentDefinition, debug, initialMcpClients, ideInstallationStatus, dynamicMcpConfig, theme, allowedAgentTypes, store, setAppState, reverify, addNotification, setMessages, onChangeDynamicMcpConfig, resume, requestPrompt, disabled, customSystemPrompt, appendSystemPrompt, setConversationId]);
2524  
2525    // Session backgrounding (Ctrl+B to background/foreground)
2526    const handleBackgroundQuery = useCallback(() => {
2527      // Stop the foreground query so the background one takes over
2528      abortController?.abort('background');
2529      // Aborting subagents may produce task-completed notifications.
2530      // Clear task notifications so the queue processor doesn't immediately
2531      // start a new foreground query; forward them to the background session.
2532      const removedNotifications = removeByFilter(cmd => cmd.mode === 'task-notification');
2533      void (async () => {
2534        const toolUseContext = getToolUseContext(messagesRef.current, [], new AbortController(), mainLoopModel);
2535        const [defaultSystemPrompt, userContext, systemContext] = await Promise.all([getSystemPrompt(toolUseContext.options.tools, mainLoopModel, Array.from(toolPermissionContext.additionalWorkingDirectories.keys()), toolUseContext.options.mcpClients), getUserContext(), getSystemContext()]);
2536        const systemPrompt = buildEffectiveSystemPrompt({
2537          mainThreadAgentDefinition,
2538          toolUseContext,
2539          customSystemPrompt,
2540          defaultSystemPrompt,
2541          appendSystemPrompt
2542        });
2543        toolUseContext.renderedSystemPrompt = systemPrompt;
2544        const notificationAttachments = await getQueuedCommandAttachments(removedNotifications).catch(() => []);
2545        const notificationMessages = notificationAttachments.map(createAttachmentMessage);
2546  
2547        // Deduplicate: if the query loop already yielded a notification into
2548        // messagesRef before we removed it from the queue, skip duplicates.
2549        // We use prompt text for dedup because source_uuid is not set on
2550        // task-notification QueuedCommands (enqueuePendingNotification callers
2551        // don't pass uuid), so it would always be undefined.
2552        const existingPrompts = new Set<string>();
2553        for (const m of messagesRef.current) {
2554          if (m.type === 'attachment' && m.attachment.type === 'queued_command' && m.attachment.commandMode === 'task-notification' && typeof m.attachment.prompt === 'string') {
2555            existingPrompts.add(m.attachment.prompt);
2556          }
2557        }
2558        const uniqueNotifications = notificationMessages.filter(m => m.attachment.type === 'queued_command' && (typeof m.attachment.prompt !== 'string' || !existingPrompts.has(m.attachment.prompt)));
2559        startBackgroundSession({
2560          messages: [...messagesRef.current, ...uniqueNotifications],
2561          queryParams: {
2562            systemPrompt,
2563            userContext,
2564            systemContext,
2565            canUseTool,
2566            toolUseContext,
2567            querySource: getQuerySourceForREPL()
2568          },
2569          description: terminalTitle,
2570          setAppState,
2571          agentDefinition: mainThreadAgentDefinition
2572        });
2573      })();
2574    }, [abortController, mainLoopModel, toolPermissionContext, mainThreadAgentDefinition, getToolUseContext, customSystemPrompt, appendSystemPrompt, canUseTool, setAppState]);
2575    const {
2576      handleBackgroundSession
2577    } = useSessionBackgrounding({
2578      setMessages,
2579      setIsLoading: setIsExternalLoading,
2580      resetLoadingState,
2581      setAbortController,
2582      onBackgroundQuery: handleBackgroundQuery
2583    });
2584    const onQueryEvent = useCallback((event: Parameters<typeof handleMessageFromStream>[0]) => {
2585      handleMessageFromStream(event, newMessage => {
2586        if (isCompactBoundaryMessage(newMessage)) {
2587          // Fullscreen: keep pre-compact messages for scrollback. query.ts
2588          // slices at the boundary for API calls, Messages.tsx skips the
2589          // boundary filter in fullscreen, and useLogMessages treats this
2590          // as an incremental append (first uuid unchanged). Cap at one
2591          // compact-interval of scrollback — normalizeMessages/applyGrouping
2592          // are O(n) per render, so drop everything before the previous
2593          // boundary to keep n bounded across multi-day sessions.
2594          if (isFullscreenEnvEnabled()) {
2595            setMessages(old => [...getMessagesAfterCompactBoundary(old, {
2596              includeSnipped: true
2597            }), newMessage]);
2598          } else {
2599            setMessages(() => [newMessage]);
2600          }
2601          // Bump conversationId so Messages.tsx row keys change and
2602          // stale memoized rows remount with post-compact content.
2603          setConversationId(randomUUID());
2604          // Compaction succeeded — clear the context-blocked flag so ticks resume
2605          if (feature('PROACTIVE') || feature('KAIROS')) {
2606            proactiveModule?.setContextBlocked(false);
2607          }
2608        } else if (newMessage.type === 'progress' && isEphemeralToolProgress(newMessage.data.type)) {
2609          // Replace the previous ephemeral progress tick for the same tool
2610          // call instead of appending. Sleep/Bash emit a tick per second and
2611          // only the last one is rendered; appending blows up the messages
2612          // array (13k+ observed) and the transcript (120MB of sleep_progress
2613          // lines). useLogMessages tracks length, so same-length replacement
2614          // also skips the transcript write.
2615          // agent_progress / hook_progress / skill_progress are NOT ephemeral
2616          // — each carries distinct state the UI needs (e.g. subagent tool
2617          // history). Replacing those leaves the AgentTool UI stuck at
2618          // "Initializing…" because it renders the full progress trail.
2619          setMessages(oldMessages => {
2620            const last = oldMessages.at(-1);
2621            if (last?.type === 'progress' && last.parentToolUseID === newMessage.parentToolUseID && last.data.type === newMessage.data.type) {
2622              const copy = oldMessages.slice();
2623              copy[copy.length - 1] = newMessage;
2624              return copy;
2625            }
2626            return [...oldMessages, newMessage];
2627          });
2628        } else {
2629          setMessages(oldMessages => [...oldMessages, newMessage]);
2630        }
2631        // Block ticks on API errors to prevent tick → error → tick
2632        // runaway loops (e.g., auth failure, rate limit, blocking limit).
2633        // Cleared on compact boundary (above) or successful response (below).
2634        if (feature('PROACTIVE') || feature('KAIROS')) {
2635          if (newMessage.type === 'assistant' && 'isApiErrorMessage' in newMessage && newMessage.isApiErrorMessage) {
2636            proactiveModule?.setContextBlocked(true);
2637          } else if (newMessage.type === 'assistant') {
2638            proactiveModule?.setContextBlocked(false);
2639          }
2640        }
2641      }, newContent => {
2642        // setResponseLength handles updating both responseLengthRef (for
2643        // spinner animation) and apiMetricsRef (endResponseLength/lastTokenTime
2644        // for OTPS). No separate metrics update needed here.
2645        setResponseLength(length => length + newContent.length);
2646      }, setStreamMode, setStreamingToolUses, tombstonedMessage => {
2647        setMessages(oldMessages => oldMessages.filter(m => m !== tombstonedMessage));
2648        void removeTranscriptMessage(tombstonedMessage.uuid);
2649      }, setStreamingThinking, metrics => {
2650        const now = Date.now();
2651        const baseline = responseLengthRef.current;
2652        apiMetricsRef.current.push({
2653          ...metrics,
2654          firstTokenTime: now,
2655          lastTokenTime: now,
2656          responseLengthBaseline: baseline,
2657          endResponseLength: baseline
2658        });
2659      }, onStreamingText);
2660    }, [setMessages, setResponseLength, setStreamMode, setStreamingToolUses, setStreamingThinking, onStreamingText]);
2661    const onQueryImpl = useCallback(async (messagesIncludingNewMessages: MessageType[], newMessages: MessageType[], abortController: AbortController, shouldQuery: boolean, additionalAllowedTools: string[], mainLoopModelParam: string, effort?: EffortValue) => {
2662      // Prepare IDE integration for new prompt. Read mcpClients fresh from
2663      // store — useManageMCPConnections may have populated it since the
2664      // render that captured this closure (same pattern as computeTools).
2665      if (shouldQuery) {
2666        const freshClients = mergeClients(initialMcpClients, store.getState().mcp.clients);
2667        void diagnosticTracker.handleQueryStart(freshClients);
2668        const ideClient = getConnectedIdeClient(freshClients);
2669        if (ideClient) {
2670          void closeOpenDiffs(ideClient);
2671        }
2672      }
2673  
2674      // Mark onboarding as complete when any user message is sent to Claude
2675      void maybeMarkProjectOnboardingComplete();
2676  
2677      // Extract a session title from the first real user message. One-shot
2678      // via ref (was tengu_birch_mist experiment: first-message-only to save
2679      // Haiku calls). The ref replaces the old `messages.length <= 1` check,
2680      // which was broken by SessionStart hook messages (prepended via
2681      // useDeferredHookMessages) and attachment messages (appended by
2682      // processTextPrompt) — both pushed length past 1 on turn one, so the
2683      // title silently fell through to the "Claude Code" default.
2684      if (!titleDisabled && !sessionTitle && !agentTitle && !haikuTitleAttemptedRef.current) {
2685        const firstUserMessage = newMessages.find(m => m.type === 'user' && !m.isMeta);
2686        const text = firstUserMessage?.type === 'user' ? getContentText(firstUserMessage.message.content) : null;
2687        // Skip synthetic breadcrumbs — slash-command output, prompt-skill
2688        // expansions (/commit → <command-message>), local-command headers
2689        // (/help → <command-name>), and bash-mode (!cmd → <bash-input>).
2690        // None of these are the user's topic; wait for real prose.
2691        if (text && !text.startsWith(`<${LOCAL_COMMAND_STDOUT_TAG}>`) && !text.startsWith(`<${COMMAND_MESSAGE_TAG}>`) && !text.startsWith(`<${COMMAND_NAME_TAG}>`) && !text.startsWith(`<${BASH_INPUT_TAG}>`)) {
2692          haikuTitleAttemptedRef.current = true;
2693          void generateSessionTitle(text, new AbortController().signal).then(title => {
2694            if (title) setHaikuTitle(title);else haikuTitleAttemptedRef.current = false;
2695          }, () => {
2696            haikuTitleAttemptedRef.current = false;
2697          });
2698        }
2699      }
2700  
2701      // Apply slash-command-scoped allowedTools (from skill frontmatter) to the
2702      // store once per turn. This also covers the reset: the next non-skill turn
2703      // passes [] and clears it. Must run before the !shouldQuery gate: forked
2704      // commands (executeForkedSlashCommand) return shouldQuery=false, and
2705      // createGetAppStateWithAllowedTools in forkedAgent.ts reads this field, so
2706      // stale skill tools would otherwise leak into forked agent permissions.
2707      // Previously this write was hidden inside getToolUseContext's getAppState
2708      // (~85 calls/turn); hoisting it here makes getAppState a pure read and stops
2709      // ephemeral contexts (permission dialog, BackgroundTasksDialog) from
2710      // accidentally clearing it mid-turn.
2711      store.setState(prev => {
2712        const cur = prev.toolPermissionContext.alwaysAllowRules.command;
2713        if (cur === additionalAllowedTools || cur?.length === additionalAllowedTools.length && cur.every((v, i) => v === additionalAllowedTools[i])) {
2714          return prev;
2715        }
2716        return {
2717          ...prev,
2718          toolPermissionContext: {
2719            ...prev.toolPermissionContext,
2720            alwaysAllowRules: {
2721              ...prev.toolPermissionContext.alwaysAllowRules,
2722              command: additionalAllowedTools
2723            }
2724          }
2725        };
2726      });
2727  
2728      // The last message is an assistant message if the user input was a bash command,
2729      // or if the user input was an invalid slash command.
2730      if (!shouldQuery) {
2731        // Manual /compact sets messages directly (shouldQuery=false) bypassing
2732        // handleMessageFromStream. Clear context-blocked if a compact boundary
2733        // is present so proactive ticks resume after compaction.
2734        if (newMessages.some(isCompactBoundaryMessage)) {
2735          // Bump conversationId so Messages.tsx row keys change and
2736          // stale memoized rows remount with post-compact content.
2737          setConversationId(randomUUID());
2738          if (feature('PROACTIVE') || feature('KAIROS')) {
2739            proactiveModule?.setContextBlocked(false);
2740          }
2741        }
2742        resetLoadingState();
2743        setAbortController(null);
2744        return;
2745      }
2746      const toolUseContext = getToolUseContext(messagesIncludingNewMessages, newMessages, abortController, mainLoopModelParam);
2747      // getToolUseContext reads tools/mcpClients fresh from store.getState()
2748      // (via computeTools/mergeClients). Use those rather than the closure-
2749      // captured `tools`/`mcpClients` — useManageMCPConnections may have
2750      // flushed new MCP state between the render that captured this closure
2751      // and now. Turn 1 via processInitialMessage is the main beneficiary.
2752      const {
2753        tools: freshTools,
2754        mcpClients: freshMcpClients
2755      } = toolUseContext.options;
2756  
2757      // Scope the skill's effort override to this turn's context only —
2758      // wrapping getAppState keeps the override out of the global store so
2759      // background agents and UI subscribers (Spinner, LogoV2) never see it.
2760      if (effort !== undefined) {
2761        const previousGetAppState = toolUseContext.getAppState;
2762        toolUseContext.getAppState = () => ({
2763          ...previousGetAppState(),
2764          effortValue: effort
2765        });
2766      }
2767      queryCheckpoint('query_context_loading_start');
2768      const [,, defaultSystemPrompt, baseUserContext, systemContext] = await Promise.all([
2769      // IMPORTANT: do this after setMessages() above, to avoid UI jank
2770      checkAndDisableBypassPermissionsIfNeeded(toolPermissionContext, setAppState),
2771      // Gated on TRANSCRIPT_CLASSIFIER so GrowthBook kill switch runs wherever auto mode is built in
2772      feature('TRANSCRIPT_CLASSIFIER') ? checkAndDisableAutoModeIfNeeded(toolPermissionContext, setAppState, store.getState().fastMode) : undefined, getSystemPrompt(freshTools, mainLoopModelParam, Array.from(toolPermissionContext.additionalWorkingDirectories.keys()), freshMcpClients), getUserContext(), getSystemContext()]);
2773      const userContext = {
2774        ...baseUserContext,
2775        ...getCoordinatorUserContext(freshMcpClients, isScratchpadEnabled() ? getScratchpadDir() : undefined),
2776        ...((feature('PROACTIVE') || feature('KAIROS')) && proactiveModule?.isProactiveActive() && !terminalFocusRef.current ? {
2777          terminalFocus: 'The terminal is unfocused \u2014 the user is not actively watching.'
2778        } : {})
2779      };
2780      queryCheckpoint('query_context_loading_end');
2781      const systemPrompt = buildEffectiveSystemPrompt({
2782        mainThreadAgentDefinition,
2783        toolUseContext,
2784        customSystemPrompt,
2785        defaultSystemPrompt,
2786        appendSystemPrompt
2787      });
2788      toolUseContext.renderedSystemPrompt = systemPrompt;
2789      queryCheckpoint('query_query_start');
2790      resetTurnHookDuration();
2791      resetTurnToolDuration();
2792      resetTurnClassifierDuration();
2793      for await (const event of query({
2794        messages: messagesIncludingNewMessages,
2795        systemPrompt,
2796        userContext,
2797        systemContext,
2798        canUseTool,
2799        toolUseContext,
2800        querySource: getQuerySourceForREPL()
2801      })) {
2802        onQueryEvent(event);
2803      }
2804      if (feature('BUDDY')) {
2805        void fireCompanionObserver(messagesRef.current, reaction => setAppState(prev => prev.companionReaction === reaction ? prev : {
2806          ...prev,
2807          companionReaction: reaction
2808        }));
2809      }
2810      queryCheckpoint('query_end');
2811  
2812      // Capture ant-only API metrics before resetLoadingState clears the ref.
2813      // For multi-request turns (tool use loops), compute P50 across all requests.
2814      if ("external" === 'ant' && apiMetricsRef.current.length > 0) {
2815        const entries = apiMetricsRef.current;
2816        const ttfts = entries.map(e => e.ttftMs);
2817        // Compute per-request OTPS using only active streaming time and
2818        // streaming-only content. endResponseLength tracks content added by
2819        // streaming deltas only, excluding subagent/compaction inflation.
2820        const otpsValues = entries.map(e => {
2821          const delta = Math.round((e.endResponseLength - e.responseLengthBaseline) / 4);
2822          const samplingMs = e.lastTokenTime - e.firstTokenTime;
2823          return samplingMs > 0 ? Math.round(delta / (samplingMs / 1000)) : 0;
2824        });
2825        const isMultiRequest = entries.length > 1;
2826        const hookMs = getTurnHookDurationMs();
2827        const hookCount = getTurnHookCount();
2828        const toolMs = getTurnToolDurationMs();
2829        const toolCount = getTurnToolCount();
2830        const classifierMs = getTurnClassifierDurationMs();
2831        const classifierCount = getTurnClassifierCount();
2832        const turnMs = Date.now() - loadingStartTimeRef.current;
2833        setMessages(prev => [...prev, createApiMetricsMessage({
2834          ttftMs: isMultiRequest ? median(ttfts) : ttfts[0]!,
2835          otps: isMultiRequest ? median(otpsValues) : otpsValues[0]!,
2836          isP50: isMultiRequest,
2837          hookDurationMs: hookMs > 0 ? hookMs : undefined,
2838          hookCount: hookCount > 0 ? hookCount : undefined,
2839          turnDurationMs: turnMs > 0 ? turnMs : undefined,
2840          toolDurationMs: toolMs > 0 ? toolMs : undefined,
2841          toolCount: toolCount > 0 ? toolCount : undefined,
2842          classifierDurationMs: classifierMs > 0 ? classifierMs : undefined,
2843          classifierCount: classifierCount > 0 ? classifierCount : undefined,
2844          configWriteCount: getGlobalConfigWriteCount()
2845        })]);
2846      }
2847      resetLoadingState();
2848  
2849      // Log query profiling report if enabled
2850      logQueryProfileReport();
2851  
2852      // Signal that a query turn has completed successfully
2853      await onTurnComplete?.(messagesRef.current);
2854    }, [initialMcpClients, resetLoadingState, getToolUseContext, toolPermissionContext, setAppState, customSystemPrompt, onTurnComplete, appendSystemPrompt, canUseTool, mainThreadAgentDefinition, onQueryEvent, sessionTitle, titleDisabled]);
2855    const onQuery = useCallback(async (newMessages: MessageType[], abortController: AbortController, shouldQuery: boolean, additionalAllowedTools: string[], mainLoopModelParam: string, onBeforeQueryCallback?: (input: string, newMessages: MessageType[]) => Promise<boolean>, input?: string, effort?: EffortValue): Promise<void> => {
2856      // If this is a teammate, mark them as active when starting a turn
2857      if (isAgentSwarmsEnabled()) {
2858        const teamName = getTeamName();
2859        const agentName = getAgentName();
2860        if (teamName && agentName) {
2861          // Fire and forget - turn starts immediately, write happens in background
2862          void setMemberActive(teamName, agentName, true);
2863        }
2864      }
2865  
2866      // Concurrent guard via state machine. tryStart() atomically checks
2867      // and transitions idle→running, returning the generation number.
2868      // Returns null if already running — no separate check-then-set.
2869      const thisGeneration = queryGuard.tryStart();
2870      if (thisGeneration === null) {
2871        logEvent('tengu_concurrent_onquery_detected', {});
2872  
2873        // Extract and enqueue user message text, skipping meta messages
2874        // (e.g. expanded skill content, tick prompts) that should not be
2875        // replayed as user-visible text.
2876        newMessages.filter((m): m is UserMessage => m.type === 'user' && !m.isMeta).map(_ => getContentText(_.message.content)).filter(_ => _ !== null).forEach((msg, i) => {
2877          enqueue({
2878            value: msg,
2879            mode: 'prompt'
2880          });
2881          if (i === 0) {
2882            logEvent('tengu_concurrent_onquery_enqueued', {});
2883          }
2884        });
2885        return;
2886      }
2887      try {
2888        // isLoading is derived from queryGuard — tryStart() above already
2889        // transitioned dispatching→running, so no setter call needed here.
2890        resetTimingRefs();
2891        setMessages(oldMessages => [...oldMessages, ...newMessages]);
2892        responseLengthRef.current = 0;
2893        if (feature('TOKEN_BUDGET')) {
2894          const parsedBudget = input ? parseTokenBudget(input) : null;
2895          snapshotOutputTokensForTurn(parsedBudget ?? getCurrentTurnTokenBudget());
2896        }
2897        apiMetricsRef.current = [];
2898        setStreamingToolUses([]);
2899        setStreamingText(null);
2900  
2901        // messagesRef is updated synchronously by the setMessages wrapper
2902        // above, so it already includes newMessages from the append at the
2903        // top of this try block.  No reconstruction needed, no waiting for
2904        // React's scheduler (previously cost 20-56ms per prompt; the 56ms
2905        // case was a GC pause caught during the await).
2906        const latestMessages = messagesRef.current;
2907        if (input) {
2908          await mrOnBeforeQuery(input, latestMessages, newMessages.length);
2909        }
2910  
2911        // Pass full conversation history to callback
2912        if (onBeforeQueryCallback && input) {
2913          const shouldProceed = await onBeforeQueryCallback(input, latestMessages);
2914          if (!shouldProceed) {
2915            return;
2916          }
2917        }
2918        await onQueryImpl(latestMessages, newMessages, abortController, shouldQuery, additionalAllowedTools, mainLoopModelParam, effort);
2919      } finally {
2920        // queryGuard.end() atomically checks generation and transitions
2921        // running→idle. Returns false if a newer query owns the guard
2922        // (cancel+resubmit race where the stale finally fires as a microtask).
2923        if (queryGuard.end(thisGeneration)) {
2924          setLastQueryCompletionTime(Date.now());
2925          skipIdleCheckRef.current = false;
2926          // Always reset loading state in finally - this ensures cleanup even
2927          // if onQueryImpl throws. onTurnComplete is called separately in
2928          // onQueryImpl only on successful completion.
2929          resetLoadingState();
2930          await mrOnTurnComplete(messagesRef.current, abortController.signal.aborted);
2931  
2932          // Notify bridge clients that the turn is complete so mobile apps
2933          // can stop the spark animation and show post-turn UI.
2934          sendBridgeResultRef.current();
2935  
2936          // Auto-hide tungsten panel content at turn end (ant-only), but keep
2937          // tungstenActiveSession set so the pill stays in the footer and the user
2938          // can reopen the panel. Background tmux tasks (e.g. /hunter) run for
2939          // minutes — wiping the session made the pill disappear entirely, forcing
2940          // the user to re-invoke Tmux just to peek. Skip on abort so the panel
2941          // stays open for inspection (matches the turn-duration guard below).
2942          if ("external" === 'ant' && !abortController.signal.aborted) {
2943            setAppState(prev => {
2944              if (prev.tungstenActiveSession === undefined) return prev;
2945              if (prev.tungstenPanelAutoHidden === true) return prev;
2946              return {
2947                ...prev,
2948                tungstenPanelAutoHidden: true
2949              };
2950            });
2951          }
2952  
2953          // Capture budget info before clearing (ant-only)
2954          let budgetInfo: {
2955            tokens: number;
2956            limit: number;
2957            nudges: number;
2958          } | undefined;
2959          if (feature('TOKEN_BUDGET')) {
2960            if (getCurrentTurnTokenBudget() !== null && getCurrentTurnTokenBudget()! > 0 && !abortController.signal.aborted) {
2961              budgetInfo = {
2962                tokens: getTurnOutputTokens(),
2963                limit: getCurrentTurnTokenBudget()!,
2964                nudges: getBudgetContinuationCount()
2965              };
2966            }
2967            snapshotOutputTokensForTurn(null);
2968          }
2969  
2970          // Add turn duration message for turns longer than 30s or with a budget
2971          // Skip if user aborted or if in loop mode (too noisy between ticks)
2972          // Defer if swarm teammates are still running (show when they finish)
2973          const turnDurationMs = Date.now() - loadingStartTimeRef.current - totalPausedMsRef.current;
2974          if ((turnDurationMs > 30000 || budgetInfo !== undefined) && !abortController.signal.aborted && !proactiveActive) {
2975            const hasRunningSwarmAgents = getAllInProcessTeammateTasks(store.getState().tasks).some(t => t.status === 'running');
2976            if (hasRunningSwarmAgents) {
2977              // Only record start time on the first deferred turn
2978              if (swarmStartTimeRef.current === null) {
2979                swarmStartTimeRef.current = loadingStartTimeRef.current;
2980              }
2981              // Always update budget — later turns may carry the actual budget
2982              if (budgetInfo) {
2983                swarmBudgetInfoRef.current = budgetInfo;
2984              }
2985            } else {
2986              setMessages(prev => [...prev, createTurnDurationMessage(turnDurationMs, budgetInfo, count(prev, isLoggableMessage))]);
2987            }
2988          }
2989          // Clear the controller so CancelRequestHandler's canCancelRunningTask
2990          // reads false at the idle prompt. Without this, the stale non-aborted
2991          // controller makes ctrl+c fire onCancel() (aborting nothing) instead of
2992          // propagating to the double-press exit flow.
2993          setAbortController(null);
2994        }
2995  
2996        // Auto-restore: if the user interrupted before any meaningful response
2997        // arrived, rewind the conversation and restore their prompt — same as
2998        // opening the message selector and picking the last message.
2999        // This runs OUTSIDE the queryGuard.end() check because onCancel calls
3000        // forceEnd(), which bumps the generation so end() returns false above.
3001        // Guards: reason === 'user-cancel' (onCancel/Esc; programmatic aborts
3002        // use 'background'/'interrupt' and must not rewind — note abort() with
3003        // no args sets reason to a DOMException, not undefined), !isActive (no
3004        // newer query started — cancel+resubmit race), empty input (don't
3005        // clobber text typed during loading), no queued commands (user queued
3006        // B while A was loading → they've moved on, don't restore A; also
3007        // avoids removeLastFromHistory removing B's entry instead of A's),
3008        // not viewing a teammate (messagesRef is the main conversation — the
3009        // old Up-arrow quick-restore had this guard, preserve it).
3010        if (abortController.signal.reason === 'user-cancel' && !queryGuard.isActive && inputValueRef.current === '' && getCommandQueueLength() === 0 && !store.getState().viewingAgentTaskId) {
3011          const msgs = messagesRef.current;
3012          const lastUserMsg = msgs.findLast(selectableUserMessagesFilter);
3013          if (lastUserMsg) {
3014            const idx = msgs.lastIndexOf(lastUserMsg);
3015            if (messagesAfterAreOnlySynthetic(msgs, idx)) {
3016              // The submit is being undone — undo its history entry too,
3017              // otherwise Up-arrow shows the restored text twice.
3018              removeLastFromHistory();
3019              restoreMessageSyncRef.current(lastUserMsg);
3020            }
3021          }
3022        }
3023      }
3024    }, [onQueryImpl, setAppState, resetLoadingState, queryGuard, mrOnBeforeQuery, mrOnTurnComplete]);
3025  
3026    // Handle initial message (from CLI args or plan mode exit with context clear)
3027    // This effect runs when isLoading becomes false and there's a pending message
3028    const initialMessageRef = useRef(false);
3029    useEffect(() => {
3030      const pending = initialMessage;
3031      if (!pending || isLoading || initialMessageRef.current) return;
3032  
3033      // Mark as processing to prevent re-entry
3034      initialMessageRef.current = true;
3035      async function processInitialMessage(initialMsg: NonNullable<typeof pending>) {
3036        // Clear context if requested (plan mode exit)
3037        if (initialMsg.clearContext) {
3038          // Preserve the plan slug before clearing context, so the new session
3039          // can access the same plan file after regenerateSessionId()
3040          const oldPlanSlug = initialMsg.message.planContent ? getPlanSlug() : undefined;
3041          const {
3042            clearConversation
3043          } = await import('../commands/clear/conversation.js');
3044          await clearConversation({
3045            setMessages,
3046            readFileState: readFileState.current,
3047            discoveredSkillNames: discoveredSkillNamesRef.current,
3048            loadedNestedMemoryPaths: loadedNestedMemoryPathsRef.current,
3049            getAppState: () => store.getState(),
3050            setAppState,
3051            setConversationId
3052          });
3053          haikuTitleAttemptedRef.current = false;
3054          setHaikuTitle(undefined);
3055          bashTools.current.clear();
3056          bashToolsProcessedIdx.current = 0;
3057  
3058          // Restore the plan slug for the new session so getPlan() finds the file
3059          if (oldPlanSlug) {
3060            setPlanSlug(getSessionId(), oldPlanSlug);
3061          }
3062        }
3063  
3064        // Atomically: clear initial message, set permission mode and rules, and store plan for verification
3065        const shouldStorePlanForVerification = initialMsg.message.planContent && "external" === 'ant' && isEnvTruthy(undefined);
3066        setAppState(prev => {
3067          // Build and apply permission updates (mode + allowedPrompts rules)
3068          let updatedToolPermissionContext = initialMsg.mode ? applyPermissionUpdates(prev.toolPermissionContext, buildPermissionUpdates(initialMsg.mode, initialMsg.allowedPrompts)) : prev.toolPermissionContext;
3069          // For auto, override the mode (buildPermissionUpdates maps
3070          // it to 'default' via toExternalPermissionMode) and strip dangerous rules
3071          if (feature('TRANSCRIPT_CLASSIFIER') && initialMsg.mode === 'auto') {
3072            updatedToolPermissionContext = stripDangerousPermissionsForAutoMode({
3073              ...updatedToolPermissionContext,
3074              mode: 'auto',
3075              prePlanMode: undefined
3076            });
3077          }
3078          return {
3079            ...prev,
3080            initialMessage: null,
3081            toolPermissionContext: updatedToolPermissionContext,
3082            ...(shouldStorePlanForVerification && {
3083              pendingPlanVerification: {
3084                plan: initialMsg.message.planContent!,
3085                verificationStarted: false,
3086                verificationCompleted: false
3087              }
3088            })
3089          };
3090        });
3091  
3092        // Create file history snapshot for code rewind
3093        if (fileHistoryEnabled()) {
3094          void fileHistoryMakeSnapshot((updater: (prev: FileHistoryState) => FileHistoryState) => {
3095            setAppState(prev => ({
3096              ...prev,
3097              fileHistory: updater(prev.fileHistory)
3098            }));
3099          }, initialMsg.message.uuid);
3100        }
3101  
3102        // Ensure SessionStart hook context is available before the first API
3103        // call. onSubmit calls this internally but the onQuery path below
3104        // bypasses onSubmit — hoist here so both paths see hook messages.
3105        await awaitPendingHooks();
3106  
3107        // Route all initial prompts through onSubmit to ensure UserPromptSubmit hooks fire
3108        // TODO: Simplify by always routing through onSubmit once it supports
3109        // ContentBlockParam arrays (images) as input
3110        const content = initialMsg.message.message.content;
3111  
3112        // Route all string content through onSubmit to ensure hooks fire
3113        // For complex content (images, etc.), fall back to direct onQuery
3114        // Plan messages bypass onSubmit to preserve planContent metadata for rendering
3115        if (typeof content === 'string' && !initialMsg.message.planContent) {
3116          // Route through onSubmit for proper processing including UserPromptSubmit hooks
3117          void onSubmit(content, {
3118            setCursorOffset: () => {},
3119            clearBuffer: () => {},
3120            resetHistory: () => {}
3121          });
3122        } else {
3123          // Plan messages or complex content (images, etc.) - send directly to model
3124          // Plan messages use onQuery to preserve planContent metadata for rendering
3125          // TODO: Once onSubmit supports ContentBlockParam arrays, remove this branch
3126          const newAbortController = createAbortController();
3127          setAbortController(newAbortController);
3128          void onQuery([initialMsg.message], newAbortController, true,
3129          // shouldQuery
3130          [],
3131          // additionalAllowedTools
3132          mainLoopModel);
3133        }
3134  
3135        // Reset ref after a delay to allow new initial messages
3136        setTimeout(ref => {
3137          ref.current = false;
3138        }, 100, initialMessageRef);
3139      }
3140      void processInitialMessage(pending);
3141    }, [initialMessage, isLoading, setMessages, setAppState, onQuery, mainLoopModel, tools]);
3142    const onSubmit = useCallback(async (input: string, helpers: PromptInputHelpers, speculationAccept?: {
3143      state: ActiveSpeculationState;
3144      speculationSessionTimeSavedMs: number;
3145      setAppState: SetAppState;
3146    }, options?: {
3147      fromKeybinding?: boolean;
3148    }) => {
3149      // Re-pin scroll to bottom on submit so the user always sees the new
3150      // exchange (matches OpenCode's auto-scroll behavior).
3151      repinScroll();
3152  
3153      // Resume loop mode if paused
3154      if (feature('PROACTIVE') || feature('KAIROS')) {
3155        proactiveModule?.resumeProactive();
3156      }
3157  
3158      // Handle immediate commands - these bypass the queue and execute right away
3159      // even while Claude is processing. Commands opt-in via `immediate: true`.
3160      // Commands triggered via keybindings are always treated as immediate.
3161      if (!speculationAccept && input.trim().startsWith('/')) {
3162        // Expand [Pasted text #N] refs so immediate commands (e.g. /btw) receive
3163        // the pasted content, not the placeholder. The non-immediate path gets
3164        // this expansion later in handlePromptSubmit.
3165        const trimmedInput = expandPastedTextRefs(input, pastedContents).trim();
3166        const spaceIndex = trimmedInput.indexOf(' ');
3167        const commandName = spaceIndex === -1 ? trimmedInput.slice(1) : trimmedInput.slice(1, spaceIndex);
3168        const commandArgs = spaceIndex === -1 ? '' : trimmedInput.slice(spaceIndex + 1).trim();
3169  
3170        // Find matching command - treat as immediate if:
3171        // 1. Command has `immediate: true`, OR
3172        // 2. Command was triggered via keybinding (fromKeybinding option)
3173        const matchingCommand = commands.find(cmd => isCommandEnabled(cmd) && (cmd.name === commandName || cmd.aliases?.includes(commandName) || getCommandName(cmd) === commandName));
3174        if (matchingCommand?.name === 'clear' && idleHintShownRef.current) {
3175          logEvent('tengu_idle_return_action', {
3176            action: 'hint_converted' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
3177            variant: idleHintShownRef.current as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
3178            idleMinutes: Math.round((Date.now() - lastQueryCompletionTimeRef.current) / 60_000),
3179            messageCount: messagesRef.current.length,
3180            totalInputTokens: getTotalInputTokens()
3181          });
3182          idleHintShownRef.current = false;
3183        }
3184        const shouldTreatAsImmediate = queryGuard.isActive && (matchingCommand?.immediate || options?.fromKeybinding);
3185        if (matchingCommand && shouldTreatAsImmediate && matchingCommand.type === 'local-jsx') {
3186          // Only clear input if the submitted text matches what's in the prompt.
3187          // When a command keybinding fires, input is "/<command>" but the actual
3188          // input value is the user's existing text - don't clear it in that case.
3189          if (input.trim() === inputValueRef.current.trim()) {
3190            setInputValue('');
3191            helpers.setCursorOffset(0);
3192            helpers.clearBuffer();
3193            setPastedContents({});
3194          }
3195          const pastedTextRefs = parseReferences(input).filter(r => pastedContents[r.id]?.type === 'text');
3196          const pastedTextCount = pastedTextRefs.length;
3197          const pastedTextBytes = pastedTextRefs.reduce((sum, r) => sum + (pastedContents[r.id]?.content.length ?? 0), 0);
3198          logEvent('tengu_paste_text', {
3199            pastedTextCount,
3200            pastedTextBytes
3201          });
3202          logEvent('tengu_immediate_command_executed', {
3203            commandName: matchingCommand.name as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
3204            fromKeybinding: options?.fromKeybinding ?? false
3205          });
3206  
3207          // Execute the command directly
3208          const executeImmediateCommand = async (): Promise<void> => {
3209            let doneWasCalled = false;
3210            const onDone = (result?: string, doneOptions?: {
3211              display?: CommandResultDisplay;
3212              metaMessages?: string[];
3213            }): void => {
3214              doneWasCalled = true;
3215              setToolJSX({
3216                jsx: null,
3217                shouldHidePromptInput: false,
3218                clearLocalJSX: true
3219              });
3220              const newMessages: MessageType[] = [];
3221              if (result && doneOptions?.display !== 'skip') {
3222                addNotification({
3223                  key: `immediate-${matchingCommand.name}`,
3224                  text: result,
3225                  priority: 'immediate'
3226                });
3227                // In fullscreen the command just showed as a centered modal
3228                // pane — the notification above is enough feedback. Adding
3229                // "❯ /config" + "⎿ dismissed" to the transcript is clutter
3230                // (those messages are type:system subtype:local_command —
3231                // user-visible but NOT sent to the model, so skipping them
3232                // doesn't change model context). Outside fullscreen the
3233                // transcript entry stays so scrollback shows what ran.
3234                if (!isFullscreenEnvEnabled()) {
3235                  newMessages.push(createCommandInputMessage(formatCommandInputTags(getCommandName(matchingCommand), commandArgs)), createCommandInputMessage(`<${LOCAL_COMMAND_STDOUT_TAG}>${escapeXml(result)}</${LOCAL_COMMAND_STDOUT_TAG}>`));
3236                }
3237              }
3238              // Inject meta messages (model-visible, user-hidden) into the transcript
3239              if (doneOptions?.metaMessages?.length) {
3240                newMessages.push(...doneOptions.metaMessages.map(content => createUserMessage({
3241                  content,
3242                  isMeta: true
3243                })));
3244              }
3245              if (newMessages.length) {
3246                setMessages(prev => [...prev, ...newMessages]);
3247              }
3248              // Restore stashed prompt after local-jsx command completes.
3249              // The normal stash restoration path (below) is skipped because
3250              // local-jsx commands return early from onSubmit.
3251              if (stashedPrompt !== undefined) {
3252                setInputValue(stashedPrompt.text);
3253                helpers.setCursorOffset(stashedPrompt.cursorOffset);
3254                setPastedContents(stashedPrompt.pastedContents);
3255                setStashedPrompt(undefined);
3256              }
3257            };
3258  
3259            // Build context for the command (reuses existing getToolUseContext).
3260            // Read messages via ref to keep onSubmit stable across message
3261            // updates — matches the pattern at L2384/L2400/L2662 and avoids
3262            // pinning stale REPL render scopes in downstream closures.
3263            const context = getToolUseContext(messagesRef.current, [], createAbortController(), mainLoopModel);
3264            const mod = await matchingCommand.load();
3265            const jsx = await mod.call(onDone, context, commandArgs);
3266  
3267            // Skip if onDone already fired — prevents stuck isLocalJSXCommand
3268            // (see processSlashCommand.tsx local-jsx case for full mechanism).
3269            if (jsx && !doneWasCalled) {
3270              // shouldHidePromptInput: false keeps Notifications mounted
3271              // so the onDone result isn't lost
3272              setToolJSX({
3273                jsx,
3274                shouldHidePromptInput: false,
3275                isLocalJSXCommand: true
3276              });
3277            }
3278          };
3279          void executeImmediateCommand();
3280          return; // Always return early - don't add to history or queue
3281        }
3282      }
3283  
3284      // Remote mode: skip empty input early before any state mutations
3285      if (activeRemote.isRemoteMode && !input.trim()) {
3286        return;
3287      }
3288  
3289      // Idle-return: prompt returning users to start fresh when the
3290      // conversation is large and the cache is cold. tengu_willow_mode
3291      // controls treatment: "dialog" (blocking), "hint" (notification), "off".
3292      {
3293        const willowMode = getFeatureValue_CACHED_MAY_BE_STALE('tengu_willow_mode', 'off');
3294        const idleThresholdMin = Number(process.env.CLAUDE_CODE_IDLE_THRESHOLD_MINUTES ?? 75);
3295        const tokenThreshold = Number(process.env.CLAUDE_CODE_IDLE_TOKEN_THRESHOLD ?? 100_000);
3296        if (willowMode !== 'off' && !getGlobalConfig().idleReturnDismissed && !skipIdleCheckRef.current && !speculationAccept && !input.trim().startsWith('/') && lastQueryCompletionTimeRef.current > 0 && getTotalInputTokens() >= tokenThreshold) {
3297          const idleMs = Date.now() - lastQueryCompletionTimeRef.current;
3298          const idleMinutes = idleMs / 60_000;
3299          if (idleMinutes >= idleThresholdMin && willowMode === 'dialog') {
3300            setIdleReturnPending({
3301              input,
3302              idleMinutes
3303            });
3304            setInputValue('');
3305            helpers.setCursorOffset(0);
3306            helpers.clearBuffer();
3307            return;
3308          }
3309        }
3310      }
3311  
3312      // Add to history for direct user submissions.
3313      // Queued command processing (executeQueuedInput) doesn't call onSubmit,
3314      // so notifications and already-queued user input won't be added to history here.
3315      // Skip history for keybinding-triggered commands (user didn't type the command).
3316      if (!options?.fromKeybinding) {
3317        addToHistory({
3318          display: speculationAccept ? input : prependModeCharacterToInput(input, inputMode),
3319          pastedContents: speculationAccept ? {} : pastedContents
3320        });
3321        // Add the just-submitted command to the front of the ghost-text
3322        // cache so it's suggested immediately (not after the 60s TTL).
3323        if (inputMode === 'bash') {
3324          prependToShellHistoryCache(input.trim());
3325        }
3326      }
3327  
3328      // Restore stash if present, but NOT for slash commands or when loading.
3329      // - Slash commands (especially interactive ones like /model, /context) hide
3330      //   the prompt and show a picker UI. Restoring the stash during a command would
3331      //   place the text in a hidden input, and the user would lose it by typing the
3332      //   next command. Instead, preserve the stash so it survives across command runs.
3333      // - When loading, the submitted input will be queued and handlePromptSubmit
3334      //   will clear the input field (onInputChange('')), which would clobber the
3335      //   restored stash. Defer restoration to after handlePromptSubmit (below).
3336      //   Remote mode is exempt: it sends via WebSocket and returns early without
3337      //   calling handlePromptSubmit, so there's no clobbering risk — restore eagerly.
3338      // In both deferred cases, the stash is restored after await handlePromptSubmit.
3339      const isSlashCommand = !speculationAccept && input.trim().startsWith('/');
3340      // Submit runs "now" (not queued) when not already loading, or when
3341      // accepting speculation, or in remote mode (which sends via WS and
3342      // returns early without calling handlePromptSubmit).
3343      const submitsNow = !isLoading || speculationAccept || activeRemote.isRemoteMode;
3344      if (stashedPrompt !== undefined && !isSlashCommand && submitsNow) {
3345        setInputValue(stashedPrompt.text);
3346        helpers.setCursorOffset(stashedPrompt.cursorOffset);
3347        setPastedContents(stashedPrompt.pastedContents);
3348        setStashedPrompt(undefined);
3349      } else if (submitsNow) {
3350        if (!options?.fromKeybinding) {
3351          // Clear input when not loading or accepting speculation.
3352          // Preserve input for keybinding-triggered commands.
3353          setInputValue('');
3354          helpers.setCursorOffset(0);
3355        }
3356        setPastedContents({});
3357      }
3358      if (submitsNow) {
3359        setInputMode('prompt');
3360        setIDESelection(undefined);
3361        setSubmitCount(_ => _ + 1);
3362        helpers.clearBuffer();
3363        tipPickedThisTurnRef.current = false;
3364  
3365        // Show the placeholder in the same React batch as setInputValue('').
3366        // Skip for slash/bash (they have their own echo), speculation and remote
3367        // mode (both setMessages directly with no gap to bridge).
3368        if (!isSlashCommand && inputMode === 'prompt' && !speculationAccept && !activeRemote.isRemoteMode) {
3369          setUserInputOnProcessing(input);
3370          // showSpinner includes userInputOnProcessing, so the spinner appears
3371          // on this render. Reset timing refs now (before queryGuard.reserve()
3372          // would) so elapsed time doesn't read as Date.now() - 0. The
3373          // isQueryActive transition above does the same reset — idempotent.
3374          resetTimingRefs();
3375        }
3376  
3377        // Increment prompt count for attribution tracking and save snapshot
3378        // The snapshot persists promptCount so it survives compaction
3379        if (feature('COMMIT_ATTRIBUTION')) {
3380          setAppState(prev => ({
3381            ...prev,
3382            attribution: incrementPromptCount(prev.attribution, snapshot => {
3383              void recordAttributionSnapshot(snapshot).catch(error => {
3384                logForDebugging(`Attribution: Failed to save snapshot: ${error}`);
3385              });
3386            })
3387          }));
3388        }
3389      }
3390  
3391      // Handle speculation acceptance
3392      if (speculationAccept) {
3393        const {
3394          queryRequired
3395        } = await handleSpeculationAccept(speculationAccept.state, speculationAccept.speculationSessionTimeSavedMs, speculationAccept.setAppState, input, {
3396          setMessages,
3397          readFileState,
3398          cwd: getOriginalCwd()
3399        });
3400        if (queryRequired) {
3401          const newAbortController = createAbortController();
3402          setAbortController(newAbortController);
3403          void onQuery([], newAbortController, true, [], mainLoopModel);
3404        }
3405        return;
3406      }
3407  
3408      // Remote mode: send input via stream-json instead of local query.
3409      // Permission requests from the remote are bridged into toolUseConfirmQueue
3410      // and rendered using the standard PermissionRequest component.
3411      //
3412      // local-jsx slash commands (e.g. /agents, /config) render UI in THIS
3413      // process — they have no remote equivalent. Let those fall through to
3414      // handlePromptSubmit so they execute locally. Prompt commands and
3415      // plain text go to the remote.
3416      if (activeRemote.isRemoteMode && !(isSlashCommand && commands.find(c => {
3417        const name = input.trim().slice(1).split(/\s/)[0];
3418        return isCommandEnabled(c) && (c.name === name || c.aliases?.includes(name!) || getCommandName(c) === name);
3419      })?.type === 'local-jsx')) {
3420        // Build content blocks when there are pasted attachments (images)
3421        const pastedValues = Object.values(pastedContents);
3422        const imageContents = pastedValues.filter(c => c.type === 'image');
3423        const imagePasteIds = imageContents.length > 0 ? imageContents.map(c => c.id) : undefined;
3424        let messageContent: string | ContentBlockParam[] = input.trim();
3425        let remoteContent: RemoteMessageContent = input.trim();
3426        if (pastedValues.length > 0) {
3427          const contentBlocks: ContentBlockParam[] = [];
3428          const remoteBlocks: Array<{
3429            type: string;
3430            [key: string]: unknown;
3431          }> = [];
3432          const trimmedInput = input.trim();
3433          if (trimmedInput) {
3434            contentBlocks.push({
3435              type: 'text',
3436              text: trimmedInput
3437            });
3438            remoteBlocks.push({
3439              type: 'text',
3440              text: trimmedInput
3441            });
3442          }
3443          for (const pasted of pastedValues) {
3444            if (pasted.type === 'image') {
3445              const source = {
3446                type: 'base64' as const,
3447                media_type: (pasted.mediaType ?? 'image/png') as 'image/jpeg' | 'image/png' | 'image/gif' | 'image/webp',
3448                data: pasted.content
3449              };
3450              contentBlocks.push({
3451                type: 'image',
3452                source
3453              });
3454              remoteBlocks.push({
3455                type: 'image',
3456                source
3457              });
3458            } else {
3459              contentBlocks.push({
3460                type: 'text',
3461                text: pasted.content
3462              });
3463              remoteBlocks.push({
3464                type: 'text',
3465                text: pasted.content
3466              });
3467            }
3468          }
3469          messageContent = contentBlocks;
3470          remoteContent = remoteBlocks;
3471        }
3472  
3473        // Create and add user message to UI
3474        // Note: empty input already handled by early return above
3475        const userMessage = createUserMessage({
3476          content: messageContent,
3477          imagePasteIds
3478        });
3479        setMessages(prev => [...prev, userMessage]);
3480  
3481        // Send to remote session
3482        await activeRemote.sendMessage(remoteContent, {
3483          uuid: userMessage.uuid
3484        });
3485        return;
3486      }
3487  
3488      // Ensure SessionStart hook context is available before the first API call.
3489      await awaitPendingHooks();
3490      await handlePromptSubmit({
3491        input,
3492        helpers,
3493        queryGuard,
3494        isExternalLoading,
3495        mode: inputMode,
3496        commands,
3497        onInputChange: setInputValue,
3498        setPastedContents,
3499        setToolJSX,
3500        getToolUseContext,
3501        messages: messagesRef.current,
3502        mainLoopModel,
3503        pastedContents,
3504        ideSelection,
3505        setUserInputOnProcessing,
3506        setAbortController,
3507        abortController,
3508        onQuery,
3509        setAppState,
3510        querySource: getQuerySourceForREPL(),
3511        onBeforeQuery,
3512        canUseTool,
3513        addNotification,
3514        setMessages,
3515        // Read via ref so streamMode can be dropped from onSubmit deps —
3516        // handlePromptSubmit only uses it for debug log + telemetry event.
3517        streamMode: streamModeRef.current,
3518        hasInterruptibleToolInProgress: hasInterruptibleToolInProgressRef.current
3519      });
3520  
3521      // Restore stash that was deferred above. Two cases:
3522      // - Slash command: handlePromptSubmit awaited the full command execution
3523      //   (including interactive pickers). Restoring now places the stash back in
3524      //   the visible input.
3525      // - Loading (queued): handlePromptSubmit enqueued + cleared input, then
3526      //   returned quickly. Restoring now places the stash back after the clear.
3527      if ((isSlashCommand || isLoading) && stashedPrompt !== undefined) {
3528        setInputValue(stashedPrompt.text);
3529        helpers.setCursorOffset(stashedPrompt.cursorOffset);
3530        setPastedContents(stashedPrompt.pastedContents);
3531        setStashedPrompt(undefined);
3532      }
3533    }, [queryGuard,
3534    // isLoading is read at the !isLoading checks above for input-clearing
3535    // and submitCount gating. It's derived from isQueryActive || isExternalLoading,
3536    // so including it here ensures the closure captures the fresh value.
3537    isLoading, isExternalLoading, inputMode, commands, setInputValue, setInputMode, setPastedContents, setSubmitCount, setIDESelection, setToolJSX, getToolUseContext,
3538    // messages is read via messagesRef.current inside the callback to
3539    // keep onSubmit stable across message updates (see L2384/L2400/L2662).
3540    // Without this, each setMessages call (~30× per turn) recreates
3541    // onSubmit, pinning the REPL render scope (1776B) + that render's
3542    // messages array in downstream closures (PromptInput, handleAutoRunIssue).
3543    // Heap analysis showed ~9 REPL scopes and ~15 messages array versions
3544    // accumulating after #20174/#20175, all traced to this dep.
3545    mainLoopModel, pastedContents, ideSelection, setUserInputOnProcessing, setAbortController, addNotification, onQuery, stashedPrompt, setStashedPrompt, setAppState, onBeforeQuery, canUseTool, remoteSession, setMessages, awaitPendingHooks, repinScroll]);
3546  
3547    // Callback for when user submits input while viewing a teammate's transcript
3548    const onAgentSubmit = useCallback(async (input: string, task: InProcessTeammateTaskState | LocalAgentTaskState, helpers: PromptInputHelpers) => {
3549      if (isLocalAgentTask(task)) {
3550        appendMessageToLocalAgent(task.id, createUserMessage({
3551          content: input
3552        }), setAppState);
3553        if (task.status === 'running') {
3554          queuePendingMessage(task.id, input, setAppState);
3555        } else {
3556          void resumeAgentBackground({
3557            agentId: task.id,
3558            prompt: input,
3559            toolUseContext: getToolUseContext(messagesRef.current, [], new AbortController(), mainLoopModel),
3560            canUseTool
3561          }).catch(err => {
3562            logForDebugging(`resumeAgentBackground failed: ${errorMessage(err)}`);
3563            addNotification({
3564              key: `resume-agent-failed-${task.id}`,
3565              jsx: <Text color="error">
3566                    Failed to resume agent: {errorMessage(err)}
3567                  </Text>,
3568              priority: 'low'
3569            });
3570          });
3571        }
3572      } else {
3573        injectUserMessageToTeammate(task.id, input, setAppState);
3574      }
3575      setInputValue('');
3576      helpers.setCursorOffset(0);
3577      helpers.clearBuffer();
3578    }, [setAppState, setInputValue, getToolUseContext, canUseTool, mainLoopModel, addNotification]);
3579  
3580    // Handlers for auto-run /issue or /good-claude (defined after onSubmit)
3581    const handleAutoRunIssue = useCallback(() => {
3582      const command = autoRunIssueReason ? getAutoRunCommand(autoRunIssueReason) : '/issue';
3583      setAutoRunIssueReason(null); // Clear the state
3584      onSubmit(command, {
3585        setCursorOffset: () => {},
3586        clearBuffer: () => {},
3587        resetHistory: () => {}
3588      }).catch(err => {
3589        logForDebugging(`Auto-run ${command} failed: ${errorMessage(err)}`);
3590      });
3591    }, [onSubmit, autoRunIssueReason]);
3592    const handleCancelAutoRunIssue = useCallback(() => {
3593      setAutoRunIssueReason(null);
3594    }, []);
3595  
3596    // Handler for when user presses 1 on survey thanks screen to share details
3597    const handleSurveyRequestFeedback = useCallback(() => {
3598      const command = "external" === 'ant' ? '/issue' : '/feedback';
3599      onSubmit(command, {
3600        setCursorOffset: () => {},
3601        clearBuffer: () => {},
3602        resetHistory: () => {}
3603      }).catch(err => {
3604        logForDebugging(`Survey feedback request failed: ${err instanceof Error ? err.message : String(err)}`);
3605      });
3606    }, [onSubmit]);
3607  
3608    // onSubmit is unstable (deps include `messages` which changes every turn).
3609    // `handleOpenRateLimitOptions` is prop-drilled to every MessageRow, and each
3610    // MessageRow fiber pins the closure (and transitively the entire REPL render
3611    // scope, ~1.8KB) at mount time. Using a ref keeps this callback stable so
3612    // old REPL scopes can be GC'd — saves ~35MB over a 1000-turn session.
3613    const onSubmitRef = useRef(onSubmit);
3614    onSubmitRef.current = onSubmit;
3615    const handleOpenRateLimitOptions = useCallback(() => {
3616      void onSubmitRef.current('/rate-limit-options', {
3617        setCursorOffset: () => {},
3618        clearBuffer: () => {},
3619        resetHistory: () => {}
3620      });
3621    }, []);
3622    const handleExit = useCallback(async () => {
3623      setIsExiting(true);
3624      // In bg sessions, always detach instead of kill — even when a worktree is
3625      // active. Without this guard, the worktree branch below short-circuits into
3626      // ExitFlow (which calls gracefulShutdown) before exit.tsx is ever loaded.
3627      if (feature('BG_SESSIONS') && isBgSession()) {
3628        spawnSync('tmux', ['detach-client'], {
3629          stdio: 'ignore'
3630        });
3631        setIsExiting(false);
3632        return;
3633      }
3634      const showWorktree = getCurrentWorktreeSession() !== null;
3635      if (showWorktree) {
3636        setExitFlow(<ExitFlow showWorktree onDone={() => {}} onCancel={() => {
3637          setExitFlow(null);
3638          setIsExiting(false);
3639        }} />);
3640        return;
3641      }
3642      const exitMod = await exit.load();
3643      const exitFlowResult = await exitMod.call(() => {});
3644      setExitFlow(exitFlowResult);
3645      // If call() returned without killing the process (bg session detach),
3646      // clear isExiting so the UI is usable on reattach. No-op on the normal
3647      // path — gracefulShutdown's process.exit() means we never get here.
3648      if (exitFlowResult === null) {
3649        setIsExiting(false);
3650      }
3651    }, []);
3652    const handleShowMessageSelector = useCallback(() => {
3653      setIsMessageSelectorVisible(prev => !prev);
3654    }, []);
3655  
3656    // Rewind conversation state to just before `message`: slice messages,
3657    // reset conversation ID, microcompact state, permission mode, prompt suggestion.
3658    // Does NOT touch the prompt input. Index is computed from messagesRef (always
3659    // fresh via the setMessages wrapper) so callers don't need to worry about
3660    // stale closures.
3661    const rewindConversationTo = useCallback((message: UserMessage) => {
3662      const prev = messagesRef.current;
3663      const messageIndex = prev.lastIndexOf(message);
3664      if (messageIndex === -1) return;
3665      logEvent('tengu_conversation_rewind', {
3666        preRewindMessageCount: prev.length,
3667        postRewindMessageCount: messageIndex,
3668        messagesRemoved: prev.length - messageIndex,
3669        rewindToMessageIndex: messageIndex
3670      });
3671      setMessages(prev.slice(0, messageIndex));
3672      // Careful, this has to happen after setMessages
3673      setConversationId(randomUUID());
3674      // Reset cached microcompact state so stale pinned cache edits
3675      // don't reference tool_use_ids from truncated messages
3676      resetMicrocompactState();
3677      if (feature('CONTEXT_COLLAPSE')) {
3678        // Rewind truncates the REPL array. Commits whose archived span
3679        // was past the rewind point can't be projected anymore
3680        // (projectView silently skips them) but the staged queue and ID
3681        // maps reference stale uuids. Simplest safe reset: drop
3682        // everything. The ctx-agent will re-stage on the next
3683        // threshold crossing.
3684        /* eslint-disable @typescript-eslint/no-require-imports */
3685        ;
3686        (require('../services/contextCollapse/index.js') as typeof import('../services/contextCollapse/index.js')).resetContextCollapse();
3687        /* eslint-enable @typescript-eslint/no-require-imports */
3688      }
3689  
3690      // Restore state from the message we're rewinding to
3691      setAppState(prev => ({
3692        ...prev,
3693        // Restore permission mode from the message
3694        toolPermissionContext: message.permissionMode && prev.toolPermissionContext.mode !== message.permissionMode ? {
3695          ...prev.toolPermissionContext,
3696          mode: message.permissionMode
3697        } : prev.toolPermissionContext,
3698        // Clear stale prompt suggestion from previous conversation state
3699        promptSuggestion: {
3700          text: null,
3701          promptId: null,
3702          shownAt: 0,
3703          acceptedAt: 0,
3704          generationRequestId: null
3705        }
3706      }));
3707    }, [setMessages, setAppState]);
3708  
3709    // Synchronous rewind + input population. Used directly by auto-restore on
3710    // interrupt (so React batches with the abort's setMessages → single render,
3711    // no flicker). MessageSelector wraps this in setImmediate via handleRestoreMessage.
3712    const restoreMessageSync = useCallback((message: UserMessage) => {
3713      rewindConversationTo(message);
3714      const r = textForResubmit(message);
3715      if (r) {
3716        setInputValue(r.text);
3717        setInputMode(r.mode);
3718      }
3719  
3720      // Restore pasted images
3721      if (Array.isArray(message.message.content) && message.message.content.some(block => block.type === 'image')) {
3722        const imageBlocks: Array<ImageBlockParam> = message.message.content.filter(block => block.type === 'image');
3723        if (imageBlocks.length > 0) {
3724          const newPastedContents: Record<number, PastedContent> = {};
3725          imageBlocks.forEach((block, index) => {
3726            if (block.source.type === 'base64') {
3727              const id = message.imagePasteIds?.[index] ?? index + 1;
3728              newPastedContents[id] = {
3729                id,
3730                type: 'image',
3731                content: block.source.data,
3732                mediaType: block.source.media_type
3733              };
3734            }
3735          });
3736          setPastedContents(newPastedContents);
3737        }
3738      }
3739    }, [rewindConversationTo, setInputValue]);
3740    restoreMessageSyncRef.current = restoreMessageSync;
3741  
3742    // MessageSelector path: defer via setImmediate so the "Interrupted" message
3743    // renders to static output before rewind — otherwise it remains vestigial
3744    // at the top of the screen.
3745    const handleRestoreMessage = useCallback(async (message: UserMessage) => {
3746      setImmediate((restore, message) => restore(message), restoreMessageSync, message);
3747    }, [restoreMessageSync]);
3748  
3749    // Not memoized — hook stores caps via ref, reads latest closure at dispatch.
3750    // 24-char prefix: deriveUUID preserves first 24, renderable uuid prefix-matches raw source.
3751    const findRawIndex = (uuid: string) => {
3752      const prefix = uuid.slice(0, 24);
3753      return messages.findIndex(m => m.uuid.slice(0, 24) === prefix);
3754    };
3755    const messageActionCaps: MessageActionCaps = {
3756      copy: text =>
3757      // setClipboard RETURNS OSC 52 — caller must stdout.write (tmux side-effects load-buffer, but that's tmux-only).
3758      void setClipboard(text).then(raw => {
3759        if (raw) process.stdout.write(raw);
3760        addNotification({
3761          // Same key as text-selection copy — repeated copies replace toast, don't queue.
3762          key: 'selection-copied',
3763          text: 'copied',
3764          color: 'success',
3765          priority: 'immediate',
3766          timeoutMs: 2000
3767        });
3768      }),
3769      edit: async msg => {
3770        // Same skip-confirm check as /rewind: lossless → direct, else confirm dialog.
3771        const rawIdx = findRawIndex(msg.uuid);
3772        const raw = rawIdx >= 0 ? messages[rawIdx] : undefined;
3773        if (!raw || !selectableUserMessagesFilter(raw)) return;
3774        const noFileChanges = !(await fileHistoryHasAnyChanges(fileHistory, raw.uuid));
3775        const onlySynthetic = messagesAfterAreOnlySynthetic(messages, rawIdx);
3776        if (noFileChanges && onlySynthetic) {
3777          // rewindConversationTo's setMessages races stream appends — cancel first (idempotent).
3778          onCancel();
3779          // handleRestoreMessage also restores pasted images.
3780          void handleRestoreMessage(raw);
3781        } else {
3782          // Dialog path: onPreRestore (= onCancel) fires when user CONFIRMS, not on nevermind.
3783          setMessageSelectorPreselect(raw);
3784          setIsMessageSelectorVisible(true);
3785        }
3786      }
3787    };
3788    const {
3789      enter: enterMessageActions,
3790      handlers: messageActionHandlers
3791    } = useMessageActions(cursor, setCursor, cursorNavRef, messageActionCaps);
3792    async function onInit() {
3793      // Always verify API key on startup, so we can show the user an error in the
3794      // bottom right corner of the screen if the API key is invalid.
3795      void reverify();
3796  
3797      // Populate readFileState with CLAUDE.md files at startup
3798      const memoryFiles = await getMemoryFiles();
3799      if (memoryFiles.length > 0) {
3800        const fileList = memoryFiles.map(f => `  [${f.type}] ${f.path} (${f.content.length} chars)${f.parent ? ` (included by ${f.parent})` : ''}`).join('\n');
3801        logForDebugging(`Loaded ${memoryFiles.length} CLAUDE.md/rules files:\n${fileList}`);
3802      } else {
3803        logForDebugging('No CLAUDE.md/rules files found');
3804      }
3805      for (const file of memoryFiles) {
3806        // When the injected content doesn't match disk (stripped HTML comments,
3807        // stripped frontmatter, MEMORY.md truncation), cache the RAW disk bytes
3808        // with isPartialView so Edit/Write require a real Read first while
3809        // getChangedFiles + nested_memory dedup still work.
3810        readFileState.current.set(file.path, {
3811          content: file.contentDiffersFromDisk ? file.rawContent ?? file.content : file.content,
3812          timestamp: Date.now(),
3813          offset: undefined,
3814          limit: undefined,
3815          isPartialView: file.contentDiffersFromDisk
3816        });
3817      }
3818  
3819      // Initial message handling is done via the initialMessage effect
3820    }
3821  
3822    // Register cost summary tracker
3823    useCostSummary(useFpsMetrics());
3824  
3825    // Record transcripts locally, for debugging and conversation recovery
3826    // Don't record conversation if we only have initial messages; optimizes
3827    // the case where user resumes a conversation then quites before doing
3828    // anything else
3829    useLogMessages(messages, messages.length === initialMessages?.length);
3830  
3831    // REPL Bridge: replicate user/assistant messages to the bridge session
3832    // for remote access via claude.ai. No-op in external builds or when not enabled.
3833    const {
3834      sendBridgeResult
3835    } = useReplBridge(messages, setMessages, abortControllerRef, commands, mainLoopModel);
3836    sendBridgeResultRef.current = sendBridgeResult;
3837    useAfterFirstRender();
3838  
3839    // Track prompt queue usage for analytics. Fire once per transition from
3840    // empty to non-empty, not on every length change -- otherwise a render loop
3841    // (concurrent onQuery thrashing, etc.) spams saveGlobalConfig, which hits
3842    // ELOCKED under concurrent sessions and falls back to unlocked writes.
3843    // That write storm is the primary trigger for ~/.claude.json corruption
3844    // (GH #3117).
3845    const hasCountedQueueUseRef = useRef(false);
3846    useEffect(() => {
3847      if (queuedCommands.length < 1) {
3848        hasCountedQueueUseRef.current = false;
3849        return;
3850      }
3851      if (hasCountedQueueUseRef.current) return;
3852      hasCountedQueueUseRef.current = true;
3853      saveGlobalConfig(current => ({
3854        ...current,
3855        promptQueueUseCount: (current.promptQueueUseCount ?? 0) + 1
3856      }));
3857    }, [queuedCommands.length]);
3858  
3859    // Process queued commands when query completes and queue has items
3860  
3861    const executeQueuedInput = useCallback(async (queuedCommands: QueuedCommand[]) => {
3862      await handlePromptSubmit({
3863        helpers: {
3864          setCursorOffset: () => {},
3865          clearBuffer: () => {},
3866          resetHistory: () => {}
3867        },
3868        queryGuard,
3869        commands,
3870        onInputChange: () => {},
3871        setPastedContents: () => {},
3872        setToolJSX,
3873        getToolUseContext,
3874        messages,
3875        mainLoopModel,
3876        ideSelection,
3877        setUserInputOnProcessing,
3878        setAbortController,
3879        onQuery,
3880        setAppState,
3881        querySource: getQuerySourceForREPL(),
3882        onBeforeQuery,
3883        canUseTool,
3884        addNotification,
3885        setMessages,
3886        queuedCommands
3887      });
3888    }, [queryGuard, commands, setToolJSX, getToolUseContext, messages, mainLoopModel, ideSelection, setUserInputOnProcessing, canUseTool, setAbortController, onQuery, addNotification, setAppState, onBeforeQuery]);
3889    useQueueProcessor({
3890      executeQueuedInput,
3891      hasActiveLocalJsxUI: isShowingLocalJSXCommand,
3892      queryGuard
3893    });
3894  
3895    // We'll use the global lastInteractionTime from state.ts
3896  
3897    // Update last interaction time when input changes.
3898    // Must be immediate because useEffect runs after the Ink render cycle flush.
3899    useEffect(() => {
3900      activityManager.recordUserActivity();
3901      updateLastInteractionTime(true);
3902    }, [inputValue, submitCount]);
3903    useEffect(() => {
3904      if (submitCount === 1) {
3905        startBackgroundHousekeeping();
3906      }
3907    }, [submitCount]);
3908  
3909    // Show notification when Claude is done responding and user is idle
3910    useEffect(() => {
3911      // Don't set up notification if Claude is busy
3912      if (isLoading) return;
3913  
3914      // Only enable notifications after the first new interaction in this session
3915      if (submitCount === 0) return;
3916  
3917      // No query has completed yet
3918      if (lastQueryCompletionTime === 0) return;
3919  
3920      // Set timeout to check idle state
3921      const timer = setTimeout((lastQueryCompletionTime, isLoading, toolJSX, focusedInputDialogRef, terminal) => {
3922        // Check if user has interacted since the response ended
3923        const lastUserInteraction = getLastInteractionTime();
3924        if (lastUserInteraction > lastQueryCompletionTime) {
3925          // User has interacted since Claude finished - they're not idle, don't notify
3926          return;
3927        }
3928  
3929        // User hasn't interacted since response ended, check other conditions
3930        const idleTimeSinceResponse = Date.now() - lastQueryCompletionTime;
3931        if (!isLoading && !toolJSX &&
3932        // Use ref to get current dialog state, avoiding stale closure
3933        focusedInputDialogRef.current === undefined && idleTimeSinceResponse >= getGlobalConfig().messageIdleNotifThresholdMs) {
3934          void sendNotification({
3935            message: 'Claude is waiting for your input',
3936            notificationType: 'idle_prompt'
3937          }, terminal);
3938        }
3939      }, getGlobalConfig().messageIdleNotifThresholdMs, lastQueryCompletionTime, isLoading, toolJSX, focusedInputDialogRef, terminal);
3940      return () => clearTimeout(timer);
3941    }, [isLoading, toolJSX, submitCount, lastQueryCompletionTime, terminal]);
3942  
3943    // Idle-return hint: show notification when idle threshold is exceeded.
3944    // Timer fires after the configured idle period; notification persists until
3945    // dismissed or the user submits.
3946    useEffect(() => {
3947      if (lastQueryCompletionTime === 0) return;
3948      if (isLoading) return;
3949      const willowMode: string = getFeatureValue_CACHED_MAY_BE_STALE('tengu_willow_mode', 'off');
3950      if (willowMode !== 'hint' && willowMode !== 'hint_v2') return;
3951      if (getGlobalConfig().idleReturnDismissed) return;
3952      const tokenThreshold = Number(process.env.CLAUDE_CODE_IDLE_TOKEN_THRESHOLD ?? 100_000);
3953      if (getTotalInputTokens() < tokenThreshold) return;
3954      const idleThresholdMs = Number(process.env.CLAUDE_CODE_IDLE_THRESHOLD_MINUTES ?? 75) * 60_000;
3955      const elapsed = Date.now() - lastQueryCompletionTime;
3956      const remaining = idleThresholdMs - elapsed;
3957      const timer = setTimeout((lqct, addNotif, msgsRef, mode, hintRef) => {
3958        if (msgsRef.current.length === 0) return;
3959        const totalTokens = getTotalInputTokens();
3960        const formattedTokens = formatTokens(totalTokens);
3961        const idleMinutes = (Date.now() - lqct) / 60_000;
3962        addNotif({
3963          key: 'idle-return-hint',
3964          jsx: mode === 'hint_v2' ? <>
3965                  <Text dimColor>new task? </Text>
3966                  <Text color="suggestion">/clear</Text>
3967                  <Text dimColor> to save </Text>
3968                  <Text color="suggestion">{formattedTokens} tokens</Text>
3969                </> : <Text color="warning">
3970                  new task? /clear to save {formattedTokens} tokens
3971                </Text>,
3972          priority: 'medium',
3973          // Persist until submit — the hint fires at T+75min idle, user may
3974          // not return for hours. removeNotification in useEffect cleanup
3975          // handles dismissal. 0x7FFFFFFF = setTimeout max (~24.8 days).
3976          timeoutMs: 0x7fffffff
3977        });
3978        hintRef.current = mode;
3979        logEvent('tengu_idle_return_action', {
3980          action: 'hint_shown' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
3981          variant: mode as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
3982          idleMinutes: Math.round(idleMinutes),
3983          messageCount: msgsRef.current.length,
3984          totalInputTokens: totalTokens
3985        });
3986      }, Math.max(0, remaining), lastQueryCompletionTime, addNotification, messagesRef, willowMode, idleHintShownRef);
3987      return () => {
3988        clearTimeout(timer);
3989        removeNotification('idle-return-hint');
3990        idleHintShownRef.current = false;
3991      };
3992    }, [lastQueryCompletionTime, isLoading, addNotification, removeNotification]);
3993  
3994    // Submits incoming prompts from teammate messages or tasks mode as new turns
3995    // Returns true if submission succeeded, false if a query is already running
3996    const handleIncomingPrompt = useCallback((content: string, options?: {
3997      isMeta?: boolean;
3998    }): boolean => {
3999      if (queryGuard.isActive) return false;
4000  
4001      // Defer to user-queued commands — user input always takes priority
4002      // over system messages (teammate messages, task list items, etc.)
4003      // Read from the module-level store at call time (not the render-time
4004      // snapshot) to avoid a stale closure — this callback's deps don't
4005      // include the queue.
4006      if (getCommandQueue().some(cmd => cmd.mode === 'prompt' || cmd.mode === 'bash')) {
4007        return false;
4008      }
4009      const newAbortController = createAbortController();
4010      setAbortController(newAbortController);
4011  
4012      // Create a user message with the formatted content (includes XML wrapper)
4013      const userMessage = createUserMessage({
4014        content,
4015        isMeta: options?.isMeta ? true : undefined
4016      });
4017      void onQuery([userMessage], newAbortController, true, [], mainLoopModel);
4018      return true;
4019    }, [onQuery, mainLoopModel, store]);
4020  
4021    // Voice input integration (VOICE_MODE builds only)
4022    const voice = feature('VOICE_MODE') ?
4023    // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
4024    useVoiceIntegration({
4025      setInputValueRaw,
4026      inputValueRef,
4027      insertTextRef
4028    }) : {
4029      stripTrailing: () => 0,
4030      handleKeyEvent: () => {},
4031      resetAnchor: () => {},
4032      interimRange: null
4033    };
4034    useInboxPoller({
4035      enabled: isAgentSwarmsEnabled(),
4036      isLoading,
4037      focusedInputDialog,
4038      onSubmitMessage: handleIncomingPrompt
4039    });
4040    useMailboxBridge({
4041      isLoading,
4042      onSubmitMessage: handleIncomingPrompt
4043    });
4044  
4045    // Scheduled tasks from .claude/scheduled_tasks.json (CronCreate/Delete/List)
4046    if (feature('AGENT_TRIGGERS')) {
4047      // Assistant mode bypasses the isLoading gate (the proactive tick →
4048      // Sleep → tick loop would otherwise starve the scheduler).
4049      // kairosEnabled is set once in initialState (main.tsx) and never mutated — no
4050      // subscription needed. The tengu_kairos_cron runtime gate is checked inside
4051      // useScheduledTasks's effect (not here) since wrapping a hook call in a dynamic
4052      // condition would break rules-of-hooks.
4053      const assistantMode = store.getState().kairosEnabled;
4054      // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
4055      useScheduledTasks!({
4056        isLoading,
4057        assistantMode,
4058        setMessages
4059      });
4060    }
4061  
4062    // Note: Permission polling is now handled by useInboxPoller
4063    // - Workers receive permission responses via mailbox messages
4064    // - Leaders receive permission requests via mailbox messages
4065  
4066    if ("external" === 'ant') {
4067      // Tasks mode: watch for tasks and auto-process them
4068      // eslint-disable-next-line react-hooks/rules-of-hooks
4069      // biome-ignore lint/correctness/useHookAtTopLevel: conditional for dead code elimination in external builds
4070      useTaskListWatcher({
4071        taskListId,
4072        isLoading,
4073        onSubmitTask: handleIncomingPrompt
4074      });
4075  
4076      // Loop mode: auto-tick when enabled (via /job command)
4077      // eslint-disable-next-line react-hooks/rules-of-hooks
4078      // biome-ignore lint/correctness/useHookAtTopLevel: conditional for dead code elimination in external builds
4079      useProactive?.({
4080        // Suppress ticks while an initial message is pending — the initial
4081        // message will be processed asynchronously and a premature tick would
4082        // race with it, causing concurrent-query enqueue of expanded skill text.
4083        isLoading: isLoading || initialMessage !== null,
4084        queuedCommandsLength: queuedCommands.length,
4085        hasActiveLocalJsxUI: isShowingLocalJSXCommand,
4086        isInPlanMode: toolPermissionContext.mode === 'plan',
4087        onSubmitTick: (prompt: string) => handleIncomingPrompt(prompt, {
4088          isMeta: true
4089        }),
4090        onQueueTick: (prompt: string) => enqueue({
4091          mode: 'prompt',
4092          value: prompt,
4093          isMeta: true
4094        })
4095      });
4096    }
4097  
4098    // Abort the current operation when a 'now' priority message arrives
4099    // (e.g. from a chat UI client via UDS).
4100    useEffect(() => {
4101      if (queuedCommands.some(cmd => cmd.priority === 'now')) {
4102        abortControllerRef.current?.abort('interrupt');
4103      }
4104    }, [queuedCommands]);
4105  
4106    // Initial load
4107    useEffect(() => {
4108      void onInit();
4109  
4110      // Cleanup on unmount
4111      return () => {
4112        void diagnosticTracker.shutdown();
4113      };
4114      // TODO: fix this
4115      // eslint-disable-next-line react-hooks/exhaustive-deps
4116    }, []);
4117  
4118    // Listen for suspend/resume events
4119    const {
4120      internal_eventEmitter
4121    } = useStdin();
4122    const [remountKey, setRemountKey] = useState(0);
4123    useEffect(() => {
4124      const handleSuspend = () => {
4125        // Print suspension instructions
4126        process.stdout.write(`\nClaude Code has been suspended. Run \`fg\` to bring Claude Code back.\nNote: ctrl + z now suspends Claude Code, ctrl + _ undoes input.\n`);
4127      };
4128      const handleResume = () => {
4129        // Force complete component tree replacement instead of terminal clear
4130        // Ink now handles line count reset internally on SIGCONT
4131        setRemountKey(prev => prev + 1);
4132      };
4133      internal_eventEmitter?.on('suspend', handleSuspend);
4134      internal_eventEmitter?.on('resume', handleResume);
4135      return () => {
4136        internal_eventEmitter?.off('suspend', handleSuspend);
4137        internal_eventEmitter?.off('resume', handleResume);
4138      };
4139    }, [internal_eventEmitter]);
4140  
4141    // Derive stop hook spinner suffix from messages state
4142    const stopHookSpinnerSuffix = useMemo(() => {
4143      if (!isLoading) return null;
4144  
4145      // Find stop hook progress messages
4146      const progressMsgs = messages.filter((m): m is ProgressMessage<HookProgress> => m.type === 'progress' && m.data.type === 'hook_progress' && (m.data.hookEvent === 'Stop' || m.data.hookEvent === 'SubagentStop'));
4147      if (progressMsgs.length === 0) return null;
4148  
4149      // Get the most recent stop hook execution
4150      const currentToolUseID = progressMsgs.at(-1)?.toolUseID;
4151      if (!currentToolUseID) return null;
4152  
4153      // Check if there's already a summary message for this execution (hooks completed)
4154      const hasSummaryForCurrentExecution = messages.some(m => m.type === 'system' && m.subtype === 'stop_hook_summary' && m.toolUseID === currentToolUseID);
4155      if (hasSummaryForCurrentExecution) return null;
4156      const currentHooks = progressMsgs.filter(p => p.toolUseID === currentToolUseID);
4157      const total = currentHooks.length;
4158  
4159      // Count completed hooks
4160      const completedCount = count(messages, m => {
4161        if (m.type !== 'attachment') return false;
4162        const attachment = m.attachment;
4163        return 'hookEvent' in attachment && (attachment.hookEvent === 'Stop' || attachment.hookEvent === 'SubagentStop') && 'toolUseID' in attachment && attachment.toolUseID === currentToolUseID;
4164      });
4165  
4166      // Check if any hook has a custom status message
4167      const customMessage = currentHooks.find(p => p.data.statusMessage)?.data.statusMessage;
4168      if (customMessage) {
4169        // Use custom message with progress counter if multiple hooks
4170        return total === 1 ? `${customMessage}…` : `${customMessage}… ${completedCount}/${total}`;
4171      }
4172  
4173      // Fall back to default behavior
4174      const hookType = currentHooks[0]?.data.hookEvent === 'SubagentStop' ? 'subagent stop' : 'stop';
4175      if ("external" === 'ant') {
4176        const cmd = currentHooks[completedCount]?.data.command;
4177        const label = cmd ? ` '${truncateToWidth(cmd, 40)}'` : '';
4178        return total === 1 ? `running ${hookType} hook${label}` : `running ${hookType} hook${label}\u2026 ${completedCount}/${total}`;
4179      }
4180      return total === 1 ? `running ${hookType} hook` : `running stop hooks… ${completedCount}/${total}`;
4181    }, [messages, isLoading]);
4182  
4183    // Callback to capture frozen state when entering transcript mode
4184    const handleEnterTranscript = useCallback(() => {
4185      setFrozenTranscriptState({
4186        messagesLength: messages.length,
4187        streamingToolUsesLength: streamingToolUses.length
4188      });
4189    }, [messages.length, streamingToolUses.length]);
4190  
4191    // Callback to clear frozen state when exiting transcript mode
4192    const handleExitTranscript = useCallback(() => {
4193      setFrozenTranscriptState(null);
4194    }, []);
4195  
4196    // Props for GlobalKeybindingHandlers component (rendered inside KeybindingSetup)
4197    const virtualScrollActive = isFullscreenEnvEnabled() && !disableVirtualScroll;
4198  
4199    // Transcript search state. Hooks must be unconditional so they live here
4200    // (not inside the `if (screen === 'transcript')` branch below); isActive
4201    // gates the useInput. Query persists across bar open/close so n/N keep
4202    // working after Enter dismisses the bar (less semantics).
4203    const jumpRef = useRef<JumpHandle | null>(null);
4204    const [searchOpen, setSearchOpen] = useState(false);
4205    const [searchQuery, setSearchQuery] = useState('');
4206    const [searchCount, setSearchCount] = useState(0);
4207    const [searchCurrent, setSearchCurrent] = useState(0);
4208    const onSearchMatchesChange = useCallback((count: number, current: number) => {
4209      setSearchCount(count);
4210      setSearchCurrent(current);
4211    }, []);
4212    useInput((input, key, event) => {
4213      if (key.ctrl || key.meta) return;
4214      // No Esc handling here — less has no navigating mode. Search state
4215      // (highlights, n/N) is just state. Esc/q/ctrl+c → transcript:exit
4216      // (ungated). Highlights clear on exit via the screen-change effect.
4217      if (input === '/') {
4218        // Capture scrollTop NOW — typing is a preview, 0-matches snaps
4219        // back here. Synchronous ref write, fires before the bar's
4220        // mount-effect calls setSearchQuery.
4221        jumpRef.current?.setAnchor();
4222        setSearchOpen(true);
4223        event.stopImmediatePropagation();
4224        return;
4225      }
4226      // Held-key batching: tokenizer coalesces to 'nnn'. Same uniform-batch
4227      // pattern as modalPagerAction in ScrollKeybindingHandler.tsx. Each
4228      // repeat is a step (n isn't idempotent like g).
4229      const c = input[0];
4230      if ((c === 'n' || c === 'N') && input === c.repeat(input.length) && searchCount > 0) {
4231        const fn = c === 'n' ? jumpRef.current?.nextMatch : jumpRef.current?.prevMatch;
4232        if (fn) for (let i = 0; i < input.length; i++) fn();
4233        event.stopImmediatePropagation();
4234      }
4235    },
4236    // Search needs virtual scroll (jumpRef drives VirtualMessageList). [
4237    // kills it, so !dumpMode — after [ there's nothing to jump in.
4238    {
4239      isActive: screen === 'transcript' && virtualScrollActive && !searchOpen && !dumpMode
4240    });
4241    const {
4242      setQuery: setHighlight,
4243      scanElement,
4244      setPositions
4245    } = useSearchHighlight();
4246  
4247    // Resize → abort search. Positions are (msg, query, WIDTH)-keyed —
4248    // cached positions are stale after a width change (new layout, new
4249    // wrapping). Clearing searchQuery triggers VML's setSearchQuery('')
4250    // which clears positionsCache + setPositions(null). Bar closes.
4251    // User hits / again → fresh everything.
4252    const transcriptCols = useTerminalSize().columns;
4253    const prevColsRef = React.useRef(transcriptCols);
4254    React.useEffect(() => {
4255      if (prevColsRef.current !== transcriptCols) {
4256        prevColsRef.current = transcriptCols;
4257        if (searchQuery || searchOpen) {
4258          setSearchOpen(false);
4259          setSearchQuery('');
4260          setSearchCount(0);
4261          setSearchCurrent(0);
4262          jumpRef.current?.disarmSearch();
4263          setHighlight('');
4264        }
4265      }
4266    }, [transcriptCols, searchQuery, searchOpen, setHighlight]);
4267  
4268    // Transcript escape hatches. Bare letters in modal context (no prompt
4269    // competing for input) — same class as g/G/j/k in ScrollKeybindingHandler.
4270    useInput((input, key, event) => {
4271      if (key.ctrl || key.meta) return;
4272      if (input === 'q') {
4273        // less: q quits the pager. ctrl+o toggles; q is the lineage exit.
4274        handleExitTranscript();
4275        event.stopImmediatePropagation();
4276        return;
4277      }
4278      if (input === '[' && !dumpMode) {
4279        // Force dump-to-scrollback. Also expand + uncap — no point dumping
4280        // a subset. Terminal/tmux cmd-F can now find anything. Guard here
4281        // (not in isActive) so v still works post-[ — dump-mode footer at
4282        // ~4898 wires editorStatus, confirming v is meant to stay live.
4283        setDumpMode(true);
4284        setShowAllInTranscript(true);
4285        event.stopImmediatePropagation();
4286      } else if (input === 'v') {
4287        // less-style: v opens the file in $VISUAL/$EDITOR. Render the full
4288        // transcript (same path /export uses), write to tmp, hand off.
4289        // openFileInExternalEditor handles alt-screen suspend/resume for
4290        // terminal editors; GUI editors spawn detached.
4291        event.stopImmediatePropagation();
4292        // Drop double-taps: the render is async and a second press before it
4293        // completes would run a second parallel render (double memory, two
4294        // tempfiles, two editor spawns). editorGenRef only guards
4295        // transcript-exit staleness, not same-session concurrency.
4296        if (editorRenderingRef.current) return;
4297        editorRenderingRef.current = true;
4298        // Capture generation + make a staleness-aware setter. Each write
4299        // checks gen (transcript exit bumps it → late writes from the
4300        // async render go silent).
4301        const gen = editorGenRef.current;
4302        const setStatus = (s: string): void => {
4303          if (gen !== editorGenRef.current) return;
4304          clearTimeout(editorTimerRef.current);
4305          setEditorStatus(s);
4306        };
4307        setStatus(`rendering ${deferredMessages.length} messages…`);
4308        void (async () => {
4309          try {
4310            // Width = terminal minus vim's line-number gutter (4 digits +
4311            // space + slack). Floor at 80. PassThrough has no .columns so
4312            // without this Ink defaults to 80. Trailing-space strip: right-
4313            // aligned timestamps still leave a flexbox spacer run at EOL.
4314            // eslint-disable-next-line custom-rules/prefer-use-terminal-size -- one-shot at keypress time, not a reactive render dep
4315            const w = Math.max(80, (process.stdout.columns ?? 80) - 6);
4316            const raw = await renderMessagesToPlainText(deferredMessages, tools, w);
4317            const text = raw.replace(/[ \t]+$/gm, '');
4318            const path = join(tmpdir(), `cc-transcript-${Date.now()}.txt`);
4319            await writeFile(path, text);
4320            const opened = openFileInExternalEditor(path);
4321            setStatus(opened ? `opening ${path}` : `wrote ${path} · no $VISUAL/$EDITOR set`);
4322          } catch (e) {
4323            setStatus(`render failed: ${e instanceof Error ? e.message : String(e)}`);
4324          }
4325          editorRenderingRef.current = false;
4326          if (gen !== editorGenRef.current) return;
4327          editorTimerRef.current = setTimeout(s => s(''), 4000, setEditorStatus);
4328        })();
4329      }
4330    },
4331    // !searchOpen: typing 'v' or '[' in the search bar is search input, not
4332    // a command. No !dumpMode here — v should work after [ (the [ handler
4333    // guards itself inline).
4334    {
4335      isActive: screen === 'transcript' && virtualScrollActive && !searchOpen
4336    });
4337  
4338    // Fresh `less` per transcript entry. Prevents stale highlights matching
4339    // unrelated normal-mode text (overlay is alt-screen-global) and avoids
4340    // surprise n/N on re-entry. Same exit resets [ dump mode — each ctrl+o
4341    // entry is a fresh instance.
4342    const inTranscript = screen === 'transcript' && virtualScrollActive;
4343    useEffect(() => {
4344      if (!inTranscript) {
4345        setSearchQuery('');
4346        setSearchCount(0);
4347        setSearchCurrent(0);
4348        setSearchOpen(false);
4349        editorGenRef.current++;
4350        clearTimeout(editorTimerRef.current);
4351        setDumpMode(false);
4352        setEditorStatus('');
4353      }
4354    }, [inTranscript]);
4355    useEffect(() => {
4356      setHighlight(inTranscript ? searchQuery : '');
4357      // Clear the position-based CURRENT (yellow) overlay too. setHighlight
4358      // only clears the scan-based inverse. Without this, the yellow box
4359      // persists at its last screen coords after ctrl-c exits transcript.
4360      if (!inTranscript) setPositions(null);
4361    }, [inTranscript, searchQuery, setHighlight, setPositions]);
4362    const globalKeybindingProps = {
4363      screen,
4364      setScreen,
4365      showAllInTranscript,
4366      setShowAllInTranscript,
4367      messageCount: messages.length,
4368      onEnterTranscript: handleEnterTranscript,
4369      onExitTranscript: handleExitTranscript,
4370      virtualScrollActive,
4371      // Bar-open is a mode (owns keystrokes — j/k type, Esc cancels).
4372      // Navigating (query set, bar closed) is NOT — Esc exits transcript,
4373      // same as less q with highlights still visible. useSearchInput
4374      // doesn't stopPropagation, so without this gate transcript:exit
4375      // would fire on the same Esc that cancels the bar (child registers
4376      // first, fires first, bubbles).
4377      searchBarOpen: searchOpen
4378    };
4379  
4380    // Use frozen lengths to slice arrays, avoiding memory overhead of cloning
4381    const transcriptMessages = frozenTranscriptState ? deferredMessages.slice(0, frozenTranscriptState.messagesLength) : deferredMessages;
4382    const transcriptStreamingToolUses = frozenTranscriptState ? streamingToolUses.slice(0, frozenTranscriptState.streamingToolUsesLength) : streamingToolUses;
4383  
4384    // Handle shift+down for teammate navigation and background task management.
4385    // Guard onOpenBackgroundTasks when a local-jsx dialog (e.g. /mcp) is open —
4386    // otherwise Shift+Down stacks BackgroundTasksDialog on top and deadlocks input.
4387    useBackgroundTaskNavigation({
4388      onOpenBackgroundTasks: isShowingLocalJSXCommand ? undefined : () => setShowBashesDialog(true)
4389    });
4390    // Auto-exit viewing mode when teammate completes or errors
4391    useTeammateViewAutoExit();
4392    if (screen === 'transcript') {
4393      // Virtual scroll replaces the 30-message cap: everything is scrollable
4394      // and memory is bounded by the viewport. Without it, wrapping transcript
4395      // in a ScrollBox would mount all messages (~250 MB on long sessions —
4396      // the exact problem), so the kill switch and non-fullscreen paths must
4397      // fall through to the legacy render: no alt screen, dump to terminal
4398      // scrollback, 30-cap + Ctrl+E. Reusing scrollRef is safe — normal-mode
4399      // and transcript-mode are mutually exclusive (this early return), so
4400      // only one ScrollBox is ever mounted at a time.
4401      const transcriptScrollRef = isFullscreenEnvEnabled() && !disableVirtualScroll && !dumpMode ? scrollRef : undefined;
4402      const transcriptMessagesElement = <Messages messages={transcriptMessages} tools={tools} commands={commands} verbose={true} toolJSX={null} toolUseConfirmQueue={[]} inProgressToolUseIDs={inProgressToolUseIDs} isMessageSelectorVisible={false} conversationId={conversationId} screen={screen} agentDefinitions={agentDefinitions} streamingToolUses={transcriptStreamingToolUses} showAllInTranscript={showAllInTranscript} onOpenRateLimitOptions={handleOpenRateLimitOptions} isLoading={isLoading} hidePastThinking={true} streamingThinking={streamingThinking} scrollRef={transcriptScrollRef} jumpRef={jumpRef} onSearchMatchesChange={onSearchMatchesChange} scanElement={scanElement} setPositions={setPositions} disableRenderCap={dumpMode} />;
4403      const transcriptToolJSX = toolJSX && <Box flexDirection="column" width="100%">
4404          {toolJSX.jsx}
4405        </Box>;
4406      const transcriptReturn = <KeybindingSetup>
4407          <AnimatedTerminalTitle isAnimating={titleIsAnimating} title={terminalTitle} disabled={titleDisabled} noPrefix={showStatusInTerminalTab} />
4408          <GlobalKeybindingHandlers {...globalKeybindingProps} />
4409          {feature('VOICE_MODE') ? <VoiceKeybindingHandler voiceHandleKeyEvent={voice.handleKeyEvent} stripTrailing={voice.stripTrailing} resetAnchor={voice.resetAnchor} isActive={!toolJSX?.isLocalJSXCommand} /> : null}
4410          <CommandKeybindingHandlers onSubmit={onSubmit} isActive={!toolJSX?.isLocalJSXCommand} />
4411          {transcriptScrollRef ?
4412        // ScrollKeybindingHandler must mount before CancelRequestHandler so
4413        // ctrl+c-with-selection copies instead of cancelling the active task.
4414        // Its raw useInput handler only stops propagation when a selection
4415        // exists — without one, ctrl+c falls through to CancelRequestHandler.
4416        <ScrollKeybindingHandler scrollRef={scrollRef}
4417        // Yield wheel/ctrl+u/d to UltraplanChoiceDialog's own scroll
4418        // handler while the modal is showing.
4419        isActive={focusedInputDialog !== 'ultraplan-choice'}
4420        // g/G/j/k/ctrl+u/ctrl+d would eat keystrokes the search bar
4421        // wants. Off while searching.
4422        isModal={!searchOpen}
4423        // Manual scroll exits the search context — clear the yellow
4424        // current-match marker. Positions are (msg, rowOffset)-keyed;
4425        // j/k changes scrollTop so rowOffset is stale → wrong row
4426        // gets yellow. Next n/N re-establishes via step()→jump().
4427        onScroll={() => jumpRef.current?.disarmSearch()} /> : null}
4428          <CancelRequestHandler {...cancelRequestProps} />
4429          {transcriptScrollRef ? <FullscreenLayout scrollRef={scrollRef} scrollable={<>
4430                  {transcriptMessagesElement}
4431                  {transcriptToolJSX}
4432                  <SandboxViolationExpandedView />
4433                </>} bottom={searchOpen ? <TranscriptSearchBar jumpRef={jumpRef}
4434        // Seed was tried (c01578c8) — broke /hello muscle
4435        // memory (cursor lands after 'foo', /hello → foohello).
4436        // Cancel-restore handles the 'don't lose prior search'
4437        // concern differently (onCancel re-applies searchQuery).
4438        initialQuery="" count={searchCount} current={searchCurrent} onClose={q => {
4439          // Enter — commit. 0-match guard: junk query shouldn't
4440          // persist (badge hidden, n/N dead anyway).
4441          setSearchQuery(searchCount > 0 ? q : '');
4442          setSearchOpen(false);
4443          // onCancel path: bar unmounts before its useEffect([query])
4444          // can fire with ''. Without this, searchCount stays stale
4445          // (n guard at :4956 passes) and VML's matches[] too
4446          // (nextMatch walks the old array). Phantom nav, no
4447          // highlight. onExit (Enter, q non-empty) still commits.
4448          if (!q) {
4449            setSearchCount(0);
4450            setSearchCurrent(0);
4451            jumpRef.current?.setSearchQuery('');
4452          }
4453        }} onCancel={() => {
4454          // Esc/ctrl+c/ctrl+g — undo. Bar's effect last fired
4455          // with whatever was typed. searchQuery (REPL state)
4456          // is unchanged since / (onClose = commit, didn't run).
4457          // Two VML calls: '' restores anchor (0-match else-
4458          // branch), then searchQuery re-scans from anchor's
4459          // nearest. Both synchronous — one React batch.
4460          // setHighlight explicit: REPL's sync-effect dep is
4461          // searchQuery (unchanged), wouldn't re-fire.
4462          setSearchOpen(false);
4463          jumpRef.current?.setSearchQuery('');
4464          jumpRef.current?.setSearchQuery(searchQuery);
4465          setHighlight(searchQuery);
4466        }} setHighlight={setHighlight} /> : <TranscriptModeFooter showAllInTranscript={showAllInTranscript} virtualScroll={true} status={editorStatus || undefined} searchBadge={searchQuery && searchCount > 0 ? {
4467          current: searchCurrent,
4468          count: searchCount
4469        } : undefined} />} /> : <>
4470              {transcriptMessagesElement}
4471              {transcriptToolJSX}
4472              <SandboxViolationExpandedView />
4473              <TranscriptModeFooter showAllInTranscript={showAllInTranscript} virtualScroll={false} suppressShowAll={dumpMode} status={editorStatus || undefined} />
4474            </>}
4475        </KeybindingSetup>;
4476      // The virtual-scroll branch (FullscreenLayout above) needs
4477      // <AlternateScreen>'s <Box height={rows}> constraint — without it,
4478      // ScrollBox's flexGrow has no ceiling, viewport = content height,
4479      // scrollTop pins at 0, and Ink's screen buffer sizes to the full
4480      // spacer (200×5k+ rows on long sessions). Same root type + props as
4481      // normal mode's wrap below so React reconciles and the alt buffer
4482      // stays entered across toggle. The 30-cap dump branch stays
4483      // unwrapped — it wants native terminal scrollback.
4484      if (transcriptScrollRef) {
4485        return <AlternateScreen mouseTracking={isMouseTrackingEnabled()}>
4486            {transcriptReturn}
4487          </AlternateScreen>;
4488      }
4489      return transcriptReturn;
4490    }
4491  
4492    // Get viewed agent task (inlined from selectors for explicit data flow).
4493    // viewedAgentTask: teammate OR local_agent — drives the boolean checks
4494    // below. viewedTeammateTask: teammate-only narrowed, for teammate-specific
4495    // field access (inProgressToolUseIDs).
4496    const viewedTask = viewingAgentTaskId ? tasks[viewingAgentTaskId] : undefined;
4497    const viewedTeammateTask = viewedTask && isInProcessTeammateTask(viewedTask) ? viewedTask : undefined;
4498    const viewedAgentTask = viewedTeammateTask ?? (viewedTask && isLocalAgentTask(viewedTask) ? viewedTask : undefined);
4499  
4500    // Bypass useDeferredValue when streaming text is showing so Messages renders
4501    // the final message in the same frame streaming text clears. Also bypass when
4502    // not loading — deferredMessages only matters during streaming (keeps input
4503    // responsive); after the turn ends, showing messages immediately prevents a
4504    // jitter gap where the spinner is gone but the answer hasn't appeared yet.
4505    // Only reducedMotion users keep the deferred path during loading.
4506    const usesSyncMessages = showStreamingText || !isLoading;
4507    // When viewing an agent, never fall through to leader — empty until
4508    // bootstrap/stream fills. Closes the see-leader-type-agent footgun.
4509    const displayedMessages = viewedAgentTask ? viewedAgentTask.messages ?? [] : usesSyncMessages ? messages : deferredMessages;
4510    // Show the placeholder until the real user message appears in
4511    // displayedMessages. userInputOnProcessing stays set for the whole turn
4512    // (cleared in resetLoadingState); this length check hides it once
4513    // displayedMessages grows past the baseline captured at submit time.
4514    // Covers both gaps: before setMessages is called (processUserInput), and
4515    // while deferredMessages lags behind messages. Suppressed when viewing an
4516    // agent — displayedMessages is a different array there, and onAgentSubmit
4517    // doesn't use the placeholder anyway.
4518    const placeholderText = userInputOnProcessing && !viewedAgentTask && displayedMessages.length <= userInputBaselineRef.current ? userInputOnProcessing : undefined;
4519    const toolPermissionOverlay = focusedInputDialog === 'tool-permission' ? <PermissionRequest key={toolUseConfirmQueue[0]?.toolUseID} onDone={() => setToolUseConfirmQueue(([_, ...tail]) => tail)} onReject={handleQueuedCommandOnCancel} toolUseConfirm={toolUseConfirmQueue[0]!} toolUseContext={getToolUseContext(messages, messages, abortController ?? createAbortController(), mainLoopModel)} verbose={verbose} workerBadge={toolUseConfirmQueue[0]?.workerBadge} setStickyFooter={isFullscreenEnvEnabled() ? setPermissionStickyFooter : undefined} /> : null;
4520  
4521    // Narrow terminals: companion collapses to a one-liner that REPL stacks
4522    // on its own row (above input in fullscreen, below in scrollback) instead
4523    // of row-beside. Wide terminals keep the row layout with sprite on the right.
4524    const companionNarrow = transcriptCols < MIN_COLS_FOR_FULL_SPRITE;
4525    // Hide the sprite when PromptInput early-returns BackgroundTasksDialog.
4526    // The sprite sits as a row sibling of PromptInput, so the dialog's Pane
4527    // divider draws at useTerminalSize() width but only gets terminalWidth -
4528    // spriteWidth — divider stops short and dialog text wraps early. Don't
4529    // check footerSelection: pill FOCUS (arrow-down to tasks pill) must keep
4530    // the sprite visible so arrow-right can navigate to it.
4531    const companionVisible = !toolJSX?.shouldHidePromptInput && !focusedInputDialog && !showBashesDialog;
4532  
4533    // In fullscreen, ALL local-jsx slash commands float in the modal slot —
4534    // FullscreenLayout wraps them in an absolute-positioned bottom-anchored
4535    // pane (▔ divider, ModalContext). Pane/Dialog inside detect the context
4536    // and skip their own top-level frame. Non-fullscreen keeps the inline
4537    // render paths below. Commands that used to route through bottom
4538    // (immediate: /model, /mcp, /btw, ...) and scrollable (non-immediate:
4539    // /config, /theme, /diff, ...) both go here now.
4540    const toolJsxCentered = isFullscreenEnvEnabled() && toolJSX?.isLocalJSXCommand === true;
4541    const centeredModal: React.ReactNode = toolJsxCentered ? toolJSX!.jsx : null;
4542  
4543    // <AlternateScreen> at the root: everything below is inside its
4544    // <Box height={rows}>. Handlers/contexts are zero-height so ScrollBox's
4545    // flexGrow in FullscreenLayout resolves against this Box. The transcript
4546    // early return above wraps its virtual-scroll branch the same way; only
4547    // the 30-cap dump branch stays unwrapped for native terminal scrollback.
4548    const mainReturn = <KeybindingSetup>
4549        <AnimatedTerminalTitle isAnimating={titleIsAnimating} title={terminalTitle} disabled={titleDisabled} noPrefix={showStatusInTerminalTab} />
4550        <GlobalKeybindingHandlers {...globalKeybindingProps} />
4551        {feature('VOICE_MODE') ? <VoiceKeybindingHandler voiceHandleKeyEvent={voice.handleKeyEvent} stripTrailing={voice.stripTrailing} resetAnchor={voice.resetAnchor} isActive={!toolJSX?.isLocalJSXCommand} /> : null}
4552        <CommandKeybindingHandlers onSubmit={onSubmit} isActive={!toolJSX?.isLocalJSXCommand} />
4553        {/* ScrollKeybindingHandler must mount before CancelRequestHandler so
4554            ctrl+c-with-selection copies instead of cancelling the active task.
4555            Its raw useInput handler only stops propagation when a selection
4556            exists — without one, ctrl+c falls through to CancelRequestHandler.
4557            PgUp/PgDn/wheel always scroll the transcript behind the modal —
4558            the modal's inner ScrollBox is not keyboard-driven. onScroll
4559            stays suppressed while a modal is showing so scroll doesn't
4560            stamp divider/pill state. */}
4561        <ScrollKeybindingHandler scrollRef={scrollRef} isActive={isFullscreenEnvEnabled() && (centeredModal != null || !focusedInputDialog || focusedInputDialog === 'tool-permission')} onScroll={centeredModal || toolPermissionOverlay || viewedAgentTask ? undefined : composedOnScroll} />
4562        {feature('MESSAGE_ACTIONS') && isFullscreenEnvEnabled() && !disableMessageActions ? <MessageActionsKeybindings handlers={messageActionHandlers} isActive={cursor !== null} /> : null}
4563        <CancelRequestHandler {...cancelRequestProps} />
4564        <MCPConnectionManager key={remountKey} dynamicMcpConfig={dynamicMcpConfig} isStrictMcpConfig={strictMcpConfig}>
4565          <FullscreenLayout scrollRef={scrollRef} overlay={toolPermissionOverlay} bottomFloat={feature('BUDDY') && companionVisible && !companionNarrow ? <CompanionFloatingBubble /> : undefined} modal={centeredModal} modalScrollRef={modalScrollRef} dividerYRef={dividerYRef} hidePill={!!viewedAgentTask} hideSticky={!!viewedTeammateTask} newMessageCount={unseenDivider?.count ?? 0} onPillClick={() => {
4566          setCursor(null);
4567          jumpToNew(scrollRef.current);
4568        }} scrollable={<>
4569                <TeammateViewHeader />
4570                <Messages messages={displayedMessages} tools={tools} commands={commands} verbose={verbose} toolJSX={toolJSX} toolUseConfirmQueue={toolUseConfirmQueue} inProgressToolUseIDs={viewedTeammateTask ? viewedTeammateTask.inProgressToolUseIDs ?? new Set() : inProgressToolUseIDs} isMessageSelectorVisible={isMessageSelectorVisible} conversationId={conversationId} screen={screen} streamingToolUses={streamingToolUses} showAllInTranscript={showAllInTranscript} agentDefinitions={agentDefinitions} onOpenRateLimitOptions={handleOpenRateLimitOptions} isLoading={isLoading} streamingText={isLoading && !viewedAgentTask ? visibleStreamingText : null} isBriefOnly={viewedAgentTask ? false : isBriefOnly} unseenDivider={viewedAgentTask ? undefined : unseenDivider} scrollRef={isFullscreenEnvEnabled() ? scrollRef : undefined} trackStickyPrompt={isFullscreenEnvEnabled() ? true : undefined} cursor={cursor} setCursor={setCursor} cursorNavRef={cursorNavRef} />
4571                <AwsAuthStatusBox />
4572                {/* Hide the processing placeholder while a modal is showing —
4573                    it would sit at the last visible transcript row right above
4574                    the ▔ divider, showing "❯ /config" as redundant clutter
4575                    (the modal IS the /config UI). Outside modals it stays so
4576                    the user sees their input echoed while Claude processes. */}
4577                {!disabled && placeholderText && !centeredModal && <UserTextMessage param={{
4578            text: placeholderText,
4579            type: 'text'
4580          }} addMargin={true} verbose={verbose} />}
4581                {toolJSX && !(toolJSX.isLocalJSXCommand && toolJSX.isImmediate) && !toolJsxCentered && <Box flexDirection="column" width="100%">
4582                      {toolJSX.jsx}
4583                    </Box>}
4584                {"external" === 'ant' && <TungstenLiveMonitor />}
4585                {feature('WEB_BROWSER_TOOL') ? WebBrowserPanelModule && <WebBrowserPanelModule.WebBrowserPanel /> : null}
4586                <Box flexGrow={1} />
4587                {showSpinner && <SpinnerWithVerb mode={streamMode} spinnerTip={spinnerTip} responseLengthRef={responseLengthRef} apiMetricsRef={apiMetricsRef} overrideMessage={spinnerMessage} spinnerSuffix={stopHookSpinnerSuffix} verbose={verbose} loadingStartTimeRef={loadingStartTimeRef} totalPausedMsRef={totalPausedMsRef} pauseStartTimeRef={pauseStartTimeRef} overrideColor={spinnerColor} overrideShimmerColor={spinnerShimmerColor} hasActiveTools={inProgressToolUseIDs.size > 0} leaderIsIdle={!isLoading} />}
4588                {!showSpinner && !isLoading && !userInputOnProcessing && !hasRunningTeammates && isBriefOnly && !viewedAgentTask && <BriefIdleStatus />}
4589                {isFullscreenEnvEnabled() && <PromptInputQueuedCommands />}
4590              </>} bottom={<Box flexDirection={feature('BUDDY') && companionNarrow ? 'column' : 'row'} width="100%" alignItems={feature('BUDDY') && companionNarrow ? undefined : 'flex-end'}>
4591                {feature('BUDDY') && companionNarrow && isFullscreenEnvEnabled() && companionVisible ? <CompanionSprite /> : null}
4592                <Box flexDirection="column" flexGrow={1}>
4593                  {permissionStickyFooter}
4594                  {/* Immediate local-jsx commands (/btw, /sandbox, /assistant,
4595                    /issue) render here, NOT inside scrollable. They stay mounted
4596                    while the main conversation streams behind them, so ScrollBox
4597                    relayouts on each new message would drag them around. bottom
4598                    is flexShrink={0} outside the ScrollBox — it never moves.
4599                    Non-immediate local-jsx (/diff, /status, /theme, ~40 others)
4600                    stays in scrollable: the main loop is paused so no jiggle,
4601                    and their tall content (DiffDetailView renders up to 400
4602                    lines with no internal scroll) needs the outer ScrollBox. */}
4603                  {toolJSX?.isLocalJSXCommand && toolJSX.isImmediate && !toolJsxCentered && <Box flexDirection="column" width="100%">
4604                        {toolJSX.jsx}
4605                      </Box>}
4606                  {!showSpinner && !toolJSX?.isLocalJSXCommand && showExpandedTodos && tasksV2 && tasksV2.length > 0 && <Box width="100%" flexDirection="column">
4607                        <TaskListV2 tasks={tasksV2} isStandalone={true} />
4608                      </Box>}
4609                  {focusedInputDialog === 'sandbox-permission' && <SandboxPermissionRequest key={sandboxPermissionRequestQueue[0]!.hostPattern.host} hostPattern={sandboxPermissionRequestQueue[0]!.hostPattern} onUserResponse={(response: {
4610              allow: boolean;
4611              persistToSettings: boolean;
4612            }) => {
4613              const {
4614                allow,
4615                persistToSettings
4616              } = response;
4617              const currentRequest = sandboxPermissionRequestQueue[0];
4618              if (!currentRequest) return;
4619              const approvedHost = currentRequest.hostPattern.host;
4620              if (persistToSettings) {
4621                const update = {
4622                  type: 'addRules' as const,
4623                  rules: [{
4624                    toolName: WEB_FETCH_TOOL_NAME,
4625                    ruleContent: `domain:${approvedHost}`
4626                  }],
4627                  behavior: (allow ? 'allow' : 'deny') as 'allow' | 'deny',
4628                  destination: 'localSettings' as const
4629                };
4630                setAppState(prev => ({
4631                  ...prev,
4632                  toolPermissionContext: applyPermissionUpdate(prev.toolPermissionContext, update)
4633                }));
4634                persistPermissionUpdate(update);
4635  
4636                // Immediately update sandbox in-memory config to prevent race conditions
4637                // where pending requests slip through before settings change is detected
4638                SandboxManager.refreshConfig();
4639              }
4640  
4641              // Resolve ALL pending requests for the same host (not just the first one)
4642              // This handles the case where multiple parallel requests came in for the same domain
4643              setSandboxPermissionRequestQueue(queue => {
4644                queue.filter(item => item.hostPattern.host === approvedHost).forEach(item => item.resolvePromise(allow));
4645                return queue.filter(item => item.hostPattern.host !== approvedHost);
4646              });
4647  
4648              // Clean up bridge subscriptions and cancel remote prompts
4649              // for this host since the local user already responded.
4650              const cleanups = sandboxBridgeCleanupRef.current.get(approvedHost);
4651              if (cleanups) {
4652                for (const fn of cleanups) {
4653                  fn();
4654                }
4655                sandboxBridgeCleanupRef.current.delete(approvedHost);
4656              }
4657            }} />}
4658                  {focusedInputDialog === 'prompt' && <PromptDialog key={promptQueue[0]!.request.prompt} title={promptQueue[0]!.title} toolInputSummary={promptQueue[0]!.toolInputSummary} request={promptQueue[0]!.request} onRespond={selectedKey => {
4659              const item = promptQueue[0];
4660              if (!item) return;
4661              item.resolve({
4662                prompt_response: item.request.prompt,
4663                selected: selectedKey
4664              });
4665              setPromptQueue(([, ...tail]) => tail);
4666            }} onAbort={() => {
4667              const item = promptQueue[0];
4668              if (!item) return;
4669              item.reject(new Error('Prompt cancelled by user'));
4670              setPromptQueue(([, ...tail]) => tail);
4671            }} />}
4672                  {/* Show pending indicator on worker while waiting for leader approval */}
4673                  {pendingWorkerRequest && <WorkerPendingPermission toolName={pendingWorkerRequest.toolName} description={pendingWorkerRequest.description} />}
4674                  {/* Show pending indicator for sandbox permission on worker side */}
4675                  {pendingSandboxRequest && <WorkerPendingPermission toolName="Network Access" description={`Waiting for leader to approve network access to ${pendingSandboxRequest.host}`} />}
4676                  {/* Worker sandbox permission requests from swarm workers */}
4677                  {focusedInputDialog === 'worker-sandbox-permission' && <SandboxPermissionRequest key={workerSandboxPermissions.queue[0]!.requestId} hostPattern={{
4678              host: workerSandboxPermissions.queue[0]!.host,
4679              port: undefined
4680            } as NetworkHostPattern} onUserResponse={(response: {
4681              allow: boolean;
4682              persistToSettings: boolean;
4683            }) => {
4684              const {
4685                allow,
4686                persistToSettings
4687              } = response;
4688              const currentRequest = workerSandboxPermissions.queue[0];
4689              if (!currentRequest) return;
4690              const approvedHost = currentRequest.host;
4691  
4692              // Send response via mailbox to the worker
4693              void sendSandboxPermissionResponseViaMailbox(currentRequest.workerName, currentRequest.requestId, approvedHost, allow, teamContext?.teamName);
4694              if (persistToSettings && allow) {
4695                const update = {
4696                  type: 'addRules' as const,
4697                  rules: [{
4698                    toolName: WEB_FETCH_TOOL_NAME,
4699                    ruleContent: `domain:${approvedHost}`
4700                  }],
4701                  behavior: 'allow' as const,
4702                  destination: 'localSettings' as const
4703                };
4704                setAppState(prev => ({
4705                  ...prev,
4706                  toolPermissionContext: applyPermissionUpdate(prev.toolPermissionContext, update)
4707                }));
4708                persistPermissionUpdate(update);
4709                SandboxManager.refreshConfig();
4710              }
4711  
4712              // Remove from queue
4713              setAppState(prev => ({
4714                ...prev,
4715                workerSandboxPermissions: {
4716                  ...prev.workerSandboxPermissions,
4717                  queue: prev.workerSandboxPermissions.queue.slice(1)
4718                }
4719              }));
4720            }} />}
4721                  {focusedInputDialog === 'elicitation' && <ElicitationDialog key={elicitation.queue[0]!.serverName + ':' + String(elicitation.queue[0]!.requestId)} event={elicitation.queue[0]!} onResponse={(action, content) => {
4722              const currentRequest = elicitation.queue[0];
4723              if (!currentRequest) return;
4724              // Call respond callback to resolve Promise
4725              currentRequest.respond({
4726                action,
4727                content
4728              });
4729              // For URL accept, keep in queue for phase 2
4730              const isUrlAccept = currentRequest.params.mode === 'url' && action === 'accept';
4731              if (!isUrlAccept) {
4732                setAppState(prev => ({
4733                  ...prev,
4734                  elicitation: {
4735                    queue: prev.elicitation.queue.slice(1)
4736                  }
4737                }));
4738              }
4739            }} onWaitingDismiss={action => {
4740              const currentRequest = elicitation.queue[0];
4741              // Remove from queue
4742              setAppState(prev => ({
4743                ...prev,
4744                elicitation: {
4745                  queue: prev.elicitation.queue.slice(1)
4746                }
4747              }));
4748              currentRequest?.onWaitingDismiss?.(action);
4749            }} />}
4750                  {focusedInputDialog === 'cost' && <CostThresholdDialog onDone={() => {
4751              setShowCostDialog(false);
4752              setHaveShownCostDialog(true);
4753              saveGlobalConfig(current => ({
4754                ...current,
4755                hasAcknowledgedCostThreshold: true
4756              }));
4757              logEvent('tengu_cost_threshold_acknowledged', {});
4758            }} />}
4759                  {focusedInputDialog === 'idle-return' && idleReturnPending && <IdleReturnDialog idleMinutes={idleReturnPending.idleMinutes} totalInputTokens={getTotalInputTokens()} onDone={async action => {
4760              const pending = idleReturnPending;
4761              setIdleReturnPending(null);
4762              logEvent('tengu_idle_return_action', {
4763                action: action as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
4764                idleMinutes: Math.round(pending.idleMinutes),
4765                messageCount: messagesRef.current.length,
4766                totalInputTokens: getTotalInputTokens()
4767              });
4768              if (action === 'dismiss') {
4769                setInputValue(pending.input);
4770                return;
4771              }
4772              if (action === 'never') {
4773                saveGlobalConfig(current => {
4774                  if (current.idleReturnDismissed) return current;
4775                  return {
4776                    ...current,
4777                    idleReturnDismissed: true
4778                  };
4779                });
4780              }
4781              if (action === 'clear') {
4782                const {
4783                  clearConversation
4784                } = await import('../commands/clear/conversation.js');
4785                await clearConversation({
4786                  setMessages,
4787                  readFileState: readFileState.current,
4788                  discoveredSkillNames: discoveredSkillNamesRef.current,
4789                  loadedNestedMemoryPaths: loadedNestedMemoryPathsRef.current,
4790                  getAppState: () => store.getState(),
4791                  setAppState,
4792                  setConversationId
4793                });
4794                haikuTitleAttemptedRef.current = false;
4795                setHaikuTitle(undefined);
4796                bashTools.current.clear();
4797                bashToolsProcessedIdx.current = 0;
4798              }
4799              skipIdleCheckRef.current = true;
4800              void onSubmitRef.current(pending.input, {
4801                setCursorOffset: () => {},
4802                clearBuffer: () => {},
4803                resetHistory: () => {}
4804              });
4805            }} />}
4806                  {focusedInputDialog === 'ide-onboarding' && <IdeOnboardingDialog onDone={() => setShowIdeOnboarding(false)} installationStatus={ideInstallationStatus} />}
4807                  {"external" === 'ant' && focusedInputDialog === 'model-switch' && AntModelSwitchCallout && <AntModelSwitchCallout onDone={(selection: string, modelAlias?: string) => {
4808              setShowModelSwitchCallout(false);
4809              if (selection === 'switch' && modelAlias) {
4810                setAppState(prev => ({
4811                  ...prev,
4812                  mainLoopModel: modelAlias,
4813                  mainLoopModelForSession: null
4814                }));
4815              }
4816            }} />}
4817                  {"external" === 'ant' && focusedInputDialog === 'undercover-callout' && UndercoverAutoCallout && <UndercoverAutoCallout onDone={() => setShowUndercoverCallout(false)} />}
4818                  {focusedInputDialog === 'effort-callout' && <EffortCallout model={mainLoopModel} onDone={selection => {
4819              setShowEffortCallout(false);
4820              if (selection !== 'dismiss') {
4821                setAppState(prev => ({
4822                  ...prev,
4823                  effortValue: selection
4824                }));
4825              }
4826            }} />}
4827                  {focusedInputDialog === 'remote-callout' && <RemoteCallout onDone={selection => {
4828              setAppState(prev => {
4829                if (!prev.showRemoteCallout) return prev;
4830                return {
4831                  ...prev,
4832                  showRemoteCallout: false,
4833                  ...(selection === 'enable' && {
4834                    replBridgeEnabled: true,
4835                    replBridgeExplicit: true,
4836                    replBridgeOutboundOnly: false
4837                  })
4838                };
4839              });
4840            }} />}
4841  
4842                  {exitFlow}
4843  
4844                  {focusedInputDialog === 'plugin-hint' && hintRecommendation && <PluginHintMenu pluginName={hintRecommendation.pluginName} pluginDescription={hintRecommendation.pluginDescription} marketplaceName={hintRecommendation.marketplaceName} sourceCommand={hintRecommendation.sourceCommand} onResponse={handleHintResponse} />}
4845  
4846                  {focusedInputDialog === 'lsp-recommendation' && lspRecommendation && <LspRecommendationMenu pluginName={lspRecommendation.pluginName} pluginDescription={lspRecommendation.pluginDescription} fileExtension={lspRecommendation.fileExtension} onResponse={handleLspResponse} />}
4847  
4848                  {focusedInputDialog === 'desktop-upsell' && <DesktopUpsellStartup onDone={() => setShowDesktopUpsellStartup(false)} />}
4849  
4850                  {feature('ULTRAPLAN') ? focusedInputDialog === 'ultraplan-choice' && ultraplanPendingChoice && <UltraplanChoiceDialog plan={ultraplanPendingChoice.plan} sessionId={ultraplanPendingChoice.sessionId} taskId={ultraplanPendingChoice.taskId} setMessages={setMessages} readFileState={readFileState.current} getAppState={() => store.getState()} setConversationId={setConversationId} /> : null}
4851  
4852                  {feature('ULTRAPLAN') ? focusedInputDialog === 'ultraplan-launch' && ultraplanLaunchPending && <UltraplanLaunchDialog onChoice={(choice, opts) => {
4853              const blurb = ultraplanLaunchPending.blurb;
4854              setAppState(prev => prev.ultraplanLaunchPending ? {
4855                ...prev,
4856                ultraplanLaunchPending: undefined
4857              } : prev);
4858              if (choice === 'cancel') return;
4859              // Command's onDone used display:'skip', so add the
4860              // echo here — gives immediate feedback before the
4861              // ~5s teleportToRemote resolves.
4862              setMessages(prev => [...prev, createCommandInputMessage(formatCommandInputTags('ultraplan', blurb))]);
4863              const appendStdout = (msg: string) => setMessages(prev => [...prev, createCommandInputMessage(`<${LOCAL_COMMAND_STDOUT_TAG}>${escapeXml(msg)}</${LOCAL_COMMAND_STDOUT_TAG}>`)]);
4864              // Defer the second message if a query is mid-turn
4865              // so it lands after the assistant reply, not
4866              // between the user's prompt and the reply.
4867              const appendWhenIdle = (msg: string) => {
4868                if (!queryGuard.isActive) {
4869                  appendStdout(msg);
4870                  return;
4871                }
4872                const unsub = queryGuard.subscribe(() => {
4873                  if (queryGuard.isActive) return;
4874                  unsub();
4875                  // Skip if the user stopped ultraplan while we
4876                  // were waiting — avoids a stale "Monitoring
4877                  // <url>" message for a session that's gone.
4878                  if (!store.getState().ultraplanSessionUrl) return;
4879                  appendStdout(msg);
4880                });
4881              };
4882              void launchUltraplan({
4883                blurb,
4884                getAppState: () => store.getState(),
4885                setAppState,
4886                signal: createAbortController().signal,
4887                disconnectedBridge: opts?.disconnectedBridge,
4888                onSessionReady: appendWhenIdle
4889              }).then(appendStdout).catch(logError);
4890            }} /> : null}
4891  
4892                  {mrRender()}
4893  
4894                  {!toolJSX?.shouldHidePromptInput && !focusedInputDialog && !isExiting && !disabled && !cursor && <>
4895                        {autoRunIssueReason && <AutoRunIssueNotification onRun={handleAutoRunIssue} onCancel={handleCancelAutoRunIssue} reason={getAutoRunIssueReasonText(autoRunIssueReason)} />}
4896                        {postCompactSurvey.state !== 'closed' ? <FeedbackSurvey state={postCompactSurvey.state} lastResponse={postCompactSurvey.lastResponse} handleSelect={postCompactSurvey.handleSelect} inputValue={inputValue} setInputValue={setInputValue} onRequestFeedback={handleSurveyRequestFeedback} /> : memorySurvey.state !== 'closed' ? <FeedbackSurvey state={memorySurvey.state} lastResponse={memorySurvey.lastResponse} handleSelect={memorySurvey.handleSelect} handleTranscriptSelect={memorySurvey.handleTranscriptSelect} inputValue={inputValue} setInputValue={setInputValue} onRequestFeedback={handleSurveyRequestFeedback} message="How well did Claude use its memory? (optional)" /> : <FeedbackSurvey state={feedbackSurvey.state} lastResponse={feedbackSurvey.lastResponse} handleSelect={feedbackSurvey.handleSelect} handleTranscriptSelect={feedbackSurvey.handleTranscriptSelect} inputValue={inputValue} setInputValue={setInputValue} onRequestFeedback={didAutoRunIssueRef.current ? undefined : handleSurveyRequestFeedback} />}
4897                        {/* Frustration-triggered transcript sharing prompt */}
4898                        {frustrationDetection.state !== 'closed' && <FeedbackSurvey state={frustrationDetection.state} lastResponse={null} handleSelect={() => {}} handleTranscriptSelect={frustrationDetection.handleTranscriptSelect} inputValue={inputValue} setInputValue={setInputValue} />}
4899                        {/* Skill improvement survey - appears when improvements detected (ant-only) */}
4900                        {"external" === 'ant' && skillImprovementSurvey.suggestion && <SkillImprovementSurvey isOpen={skillImprovementSurvey.isOpen} skillName={skillImprovementSurvey.suggestion.skillName} updates={skillImprovementSurvey.suggestion.updates} handleSelect={skillImprovementSurvey.handleSelect} inputValue={inputValue} setInputValue={setInputValue} />}
4901                        {showIssueFlagBanner && <IssueFlagBanner />}
4902                        {}
4903                        <PromptInput debug={debug} ideSelection={ideSelection} hasSuppressedDialogs={!!hasSuppressedDialogs} isLocalJSXCommandActive={isShowingLocalJSXCommand} getToolUseContext={getToolUseContext} toolPermissionContext={toolPermissionContext} setToolPermissionContext={setToolPermissionContext} apiKeyStatus={apiKeyStatus} commands={commands} agents={agentDefinitions.activeAgents} isLoading={isLoading} onExit={handleExit} verbose={verbose} messages={messages} onAutoUpdaterResult={setAutoUpdaterResult} autoUpdaterResult={autoUpdaterResult} input={inputValue} onInputChange={setInputValue} mode={inputMode} onModeChange={setInputMode} stashedPrompt={stashedPrompt} setStashedPrompt={setStashedPrompt} submitCount={submitCount} onShowMessageSelector={handleShowMessageSelector} onMessageActionsEnter={
4904              // Works during isLoading — edit cancels first; uuid selection survives appends.
4905              feature('MESSAGE_ACTIONS') && isFullscreenEnvEnabled() && !disableMessageActions ? enterMessageActions : undefined} mcpClients={mcpClients} pastedContents={pastedContents} setPastedContents={setPastedContents} vimMode={vimMode} setVimMode={setVimMode} showBashesDialog={showBashesDialog} setShowBashesDialog={setShowBashesDialog} onSubmit={onSubmit} onAgentSubmit={onAgentSubmit} isSearchingHistory={isSearchingHistory} setIsSearchingHistory={setIsSearchingHistory} helpOpen={isHelpOpen} setHelpOpen={setIsHelpOpen} insertTextRef={feature('VOICE_MODE') ? insertTextRef : undefined} voiceInterimRange={voice.interimRange} />
4906                        <SessionBackgroundHint onBackgroundSession={handleBackgroundSession} isLoading={isLoading} />
4907                      </>}
4908                  {cursor &&
4909            // inputValue is REPL state; typed text survives the round-trip.
4910            <MessageActionsBar cursor={cursor} />}
4911                  {focusedInputDialog === 'message-selector' && <MessageSelector messages={messages} preselectedMessage={messageSelectorPreselect} onPreRestore={onCancel} onRestoreCode={async (message: UserMessage) => {
4912              await fileHistoryRewind((updater: (prev: FileHistoryState) => FileHistoryState) => {
4913                setAppState(prev => ({
4914                  ...prev,
4915                  fileHistory: updater(prev.fileHistory)
4916                }));
4917              }, message.uuid);
4918            }} onSummarize={async (message: UserMessage, feedback?: string, direction: PartialCompactDirection = 'from') => {
4919              // Project snipped messages so the compact model
4920              // doesn't summarize content that was intentionally removed.
4921              const compactMessages = getMessagesAfterCompactBoundary(messages);
4922              const messageIndex = compactMessages.indexOf(message);
4923              if (messageIndex === -1) {
4924                // Selected a snipped or pre-compact message that the
4925                // selector still shows (REPL keeps full history for
4926                // scrollback). Surface why nothing happened instead
4927                // of silently no-oping.
4928                setMessages(prev => [...prev, createSystemMessage('That message is no longer in the active context (snipped or pre-compact). Choose a more recent message.', 'warning')]);
4929                return;
4930              }
4931              const newAbortController = createAbortController();
4932              const context = getToolUseContext(compactMessages, [], newAbortController, mainLoopModel);
4933              const appState = context.getAppState();
4934              const defaultSysPrompt = await getSystemPrompt(context.options.tools, context.options.mainLoopModel, Array.from(appState.toolPermissionContext.additionalWorkingDirectories.keys()), context.options.mcpClients);
4935              const systemPrompt = buildEffectiveSystemPrompt({
4936                mainThreadAgentDefinition: undefined,
4937                toolUseContext: context,
4938                customSystemPrompt: context.options.customSystemPrompt,
4939                defaultSystemPrompt: defaultSysPrompt,
4940                appendSystemPrompt: context.options.appendSystemPrompt
4941              });
4942              const [userContext, systemContext] = await Promise.all([getUserContext(), getSystemContext()]);
4943              const result = await partialCompactConversation(compactMessages, messageIndex, context, {
4944                systemPrompt,
4945                userContext,
4946                systemContext,
4947                toolUseContext: context,
4948                forkContextMessages: compactMessages
4949              }, feedback, direction);
4950              const kept = result.messagesToKeep ?? [];
4951              const ordered = direction === 'up_to' ? [...result.summaryMessages, ...kept] : [...kept, ...result.summaryMessages];
4952              const postCompact = [result.boundaryMarker, ...ordered, ...result.attachments, ...result.hookResults];
4953              // Fullscreen 'from' keeps scrollback; 'up_to' must not
4954              // (old[0] unchanged + grown array means incremental
4955              // useLogMessages path, so boundary never persisted).
4956              // Find by uuid since old is raw REPL history and snipped
4957              // entries can shift the projected messageIndex.
4958              if (isFullscreenEnvEnabled() && direction === 'from') {
4959                setMessages(old => {
4960                  const rawIdx = old.findIndex(m => m.uuid === message.uuid);
4961                  return [...old.slice(0, rawIdx === -1 ? 0 : rawIdx), ...postCompact];
4962                });
4963              } else {
4964                setMessages(postCompact);
4965              }
4966              // Partial compact bypasses handleMessageFromStream — clear
4967              // the context-blocked flag so proactive ticks resume.
4968              if (feature('PROACTIVE') || feature('KAIROS')) {
4969                proactiveModule?.setContextBlocked(false);
4970              }
4971              setConversationId(randomUUID());
4972              runPostCompactCleanup(context.options.querySource);
4973              if (direction === 'from') {
4974                const r = textForResubmit(message);
4975                if (r) {
4976                  setInputValue(r.text);
4977                  setInputMode(r.mode);
4978                }
4979              }
4980  
4981              // Show notification with ctrl+o hint
4982              const historyShortcut = getShortcutDisplay('app:toggleTranscript', 'Global', 'ctrl+o');
4983              addNotification({
4984                key: 'summarize-ctrl-o-hint',
4985                text: `Conversation summarized (${historyShortcut} for history)`,
4986                priority: 'medium',
4987                timeoutMs: 8000
4988              });
4989            }} onRestoreMessage={handleRestoreMessage} onClose={() => {
4990              setIsMessageSelectorVisible(false);
4991              setMessageSelectorPreselect(undefined);
4992            }} />}
4993                  {"external" === 'ant' && <DevBar />}
4994                </Box>
4995                {feature('BUDDY') && !(companionNarrow && isFullscreenEnvEnabled()) && companionVisible ? <CompanionSprite /> : null}
4996              </Box>} />
4997        </MCPConnectionManager>
4998      </KeybindingSetup>;
4999    if (isFullscreenEnvEnabled()) {
5000      return <AlternateScreen mouseTracking={isMouseTrackingEnabled()}>
5001          {mainReturn}
5002        </AlternateScreen>;
5003    }
5004    return mainReturn;
5005  }
5006  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmZWF0dXJlIiwic3Bhd25TeW5jIiwic25hcHNob3RPdXRwdXRUb2tlbnNGb3JUdXJuIiwiZ2V0Q3VycmVudFR1cm5Ub2tlbkJ1ZGdldCIsImdldFR1cm5PdXRwdXRUb2tlbnMiLCJnZXRCdWRnZXRDb250aW51YXRpb25Db3VudCIsImdldFRvdGFsSW5wdXRUb2tlbnMiLCJwYXJzZVRva2VuQnVkZ2V0IiwiY291bnQiLCJkaXJuYW1lIiwiam9pbiIsInRtcGRpciIsImZpZ3VyZXMiLCJ1c2VJbnB1dCIsInVzZVNlYXJjaElucHV0IiwidXNlVGVybWluYWxTaXplIiwidXNlU2VhcmNoSGlnaGxpZ2h0IiwiSnVtcEhhbmRsZSIsInJlbmRlck1lc3NhZ2VzVG9QbGFpblRleHQiLCJvcGVuRmlsZUluRXh0ZXJuYWxFZGl0b3IiLCJ3cml0ZUZpbGUiLCJCb3giLCJUZXh0IiwidXNlU3RkaW4iLCJ1c2VUaGVtZSIsInVzZVRlcm1pbmFsRm9jdXMiLCJ1c2VUZXJtaW5hbFRpdGxlIiwidXNlVGFiU3RhdHVzIiwiVGFiU3RhdHVzS2luZCIsIkNvc3RUaHJlc2hvbGREaWFsb2ciLCJJZGxlUmV0dXJuRGlhbG9nIiwiUmVhY3QiLCJ1c2VFZmZlY3QiLCJ1c2VNZW1vIiwidXNlUmVmIiwidXNlU3RhdGUiLCJ1c2VDYWxsYmFjayIsInVzZURlZmVycmVkVmFsdWUiLCJ1c2VMYXlvdXRFZmZlY3QiLCJSZWZPYmplY3QiLCJ1c2VOb3RpZmljYXRpb25zIiwic2VuZE5vdGlmaWNhdGlvbiIsInN0YXJ0UHJldmVudFNsZWVwIiwic3RvcFByZXZlbnRTbGVlcCIsInVzZVRlcm1pbmFsTm90aWZpY2F0aW9uIiwiaGFzQ3Vyc29yVXBWaWV3cG9ydFlhbmtCdWciLCJjcmVhdGVGaWxlU3RhdGVDYWNoZVdpdGhTaXplTGltaXQiLCJtZXJnZUZpbGVTdGF0ZUNhY2hlcyIsIlJFQURfRklMRV9TVEFURV9DQUNIRV9TSVpFIiwidXBkYXRlTGFzdEludGVyYWN0aW9uVGltZSIsImdldExhc3RJbnRlcmFjdGlvblRpbWUiLCJnZXRPcmlnaW5hbEN3ZCIsImdldFByb2plY3RSb290IiwiZ2V0U2Vzc2lvbklkIiwic3dpdGNoU2Vzc2lvbiIsInNldENvc3RTdGF0ZUZvclJlc3RvcmUiLCJnZXRUdXJuSG9va0R1cmF0aW9uTXMiLCJnZXRUdXJuSG9va0NvdW50IiwicmVzZXRUdXJuSG9va0R1cmF0aW9uIiwiZ2V0VHVyblRvb2xEdXJhdGlvbk1zIiwiZ2V0VHVyblRvb2xDb3VudCIsInJlc2V0VHVyblRvb2xEdXJhdGlvbiIsImdldFR1cm5DbGFzc2lmaWVyRHVyYXRpb25NcyIsImdldFR1cm5DbGFzc2lmaWVyQ291bnQiLCJyZXNldFR1cm5DbGFzc2lmaWVyRHVyYXRpb24iLCJhc1Nlc3Npb25JZCIsImFzQWdlbnRJZCIsImxvZ0ZvckRlYnVnZ2luZyIsIlF1ZXJ5R3VhcmQiLCJpc0VudlRydXRoeSIsImZvcm1hdFRva2VucyIsInRydW5jYXRlVG9XaWR0aCIsImNvbnN1bWVFYXJseUlucHV0Iiwic2V0TWVtYmVyQWN0aXZlIiwiaXNTd2FybVdvcmtlciIsImdlbmVyYXRlU2FuZGJveFJlcXVlc3RJZCIsInNlbmRTYW5kYm94UGVybWlzc2lvblJlcXVlc3RWaWFNYWlsYm94Iiwic2VuZFNhbmRib3hQZXJtaXNzaW9uUmVzcG9uc2VWaWFNYWlsYm94IiwicmVnaXN0ZXJTYW5kYm94UGVybWlzc2lvbkNhbGxiYWNrIiwiZ2V0VGVhbU5hbWUiLCJnZXRBZ2VudE5hbWUiLCJXb3JrZXJQZW5kaW5nUGVybWlzc2lvbiIsImluamVjdFVzZXJNZXNzYWdlVG9UZWFtbWF0ZSIsImdldEFsbEluUHJvY2Vzc1RlYW1tYXRlVGFza3MiLCJpc0xvY2FsQWdlbnRUYXNrIiwicXVldWVQZW5kaW5nTWVzc2FnZSIsImFwcGVuZE1lc3NhZ2VUb0xvY2FsQWdlbnQiLCJMb2NhbEFnZW50VGFza1N0YXRlIiwicmVnaXN0ZXJMZWFkZXJUb29sVXNlQ29uZmlybVF1ZXVlIiwidW5yZWdpc3RlckxlYWRlclRvb2xVc2VDb25maXJtUXVldWUiLCJyZWdpc3RlckxlYWRlclNldFRvb2xQZXJtaXNzaW9uQ29udGV4dCIsInVucmVnaXN0ZXJMZWFkZXJTZXRUb29sUGVybWlzc2lvbkNvbnRleHQiLCJlbmRJbnRlcmFjdGlvblNwYW4iLCJ1c2VMb2dNZXNzYWdlcyIsInVzZVJlcGxCcmlkZ2UiLCJDb21tYW5kIiwiQ29tbWFuZFJlc3VsdERpc3BsYXkiLCJSZXN1bWVFbnRyeXBvaW50IiwiZ2V0Q29tbWFuZE5hbWUiLCJpc0NvbW1hbmRFbmFibGVkIiwiUHJvbXB0SW5wdXRNb2RlIiwiUXVldWVkQ29tbWFuZCIsIlZpbU1vZGUiLCJNZXNzYWdlU2VsZWN0b3IiLCJzZWxlY3RhYmxlVXNlck1lc3NhZ2VzRmlsdGVyIiwibWVzc2FnZXNBZnRlckFyZU9ubHlTeW50aGV0aWMiLCJ1c2VJZGVMb2dnaW5nIiwiUGVybWlzc2lvblJlcXVlc3QiLCJUb29sVXNlQ29uZmlybSIsIkVsaWNpdGF0aW9uRGlhbG9nIiwiUHJvbXB0RGlhbG9nIiwiUHJvbXB0UmVxdWVzdCIsIlByb21wdFJlc3BvbnNlIiwiUHJvbXB0SW5wdXQiLCJQcm9tcHRJbnB1dFF1ZXVlZENvbW1hbmRzIiwidXNlUmVtb3RlU2Vzc2lvbiIsInVzZURpcmVjdENvbm5lY3QiLCJEaXJlY3RDb25uZWN0Q29uZmlnIiwidXNlU1NIU2Vzc2lvbiIsInVzZUFzc2lzdGFudEhpc3RvcnkiLCJTU0hTZXNzaW9uIiwiU2tpbGxJbXByb3ZlbWVudFN1cnZleSIsInVzZVNraWxsSW1wcm92ZW1lbnRTdXJ2ZXkiLCJ1c2VNb3JlUmlnaHQiLCJTcGlubmVyV2l0aFZlcmIiLCJCcmllZklkbGVTdGF0dXMiLCJTcGlubmVyTW9kZSIsImdldFN5c3RlbVByb21wdCIsImJ1aWxkRWZmZWN0aXZlU3lzdGVtUHJvbXB0IiwiZ2V0U3lzdGVtQ29udGV4dCIsImdldFVzZXJDb250ZXh0IiwiZ2V0TWVtb3J5RmlsZXMiLCJzdGFydEJhY2tncm91bmRIb3VzZWtlZXBpbmciLCJnZXRUb3RhbENvc3QiLCJzYXZlQ3VycmVudFNlc3Npb25Db3N0cyIsInJlc2V0Q29zdFN0YXRlIiwiZ2V0U3RvcmVkU2Vzc2lvbkNvc3RzIiwidXNlQ29zdFN1bW1hcnkiLCJ1c2VGcHNNZXRyaWNzIiwidXNlQWZ0ZXJGaXJzdFJlbmRlciIsInVzZURlZmVycmVkSG9va01lc3NhZ2VzIiwiYWRkVG9IaXN0b3J5IiwicmVtb3ZlTGFzdEZyb21IaXN0b3J5IiwiZXhwYW5kUGFzdGVkVGV4dFJlZnMiLCJwYXJzZVJlZmVyZW5jZXMiLCJwcmVwZW5kTW9kZUNoYXJhY3RlclRvSW5wdXQiLCJwcmVwZW5kVG9TaGVsbEhpc3RvcnlDYWNoZSIsInVzZUFwaUtleVZlcmlmaWNhdGlvbiIsIkdsb2JhbEtleWJpbmRpbmdIYW5kbGVycyIsIkNvbW1hbmRLZXliaW5kaW5nSGFuZGxlcnMiLCJLZXliaW5kaW5nU2V0dXAiLCJ1c2VTaG9ydGN1dERpc3BsYXkiLCJnZXRTaG9ydGN1dERpc3BsYXkiLCJDYW5jZWxSZXF1ZXN0SGFuZGxlciIsInVzZUJhY2tncm91bmRUYXNrTmF2aWdhdGlvbiIsInVzZVN3YXJtSW5pdGlhbGl6YXRpb24iLCJ1c2VUZWFtbWF0ZVZpZXdBdXRvRXhpdCIsImVycm9yTWVzc2FnZSIsImlzSHVtYW5UdXJuIiwibG9nRXJyb3IiLCJ1c2VWb2ljZUludGVncmF0aW9uIiwicmVxdWlyZSIsInN0cmlwVHJhaWxpbmciLCJoYW5kbGVLZXlFdmVudCIsInJlc2V0QW5jaG9yIiwiVm9pY2VLZXliaW5kaW5nSGFuZGxlciIsInVzZUZydXN0cmF0aW9uRGV0ZWN0aW9uIiwic3RhdGUiLCJoYW5kbGVUcmFuc2NyaXB0U2VsZWN0IiwidXNlQW50T3JnV2FybmluZ05vdGlmaWNhdGlvbiIsImdldENvb3JkaW5hdG9yVXNlckNvbnRleHQiLCJtY3BDbGllbnRzIiwiUmVhZG9ubHlBcnJheSIsIm5hbWUiLCJzY3JhdGNocGFkRGlyIiwiayIsInVzZUNhblVzZVRvb2wiLCJUb29sUGVybWlzc2lvbkNvbnRleHQiLCJUb29sIiwiYXBwbHlQZXJtaXNzaW9uVXBkYXRlIiwiYXBwbHlQZXJtaXNzaW9uVXBkYXRlcyIsInBlcnNpc3RQZXJtaXNzaW9uVXBkYXRlIiwiYnVpbGRQZXJtaXNzaW9uVXBkYXRlcyIsInN0cmlwRGFuZ2Vyb3VzUGVybWlzc2lvbnNGb3JBdXRvTW9kZSIsImdldFNjcmF0Y2hwYWREaXIiLCJpc1NjcmF0Y2hwYWRFbmFibGVkIiwiV0VCX0ZFVENIX1RPT0xfTkFNRSIsIlNMRUVQX1RPT0xfTkFNRSIsImNsZWFyU3BlY3VsYXRpdmVDaGVja3MiLCJBdXRvVXBkYXRlclJlc3VsdCIsImdldEdsb2JhbENvbmZpZyIsInNhdmVHbG9iYWxDb25maWciLCJnZXRHbG9iYWxDb25maWdXcml0ZUNvdW50IiwiaGFzQ29uc29sZUJpbGxpbmdBY2Nlc3MiLCJsb2dFdmVudCIsIkFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMiLCJnZXRGZWF0dXJlVmFsdWVfQ0FDSEVEX01BWV9CRV9TVEFMRSIsInRleHRGb3JSZXN1Ym1pdCIsImhhbmRsZU1lc3NhZ2VGcm9tU3RyZWFtIiwiU3RyZWFtaW5nVG9vbFVzZSIsIlN0cmVhbWluZ1RoaW5raW5nIiwiaXNDb21wYWN0Qm91bmRhcnlNZXNzYWdlIiwiZ2V0TWVzc2FnZXNBZnRlckNvbXBhY3RCb3VuZGFyeSIsImdldENvbnRlbnRUZXh0IiwiY3JlYXRlVXNlck1lc3NhZ2UiLCJjcmVhdGVBc3Npc3RhbnRNZXNzYWdlIiwiY3JlYXRlVHVybkR1cmF0aW9uTWVzc2FnZSIsImNyZWF0ZUFnZW50c0tpbGxlZE1lc3NhZ2UiLCJjcmVhdGVBcGlNZXRyaWNzTWVzc2FnZSIsImNyZWF0ZVN5c3RlbU1lc3NhZ2UiLCJjcmVhdGVDb21tYW5kSW5wdXRNZXNzYWdlIiwiZm9ybWF0Q29tbWFuZElucHV0VGFncyIsImdlbmVyYXRlU2Vzc2lvblRpdGxlIiwiQkFTSF9JTlBVVF9UQUciLCJDT01NQU5EX01FU1NBR0VfVEFHIiwiQ09NTUFORF9OQU1FX1RBRyIsIkxPQ0FMX0NPTU1BTkRfU1RET1VUX1RBRyIsImVzY2FwZVhtbCIsIlRoaW5raW5nQ29uZmlnIiwiZ3JhY2VmdWxTaHV0ZG93blN5bmMiLCJoYW5kbGVQcm9tcHRTdWJtaXQiLCJQcm9tcHRJbnB1dEhlbHBlcnMiLCJ1c2VRdWV1ZVByb2Nlc3NvciIsInVzZU1haWxib3hCcmlkZ2UiLCJxdWVyeUNoZWNrcG9pbnQiLCJsb2dRdWVyeVByb2ZpbGVSZXBvcnQiLCJNZXNzYWdlIiwiTWVzc2FnZVR5cGUiLCJVc2VyTWVzc2FnZSIsIlByb2dyZXNzTWVzc2FnZSIsIkhvb2tSZXN1bHRNZXNzYWdlIiwiUGFydGlhbENvbXBhY3REaXJlY3Rpb24iLCJxdWVyeSIsIm1lcmdlQ2xpZW50cyIsInVzZU1lcmdlZENsaWVudHMiLCJnZXRRdWVyeVNvdXJjZUZvclJFUEwiLCJ1c2VNZXJnZWRUb29scyIsIm1lcmdlQW5kRmlsdGVyVG9vbHMiLCJ1c2VNZXJnZWRDb21tYW5kcyIsInVzZVNraWxsc0NoYW5nZSIsInVzZU1hbmFnZVBsdWdpbnMiLCJNZXNzYWdlcyIsIlRhc2tMaXN0VjIiLCJUZWFtbWF0ZVZpZXdIZWFkZXIiLCJ1c2VUYXNrc1YyV2l0aENvbGxhcHNlRWZmZWN0IiwibWF5YmVNYXJrUHJvamVjdE9uYm9hcmRpbmdDb21wbGV0ZSIsIk1DUFNlcnZlckNvbm5lY3Rpb24iLCJTY29wZWRNY3BTZXJ2ZXJDb25maWciLCJyYW5kb21VVUlEIiwiVVVJRCIsInByb2Nlc3NTZXNzaW9uU3RhcnRIb29rcyIsImV4ZWN1dGVTZXNzaW9uRW5kSG9va3MiLCJnZXRTZXNzaW9uRW5kSG9va1RpbWVvdXRNcyIsIklERVNlbGVjdGlvbiIsInVzZUlkZVNlbGVjdGlvbiIsImdldFRvb2xzIiwiYXNzZW1ibGVUb29sUG9vbCIsIkFnZW50RGVmaW5pdGlvbiIsInJlc29sdmVBZ2VudFRvb2xzIiwicmVzdW1lQWdlbnRCYWNrZ3JvdW5kIiwidXNlTWFpbkxvb3BNb2RlbCIsInVzZUFwcFN0YXRlIiwidXNlU2V0QXBwU3RhdGUiLCJ1c2VBcHBTdGF0ZVN0b3JlIiwiQ29udGVudEJsb2NrUGFyYW0iLCJJbWFnZUJsb2NrUGFyYW0iLCJQcm9jZXNzVXNlcklucHV0Q29udGV4dCIsIlBhc3RlZENvbnRlbnQiLCJjb3B5UGxhbkZvckZvcmsiLCJjb3B5UGxhbkZvclJlc3VtZSIsImdldFBsYW5TbHVnIiwic2V0UGxhblNsdWciLCJjbGVhclNlc3Npb25NZXRhZGF0YSIsInJlc2V0U2Vzc2lvbkZpbGVQb2ludGVyIiwiYWRvcHRSZXN1bWVkU2Vzc2lvbkZpbGUiLCJyZW1vdmVUcmFuc2NyaXB0TWVzc2FnZSIsInJlc3RvcmVTZXNzaW9uTWV0YWRhdGEiLCJnZXRDdXJyZW50U2Vzc2lvblRpdGxlIiwiaXNFcGhlbWVyYWxUb29sUHJvZ3Jlc3MiLCJpc0xvZ2dhYmxlTWVzc2FnZSIsInNhdmVXb3JrdHJlZVN0YXRlIiwiZ2V0QWdlbnRUcmFuc2NyaXB0IiwiZGVzZXJpYWxpemVNZXNzYWdlcyIsImV4dHJhY3RSZWFkRmlsZXNGcm9tTWVzc2FnZXMiLCJleHRyYWN0QmFzaFRvb2xzRnJvbU1lc3NhZ2VzIiwicmVzZXRNaWNyb2NvbXBhY3RTdGF0ZSIsInJ1blBvc3RDb21wYWN0Q2xlYW51cCIsInByb3Zpc2lvbkNvbnRlbnRSZXBsYWNlbWVudFN0YXRlIiwicmVjb25zdHJ1Y3RDb250ZW50UmVwbGFjZW1lbnRTdGF0ZSIsIkNvbnRlbnRSZXBsYWNlbWVudFJlY29yZCIsInBhcnRpYWxDb21wYWN0Q29udmVyc2F0aW9uIiwiTG9nT3B0aW9uIiwiQWdlbnRDb2xvck5hbWUiLCJmaWxlSGlzdG9yeU1ha2VTbmFwc2hvdCIsIkZpbGVIaXN0b3J5U3RhdGUiLCJmaWxlSGlzdG9yeVJld2luZCIsIkZpbGVIaXN0b3J5U25hcHNob3QiLCJjb3B5RmlsZUhpc3RvcnlGb3JSZXN1bWUiLCJmaWxlSGlzdG9yeUVuYWJsZWQiLCJmaWxlSGlzdG9yeUhhc0FueUNoYW5nZXMiLCJBdHRyaWJ1dGlvblN0YXRlIiwiaW5jcmVtZW50UHJvbXB0Q291bnQiLCJyZWNvcmRBdHRyaWJ1dGlvblNuYXBzaG90IiwiY29tcHV0ZVN0YW5kYWxvbmVBZ2VudENvbnRleHQiLCJyZXN0b3JlQWdlbnRGcm9tU2Vzc2lvbiIsInJlc3RvcmVTZXNzaW9uU3RhdGVGcm9tTG9nIiwicmVzdG9yZVdvcmt0cmVlRm9yUmVzdW1lIiwiZXhpdFJlc3RvcmVkV29ya3RyZWUiLCJpc0JnU2Vzc2lvbiIsInVwZGF0ZVNlc3Npb25OYW1lIiwidXBkYXRlU2Vzc2lvbkFjdGl2aXR5IiwiaXNJblByb2Nlc3NUZWFtbWF0ZVRhc2siLCJJblByb2Nlc3NUZWFtbWF0ZVRhc2tTdGF0ZSIsInJlc3RvcmVSZW1vdGVBZ2VudFRhc2tzIiwidXNlSW5ib3hQb2xsZXIiLCJwcm9hY3RpdmVNb2R1bGUiLCJQUk9BQ1RJVkVfTk9fT1BfU1VCU0NSSUJFIiwiX2NiIiwiUFJPQUNUSVZFX0ZBTFNFIiwiU1VHR0VTVF9CR19QUl9OT09QIiwiX3AiLCJfbiIsInVzZVByb2FjdGl2ZSIsInVzZVNjaGVkdWxlZFRhc2tzIiwiaXNBZ2VudFN3YXJtc0VuYWJsZWQiLCJ1c2VUYXNrTGlzdFdhdGNoZXIiLCJTYW5kYm94QXNrQ2FsbGJhY2siLCJOZXR3b3JrSG9zdFBhdHRlcm4iLCJJREVFeHRlbnNpb25JbnN0YWxsYXRpb25TdGF0dXMiLCJjbG9zZU9wZW5EaWZmcyIsImdldENvbm5lY3RlZElkZUNsaWVudCIsIklkZVR5cGUiLCJ1c2VJREVJbnRlZ3JhdGlvbiIsImV4aXQiLCJFeGl0RmxvdyIsImdldEN1cnJlbnRXb3JrdHJlZVNlc3Npb24iLCJwb3BBbGxFZGl0YWJsZSIsImVucXVldWUiLCJTZXRBcHBTdGF0ZSIsImdldENvbW1hbmRRdWV1ZSIsImdldENvbW1hbmRRdWV1ZUxlbmd0aCIsInJlbW92ZUJ5RmlsdGVyIiwidXNlQ29tbWFuZFF1ZXVlIiwiU2Vzc2lvbkJhY2tncm91bmRIaW50Iiwic3RhcnRCYWNrZ3JvdW5kU2Vzc2lvbiIsInVzZVNlc3Npb25CYWNrZ3JvdW5kaW5nIiwiZGlhZ25vc3RpY1RyYWNrZXIiLCJoYW5kbGVTcGVjdWxhdGlvbkFjY2VwdCIsIkFjdGl2ZVNwZWN1bGF0aW9uU3RhdGUiLCJJZGVPbmJvYXJkaW5nRGlhbG9nIiwiRWZmb3J0Q2FsbG91dCIsInNob3VsZFNob3dFZmZvcnRDYWxsb3V0IiwiRWZmb3J0VmFsdWUiLCJSZW1vdGVDYWxsb3V0IiwiQW50TW9kZWxTd2l0Y2hDYWxsb3V0Iiwic2hvdWxkU2hvd0FudE1vZGVsU3dpdGNoIiwic2hvdWxkU2hvd01vZGVsU3dpdGNoQ2FsbG91dCIsIlVuZGVyY292ZXJBdXRvQ2FsbG91dCIsImFjdGl2aXR5TWFuYWdlciIsImNyZWF0ZUFib3J0Q29udHJvbGxlciIsIk1DUENvbm5lY3Rpb25NYW5hZ2VyIiwidXNlRmVlZGJhY2tTdXJ2ZXkiLCJ1c2VNZW1vcnlTdXJ2ZXkiLCJ1c2VQb3N0Q29tcGFjdFN1cnZleSIsIkZlZWRiYWNrU3VydmV5IiwidXNlSW5zdGFsbE1lc3NhZ2VzIiwidXNlQXdheVN1bW1hcnkiLCJ1c2VDaHJvbWVFeHRlbnNpb25Ob3RpZmljYXRpb24iLCJ1c2VPZmZpY2lhbE1hcmtldHBsYWNlTm90aWZpY2F0aW9uIiwidXNlUHJvbXB0c0Zyb21DbGF1ZGVJbkNocm9tZSIsImdldFRpcFRvU2hvd09uU3Bpbm5lciIsInJlY29yZFNob3duVGlwIiwiVGhlbWUiLCJjaGVja0FuZERpc2FibGVCeXBhc3NQZXJtaXNzaW9uc0lmTmVlZGVkIiwiY2hlY2tBbmREaXNhYmxlQXV0b01vZGVJZk5lZWRlZCIsInVzZUtpY2tPZmZDaGVja0FuZERpc2FibGVCeXBhc3NQZXJtaXNzaW9uc0lmTmVlZGVkIiwidXNlS2lja09mZkNoZWNrQW5kRGlzYWJsZUF1dG9Nb2RlSWZOZWVkZWQiLCJTYW5kYm94TWFuYWdlciIsIlNBTkRCT1hfTkVUV09SS19BQ0NFU1NfVE9PTF9OQU1FIiwidXNlRmlsZUhpc3RvcnlTbmFwc2hvdEluaXQiLCJTYW5kYm94UGVybWlzc2lvblJlcXVlc3QiLCJTYW5kYm94VmlvbGF0aW9uRXhwYW5kZWRWaWV3IiwidXNlU2V0dGluZ3NFcnJvcnMiLCJ1c2VNY3BDb25uZWN0aXZpdHlTdGF0dXMiLCJ1c2VBdXRvTW9kZVVuYXZhaWxhYmxlTm90aWZpY2F0aW9uIiwiQVVUT19NT0RFX0RFU0NSSVBUSU9OIiwidXNlTHNwSW5pdGlhbGl6YXRpb25Ob3RpZmljYXRpb24iLCJ1c2VMc3BQbHVnaW5SZWNvbW1lbmRhdGlvbiIsIkxzcFJlY29tbWVuZGF0aW9uTWVudSIsInVzZUNsYXVkZUNvZGVIaW50UmVjb21tZW5kYXRpb24iLCJQbHVnaW5IaW50TWVudSIsIkRlc2t0b3BVcHNlbGxTdGFydHVwIiwic2hvdWxkU2hvd0Rlc2t0b3BVcHNlbGxTdGFydHVwIiwidXNlUGx1Z2luSW5zdGFsbGF0aW9uU3RhdHVzIiwidXNlUGx1Z2luQXV0b3VwZGF0ZU5vdGlmaWNhdGlvbiIsInBlcmZvcm1TdGFydHVwQ2hlY2tzIiwiVXNlclRleHRNZXNzYWdlIiwiQXdzQXV0aFN0YXR1c0JveCIsInVzZVJhdGVMaW1pdFdhcm5pbmdOb3RpZmljYXRpb24iLCJ1c2VEZXByZWNhdGlvbldhcm5pbmdOb3RpZmljYXRpb24iLCJ1c2VOcG1EZXByZWNhdGlvbk5vdGlmaWNhdGlvbiIsInVzZUlERVN0YXR1c0luZGljYXRvciIsInVzZU1vZGVsTWlncmF0aW9uTm90aWZpY2F0aW9ucyIsInVzZUNhblN3aXRjaFRvRXhpc3RpbmdTdWJzY3JpcHRpb24iLCJ1c2VUZWFtbWF0ZUxpZmVjeWNsZU5vdGlmaWNhdGlvbiIsInVzZUZhc3RNb2RlTm90aWZpY2F0aW9uIiwiQXV0b1J1bklzc3VlTm90aWZpY2F0aW9uIiwic2hvdWxkQXV0b1J1bklzc3VlIiwiZ2V0QXV0b1J1bklzc3VlUmVhc29uVGV4dCIsImdldEF1dG9SdW5Db21tYW5kIiwiQXV0b1J1bklzc3VlUmVhc29uIiwiSG9va1Byb2dyZXNzIiwiVHVuZ3N0ZW5MaXZlTW9uaXRvciIsIldlYkJyb3dzZXJQYW5lbE1vZHVsZSIsIklzc3VlRmxhZ0Jhbm5lciIsInVzZUlzc3VlRmxhZ0Jhbm5lciIsIkNvbXBhbmlvblNwcml0ZSIsIkNvbXBhbmlvbkZsb2F0aW5nQnViYmxlIiwiTUlOX0NPTFNfRk9SX0ZVTExfU1BSSVRFIiwiRGV2QmFyIiwiUmVtb3RlU2Vzc2lvbkNvbmZpZyIsIlJFTU9URV9TQUZFX0NPTU1BTkRTIiwiUmVtb3RlTWVzc2FnZUNvbnRlbnQiLCJGdWxsc2NyZWVuTGF5b3V0IiwidXNlVW5zZWVuRGl2aWRlciIsImNvbXB1dGVVbnNlZW5EaXZpZGVyIiwiaXNGdWxsc2NyZWVuRW52RW5hYmxlZCIsIm1heWJlR2V0VG11eE1vdXNlSGludCIsImlzTW91c2VUcmFja2luZ0VuYWJsZWQiLCJBbHRlcm5hdGVTY3JlZW4iLCJTY3JvbGxLZXliaW5kaW5nSGFuZGxlciIsInVzZU1lc3NhZ2VBY3Rpb25zIiwiTWVzc2FnZUFjdGlvbnNLZXliaW5kaW5ncyIsIk1lc3NhZ2VBY3Rpb25zQmFyIiwiTWVzc2FnZUFjdGlvbnNTdGF0ZSIsIk1lc3NhZ2VBY3Rpb25zTmF2IiwiTWVzc2FnZUFjdGlvbkNhcHMiLCJzZXRDbGlwYm9hcmQiLCJTY3JvbGxCb3hIYW5kbGUiLCJjcmVhdGVBdHRhY2htZW50TWVzc2FnZSIsImdldFF1ZXVlZENvbW1hbmRBdHRhY2htZW50cyIsIkVNUFRZX01DUF9DTElFTlRTIiwiSElTVE9SWV9TVFVCIiwibWF5YmVMb2FkT2xkZXIiLCJfIiwiUkVDRU5UX1NDUk9MTF9SRVBJTl9XSU5ET1dfTVMiLCJtZWRpYW4iLCJ2YWx1ZXMiLCJzb3J0ZWQiLCJzb3J0IiwiYSIsImIiLCJtaWQiLCJNYXRoIiwiZmxvb3IiLCJsZW5ndGgiLCJyb3VuZCIsIlRyYW5zY3JpcHRNb2RlRm9vdGVyIiwidDAiLCIkIiwiX2MiLCJzaG93QWxsSW5UcmFuc2NyaXB0IiwidmlydHVhbFNjcm9sbCIsInNlYXJjaEJhZGdlIiwic3VwcHJlc3NTaG93QWxsIiwidDEiLCJzdGF0dXMiLCJ1bmRlZmluZWQiLCJ0b2dnbGVTaG9ydGN1dCIsInNob3dBbGxTaG9ydGN1dCIsInQyIiwiYXJyb3dVcCIsImFycm93RG93biIsInQzIiwidDQiLCJjdXJyZW50IiwidDUiLCJUcmFuc2NyaXB0U2VhcmNoQmFyIiwianVtcFJlZiIsIm9uQ2xvc2UiLCJvbkNhbmNlbCIsInNldEhpZ2hsaWdodCIsImluaXRpYWxRdWVyeSIsImxhc3RRdWVyeSIsIlJlYWN0Tm9kZSIsImN1cnNvck9mZnNldCIsImlzQWN0aXZlIiwib25FeGl0IiwiaW5kZXhTdGF0dXMiLCJzZXRJbmRleFN0YXR1cyIsIm1zIiwiYWxpdmUiLCJ3YXJtIiwid2FybVNlYXJjaEluZGV4IiwidGhlbiIsInNldFRpbWVvdXQiLCJ3YXJtRG9uZSIsInNldFNlYXJjaFF1ZXJ5Iiwib2ZmIiwiY3Vyc29yQ2hhciIsInNsaWNlIiwiVElUTEVfQU5JTUFUSU9OX0ZSQU1FUyIsIlRJVExFX1NUQVRJQ19QUkVGSVgiLCJUSVRMRV9BTklNQVRJT05fSU5URVJWQUxfTVMiLCJBbmltYXRlZFRlcm1pbmFsVGl0bGUiLCJpc0FuaW1hdGluZyIsInRpdGxlIiwiZGlzYWJsZWQiLCJub1ByZWZpeCIsInRlcm1pbmFsRm9jdXNlZCIsImZyYW1lIiwic2V0RnJhbWUiLCJpbnRlcnZhbCIsInNldEludGVydmFsIiwiX3RlbXAyIiwiY2xlYXJJbnRlcnZhbCIsInByZWZpeCIsInNldEZyYW1lXzAiLCJfdGVtcCIsImYiLCJQcm9wcyIsImNvbW1hbmRzIiwiZGVidWciLCJpbml0aWFsVG9vbHMiLCJpbml0aWFsTWVzc2FnZXMiLCJwZW5kaW5nSG9va01lc3NhZ2VzIiwiUHJvbWlzZSIsImluaXRpYWxGaWxlSGlzdG9yeVNuYXBzaG90cyIsImluaXRpYWxDb250ZW50UmVwbGFjZW1lbnRzIiwiaW5pdGlhbEFnZW50TmFtZSIsImluaXRpYWxBZ2VudENvbG9yIiwiZHluYW1pY01jcENvbmZpZyIsIlJlY29yZCIsImF1dG9Db25uZWN0SWRlRmxhZyIsInN0cmljdE1jcENvbmZpZyIsInN5c3RlbVByb21wdCIsImFwcGVuZFN5c3RlbVByb21wdCIsIm9uQmVmb3JlUXVlcnkiLCJpbnB1dCIsIm5ld01lc3NhZ2VzIiwib25UdXJuQ29tcGxldGUiLCJtZXNzYWdlcyIsIm1haW5UaHJlYWRBZ2VudERlZmluaXRpb24iLCJkaXNhYmxlU2xhc2hDb21tYW5kcyIsInRhc2tMaXN0SWQiLCJyZW1vdGVTZXNzaW9uQ29uZmlnIiwiZGlyZWN0Q29ubmVjdENvbmZpZyIsInNzaFNlc3Npb24iLCJ0aGlua2luZ0NvbmZpZyIsIlNjcmVlbiIsIlJFUEwiLCJpbml0aWFsQ29tbWFuZHMiLCJpbml0aWFsTWNwQ2xpZW50cyIsImluaXRpYWxEeW5hbWljTWNwQ29uZmlnIiwiY3VzdG9tU3lzdGVtUHJvbXB0IiwiaW5pdGlhbE1haW5UaHJlYWRBZ2VudERlZmluaXRpb24iLCJpc1JlbW90ZVNlc3Npb24iLCJ0aXRsZURpc2FibGVkIiwicHJvY2VzcyIsImVudiIsIkNMQVVERV9DT0RFX0RJU0FCTEVfVEVSTUlOQUxfVElUTEUiLCJtb3JlUmlnaHRFbmFibGVkIiwiQ0xBVURFX01PUkVSSUdIVCIsImRpc2FibGVWaXJ0dWFsU2Nyb2xsIiwiQ0xBVURFX0NPREVfRElTQUJMRV9WSVJUVUFMX1NDUk9MTCIsImRpc2FibGVNZXNzYWdlQWN0aW9ucyIsIkNMQVVERV9DT0RFX0RJU0FCTEVfTUVTU0FHRV9BQ1RJT05TIiwic2V0TWFpblRocmVhZEFnZW50RGVmaW5pdGlvbiIsInRvb2xQZXJtaXNzaW9uQ29udGV4dCIsInMiLCJ2ZXJib3NlIiwibWNwIiwicGx1Z2lucyIsImFnZW50RGVmaW5pdGlvbnMiLCJmaWxlSGlzdG9yeSIsImluaXRpYWxNZXNzYWdlIiwicXVldWVkQ29tbWFuZHMiLCJzcGlubmVyVGlwIiwic2hvd0V4cGFuZGVkVG9kb3MiLCJleHBhbmRlZFZpZXciLCJwZW5kaW5nV29ya2VyUmVxdWVzdCIsInBlbmRpbmdTYW5kYm94UmVxdWVzdCIsInRlYW1Db250ZXh0IiwidGFza3MiLCJ3b3JrZXJTYW5kYm94UGVybWlzc2lvbnMiLCJlbGljaXRhdGlvbiIsInVsdHJhcGxhblBlbmRpbmdDaG9pY2UiLCJ1bHRyYXBsYW5MYXVuY2hQZW5kaW5nIiwidmlld2luZ0FnZW50VGFza0lkIiwic2V0QXBwU3RhdGUiLCJ2aWV3ZWRMb2NhbEFnZW50IiwibmVlZHNCb290c3RyYXAiLCJyZXRhaW4iLCJkaXNrTG9hZGVkIiwidGFza0lkIiwicmVzdWx0IiwicHJldiIsInQiLCJsaXZlIiwibGl2ZVV1aWRzIiwiU2V0IiwibWFwIiwibSIsInV1aWQiLCJkaXNrT25seSIsImZpbHRlciIsImhhcyIsInN0b3JlIiwidGVybWluYWwiLCJtYWluTG9vcE1vZGVsIiwibG9jYWxDb21tYW5kcyIsInNldExvY2FsQ29tbWFuZHMiLCJwcm9hY3RpdmVBY3RpdmUiLCJ1c2VTeW5jRXh0ZXJuYWxTdG9yZSIsInN1YnNjcmliZVRvUHJvYWN0aXZlQ2hhbmdlcyIsImlzUHJvYWN0aXZlQWN0aXZlIiwiaXNCcmllZk9ubHkiLCJsb2NhbFRvb2xzIiwic2V0RHluYW1pY01jcENvbmZpZyIsIm9uQ2hhbmdlRHluYW1pY01jcENvbmZpZyIsImNvbmZpZyIsInNjcmVlbiIsInNldFNjcmVlbiIsInNldFNob3dBbGxJblRyYW5zY3JpcHQiLCJkdW1wTW9kZSIsInNldER1bXBNb2RlIiwiZWRpdG9yU3RhdHVzIiwic2V0RWRpdG9yU3RhdHVzIiwiZWRpdG9yR2VuUmVmIiwiZWRpdG9yVGltZXJSZWYiLCJSZXR1cm5UeXBlIiwiZWRpdG9yUmVuZGVyaW5nUmVmIiwiYWRkTm90aWZpY2F0aW9uIiwicmVtb3ZlTm90aWZpY2F0aW9uIiwidHJ5U3VnZ2VzdEJnUFJJbnRlcmNlcHQiLCJjbGllbnRzIiwiaWRlU2VsZWN0aW9uIiwic2V0SURFU2VsZWN0aW9uIiwiaWRlVG9JbnN0YWxsRXh0ZW5zaW9uIiwic2V0SURFVG9JbnN0YWxsRXh0ZW5zaW9uIiwiaWRlSW5zdGFsbGF0aW9uU3RhdHVzIiwic2V0SURFSW5zdGFsbGF0aW9uU3RhdHVzIiwic2hvd0lkZU9uYm9hcmRpbmciLCJzZXRTaG93SWRlT25ib2FyZGluZyIsInNob3dNb2RlbFN3aXRjaENhbGxvdXQiLCJzZXRTaG93TW9kZWxTd2l0Y2hDYWxsb3V0Iiwic2hvd0VmZm9ydENhbGxvdXQiLCJzZXRTaG93RWZmb3J0Q2FsbG91dCIsInNob3dSZW1vdGVDYWxsb3V0Iiwic2hvd0Rlc2t0b3BVcHNlbGxTdGFydHVwIiwic2V0U2hvd0Rlc2t0b3BVcHNlbGxTdGFydHVwIiwicmVjb21tZW5kYXRpb24iLCJsc3BSZWNvbW1lbmRhdGlvbiIsImhhbmRsZVJlc3BvbnNlIiwiaGFuZGxlTHNwUmVzcG9uc2UiLCJoaW50UmVjb21tZW5kYXRpb24iLCJoYW5kbGVIaW50UmVzcG9uc2UiLCJjb21iaW5lZEluaXRpYWxUb29scyIsImVuYWJsZWQiLCJ0YXNrc1YyIiwibW9kZSIsIm1lcmdlZFRvb2xzIiwidG9vbHMiLCJhbGxvd2VkQWdlbnRUeXBlcyIsInJlc29sdmVkIiwicmVzb2x2ZWRUb29scyIsImNvbW1hbmRzV2l0aFBsdWdpbnMiLCJtZXJnZWRDb21tYW5kcyIsInN0cmVhbU1vZGUiLCJzZXRTdHJlYW1Nb2RlIiwic3RyZWFtTW9kZVJlZiIsInN0cmVhbWluZ1Rvb2xVc2VzIiwic2V0U3RyZWFtaW5nVG9vbFVzZXMiLCJzdHJlYW1pbmdUaGlua2luZyIsInNldFN0cmVhbWluZ1RoaW5raW5nIiwiaXNTdHJlYW1pbmciLCJzdHJlYW1pbmdFbmRlZEF0IiwiZWxhcHNlZCIsIkRhdGUiLCJub3ciLCJyZW1haW5pbmciLCJ0aW1lciIsImNsZWFyVGltZW91dCIsImFib3J0Q29udHJvbGxlciIsInNldEFib3J0Q29udHJvbGxlciIsIkFib3J0Q29udHJvbGxlciIsImFib3J0Q29udHJvbGxlclJlZiIsInNlbmRCcmlkZ2VSZXN1bHRSZWYiLCJyZXN0b3JlTWVzc2FnZVN5bmNSZWYiLCJzY3JvbGxSZWYiLCJtb2RhbFNjcm9sbFJlZiIsImxhc3RVc2VyU2Nyb2xsVHNSZWYiLCJxdWVyeUd1YXJkIiwiaXNRdWVyeUFjdGl2ZSIsInN1YnNjcmliZSIsImdldFNuYXBzaG90IiwiaXNFeHRlcm5hbExvYWRpbmciLCJzZXRJc0V4dGVybmFsTG9hZGluZ1JhdyIsImhhc0luaXRpYWxQcm9tcHQiLCJpc0xvYWRpbmciLCJ1c2VySW5wdXRPblByb2Nlc3NpbmciLCJzZXRVc2VySW5wdXRPblByb2Nlc3NpbmdSYXciLCJ1c2VySW5wdXRCYXNlbGluZVJlZiIsInVzZXJNZXNzYWdlUGVuZGluZ1JlZiIsImxvYWRpbmdTdGFydFRpbWVSZWYiLCJ0b3RhbFBhdXNlZE1zUmVmIiwicGF1c2VTdGFydFRpbWVSZWYiLCJyZXNldFRpbWluZ1JlZnMiLCJ3YXNRdWVyeUFjdGl2ZVJlZiIsInNldElzRXh0ZXJuYWxMb2FkaW5nIiwidmFsdWUiLCJzd2FybVN0YXJ0VGltZVJlZiIsInN3YXJtQnVkZ2V0SW5mb1JlZiIsInRva2VucyIsImxpbWl0IiwibnVkZ2VzIiwiZm9jdXNlZElucHV0RGlhbG9nUmVmIiwiZ2V0Rm9jdXNlZElucHV0RGlhbG9nIiwiUFJPTVBUX1NVUFBSRVNTSU9OX01TIiwiaXNQcm9tcHRJbnB1dEFjdGl2ZSIsInNldElzUHJvbXB0SW5wdXRBY3RpdmUiLCJhdXRvVXBkYXRlclJlc3VsdCIsInNldEF1dG9VcGRhdGVyUmVzdWx0Iiwibm90aWZpY2F0aW9ucyIsImZvckVhY2giLCJub3RpZmljYXRpb24iLCJrZXkiLCJ0ZXh0IiwicHJpb3JpdHkiLCJoaW50Iiwic2hvd1VuZGVyY292ZXJDYWxsb3V0Iiwic2V0U2hvd1VuZGVyY292ZXJDYWxsb3V0IiwiaXNJbnRlcm5hbE1vZGVsUmVwbyIsInNob3VsZFNob3dVbmRlcmNvdmVyQXV0b05vdGljZSIsInRvb2xKU1giLCJzZXRUb29sSlNYSW50ZXJuYWwiLCJqc3giLCJzaG91bGRIaWRlUHJvbXB0SW5wdXQiLCJzaG91bGRDb250aW51ZUFuaW1hdGlvbiIsInNob3dTcGlubmVyIiwiaXNMb2NhbEpTWENvbW1hbmQiLCJpc0ltbWVkaWF0ZSIsImxvY2FsSlNYQ29tbWFuZFJlZiIsInNldFRvb2xKU1giLCJhcmdzIiwiY2xlYXJMb2NhbEpTWCIsInJlc3QiLCJ0b29sVXNlQ29uZmlybVF1ZXVlIiwic2V0VG9vbFVzZUNvbmZpcm1RdWV1ZSIsInBlcm1pc3Npb25TdGlja3lGb290ZXIiLCJzZXRQZXJtaXNzaW9uU3RpY2t5Rm9vdGVyIiwic2FuZGJveFBlcm1pc3Npb25SZXF1ZXN0UXVldWUiLCJzZXRTYW5kYm94UGVybWlzc2lvblJlcXVlc3RRdWV1ZSIsIkFycmF5IiwiaG9zdFBhdHRlcm4iLCJyZXNvbHZlUHJvbWlzZSIsImFsbG93Q29ubmVjdGlvbiIsInByb21wdFF1ZXVlIiwic2V0UHJvbXB0UXVldWUiLCJyZXF1ZXN0IiwidG9vbElucHV0U3VtbWFyeSIsInJlc29sdmUiLCJyZXNwb25zZSIsInJlamVjdCIsImVycm9yIiwiRXJyb3IiLCJzYW5kYm94QnJpZGdlQ2xlYW51cFJlZiIsIk1hcCIsInRlcm1pbmFsVGl0bGVGcm9tUmVuYW1lIiwic2V0dGluZ3MiLCJzZXNzaW9uVGl0bGUiLCJoYWlrdVRpdGxlIiwic2V0SGFpa3VUaXRsZSIsImhhaWt1VGl0bGVBdHRlbXB0ZWRSZWYiLCJhZ2VudFRpdGxlIiwiYWdlbnRUeXBlIiwidGVybWluYWxUaXRsZSIsImlzV2FpdGluZ0ZvckFwcHJvdmFsIiwiaXNTaG93aW5nTG9jYWxKU1hDb21tYW5kIiwidGl0bGVJc0FuaW1hdGluZyIsInNlc3Npb25TdGF0dXMiLCJ3YWl0aW5nRm9yIiwidG9vbCIsInRhYlN0YXR1c0dhdGVFbmFibGVkIiwic2hvd1N0YXR1c0luVGVybWluYWxUYWIiLCJyYXdTZXRNZXNzYWdlcyIsIm1lc3NhZ2VzUmVmIiwiaWRsZUhpbnRTaG93blJlZiIsInNldE1lc3NhZ2VzIiwiYWN0aW9uIiwiU2V0U3RhdGVBY3Rpb24iLCJuZXh0IiwiZGVsdGEiLCJhZGRlZCIsInNvbWUiLCJzZXRVc2VySW5wdXRPblByb2Nlc3NpbmciLCJkaXZpZGVySW5kZXgiLCJkaXZpZGVyWVJlZiIsIm9uU2Nyb2xsQXdheSIsIm9uUmVwaW4iLCJqdW1wVG9OZXciLCJzaGlmdERpdmlkZXIiLCJjdXJzb3IiLCJzZXRDdXJzb3IiLCJjdXJzb3JOYXZSZWYiLCJ1bnNlZW5EaXZpZGVyIiwicmVwaW5TY3JvbGwiLCJzY3JvbGxUb0JvdHRvbSIsImxhc3RNc2ciLCJhdCIsImxhc3RNc2dJc0h1bWFuIiwib25QcmVwZW5kIiwiY29tcG9zZWRPblNjcm9sbCIsInN0aWNreSIsImhhbmRsZSIsImNvbXBhbmlvblJlYWN0aW9uIiwiYXdhaXRQZW5kaW5nSG9va3MiLCJkZWZlcnJlZE1lc3NhZ2VzIiwiZGVmZXJyZWRCZWhpbmQiLCJmcm96ZW5UcmFuc2NyaXB0U3RhdGUiLCJzZXRGcm96ZW5UcmFuc2NyaXB0U3RhdGUiLCJtZXNzYWdlc0xlbmd0aCIsInN0cmVhbWluZ1Rvb2xVc2VzTGVuZ3RoIiwiaW5wdXRWYWx1ZSIsInNldElucHV0VmFsdWVSYXciLCJpbnB1dFZhbHVlUmVmIiwiaW5zZXJ0VGV4dFJlZiIsImluc2VydCIsInNldElucHV0V2l0aEN1cnNvciIsInNldElucHV0VmFsdWUiLCJ0cmltIiwiaW5wdXRNb2RlIiwic2V0SW5wdXRNb2RlIiwic3Rhc2hlZFByb21wdCIsInNldFN0YXNoZWRQcm9tcHQiLCJwYXN0ZWRDb250ZW50cyIsImhhbmRsZVJlbW90ZUluaXQiLCJyZW1vdGVTbGFzaENvbW1hbmRzIiwicmVtb3RlQ29tbWFuZFNldCIsImNtZCIsImluUHJvZ3Jlc3NUb29sVXNlSURzIiwic2V0SW5Qcm9ncmVzc1Rvb2xVc2VJRHMiLCJoYXNJbnRlcnJ1cHRpYmxlVG9vbEluUHJvZ3Jlc3NSZWYiLCJyZW1vdGVTZXNzaW9uIiwic2V0SXNMb2FkaW5nIiwib25Jbml0IiwiZGlyZWN0Q29ubmVjdCIsInNzaFJlbW90ZSIsInNlc3Npb24iLCJhY3RpdmVSZW1vdGUiLCJpc1JlbW90ZU1vZGUiLCJzZXRQYXN0ZWRDb250ZW50cyIsInN1Ym1pdENvdW50Iiwic2V0U3VibWl0Q291bnQiLCJyZXNwb25zZUxlbmd0aFJlZiIsImFwaU1ldHJpY3NSZWYiLCJ0dGZ0TXMiLCJmaXJzdFRva2VuVGltZSIsImxhc3RUb2tlblRpbWUiLCJyZXNwb25zZUxlbmd0aEJhc2VsaW5lIiwiZW5kUmVzcG9uc2VMZW5ndGgiLCJzZXRSZXNwb25zZUxlbmd0aCIsImVudHJpZXMiLCJsYXN0RW50cnkiLCJzdHJlYW1pbmdUZXh0Iiwic2V0U3RyZWFtaW5nVGV4dCIsInJlZHVjZWRNb3Rpb24iLCJwcmVmZXJzUmVkdWNlZE1vdGlvbiIsInNob3dTdHJlYW1pbmdUZXh0Iiwib25TdHJlYW1pbmdUZXh0IiwidmlzaWJsZVN0cmVhbWluZ1RleHQiLCJzdWJzdHJpbmciLCJsYXN0SW5kZXhPZiIsImxhc3RRdWVyeUNvbXBsZXRpb25UaW1lIiwic2V0TGFzdFF1ZXJ5Q29tcGxldGlvblRpbWUiLCJzcGlubmVyTWVzc2FnZSIsInNldFNwaW5uZXJNZXNzYWdlIiwic3Bpbm5lckNvbG9yIiwic2V0U3Bpbm5lckNvbG9yIiwic3Bpbm5lclNoaW1tZXJDb2xvciIsInNldFNwaW5uZXJTaGltbWVyQ29sb3IiLCJpc01lc3NhZ2VTZWxlY3RvclZpc2libGUiLCJzZXRJc01lc3NhZ2VTZWxlY3RvclZpc2libGUiLCJtZXNzYWdlU2VsZWN0b3JQcmVzZWxlY3QiLCJzZXRNZXNzYWdlU2VsZWN0b3JQcmVzZWxlY3QiLCJzaG93Q29zdERpYWxvZyIsInNldFNob3dDb3N0RGlhbG9nIiwiY29udmVyc2F0aW9uSWQiLCJzZXRDb252ZXJzYXRpb25JZCIsImlkbGVSZXR1cm5QZW5kaW5nIiwic2V0SWRsZVJldHVyblBlbmRpbmciLCJpZGxlTWludXRlcyIsInNraXBJZGxlQ2hlY2tSZWYiLCJsYXN0UXVlcnlDb21wbGV0aW9uVGltZVJlZiIsImNvbnRlbnRSZXBsYWNlbWVudFN0YXRlUmVmIiwiaGF2ZVNob3duQ29zdERpYWxvZyIsInNldEhhdmVTaG93bkNvc3REaWFsb2ciLCJoYXNBY2tub3dsZWRnZWRDb3N0VGhyZXNob2xkIiwidmltTW9kZSIsInNldFZpbU1vZGUiLCJzaG93QmFzaGVzRGlhbG9nIiwic2V0U2hvd0Jhc2hlc0RpYWxvZyIsImlzU2VhcmNoaW5nSGlzdG9yeSIsInNldElzU2VhcmNoaW5nSGlzdG9yeSIsImlzSGVscE9wZW4iLCJzZXRJc0hlbHBPcGVuIiwiaXNUZXJtaW5hbEZvY3VzZWQiLCJ0ZXJtaW5hbEZvY3VzUmVmIiwidGhlbWUiLCJ0aXBQaWNrZWRUaGlzVHVyblJlZiIsInBpY2tOZXdTcGlubmVyVGlwIiwiYmFzaFRvb2xzUHJvY2Vzc2VkSWR4IiwiYmFzaFRvb2xzIiwiYWRkIiwicmVhZEZpbGVTdGF0ZSIsInRpcCIsImNvbnRlbnQiLCJyZXNldExvYWRpbmdTdGF0ZSIsImhhc1J1bm5pbmdUZWFtbWF0ZXMiLCJ0b3RhbE1zIiwiZGVmZXJyZWRCdWRnZXQiLCJzYWZlWW9sb01lc3NhZ2VTaG93blJlZiIsImF1dG9QZXJtaXNzaW9uc05vdGlmaWNhdGlvbkNvdW50IiwicmVmIiwicHJldkNvdW50Iiwid29ya3RyZWVUaXBTaG93blJlZiIsInd0IiwiY3JlYXRpb25EdXJhdGlvbk1zIiwidXNlZFNwYXJzZVBhdGhzIiwic2VjcyIsIm9ubHlTbGVlcFRvb2xBY3RpdmUiLCJsYXN0QXNzaXN0YW50IiwiZmluZExhc3QiLCJ0eXBlIiwiaW5Qcm9ncmVzc1Rvb2xVc2VzIiwibWVzc2FnZSIsImlkIiwiZXZlcnkiLCJtck9uQmVmb3JlUXVlcnkiLCJtck9uVHVybkNvbXBsZXRlIiwicmVuZGVyIiwibXJSZW5kZXIiLCJoYXNBY3RpdmVQcm9tcHQiLCJxdWV1ZSIsImZlZWRiYWNrU3VydmV5T3JpZ2luYWwiLCJza2lsbEltcHJvdmVtZW50U3VydmV5Iiwic2hvd0lzc3VlRmxhZ0Jhbm5lciIsImZlZWRiYWNrU3VydmV5IiwiaGFuZGxlU2VsZWN0Iiwic2VsZWN0ZWQiLCJkaWRBdXRvUnVuSXNzdWVSZWYiLCJzaG93ZWRUcmFuc2NyaXB0UHJvbXB0Iiwic2V0QXV0b1J1bklzc3VlUmVhc29uIiwicG9zdENvbXBhY3RTdXJ2ZXkiLCJtZW1vcnlTdXJ2ZXkiLCJmcnVzdHJhdGlvbkRldGVjdGlvbiIsInNldElERUluc3RhbGxhdGlvblN0YXRlIiwiZmlsZUhpc3RvcnlTdGF0ZSIsInJlc3VtZSIsInNlc3Npb25JZCIsImxvZyIsImVudHJ5cG9pbnQiLCJyZXN1bWVTdGFydCIsInBlcmZvcm1hbmNlIiwiY29vcmRpbmF0b3JNb2R1bGUiLCJ3YXJuaW5nIiwibWF0Y2hTZXNzaW9uTW9kZSIsImdldEFnZW50RGVmaW5pdGlvbnNXaXRoT3ZlcnJpZGVzIiwiZ2V0QWN0aXZlQWdlbnRzRnJvbUxpc3QiLCJjYWNoZSIsImNsZWFyIiwiZnJlc2hBZ2VudERlZnMiLCJhbGxBZ2VudHMiLCJhY3RpdmVBZ2VudHMiLCJwdXNoIiwic2Vzc2lvbkVuZFRpbWVvdXRNcyIsImdldEFwcFN0YXRlIiwiZ2V0U3RhdGUiLCJzaWduYWwiLCJBYm9ydFNpZ25hbCIsInRpbWVvdXQiLCJ0aW1lb3V0TXMiLCJob29rTWVzc2FnZXMiLCJtb2RlbCIsImZpbGVIaXN0b3J5U25hcHNob3RzIiwiYWdlbnREZWZpbml0aW9uIiwicmVzdG9yZWRBZ2VudCIsImFnZW50U2V0dGluZyIsImFnZW50Iiwic3RhbmRhbG9uZUFnZW50Q29udGV4dCIsImFnZW50TmFtZSIsImFnZW50Q29sb3IiLCJyZXN0b3JlUmVhZEZpbGVTdGF0ZSIsInByb2plY3RQYXRoIiwidGFyZ2V0U2Vzc2lvbkNvc3RzIiwiZnVsbFBhdGgiLCJyZW5hbWVSZWNvcmRpbmdGb3JTZXNzaW9uIiwid29ya3RyZWVTZXNzaW9uIiwid3MiLCJzYXZlTW9kZSIsImlzQ29vcmRpbmF0b3JNb2RlIiwiY29udGVudFJlcGxhY2VtZW50cyIsInN1Y2Nlc3MiLCJyZXN1bWVfZHVyYXRpb25fbXMiLCJpbml0aWFsUmVhZEZpbGVTdGF0ZSIsImRpc2NvdmVyZWRTa2lsbE5hbWVzUmVmIiwibG9hZGVkTmVzdGVkTWVtb3J5UGF0aHNSZWYiLCJjd2QiLCJleHRyYWN0ZWQiLCJhcGlLZXlTdGF0dXMiLCJyZXZlcmlmeSIsImF1dG9SdW5Jc3N1ZVJlYXNvbiIsImV4aXRGbG93Iiwic2V0RXhpdEZsb3ciLCJpc0V4aXRpbmciLCJzZXRJc0V4aXRpbmciLCJzaG93aW5nQ29zdERpYWxvZyIsImFsbG93RGlhbG9nc1dpdGhBbmltYXRpb24iLCJmb2N1c2VkSW5wdXREaWFsb2ciLCJoYXNTdXBwcmVzc2VkRGlhbG9ncyIsImlzUGF1c2VkIiwicHJldkRpYWxvZ1JlZiIsIndhcyIsInBhdXNlUHJvYWN0aXZlIiwiZm9yY2VFbmQiLCJvbkFib3J0IiwiaXRlbSIsImFib3J0IiwiY2FuY2VsUmVxdWVzdCIsImhhbmRsZVF1ZXVlZENvbW1hbmRPbkNhbmNlbCIsImltYWdlcyIsIm5ld0NvbnRlbnRzIiwiaW1hZ2UiLCJjYW5jZWxSZXF1ZXN0UHJvcHMiLCJvbkFnZW50c0tpbGxlZCIsImFib3J0U2lnbmFsIiwicG9wQ29tbWFuZEZyb21RdWV1ZSIsInRvdGFsQ29zdCIsInNhbmRib3hBc2tDYWxsYmFjayIsInJlcXVlc3RJZCIsInNlbnQiLCJob3N0IiwicmVzb2x2ZVNob3VsZEFsbG93SG9zdCIsInJlc29sdmVPbmNlIiwiYWxsb3ciLCJicmlkZ2VDYWxsYmFja3MiLCJyZXBsQnJpZGdlUGVybWlzc2lvbkNhbGxiYWNrcyIsImJyaWRnZVJlcXVlc3RJZCIsInNlbmRSZXF1ZXN0IiwidW5zdWJzY3JpYmUiLCJvblJlc3BvbnNlIiwiYmVoYXZpb3IiLCJzaWJsaW5nQ2xlYW51cHMiLCJnZXQiLCJmbiIsImRlbGV0ZSIsImNsZWFudXAiLCJleGlzdGluZyIsInNldCIsInJlYXNvbiIsImdldFNhbmRib3hVbmF2YWlsYWJsZVJlYXNvbiIsImlzU2FuZGJveFJlcXVpcmVkIiwic3RkZXJyIiwid3JpdGUiLCJsZXZlbCIsImlzU2FuZGJveGluZ0VuYWJsZWQiLCJpbml0aWFsaXplIiwiY2F0Y2giLCJlcnIiLCJzZXRUb29sUGVybWlzc2lvbkNvbnRleHQiLCJjb250ZXh0Iiwib3B0aW9ucyIsInByZXNlcnZlTW9kZSIsInNldEltbWVkaWF0ZSIsImN1cnJlbnRRdWV1ZSIsInJlY2hlY2tQZXJtaXNzaW9uIiwiY2FuVXNlVG9vbCIsInJlcXVlc3RQcm9tcHQiLCJnZXRUb29sVXNlQ29udGV4dCIsImNvbXB1dGVUb29scyIsImFzc2VtYmxlZCIsIm1lcmdlZCIsInRoaW5raW5nRW5hYmxlZCIsIm1jcFJlc291cmNlcyIsInJlc291cmNlcyIsImlzTm9uSW50ZXJhY3RpdmVTZXNzaW9uIiwicmVmcmVzaFRvb2xzIiwidXBkYXRlRmlsZUhpc3RvcnlTdGF0ZSIsInVwZGF0ZXIiLCJ1cGRhdGVkIiwidXBkYXRlQXR0cmlidXRpb25TdGF0ZSIsImF0dHJpYnV0aW9uIiwib3Blbk1lc3NhZ2VTZWxlY3RvciIsIm9uQ2hhbmdlQVBJS2V5IiwiYXBwZW5kU3lzdGVtTWVzc2FnZSIsIm1zZyIsInNlbmRPU05vdGlmaWNhdGlvbiIsIm9wdHMiLCJvbkluc3RhbGxJREVFeHRlbnNpb24iLCJuZXN0ZWRNZW1vcnlBdHRhY2htZW50VHJpZ2dlcnMiLCJsb2FkZWROZXN0ZWRNZW1vcnlQYXRocyIsImR5bmFtaWNTa2lsbERpclRyaWdnZXJzIiwiZGlzY292ZXJlZFNraWxsTmFtZXMiLCJwdXNoQXBpTWV0cmljc0VudHJ5IiwiYmFzZWxpbmUiLCJvbkNvbXBhY3RQcm9ncmVzcyIsImV2ZW50IiwiaG9va1R5cGUiLCJzZXRIYXNJbnRlcnJ1cHRpYmxlVG9vbEluUHJvZ3Jlc3MiLCJ2IiwiY29udGVudFJlcGxhY2VtZW50U3RhdGUiLCJoYW5kbGVCYWNrZ3JvdW5kUXVlcnkiLCJyZW1vdmVkTm90aWZpY2F0aW9ucyIsInRvb2xVc2VDb250ZXh0IiwiZGVmYXVsdFN5c3RlbVByb21wdCIsInVzZXJDb250ZXh0Iiwic3lzdGVtQ29udGV4dCIsImFsbCIsImZyb20iLCJhZGRpdGlvbmFsV29ya2luZ0RpcmVjdG9yaWVzIiwia2V5cyIsInJlbmRlcmVkU3lzdGVtUHJvbXB0Iiwibm90aWZpY2F0aW9uQXR0YWNobWVudHMiLCJub3RpZmljYXRpb25NZXNzYWdlcyIsImV4aXN0aW5nUHJvbXB0cyIsImF0dGFjaG1lbnQiLCJjb21tYW5kTW9kZSIsInByb21wdCIsInVuaXF1ZU5vdGlmaWNhdGlvbnMiLCJxdWVyeVBhcmFtcyIsInF1ZXJ5U291cmNlIiwiZGVzY3JpcHRpb24iLCJoYW5kbGVCYWNrZ3JvdW5kU2Vzc2lvbiIsIm9uQmFja2dyb3VuZFF1ZXJ5Iiwib25RdWVyeUV2ZW50IiwiUGFyYW1ldGVycyIsIm5ld01lc3NhZ2UiLCJvbGQiLCJpbmNsdWRlU25pcHBlZCIsInNldENvbnRleHRCbG9ja2VkIiwiZGF0YSIsIm9sZE1lc3NhZ2VzIiwibGFzdCIsInBhcmVudFRvb2xVc2VJRCIsImNvcHkiLCJpc0FwaUVycm9yTWVzc2FnZSIsIm5ld0NvbnRlbnQiLCJ0b21ic3RvbmVkTWVzc2FnZSIsIm1ldHJpY3MiLCJvblF1ZXJ5SW1wbCIsIm1lc3NhZ2VzSW5jbHVkaW5nTmV3TWVzc2FnZXMiLCJzaG91bGRRdWVyeSIsImFkZGl0aW9uYWxBbGxvd2VkVG9vbHMiLCJtYWluTG9vcE1vZGVsUGFyYW0iLCJlZmZvcnQiLCJmcmVzaENsaWVudHMiLCJoYW5kbGVRdWVyeVN0YXJ0IiwiaWRlQ2xpZW50IiwiZmlyc3RVc2VyTWVzc2FnZSIsImZpbmQiLCJpc01ldGEiLCJzdGFydHNXaXRoIiwic2V0U3RhdGUiLCJjdXIiLCJhbHdheXNBbGxvd1J1bGVzIiwiY29tbWFuZCIsImkiLCJmcmVzaFRvb2xzIiwiZnJlc2hNY3BDbGllbnRzIiwicHJldmlvdXNHZXRBcHBTdGF0ZSIsImVmZm9ydFZhbHVlIiwiYmFzZVVzZXJDb250ZXh0IiwiZmFzdE1vZGUiLCJ0ZXJtaW5hbEZvY3VzIiwiZmlyZUNvbXBhbmlvbk9ic2VydmVyIiwicmVhY3Rpb24iLCJ0dGZ0cyIsImUiLCJvdHBzVmFsdWVzIiwic2FtcGxpbmdNcyIsImlzTXVsdGlSZXF1ZXN0IiwiaG9va01zIiwiaG9va0NvdW50IiwidG9vbE1zIiwidG9vbENvdW50IiwiY2xhc3NpZmllck1zIiwiY2xhc3NpZmllckNvdW50IiwidHVybk1zIiwib3RwcyIsImlzUDUwIiwiaG9va0R1cmF0aW9uTXMiLCJ0dXJuRHVyYXRpb25NcyIsInRvb2xEdXJhdGlvbk1zIiwiY2xhc3NpZmllckR1cmF0aW9uTXMiLCJjb25maWdXcml0ZUNvdW50Iiwib25RdWVyeSIsIm9uQmVmb3JlUXVlcnlDYWxsYmFjayIsInRlYW1OYW1lIiwidGhpc0dlbmVyYXRpb24iLCJ0cnlTdGFydCIsInBhcnNlZEJ1ZGdldCIsImxhdGVzdE1lc3NhZ2VzIiwic2hvdWxkUHJvY2VlZCIsImVuZCIsImFib3J0ZWQiLCJ0dW5nc3RlbkFjdGl2ZVNlc3Npb24iLCJ0dW5nc3RlblBhbmVsQXV0b0hpZGRlbiIsImJ1ZGdldEluZm8iLCJoYXNSdW5uaW5nU3dhcm1BZ2VudHMiLCJtc2dzIiwibGFzdFVzZXJNc2ciLCJpZHgiLCJpbml0aWFsTWVzc2FnZVJlZiIsInBlbmRpbmciLCJwcm9jZXNzSW5pdGlhbE1lc3NhZ2UiLCJpbml0aWFsTXNnIiwiTm9uTnVsbGFibGUiLCJjbGVhckNvbnRleHQiLCJvbGRQbGFuU2x1ZyIsInBsYW5Db250ZW50IiwiY2xlYXJDb252ZXJzYXRpb24iLCJzaG91bGRTdG9yZVBsYW5Gb3JWZXJpZmljYXRpb24iLCJ1cGRhdGVkVG9vbFBlcm1pc3Npb25Db250ZXh0IiwiYWxsb3dlZFByb21wdHMiLCJwcmVQbGFuTW9kZSIsInBlbmRpbmdQbGFuVmVyaWZpY2F0aW9uIiwicGxhbiIsInZlcmlmaWNhdGlvblN0YXJ0ZWQiLCJ2ZXJpZmljYXRpb25Db21wbGV0ZWQiLCJvblN1Ym1pdCIsInNldEN1cnNvck9mZnNldCIsImNsZWFyQnVmZmVyIiwicmVzZXRIaXN0b3J5IiwibmV3QWJvcnRDb250cm9sbGVyIiwiaGVscGVycyIsInNwZWN1bGF0aW9uQWNjZXB0Iiwic3BlY3VsYXRpb25TZXNzaW9uVGltZVNhdmVkTXMiLCJmcm9tS2V5YmluZGluZyIsInJlc3VtZVByb2FjdGl2ZSIsInRyaW1tZWRJbnB1dCIsInNwYWNlSW5kZXgiLCJpbmRleE9mIiwiY29tbWFuZE5hbWUiLCJjb21tYW5kQXJncyIsIm1hdGNoaW5nQ29tbWFuZCIsImFsaWFzZXMiLCJpbmNsdWRlcyIsInZhcmlhbnQiLCJtZXNzYWdlQ291bnQiLCJ0b3RhbElucHV0VG9rZW5zIiwic2hvdWxkVHJlYXRBc0ltbWVkaWF0ZSIsImltbWVkaWF0ZSIsInBhc3RlZFRleHRSZWZzIiwiciIsInBhc3RlZFRleHRDb3VudCIsInBhc3RlZFRleHRCeXRlcyIsInJlZHVjZSIsInN1bSIsImV4ZWN1dGVJbW1lZGlhdGVDb21tYW5kIiwiZG9uZVdhc0NhbGxlZCIsIm9uRG9uZSIsImRvbmVPcHRpb25zIiwiZGlzcGxheSIsIm1ldGFNZXNzYWdlcyIsIm1vZCIsImxvYWQiLCJjYWxsIiwid2lsbG93TW9kZSIsImlkbGVUaHJlc2hvbGRNaW4iLCJOdW1iZXIiLCJDTEFVREVfQ09ERV9JRExFX1RIUkVTSE9MRF9NSU5VVEVTIiwidG9rZW5UaHJlc2hvbGQiLCJDTEFVREVfQ09ERV9JRExFX1RPS0VOX1RIUkVTSE9MRCIsImlkbGVSZXR1cm5EaXNtaXNzZWQiLCJpZGxlTXMiLCJpc1NsYXNoQ29tbWFuZCIsInN1Ym1pdHNOb3ciLCJzbmFwc2hvdCIsInF1ZXJ5UmVxdWlyZWQiLCJjIiwic3BsaXQiLCJwYXN0ZWRWYWx1ZXMiLCJPYmplY3QiLCJpbWFnZUNvbnRlbnRzIiwiaW1hZ2VQYXN0ZUlkcyIsIm1lc3NhZ2VDb250ZW50IiwicmVtb3RlQ29udGVudCIsImNvbnRlbnRCbG9ja3MiLCJyZW1vdGVCbG9ja3MiLCJwYXN0ZWQiLCJzb3VyY2UiLCJjb25zdCIsIm1lZGlhX3R5cGUiLCJtZWRpYVR5cGUiLCJ1c2VyTWVzc2FnZSIsInNlbmRNZXNzYWdlIiwib25JbnB1dENoYW5nZSIsImhhc0ludGVycnVwdGlibGVUb29sSW5Qcm9ncmVzcyIsIm9uQWdlbnRTdWJtaXQiLCJ0YXNrIiwiYWdlbnRJZCIsImhhbmRsZUF1dG9SdW5Jc3N1ZSIsImhhbmRsZUNhbmNlbEF1dG9SdW5Jc3N1ZSIsImhhbmRsZVN1cnZleVJlcXVlc3RGZWVkYmFjayIsIlN0cmluZyIsIm9uU3VibWl0UmVmIiwiaGFuZGxlT3BlblJhdGVMaW1pdE9wdGlvbnMiLCJoYW5kbGVFeGl0Iiwic3RkaW8iLCJzaG93V29ya3RyZWUiLCJleGl0TW9kIiwiZXhpdEZsb3dSZXN1bHQiLCJoYW5kbGVTaG93TWVzc2FnZVNlbGVjdG9yIiwicmV3aW5kQ29udmVyc2F0aW9uVG8iLCJtZXNzYWdlSW5kZXgiLCJwcmVSZXdpbmRNZXNzYWdlQ291bnQiLCJwb3N0UmV3aW5kTWVzc2FnZUNvdW50IiwibWVzc2FnZXNSZW1vdmVkIiwicmV3aW5kVG9NZXNzYWdlSW5kZXgiLCJyZXNldENvbnRleHRDb2xsYXBzZSIsInBlcm1pc3Npb25Nb2RlIiwicHJvbXB0U3VnZ2VzdGlvbiIsInByb21wdElkIiwic2hvd25BdCIsImFjY2VwdGVkQXQiLCJnZW5lcmF0aW9uUmVxdWVzdElkIiwicmVzdG9yZU1lc3NhZ2VTeW5jIiwiaXNBcnJheSIsImJsb2NrIiwiaW1hZ2VCbG9ja3MiLCJuZXdQYXN0ZWRDb250ZW50cyIsImluZGV4IiwiaGFuZGxlUmVzdG9yZU1lc3NhZ2UiLCJyZXN0b3JlIiwiZmluZFJhd0luZGV4IiwiZmluZEluZGV4IiwibWVzc2FnZUFjdGlvbkNhcHMiLCJyYXciLCJzdGRvdXQiLCJjb2xvciIsImVkaXQiLCJyYXdJZHgiLCJub0ZpbGVDaGFuZ2VzIiwib25seVN5bnRoZXRpYyIsImVudGVyIiwiZW50ZXJNZXNzYWdlQWN0aW9ucyIsImhhbmRsZXJzIiwibWVzc2FnZUFjdGlvbkhhbmRsZXJzIiwibWVtb3J5RmlsZXMiLCJmaWxlTGlzdCIsInBhdGgiLCJwYXJlbnQiLCJmaWxlIiwiY29udGVudERpZmZlcnNGcm9tRGlzayIsInJhd0NvbnRlbnQiLCJ0aW1lc3RhbXAiLCJvZmZzZXQiLCJpc1BhcnRpYWxWaWV3Iiwic2VuZEJyaWRnZVJlc3VsdCIsImhhc0NvdW50ZWRRdWV1ZVVzZVJlZiIsInByb21wdFF1ZXVlVXNlQ291bnQiLCJleGVjdXRlUXVldWVkSW5wdXQiLCJoYXNBY3RpdmVMb2NhbEpzeFVJIiwicmVjb3JkVXNlckFjdGl2aXR5IiwibGFzdFVzZXJJbnRlcmFjdGlvbiIsImlkbGVUaW1lU2luY2VSZXNwb25zZSIsIm1lc3NhZ2VJZGxlTm90aWZUaHJlc2hvbGRNcyIsIm5vdGlmaWNhdGlvblR5cGUiLCJpZGxlVGhyZXNob2xkTXMiLCJscWN0IiwiYWRkTm90aWYiLCJtc2dzUmVmIiwiaGludFJlZiIsInRvdGFsVG9rZW5zIiwiZm9ybWF0dGVkVG9rZW5zIiwibWF4IiwiaGFuZGxlSW5jb21pbmdQcm9tcHQiLCJ2b2ljZSIsImludGVyaW1SYW5nZSIsIm9uU3VibWl0TWVzc2FnZSIsImFzc2lzdGFudE1vZGUiLCJrYWlyb3NFbmFibGVkIiwib25TdWJtaXRUYXNrIiwicXVldWVkQ29tbWFuZHNMZW5ndGgiLCJpc0luUGxhbk1vZGUiLCJvblN1Ym1pdFRpY2siLCJvblF1ZXVlVGljayIsInNodXRkb3duIiwiaW50ZXJuYWxfZXZlbnRFbWl0dGVyIiwicmVtb3VudEtleSIsInNldFJlbW91bnRLZXkiLCJoYW5kbGVTdXNwZW5kIiwiaGFuZGxlUmVzdW1lIiwib24iLCJzdG9wSG9va1NwaW5uZXJTdWZmaXgiLCJwcm9ncmVzc01zZ3MiLCJob29rRXZlbnQiLCJjdXJyZW50VG9vbFVzZUlEIiwidG9vbFVzZUlEIiwiaGFzU3VtbWFyeUZvckN1cnJlbnRFeGVjdXRpb24iLCJzdWJ0eXBlIiwiY3VycmVudEhvb2tzIiwicCIsInRvdGFsIiwiY29tcGxldGVkQ291bnQiLCJjdXN0b21NZXNzYWdlIiwic3RhdHVzTWVzc2FnZSIsImxhYmVsIiwiaGFuZGxlRW50ZXJUcmFuc2NyaXB0IiwiaGFuZGxlRXhpdFRyYW5zY3JpcHQiLCJ2aXJ0dWFsU2Nyb2xsQWN0aXZlIiwic2VhcmNoT3BlbiIsInNldFNlYXJjaE9wZW4iLCJzZWFyY2hRdWVyeSIsInNlYXJjaENvdW50Iiwic2V0U2VhcmNoQ291bnQiLCJzZWFyY2hDdXJyZW50Iiwic2V0U2VhcmNoQ3VycmVudCIsIm9uU2VhcmNoTWF0Y2hlc0NoYW5nZSIsImN0cmwiLCJtZXRhIiwic2V0QW5jaG9yIiwic3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uIiwicmVwZWF0IiwibmV4dE1hdGNoIiwicHJldk1hdGNoIiwic2V0UXVlcnkiLCJzY2FuRWxlbWVudCIsInNldFBvc2l0aW9ucyIsInRyYW5zY3JpcHRDb2xzIiwiY29sdW1ucyIsInByZXZDb2xzUmVmIiwiZGlzYXJtU2VhcmNoIiwiZ2VuIiwic2V0U3RhdHVzIiwidyIsInJlcGxhY2UiLCJvcGVuZWQiLCJpblRyYW5zY3JpcHQiLCJnbG9iYWxLZXliaW5kaW5nUHJvcHMiLCJvbkVudGVyVHJhbnNjcmlwdCIsIm9uRXhpdFRyYW5zY3JpcHQiLCJzZWFyY2hCYXJPcGVuIiwidHJhbnNjcmlwdE1lc3NhZ2VzIiwidHJhbnNjcmlwdFN0cmVhbWluZ1Rvb2xVc2VzIiwib25PcGVuQmFja2dyb3VuZFRhc2tzIiwidHJhbnNjcmlwdFNjcm9sbFJlZiIsInRyYW5zY3JpcHRNZXNzYWdlc0VsZW1lbnQiLCJ0cmFuc2NyaXB0VG9vbEpTWCIsInRyYW5zY3JpcHRSZXR1cm4iLCJxIiwidmlld2VkVGFzayIsInZpZXdlZFRlYW1tYXRlVGFzayIsInZpZXdlZEFnZW50VGFzayIsInVzZXNTeW5jTWVzc2FnZXMiLCJkaXNwbGF5ZWRNZXNzYWdlcyIsInBsYWNlaG9sZGVyVGV4dCIsInRvb2xQZXJtaXNzaW9uT3ZlcmxheSIsInRhaWwiLCJ3b3JrZXJCYWRnZSIsImNvbXBhbmlvbk5hcnJvdyIsImNvbXBhbmlvblZpc2libGUiLCJ0b29sSnN4Q2VudGVyZWQiLCJjZW50ZXJlZE1vZGFsIiwibWFpblJldHVybiIsInNpemUiLCJwZXJzaXN0VG9TZXR0aW5ncyIsImN1cnJlbnRSZXF1ZXN0IiwiYXBwcm92ZWRIb3N0IiwidXBkYXRlIiwicnVsZXMiLCJ0b29sTmFtZSIsInJ1bGVDb250ZW50IiwiZGVzdGluYXRpb24iLCJyZWZyZXNoQ29uZmlnIiwiY2xlYW51cHMiLCJzZWxlY3RlZEtleSIsInByb21wdF9yZXNwb25zZSIsInBvcnQiLCJ3b3JrZXJOYW1lIiwic2VydmVyTmFtZSIsInJlc3BvbmQiLCJpc1VybEFjY2VwdCIsInBhcmFtcyIsIm9uV2FpdGluZ0Rpc21pc3MiLCJzZWxlY3Rpb24iLCJtb2RlbEFsaWFzIiwibWFpbkxvb3BNb2RlbEZvclNlc3Npb24iLCJyZXBsQnJpZGdlRW5hYmxlZCIsInJlcGxCcmlkZ2VFeHBsaWNpdCIsInJlcGxCcmlkZ2VPdXRib3VuZE9ubHkiLCJwbHVnaW5OYW1lIiwicGx1Z2luRGVzY3JpcHRpb24iLCJtYXJrZXRwbGFjZU5hbWUiLCJzb3VyY2VDb21tYW5kIiwiZmlsZUV4dGVuc2lvbiIsImNob2ljZSIsImJsdXJiIiwiYXBwZW5kU3Rkb3V0IiwiYXBwZW5kV2hlbklkbGUiLCJ1bnN1YiIsInVsdHJhcGxhblNlc3Npb25VcmwiLCJsYXVuY2hVbHRyYXBsYW4iLCJkaXNjb25uZWN0ZWRCcmlkZ2UiLCJvblNlc3Npb25SZWFkeSIsImxhc3RSZXNwb25zZSIsInN1Z2dlc3Rpb24iLCJpc09wZW4iLCJza2lsbE5hbWUiLCJ1cGRhdGVzIiwiZmVlZGJhY2siLCJkaXJlY3Rpb24iLCJjb21wYWN0TWVzc2FnZXMiLCJhcHBTdGF0ZSIsImRlZmF1bHRTeXNQcm9tcHQiLCJmb3JrQ29udGV4dE1lc3NhZ2VzIiwia2VwdCIsIm1lc3NhZ2VzVG9LZWVwIiwib3JkZXJlZCIsInN1bW1hcnlNZXNzYWdlcyIsInBvc3RDb21wYWN0IiwiYm91bmRhcnlNYXJrZXIiLCJhdHRhY2htZW50cyIsImhvb2tSZXN1bHRzIiwiaGlzdG9yeVNob3J0Y3V0Il0sInNvdXJjZXMiOlsiUkVQTC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiLy8gYmlvbWUtaWdub3JlLWFsbCBhc3Npc3Qvc291cmNlL29yZ2FuaXplSW1wb3J0czogQU5ULU9OTFkgaW1wb3J0IG1hcmtlcnMgbXVzdCBub3QgYmUgcmVvcmRlcmVkXG5pbXBvcnQgeyBmZWF0dXJlIH0gZnJvbSAnYnVuOmJ1bmRsZSdcbmltcG9ydCB7IHNwYXduU3luYyB9IGZyb20gJ2NoaWxkX3Byb2Nlc3MnXG5pbXBvcnQge1xuICBzbmFwc2hvdE91dHB1dFRva2Vuc0ZvclR1cm4sXG4gIGdldEN1cnJlbnRUdXJuVG9rZW5CdWRnZXQsXG4gIGdldFR1cm5PdXRwdXRUb2tlbnMsXG4gIGdldEJ1ZGdldENvbnRpbnVhdGlvbkNvdW50LFxuICBnZXRUb3RhbElucHV0VG9rZW5zLFxufSBmcm9tICcuLi9ib290c3RyYXAvc3RhdGUuanMnXG5pbXBvcnQgeyBwYXJzZVRva2VuQnVkZ2V0IH0gZnJvbSAnLi4vdXRpbHMvdG9rZW5CdWRnZXQuanMnXG5pbXBvcnQgeyBjb3VudCB9IGZyb20gJy4uL3V0aWxzL2FycmF5LmpzJ1xuaW1wb3J0IHsgZGlybmFtZSwgam9pbiB9IGZyb20gJ3BhdGgnXG5pbXBvcnQgeyB0bXBkaXIgfSBmcm9tICdvcydcbmltcG9ydCBmaWd1cmVzIGZyb20gJ2ZpZ3VyZXMnXG4vLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgY3VzdG9tLXJ1bGVzL3ByZWZlci11c2Uta2V5YmluZGluZ3MgLS0gLyBuIE4gRXNjIFsgdiBhcmUgYmFyZSBsZXR0ZXJzIGluIHRyYW5zY3JpcHQgbW9kYWwgY29udGV4dCwgc2FtZSBjbGFzcyBhcyBnL0cvai9rIGluIFNjcm9sbEtleWJpbmRpbmdIYW5kbGVyXG5pbXBvcnQgeyB1c2VJbnB1dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB7IHVzZVNlYXJjaElucHV0IH0gZnJvbSAnLi4vaG9va3MvdXNlU2VhcmNoSW5wdXQuanMnXG5pbXBvcnQgeyB1c2VUZXJtaW5hbFNpemUgfSBmcm9tICcuLi9ob29rcy91c2VUZXJtaW5hbFNpemUuanMnXG5pbXBvcnQgeyB1c2VTZWFyY2hIaWdobGlnaHQgfSBmcm9tICcuLi9pbmsvaG9va3MvdXNlLXNlYXJjaC1oaWdobGlnaHQuanMnXG5pbXBvcnQgdHlwZSB7IEp1bXBIYW5kbGUgfSBmcm9tICcuLi9jb21wb25lbnRzL1ZpcnR1YWxNZXNzYWdlTGlzdC5qcydcbmltcG9ydCB7IHJlbmRlck1lc3NhZ2VzVG9QbGFpblRleHQgfSBmcm9tICcuLi91dGlscy9leHBvcnRSZW5kZXJlci5qcydcbmltcG9ydCB7IG9wZW5GaWxlSW5FeHRlcm5hbEVkaXRvciB9IGZyb20gJy4uL3V0aWxzL2VkaXRvci5qcydcbmltcG9ydCB7IHdyaXRlRmlsZSB9IGZyb20gJ2ZzL3Byb21pc2VzJ1xuaW1wb3J0IHtcbiAgQm94LFxuICBUZXh0LFxuICB1c2VTdGRpbixcbiAgdXNlVGhlbWUsXG4gIHVzZVRlcm1pbmFsRm9jdXMsXG4gIHVzZVRlcm1pbmFsVGl0bGUsXG4gIHVzZVRhYlN0YXR1cyxcbn0gZnJvbSAnLi4vaW5rLmpzJ1xuaW1wb3J0IHR5cGUgeyBUYWJTdGF0dXNLaW5kIH0gZnJvbSAnLi4vaW5rL2hvb2tzL3VzZS10YWItc3RhdHVzLmpzJ1xuaW1wb3J0IHsgQ29zdFRocmVzaG9sZERpYWxvZyB9IGZyb20gJy4uL2NvbXBvbmVudHMvQ29zdFRocmVzaG9sZERpYWxvZy5qcydcbmltcG9ydCB7IElkbGVSZXR1cm5EaWFsb2cgfSBmcm9tICcuLi9jb21wb25lbnRzL0lkbGVSZXR1cm5EaWFsb2cuanMnXG5pbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7XG4gIHVzZUVmZmVjdCxcbiAgdXNlTWVtbyxcbiAgdXNlUmVmLFxuICB1c2VTdGF0ZSxcbiAgdXNlQ2FsbGJhY2ssXG4gIHVzZURlZmVycmVkVmFsdWUsXG4gIHVzZUxheW91dEVmZmVjdCxcbiAgdHlwZSBSZWZPYmplY3QsXG59IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlTm90aWZpY2F0aW9ucyB9IGZyb20gJy4uL2NvbnRleHQvbm90aWZpY2F0aW9ucy5qcydcbmltcG9ydCB7IHNlbmROb3RpZmljYXRpb24gfSBmcm9tICcuLi9zZXJ2aWNlcy9ub3RpZmllci5qcydcbmltcG9ydCB7XG4gIHN0YXJ0UHJldmVudFNsZWVwLFxuICBzdG9wUHJldmVudFNsZWVwLFxufSBmcm9tICcuLi9zZXJ2aWNlcy9wcmV2ZW50U2xlZXAuanMnXG5pbXBvcnQgeyB1c2VUZXJtaW5hbE5vdGlmaWNhdGlvbiB9IGZyb20gJy4uL2luay91c2VUZXJtaW5hbE5vdGlmaWNhdGlvbi5qcydcbmltcG9ydCB7IGhhc0N1cnNvclVwVmlld3BvcnRZYW5rQnVnIH0gZnJvbSAnLi4vaW5rL3Rlcm1pbmFsLmpzJ1xuaW1wb3J0IHtcbiAgY3JlYXRlRmlsZVN0YXRlQ2FjaGVXaXRoU2l6ZUxpbWl0LFxuICBtZXJnZUZpbGVTdGF0ZUNhY2hlcyxcbiAgUkVBRF9GSUxFX1NUQVRFX0NBQ0hFX1NJWkUsXG59IGZyb20gJy4uL3V0aWxzL2ZpbGVTdGF0ZUNhY2hlLmpzJ1xuaW1wb3J0IHtcbiAgdXBkYXRlTGFzdEludGVyYWN0aW9uVGltZSxcbiAgZ2V0TGFzdEludGVyYWN0aW9uVGltZSxcbiAgZ2V0T3JpZ2luYWxDd2QsXG4gIGdldFByb2plY3RSb290LFxuICBnZXRTZXNzaW9uSWQsXG4gIHN3aXRjaFNlc3Npb24sXG4gIHNldENvc3RTdGF0ZUZvclJlc3RvcmUsXG4gIGdldFR1cm5Ib29rRHVyYXRpb25NcyxcbiAgZ2V0VHVybkhvb2tDb3VudCxcbiAgcmVzZXRUdXJuSG9va0R1cmF0aW9uLFxuICBnZXRUdXJuVG9vbER1cmF0aW9uTXMsXG4gIGdldFR1cm5Ub29sQ291bnQsXG4gIHJlc2V0VHVyblRvb2xEdXJhdGlvbixcbiAgZ2V0VHVybkNsYXNzaWZpZXJEdXJhdGlvbk1zLFxuICBnZXRUdXJuQ2xhc3NpZmllckNvdW50LFxuICByZXNldFR1cm5DbGFzc2lmaWVyRHVyYXRpb24sXG59IGZyb20gJy4uL2Jvb3RzdHJhcC9zdGF0ZS5qcydcbmltcG9ydCB7IGFzU2Vzc2lvbklkLCBhc0FnZW50SWQgfSBmcm9tICcuLi90eXBlcy9pZHMuanMnXG5pbXBvcnQgeyBsb2dGb3JEZWJ1Z2dpbmcgfSBmcm9tICcuLi91dGlscy9kZWJ1Zy5qcydcbmltcG9ydCB7IFF1ZXJ5R3VhcmQgfSBmcm9tICcuLi91dGlscy9RdWVyeUd1YXJkLmpzJ1xuaW1wb3J0IHsgaXNFbnZUcnV0aHkgfSBmcm9tICcuLi91dGlscy9lbnZVdGlscy5qcydcbmltcG9ydCB7IGZvcm1hdFRva2VucywgdHJ1bmNhdGVUb1dpZHRoIH0gZnJvbSAnLi4vdXRpbHMvZm9ybWF0LmpzJ1xuaW1wb3J0IHsgY29uc3VtZUVhcmx5SW5wdXQgfSBmcm9tICcuLi91dGlscy9lYXJseUlucHV0LmpzJ1xuXG5pbXBvcnQgeyBzZXRNZW1iZXJBY3RpdmUgfSBmcm9tICcuLi91dGlscy9zd2FybS90ZWFtSGVscGVycy5qcydcbmltcG9ydCB7XG4gIGlzU3dhcm1Xb3JrZXIsXG4gIGdlbmVyYXRlU2FuZGJveFJlcXVlc3RJZCxcbiAgc2VuZFNhbmRib3hQZXJtaXNzaW9uUmVxdWVzdFZpYU1haWxib3gsXG4gIHNlbmRTYW5kYm94UGVybWlzc2lvblJlc3BvbnNlVmlhTWFpbGJveCxcbn0gZnJvbSAnLi4vdXRpbHMvc3dhcm0vcGVybWlzc2lvblN5bmMuanMnXG5pbXBvcnQgeyByZWdpc3RlclNhbmRib3hQZXJtaXNzaW9uQ2FsbGJhY2sgfSBmcm9tICcuLi9ob29rcy91c2VTd2FybVBlcm1pc3Npb25Qb2xsZXIuanMnXG5pbXBvcnQgeyBnZXRUZWFtTmFtZSwgZ2V0QWdlbnROYW1lIH0gZnJvbSAnLi4vdXRpbHMvdGVhbW1hdGUuanMnXG5pbXBvcnQgeyBXb3JrZXJQZW5kaW5nUGVybWlzc2lvbiB9IGZyb20gJy4uL2NvbXBvbmVudHMvcGVybWlzc2lvbnMvV29ya2VyUGVuZGluZ1Blcm1pc3Npb24uanMnXG5pbXBvcnQge1xuICBpbmplY3RVc2VyTWVzc2FnZVRvVGVhbW1hdGUsXG4gIGdldEFsbEluUHJvY2Vzc1RlYW1tYXRlVGFza3MsXG59IGZyb20gJy4uL3Rhc2tzL0luUHJvY2Vzc1RlYW1tYXRlVGFzay9JblByb2Nlc3NUZWFtbWF0ZVRhc2suanMnXG5pbXBvcnQge1xuICBpc0xvY2FsQWdlbnRUYXNrLFxuICBxdWV1ZVBlbmRpbmdNZXNzYWdlLFxuICBhcHBlbmRNZXNzYWdlVG9Mb2NhbEFnZW50LFxuICB0eXBlIExvY2FsQWdlbnRUYXNrU3RhdGUsXG59IGZyb20gJy4uL3Rhc2tzL0xvY2FsQWdlbnRUYXNrL0xvY2FsQWdlbnRUYXNrLmpzJ1xuaW1wb3J0IHtcbiAgcmVnaXN0ZXJMZWFkZXJUb29sVXNlQ29uZmlybVF1ZXVlLFxuICB1bnJlZ2lzdGVyTGVhZGVyVG9vbFVzZUNvbmZpcm1RdWV1ZSxcbiAgcmVnaXN0ZXJMZWFkZXJTZXRUb29sUGVybWlzc2lvbkNvbnRleHQsXG4gIHVucmVnaXN0ZXJMZWFkZXJTZXRUb29sUGVybWlzc2lvbkNvbnRleHQsXG59IGZyb20gJy4uL3V0aWxzL3N3YXJtL2xlYWRlclBlcm1pc3Npb25CcmlkZ2UuanMnXG5pbXBvcnQgeyBlbmRJbnRlcmFjdGlvblNwYW4gfSBmcm9tICcuLi91dGlscy90ZWxlbWV0cnkvc2Vzc2lvblRyYWNpbmcuanMnXG5pbXBvcnQgeyB1c2VMb2dNZXNzYWdlcyB9IGZyb20gJy4uL2hvb2tzL3VzZUxvZ01lc3NhZ2VzLmpzJ1xuaW1wb3J0IHsgdXNlUmVwbEJyaWRnZSB9IGZyb20gJy4uL2hvb2tzL3VzZVJlcGxCcmlkZ2UuanMnXG5pbXBvcnQge1xuICB0eXBlIENvbW1hbmQsXG4gIHR5cGUgQ29tbWFuZFJlc3VsdERpc3BsYXksXG4gIHR5cGUgUmVzdW1lRW50cnlwb2ludCxcbiAgZ2V0Q29tbWFuZE5hbWUsXG4gIGlzQ29tbWFuZEVuYWJsZWQsXG59IGZyb20gJy4uL2NvbW1hbmRzLmpzJ1xuaW1wb3J0IHR5cGUge1xuICBQcm9tcHRJbnB1dE1vZGUsXG4gIFF1ZXVlZENvbW1hbmQsXG4gIFZpbU1vZGUsXG59IGZyb20gJy4uL3R5cGVzL3RleHRJbnB1dFR5cGVzLmpzJ1xuaW1wb3J0IHtcbiAgTWVzc2FnZVNlbGVjdG9yLFxuICBzZWxlY3RhYmxlVXNlck1lc3NhZ2VzRmlsdGVyLFxuICBtZXNzYWdlc0FmdGVyQXJlT25seVN5bnRoZXRpYyxcbn0gZnJvbSAnLi4vY29tcG9uZW50cy9NZXNzYWdlU2VsZWN0b3IuanMnXG5pbXBvcnQgeyB1c2VJZGVMb2dnaW5nIH0gZnJvbSAnLi4vaG9va3MvdXNlSWRlTG9nZ2luZy5qcydcbmltcG9ydCB7XG4gIFBlcm1pc3Npb25SZXF1ZXN0LFxuICB0eXBlIFRvb2xVc2VDb25maXJtLFxufSBmcm9tICcuLi9jb21wb25lbnRzL3Blcm1pc3Npb25zL1Blcm1pc3Npb25SZXF1ZXN0LmpzJ1xuaW1wb3J0IHsgRWxpY2l0YXRpb25EaWFsb2cgfSBmcm9tICcuLi9jb21wb25lbnRzL21jcC9FbGljaXRhdGlvbkRpYWxvZy5qcydcbmltcG9ydCB7IFByb21wdERpYWxvZyB9IGZyb20gJy4uL2NvbXBvbmVudHMvaG9va3MvUHJvbXB0RGlhbG9nLmpzJ1xuaW1wb3J0IHR5cGUgeyBQcm9tcHRSZXF1ZXN0LCBQcm9tcHRSZXNwb25zZSB9IGZyb20gJy4uL3R5cGVzL2hvb2tzLmpzJ1xuaW1wb3J0IFByb21wdElucHV0IGZyb20gJy4uL2NvbXBvbmVudHMvUHJvbXB0SW5wdXQvUHJvbXB0SW5wdXQuanMnXG5pbXBvcnQgeyBQcm9tcHRJbnB1dFF1ZXVlZENvbW1hbmRzIH0gZnJvbSAnLi4vY29tcG9uZW50cy9Qcm9tcHRJbnB1dC9Qcm9tcHRJbnB1dFF1ZXVlZENvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgdXNlUmVtb3RlU2Vzc2lvbiB9IGZyb20gJy4uL2hvb2tzL3VzZVJlbW90ZVNlc3Npb24uanMnXG5pbXBvcnQgeyB1c2VEaXJlY3RDb25uZWN0IH0gZnJvbSAnLi4vaG9va3MvdXNlRGlyZWN0Q29ubmVjdC5qcydcbmltcG9ydCB0eXBlIHsgRGlyZWN0Q29ubmVjdENvbmZpZyB9IGZyb20gJy4uL3NlcnZlci9kaXJlY3RDb25uZWN0TWFuYWdlci5qcydcbmltcG9ydCB7IHVzZVNTSFNlc3Npb24gfSBmcm9tICcuLi9ob29rcy91c2VTU0hTZXNzaW9uLmpzJ1xuaW1wb3J0IHsgdXNlQXNzaXN0YW50SGlzdG9yeSB9IGZyb20gJy4uL2hvb2tzL3VzZUFzc2lzdGFudEhpc3RvcnkuanMnXG5pbXBvcnQgdHlwZSB7IFNTSFNlc3Npb24gfSBmcm9tICcuLi9zc2gvY3JlYXRlU1NIU2Vzc2lvbi5qcydcbmltcG9ydCB7IFNraWxsSW1wcm92ZW1lbnRTdXJ2ZXkgfSBmcm9tICcuLi9jb21wb25lbnRzL1NraWxsSW1wcm92ZW1lbnRTdXJ2ZXkuanMnXG5pbXBvcnQgeyB1c2VTa2lsbEltcHJvdmVtZW50U3VydmV5IH0gZnJvbSAnLi4vaG9va3MvdXNlU2tpbGxJbXByb3ZlbWVudFN1cnZleS5qcydcbmltcG9ydCB7IHVzZU1vcmVSaWdodCB9IGZyb20gJy4uL21vcmVyaWdodC91c2VNb3JlUmlnaHQuanMnXG5pbXBvcnQge1xuICBTcGlubmVyV2l0aFZlcmIsXG4gIEJyaWVmSWRsZVN0YXR1cyxcbiAgdHlwZSBTcGlubmVyTW9kZSxcbn0gZnJvbSAnLi4vY29tcG9uZW50cy9TcGlubmVyLmpzJ1xuaW1wb3J0IHsgZ2V0U3lzdGVtUHJvbXB0IH0gZnJvbSAnLi4vY29uc3RhbnRzL3Byb21wdHMuanMnXG5pbXBvcnQgeyBidWlsZEVmZmVjdGl2ZVN5c3RlbVByb21wdCB9IGZyb20gJy4uL3V0aWxzL3N5c3RlbVByb21wdC5qcydcbmltcG9ydCB7IGdldFN5c3RlbUNvbnRleHQsIGdldFVzZXJDb250ZXh0IH0gZnJvbSAnLi4vY29udGV4dC5qcydcbmltcG9ydCB7IGdldE1lbW9yeUZpbGVzIH0gZnJvbSAnLi4vdXRpbHMvY2xhdWRlbWQuanMnXG5pbXBvcnQgeyBzdGFydEJhY2tncm91bmRIb3VzZWtlZXBpbmcgfSBmcm9tICcuLi91dGlscy9iYWNrZ3JvdW5kSG91c2VrZWVwaW5nLmpzJ1xuaW1wb3J0IHtcbiAgZ2V0VG90YWxDb3N0LFxuICBzYXZlQ3VycmVudFNlc3Npb25Db3N0cyxcbiAgcmVzZXRDb3N0U3RhdGUsXG4gIGdldFN0b3JlZFNlc3Npb25Db3N0cyxcbn0gZnJvbSAnLi4vY29zdC10cmFja2VyLmpzJ1xuaW1wb3J0IHsgdXNlQ29zdFN1bW1hcnkgfSBmcm9tICcuLi9jb3N0SG9vay5qcydcbmltcG9ydCB7IHVzZUZwc01ldHJpY3MgfSBmcm9tICcuLi9jb250ZXh0L2Zwc01ldHJpY3MuanMnXG5pbXBvcnQgeyB1c2VBZnRlckZpcnN0UmVuZGVyIH0gZnJvbSAnLi4vaG9va3MvdXNlQWZ0ZXJGaXJzdFJlbmRlci5qcydcbmltcG9ydCB7IHVzZURlZmVycmVkSG9va01lc3NhZ2VzIH0gZnJvbSAnLi4vaG9va3MvdXNlRGVmZXJyZWRIb29rTWVzc2FnZXMuanMnXG5pbXBvcnQge1xuICBhZGRUb0hpc3RvcnksXG4gIHJlbW92ZUxhc3RGcm9tSGlzdG9yeSxcbiAgZXhwYW5kUGFzdGVkVGV4dFJlZnMsXG4gIHBhcnNlUmVmZXJlbmNlcyxcbn0gZnJvbSAnLi4vaGlzdG9yeS5qcydcbmltcG9ydCB7IHByZXBlbmRNb2RlQ2hhcmFjdGVyVG9JbnB1dCB9IGZyb20gJy4uL2NvbXBvbmVudHMvUHJvbXB0SW5wdXQvaW5wdXRNb2Rlcy5qcydcbmltcG9ydCB7IHByZXBlbmRUb1NoZWxsSGlzdG9yeUNhY2hlIH0gZnJvbSAnLi4vdXRpbHMvc3VnZ2VzdGlvbnMvc2hlbGxIaXN0b3J5Q29tcGxldGlvbi5qcydcbmltcG9ydCB7IHVzZUFwaUtleVZlcmlmaWNhdGlvbiB9IGZyb20gJy4uL2hvb2tzL3VzZUFwaUtleVZlcmlmaWNhdGlvbi5qcydcbmltcG9ydCB7IEdsb2JhbEtleWJpbmRpbmdIYW5kbGVycyB9IGZyb20gJy4uL2hvb2tzL3VzZUdsb2JhbEtleWJpbmRpbmdzLmpzJ1xuaW1wb3J0IHsgQ29tbWFuZEtleWJpbmRpbmdIYW5kbGVycyB9IGZyb20gJy4uL2hvb2tzL3VzZUNvbW1hbmRLZXliaW5kaW5ncy5qcydcbmltcG9ydCB7IEtleWJpbmRpbmdTZXR1cCB9IGZyb20gJy4uL2tleWJpbmRpbmdzL0tleWJpbmRpbmdQcm92aWRlclNldHVwLmpzJ1xuaW1wb3J0IHsgdXNlU2hvcnRjdXREaXNwbGF5IH0gZnJvbSAnLi4va2V5YmluZGluZ3MvdXNlU2hvcnRjdXREaXNwbGF5LmpzJ1xuaW1wb3J0IHsgZ2V0U2hvcnRjdXREaXNwbGF5IH0gZnJvbSAnLi4va2V5YmluZGluZ3Mvc2hvcnRjdXRGb3JtYXQuanMnXG5pbXBvcnQgeyBDYW5jZWxSZXF1ZXN0SGFuZGxlciB9IGZyb20gJy4uL2hvb2tzL3VzZUNhbmNlbFJlcXVlc3QuanMnXG5pbXBvcnQgeyB1c2VCYWNrZ3JvdW5kVGFza05hdmlnYXRpb24gfSBmcm9tICcuLi9ob29rcy91c2VCYWNrZ3JvdW5kVGFza05hdmlnYXRpb24uanMnXG5pbXBvcnQgeyB1c2VTd2FybUluaXRpYWxpemF0aW9uIH0gZnJvbSAnLi4vaG9va3MvdXNlU3dhcm1Jbml0aWFsaXphdGlvbi5qcydcbmltcG9ydCB7IHVzZVRlYW1tYXRlVmlld0F1dG9FeGl0IH0gZnJvbSAnLi4vaG9va3MvdXNlVGVhbW1hdGVWaWV3QXV0b0V4aXQuanMnXG5pbXBvcnQgeyBlcnJvck1lc3NhZ2UgfSBmcm9tICcuLi91dGlscy9lcnJvcnMuanMnXG5pbXBvcnQgeyBpc0h1bWFuVHVybiB9IGZyb20gJy4uL3V0aWxzL21lc3NhZ2VQcmVkaWNhdGVzLmpzJ1xuaW1wb3J0IHsgbG9nRXJyb3IgfSBmcm9tICcuLi91dGlscy9sb2cuanMnXG4vLyBEZWFkIGNvZGUgZWxpbWluYXRpb246IGNvbmRpdGlvbmFsIGltcG9ydHNcbi8qIGVzbGludC1kaXNhYmxlIGN1c3RvbS1ydWxlcy9uby1wcm9jZXNzLWVudi10b3AtbGV2ZWwsIEB0eXBlc2NyaXB0LWVzbGludC9uby1yZXF1aXJlLWltcG9ydHMgKi9cbmNvbnN0IHVzZVZvaWNlSW50ZWdyYXRpb246IHR5cGVvZiBpbXBvcnQoJy4uL2hvb2tzL3VzZVZvaWNlSW50ZWdyYXRpb24uanMnKS51c2VWb2ljZUludGVncmF0aW9uID1cbiAgZmVhdHVyZSgnVk9JQ0VfTU9ERScpXG4gICAgPyByZXF1aXJlKCcuLi9ob29rcy91c2VWb2ljZUludGVncmF0aW9uLmpzJykudXNlVm9pY2VJbnRlZ3JhdGlvblxuICAgIDogKCkgPT4gKHtcbiAgICAgICAgc3RyaXBUcmFpbGluZzogKCkgPT4gMCxcbiAgICAgICAgaGFuZGxlS2V5RXZlbnQ6ICgpID0+IHt9LFxuICAgICAgICByZXNldEFuY2hvcjogKCkgPT4ge30sXG4gICAgICB9KVxuY29uc3QgVm9pY2VLZXliaW5kaW5nSGFuZGxlcjogdHlwZW9mIGltcG9ydCgnLi4vaG9va3MvdXNlVm9pY2VJbnRlZ3JhdGlvbi5qcycpLlZvaWNlS2V5YmluZGluZ0hhbmRsZXIgPVxuICBmZWF0dXJlKCdWT0lDRV9NT0RFJylcbiAgICA/IHJlcXVpcmUoJy4uL2hvb2tzL3VzZVZvaWNlSW50ZWdyYXRpb24uanMnKS5Wb2ljZUtleWJpbmRpbmdIYW5kbGVyXG4gICAgOiAoKSA9PiBudWxsXG4vLyBGcnVzdHJhdGlvbiBkZXRlY3Rpb24gaXMgYW50LW9ubHkgKGRvZ2Zvb2RpbmcpLiBDb25kaXRpb25hbCByZXF1aXJlIHNvIGV4dGVybmFsXG4vLyBidWlsZHMgZWxpbWluYXRlIHRoZSBtb2R1bGUgZW50aXJlbHkgKGluY2x1ZGluZyBpdHMgdHdvIE8obikgdXNlTWVtb3MgdGhhdCBydW5cbi8vIG9uIGV2ZXJ5IG1lc3NhZ2VzIGNoYW5nZSwgcGx1cyB0aGUgR3Jvd3RoQm9vayBmZXRjaCkuXG5jb25zdCB1c2VGcnVzdHJhdGlvbkRldGVjdGlvbjogdHlwZW9mIGltcG9ydCgnLi4vY29tcG9uZW50cy9GZWVkYmFja1N1cnZleS91c2VGcnVzdHJhdGlvbkRldGVjdGlvbi5qcycpLnVzZUZydXN0cmF0aW9uRGV0ZWN0aW9uID1cbiAgXCJleHRlcm5hbFwiID09PSAnYW50J1xuICAgID8gcmVxdWlyZSgnLi4vY29tcG9uZW50cy9GZWVkYmFja1N1cnZleS91c2VGcnVzdHJhdGlvbkRldGVjdGlvbi5qcycpXG4gICAgICAgIC51c2VGcnVzdHJhdGlvbkRldGVjdGlvblxuICAgIDogKCkgPT4gKHsgc3RhdGU6ICdjbG9zZWQnLCBoYW5kbGVUcmFuc2NyaXB0U2VsZWN0OiAoKSA9PiB7fSB9KVxuLy8gQW50LW9ubHkgb3JnIHdhcm5pbmcuIENvbmRpdGlvbmFsIHJlcXVpcmUgc28gdGhlIG9yZyBVVUlEIGxpc3QgaXNcbi8vIGVsaW1pbmF0ZWQgZnJvbSBleHRlcm5hbCBidWlsZHMgKG9uZSBVVUlEIGlzIG9uIGV4Y2x1ZGVkLXN0cmluZ3MpLlxuY29uc3QgdXNlQW50T3JnV2FybmluZ05vdGlmaWNhdGlvbjogdHlwZW9mIGltcG9ydCgnLi4vaG9va3Mvbm90aWZzL3VzZUFudE9yZ1dhcm5pbmdOb3RpZmljYXRpb24uanMnKS51c2VBbnRPcmdXYXJuaW5nTm90aWZpY2F0aW9uID1cbiAgXCJleHRlcm5hbFwiID09PSAnYW50J1xuICAgID8gcmVxdWlyZSgnLi4vaG9va3Mvbm90aWZzL3VzZUFudE9yZ1dhcm5pbmdOb3RpZmljYXRpb24uanMnKVxuICAgICAgICAudXNlQW50T3JnV2FybmluZ05vdGlmaWNhdGlvblxuICAgIDogKCkgPT4ge31cbi8vIERlYWQgY29kZSBlbGltaW5hdGlvbjogY29uZGl0aW9uYWwgaW1wb3J0IGZvciBjb29yZGluYXRvciBtb2RlXG5jb25zdCBnZXRDb29yZGluYXRvclVzZXJDb250ZXh0OiAoXG4gIG1jcENsaWVudHM6IFJlYWRvbmx5QXJyYXk8eyBuYW1lOiBzdHJpbmcgfT4sXG4gIHNjcmF0Y2hwYWREaXI/OiBzdHJpbmcsXG4pID0+IHsgW2s6IHN0cmluZ106IHN0cmluZyB9ID0gZmVhdHVyZSgnQ09PUkRJTkFUT1JfTU9ERScpXG4gID8gcmVxdWlyZSgnLi4vY29vcmRpbmF0b3IvY29vcmRpbmF0b3JNb2RlLmpzJykuZ2V0Q29vcmRpbmF0b3JVc2VyQ29udGV4dFxuICA6ICgpID0+ICh7fSlcbi8qIGVzbGludC1lbmFibGUgY3VzdG9tLXJ1bGVzL25vLXByb2Nlc3MtZW52LXRvcC1sZXZlbCwgQHR5cGVzY3JpcHQtZXNsaW50L25vLXJlcXVpcmUtaW1wb3J0cyAqL1xuaW1wb3J0IHVzZUNhblVzZVRvb2wgZnJvbSAnLi4vaG9va3MvdXNlQ2FuVXNlVG9vbC5qcydcbmltcG9ydCB0eXBlIHsgVG9vbFBlcm1pc3Npb25Db250ZXh0LCBUb29sIH0gZnJvbSAnLi4vVG9vbC5qcydcbmltcG9ydCB7XG4gIGFwcGx5UGVybWlzc2lvblVwZGF0ZSxcbiAgYXBwbHlQZXJtaXNzaW9uVXBkYXRlcyxcbiAgcGVyc2lzdFBlcm1pc3Npb25VcGRhdGUsXG59IGZyb20gJy4uL3V0aWxzL3Blcm1pc3Npb25zL1Blcm1pc3Npb25VcGRhdGUuanMnXG5pbXBvcnQgeyBidWlsZFBlcm1pc3Npb25VcGRhdGVzIH0gZnJvbSAnLi4vY29tcG9uZW50cy9wZXJtaXNzaW9ucy9FeGl0UGxhbk1vZGVQZXJtaXNzaW9uUmVxdWVzdC9FeGl0UGxhbk1vZGVQZXJtaXNzaW9uUmVxdWVzdC5qcydcbmltcG9ydCB7IHN0cmlwRGFuZ2Vyb3VzUGVybWlzc2lvbnNGb3JBdXRvTW9kZSB9IGZyb20gJy4uL3V0aWxzL3Blcm1pc3Npb25zL3Blcm1pc3Npb25TZXR1cC5qcydcbmltcG9ydCB7XG4gIGdldFNjcmF0Y2hwYWREaXIsXG4gIGlzU2NyYXRjaHBhZEVuYWJsZWQsXG59IGZyb20gJy4uL3V0aWxzL3Blcm1pc3Npb25zL2ZpbGVzeXN0ZW0uanMnXG5pbXBvcnQgeyBXRUJfRkVUQ0hfVE9PTF9OQU1FIH0gZnJvbSAnLi4vdG9vbHMvV2ViRmV0Y2hUb29sL3Byb21wdC5qcydcbmltcG9ydCB7IFNMRUVQX1RPT0xfTkFNRSB9IGZyb20gJy4uL3Rvb2xzL1NsZWVwVG9vbC9wcm9tcHQuanMnXG5pbXBvcnQgeyBjbGVhclNwZWN1bGF0aXZlQ2hlY2tzIH0gZnJvbSAnLi4vdG9vbHMvQmFzaFRvb2wvYmFzaFBlcm1pc3Npb25zLmpzJ1xuaW1wb3J0IHR5cGUgeyBBdXRvVXBkYXRlclJlc3VsdCB9IGZyb20gJy4uL3V0aWxzL2F1dG9VcGRhdGVyLmpzJ1xuaW1wb3J0IHtcbiAgZ2V0R2xvYmFsQ29uZmlnLFxuICBzYXZlR2xvYmFsQ29uZmlnLFxuICBnZXRHbG9iYWxDb25maWdXcml0ZUNvdW50LFxufSBmcm9tICcuLi91dGlscy9jb25maWcuanMnXG5pbXBvcnQgeyBoYXNDb25zb2xlQmlsbGluZ0FjY2VzcyB9IGZyb20gJy4uL3V0aWxzL2JpbGxpbmcuanMnXG5pbXBvcnQge1xuICBsb2dFdmVudCxcbiAgdHlwZSBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxufSBmcm9tICdzcmMvc2VydmljZXMvYW5hbHl0aWNzL2luZGV4LmpzJ1xuaW1wb3J0IHsgZ2V0RmVhdHVyZVZhbHVlX0NBQ0hFRF9NQVlfQkVfU1RBTEUgfSBmcm9tICdzcmMvc2VydmljZXMvYW5hbHl0aWNzL2dyb3d0aGJvb2suanMnXG5pbXBvcnQge1xuICB0ZXh0Rm9yUmVzdWJtaXQsXG4gIGhhbmRsZU1lc3NhZ2VGcm9tU3RyZWFtLFxuICB0eXBlIFN0cmVhbWluZ1Rvb2xVc2UsXG4gIHR5cGUgU3RyZWFtaW5nVGhpbmtpbmcsXG4gIGlzQ29tcGFjdEJvdW5kYXJ5TWVzc2FnZSxcbiAgZ2V0TWVzc2FnZXNBZnRlckNvbXBhY3RCb3VuZGFyeSxcbiAgZ2V0Q29udGVudFRleHQsXG4gIGNyZWF0ZVVzZXJNZXNzYWdlLFxuICBjcmVhdGVBc3Npc3RhbnRNZXNzYWdlLFxuICBjcmVhdGVUdXJuRHVyYXRpb25NZXNzYWdlLFxuICBjcmVhdGVBZ2VudHNLaWxsZWRNZXNzYWdlLFxuICBjcmVhdGVBcGlNZXRyaWNzTWVzc2FnZSxcbiAgY3JlYXRlU3lzdGVtTWVzc2FnZSxcbiAgY3JlYXRlQ29tbWFuZElucHV0TWVzc2FnZSxcbiAgZm9ybWF0Q29tbWFuZElucHV0VGFncyxcbn0gZnJvbSAnLi4vdXRpbHMvbWVzc2FnZXMuanMnXG5pbXBvcnQgeyBnZW5lcmF0ZVNlc3Npb25UaXRsZSB9IGZyb20gJy4uL3V0aWxzL3Nlc3Npb25UaXRsZS5qcydcbmltcG9ydCB7XG4gIEJBU0hfSU5QVVRfVEFHLFxuICBDT01NQU5EX01FU1NBR0VfVEFHLFxuICBDT01NQU5EX05BTUVfVEFHLFxuICBMT0NBTF9DT01NQU5EX1NURE9VVF9UQUcsXG59IGZyb20gJy4uL2NvbnN0YW50cy94bWwuanMnXG5pbXBvcnQgeyBlc2NhcGVYbWwgfSBmcm9tICcuLi91dGlscy94bWwuanMnXG5pbXBvcnQgdHlwZSB7IFRoaW5raW5nQ29uZmlnIH0gZnJvbSAnLi4vdXRpbHMvdGhpbmtpbmcuanMnXG5pbXBvcnQgeyBncmFjZWZ1bFNodXRkb3duU3luYyB9IGZyb20gJy4uL3V0aWxzL2dyYWNlZnVsU2h1dGRvd24uanMnXG5pbXBvcnQge1xuICBoYW5kbGVQcm9tcHRTdWJtaXQsXG4gIHR5cGUgUHJvbXB0SW5wdXRIZWxwZXJzLFxufSBmcm9tICcuLi91dGlscy9oYW5kbGVQcm9tcHRTdWJtaXQuanMnXG5pbXBvcnQgeyB1c2VRdWV1ZVByb2Nlc3NvciB9IGZyb20gJy4uL2hvb2tzL3VzZVF1ZXVlUHJvY2Vzc29yLmpzJ1xuaW1wb3J0IHsgdXNlTWFpbGJveEJyaWRnZSB9IGZyb20gJy4uL2hvb2tzL3VzZU1haWxib3hCcmlkZ2UuanMnXG5pbXBvcnQge1xuICBxdWVyeUNoZWNrcG9pbnQsXG4gIGxvZ1F1ZXJ5UHJvZmlsZVJlcG9ydCxcbn0gZnJvbSAnLi4vdXRpbHMvcXVlcnlQcm9maWxlci5qcydcbmltcG9ydCB0eXBlIHtcbiAgTWVzc2FnZSBhcyBNZXNzYWdlVHlwZSxcbiAgVXNlck1lc3NhZ2UsXG4gIFByb2dyZXNzTWVzc2FnZSxcbiAgSG9va1Jlc3VsdE1lc3NhZ2UsXG4gIFBhcnRpYWxDb21wYWN0RGlyZWN0aW9uLFxufSBmcm9tICcuLi90eXBlcy9tZXNzYWdlLmpzJ1xuaW1wb3J0IHsgcXVlcnkgfSBmcm9tICcuLi9xdWVyeS5qcydcbmltcG9ydCB7IG1lcmdlQ2xpZW50cywgdXNlTWVyZ2VkQ2xpZW50cyB9IGZyb20gJy4uL2hvb2tzL3VzZU1lcmdlZENsaWVudHMuanMnXG5pbXBvcnQgeyBnZXRRdWVyeVNvdXJjZUZvclJFUEwgfSBmcm9tICcuLi91dGlscy9wcm9tcHRDYXRlZ29yeS5qcydcbmltcG9ydCB7IHVzZU1lcmdlZFRvb2xzIH0gZnJvbSAnLi4vaG9va3MvdXNlTWVyZ2VkVG9vbHMuanMnXG5pbXBvcnQgeyBtZXJnZUFuZEZpbHRlclRvb2xzIH0gZnJvbSAnLi4vdXRpbHMvdG9vbFBvb2wuanMnXG5pbXBvcnQgeyB1c2VNZXJnZWRDb21tYW5kcyB9IGZyb20gJy4uL2hvb2tzL3VzZU1lcmdlZENvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgdXNlU2tpbGxzQ2hhbmdlIH0gZnJvbSAnLi4vaG9va3MvdXNlU2tpbGxzQ2hhbmdlLmpzJ1xuaW1wb3J0IHsgdXNlTWFuYWdlUGx1Z2lucyB9IGZyb20gJy4uL2hvb2tzL3VzZU1hbmFnZVBsdWdpbnMuanMnXG5pbXBvcnQgeyBNZXNzYWdlcyB9IGZyb20gJy4uL2NvbXBvbmVudHMvTWVzc2FnZXMuanMnXG5pbXBvcnQgeyBUYXNrTGlzdFYyIH0gZnJvbSAnLi4vY29tcG9uZW50cy9UYXNrTGlzdFYyLmpzJ1xuaW1wb3J0IHsgVGVhbW1hdGVWaWV3SGVhZGVyIH0gZnJvbSAnLi4vY29tcG9uZW50cy9UZWFtbWF0ZVZpZXdIZWFkZXIuanMnXG5pbXBvcnQgeyB1c2VUYXNrc1YyV2l0aENvbGxhcHNlRWZmZWN0IH0gZnJvbSAnLi4vaG9va3MvdXNlVGFza3NWMi5qcydcbmltcG9ydCB7IG1heWJlTWFya1Byb2plY3RPbmJvYXJkaW5nQ29tcGxldGUgfSBmcm9tICcuLi9wcm9qZWN0T25ib2FyZGluZ1N0YXRlLmpzJ1xuaW1wb3J0IHR5cGUgeyBNQ1BTZXJ2ZXJDb25uZWN0aW9uIH0gZnJvbSAnLi4vc2VydmljZXMvbWNwL3R5cGVzLmpzJ1xuaW1wb3J0IHR5cGUgeyBTY29wZWRNY3BTZXJ2ZXJDb25maWcgfSBmcm9tICcuLi9zZXJ2aWNlcy9tY3AvdHlwZXMuanMnXG5pbXBvcnQgeyByYW5kb21VVUlELCB0eXBlIFVVSUQgfSBmcm9tICdjcnlwdG8nXG5pbXBvcnQgeyBwcm9jZXNzU2Vzc2lvblN0YXJ0SG9va3MgfSBmcm9tICcuLi91dGlscy9zZXNzaW9uU3RhcnQuanMnXG5pbXBvcnQge1xuICBleGVjdXRlU2Vzc2lvbkVuZEhvb2tzLFxuICBnZXRTZXNzaW9uRW5kSG9va1RpbWVvdXRNcyxcbn0gZnJvbSAnLi4vdXRpbHMvaG9va3MuanMnXG5pbXBvcnQgeyB0eXBlIElERVNlbGVjdGlvbiwgdXNlSWRlU2VsZWN0aW9uIH0gZnJvbSAnLi4vaG9va3MvdXNlSWRlU2VsZWN0aW9uLmpzJ1xuaW1wb3J0IHsgZ2V0VG9vbHMsIGFzc2VtYmxlVG9vbFBvb2wgfSBmcm9tICcuLi90b29scy5qcydcbmltcG9ydCB0eXBlIHsgQWdlbnREZWZpbml0aW9uIH0gZnJvbSAnLi4vdG9vbHMvQWdlbnRUb29sL2xvYWRBZ2VudHNEaXIuanMnXG5pbXBvcnQgeyByZXNvbHZlQWdlbnRUb29scyB9IGZyb20gJy4uL3Rvb2xzL0FnZW50VG9vbC9hZ2VudFRvb2xVdGlscy5qcydcbmltcG9ydCB7IHJlc3VtZUFnZW50QmFja2dyb3VuZCB9IGZyb20gJy4uL3Rvb2xzL0FnZW50VG9vbC9yZXN1bWVBZ2VudC5qcydcbmltcG9ydCB7IHVzZU1haW5Mb29wTW9kZWwgfSBmcm9tICcuLi9ob29rcy91c2VNYWluTG9vcE1vZGVsLmpzJ1xuaW1wb3J0IHtcbiAgdXNlQXBwU3RhdGUsXG4gIHVzZVNldEFwcFN0YXRlLFxuICB1c2VBcHBTdGF0ZVN0b3JlLFxufSBmcm9tICcuLi9zdGF0ZS9BcHBTdGF0ZS5qcydcbmltcG9ydCB0eXBlIHtcbiAgQ29udGVudEJsb2NrUGFyYW0sXG4gIEltYWdlQmxvY2tQYXJhbSxcbn0gZnJvbSAnQGFudGhyb3BpYy1haS9zZGsvcmVzb3VyY2VzL21lc3NhZ2VzLm1qcydcbmltcG9ydCB0eXBlIHsgUHJvY2Vzc1VzZXJJbnB1dENvbnRleHQgfSBmcm9tICcuLi91dGlscy9wcm9jZXNzVXNlcklucHV0L3Byb2Nlc3NVc2VySW5wdXQuanMnXG5pbXBvcnQgdHlwZSB7IFBhc3RlZENvbnRlbnQgfSBmcm9tICcuLi91dGlscy9jb25maWcuanMnXG5pbXBvcnQge1xuICBjb3B5UGxhbkZvckZvcmssXG4gIGNvcHlQbGFuRm9yUmVzdW1lLFxuICBnZXRQbGFuU2x1ZyxcbiAgc2V0UGxhblNsdWcsXG59IGZyb20gJy4uL3V0aWxzL3BsYW5zLmpzJ1xuaW1wb3J0IHtcbiAgY2xlYXJTZXNzaW9uTWV0YWRhdGEsXG4gIHJlc2V0U2Vzc2lvbkZpbGVQb2ludGVyLFxuICBhZG9wdFJlc3VtZWRTZXNzaW9uRmlsZSxcbiAgcmVtb3ZlVHJhbnNjcmlwdE1lc3NhZ2UsXG4gIHJlc3RvcmVTZXNzaW9uTWV0YWRhdGEsXG4gIGdldEN1cnJlbnRTZXNzaW9uVGl0bGUsXG4gIGlzRXBoZW1lcmFsVG9vbFByb2dyZXNzLFxuICBpc0xvZ2dhYmxlTWVzc2FnZSxcbiAgc2F2ZVdvcmt0cmVlU3RhdGUsXG4gIGdldEFnZW50VHJhbnNjcmlwdCxcbn0gZnJvbSAnLi4vdXRpbHMvc2Vzc2lvblN0b3JhZ2UuanMnXG5pbXBvcnQgeyBkZXNlcmlhbGl6ZU1lc3NhZ2VzIH0gZnJvbSAnLi4vdXRpbHMvY29udmVyc2F0aW9uUmVjb3ZlcnkuanMnXG5pbXBvcnQge1xuICBleHRyYWN0UmVhZEZpbGVzRnJvbU1lc3NhZ2VzLFxuICBleHRyYWN0QmFzaFRvb2xzRnJvbU1lc3NhZ2VzLFxufSBmcm9tICcuLi91dGlscy9xdWVyeUhlbHBlcnMuanMnXG5pbXBvcnQgeyByZXNldE1pY3JvY29tcGFjdFN0YXRlIH0gZnJvbSAnLi4vc2VydmljZXMvY29tcGFjdC9taWNyb0NvbXBhY3QuanMnXG5pbXBvcnQgeyBydW5Qb3N0Q29tcGFjdENsZWFudXAgfSBmcm9tICcuLi9zZXJ2aWNlcy9jb21wYWN0L3Bvc3RDb21wYWN0Q2xlYW51cC5qcydcbmltcG9ydCB7XG4gIHByb3Zpc2lvbkNvbnRlbnRSZXBsYWNlbWVudFN0YXRlLFxuICByZWNvbnN0cnVjdENvbnRlbnRSZXBsYWNlbWVudFN0YXRlLFxuICB0eXBlIENvbnRlbnRSZXBsYWNlbWVudFJlY29yZCxcbn0gZnJvbSAnLi4vdXRpbHMvdG9vbFJlc3VsdFN0b3JhZ2UuanMnXG5pbXBvcnQgeyBwYXJ0aWFsQ29tcGFjdENvbnZlcnNhdGlvbiB9IGZyb20gJy4uL3NlcnZpY2VzL2NvbXBhY3QvY29tcGFjdC5qcydcbmltcG9ydCB0eXBlIHsgTG9nT3B0aW9uIH0gZnJvbSAnLi4vdHlwZXMvbG9ncy5qcydcbmltcG9ydCB0eXBlIHsgQWdlbnRDb2xvck5hbWUgfSBmcm9tICcuLi90b29scy9BZ2VudFRvb2wvYWdlbnRDb2xvck1hbmFnZXIuanMnXG5pbXBvcnQge1xuICBmaWxlSGlzdG9yeU1ha2VTbmFwc2hvdCxcbiAgdHlwZSBGaWxlSGlzdG9yeVN0YXRlLFxuICBmaWxlSGlzdG9yeVJld2luZCxcbiAgdHlwZSBGaWxlSGlzdG9yeVNuYXBzaG90LFxuICBjb3B5RmlsZUhpc3RvcnlGb3JSZXN1bWUsXG4gIGZpbGVIaXN0b3J5RW5hYmxlZCxcbiAgZmlsZUhpc3RvcnlIYXNBbnlDaGFuZ2VzLFxufSBmcm9tICcuLi91dGlscy9maWxlSGlzdG9yeS5qcydcbmltcG9ydCB7XG4gIHR5cGUgQXR0cmlidXRpb25TdGF0ZSxcbiAgaW5jcmVtZW50UHJvbXB0Q291bnQsXG59IGZyb20gJy4uL3V0aWxzL2NvbW1pdEF0dHJpYnV0aW9uLmpzJ1xuaW1wb3J0IHsgcmVjb3JkQXR0cmlidXRpb25TbmFwc2hvdCB9IGZyb20gJy4uL3V0aWxzL3Nlc3Npb25TdG9yYWdlLmpzJ1xuaW1wb3J0IHtcbiAgY29tcHV0ZVN0YW5kYWxvbmVBZ2VudENvbnRleHQsXG4gIHJlc3RvcmVBZ2VudEZyb21TZXNzaW9uLFxuICByZXN0b3JlU2Vzc2lvblN0YXRlRnJvbUxvZyxcbiAgcmVzdG9yZVdvcmt0cmVlRm9yUmVzdW1lLFxuICBleGl0UmVzdG9yZWRXb3JrdHJlZSxcbn0gZnJvbSAnLi4vdXRpbHMvc2Vzc2lvblJlc3RvcmUuanMnXG5pbXBvcnQge1xuICBpc0JnU2Vzc2lvbixcbiAgdXBkYXRlU2Vzc2lvbk5hbWUsXG4gIHVwZGF0ZVNlc3Npb25BY3Rpdml0eSxcbn0gZnJvbSAnLi4vdXRpbHMvY29uY3VycmVudFNlc3Npb25zLmpzJ1xuaW1wb3J0IHtcbiAgaXNJblByb2Nlc3NUZWFtbWF0ZVRhc2ssXG4gIHR5cGUgSW5Qcm9jZXNzVGVhbW1hdGVUYXNrU3RhdGUsXG59IGZyb20gJy4uL3Rhc2tzL0luUHJvY2Vzc1RlYW1tYXRlVGFzay90eXBlcy5qcydcbmltcG9ydCB7IHJlc3RvcmVSZW1vdGVBZ2VudFRhc2tzIH0gZnJvbSAnLi4vdGFza3MvUmVtb3RlQWdlbnRUYXNrL1JlbW90ZUFnZW50VGFzay5qcydcbmltcG9ydCB7IHVzZUluYm94UG9sbGVyIH0gZnJvbSAnLi4vaG9va3MvdXNlSW5ib3hQb2xsZXIuanMnXG4vLyBEZWFkIGNvZGUgZWxpbWluYXRpb246IGNvbmRpdGlvbmFsIGltcG9ydCBmb3IgbG9vcCBtb2RlXG4vKiBlc2xpbnQtZGlzYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tcmVxdWlyZS1pbXBvcnRzICovXG5jb25zdCBwcm9hY3RpdmVNb2R1bGUgPVxuICBmZWF0dXJlKCdQUk9BQ1RJVkUnKSB8fCBmZWF0dXJlKCdLQUlST1MnKVxuICAgID8gcmVxdWlyZSgnLi4vcHJvYWN0aXZlL2luZGV4LmpzJylcbiAgICA6IG51bGxcbmNvbnN0IFBST0FDVElWRV9OT19PUF9TVUJTQ1JJQkUgPSAoX2NiOiAoKSA9PiB2b2lkKSA9PiAoKSA9PiB7fVxuY29uc3QgUFJPQUNUSVZFX0ZBTFNFID0gKCkgPT4gZmFsc2VcbmNvbnN0IFNVR0dFU1RfQkdfUFJfTk9PUCA9IChfcDogc3RyaW5nLCBfbjogc3RyaW5nKTogYm9vbGVhbiA9PiBmYWxzZVxuY29uc3QgdXNlUHJvYWN0aXZlID1cbiAgZmVhdHVyZSgnUFJPQUNUSVZFJykgfHwgZmVhdHVyZSgnS0FJUk9TJylcbiAgICA/IHJlcXVpcmUoJy4uL3Byb2FjdGl2ZS91c2VQcm9hY3RpdmUuanMnKS51c2VQcm9hY3RpdmVcbiAgICA6IG51bGxcbmNvbnN0IHVzZVNjaGVkdWxlZFRhc2tzID0gZmVhdHVyZSgnQUdFTlRfVFJJR0dFUlMnKVxuICA/IHJlcXVpcmUoJy4uL2hvb2tzL3VzZVNjaGVkdWxlZFRhc2tzLmpzJykudXNlU2NoZWR1bGVkVGFza3NcbiAgOiBudWxsXG4vKiBlc2xpbnQtZW5hYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby1yZXF1aXJlLWltcG9ydHMgKi9cbmltcG9ydCB7IGlzQWdlbnRTd2FybXNFbmFibGVkIH0gZnJvbSAnLi4vdXRpbHMvYWdlbnRTd2FybXNFbmFibGVkLmpzJ1xuaW1wb3J0IHsgdXNlVGFza0xpc3RXYXRjaGVyIH0gZnJvbSAnLi4vaG9va3MvdXNlVGFza0xpc3RXYXRjaGVyLmpzJ1xuaW1wb3J0IHR5cGUge1xuICBTYW5kYm94QXNrQ2FsbGJhY2ssXG4gIE5ldHdvcmtIb3N0UGF0dGVybixcbn0gZnJvbSAnLi4vdXRpbHMvc2FuZGJveC9zYW5kYm94LWFkYXB0ZXIuanMnXG5cbmltcG9ydCB7XG4gIHR5cGUgSURFRXh0ZW5zaW9uSW5zdGFsbGF0aW9uU3RhdHVzLFxuICBjbG9zZU9wZW5EaWZmcyxcbiAgZ2V0Q29ubmVjdGVkSWRlQ2xpZW50LFxuICB0eXBlIElkZVR5cGUsXG59IGZyb20gJy4uL3V0aWxzL2lkZS5qcydcbmltcG9ydCB7IHVzZUlERUludGVncmF0aW9uIH0gZnJvbSAnLi4vaG9va3MvdXNlSURFSW50ZWdyYXRpb24uanMnXG5pbXBvcnQgZXhpdCBmcm9tICcuLi9jb21tYW5kcy9leGl0L2luZGV4LmpzJ1xuaW1wb3J0IHsgRXhpdEZsb3cgfSBmcm9tICcuLi9jb21wb25lbnRzL0V4aXRGbG93LmpzJ1xuaW1wb3J0IHsgZ2V0Q3VycmVudFdvcmt0cmVlU2Vzc2lvbiB9IGZyb20gJy4uL3V0aWxzL3dvcmt0cmVlLmpzJ1xuaW1wb3J0IHtcbiAgcG9wQWxsRWRpdGFibGUsXG4gIGVucXVldWUsXG4gIHR5cGUgU2V0QXBwU3RhdGUsXG4gIGdldENvbW1hbmRRdWV1ZSxcbiAgZ2V0Q29tbWFuZFF1ZXVlTGVuZ3RoLFxuICByZW1vdmVCeUZpbHRlcixcbn0gZnJvbSAnLi4vdXRpbHMvbWVzc2FnZVF1ZXVlTWFuYWdlci5qcydcbmltcG9ydCB7IHVzZUNvbW1hbmRRdWV1ZSB9IGZyb20gJy4uL2hvb2tzL3VzZUNvbW1hbmRRdWV1ZS5qcydcbmltcG9ydCB7IFNlc3Npb25CYWNrZ3JvdW5kSGludCB9IGZyb20gJy4uL2NvbXBvbmVudHMvU2Vzc2lvbkJhY2tncm91bmRIaW50LmpzJ1xuaW1wb3J0IHsgc3RhcnRCYWNrZ3JvdW5kU2Vzc2lvbiB9IGZyb20gJy4uL3Rhc2tzL0xvY2FsTWFpblNlc3Npb25UYXNrLmpzJ1xuaW1wb3J0IHsgdXNlU2Vzc2lvbkJhY2tncm91bmRpbmcgfSBmcm9tICcuLi9ob29rcy91c2VTZXNzaW9uQmFja2dyb3VuZGluZy5qcydcbmltcG9ydCB7IGRpYWdub3N0aWNUcmFja2VyIH0gZnJvbSAnLi4vc2VydmljZXMvZGlhZ25vc3RpY1RyYWNraW5nLmpzJ1xuaW1wb3J0IHtcbiAgaGFuZGxlU3BlY3VsYXRpb25BY2NlcHQsXG4gIHR5cGUgQWN0aXZlU3BlY3VsYXRpb25TdGF0ZSxcbn0gZnJvbSAnLi4vc2VydmljZXMvUHJvbXB0U3VnZ2VzdGlvbi9zcGVjdWxhdGlvbi5qcydcbmltcG9ydCB7IElkZU9uYm9hcmRpbmdEaWFsb2cgfSBmcm9tICcuLi9jb21wb25lbnRzL0lkZU9uYm9hcmRpbmdEaWFsb2cuanMnXG5pbXBvcnQge1xuICBFZmZvcnRDYWxsb3V0LFxuICBzaG91bGRTaG93RWZmb3J0Q2FsbG91dCxcbn0gZnJvbSAnLi4vY29tcG9uZW50cy9FZmZvcnRDYWxsb3V0LmpzJ1xuaW1wb3J0IHR5cGUgeyBFZmZvcnRWYWx1ZSB9IGZyb20gJy4uL3V0aWxzL2VmZm9ydC5qcydcbmltcG9ydCB7IFJlbW90ZUNhbGxvdXQgfSBmcm9tICcuLi9jb21wb25lbnRzL1JlbW90ZUNhbGxvdXQuanMnXG4vKiBlc2xpbnQtZGlzYWJsZSBjdXN0b20tcnVsZXMvbm8tcHJvY2Vzcy1lbnYtdG9wLWxldmVsLCBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tcmVxdWlyZS1pbXBvcnRzICovXG5jb25zdCBBbnRNb2RlbFN3aXRjaENhbGxvdXQgPVxuICBcImV4dGVybmFsXCIgPT09ICdhbnQnXG4gICAgPyByZXF1aXJlKCcuLi9jb21wb25lbnRzL0FudE1vZGVsU3dpdGNoQ2FsbG91dC5qcycpLkFudE1vZGVsU3dpdGNoQ2FsbG91dFxuICAgIDogbnVsbFxuY29uc3Qgc2hvdWxkU2hvd0FudE1vZGVsU3dpdGNoID1cbiAgXCJleHRlcm5hbFwiID09PSAnYW50J1xuICAgID8gcmVxdWlyZSgnLi4vY29tcG9uZW50cy9BbnRNb2RlbFN3aXRjaENhbGxvdXQuanMnKVxuICAgICAgICAuc2hvdWxkU2hvd01vZGVsU3dpdGNoQ2FsbG91dFxuICAgIDogKCk6IGJvb2xlYW4gPT4gZmFsc2VcbmNvbnN0IFVuZGVyY292ZXJBdXRvQ2FsbG91dCA9XG4gIFwiZXh0ZXJuYWxcIiA9PT0gJ2FudCdcbiAgICA/IHJlcXVpcmUoJy4uL2NvbXBvbmVudHMvVW5kZXJjb3ZlckF1dG9DYWxsb3V0LmpzJykuVW5kZXJjb3ZlckF1dG9DYWxsb3V0XG4gICAgOiBudWxsXG4vKiBlc2xpbnQtZW5hYmxlIGN1c3RvbS1ydWxlcy9uby1wcm9jZXNzLWVudi10b3AtbGV2ZWwsIEB0eXBlc2NyaXB0LWVzbGludC9uby1yZXF1aXJlLWltcG9ydHMgKi9cbmltcG9ydCB7IGFjdGl2aXR5TWFuYWdlciB9IGZyb20gJy4uL3V0aWxzL2FjdGl2aXR5TWFuYWdlci5qcydcbmltcG9ydCB7IGNyZWF0ZUFib3J0Q29udHJvbGxlciB9IGZyb20gJy4uL3V0aWxzL2Fib3J0Q29udHJvbGxlci5qcydcbmltcG9ydCB7IE1DUENvbm5lY3Rpb25NYW5hZ2VyIH0gZnJvbSAnc3JjL3NlcnZpY2VzL21jcC9NQ1BDb25uZWN0aW9uTWFuYWdlci5qcydcbmltcG9ydCB7IHVzZUZlZWRiYWNrU3VydmV5IH0gZnJvbSAnc3JjL2NvbXBvbmVudHMvRmVlZGJhY2tTdXJ2ZXkvdXNlRmVlZGJhY2tTdXJ2ZXkuanMnXG5pbXBvcnQgeyB1c2VNZW1vcnlTdXJ2ZXkgfSBmcm9tICdzcmMvY29tcG9uZW50cy9GZWVkYmFja1N1cnZleS91c2VNZW1vcnlTdXJ2ZXkuanMnXG5pbXBvcnQgeyB1c2VQb3N0Q29tcGFjdFN1cnZleSB9IGZyb20gJ3NyYy9jb21wb25lbnRzL0ZlZWRiYWNrU3VydmV5L3VzZVBvc3RDb21wYWN0U3VydmV5LmpzJ1xuaW1wb3J0IHsgRmVlZGJhY2tTdXJ2ZXkgfSBmcm9tICdzcmMvY29tcG9uZW50cy9GZWVkYmFja1N1cnZleS9GZWVkYmFja1N1cnZleS5qcydcbmltcG9ydCB7IHVzZUluc3RhbGxNZXNzYWdlcyB9IGZyb20gJ3NyYy9ob29rcy9ub3RpZnMvdXNlSW5zdGFsbE1lc3NhZ2VzLmpzJ1xuaW1wb3J0IHsgdXNlQXdheVN1bW1hcnkgfSBmcm9tICdzcmMvaG9va3MvdXNlQXdheVN1bW1hcnkuanMnXG5pbXBvcnQgeyB1c2VDaHJvbWVFeHRlbnNpb25Ob3RpZmljYXRpb24gfSBmcm9tICdzcmMvaG9va3MvdXNlQ2hyb21lRXh0ZW5zaW9uTm90aWZpY2F0aW9uLmpzJ1xuaW1wb3J0IHsgdXNlT2ZmaWNpYWxNYXJrZXRwbGFjZU5vdGlmaWNhdGlvbiB9IGZyb20gJ3NyYy9ob29rcy91c2VPZmZpY2lhbE1hcmtldHBsYWNlTm90aWZpY2F0aW9uLmpzJ1xuaW1wb3J0IHsgdXNlUHJvbXB0c0Zyb21DbGF1ZGVJbkNocm9tZSB9IGZyb20gJ3NyYy9ob29rcy91c2VQcm9tcHRzRnJvbUNsYXVkZUluQ2hyb21lLmpzJ1xuaW1wb3J0IHtcbiAgZ2V0VGlwVG9TaG93T25TcGlubmVyLFxuICByZWNvcmRTaG93blRpcCxcbn0gZnJvbSAnc3JjL3NlcnZpY2VzL3RpcHMvdGlwU2NoZWR1bGVyLmpzJ1xuaW1wb3J0IHR5cGUgeyBUaGVtZSB9IGZyb20gJ3NyYy91dGlscy90aGVtZS5qcydcbmltcG9ydCB7XG4gIGNoZWNrQW5kRGlzYWJsZUJ5cGFzc1Blcm1pc3Npb25zSWZOZWVkZWQsXG4gIGNoZWNrQW5kRGlzYWJsZUF1dG9Nb2RlSWZOZWVkZWQsXG4gIHVzZUtpY2tPZmZDaGVja0FuZERpc2FibGVCeXBhc3NQZXJtaXNzaW9uc0lmTmVlZGVkLFxuICB1c2VLaWNrT2ZmQ2hlY2tBbmREaXNhYmxlQXV0b01vZGVJZk5lZWRlZCxcbn0gZnJvbSAnc3JjL3V0aWxzL3Blcm1pc3Npb25zL2J5cGFzc1Blcm1pc3Npb25zS2lsbHN3aXRjaC5qcydcbmltcG9ydCB7IFNhbmRib3hNYW5hZ2VyIH0gZnJvbSAnc3JjL3V0aWxzL3NhbmRib3gvc2FuZGJveC1hZGFwdGVyLmpzJ1xuaW1wb3J0IHsgU0FOREJPWF9ORVRXT1JLX0FDQ0VTU19UT09MX05BTUUgfSBmcm9tICdzcmMvY2xpL3N0cnVjdHVyZWRJTy5qcydcbmltcG9ydCB7IHVzZUZpbGVIaXN0b3J5U25hcHNob3RJbml0IH0gZnJvbSAnc3JjL2hvb2tzL3VzZUZpbGVIaXN0b3J5U25hcHNob3RJbml0LmpzJ1xuaW1wb3J0IHsgU2FuZGJveFBlcm1pc3Npb25SZXF1ZXN0IH0gZnJvbSAnc3JjL2NvbXBvbmVudHMvcGVybWlzc2lvbnMvU2FuZGJveFBlcm1pc3Npb25SZXF1ZXN0LmpzJ1xuaW1wb3J0IHsgU2FuZGJveFZpb2xhdGlvbkV4cGFuZGVkVmlldyB9IGZyb20gJ3NyYy9jb21wb25lbnRzL1NhbmRib3hWaW9sYXRpb25FeHBhbmRlZFZpZXcuanMnXG5pbXBvcnQgeyB1c2VTZXR0aW5nc0Vycm9ycyB9IGZyb20gJ3NyYy9ob29rcy9ub3RpZnMvdXNlU2V0dGluZ3NFcnJvcnMuanMnXG5pbXBvcnQgeyB1c2VNY3BDb25uZWN0aXZpdHlTdGF0dXMgfSBmcm9tICdzcmMvaG9va3Mvbm90aWZzL3VzZU1jcENvbm5lY3Rpdml0eVN0YXR1cy5qcydcbmltcG9ydCB7IHVzZUF1dG9Nb2RlVW5hdmFpbGFibGVOb3RpZmljYXRpb24gfSBmcm9tICdzcmMvaG9va3Mvbm90aWZzL3VzZUF1dG9Nb2RlVW5hdmFpbGFibGVOb3RpZmljYXRpb24uanMnXG5pbXBvcnQgeyBBVVRPX01PREVfREVTQ1JJUFRJT04gfSBmcm9tICdzcmMvY29tcG9uZW50cy9BdXRvTW9kZU9wdEluRGlhbG9nLmpzJ1xuaW1wb3J0IHsgdXNlTHNwSW5pdGlhbGl6YXRpb25Ob3RpZmljYXRpb24gfSBmcm9tICdzcmMvaG9va3Mvbm90aWZzL3VzZUxzcEluaXRpYWxpemF0aW9uTm90aWZpY2F0aW9uLmpzJ1xuaW1wb3J0IHsgdXNlTHNwUGx1Z2luUmVjb21tZW5kYXRpb24gfSBmcm9tICdzcmMvaG9va3MvdXNlTHNwUGx1Z2luUmVjb21tZW5kYXRpb24uanMnXG5pbXBvcnQgeyBMc3BSZWNvbW1lbmRhdGlvbk1lbnUgfSBmcm9tICdzcmMvY29tcG9uZW50cy9Mc3BSZWNvbW1lbmRhdGlvbi9Mc3BSZWNvbW1lbmRhdGlvbk1lbnUuanMnXG5pbXBvcnQgeyB1c2VDbGF1ZGVDb2RlSGludFJlY29tbWVuZGF0aW9uIH0gZnJvbSAnc3JjL2hvb2tzL3VzZUNsYXVkZUNvZGVIaW50UmVjb21tZW5kYXRpb24uanMnXG5pbXBvcnQgeyBQbHVnaW5IaW50TWVudSB9IGZyb20gJ3NyYy9jb21wb25lbnRzL0NsYXVkZUNvZGVIaW50L1BsdWdpbkhpbnRNZW51LmpzJ1xuaW1wb3J0IHtcbiAgRGVza3RvcFVwc2VsbFN0YXJ0dXAsXG4gIHNob3VsZFNob3dEZXNrdG9wVXBzZWxsU3RhcnR1cCxcbn0gZnJvbSAnc3JjL2NvbXBvbmVudHMvRGVza3RvcFVwc2VsbC9EZXNrdG9wVXBzZWxsU3RhcnR1cC5qcydcbmltcG9ydCB7IHVzZVBsdWdpbkluc3RhbGxhdGlvblN0YXR1cyB9IGZyb20gJ3NyYy9ob29rcy9ub3RpZnMvdXNlUGx1Z2luSW5zdGFsbGF0aW9uU3RhdHVzLmpzJ1xuaW1wb3J0IHsgdXNlUGx1Z2luQXV0b3VwZGF0ZU5vdGlmaWNhdGlvbiB9IGZyb20gJ3NyYy9ob29rcy9ub3RpZnMvdXNlUGx1Z2luQXV0b3VwZGF0ZU5vdGlmaWNhdGlvbi5qcydcbmltcG9ydCB7IHBlcmZvcm1TdGFydHVwQ2hlY2tzIH0gZnJvbSAnc3JjL3V0aWxzL3BsdWdpbnMvcGVyZm9ybVN0YXJ0dXBDaGVja3MuanMnXG5pbXBvcnQgeyBVc2VyVGV4dE1lc3NhZ2UgfSBmcm9tICdzcmMvY29tcG9uZW50cy9tZXNzYWdlcy9Vc2VyVGV4dE1lc3NhZ2UuanMnXG5pbXBvcnQgeyBBd3NBdXRoU3RhdHVzQm94IH0gZnJvbSAnLi4vY29tcG9uZW50cy9Bd3NBdXRoU3RhdHVzQm94LmpzJ1xuaW1wb3J0IHsgdXNlUmF0ZUxpbWl0V2FybmluZ05vdGlmaWNhdGlvbiB9IGZyb20gJ3NyYy9ob29rcy9ub3RpZnMvdXNlUmF0ZUxpbWl0V2FybmluZ05vdGlmaWNhdGlvbi5qcydcbmltcG9ydCB7IHVzZURlcHJlY2F0aW9uV2FybmluZ05vdGlmaWNhdGlvbiB9IGZyb20gJ3NyYy9ob29rcy9ub3RpZnMvdXNlRGVwcmVjYXRpb25XYXJuaW5nTm90aWZpY2F0aW9uLmpzJ1xuaW1wb3J0IHsgdXNlTnBtRGVwcmVjYXRpb25Ob3RpZmljYXRpb24gfSBmcm9tICdzcmMvaG9va3Mvbm90aWZzL3VzZU5wbURlcHJlY2F0aW9uTm90aWZpY2F0aW9uLmpzJ1xuaW1wb3J0IHsgdXNlSURFU3RhdHVzSW5kaWNhdG9yIH0gZnJvbSAnc3JjL2hvb2tzL25vdGlmcy91c2VJREVTdGF0dXNJbmRpY2F0b3IuanMnXG5pbXBvcnQgeyB1c2VNb2RlbE1pZ3JhdGlvbk5vdGlmaWNhdGlvbnMgfSBmcm9tICdzcmMvaG9va3Mvbm90aWZzL3VzZU1vZGVsTWlncmF0aW9uTm90aWZpY2F0aW9ucy5qcydcbmltcG9ydCB7IHVzZUNhblN3aXRjaFRvRXhpc3RpbmdTdWJzY3JpcHRpb24gfSBmcm9tICdzcmMvaG9va3Mvbm90aWZzL3VzZUNhblN3aXRjaFRvRXhpc3RpbmdTdWJzY3JpcHRpb24uanMnXG5pbXBvcnQgeyB1c2VUZWFtbWF0ZUxpZmVjeWNsZU5vdGlmaWNhdGlvbiB9IGZyb20gJ3NyYy9ob29rcy9ub3RpZnMvdXNlVGVhbW1hdGVTaHV0ZG93bk5vdGlmaWNhdGlvbi5qcydcbmltcG9ydCB7IHVzZUZhc3RNb2RlTm90aWZpY2F0aW9uIH0gZnJvbSAnc3JjL2hvb2tzL25vdGlmcy91c2VGYXN0TW9kZU5vdGlmaWNhdGlvbi5qcydcbmltcG9ydCB7XG4gIEF1dG9SdW5Jc3N1ZU5vdGlmaWNhdGlvbixcbiAgc2hvdWxkQXV0b1J1bklzc3VlLFxuICBnZXRBdXRvUnVuSXNzdWVSZWFzb25UZXh0LFxuICBnZXRBdXRvUnVuQ29tbWFuZCxcbiAgdHlwZSBBdXRvUnVuSXNzdWVSZWFzb24sXG59IGZyb20gJy4uL3V0aWxzL2F1dG9SdW5Jc3N1ZS5qcydcbmltcG9ydCB0eXBlIHsgSG9va1Byb2dyZXNzIH0gZnJvbSAnLi4vdHlwZXMvaG9va3MuanMnXG5pbXBvcnQgeyBUdW5nc3RlbkxpdmVNb25pdG9yIH0gZnJvbSAnLi4vdG9vbHMvVHVuZ3N0ZW5Ub29sL1R1bmdzdGVuTGl2ZU1vbml0b3IuanMnXG4vKiBlc2xpbnQtZGlzYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tcmVxdWlyZS1pbXBvcnRzICovXG5jb25zdCBXZWJCcm93c2VyUGFuZWxNb2R1bGUgPSBmZWF0dXJlKCdXRUJfQlJPV1NFUl9UT09MJylcbiAgPyAocmVxdWlyZSgnLi4vdG9vbHMvV2ViQnJvd3NlclRvb2wvV2ViQnJvd3NlclBhbmVsLmpzJykgYXMgdHlwZW9mIGltcG9ydCgnLi4vdG9vbHMvV2ViQnJvd3NlclRvb2wvV2ViQnJvd3NlclBhbmVsLmpzJykpXG4gIDogbnVsbFxuLyogZXNsaW50LWVuYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tcmVxdWlyZS1pbXBvcnRzICovXG5pbXBvcnQgeyBJc3N1ZUZsYWdCYW5uZXIgfSBmcm9tICcuLi9jb21wb25lbnRzL1Byb21wdElucHV0L0lzc3VlRmxhZ0Jhbm5lci5qcydcbmltcG9ydCB7IHVzZUlzc3VlRmxhZ0Jhbm5lciB9IGZyb20gJy4uL2hvb2tzL3VzZUlzc3VlRmxhZ0Jhbm5lci5qcydcbmltcG9ydCB7XG4gIENvbXBhbmlvblNwcml0ZSxcbiAgQ29tcGFuaW9uRmxvYXRpbmdCdWJibGUsXG4gIE1JTl9DT0xTX0ZPUl9GVUxMX1NQUklURSxcbn0gZnJvbSAnLi4vYnVkZHkvQ29tcGFuaW9uU3ByaXRlLmpzJ1xuaW1wb3J0IHsgRGV2QmFyIH0gZnJvbSAnLi4vY29tcG9uZW50cy9EZXZCYXIuanMnXG4vLyBTZXNzaW9uIG1hbmFnZXIgcmVtb3ZlZCAtIHVzaW5nIEFwcFN0YXRlIG5vd1xuaW1wb3J0IHR5cGUgeyBSZW1vdGVTZXNzaW9uQ29uZmlnIH0gZnJvbSAnLi4vcmVtb3RlL1JlbW90ZVNlc3Npb25NYW5hZ2VyLmpzJ1xuaW1wb3J0IHsgUkVNT1RFX1NBRkVfQ09NTUFORFMgfSBmcm9tICcuLi9jb21tYW5kcy5qcydcbmltcG9ydCB0eXBlIHsgUmVtb3RlTWVzc2FnZUNvbnRlbnQgfSBmcm9tICcuLi91dGlscy90ZWxlcG9ydC9hcGkuanMnXG5pbXBvcnQge1xuICBGdWxsc2NyZWVuTGF5b3V0LFxuICB1c2VVbnNlZW5EaXZpZGVyLFxuICBjb21wdXRlVW5zZWVuRGl2aWRlcixcbn0gZnJvbSAnLi4vY29tcG9uZW50cy9GdWxsc2NyZWVuTGF5b3V0LmpzJ1xuaW1wb3J0IHtcbiAgaXNGdWxsc2NyZWVuRW52RW5hYmxlZCxcbiAgbWF5YmVHZXRUbXV4TW91c2VIaW50LFxuICBpc01vdXNlVHJhY2tpbmdFbmFibGVkLFxufSBmcm9tICcuLi91dGlscy9mdWxsc2NyZWVuLmpzJ1xuaW1wb3J0IHsgQWx0ZXJuYXRlU2NyZWVuIH0gZnJvbSAnLi4vaW5rL2NvbXBvbmVudHMvQWx0ZXJuYXRlU2NyZWVuLmpzJ1xuaW1wb3J0IHsgU2Nyb2xsS2V5YmluZGluZ0hhbmRsZXIgfSBmcm9tICcuLi9jb21wb25lbnRzL1Njcm9sbEtleWJpbmRpbmdIYW5kbGVyLmpzJ1xuaW1wb3J0IHtcbiAgdXNlTWVzc2FnZUFjdGlvbnMsXG4gIE1lc3NhZ2VBY3Rpb25zS2V5YmluZGluZ3MsXG4gIE1lc3NhZ2VBY3Rpb25zQmFyLFxuICB0eXBlIE1lc3NhZ2VBY3Rpb25zU3RhdGUsXG4gIHR5cGUgTWVzc2FnZUFjdGlvbnNOYXYsXG4gIHR5cGUgTWVzc2FnZUFjdGlvbkNhcHMsXG59IGZyb20gJy4uL2NvbXBvbmVudHMvbWVzc2FnZUFjdGlvbnMuanMnXG5pbXBvcnQgeyBzZXRDbGlwYm9hcmQgfSBmcm9tICcuLi9pbmsvdGVybWlvL29zYy5qcydcbmltcG9ydCB0eXBlIHsgU2Nyb2xsQm94SGFuZGxlIH0gZnJvbSAnLi4vaW5rL2NvbXBvbmVudHMvU2Nyb2xsQm94LmpzJ1xuaW1wb3J0IHtcbiAgY3JlYXRlQXR0YWNobWVudE1lc3NhZ2UsXG4gIGdldFF1ZXVlZENvbW1hbmRBdHRhY2htZW50cyxcbn0gZnJvbSAnLi4vdXRpbHMvYXR0YWNobWVudHMuanMnXG5cbi8vIFN0YWJsZSBlbXB0eSBhcnJheSBmb3IgaG9va3MgdGhhdCBhY2NlcHQgTUNQU2VydmVyQ29ubmVjdGlvbltdIOKAlCBhdm9pZHNcbi8vIGNyZWF0aW5nIGEgbmV3IFtdIGxpdGVyYWwgb24gZXZlcnkgcmVuZGVyIGluIHJlbW90ZSBtb2RlLCB3aGljaCB3b3VsZFxuLy8gY2F1c2UgdXNlRWZmZWN0IGRlcGVuZGVuY3kgY2hhbmdlcyBhbmQgaW5maW5pdGUgcmUtcmVuZGVyIGxvb3BzLlxuY29uc3QgRU1QVFlfTUNQX0NMSUVOVFM6IE1DUFNlcnZlckNvbm5lY3Rpb25bXSA9IFtdXG5cbi8vIFN0YWJsZSBzdHViIGZvciB1c2VBc3Npc3RhbnRIaXN0b3J5J3Mgbm9uLUtBSVJPUyBicmFuY2gg4oCUIGF2b2lkcyBhIG5ld1xuLy8gZnVuY3Rpb24gaWRlbnRpdHkgZWFjaCByZW5kZXIsIHdoaWNoIHdvdWxkIGJyZWFrIGNvbXBvc2VkT25TY3JvbGwncyBtZW1vLlxuY29uc3QgSElTVE9SWV9TVFVCID0geyBtYXliZUxvYWRPbGRlcjogKF86IFNjcm9sbEJveEhhbmRsZSkgPT4ge30gfVxuLy8gV2luZG93IGFmdGVyIGEgdXNlci1pbml0aWF0ZWQgc2Nyb2xsIGR1cmluZyB3aGljaCB0eXBlLWludG8tZW1wdHkgZG9lcyBOT1Rcbi8vIHJlcGluIHRvIGJvdHRvbS4gSm9zaCBSb3NlbidzIHdvcmtmbG93OiBDbGF1ZGUgZW1pdHMgbG9uZyBvdXRwdXQg4oaSIHNjcm9sbFxuLy8gdXAgdG8gcmVhZCB0aGUgc3RhcnQg4oaSIHN0YXJ0IHR5cGluZyDihpIgYmVmb3JlIHRoaXMgZml4LCBzbmFwcGVkIHRvIGJvdHRvbS5cbi8vIGh0dHBzOi8vYW50aHJvcGljLnNsYWNrLmNvbS9hcmNoaXZlcy9DMDdWQlNIVjdFVi9wMTc3MzU0NTQ0OTg3MTczOVxuY29uc3QgUkVDRU5UX1NDUk9MTF9SRVBJTl9XSU5ET1dfTVMgPSAzMDAwXG5cbi8vIFVzZSBMUlUgY2FjaGUgdG8gcHJldmVudCB1bmJvdW5kZWQgbWVtb3J5IGdyb3d0aFxuLy8gMTAwIGZpbGVzIHNob3VsZCBiZSBzdWZmaWNpZW50IGZvciBtb3N0IGNvZGluZyBzZXNzaW9ucyB3aGlsZSBwcmV2ZW50aW5nXG4vLyBtZW1vcnkgaXNzdWVzIHdoZW4gd29ya2luZyBhY3Jvc3MgbWFueSBmaWxlcyBpbiBsYXJnZSBwcm9qZWN0c1xuXG5mdW5jdGlvbiBtZWRpYW4odmFsdWVzOiBudW1iZXJbXSk6IG51bWJlciB7XG4gIGNvbnN0IHNvcnRlZCA9IFsuLi52YWx1ZXNdLnNvcnQoKGEsIGIpID0+IGEgLSBiKVxuICBjb25zdCBtaWQgPSBNYXRoLmZsb29yKHNvcnRlZC5sZW5ndGggLyAyKVxuICByZXR1cm4gc29ydGVkLmxlbmd0aCAlIDIgPT09IDBcbiAgICA/IE1hdGgucm91bmQoKHNvcnRlZFttaWQgLSAxXSEgKyBzb3J0ZWRbbWlkXSEpIC8gMilcbiAgICA6IHNvcnRlZFttaWRdIVxufVxuXG4vKipcbiAqIFNtYWxsIGNvbXBvbmVudCB0byBkaXNwbGF5IHRyYW5zY3JpcHQgbW9kZSBmb290ZXIgd2l0aCBkeW5hbWljIGtleWJpbmRpbmcuXG4gKiBNdXN0IGJlIHJlbmRlcmVkIGluc2lkZSBLZXliaW5kaW5nU2V0dXAgdG8gYWNjZXNzIGtleWJpbmRpbmcgY29udGV4dC5cbiAqL1xuZnVuY3Rpb24gVHJhbnNjcmlwdE1vZGVGb290ZXIoe1xuICBzaG93QWxsSW5UcmFuc2NyaXB0LFxuICB2aXJ0dWFsU2Nyb2xsLFxuICBzZWFyY2hCYWRnZSxcbiAgc3VwcHJlc3NTaG93QWxsID0gZmFsc2UsXG4gIHN0YXR1cyxcbn06IHtcbiAgc2hvd0FsbEluVHJhbnNjcmlwdDogYm9vbGVhblxuICB2aXJ0dWFsU2Nyb2xsOiBib29sZWFuXG4gIC8qKiBNaW5pbWFwIHdoaWxlIG5hdmlnYXRpbmcgYSBjbG9zZWQtYmFyIHNlYXJjaC4gU2hvd3Mgbi9OIGhpbnRzICtcbiAgICogIHJpZ2h0LWFsaWduZWQgY291bnQgaW5zdGVhZCBvZiBzY3JvbGwgaGludHMuICovXG4gIHNlYXJjaEJhZGdlPzogeyBjdXJyZW50OiBudW1iZXI7IGNvdW50OiBudW1iZXIgfVxuICAvKiogSGlkZSB0aGUgY3RybCtlIGhpbnQuIFRoZSBbIGR1bXAgcGF0aCBzaGFyZXMgdGhpcyBmb290ZXIgd2l0aFxuICAgKiAgZW52LW9wdGVkIGR1bXAgKENMQVVERV9DT0RFX05PX0ZMSUNLRVI9MCAvIERJU0FCTEVfVklSVFVBTF9TQ1JPTEw9MSksXG4gICAqICBidXQgY3RybCtlIG9ubHkgd29ya3MgaW4gdGhlIGVudiBjYXNlIOKAlCB1c2VHbG9iYWxLZXliaW5kaW5ncy50c3hcbiAgICogIGdhdGVzIG9uICF2aXJ0dWFsU2Nyb2xsQWN0aXZlIHdoaWNoIGlzIGVudi1kZXJpdmVkLCBkb2Vzbid0IGtub3dcbiAgICogIFsgaGFwcGVuZWQuICovXG4gIHN1cHByZXNzU2hvd0FsbD86IGJvb2xlYW5cbiAgLyoqIFRyYW5zaWVudCBzdGF0dXMgKHYtZm9yLWVkaXRvciBwcm9ncmVzcykuIE5vdGlmaWNhdGlvbnMgcmVuZGVyIGluc2lkZVxuICAgKiAgUHJvbXB0SW5wdXQgd2hpY2ggaXNuJ3QgbW91bnRlZCBpbiB0cmFuc2NyaXB0IOKAlCBhZGROb3RpZmljYXRpb24gcXVldWVzXG4gICAqICBidXQgbm90aGluZyBkcmF3cyBpdC4gKi9cbiAgc3RhdHVzPzogc3RyaW5nXG59KTogUmVhY3QuUmVhY3ROb2RlIHtcbiAgY29uc3QgdG9nZ2xlU2hvcnRjdXQgPSB1c2VTaG9ydGN1dERpc3BsYXkoXG4gICAgJ2FwcDp0b2dnbGVUcmFuc2NyaXB0JyxcbiAgICAnR2xvYmFsJyxcbiAgICAnY3RybCtvJyxcbiAgKVxuICBjb25zdCBzaG93QWxsU2hvcnRjdXQgPSB1c2VTaG9ydGN1dERpc3BsYXkoXG4gICAgJ3RyYW5zY3JpcHQ6dG9nZ2xlU2hvd0FsbCcsXG4gICAgJ1RyYW5zY3JpcHQnLFxuICAgICdjdHJsK2UnLFxuICApXG4gIHJldHVybiAoXG4gICAgPEJveFxuICAgICAgbm9TZWxlY3RcbiAgICAgIGFsaWduSXRlbXM9XCJjZW50ZXJcIlxuICAgICAgYWxpZ25TZWxmPVwiY2VudGVyXCJcbiAgICAgIGJvcmRlclRvcERpbUNvbG9yXG4gICAgICBib3JkZXJCb3R0b209e2ZhbHNlfVxuICAgICAgYm9yZGVyTGVmdD17ZmFsc2V9XG4gICAgICBib3JkZXJSaWdodD17ZmFsc2V9XG4gICAgICBib3JkZXJTdHlsZT1cInNpbmdsZVwiXG4gICAgICBtYXJnaW5Ub3A9ezF9XG4gICAgICBwYWRkaW5nTGVmdD17Mn1cbiAgICAgIHdpZHRoPVwiMTAwJVwiXG4gICAgPlxuICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgIFNob3dpbmcgZGV0YWlsZWQgdHJhbnNjcmlwdCDCtyB7dG9nZ2xlU2hvcnRjdXR9IHRvIHRvZ2dsZVxuICAgICAgICB7c2VhcmNoQmFkZ2VcbiAgICAgICAgICA/ICcgwrcgbi9OIHRvIG5hdmlnYXRlJ1xuICAgICAgICAgIDogdmlydHVhbFNjcm9sbFxuICAgICAgICAgICAgPyBgIMK3ICR7ZmlndXJlcy5hcnJvd1VwfSR7ZmlndXJlcy5hcnJvd0Rvd259IHNjcm9sbCDCtyBob21lL2VuZCB0b3AvYm90dG9tYFxuICAgICAgICAgICAgOiBzdXBwcmVzc1Nob3dBbGxcbiAgICAgICAgICAgICAgPyAnJ1xuICAgICAgICAgICAgICA6IGAgwrcgJHtzaG93QWxsU2hvcnRjdXR9IHRvICR7c2hvd0FsbEluVHJhbnNjcmlwdCA/ICdjb2xsYXBzZScgOiAnc2hvdyBhbGwnfWB9XG4gICAgICA8L1RleHQ+XG4gICAgICB7c3RhdHVzID8gKFxuICAgICAgICAvLyB2LWZvci1lZGl0b3IgcmVuZGVyIHByb2dyZXNzIOKAlCB0cmFuc2llbnQsIHByZWVtcHRzIHRoZSBzZWFyY2hcbiAgICAgICAgLy8gYmFkZ2Ugc2luY2UgdGhlIHVzZXIganVzdCBwcmVzc2VkIHYgYW5kIHdhbnRzIHRvIHNlZSB3aGF0J3NcbiAgICAgICAgLy8gaGFwcGVuaW5nLiBDbGVhcnMgYWZ0ZXIgNHMuXG4gICAgICAgIDw+XG4gICAgICAgICAgPEJveCBmbGV4R3Jvdz17MX0gLz5cbiAgICAgICAgICA8VGV4dD57c3RhdHVzfSA8L1RleHQ+XG4gICAgICAgIDwvPlxuICAgICAgKSA6IHNlYXJjaEJhZGdlID8gKFxuICAgICAgICAvLyBFbmdpbmUtY291bnRlZCDigJQgY2xvc2UgZW5vdWdoIGZvciBhIHJvdWdoIGxvY2F0aW9uIGhpbnQuIE1heVxuICAgICAgICAvLyBkcmlmdCBmcm9tIHJlbmRlci1jb3VudCBmb3IgZ2hvc3QvcGhhbnRvbSBtZXNzYWdlcy5cbiAgICAgICAgPD5cbiAgICAgICAgICA8Qm94IGZsZXhHcm93PXsxfSAvPlxuICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPlxuICAgICAgICAgICAge3NlYXJjaEJhZGdlLmN1cnJlbnR9L3tzZWFyY2hCYWRnZS5jb3VudH1cbiAgICAgICAgICAgIHsnICAnfVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC8+XG4gICAgICApIDogbnVsbH1cbiAgICA8L0JveD5cbiAgKVxufVxuXG4vKiogbGVzcy1zdHlsZSAvIGJhci4gMS1yb3csIHNhbWUgYm9yZGVyLXRvcCBzdHlsaW5nIGFzIFRyYW5zY3JpcHRNb2RlRm9vdGVyXG4gKiAgc28gc3dhcHBpbmcgdGhlbSBpbiB0aGUgYm90dG9tIHNsb3QgZG9lc24ndCBzaGlmdCBTY3JvbGxCb3ggaGVpZ2h0LlxuICogIHVzZVNlYXJjaElucHV0IGhhbmRsZXMgcmVhZGxpbmUgZWRpdGluZzsgd2UgcmVwb3J0IHF1ZXJ5IGNoYW5nZXMgYW5kXG4gKiAgcmVuZGVyIHRoZSBjb3VudGVyLiBJbmNyZW1lbnRhbCDigJQgcmUtc2VhcmNoICsgaGlnaGxpZ2h0IHBlciBrZXlzdHJva2UuICovXG5mdW5jdGlvbiBUcmFuc2NyaXB0U2VhcmNoQmFyKHtcbiAganVtcFJlZixcbiAgY291bnQsXG4gIGN1cnJlbnQsXG4gIG9uQ2xvc2UsXG4gIG9uQ2FuY2VsLFxuICBzZXRIaWdobGlnaHQsXG4gIGluaXRpYWxRdWVyeSxcbn06IHtcbiAganVtcFJlZjogUmVmT2JqZWN0PEp1bXBIYW5kbGUgfCBudWxsPlxuICBjb3VudDogbnVtYmVyXG4gIGN1cnJlbnQ6IG51bWJlclxuICAvKiogRW50ZXIg4oCUIGNvbW1pdC4gUXVlcnkgcGVyc2lzdHMgZm9yIG4vTi4gKi9cbiAgb25DbG9zZTogKGxhc3RRdWVyeTogc3RyaW5nKSA9PiB2b2lkXG4gIC8qKiBFc2MvY3RybCtjL2N0cmwrZyDigJQgdW5kbyB0byBwcmUtLyBzdGF0ZS4gKi9cbiAgb25DYW5jZWw6ICgpID0+IHZvaWRcbiAgc2V0SGlnaGxpZ2h0OiAocXVlcnk6IHN0cmluZykgPT4gdm9pZFxuICAvLyBTZWVkIHdpdGggdGhlIHByZXZpb3VzIHF1ZXJ5IChsZXNzOiAvIHNob3dzIGxhc3QgcGF0dGVybikuIE1vdW50LWZpcmVcbiAgLy8gb2YgdGhlIGVmZmVjdCByZS1zY2FucyB3aXRoIHRoZSBzYW1lIHF1ZXJ5IOKAlCBpZGVtcG90ZW50IChzYW1lIG1hdGNoZXMsXG4gIC8vIG5lYXJlc3QtcHRyLCBzYW1lIGhpZ2hsaWdodHMpLiBVc2VyIGNhbiBlZGl0IG9yIGNsZWFyLlxuICBpbml0aWFsUXVlcnk6IHN0cmluZ1xufSk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IHsgcXVlcnksIGN1cnNvck9mZnNldCB9ID0gdXNlU2VhcmNoSW5wdXQoe1xuICAgIGlzQWN0aXZlOiB0cnVlLFxuICAgIGluaXRpYWxRdWVyeSxcbiAgICBvbkV4aXQ6ICgpID0+IG9uQ2xvc2UocXVlcnkpLFxuICAgIG9uQ2FuY2VsLFxuICB9KVxuICAvLyBJbmRleCB3YXJtLXVwIHJ1bnMgYmVmb3JlIHRoZSBxdWVyeSBlZmZlY3Qgc28gaXQgbWVhc3VyZXMgdGhlIHJlYWxcbiAgLy8gY29zdCDigJQgb3RoZXJ3aXNlIHNldFNlYXJjaFF1ZXJ5IGZpbGxzIHRoZSBjYWNoZSBmaXJzdCBhbmQgd2FybVxuICAvLyByZXBvcnRzIH4wbXMgd2hpbGUgdGhlIHVzZXIgZmVsdCB0aGUgYWN0dWFsIGxhZy5cbiAgLy8gRmlyc3QgLyBpbiBhIHRyYW5zY3JpcHQgc2Vzc2lvbiBwYXlzIHRoZSBleHRyYWN0U2VhcmNoVGV4dCBjb3N0LlxuICAvLyBTdWJzZXF1ZW50IC8gcmV0dXJuIDAgaW1tZWRpYXRlbHkgKGluZGV4V2FybWVkIHJlZiBpbiBWTUwpLlxuICAvLyBUcmFuc2NyaXB0IGlzIGZyb3plbiBhdCBjdHJsK28gc28gdGhlIGNhY2hlIHN0YXlzIHZhbGlkLlxuICAvLyBJbml0aWFsICdidWlsZGluZycgc28gd2FybURvbmUgaXMgZmFsc2Ugb24gbW91bnQg4oCUIHRoZSBbcXVlcnldIGVmZmVjdFxuICAvLyB3YWl0cyBmb3IgdGhlIHdhcm0gZWZmZWN0J3MgZmlyc3QgcmVzb2x2ZSBpbnN0ZWFkIG9mIHJhY2luZyBpdC4gV2l0aFxuICAvLyBudWxsIGluaXRpYWwsIHdhcm1Eb25lIHdvdWxkIGJlIHRydWUgb24gbW91bnQg4oaSIFtxdWVyeV0gZmlyZXMg4oaSXG4gIC8vIHNldFNlYXJjaFF1ZXJ5IGZpbGxzIGNhY2hlIOKGkiB3YXJtIHJlcG9ydHMgfjBtcyB3aGlsZSB0aGUgdXNlciBmZWx0XG4gIC8vIHRoZSByZWFsIGxhZy5cbiAgY29uc3QgW2luZGV4U3RhdHVzLCBzZXRJbmRleFN0YXR1c10gPSBSZWFjdC51c2VTdGF0ZTxcbiAgICAnYnVpbGRpbmcnIHwgeyBtczogbnVtYmVyIH0gfCBudWxsXG4gID4oJ2J1aWxkaW5nJylcbiAgUmVhY3QudXNlRWZmZWN0KCgpID0+IHtcbiAgICBsZXQgYWxpdmUgPSB0cnVlXG4gICAgY29uc3Qgd2FybSA9IGp1bXBSZWYuY3VycmVudD8ud2FybVNlYXJjaEluZGV4XG4gICAgaWYgKCF3YXJtKSB7XG4gICAgICBzZXRJbmRleFN0YXR1cyhudWxsKSAvLyBWTUwgbm90IG1vdW50ZWQgeWV0IOKAlCByYXJlLCBza2lwIGluZGljYXRvclxuICAgICAgcmV0dXJuXG4gICAgfVxuICAgIHNldEluZGV4U3RhdHVzKCdidWlsZGluZycpXG4gICAgd2FybSgpLnRoZW4obXMgPT4ge1xuICAgICAgaWYgKCFhbGl2ZSkgcmV0dXJuXG4gICAgICAvLyA8MjBtcyA9IGltcGVyY2VwdGlibGUuIE5vIHBvaW50IHNob3dpbmcgXCJpbmRleGVkIGluIDNtc1wiLlxuICAgICAgaWYgKG1zIDwgMjApIHtcbiAgICAgICAgc2V0SW5kZXhTdGF0dXMobnVsbClcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHNldEluZGV4U3RhdHVzKHsgbXMgfSlcbiAgICAgICAgc2V0VGltZW91dCgoKSA9PiBhbGl2ZSAmJiBzZXRJbmRleFN0YXR1cyhudWxsKSwgMjAwMClcbiAgICAgIH1cbiAgICB9KVxuICAgIHJldHVybiAoKSA9PiB7XG4gICAgICBhbGl2ZSA9IGZhbHNlXG4gICAgfVxuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSByZWFjdC1ob29rcy9leGhhdXN0aXZlLWRlcHNcbiAgfSwgW10pIC8vIG1vdW50LW9ubHk6IGJhciBvcGVucyBvbmNlIHBlciAvXG4gIC8vIEdhdGUgdGhlIHF1ZXJ5IGVmZmVjdCBvbiB3YXJtIGNvbXBsZXRpb24uIHNldEhpZ2hsaWdodCBzdGF5cyBpbnN0YW50XG4gIC8vIChzY3JlZW4tc3BhY2Ugb3ZlcmxheSwgbm8gaW5kZXhpbmcpLiBzZXRTZWFyY2hRdWVyeSAodGhlIHNjYW4pIHdhaXRzLlxuICBjb25zdCB3YXJtRG9uZSA9IGluZGV4U3RhdHVzICE9PSAnYnVpbGRpbmcnXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKCF3YXJtRG9uZSkgcmV0dXJuXG4gICAganVtcFJlZi5jdXJyZW50Py5zZXRTZWFyY2hRdWVyeShxdWVyeSlcbiAgICBzZXRIaWdobGlnaHQocXVlcnkpXG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIHJlYWN0LWhvb2tzL2V4aGF1c3RpdmUtZGVwc1xuICB9LCBbcXVlcnksIHdhcm1Eb25lXSlcbiAgY29uc3Qgb2ZmID0gY3Vyc29yT2Zmc2V0XG4gIGNvbnN0IGN1cnNvckNoYXIgPSBvZmYgPCBxdWVyeS5sZW5ndGggPyBxdWVyeVtvZmZdIDogJyAnXG4gIHJldHVybiAoXG4gICAgPEJveFxuICAgICAgYm9yZGVyVG9wRGltQ29sb3JcbiAgICAgIGJvcmRlckJvdHRvbT17ZmFsc2V9XG4gICAgICBib3JkZXJMZWZ0PXtmYWxzZX1cbiAgICAgIGJvcmRlclJpZ2h0PXtmYWxzZX1cbiAgICAgIGJvcmRlclN0eWxlPVwic2luZ2xlXCJcbiAgICAgIG1hcmdpblRvcD17MX1cbiAgICAgIHBhZGRpbmdMZWZ0PXsyfVxuICAgICAgd2lkdGg9XCIxMDAlXCJcbiAgICAgIC8vIGFwcGx5U2VhcmNoSGlnaGxpZ2h0IHNjYW5zIHRoZSB3aG9sZSBzY3JlZW4gYnVmZmVyLiBUaGUgcXVlcnlcbiAgICAgIC8vIHRleHQgcmVuZGVyZWQgaGVyZSBJUyBvbiBzY3JlZW4g4oCUIC9mb28gbWF0Y2hlcyBpdHMgb3duICdmb28nIGluXG4gICAgICAvLyB0aGUgYmFyLiBXaXRoIG5vIGNvbnRlbnQgbWF0Y2hlcyB0aGF0J3MgdGhlIE9OTFkgdmlzaWJsZSBtYXRjaCDihpJcbiAgICAgIC8vIGdldHMgQ1VSUkVOVCDihpIgdW5kZXJsaW5lZC4gbm9TZWxlY3QgbWFrZXMgc2VhcmNoSGlnaGxpZ2h0LnRzOjc2XG4gICAgICAvLyBza2lwIHRoZXNlIGNlbGxzIChzYW1lIGV4Y2x1c2lvbiBhcyBndXR0ZXJzKS4gWW91IGNhbid0IHRleHQtXG4gICAgICAvLyBzZWxlY3QgdGhlIGJhciBlaXRoZXI7IGl0J3MgdHJhbnNpZW50IGNocm9tZSwgZmluZS5cbiAgICAgIG5vU2VsZWN0XG4gICAgPlxuICAgICAgPFRleHQ+LzwvVGV4dD5cbiAgICAgIDxUZXh0PntxdWVyeS5zbGljZSgwLCBvZmYpfTwvVGV4dD5cbiAgICAgIDxUZXh0IGludmVyc2U+e2N1cnNvckNoYXJ9PC9UZXh0PlxuICAgICAge29mZiA8IHF1ZXJ5Lmxlbmd0aCAmJiA8VGV4dD57cXVlcnkuc2xpY2Uob2ZmICsgMSl9PC9UZXh0Pn1cbiAgICAgIDxCb3ggZmxleEdyb3c9ezF9IC8+XG4gICAgICB7aW5kZXhTdGF0dXMgPT09ICdidWlsZGluZycgPyAoXG4gICAgICAgIDxUZXh0IGRpbUNvbG9yPmluZGV4aW5n4oCmIDwvVGV4dD5cbiAgICAgICkgOiBpbmRleFN0YXR1cyA/IChcbiAgICAgICAgPFRleHQgZGltQ29sb3I+aW5kZXhlZCBpbiB7aW5kZXhTdGF0dXMubXN9bXMgPC9UZXh0PlxuICAgICAgKSA6IGNvdW50ID09PSAwICYmIHF1ZXJ5ID8gKFxuICAgICAgICA8VGV4dCBjb2xvcj1cImVycm9yXCI+bm8gbWF0Y2hlcyA8L1RleHQ+XG4gICAgICApIDogY291bnQgPiAwID8gKFxuICAgICAgICAvLyBFbmdpbmUtY291bnRlZCAoaW5kZXhPZiBvbiBleHRyYWN0U2VhcmNoVGV4dCkuIE1heSBkcmlmdCBmcm9tXG4gICAgICAgIC8vIHJlbmRlci1jb3VudCBmb3IgZ2hvc3QvcGhhbnRvbSBtZXNzYWdlcyDigJQgYmFkZ2UgaXMgYSByb3VnaFxuICAgICAgICAvLyBsb2NhdGlvbiBoaW50LiBzY2FuRWxlbWVudCBnaXZlcyBleGFjdCBwZXItbWVzc2FnZSBwb3NpdGlvbnNcbiAgICAgICAgLy8gYnV0IGNvdW50aW5nIEFMTCB3b3VsZCBjb3N0IH4xLTNtcyDDlyBtYXRjaGVkLW1lc3NhZ2VzLlxuICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICB7Y3VycmVudH0ve2NvdW50fVxuICAgICAgICAgIHsnICAnfVxuICAgICAgICA8L1RleHQ+XG4gICAgICApIDogbnVsbH1cbiAgICA8L0JveD5cbiAgKVxufVxuXG5jb25zdCBUSVRMRV9BTklNQVRJT05fRlJBTUVTID0gWyfioIInLCAn4qCQJ11cbmNvbnN0IFRJVExFX1NUQVRJQ19QUkVGSVggPSAn4pyzJ1xuY29uc3QgVElUTEVfQU5JTUFUSU9OX0lOVEVSVkFMX01TID0gOTYwXG5cbi8qKlxuICogU2V0cyB0aGUgdGVybWluYWwgdGFiIHRpdGxlLCB3aXRoIGFuIGFuaW1hdGVkIHByZWZpeCBnbHlwaCB3aGlsZSBhIHF1ZXJ5XG4gKiBpcyBydW5uaW5nLiBJc29sYXRlZCBmcm9tIFJFUEwgc28gdGhlIDk2MG1zIGFuaW1hdGlvbiB0aWNrIHJlLXJlbmRlcnMgb25seVxuICogdGhpcyBsZWFmIGNvbXBvbmVudCAod2hpY2ggcmV0dXJucyBudWxsIOKAlCBwdXJlIHNpZGUtZWZmZWN0KSBpbnN0ZWFkIG9mIHRoZVxuICogZW50aXJlIFJFUEwgdHJlZS4gQmVmb3JlIGV4dHJhY3Rpb24sIHRoZSB0aWNrIHdhcyB+MSBSRVBMIHJlbmRlci9zZWMgZm9yXG4gKiB0aGUgZHVyYXRpb24gb2YgZXZlcnkgdHVybiwgZHJhZ2dpbmcgUHJvbXB0SW5wdXQgYW5kIGZyaWVuZHMgYWxvbmcuXG4gKi9cbmZ1bmN0aW9uIEFuaW1hdGVkVGVybWluYWxUaXRsZSh7XG4gIGlzQW5pbWF0aW5nLFxuICB0aXRsZSxcbiAgZGlzYWJsZWQsXG4gIG5vUHJlZml4LFxufToge1xuICBpc0FuaW1hdGluZzogYm9vbGVhblxuICB0aXRsZTogc3RyaW5nXG4gIGRpc2FibGVkOiBib29sZWFuXG4gIG5vUHJlZml4OiBib29sZWFuXG59KTogbnVsbCB7XG4gIGNvbnN0IHRlcm1pbmFsRm9jdXNlZCA9IHVzZVRlcm1pbmFsRm9jdXMoKVxuICBjb25zdCBbZnJhbWUsIHNldEZyYW1lXSA9IHVzZVN0YXRlKDApXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKGRpc2FibGVkIHx8IG5vUHJlZml4IHx8ICFpc0FuaW1hdGluZyB8fCAhdGVybWluYWxGb2N1c2VkKSByZXR1cm5cbiAgICBjb25zdCBpbnRlcnZhbCA9IHNldEludGVydmFsKFxuICAgICAgc2V0RnJhbWUgPT4gc2V0RnJhbWUoZiA9PiAoZiArIDEpICUgVElUTEVfQU5JTUFUSU9OX0ZSQU1FUy5sZW5ndGgpLFxuICAgICAgVElUTEVfQU5JTUFUSU9OX0lOVEVSVkFMX01TLFxuICAgICAgc2V0RnJhbWUsXG4gICAgKVxuICAgIHJldHVybiAoKSA9PiBjbGVhckludGVydmFsKGludGVydmFsKVxuICB9LCBbZGlzYWJsZWQsIG5vUHJlZml4LCBpc0FuaW1hdGluZywgdGVybWluYWxGb2N1c2VkXSlcbiAgY29uc3QgcHJlZml4ID0gaXNBbmltYXRpbmdcbiAgICA/IChUSVRMRV9BTklNQVRJT05fRlJBTUVTW2ZyYW1lXSA/PyBUSVRMRV9TVEFUSUNfUFJFRklYKVxuICAgIDogVElUTEVfU1RBVElDX1BSRUZJWFxuICB1c2VUZXJtaW5hbFRpdGxlKGRpc2FibGVkID8gbnVsbCA6IG5vUHJlZml4ID8gdGl0bGUgOiBgJHtwcmVmaXh9ICR7dGl0bGV9YClcbiAgcmV0dXJuIG51bGxcbn1cblxuZXhwb3J0IHR5cGUgUHJvcHMgPSB7XG4gIGNvbW1hbmRzOiBDb21tYW5kW11cbiAgZGVidWc6IGJvb2xlYW5cbiAgaW5pdGlhbFRvb2xzOiBUb29sW11cbiAgLy8gSW5pdGlhbCBtZXNzYWdlcyB0byBwb3B1bGF0ZSB0aGUgUkVQTCB3aXRoXG4gIGluaXRpYWxNZXNzYWdlcz86IE1lc3NhZ2VUeXBlW11cbiAgLy8gRGVmZXJyZWQgaG9vayBtZXNzYWdlcyBwcm9taXNlIOKAlCBSRVBMIHJlbmRlcnMgaW1tZWRpYXRlbHkgYW5kIGluamVjdHNcbiAgLy8gaG9vayBtZXNzYWdlcyB3aGVuIHRoZXkgcmVzb2x2ZS4gQXdhaXRlZCBiZWZvcmUgdGhlIGZpcnN0IEFQSSBjYWxsLlxuICBwZW5kaW5nSG9va01lc3NhZ2VzPzogUHJvbWlzZTxIb29rUmVzdWx0TWVzc2FnZVtdPlxuICBpbml0aWFsRmlsZUhpc3RvcnlTbmFwc2hvdHM/OiBGaWxlSGlzdG9yeVNuYXBzaG90W11cbiAgLy8gQ29udGVudC1yZXBsYWNlbWVudCByZWNvcmRzIGZyb20gYSByZXN1bWVkIHNlc3Npb24ncyB0cmFuc2NyaXB0IOKAlCB1c2VkIHRvXG4gIC8vIHJlY29uc3RydWN0IGNvbnRlbnRSZXBsYWNlbWVudFN0YXRlIHNvIHRoZSBzYW1lIHJlc3VsdHMgYXJlIHJlLXJlcGxhY2VkXG4gIGluaXRpYWxDb250ZW50UmVwbGFjZW1lbnRzPzogQ29udGVudFJlcGxhY2VtZW50UmVjb3JkW11cbiAgLy8gSW5pdGlhbCBhZ2VudCBjb250ZXh0IGZvciBzZXNzaW9uIHJlc3VtZSAobmFtZS9jb2xvciBzZXQgdmlhIC9yZW5hbWUgb3IgL2NvbG9yKVxuICBpbml0aWFsQWdlbnROYW1lPzogc3RyaW5nXG4gIGluaXRpYWxBZ2VudENvbG9yPzogQWdlbnRDb2xvck5hbWVcbiAgbWNwQ2xpZW50cz86IE1DUFNlcnZlckNvbm5lY3Rpb25bXVxuICBkeW5hbWljTWNwQ29uZmlnPzogUmVjb3JkPHN0cmluZywgU2NvcGVkTWNwU2VydmVyQ29uZmlnPlxuICBhdXRvQ29ubmVjdElkZUZsYWc/OiBib29sZWFuXG4gIHN0cmljdE1jcENvbmZpZz86IGJvb2xlYW5cbiAgc3lzdGVtUHJvbXB0Pzogc3RyaW5nXG4gIGFwcGVuZFN5c3RlbVByb21wdD86IHN0cmluZ1xuICAvLyBPcHRpb25hbCBjYWxsYmFjayBpbnZva2VkIGJlZm9yZSBxdWVyeSBleGVjdXRpb25cbiAgLy8gQ2FsbGVkIGFmdGVyIHVzZXIgbWVzc2FnZSBpcyBhZGRlZCB0byBjb252ZXJzYXRpb24gYnV0IGJlZm9yZSBBUEkgY2FsbFxuICAvLyBSZXR1cm4gZmFsc2UgdG8gcHJldmVudCBxdWVyeSBleGVjdXRpb25cbiAgb25CZWZvcmVRdWVyeT86IChcbiAgICBpbnB1dDogc3RyaW5nLFxuICAgIG5ld01lc3NhZ2VzOiBNZXNzYWdlVHlwZVtdLFxuICApID0+IFByb21pc2U8Ym9vbGVhbj5cbiAgLy8gT3B0aW9uYWwgY2FsbGJhY2sgd2hlbiBhIHR1cm4gY29tcGxldGVzIChtb2RlbCBmaW5pc2hlcyByZXNwb25kaW5nKVxuICBvblR1cm5Db21wbGV0ZT86IChtZXNzYWdlczogTWVzc2FnZVR5cGVbXSkgPT4gdm9pZCB8IFByb21pc2U8dm9pZD5cbiAgLy8gV2hlbiB0cnVlLCBkaXNhYmxlcyBSRVBMIGlucHV0IChoaWRlcyBwcm9tcHQgYW5kIHByZXZlbnRzIG1lc3NhZ2Ugc2VsZWN0b3IpXG4gIGRpc2FibGVkPzogYm9vbGVhblxuICAvLyBPcHRpb25hbCBhZ2VudCBkZWZpbml0aW9uIHRvIHVzZSBmb3IgdGhlIG1haW4gdGhyZWFkXG4gIG1haW5UaHJlYWRBZ2VudERlZmluaXRpb24/OiBBZ2VudERlZmluaXRpb25cbiAgLy8gV2hlbiB0cnVlLCBkaXNhYmxlcyBhbGwgc2xhc2ggY29tbWFuZHNcbiAgZGlzYWJsZVNsYXNoQ29tbWFuZHM/OiBib29sZWFuXG4gIC8vIFRhc2sgbGlzdCBpZDogd2hlbiBzZXQsIGVuYWJsZXMgdGFza3MgbW9kZSB0aGF0IHdhdGNoZXMgYSB0YXNrIGxpc3QgYW5kIGF1dG8tcHJvY2Vzc2VzIHRhc2tzLlxuICB0YXNrTGlzdElkPzogc3RyaW5nXG4gIC8vIFJlbW90ZSBzZXNzaW9uIGNvbmZpZyBmb3IgLS1yZW1vdGUgbW9kZSAodXNlcyBDQ1IgYXMgZXhlY3V0aW9uIGVuZ2luZSlcbiAgcmVtb3RlU2Vzc2lvbkNvbmZpZz86IFJlbW90ZVNlc3Npb25Db25maWdcbiAgLy8gRGlyZWN0IGNvbm5lY3QgY29uZmlnIGZvciBgY2xhdWRlIGNvbm5lY3RgIG1vZGUgKGNvbm5lY3RzIHRvIGEgY2xhdWRlIHNlcnZlcilcbiAgZGlyZWN0Q29ubmVjdENvbmZpZz86IERpcmVjdENvbm5lY3RDb25maWdcbiAgLy8gU1NIIHNlc3Npb24gZm9yIGBjbGF1ZGUgc3NoYCBtb2RlIChsb2NhbCBSRVBMLCByZW1vdGUgdG9vbHMgb3ZlciBzc2gpXG4gIHNzaFNlc3Npb24/OiBTU0hTZXNzaW9uXG4gIC8vIFRoaW5raW5nIGNvbmZpZ3VyYXRpb24gdG8gdXNlIHdoZW4gdGhpbmtpbmcgaXMgZW5hYmxlZFxuICB0aGlua2luZ0NvbmZpZzogVGhpbmtpbmdDb25maWdcbn1cblxuZXhwb3J0IHR5cGUgU2NyZWVuID0gJ3Byb21wdCcgfCAndHJhbnNjcmlwdCdcblxuZXhwb3J0IGZ1bmN0aW9uIFJFUEwoe1xuICBjb21tYW5kczogaW5pdGlhbENvbW1hbmRzLFxuICBkZWJ1ZyxcbiAgaW5pdGlhbFRvb2xzLFxuICBpbml0aWFsTWVzc2FnZXMsXG4gIHBlbmRpbmdIb29rTWVzc2FnZXMsXG4gIGluaXRpYWxGaWxlSGlzdG9yeVNuYXBzaG90cyxcbiAgaW5pdGlhbENvbnRlbnRSZXBsYWNlbWVudHMsXG4gIGluaXRpYWxBZ2VudE5hbWUsXG4gIGluaXRpYWxBZ2VudENvbG9yLFxuICBtY3BDbGllbnRzOiBpbml0aWFsTWNwQ2xpZW50cyxcbiAgZHluYW1pY01jcENvbmZpZzogaW5pdGlhbER5bmFtaWNNY3BDb25maWcsXG4gIGF1dG9Db25uZWN0SWRlRmxhZyxcbiAgc3RyaWN0TWNwQ29uZmlnID0gZmFsc2UsXG4gIHN5c3RlbVByb21wdDogY3VzdG9tU3lzdGVtUHJvbXB0LFxuICBhcHBlbmRTeXN0ZW1Qcm9tcHQsXG4gIG9uQmVmb3JlUXVlcnksXG4gIG9uVHVybkNvbXBsZXRlLFxuICBkaXNhYmxlZCA9IGZhbHNlLFxuICBtYWluVGhyZWFkQWdlbnREZWZpbml0aW9uOiBpbml0aWFsTWFpblRocmVhZEFnZW50RGVmaW5pdGlvbixcbiAgZGlzYWJsZVNsYXNoQ29tbWFuZHMgPSBmYWxzZSxcbiAgdGFza0xpc3RJZCxcbiAgcmVtb3RlU2Vzc2lvbkNvbmZpZyxcbiAgZGlyZWN0Q29ubmVjdENvbmZpZyxcbiAgc3NoU2Vzc2lvbixcbiAgdGhpbmtpbmdDb25maWcsXG59OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IGlzUmVtb3RlU2Vzc2lvbiA9ICEhcmVtb3RlU2Vzc2lvbkNvbmZpZ1xuXG4gIC8vIEVudi12YXIgZ2F0ZXMgaG9pc3RlZCB0byBtb3VudC10aW1lIOKAlCBpc0VudlRydXRoeSBkb2VzIHRvTG93ZXJDYXNlK3RyaW0rXG4gIC8vIGluY2x1ZGVzLCBhbmQgdGhlc2Ugd2VyZSBvbiB0aGUgcmVuZGVyIHBhdGggKGhvdCBkdXJpbmcgUGFnZVVwIHNwYW0pLlxuICBjb25zdCB0aXRsZURpc2FibGVkID0gdXNlTWVtbyhcbiAgICAoKSA9PiBpc0VudlRydXRoeShwcm9jZXNzLmVudi5DTEFVREVfQ09ERV9ESVNBQkxFX1RFUk1JTkFMX1RJVExFKSxcbiAgICBbXSxcbiAgKVxuICBjb25zdCBtb3JlUmlnaHRFbmFibGVkID0gdXNlTWVtbyhcbiAgICAoKSA9PlxuICAgICAgXCJleHRlcm5hbFwiID09PSAnYW50JyAmJlxuICAgICAgaXNFbnZUcnV0aHkocHJvY2Vzcy5lbnYuQ0xBVURFX01PUkVSSUdIVCksXG4gICAgW10sXG4gIClcbiAgY29uc3QgZGlzYWJsZVZpcnR1YWxTY3JvbGwgPSB1c2VNZW1vKFxuICAgICgpID0+IGlzRW52VHJ1dGh5KHByb2Nlc3MuZW52LkNMQVVERV9DT0RFX0RJU0FCTEVfVklSVFVBTF9TQ1JPTEwpLFxuICAgIFtdLFxuICApXG4gIGNvbnN0IGRpc2FibGVNZXNzYWdlQWN0aW9ucyA9IGZlYXR1cmUoJ01FU1NBR0VfQUNUSU9OUycpXG4gICAgPyAvLyBiaW9tZS1pZ25vcmUgbGludC9jb3JyZWN0bmVzcy91c2VIb29rQXRUb3BMZXZlbDogZmVhdHVyZSgpIGlzIGEgY29tcGlsZS10aW1lIGNvbnN0YW50XG4gICAgICB1c2VNZW1vKFxuICAgICAgICAoKSA9PiBpc0VudlRydXRoeShwcm9jZXNzLmVudi5DTEFVREVfQ09ERV9ESVNBQkxFX01FU1NBR0VfQUNUSU9OUyksXG4gICAgICAgIFtdLFxuICAgICAgKVxuICAgIDogZmFsc2VcblxuICAvLyBMb2cgUkVQTCBtb3VudC91bm1vdW50IGxpZmVjeWNsZVxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGxvZ0ZvckRlYnVnZ2luZyhgW1JFUEw6bW91bnRdIFJFUEwgbW91bnRlZCwgZGlzYWJsZWQ9JHtkaXNhYmxlZH1gKVxuICAgIHJldHVybiAoKSA9PiBsb2dGb3JEZWJ1Z2dpbmcoYFtSRVBMOnVubW91bnRdIFJFUEwgdW5tb3VudGluZ2ApXG4gIH0sIFtkaXNhYmxlZF0pXG5cbiAgLy8gQWdlbnQgZGVmaW5pdGlvbiBpcyBzdGF0ZSBzbyAvcmVzdW1lIGNhbiB1cGRhdGUgaXQgbWlkLXNlc3Npb25cbiAgY29uc3QgW21haW5UaHJlYWRBZ2VudERlZmluaXRpb24sIHNldE1haW5UaHJlYWRBZ2VudERlZmluaXRpb25dID0gdXNlU3RhdGUoXG4gICAgaW5pdGlhbE1haW5UaHJlYWRBZ2VudERlZmluaXRpb24sXG4gIClcblxuICBjb25zdCB0b29sUGVybWlzc2lvbkNvbnRleHQgPSB1c2VBcHBTdGF0ZShzID0+IHMudG9vbFBlcm1pc3Npb25Db250ZXh0KVxuICBjb25zdCB2ZXJib3NlID0gdXNlQXBwU3RhdGUocyA9PiBzLnZlcmJvc2UpXG4gIGNvbnN0IG1jcCA9IHVzZUFwcFN0YXRlKHMgPT4gcy5tY3ApXG4gIGNvbnN0IHBsdWdpbnMgPSB1c2VBcHBTdGF0ZShzID0+IHMucGx1Z2lucylcbiAgY29uc3QgYWdlbnREZWZpbml0aW9ucyA9IHVzZUFwcFN0YXRlKHMgPT4gcy5hZ2VudERlZmluaXRpb25zKVxuICBjb25zdCBmaWxlSGlzdG9yeSA9IHVzZUFwcFN0YXRlKHMgPT4gcy5maWxlSGlzdG9yeSlcbiAgY29uc3QgaW5pdGlhbE1lc3NhZ2UgPSB1c2VBcHBTdGF0ZShzID0+IHMuaW5pdGlhbE1lc3NhZ2UpXG4gIGNvbnN0IHF1ZXVlZENvbW1hbmRzID0gdXNlQ29tbWFuZFF1ZXVlKClcbiAgLy8gZmVhdHVyZSgpIGlzIGEgYnVpbGQtdGltZSBjb25zdGFudCDigJQgZGVhZCBjb2RlIGVsaW1pbmF0aW9uIHJlbW92ZXMgdGhlIGhvb2tcbiAgLy8gY2FsbCBlbnRpcmVseSBpbiBleHRlcm5hbCBidWlsZHMsIHNvIHRoaXMgaXMgc2FmZSBkZXNwaXRlIGxvb2tpbmcgY29uZGl0aW9uYWwuXG4gIC8vIFRoZXNlIGZpZWxkcyBjb250YWluIGV4Y2x1ZGVkIHN0cmluZ3MgdGhhdCBtdXN0IG5vdCBhcHBlYXIgaW4gZXh0ZXJuYWwgYnVpbGRzLlxuICBjb25zdCBzcGlubmVyVGlwID0gdXNlQXBwU3RhdGUocyA9PiBzLnNwaW5uZXJUaXApXG4gIGNvbnN0IHNob3dFeHBhbmRlZFRvZG9zID0gdXNlQXBwU3RhdGUocyA9PiBzLmV4cGFuZGVkVmlldykgPT09ICd0YXNrcydcbiAgY29uc3QgcGVuZGluZ1dvcmtlclJlcXVlc3QgPSB1c2VBcHBTdGF0ZShzID0+IHMucGVuZGluZ1dvcmtlclJlcXVlc3QpXG4gIGNvbnN0IHBlbmRpbmdTYW5kYm94UmVxdWVzdCA9IHVzZUFwcFN0YXRlKHMgPT4gcy5wZW5kaW5nU2FuZGJveFJlcXVlc3QpXG4gIGNvbnN0IHRlYW1Db250ZXh0ID0gdXNlQXBwU3RhdGUocyA9PiBzLnRlYW1Db250ZXh0KVxuICBjb25zdCB0YXNrcyA9IHVzZUFwcFN0YXRlKHMgPT4gcy50YXNrcylcbiAgY29uc3Qgd29ya2VyU2FuZGJveFBlcm1pc3Npb25zID0gdXNlQXBwU3RhdGUocyA9PiBzLndvcmtlclNhbmRib3hQZXJtaXNzaW9ucylcbiAgY29uc3QgZWxpY2l0YXRpb24gPSB1c2VBcHBTdGF0ZShzID0+IHMuZWxpY2l0YXRpb24pXG4gIGNvbnN0IHVsdHJhcGxhblBlbmRpbmdDaG9pY2UgPSB1c2VBcHBTdGF0ZShzID0+IHMudWx0cmFwbGFuUGVuZGluZ0Nob2ljZSlcbiAgY29uc3QgdWx0cmFwbGFuTGF1bmNoUGVuZGluZyA9IHVzZUFwcFN0YXRlKHMgPT4gcy51bHRyYXBsYW5MYXVuY2hQZW5kaW5nKVxuICBjb25zdCB2aWV3aW5nQWdlbnRUYXNrSWQgPSB1c2VBcHBTdGF0ZShzID0+IHMudmlld2luZ0FnZW50VGFza0lkKVxuICBjb25zdCBzZXRBcHBTdGF0ZSA9IHVzZVNldEFwcFN0YXRlKClcblxuICAvLyBCb290c3RyYXA6IHJldGFpbmVkIGxvY2FsX2FnZW50IHRoYXQgaGFzbid0IGxvYWRlZCBkaXNrIHlldCDihpIgcmVhZFxuICAvLyBzaWRlY2hhaW4gSlNPTkwgYW5kIFVVSUQtbWVyZ2Ugd2l0aCB3aGF0ZXZlciBzdHJlYW0gaGFzIGFwcGVuZGVkIHNvIGZhci5cbiAgLy8gU3RyZWFtIGFwcGVuZHMgaW1tZWRpYXRlbHkgb24gcmV0YWluIChubyBkZWZlcik7IGJvb3RzdHJhcCBmaWxscyB0aGVcbiAgLy8gcHJlZml4LiBEaXNrLXdyaXRlLWJlZm9yZS15aWVsZCBtZWFucyBsaXZlIGlzIGFsd2F5cyBhIHN1ZmZpeCBvZiBkaXNrLlxuICBjb25zdCB2aWV3ZWRMb2NhbEFnZW50ID0gdmlld2luZ0FnZW50VGFza0lkXG4gICAgPyB0YXNrc1t2aWV3aW5nQWdlbnRUYXNrSWRdXG4gICAgOiB1bmRlZmluZWRcbiAgY29uc3QgbmVlZHNCb290c3RyYXAgPVxuICAgIGlzTG9jYWxBZ2VudFRhc2sodmlld2VkTG9jYWxBZ2VudCkgJiZcbiAgICB2aWV3ZWRMb2NhbEFnZW50LnJldGFpbiAmJlxuICAgICF2aWV3ZWRMb2NhbEFnZW50LmRpc2tMb2FkZWRcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIXZpZXdpbmdBZ2VudFRhc2tJZCB8fCAhbmVlZHNCb290c3RyYXApIHJldHVyblxuICAgIGNvbnN0IHRhc2tJZCA9IHZpZXdpbmdBZ2VudFRhc2tJZFxuICAgIHZvaWQgZ2V0QWdlbnRUcmFuc2NyaXB0KGFzQWdlbnRJZCh0YXNrSWQpKS50aGVuKHJlc3VsdCA9PiB7XG4gICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+IHtcbiAgICAgICAgY29uc3QgdCA9IHByZXYudGFza3NbdGFza0lkXVxuICAgICAgICBpZiAoIWlzTG9jYWxBZ2VudFRhc2sodCkgfHwgdC5kaXNrTG9hZGVkIHx8ICF0LnJldGFpbikgcmV0dXJuIHByZXZcbiAgICAgICAgY29uc3QgbGl2ZSA9IHQubWVzc2FnZXMgPz8gW11cbiAgICAgICAgY29uc3QgbGl2ZVV1aWRzID0gbmV3IFNldChsaXZlLm1hcChtID0+IG0udXVpZCkpXG4gICAgICAgIGNvbnN0IGRpc2tPbmx5ID0gcmVzdWx0XG4gICAgICAgICAgPyByZXN1bHQubWVzc2FnZXMuZmlsdGVyKG0gPT4gIWxpdmVVdWlkcy5oYXMobS51dWlkKSlcbiAgICAgICAgICA6IFtdXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICB0YXNrczoge1xuICAgICAgICAgICAgLi4ucHJldi50YXNrcyxcbiAgICAgICAgICAgIFt0YXNrSWRdOiB7XG4gICAgICAgICAgICAgIC4uLnQsXG4gICAgICAgICAgICAgIG1lc3NhZ2VzOiBbLi4uZGlza09ubHksIC4uLmxpdmVdLFxuICAgICAgICAgICAgICBkaXNrTG9hZGVkOiB0cnVlLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICB9LFxuICAgICAgICB9XG4gICAgICB9KVxuICAgIH0pXG4gIH0sIFt2aWV3aW5nQWdlbnRUYXNrSWQsIG5lZWRzQm9vdHN0cmFwLCBzZXRBcHBTdGF0ZV0pXG5cbiAgY29uc3Qgc3RvcmUgPSB1c2VBcHBTdGF0ZVN0b3JlKClcbiAgY29uc3QgdGVybWluYWwgPSB1c2VUZXJtaW5hbE5vdGlmaWNhdGlvbigpXG4gIGNvbnN0IG1haW5Mb29wTW9kZWwgPSB1c2VNYWluTG9vcE1vZGVsKClcblxuICAvLyBOb3RlOiBzdGFuZGFsb25lQWdlbnRDb250ZXh0IGlzIGluaXRpYWxpemVkIGluIG1haW4udHN4ICh2aWEgaW5pdGlhbFN0YXRlKSBvclxuICAvLyBSZXN1bWVDb252ZXJzYXRpb24udHN4ICh2aWEgc2V0QXBwU3RhdGUgYmVmb3JlIHJlbmRlcmluZyBSRVBMKSB0byBhdm9pZFxuICAvLyB1c2VFZmZlY3QtYmFzZWQgc3RhdGUgaW5pdGlhbGl6YXRpb24gb24gbW91bnQgKHBlciBDTEFVREUubWQgZ3VpZGVsaW5lcylcblxuICAvLyBMb2NhbCBzdGF0ZSBmb3IgY29tbWFuZHMgKGhvdC1yZWxvYWRhYmxlIHdoZW4gc2tpbGwgZmlsZXMgY2hhbmdlKVxuICBjb25zdCBbbG9jYWxDb21tYW5kcywgc2V0TG9jYWxDb21tYW5kc10gPSB1c2VTdGF0ZShpbml0aWFsQ29tbWFuZHMpXG5cbiAgLy8gV2F0Y2ggZm9yIHNraWxsIGZpbGUgY2hhbmdlcyBhbmQgcmVsb2FkIGFsbCBjb21tYW5kc1xuICB1c2VTa2lsbHNDaGFuZ2UoXG4gICAgaXNSZW1vdGVTZXNzaW9uID8gdW5kZWZpbmVkIDogZ2V0UHJvamVjdFJvb3QoKSxcbiAgICBzZXRMb2NhbENvbW1hbmRzLFxuICApXG5cbiAgLy8gVHJhY2sgcHJvYWN0aXZlIG1vZGUgZm9yIHRvb2xzIGRlcGVuZGVuY3kgLSBTbGVlcFRvb2wgZmlsdGVycyBieSBwcm9hY3RpdmUgc3RhdGVcbiAgY29uc3QgcHJvYWN0aXZlQWN0aXZlID0gUmVhY3QudXNlU3luY0V4dGVybmFsU3RvcmUoXG4gICAgcHJvYWN0aXZlTW9kdWxlPy5zdWJzY3JpYmVUb1Byb2FjdGl2ZUNoYW5nZXMgPz8gUFJPQUNUSVZFX05PX09QX1NVQlNDUklCRSxcbiAgICBwcm9hY3RpdmVNb2R1bGU/LmlzUHJvYWN0aXZlQWN0aXZlID8/IFBST0FDVElWRV9GQUxTRSxcbiAgKVxuXG4gIC8vIEJyaWVmVG9vbC5pc0VuYWJsZWQoKSByZWFkcyBnZXRVc2VyTXNnT3B0SW4oKSBmcm9tIGJvb3RzdHJhcCBzdGF0ZSwgd2hpY2hcbiAgLy8gL2JyaWVmIGZsaXBzIG1pZC1zZXNzaW9uIGFsb25nc2lkZSBpc0JyaWVmT25seS4gVGhlIG1lbW8gYmVsb3cgbmVlZHMgYVxuICAvLyBSZWFjdC12aXNpYmxlIGRlcCB0byByZS1ydW4gZ2V0VG9vbHMoKSB3aGVuIHRoYXQgaGFwcGVuczsgaXNCcmllZk9ubHkgaXNcbiAgLy8gdGhlIEFwcFN0YXRlIG1pcnJvciB0aGF0IHRyaWdnZXJzIHRoZSByZS1yZW5kZXIuIFdpdGhvdXQgdGhpcywgdG9nZ2xpbmdcbiAgLy8gL2JyaWVmIG1pZC1zZXNzaW9uIGxlYXZlcyB0aGUgc3RhbGUgdG9vbCBsaXN0IChubyBTZW5kVXNlck1lc3NhZ2UpIGFuZFxuICAvLyB0aGUgbW9kZWwgZW1pdHMgcGxhaW4gdGV4dCB0aGUgYnJpZWYgZmlsdGVyIGhpZGVzLlxuICBjb25zdCBpc0JyaWVmT25seSA9IHVzZUFwcFN0YXRlKHMgPT4gcy5pc0JyaWVmT25seSlcblxuICBjb25zdCBsb2NhbFRvb2xzID0gdXNlTWVtbyhcbiAgICAoKSA9PiBnZXRUb29scyh0b29sUGVybWlzc2lvbkNvbnRleHQpLFxuICAgIFt0b29sUGVybWlzc2lvbkNvbnRleHQsIHByb2FjdGl2ZUFjdGl2ZSwgaXNCcmllZk9ubHldLFxuICApXG5cbiAgdXNlS2lja09mZkNoZWNrQW5kRGlzYWJsZUJ5cGFzc1Blcm1pc3Npb25zSWZOZWVkZWQoKVxuICB1c2VLaWNrT2ZmQ2hlY2tBbmREaXNhYmxlQXV0b01vZGVJZk5lZWRlZCgpXG5cbiAgY29uc3QgW2R5bmFtaWNNY3BDb25maWcsIHNldER5bmFtaWNNY3BDb25maWddID0gdXNlU3RhdGU8XG4gICAgUmVjb3JkPHN0cmluZywgU2NvcGVkTWNwU2VydmVyQ29uZmlnPiB8IHVuZGVmaW5lZFxuICA+KGluaXRpYWxEeW5hbWljTWNwQ29uZmlnKVxuXG4gIGNvbnN0IG9uQ2hhbmdlRHluYW1pY01jcENvbmZpZyA9IHVzZUNhbGxiYWNrKFxuICAgIChjb25maWc6IFJlY29yZDxzdHJpbmcsIFNjb3BlZE1jcFNlcnZlckNvbmZpZz4pID0+IHtcbiAgICAgIHNldER5bmFtaWNNY3BDb25maWcoY29uZmlnKVxuICAgIH0sXG4gICAgW3NldER5bmFtaWNNY3BDb25maWddLFxuICApXG5cbiAgY29uc3QgW3NjcmVlbiwgc2V0U2NyZWVuXSA9IHVzZVN0YXRlPFNjcmVlbj4oJ3Byb21wdCcpXG4gIGNvbnN0IFtzaG93QWxsSW5UcmFuc2NyaXB0LCBzZXRTaG93QWxsSW5UcmFuc2NyaXB0XSA9IHVzZVN0YXRlKGZhbHNlKVxuICAvLyBbIGZvcmNlcyB0aGUgZHVtcC10by1zY3JvbGxiYWNrIHBhdGggaW5zaWRlIHRyYW5zY3JpcHQgbW9kZS4gU2VwYXJhdGVcbiAgLy8gZnJvbSBDTEFVREVfQ09ERV9OT19GTElDS0VSPTAgKHdoaWNoIGlzIHByb2Nlc3MtbGlmZXRpbWUpIOKAlCB0aGlzIGlzXG4gIC8vIGVwaGVtZXJhbCwgcmVzZXQgb24gdHJhbnNjcmlwdCBleGl0LiBEaWFnbm9zdGljIGVzY2FwZSBoYXRjaCBzb1xuICAvLyB0ZXJtaW5hbC90bXV4IG5hdGl2ZSBjbWQtRiBjYW4gc2VhcmNoIHRoZSBmdWxsIGZsYXQgcmVuZGVyLlxuICBjb25zdCBbZHVtcE1vZGUsIHNldER1bXBNb2RlXSA9IHVzZVN0YXRlKGZhbHNlKVxuICAvLyB2LWZvci1lZGl0b3IgcmVuZGVyIHByb2dyZXNzLiBJbmxpbmUgaW4gdGhlIGZvb3RlciDigJQgbm90aWZpY2F0aW9uc1xuICAvLyByZW5kZXIgaW5zaWRlIFByb21wdElucHV0IHdoaWNoIGlzbid0IG1vdW50ZWQgaW4gdHJhbnNjcmlwdC5cbiAgY29uc3QgW2VkaXRvclN0YXR1cywgc2V0RWRpdG9yU3RhdHVzXSA9IHVzZVN0YXRlKCcnKVxuICAvLyBJbmNyZW1lbnRlZCBvbiB0cmFuc2NyaXB0IGV4aXQuIEFzeW5jIHYtcmVuZGVyIGNhcHR1cmVzIHRoaXMgYXQgc3RhcnQ7XG4gIC8vIGVhY2ggc3RhdHVzIHdyaXRlIG5vLW9wcyBpZiBzdGFsZSAodXNlciBsZWZ0IHRyYW5zY3JpcHQgbWlkLXJlbmRlciDigJRcbiAgLy8gdGhlIHN0YWJsZSBzZXRTdGF0ZSB3b3VsZCBvdGhlcndpc2Ugc3RhbXAgYSBnaG9zdCB0b2FzdCBpbnRvIHRoZSBuZXh0XG4gIC8vIHNlc3Npb24pLiBBbHNvIGNsZWFycyBhbnkgcGVuZGluZyA0cyBhdXRvLWNsZWFyLlxuICBjb25zdCBlZGl0b3JHZW5SZWYgPSB1c2VSZWYoMClcbiAgY29uc3QgZWRpdG9yVGltZXJSZWYgPSB1c2VSZWY8UmV0dXJuVHlwZTx0eXBlb2Ygc2V0VGltZW91dD4gfCB1bmRlZmluZWQ+KFxuICAgIHVuZGVmaW5lZCxcbiAgKVxuICBjb25zdCBlZGl0b3JSZW5kZXJpbmdSZWYgPSB1c2VSZWYoZmFsc2UpXG4gIGNvbnN0IHsgYWRkTm90aWZpY2F0aW9uLCByZW1vdmVOb3RpZmljYXRpb24gfSA9IHVzZU5vdGlmaWNhdGlvbnMoKVxuXG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBwcmVmZXItY29uc3RcbiAgbGV0IHRyeVN1Z2dlc3RCZ1BSSW50ZXJjZXB0ID0gU1VHR0VTVF9CR19QUl9OT09QXG5cbiAgY29uc3QgbWNwQ2xpZW50cyA9IHVzZU1lcmdlZENsaWVudHMoaW5pdGlhbE1jcENsaWVudHMsIG1jcC5jbGllbnRzKVxuXG4gIC8vIElERSBpbnRlZ3JhdGlvblxuICBjb25zdCBbaWRlU2VsZWN0aW9uLCBzZXRJREVTZWxlY3Rpb25dID0gdXNlU3RhdGU8SURFU2VsZWN0aW9uIHwgdW5kZWZpbmVkPihcbiAgICB1bmRlZmluZWQsXG4gIClcbiAgY29uc3QgW2lkZVRvSW5zdGFsbEV4dGVuc2lvbiwgc2V0SURFVG9JbnN0YWxsRXh0ZW5zaW9uXSA9XG4gICAgdXNlU3RhdGU8SWRlVHlwZSB8IG51bGw+KG51bGwpXG4gIGNvbnN0IFtpZGVJbnN0YWxsYXRpb25TdGF0dXMsIHNldElERUluc3RhbGxhdGlvblN0YXR1c10gPVxuICAgIHVzZVN0YXRlPElERUV4dGVuc2lvbkluc3RhbGxhdGlvblN0YXR1cyB8IG51bGw+KG51bGwpXG4gIGNvbnN0IFtzaG93SWRlT25ib2FyZGluZywgc2V0U2hvd0lkZU9uYm9hcmRpbmddID0gdXNlU3RhdGUoZmFsc2UpXG4gIC8vIERlYWQgY29kZSBlbGltaW5hdGlvbjogbW9kZWwgc3dpdGNoIGNhbGxvdXQgc3RhdGUgKGFudC1vbmx5KVxuICBjb25zdCBbc2hvd01vZGVsU3dpdGNoQ2FsbG91dCwgc2V0U2hvd01vZGVsU3dpdGNoQ2FsbG91dF0gPSB1c2VTdGF0ZSgoKSA9PiB7XG4gICAgaWYgKFwiZXh0ZXJuYWxcIiA9PT0gJ2FudCcpIHtcbiAgICAgIHJldHVybiBzaG91bGRTaG93QW50TW9kZWxTd2l0Y2goKVxuICAgIH1cbiAgICByZXR1cm4gZmFsc2VcbiAgfSlcbiAgY29uc3QgW3Nob3dFZmZvcnRDYWxsb3V0LCBzZXRTaG93RWZmb3J0Q2FsbG91dF0gPSB1c2VTdGF0ZSgoKSA9PlxuICAgIHNob3VsZFNob3dFZmZvcnRDYWxsb3V0KG1haW5Mb29wTW9kZWwpLFxuICApXG4gIGNvbnN0IHNob3dSZW1vdGVDYWxsb3V0ID0gdXNlQXBwU3RhdGUocyA9PiBzLnNob3dSZW1vdGVDYWxsb3V0KVxuICBjb25zdCBbc2hvd0Rlc2t0b3BVcHNlbGxTdGFydHVwLCBzZXRTaG93RGVza3RvcFVwc2VsbFN0YXJ0dXBdID0gdXNlU3RhdGUoKCkgPT5cbiAgICBzaG91bGRTaG93RGVza3RvcFVwc2VsbFN0YXJ0dXAoKSxcbiAgKVxuICAvLyBub3RpZmljYXRpb25zXG4gIHVzZU1vZGVsTWlncmF0aW9uTm90aWZpY2F0aW9ucygpXG4gIHVzZUNhblN3aXRjaFRvRXhpc3RpbmdTdWJzY3JpcHRpb24oKVxuICB1c2VJREVTdGF0dXNJbmRpY2F0b3IoeyBpZGVTZWxlY3Rpb24sIG1jcENsaWVudHMsIGlkZUluc3RhbGxhdGlvblN0YXR1cyB9KVxuICB1c2VNY3BDb25uZWN0aXZpdHlTdGF0dXMoeyBtY3BDbGllbnRzIH0pXG4gIHVzZUF1dG9Nb2RlVW5hdmFpbGFibGVOb3RpZmljYXRpb24oKVxuICB1c2VQbHVnaW5JbnN0YWxsYXRpb25TdGF0dXMoKVxuICB1c2VQbHVnaW5BdXRvdXBkYXRlTm90aWZpY2F0aW9uKClcbiAgdXNlU2V0dGluZ3NFcnJvcnMoKVxuICB1c2VSYXRlTGltaXRXYXJuaW5nTm90aWZpY2F0aW9uKG1haW5Mb29wTW9kZWwpXG4gIHVzZUZhc3RNb2RlTm90aWZpY2F0aW9uKClcbiAgdXNlRGVwcmVjYXRpb25XYXJuaW5nTm90aWZpY2F0aW9uKG1haW5Mb29wTW9kZWwpXG4gIHVzZU5wbURlcHJlY2F0aW9uTm90aWZpY2F0aW9uKClcbiAgdXNlQW50T3JnV2FybmluZ05vdGlmaWNhdGlvbigpXG4gIHVzZUluc3RhbGxNZXNzYWdlcygpXG4gIHVzZUNocm9tZUV4dGVuc2lvbk5vdGlmaWNhdGlvbigpXG4gIHVzZU9mZmljaWFsTWFya2V0cGxhY2VOb3RpZmljYXRpb24oKVxuICB1c2VMc3BJbml0aWFsaXphdGlvbk5vdGlmaWNhdGlvbigpXG4gIHVzZVRlYW1tYXRlTGlmZWN5Y2xlTm90aWZpY2F0aW9uKClcbiAgY29uc3Qge1xuICAgIHJlY29tbWVuZGF0aW9uOiBsc3BSZWNvbW1lbmRhdGlvbixcbiAgICBoYW5kbGVSZXNwb25zZTogaGFuZGxlTHNwUmVzcG9uc2UsXG4gIH0gPSB1c2VMc3BQbHVnaW5SZWNvbW1lbmRhdGlvbigpXG4gIGNvbnN0IHtcbiAgICByZWNvbW1lbmRhdGlvbjogaGludFJlY29tbWVuZGF0aW9uLFxuICAgIGhhbmRsZVJlc3BvbnNlOiBoYW5kbGVIaW50UmVzcG9uc2UsXG4gIH0gPSB1c2VDbGF1ZGVDb2RlSGludFJlY29tbWVuZGF0aW9uKClcblxuICAvLyBNZW1vaXplIHRoZSBjb21iaW5lZCBpbml0aWFsIHRvb2xzIGFycmF5IHRvIHByZXZlbnQgcmVmZXJlbmNlIGNoYW5nZXNcbiAgY29uc3QgY29tYmluZWRJbml0aWFsVG9vbHMgPSB1c2VNZW1vKCgpID0+IHtcbiAgICByZXR1cm4gWy4uLmxvY2FsVG9vbHMsIC4uLmluaXRpYWxUb29sc11cbiAgfSwgW2xvY2FsVG9vbHMsIGluaXRpYWxUb29sc10pXG5cbiAgLy8gSW5pdGlhbGl6ZSBwbHVnaW4gbWFuYWdlbWVudFxuICB1c2VNYW5hZ2VQbHVnaW5zKHsgZW5hYmxlZDogIWlzUmVtb3RlU2Vzc2lvbiB9KVxuXG4gIGNvbnN0IHRhc2tzVjIgPSB1c2VUYXNrc1YyV2l0aENvbGxhcHNlRWZmZWN0KClcblxuICAvLyBTdGFydCBiYWNrZ3JvdW5kIHBsdWdpbiBpbnN0YWxsYXRpb25zXG5cbiAgLy8gU0VDVVJJVFk6IFRoaXMgY29kZSBpcyBndWFyYW50ZWVkIHRvIHJ1biBPTkxZIGFmdGVyIHRoZSBcInRydXN0IHRoaXMgZm9sZGVyXCIgZGlhbG9nXG4gIC8vIGhhcyBiZWVuIGNvbmZpcm1lZCBieSB0aGUgdXNlci4gVGhlIHRydXN0IGRpYWxvZyBpcyBzaG93biBpbiBjbGkudHN4IChsaW5lIH4zODcpXG4gIC8vIGJlZm9yZSB0aGUgUkVQTCBjb21wb25lbnQgaXMgcmVuZGVyZWQuIFRoZSBkaWFsb2cgYmxvY2tzIGV4ZWN1dGlvbiB1bnRpbCB0aGUgdXNlclxuICAvLyBhY2NlcHRzLCBhbmQgb25seSB0aGVuIGlzIHRoZSBSRVBMIGNvbXBvbmVudCBtb3VudGVkIGFuZCB0aGlzIGVmZmVjdCBydW5zLlxuICAvLyBUaGlzIGVuc3VyZXMgdGhhdCBwbHVnaW4gaW5zdGFsbGF0aW9ucyBmcm9tIHJlcG9zaXRvcnkgYW5kIHVzZXIgc2V0dGluZ3Mgb25seVxuICAvLyBoYXBwZW4gYWZ0ZXIgZXhwbGljaXQgdXNlciBjb25zZW50IHRvIHRydXN0IHRoZSBjdXJyZW50IHdvcmtpbmcgZGlyZWN0b3J5LlxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmIChpc1JlbW90ZVNlc3Npb24pIHJldHVyblxuICAgIHZvaWQgcGVyZm9ybVN0YXJ0dXBDaGVja3Moc2V0QXBwU3RhdGUpXG4gIH0sIFtzZXRBcHBTdGF0ZSwgaXNSZW1vdGVTZXNzaW9uXSlcblxuICAvLyBBbGxvdyBDbGF1ZGUgaW4gQ2hyb21lIE1DUCB0byBzZW5kIHByb21wdHMgdGhyb3VnaCBNQ1Agbm90aWZpY2F0aW9uc1xuICAvLyBhbmQgc3luYyBwZXJtaXNzaW9uIG1vZGUgY2hhbmdlcyB0byB0aGUgQ2hyb21lIGV4dGVuc2lvblxuICB1c2VQcm9tcHRzRnJvbUNsYXVkZUluQ2hyb21lKFxuICAgIGlzUmVtb3RlU2Vzc2lvbiA/IEVNUFRZX01DUF9DTElFTlRTIDogbWNwQ2xpZW50cyxcbiAgICB0b29sUGVybWlzc2lvbkNvbnRleHQubW9kZSxcbiAgKVxuXG4gIC8vIEluaXRpYWxpemUgc3dhcm0gZmVhdHVyZXM6IHRlYW1tYXRlIGhvb2tzIGFuZCBjb250ZXh0XG4gIC8vIEhhbmRsZXMgYm90aCBmcmVzaCBzcGF3bnMgYW5kIHJlc3VtZWQgdGVhbW1hdGUgc2Vzc2lvbnNcbiAgdXNlU3dhcm1Jbml0aWFsaXphdGlvbihzZXRBcHBTdGF0ZSwgaW5pdGlhbE1lc3NhZ2VzLCB7XG4gICAgZW5hYmxlZDogIWlzUmVtb3RlU2Vzc2lvbixcbiAgfSlcblxuICBjb25zdCBtZXJnZWRUb29scyA9IHVzZU1lcmdlZFRvb2xzKFxuICAgIGNvbWJpbmVkSW5pdGlhbFRvb2xzLFxuICAgIG1jcC50b29scyxcbiAgICB0b29sUGVybWlzc2lvbkNvbnRleHQsXG4gIClcblxuICAvLyBBcHBseSBhZ2VudCB0b29sIHJlc3RyaWN0aW9ucyBpZiBtYWluVGhyZWFkQWdlbnREZWZpbml0aW9uIGlzIHNldFxuICBjb25zdCB7IHRvb2xzLCBhbGxvd2VkQWdlbnRUeXBlcyB9ID0gdXNlTWVtbygoKSA9PiB7XG4gICAgaWYgKCFtYWluVGhyZWFkQWdlbnREZWZpbml0aW9uKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICB0b29sczogbWVyZ2VkVG9vbHMsXG4gICAgICAgIGFsbG93ZWRBZ2VudFR5cGVzOiB1bmRlZmluZWQgYXMgc3RyaW5nW10gfCB1bmRlZmluZWQsXG4gICAgICB9XG4gICAgfVxuICAgIGNvbnN0IHJlc29sdmVkID0gcmVzb2x2ZUFnZW50VG9vbHMoXG4gICAgICBtYWluVGhyZWFkQWdlbnREZWZpbml0aW9uLFxuICAgICAgbWVyZ2VkVG9vbHMsXG4gICAgICBmYWxzZSxcbiAgICAgIHRydWUsXG4gICAgKVxuICAgIHJldHVybiB7XG4gICAgICB0b29sczogcmVzb2x2ZWQucmVzb2x2ZWRUb29scyxcbiAgICAgIGFsbG93ZWRBZ2VudFR5cGVzOiByZXNvbHZlZC5hbGxvd2VkQWdlbnRUeXBlcyxcbiAgICB9XG4gIH0sIFttYWluVGhyZWFkQWdlbnREZWZpbml0aW9uLCBtZXJnZWRUb29sc10pXG5cbiAgLy8gTWVyZ2UgY29tbWFuZHMgZnJvbSBsb2NhbCBzdGF0ZSwgcGx1Z2lucywgYW5kIE1DUFxuICBjb25zdCBjb21tYW5kc1dpdGhQbHVnaW5zID0gdXNlTWVyZ2VkQ29tbWFuZHMoXG4gICAgbG9jYWxDb21tYW5kcyxcbiAgICBwbHVnaW5zLmNvbW1hbmRzIGFzIENvbW1hbmRbXSxcbiAgKVxuICBjb25zdCBtZXJnZWRDb21tYW5kcyA9IHVzZU1lcmdlZENvbW1hbmRzKFxuICAgIGNvbW1hbmRzV2l0aFBsdWdpbnMsXG4gICAgbWNwLmNvbW1hbmRzIGFzIENvbW1hbmRbXSxcbiAgKVxuICAvLyBGaWx0ZXIgb3V0IGFsbCBjb21tYW5kcyBpZiBkaXNhYmxlU2xhc2hDb21tYW5kcyBpcyB0cnVlXG4gIGNvbnN0IGNvbW1hbmRzID0gdXNlTWVtbyhcbiAgICAoKSA9PiAoZGlzYWJsZVNsYXNoQ29tbWFuZHMgPyBbXSA6IG1lcmdlZENvbW1hbmRzKSxcbiAgICBbZGlzYWJsZVNsYXNoQ29tbWFuZHMsIG1lcmdlZENvbW1hbmRzXSxcbiAgKVxuXG4gIHVzZUlkZUxvZ2dpbmcoaXNSZW1vdGVTZXNzaW9uID8gRU1QVFlfTUNQX0NMSUVOVFMgOiBtY3AuY2xpZW50cylcbiAgdXNlSWRlU2VsZWN0aW9uKFxuICAgIGlzUmVtb3RlU2Vzc2lvbiA/IEVNUFRZX01DUF9DTElFTlRTIDogbWNwLmNsaWVudHMsXG4gICAgc2V0SURFU2VsZWN0aW9uLFxuICApXG5cbiAgY29uc3QgW3N0cmVhbU1vZGUsIHNldFN0cmVhbU1vZGVdID0gdXNlU3RhdGU8U3Bpbm5lck1vZGU+KCdyZXNwb25kaW5nJylcbiAgLy8gUmVmIG1pcnJvciBzbyBvblN1Ym1pdCBjYW4gcmVhZCB0aGUgbGF0ZXN0IHZhbHVlIHdpdGhvdXQgYWRkaW5nXG4gIC8vIHN0cmVhbU1vZGUgdG8gaXRzIGRlcHMuIHN0cmVhbU1vZGUgZmxpcHMgYmV0d2VlblxuICAvLyByZXF1ZXN0aW5nL3Jlc3BvbmRpbmcvdG9vbC11c2UgfjEweCBwZXIgdHVybiBkdXJpbmcgc3RyZWFtaW5nOyBoYXZpbmcgaXRcbiAgLy8gaW4gb25TdWJtaXQncyBkZXBzIHdhcyByZWNyZWF0aW5nIG9uU3VibWl0IG9uIGV2ZXJ5IGZsaXAsIHdoaWNoXG4gIC8vIGNhc2NhZGVkIGludG8gUHJvbXB0SW5wdXQgcHJvcCBjaHVybiBhbmQgZG93bnN0cmVhbSB1c2VDYWxsYmFjay91c2VNZW1vXG4gIC8vIGludmFsaWRhdGlvbi4gVGhlIG9ubHkgY29uc3VtZXJzIGluc2lkZSBjYWxsYmFja3MgYXJlIGRlYnVnIGxvZ2dpbmcgYW5kXG4gIC8vIHRlbGVtZXRyeSAoaGFuZGxlUHJvbXB0U3VibWl0LnRzKSwgc28gYSBzdGFsZS1ieS1vbmUtcmVuZGVyIHZhbHVlIGlzXG4gIC8vIGhhcm1sZXNzIOKAlCBidXQgcmVmIG1pcnJvcnMgc3luYyBvbiBldmVyeSByZW5kZXIgYW55d2F5IHNvIGl0J3MgZnJlc2guXG4gIGNvbnN0IHN0cmVhbU1vZGVSZWYgPSB1c2VSZWYoc3RyZWFtTW9kZSlcbiAgc3RyZWFtTW9kZVJlZi5jdXJyZW50ID0gc3RyZWFtTW9kZVxuICBjb25zdCBbc3RyZWFtaW5nVG9vbFVzZXMsIHNldFN0cmVhbWluZ1Rvb2xVc2VzXSA9IHVzZVN0YXRlPFxuICAgIFN0cmVhbWluZ1Rvb2xVc2VbXVxuICA+KFtdKVxuICBjb25zdCBbc3RyZWFtaW5nVGhpbmtpbmcsIHNldFN0cmVhbWluZ1RoaW5raW5nXSA9XG4gICAgdXNlU3RhdGU8U3RyZWFtaW5nVGhpbmtpbmcgfCBudWxsPihudWxsKVxuXG4gIC8vIEF1dG8taGlkZSBzdHJlYW1pbmcgdGhpbmtpbmcgYWZ0ZXIgMzAgc2Vjb25kcyBvZiBiZWluZyBjb21wbGV0ZWRcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoXG4gICAgICBzdHJlYW1pbmdUaGlua2luZyAmJlxuICAgICAgIXN0cmVhbWluZ1RoaW5raW5nLmlzU3RyZWFtaW5nICYmXG4gICAgICBzdHJlYW1pbmdUaGlua2luZy5zdHJlYW1pbmdFbmRlZEF0XG4gICAgKSB7XG4gICAgICBjb25zdCBlbGFwc2VkID0gRGF0ZS5ub3coKSAtIHN0cmVhbWluZ1RoaW5raW5nLnN0cmVhbWluZ0VuZGVkQXRcbiAgICAgIGNvbnN0IHJlbWFpbmluZyA9IDMwMDAwIC0gZWxhcHNlZFxuICAgICAgaWYgKHJlbWFpbmluZyA+IDApIHtcbiAgICAgICAgY29uc3QgdGltZXIgPSBzZXRUaW1lb3V0KHNldFN0cmVhbWluZ1RoaW5raW5nLCByZW1haW5pbmcsIG51bGwpXG4gICAgICAgIHJldHVybiAoKSA9PiBjbGVhclRpbWVvdXQodGltZXIpXG4gICAgICB9IGVsc2Uge1xuICAgICAgICBzZXRTdHJlYW1pbmdUaGlua2luZyhudWxsKVxuICAgICAgfVxuICAgIH1cbiAgfSwgW3N0cmVhbWluZ1RoaW5raW5nXSlcblxuICBjb25zdCBbYWJvcnRDb250cm9sbGVyLCBzZXRBYm9ydENvbnRyb2xsZXJdID1cbiAgICB1c2VTdGF0ZTxBYm9ydENvbnRyb2xsZXIgfCBudWxsPihudWxsKVxuICAvLyBSZWYgdGhhdCBhbHdheXMgcG9pbnRzIHRvIHRoZSBjdXJyZW50IGFib3J0IGNvbnRyb2xsZXIsIHVzZWQgYnkgdGhlXG4gIC8vIFJFUEwgYnJpZGdlIHRvIGFib3J0IHRoZSBhY3RpdmUgcXVlcnkgd2hlbiBhIHJlbW90ZSBpbnRlcnJ1cHQgYXJyaXZlcy5cbiAgY29uc3QgYWJvcnRDb250cm9sbGVyUmVmID0gdXNlUmVmPEFib3J0Q29udHJvbGxlciB8IG51bGw+KG51bGwpXG4gIGFib3J0Q29udHJvbGxlclJlZi5jdXJyZW50ID0gYWJvcnRDb250cm9sbGVyXG5cbiAgLy8gUmVmIGZvciB0aGUgYnJpZGdlIHJlc3VsdCBjYWxsYmFjayDigJQgc2V0IGFmdGVyIHVzZVJlcGxCcmlkZ2UgaW5pdGlhbGl6ZXMsXG4gIC8vIHJlYWQgaW4gdGhlIG9uUXVlcnkgZmluYWxseSBibG9jayB0byBub3RpZnkgbW9iaWxlIGNsaWVudHMgdGhhdCBhIHR1cm4gZW5kZWQuXG4gIGNvbnN0IHNlbmRCcmlkZ2VSZXN1bHRSZWYgPSB1c2VSZWY8KCkgPT4gdm9pZD4oKCkgPT4ge30pXG5cbiAgLy8gUmVmIGZvciB0aGUgc3luY2hyb25vdXMgcmVzdG9yZSBjYWxsYmFjayDigJQgc2V0IGFmdGVyIHJlc3RvcmVNZXNzYWdlU3luYyBpc1xuICAvLyBkZWZpbmVkLCByZWFkIGluIHRoZSBvblF1ZXJ5IGZpbmFsbHkgYmxvY2sgZm9yIGF1dG8tcmVzdG9yZSBvbiBpbnRlcnJ1cHQuXG4gIGNvbnN0IHJlc3RvcmVNZXNzYWdlU3luY1JlZiA9IHVzZVJlZjwobTogVXNlck1lc3NhZ2UpID0+IHZvaWQ+KCgpID0+IHt9KVxuXG4gIC8vIFJlZiB0byB0aGUgZnVsbHNjcmVlbiBsYXlvdXQncyBzY3JvbGwgYm94IGZvciBrZXlib2FyZCBzY3JvbGxpbmcuXG4gIC8vIE51bGwgd2hlbiBmdWxsc2NyZWVuIG1vZGUgaXMgZGlzYWJsZWQgKHJlZiBuZXZlciBhdHRhY2hlZCkuXG4gIGNvbnN0IHNjcm9sbFJlZiA9IHVzZVJlZjxTY3JvbGxCb3hIYW5kbGU+KG51bGwpXG4gIC8vIFNlcGFyYXRlIHJlZiBmb3IgdGhlIG1vZGFsIHNsb3QncyBpbm5lciBTY3JvbGxCb3gg4oCUIHBhc3NlZCB0aHJvdWdoXG4gIC8vIEZ1bGxzY3JlZW5MYXlvdXQg4oaSIE1vZGFsQ29udGV4dCBzbyBUYWJzIGNhbiBhdHRhY2ggaXQgdG8gaXRzIG93blxuICAvLyBTY3JvbGxCb3ggZm9yIHRhbGwgY29udGVudCAoZS5nLiAvc3RhdHVzJ3MgTUNQLXNlcnZlciBsaXN0KS4gTk9UXG4gIC8vIGtleWJvYXJkLWRyaXZlbiDigJQgU2Nyb2xsS2V5YmluZGluZ0hhbmRsZXIgc3RheXMgb24gdGhlIG91dGVyIHJlZiBzb1xuICAvLyBQZ1VwL1BnRG4vd2hlZWwgYWx3YXlzIHNjcm9sbCB0aGUgdHJhbnNjcmlwdCBiZWhpbmQgdGhlIG1vZGFsLlxuICAvLyBQbHVtYmluZyBrZXB0IGZvciBmdXR1cmUgbW9kYWwtc2Nyb2xsIHdpcmluZy5cbiAgY29uc3QgbW9kYWxTY3JvbGxSZWYgPSB1c2VSZWY8U2Nyb2xsQm94SGFuZGxlPihudWxsKVxuICAvLyBUaW1lc3RhbXAgb2YgdGhlIGxhc3QgdXNlci1pbml0aWF0ZWQgc2Nyb2xsICh3aGVlbCwgUGdVcC9QZ0RuLCBjdHJsK3UsXG4gIC8vIEVuZC9Ib21lLCBHLCBkcmFnLXRvLXNjcm9sbCkuIFN0YW1wZWQgaW4gY29tcG9zZWRPblNjcm9sbCDigJQgdGhlIHNpbmdsZVxuICAvLyBjaG9rZXBvaW50IFNjcm9sbEtleWJpbmRpbmdIYW5kbGVyIGNhbGxzIGZvciBldmVyeSB1c2VyIHNjcm9sbCBhY3Rpb24uXG4gIC8vIFByb2dyYW1tYXRpYyBzY3JvbGxzIChyZXBpblNjcm9sbCdzIHNjcm9sbFRvQm90dG9tLCBzdGlja3kgYXV0by1mb2xsb3cpXG4gIC8vIGRvIE5PVCBnbyB0aHJvdWdoIGNvbXBvc2VkT25TY3JvbGwsIHNvIHRoZXkgZG9uJ3Qgc3RhbXAgdGhpcy4gUmVmIG5vdFxuICAvLyBzdGF0ZTogbm8gcmUtcmVuZGVyIG9uIGV2ZXJ5IHdoZWVsIHRpY2suXG4gIGNvbnN0IGxhc3RVc2VyU2Nyb2xsVHNSZWYgPSB1c2VSZWYoMClcblxuICAvLyBTeW5jaHJvbm91cyBzdGF0ZSBtYWNoaW5lIGZvciB0aGUgcXVlcnkgbGlmZWN5Y2xlLiBSZXBsYWNlcyB0aGVcbiAgLy8gZXJyb3ItcHJvbmUgZHVhbC1zdGF0ZSBwYXR0ZXJuIHdoZXJlIGlzTG9hZGluZyAoUmVhY3Qgc3RhdGUsIGFzeW5jXG4gIC8vIGJhdGNoZWQpIGFuZCBpc1F1ZXJ5UnVubmluZyAocmVmLCBzeW5jKSBjb3VsZCBkZXN5bmMuIFNlZSBRdWVyeUd1YXJkLnRzLlxuICBjb25zdCBxdWVyeUd1YXJkID0gUmVhY3QudXNlUmVmKG5ldyBRdWVyeUd1YXJkKCkpLmN1cnJlbnRcblxuICAvLyBTdWJzY3JpYmUgdG8gdGhlIGd1YXJkIOKAlCB0cnVlIGR1cmluZyBkaXNwYXRjaGluZyBvciBydW5uaW5nLlxuICAvLyBUaGlzIGlzIHRoZSBzaW5nbGUgc291cmNlIG9mIHRydXRoIGZvciBcImlzIGEgbG9jYWwgcXVlcnkgaW4gZmxpZ2h0XCIuXG4gIGNvbnN0IGlzUXVlcnlBY3RpdmUgPSBSZWFjdC51c2VTeW5jRXh0ZXJuYWxTdG9yZShcbiAgICBxdWVyeUd1YXJkLnN1YnNjcmliZSxcbiAgICBxdWVyeUd1YXJkLmdldFNuYXBzaG90LFxuICApXG5cbiAgLy8gU2VwYXJhdGUgbG9hZGluZyBmbGFnIGZvciBvcGVyYXRpb25zIG91dHNpZGUgdGhlIGxvY2FsIHF1ZXJ5IGd1YXJkOlxuICAvLyByZW1vdGUgc2Vzc2lvbnMgKHVzZVJlbW90ZVNlc3Npb24gLyB1c2VEaXJlY3RDb25uZWN0KSBhbmQgZm9yZWdyb3VuZGVkXG4gIC8vIGJhY2tncm91bmQgdGFza3MgKHVzZVNlc3Npb25CYWNrZ3JvdW5kaW5nKS4gVGhlc2UgZG9uJ3Qgcm91dGUgdGhyb3VnaFxuICAvLyBvblF1ZXJ5IC8gcXVlcnlHdWFyZCwgc28gdGhleSBuZWVkIHRoZWlyIG93biBzcGlubmVyLXZpc2liaWxpdHkgc3RhdGUuXG4gIC8vIEluaXRpYWxpemUgdHJ1ZSBpZiByZW1vdGUgbW9kZSB3aXRoIGluaXRpYWwgcHJvbXB0IChDQ1IgcHJvY2Vzc2luZyBpdCkuXG4gIGNvbnN0IFtpc0V4dGVybmFsTG9hZGluZywgc2V0SXNFeHRlcm5hbExvYWRpbmdSYXddID0gUmVhY3QudXNlU3RhdGUoXG4gICAgcmVtb3RlU2Vzc2lvbkNvbmZpZz8uaGFzSW5pdGlhbFByb21wdCA/PyBmYWxzZSxcbiAgKVxuXG4gIC8vIERlcml2ZWQ6IGFueSBsb2FkaW5nIHNvdXJjZSBhY3RpdmUuIFJlYWQtb25seSDigJQgbm8gc2V0dGVyLiBMb2NhbCBxdWVyeVxuICAvLyBsb2FkaW5nIGlzIGRyaXZlbiBieSBxdWVyeUd1YXJkIChyZXNlcnZlL3RyeVN0YXJ0L2VuZC9jYW5jZWxSZXNlcnZhdGlvbiksXG4gIC8vIGV4dGVybmFsIGxvYWRpbmcgYnkgc2V0SXNFeHRlcm5hbExvYWRpbmcuXG4gIGNvbnN0IGlzTG9hZGluZyA9IGlzUXVlcnlBY3RpdmUgfHwgaXNFeHRlcm5hbExvYWRpbmdcblxuICAvLyBFbGFwc2VkIHRpbWUgaXMgY29tcHV0ZWQgYnkgU3Bpbm5lcldpdGhWZXJiIGZyb20gdGhlc2UgcmVmcyBvbiBlYWNoXG4gIC8vIGFuaW1hdGlvbiBmcmFtZSwgYXZvaWRpbmcgYSB1c2VJbnRlcnZhbCB0aGF0IHJlLXJlbmRlcnMgdGhlIGVudGlyZSBSRVBMLlxuICBjb25zdCBbdXNlcklucHV0T25Qcm9jZXNzaW5nLCBzZXRVc2VySW5wdXRPblByb2Nlc3NpbmdSYXddID0gUmVhY3QudXNlU3RhdGU8XG4gICAgc3RyaW5nIHwgdW5kZWZpbmVkXG4gID4odW5kZWZpbmVkKVxuICAvLyBtZXNzYWdlc1JlZi5jdXJyZW50Lmxlbmd0aCBhdCB0aGUgbW9tZW50IHVzZXJJbnB1dE9uUHJvY2Vzc2luZyB3YXMgc2V0LlxuICAvLyBUaGUgcGxhY2Vob2xkZXIgaGlkZXMgb25jZSBkaXNwbGF5ZWRNZXNzYWdlcyBncm93cyBwYXN0IHRoaXMg4oCUIGkuZS4gdGhlXG4gIC8vIHJlYWwgdXNlciBtZXNzYWdlIGhhcyBsYW5kZWQgaW4gdGhlIHZpc2libGUgdHJhbnNjcmlwdC5cbiAgY29uc3QgdXNlcklucHV0QmFzZWxpbmVSZWYgPSBSZWFjdC51c2VSZWYoMClcbiAgLy8gVHJ1ZSB3aGlsZSB0aGUgc3VibWl0dGVkIHByb21wdCBpcyBiZWluZyBwcm9jZXNzZWQgYnV0IGl0cyB1c2VyIG1lc3NhZ2VcbiAgLy8gaGFzbid0IHJlYWNoZWQgc2V0TWVzc2FnZXMgeWV0LiBzZXRNZXNzYWdlcyB1c2VzIHRoaXMgdG8ga2VlcCB0aGVcbiAgLy8gYmFzZWxpbmUgaW4gc3luYyB3aGVuIHVucmVsYXRlZCBhc3luYyBtZXNzYWdlcyAoYnJpZGdlIHN0YXR1cywgaG9va1xuICAvLyByZXN1bHRzLCBzY2hlZHVsZWQgdGFza3MpIGxhbmQgZHVyaW5nIHRoYXQgd2luZG93LlxuICBjb25zdCB1c2VyTWVzc2FnZVBlbmRpbmdSZWYgPSBSZWFjdC51c2VSZWYoZmFsc2UpXG5cbiAgLy8gV2FsbC1jbG9jayB0aW1lIHRyYWNraW5nIHJlZnMgZm9yIGFjY3VyYXRlIGVsYXBzZWQgdGltZSBjYWxjdWxhdGlvblxuICBjb25zdCBsb2FkaW5nU3RhcnRUaW1lUmVmID0gUmVhY3QudXNlUmVmPG51bWJlcj4oMClcbiAgY29uc3QgdG90YWxQYXVzZWRNc1JlZiA9IFJlYWN0LnVzZVJlZigwKVxuICBjb25zdCBwYXVzZVN0YXJ0VGltZVJlZiA9IFJlYWN0LnVzZVJlZjxudW1iZXIgfCBudWxsPihudWxsKVxuICBjb25zdCByZXNldFRpbWluZ1JlZnMgPSBSZWFjdC51c2VDYWxsYmFjaygoKSA9PiB7XG4gICAgbG9hZGluZ1N0YXJ0VGltZVJlZi5jdXJyZW50ID0gRGF0ZS5ub3coKVxuICAgIHRvdGFsUGF1c2VkTXNSZWYuY3VycmVudCA9IDBcbiAgICBwYXVzZVN0YXJ0VGltZVJlZi5jdXJyZW50ID0gbnVsbFxuICB9LCBbXSlcblxuICAvLyBSZXNldCB0aW1pbmcgcmVmcyBpbmxpbmUgd2hlbiBpc1F1ZXJ5QWN0aXZlIHRyYW5zaXRpb25zIGZhbHNl4oaSdHJ1ZS5cbiAgLy8gcXVlcnlHdWFyZC5yZXNlcnZlKCkgKGluIGV4ZWN1dGVVc2VySW5wdXQpIGZpcmVzIEJFRk9SRSBwcm9jZXNzVXNlcklucHV0J3NcbiAgLy8gZmlyc3QgYXdhaXQsIGJ1dCB0aGUgcmVmIHJlc2V0IGluIG9uUXVlcnkncyB0cnkgYmxvY2sgcnVucyBBRlRFUi4gRHVyaW5nXG4gIC8vIHRoYXQgZ2FwLCBSZWFjdCByZW5kZXJzIHRoZSBzcGlubmVyIHdpdGggbG9hZGluZ1N0YXJ0VGltZVJlZj0wLCBjb21wdXRpbmdcbiAgLy8gZWxhcHNlZFRpbWVNcyA9IERhdGUubm93KCkgLSAwIOKJiCA1NiB5ZWFycy4gVGhpcyBpbmxpbmUgcmVzZXQgcnVucyBvbiB0aGVcbiAgLy8gZmlyc3QgcmVuZGVyIHdoZXJlIGlzUXVlcnlBY3RpdmUgaXMgb2JzZXJ2ZWQgdHJ1ZSDigJQgdGhlIHNhbWUgcmVuZGVyIHRoYXRcbiAgLy8gZmlyc3Qgc2hvd3MgdGhlIHNwaW5uZXIg4oCUIHNvIHRoZSByZWYgaXMgY29ycmVjdCBieSB0aGUgdGltZSB0aGUgc3Bpbm5lclxuICAvLyByZWFkcyBpdC4gU2VlIElOQy00NTQ5LlxuICBjb25zdCB3YXNRdWVyeUFjdGl2ZVJlZiA9IFJlYWN0LnVzZVJlZihmYWxzZSlcbiAgaWYgKGlzUXVlcnlBY3RpdmUgJiYgIXdhc1F1ZXJ5QWN0aXZlUmVmLmN1cnJlbnQpIHtcbiAgICByZXNldFRpbWluZ1JlZnMoKVxuICB9XG4gIHdhc1F1ZXJ5QWN0aXZlUmVmLmN1cnJlbnQgPSBpc1F1ZXJ5QWN0aXZlXG5cbiAgLy8gV3JhcHBlciBmb3Igc2V0SXNFeHRlcm5hbExvYWRpbmcgdGhhdCByZXNldHMgdGltaW5nIHJlZnMgb24gdHJhbnNpdGlvblxuICAvLyB0byB0cnVlIOKAlCBTcGlubmVyV2l0aFZlcmIgcmVhZHMgdGhlc2UgZm9yIGVsYXBzZWQgdGltZSwgc28gdGhleSBtdXN0IGJlXG4gIC8vIHJlc2V0IGZvciByZW1vdGUgc2Vzc2lvbnMgLyBmb3JlZ3JvdW5kZWQgdGFza3MgdG9vIChub3QganVzdCBsb2NhbFxuICAvLyBxdWVyaWVzLCB3aGljaCByZXNldCB0aGVtIGluIG9uUXVlcnkpLiBXaXRob3V0IHRoaXMsIGEgcmVtb3RlLW9ubHlcbiAgLy8gc2Vzc2lvbiB3b3VsZCBzaG93IH41NiB5ZWFycyBlbGFwc2VkIChEYXRlLm5vdygpIC0gMCkuXG4gIGNvbnN0IHNldElzRXh0ZXJuYWxMb2FkaW5nID0gUmVhY3QudXNlQ2FsbGJhY2soXG4gICAgKHZhbHVlOiBib29sZWFuKSA9PiB7XG4gICAgICBzZXRJc0V4dGVybmFsTG9hZGluZ1Jhdyh2YWx1ZSlcbiAgICAgIGlmICh2YWx1ZSkgcmVzZXRUaW1pbmdSZWZzKClcbiAgICB9LFxuICAgIFtyZXNldFRpbWluZ1JlZnNdLFxuICApXG5cbiAgLy8gU3RhcnQgdGltZSBvZiB0aGUgZmlyc3QgdHVybiB0aGF0IGhhZCBzd2FybSB0ZWFtbWF0ZXMgcnVubmluZ1xuICAvLyBVc2VkIHRvIGNvbXB1dGUgdG90YWwgZWxhcHNlZCB0aW1lIChpbmNsdWRpbmcgdGVhbW1hdGUgZXhlY3V0aW9uKSBmb3IgdGhlIGRlZmVycmVkIG1lc3NhZ2VcbiAgY29uc3Qgc3dhcm1TdGFydFRpbWVSZWYgPSBSZWFjdC51c2VSZWY8bnVtYmVyIHwgbnVsbD4obnVsbClcbiAgY29uc3Qgc3dhcm1CdWRnZXRJbmZvUmVmID0gUmVhY3QudXNlUmVmPFxuICAgIHsgdG9rZW5zOiBudW1iZXI7IGxpbWl0OiBudW1iZXI7IG51ZGdlczogbnVtYmVyIH0gfCB1bmRlZmluZWRcbiAgPih1bmRlZmluZWQpXG5cbiAgLy8gUmVmIHRvIHRyYWNrIGN1cnJlbnQgZm9jdXNlZElucHV0RGlhbG9nIGZvciB1c2UgaW4gY2FsbGJhY2tzXG4gIC8vIFRoaXMgYXZvaWRzIHN0YWxlIGNsb3N1cmVzIHdoZW4gY2hlY2tpbmcgZGlhbG9nIHN0YXRlIGluIHRpbWVyIGNhbGxiYWNrc1xuICBjb25zdCBmb2N1c2VkSW5wdXREaWFsb2dSZWYgPVxuICAgIFJlYWN0LnVzZVJlZjxSZXR1cm5UeXBlPHR5cGVvZiBnZXRGb2N1c2VkSW5wdXREaWFsb2c+Pih1bmRlZmluZWQpXG5cbiAgLy8gSG93IGxvbmcgYWZ0ZXIgdGhlIGxhc3Qga2V5c3Ryb2tlIGJlZm9yZSBkZWZlcnJlZCBkaWFsb2dzIGFyZSBzaG93blxuICBjb25zdCBQUk9NUFRfU1VQUFJFU1NJT05fTVMgPSAxNTAwXG4gIC8vIFRydWUgd2hlbiB1c2VyIGlzIGFjdGl2ZWx5IHR5cGluZyDigJQgZGVmZXJzIGludGVycnVwdCBkaWFsb2dzIHNvIGtleXN0cm9rZXNcbiAgLy8gZG9uJ3QgYWNjaWRlbnRhbGx5IGRpc21pc3Mgb3IgYW5zd2VyIGEgcGVybWlzc2lvbiBwcm9tcHQgdGhlIHVzZXIgaGFzbid0IHJlYWQgeWV0LlxuICBjb25zdCBbaXNQcm9tcHRJbnB1dEFjdGl2ZSwgc2V0SXNQcm9tcHRJbnB1dEFjdGl2ZV0gPSBSZWFjdC51c2VTdGF0ZShmYWxzZSlcblxuICBjb25zdCBbYXV0b1VwZGF0ZXJSZXN1bHQsIHNldEF1dG9VcGRhdGVyUmVzdWx0XSA9XG4gICAgdXNlU3RhdGU8QXV0b1VwZGF0ZXJSZXN1bHQgfCBudWxsPihudWxsKVxuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKGF1dG9VcGRhdGVyUmVzdWx0Py5ub3RpZmljYXRpb25zKSB7XG4gICAgICBhdXRvVXBkYXRlclJlc3VsdC5ub3RpZmljYXRpb25zLmZvckVhY2gobm90aWZpY2F0aW9uID0+IHtcbiAgICAgICAgYWRkTm90aWZpY2F0aW9uKHtcbiAgICAgICAgICBrZXk6ICdhdXRvLXVwZGF0ZXItbm90aWZpY2F0aW9uJyxcbiAgICAgICAgICB0ZXh0OiBub3RpZmljYXRpb24sXG4gICAgICAgICAgcHJpb3JpdHk6ICdsb3cnLFxuICAgICAgICB9KVxuICAgICAgfSlcbiAgICB9XG4gIH0sIFthdXRvVXBkYXRlclJlc3VsdCwgYWRkTm90aWZpY2F0aW9uXSlcblxuICAvLyB0bXV4ICsgZnVsbHNjcmVlbiArIGBtb3VzZSBvZmZgOiBvbmUtdGltZSBoaW50IHRoYXQgd2hlZWwgd29uJ3Qgc2Nyb2xsLlxuICAvLyBXZSBubyBsb25nZXIgbXV0YXRlIHRtdXgncyBzZXNzaW9uLXNjb3BlZCBtb3VzZSBvcHRpb24gKGl0IHBvaXNvbmVkXG4gIC8vIHNpYmxpbmcgcGFuZXMpOyB0bXV4IHVzZXJzIGFscmVhZHkga25vdyB0aGlzIHRyYWRlb2ZmIGZyb20gdmltL2xlc3MuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKGlzRnVsbHNjcmVlbkVudkVuYWJsZWQoKSkge1xuICAgICAgdm9pZCBtYXliZUdldFRtdXhNb3VzZUhpbnQoKS50aGVuKGhpbnQgPT4ge1xuICAgICAgICBpZiAoaGludCkge1xuICAgICAgICAgIGFkZE5vdGlmaWNhdGlvbih7XG4gICAgICAgICAgICBrZXk6ICd0bXV4LW1vdXNlLWhpbnQnLFxuICAgICAgICAgICAgdGV4dDogaGludCxcbiAgICAgICAgICAgIHByaW9yaXR5OiAnbG93JyxcbiAgICAgICAgICB9KVxuICAgICAgICB9XG4gICAgICB9KVxuICAgIH1cbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgcmVhY3QtaG9va3MvZXhoYXVzdGl2ZS1kZXBzXG4gIH0sIFtdKVxuXG4gIGNvbnN0IFtzaG93VW5kZXJjb3ZlckNhbGxvdXQsIHNldFNob3dVbmRlcmNvdmVyQ2FsbG91dF0gPSB1c2VTdGF0ZShmYWxzZSlcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoXCJleHRlcm5hbFwiID09PSAnYW50Jykge1xuICAgICAgdm9pZCAoYXN5bmMgKCkgPT4ge1xuICAgICAgICAvLyBXYWl0IGZvciByZXBvIGNsYXNzaWZpY2F0aW9uIHRvIHNldHRsZSAobWVtb2l6ZWQsIG5vLW9wIGlmIHByaW1lZCkuXG4gICAgICAgIGNvbnN0IHsgaXNJbnRlcm5hbE1vZGVsUmVwbyB9ID0gYXdhaXQgaW1wb3J0KFxuICAgICAgICAgICcuLi91dGlscy9jb21taXRBdHRyaWJ1dGlvbi5qcydcbiAgICAgICAgKVxuICAgICAgICBhd2FpdCBpc0ludGVybmFsTW9kZWxSZXBvKClcbiAgICAgICAgY29uc3QgeyBzaG91bGRTaG93VW5kZXJjb3ZlckF1dG9Ob3RpY2UgfSA9IGF3YWl0IGltcG9ydChcbiAgICAgICAgICAnLi4vdXRpbHMvdW5kZXJjb3Zlci5qcydcbiAgICAgICAgKVxuICAgICAgICBpZiAoc2hvdWxkU2hvd1VuZGVyY292ZXJBdXRvTm90aWNlKCkpIHtcbiAgICAgICAgICBzZXRTaG93VW5kZXJjb3ZlckNhbGxvdXQodHJ1ZSlcbiAgICAgICAgfVxuICAgICAgfSkoKVxuICAgIH1cbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgcmVhY3QtaG9va3MvZXhoYXVzdGl2ZS1kZXBzXG4gIH0sIFtdKVxuXG4gIGNvbnN0IFt0b29sSlNYLCBzZXRUb29sSlNYSW50ZXJuYWxdID0gdXNlU3RhdGU8e1xuICAgIGpzeDogUmVhY3QuUmVhY3ROb2RlIHwgbnVsbFxuICAgIHNob3VsZEhpZGVQcm9tcHRJbnB1dDogYm9vbGVhblxuICAgIHNob3VsZENvbnRpbnVlQW5pbWF0aW9uPzogdHJ1ZVxuICAgIHNob3dTcGlubmVyPzogYm9vbGVhblxuICAgIGlzTG9jYWxKU1hDb21tYW5kPzogYm9vbGVhblxuICAgIGlzSW1tZWRpYXRlPzogYm9vbGVhblxuICB9IHwgbnVsbD4obnVsbClcblxuICAvLyBUcmFjayBsb2NhbCBKU1ggY29tbWFuZHMgc2VwYXJhdGVseSBzbyB0b29scyBjYW4ndCBvdmVyd3JpdGUgdGhlbS5cbiAgLy8gVGhpcyBlbmFibGVzIFwiaW1tZWRpYXRlXCIgY29tbWFuZHMgKGxpa2UgL2J0dykgdG8gcGVyc2lzdCB3aGlsZSBDbGF1ZGUgaXMgcHJvY2Vzc2luZy5cbiAgY29uc3QgbG9jYWxKU1hDb21tYW5kUmVmID0gdXNlUmVmPHtcbiAgICBqc3g6IFJlYWN0LlJlYWN0Tm9kZSB8IG51bGxcbiAgICBzaG91bGRIaWRlUHJvbXB0SW5wdXQ6IGJvb2xlYW5cbiAgICBzaG91bGRDb250aW51ZUFuaW1hdGlvbj86IHRydWVcbiAgICBzaG93U3Bpbm5lcj86IGJvb2xlYW5cbiAgICBpc0xvY2FsSlNYQ29tbWFuZDogdHJ1ZVxuICB9IHwgbnVsbD4obnVsbClcblxuICAvLyBXcmFwcGVyIGZvciBzZXRUb29sSlNYIHRoYXQgcHJlc2VydmVzIGxvY2FsIEpTWCBjb21tYW5kcyAobGlrZSAvYnR3KS5cbiAgLy8gV2hlbiBhIGxvY2FsIEpTWCBjb21tYW5kIGlzIGFjdGl2ZSwgd2UgaWdub3JlIHVwZGF0ZXMgZnJvbSB0b29sc1xuICAvLyB1bmxlc3MgdGhleSBleHBsaWNpdGx5IHNldCBjbGVhckxvY2FsSlNYOiB0cnVlIChmcm9tIG9uRG9uZSBjYWxsYmFja3MpLlxuICAvL1xuICAvLyBUTyBBREQgQSBORVcgSU1NRURJQVRFIENPTU1BTkQ6XG4gIC8vIDEuIFNldCBgaW1tZWRpYXRlOiB0cnVlYCBpbiB0aGUgY29tbWFuZCBkZWZpbml0aW9uXG4gIC8vIDIuIFNldCBgaXNMb2NhbEpTWENvbW1hbmQ6IHRydWVgIHdoZW4gY2FsbGluZyBzZXRUb29sSlNYIGluIHRoZSBjb21tYW5kJ3MgSlNYXG4gIC8vIDMuIEluIHRoZSBvbkRvbmUgY2FsbGJhY2ssIHVzZSBgc2V0VG9vbEpTWCh7IGpzeDogbnVsbCwgc2hvdWxkSGlkZVByb21wdElucHV0OiBmYWxzZSwgY2xlYXJMb2NhbEpTWDogdHJ1ZSB9KWBcbiAgLy8gICAgdG8gZXhwbGljaXRseSBjbGVhciB0aGUgb3ZlcmxheSB3aGVuIHRoZSB1c2VyIGRpc21pc3NlcyBpdFxuICBjb25zdCBzZXRUb29sSlNYID0gdXNlQ2FsbGJhY2soXG4gICAgKFxuICAgICAgYXJnczoge1xuICAgICAgICBqc3g6IFJlYWN0LlJlYWN0Tm9kZSB8IG51bGxcbiAgICAgICAgc2hvdWxkSGlkZVByb21wdElucHV0OiBib29sZWFuXG4gICAgICAgIHNob3VsZENvbnRpbnVlQW5pbWF0aW9uPzogdHJ1ZVxuICAgICAgICBzaG93U3Bpbm5lcj86IGJvb2xlYW5cbiAgICAgICAgaXNMb2NhbEpTWENvbW1hbmQ/OiBib29sZWFuXG4gICAgICAgIGNsZWFyTG9jYWxKU1g/OiBib29sZWFuXG4gICAgICB9IHwgbnVsbCxcbiAgICApID0+IHtcbiAgICAgIC8vIElmIHNldHRpbmcgYSBsb2NhbCBKU1ggY29tbWFuZCwgc3RvcmUgaXQgaW4gdGhlIHJlZlxuICAgICAgaWYgKGFyZ3M/LmlzTG9jYWxKU1hDb21tYW5kKSB7XG4gICAgICAgIGNvbnN0IHsgY2xlYXJMb2NhbEpTWDogXywgLi4ucmVzdCB9ID0gYXJnc1xuICAgICAgICBsb2NhbEpTWENvbW1hbmRSZWYuY3VycmVudCA9IHsgLi4ucmVzdCwgaXNMb2NhbEpTWENvbW1hbmQ6IHRydWUgfVxuICAgICAgICBzZXRUb29sSlNYSW50ZXJuYWwocmVzdClcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG5cbiAgICAgIC8vIElmIHRoZXJlJ3MgYW4gYWN0aXZlIGxvY2FsIEpTWCBjb21tYW5kIGluIHRoZSByZWZcbiAgICAgIGlmIChsb2NhbEpTWENvbW1hbmRSZWYuY3VycmVudCkge1xuICAgICAgICAvLyBBbGxvdyBjbGVhcmluZyBvbmx5IGlmIGV4cGxpY2l0bHkgcmVxdWVzdGVkIChmcm9tIG9uRG9uZSBjYWxsYmFja3MpXG4gICAgICAgIGlmIChhcmdzPy5jbGVhckxvY2FsSlNYKSB7XG4gICAgICAgICAgbG9jYWxKU1hDb21tYW5kUmVmLmN1cnJlbnQgPSBudWxsXG4gICAgICAgICAgc2V0VG9vbEpTWEludGVybmFsKG51bGwpXG4gICAgICAgICAgcmV0dXJuXG4gICAgICAgIH1cbiAgICAgICAgLy8gT3RoZXJ3aXNlLCBrZWVwIHRoZSBsb2NhbCBKU1ggY29tbWFuZCB2aXNpYmxlIC0gaWdub3JlIHRvb2wgdXBkYXRlc1xuICAgICAgICByZXR1cm5cbiAgICAgIH1cblxuICAgICAgLy8gTm8gYWN0aXZlIGxvY2FsIEpTWCBjb21tYW5kLCBhbGxvdyBhbnkgdXBkYXRlXG4gICAgICBpZiAoYXJncz8uY2xlYXJMb2NhbEpTWCkge1xuICAgICAgICBzZXRUb29sSlNYSW50ZXJuYWwobnVsbClcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG4gICAgICBzZXRUb29sSlNYSW50ZXJuYWwoYXJncylcbiAgICB9LFxuICAgIFtdLFxuICApXG4gIGNvbnN0IFt0b29sVXNlQ29uZmlybVF1ZXVlLCBzZXRUb29sVXNlQ29uZmlybVF1ZXVlXSA9IHVzZVN0YXRlPFxuICAgIFRvb2xVc2VDb25maXJtW11cbiAgPihbXSlcbiAgLy8gU3RpY2t5IGZvb3RlciBKU1ggcmVnaXN0ZXJlZCBieSBwZXJtaXNzaW9uIHJlcXVlc3QgY29tcG9uZW50cyAoY3VycmVudGx5XG4gIC8vIG9ubHkgRXhpdFBsYW5Nb2RlUGVybWlzc2lvblJlcXVlc3QpLiBSZW5kZXJzIGluIEZ1bGxzY3JlZW5MYXlvdXQncyBgYm90dG9tYFxuICAvLyBzbG90IHNvIHJlc3BvbnNlIG9wdGlvbnMgc3RheSB2aXNpYmxlIHdoaWxlIHRoZSB1c2VyIHNjcm9sbHMgYSBsb25nIHBsYW4uXG4gIGNvbnN0IFtwZXJtaXNzaW9uU3RpY2t5Rm9vdGVyLCBzZXRQZXJtaXNzaW9uU3RpY2t5Rm9vdGVyXSA9XG4gICAgdXNlU3RhdGU8UmVhY3QuUmVhY3ROb2RlIHwgbnVsbD4obnVsbClcbiAgY29uc3QgW3NhbmRib3hQZXJtaXNzaW9uUmVxdWVzdFF1ZXVlLCBzZXRTYW5kYm94UGVybWlzc2lvblJlcXVlc3RRdWV1ZV0gPVxuICAgIHVzZVN0YXRlPFxuICAgICAgQXJyYXk8e1xuICAgICAgICBob3N0UGF0dGVybjogTmV0d29ya0hvc3RQYXR0ZXJuXG4gICAgICAgIHJlc29sdmVQcm9taXNlOiAoYWxsb3dDb25uZWN0aW9uOiBib29sZWFuKSA9PiB2b2lkXG4gICAgICB9PlxuICAgID4oW10pXG4gIGNvbnN0IFtwcm9tcHRRdWV1ZSwgc2V0UHJvbXB0UXVldWVdID0gdXNlU3RhdGU8XG4gICAgQXJyYXk8e1xuICAgICAgcmVxdWVzdDogUHJvbXB0UmVxdWVzdFxuICAgICAgdGl0bGU6IHN0cmluZ1xuICAgICAgdG9vbElucHV0U3VtbWFyeT86IHN0cmluZyB8IG51bGxcbiAgICAgIHJlc29sdmU6IChyZXNwb25zZTogUHJvbXB0UmVzcG9uc2UpID0+IHZvaWRcbiAgICAgIHJlamVjdDogKGVycm9yOiBFcnJvcikgPT4gdm9pZFxuICAgIH0+XG4gID4oW10pXG5cbiAgLy8gVHJhY2sgYnJpZGdlIGNsZWFudXAgZnVuY3Rpb25zIGZvciBzYW5kYm94IHBlcm1pc3Npb24gcmVxdWVzdHMgc28gdGhlXG4gIC8vIGxvY2FsIGRpYWxvZyBoYW5kbGVyIGNhbiBjYW5jZWwgdGhlIHJlbW90ZSBwcm9tcHQgd2hlbiB0aGUgbG9jYWwgdXNlclxuICAvLyByZXNwb25kcyBmaXJzdC4gS2V5ZWQgYnkgaG9zdCB0byBzdXBwb3J0IGNvbmN1cnJlbnQgc2FtZS1ob3N0IHJlcXVlc3RzLlxuICBjb25zdCBzYW5kYm94QnJpZGdlQ2xlYW51cFJlZiA9IHVzZVJlZjxNYXA8c3RyaW5nLCBBcnJheTwoKSA9PiB2b2lkPj4+KFxuICAgIG5ldyBNYXAoKSxcbiAgKVxuXG4gIC8vIC0tIFRlcm1pbmFsIHRpdGxlIG1hbmFnZW1lbnRcbiAgLy8gU2Vzc2lvbiB0aXRsZSAoc2V0IHZpYSAvcmVuYW1lIG9yIHJlc3RvcmVkIG9uIHJlc3VtZSkgd2lucyBvdmVyXG4gIC8vIHRoZSBhZ2VudCBuYW1lLCB3aGljaCB3aW5zIG92ZXIgdGhlIEhhaWt1LWV4dHJhY3RlZCB0b3BpYztcbiAgLy8gYWxsIGZhbGwgYmFjayB0byB0aGUgcHJvZHVjdCBuYW1lLlxuICBjb25zdCB0ZXJtaW5hbFRpdGxlRnJvbVJlbmFtZSA9XG4gICAgdXNlQXBwU3RhdGUocyA9PiBzLnNldHRpbmdzLnRlcm1pbmFsVGl0bGVGcm9tUmVuYW1lKSAhPT0gZmFsc2VcbiAgY29uc3Qgc2Vzc2lvblRpdGxlID0gdGVybWluYWxUaXRsZUZyb21SZW5hbWVcbiAgICA/IGdldEN1cnJlbnRTZXNzaW9uVGl0bGUoZ2V0U2Vzc2lvbklkKCkpXG4gICAgOiB1bmRlZmluZWRcbiAgY29uc3QgW2hhaWt1VGl0bGUsIHNldEhhaWt1VGl0bGVdID0gdXNlU3RhdGU8c3RyaW5nPigpXG4gIC8vIEdhdGVzIHRoZSBvbmUtc2hvdCBIYWlrdSBjYWxsIHRoYXQgZ2VuZXJhdGVzIHRoZSB0YWIgdGl0bGUuIFNlZWRlZCB0cnVlXG4gIC8vIG9uIHJlc3VtZSAoaW5pdGlhbE1lc3NhZ2VzIHByZXNlbnQpIHNvIHdlIGRvbid0IHJlLXRpdGxlIGEgcmVzdW1lZFxuICAvLyBzZXNzaW9uIGZyb20gbWlkLWNvbnZlcnNhdGlvbiBjb250ZXh0LlxuICBjb25zdCBoYWlrdVRpdGxlQXR0ZW1wdGVkUmVmID0gdXNlUmVmKChpbml0aWFsTWVzc2FnZXM/Lmxlbmd0aCA/PyAwKSA+IDApXG4gIGNvbnN0IGFnZW50VGl0bGUgPSBtYWluVGhyZWFkQWdlbnREZWZpbml0aW9uPy5hZ2VudFR5cGVcbiAgY29uc3QgdGVybWluYWxUaXRsZSA9XG4gICAgc2Vzc2lvblRpdGxlID8/IGFnZW50VGl0bGUgPz8gaGFpa3VUaXRsZSA/PyAnQ2xhdWRlIENvZGUnXG4gIGNvbnN0IGlzV2FpdGluZ0ZvckFwcHJvdmFsID1cbiAgICB0b29sVXNlQ29uZmlybVF1ZXVlLmxlbmd0aCA+IDAgfHxcbiAgICBwcm9tcHRRdWV1ZS5sZW5ndGggPiAwIHx8XG4gICAgcGVuZGluZ1dvcmtlclJlcXVlc3QgfHxcbiAgICBwZW5kaW5nU2FuZGJveFJlcXVlc3RcbiAgLy8gTG9jYWwtanN4IGNvbW1hbmRzIChsaWtlIC9wbHVnaW4sIC9jb25maWcpIHNob3cgdXNlci1mYWNpbmcgZGlhbG9ncyB0aGF0XG4gIC8vIHdhaXQgZm9yIGlucHV0LiBSZXF1aXJlIGpzeCAhPSBudWxsIOKAlCBpZiB0aGUgZmxhZyBpcyBzdHVjayB0cnVlIGJ1dCBqc3hcbiAgLy8gaXMgbnVsbCwgdHJlYXQgYXMgbm90LXNob3dpbmcgc28gVGV4dElucHV0IGZvY3VzIGFuZCBxdWV1ZSBwcm9jZXNzb3JcbiAgLy8gYXJlbid0IGRlYWRsb2NrZWQgYnkgYSBwaGFudG9tIG92ZXJsYXkuXG4gIGNvbnN0IGlzU2hvd2luZ0xvY2FsSlNYQ29tbWFuZCA9XG4gICAgdG9vbEpTWD8uaXNMb2NhbEpTWENvbW1hbmQgPT09IHRydWUgJiYgdG9vbEpTWD8uanN4ICE9IG51bGxcbiAgY29uc3QgdGl0bGVJc0FuaW1hdGluZyA9XG4gICAgaXNMb2FkaW5nICYmICFpc1dhaXRpbmdGb3JBcHByb3ZhbCAmJiAhaXNTaG93aW5nTG9jYWxKU1hDb21tYW5kXG4gIC8vIFRpdGxlIGFuaW1hdGlvbiBzdGF0ZSBsaXZlcyBpbiA8QW5pbWF0ZWRUZXJtaW5hbFRpdGxlPiBzbyB0aGUgOTYwbXMgdGlja1xuICAvLyBkb2Vzbid0IHJlLXJlbmRlciBSRVBMLiB0aXRsZURpc2FibGVkL3Rlcm1pbmFsVGl0bGUgYXJlIHN0aWxsIGNvbXB1dGVkXG4gIC8vIGhlcmUgYmVjYXVzZSBvblF1ZXJ5SW1wbCByZWFkcyB0aGVtIChiYWNrZ3JvdW5kIHNlc3Npb24gZGVzY3JpcHRpb24sXG4gIC8vIGhhaWt1IHRpdGxlIGV4dHJhY3Rpb24gZ2F0ZSkuXG5cbiAgLy8gUHJldmVudCBtYWNPUyBmcm9tIHNsZWVwaW5nIHdoaWxlIENsYXVkZSBpcyB3b3JraW5nXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKGlzTG9hZGluZyAmJiAhaXNXYWl0aW5nRm9yQXBwcm92YWwgJiYgIWlzU2hvd2luZ0xvY2FsSlNYQ29tbWFuZCkge1xuICAgICAgc3RhcnRQcmV2ZW50U2xlZXAoKVxuICAgICAgcmV0dXJuICgpID0+IHN0b3BQcmV2ZW50U2xlZXAoKVxuICAgIH1cbiAgfSwgW2lzTG9hZGluZywgaXNXYWl0aW5nRm9yQXBwcm92YWwsIGlzU2hvd2luZ0xvY2FsSlNYQ29tbWFuZF0pXG5cbiAgY29uc3Qgc2Vzc2lvblN0YXR1czogVGFiU3RhdHVzS2luZCA9XG4gICAgaXNXYWl0aW5nRm9yQXBwcm92YWwgfHwgaXNTaG93aW5nTG9jYWxKU1hDb21tYW5kXG4gICAgICA/ICd3YWl0aW5nJ1xuICAgICAgOiBpc0xvYWRpbmdcbiAgICAgICAgPyAnYnVzeSdcbiAgICAgICAgOiAnaWRsZSdcblxuICBjb25zdCB3YWl0aW5nRm9yID1cbiAgICBzZXNzaW9uU3RhdHVzICE9PSAnd2FpdGluZydcbiAgICAgID8gdW5kZWZpbmVkXG4gICAgICA6IHRvb2xVc2VDb25maXJtUXVldWUubGVuZ3RoID4gMFxuICAgICAgICA/IGBhcHByb3ZlICR7dG9vbFVzZUNvbmZpcm1RdWV1ZVswXSEudG9vbC5uYW1lfWBcbiAgICAgICAgOiBwZW5kaW5nV29ya2VyUmVxdWVzdFxuICAgICAgICAgID8gJ3dvcmtlciByZXF1ZXN0J1xuICAgICAgICAgIDogcGVuZGluZ1NhbmRib3hSZXF1ZXN0XG4gICAgICAgICAgICA/ICdzYW5kYm94IHJlcXVlc3QnXG4gICAgICAgICAgICA6IGlzU2hvd2luZ0xvY2FsSlNYQ29tbWFuZFxuICAgICAgICAgICAgICA/ICdkaWFsb2cgb3BlbidcbiAgICAgICAgICAgICAgOiAnaW5wdXQgbmVlZGVkJ1xuXG4gIC8vIFB1c2ggc3RhdHVzIHRvIHRoZSBQSUQgZmlsZSBmb3IgYGNsYXVkZSBwc2AuIEZpcmUtYW5kLWZvcmdldDsgcHMgZmFsbHNcbiAgLy8gYmFjayB0byB0cmFuc2NyaXB0LXRhaWwgZGVyaXZhdGlvbiB3aGVuIHRoaXMgaXMgbWlzc2luZy9zdGFsZS5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoZmVhdHVyZSgnQkdfU0VTU0lPTlMnKSkge1xuICAgICAgdm9pZCB1cGRhdGVTZXNzaW9uQWN0aXZpdHkoeyBzdGF0dXM6IHNlc3Npb25TdGF0dXMsIHdhaXRpbmdGb3IgfSlcbiAgICB9XG4gIH0sIFtzZXNzaW9uU3RhdHVzLCB3YWl0aW5nRm9yXSlcblxuICAvLyAzUCBkZWZhdWx0OiBvZmYg4oCUIE9TQyAyMTMzNyBpcyBhbnQtb25seSB3aGlsZSB0aGUgc3BlYyBzdGFiaWxpemVzLlxuICAvLyBHYXRlZCBzbyB3ZSBjYW4gcm9sbCBiYWNrIGlmIHRoZSBzaWRlYmFyIGluZGljYXRvciBjb25mbGljdHMgd2l0aFxuICAvLyB0aGUgdGl0bGUgc3Bpbm5lciBpbiB0ZXJtaW5hbHMgdGhhdCByZW5kZXIgYm90aC4gV2hlbiB0aGUgZmxhZyBpc1xuICAvLyBvbiwgdGhlIHVzZXItZmFjaW5nIGNvbmZpZyBzZXR0aW5nIGNvbnRyb2xzIHdoZXRoZXIgaXQncyBhY3RpdmUuXG4gIGNvbnN0IHRhYlN0YXR1c0dhdGVFbmFibGVkID0gZ2V0RmVhdHVyZVZhbHVlX0NBQ0hFRF9NQVlfQkVfU1RBTEUoXG4gICAgJ3Rlbmd1X3Rlcm1pbmFsX3NpZGViYXInLFxuICAgIGZhbHNlLFxuICApXG4gIGNvbnN0IHNob3dTdGF0dXNJblRlcm1pbmFsVGFiID1cbiAgICB0YWJTdGF0dXNHYXRlRW5hYmxlZCAmJiAoZ2V0R2xvYmFsQ29uZmlnKCkuc2hvd1N0YXR1c0luVGVybWluYWxUYWIgPz8gZmFsc2UpXG4gIHVzZVRhYlN0YXR1cyh0aXRsZURpc2FibGVkIHx8ICFzaG93U3RhdHVzSW5UZXJtaW5hbFRhYiA/IG51bGwgOiBzZXNzaW9uU3RhdHVzKVxuXG4gIC8vIFJlZ2lzdGVyIHRoZSBsZWFkZXIncyBzZXRUb29sVXNlQ29uZmlybVF1ZXVlIGZvciBpbi1wcm9jZXNzIHRlYW1tYXRlc1xuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIHJlZ2lzdGVyTGVhZGVyVG9vbFVzZUNvbmZpcm1RdWV1ZShzZXRUb29sVXNlQ29uZmlybVF1ZXVlKVxuICAgIHJldHVybiAoKSA9PiB1bnJlZ2lzdGVyTGVhZGVyVG9vbFVzZUNvbmZpcm1RdWV1ZSgpXG4gIH0sIFtzZXRUb29sVXNlQ29uZmlybVF1ZXVlXSlcblxuICBjb25zdCBbbWVzc2FnZXMsIHJhd1NldE1lc3NhZ2VzXSA9IHVzZVN0YXRlPE1lc3NhZ2VUeXBlW10+KFxuICAgIGluaXRpYWxNZXNzYWdlcyA/PyBbXSxcbiAgKVxuICBjb25zdCBtZXNzYWdlc1JlZiA9IHVzZVJlZihtZXNzYWdlcylcbiAgLy8gU3RvcmVzIHRoZSB3aWxsb3dNb2RlIHZhcmlhbnQgdGhhdCB3YXMgc2hvd24gKG9yIGZhbHNlIGlmIG5vIGhpbnQgc2hvd24pLlxuICAvLyBDYXB0dXJlZCBhdCBoaW50X3Nob3duIHRpbWUgc28gaGludF9jb252ZXJ0ZWQgdGVsZW1ldHJ5IHJlcG9ydHMgdGhlIHNhbWVcbiAgLy8gdmFyaWFudCDigJQgdGhlIEdyb3d0aEJvb2sgdmFsdWUgc2hvdWxkbid0IGNoYW5nZSBtaWQtc2Vzc2lvbiwgYnV0IHJlYWRpbmdcbiAgLy8gaXQgb25jZSBndWFyYW50ZWVzIGNvbnNpc3RlbmN5IGJldHdlZW4gdGhlIHBhaXJlZCBldmVudHMuXG4gIGNvbnN0IGlkbGVIaW50U2hvd25SZWYgPSB1c2VSZWY8c3RyaW5nIHwgZmFsc2U+KGZhbHNlKVxuICAvLyBXcmFwIHNldE1lc3NhZ2VzIHNvIG1lc3NhZ2VzUmVmIGlzIGFsd2F5cyBjdXJyZW50IHRoZSBpbnN0YW50IHRoZVxuICAvLyBjYWxsIHJldHVybnMg4oCUIG5vdCB3aGVuIFJlYWN0IGxhdGVyIHByb2Nlc3NlcyB0aGUgYmF0Y2guICBBcHBseSB0aGVcbiAgLy8gdXBkYXRlciBlYWdlcmx5IGFnYWluc3QgdGhlIHJlZiwgdGhlbiBoYW5kIFJlYWN0IHRoZSBjb21wdXRlZCB2YWx1ZVxuICAvLyAobm90IHRoZSBmdW5jdGlvbikuICByYXdTZXRNZXNzYWdlcyBiYXRjaGluZyBiZWNvbWVzIGxhc3Qtd3JpdGUtd2lucyxcbiAgLy8gYW5kIHRoZSBsYXN0IHdyaXRlIGlzIGNvcnJlY3QgYmVjYXVzZSBlYWNoIGNhbGwgY29tcG9zZXMgYWdhaW5zdCB0aGVcbiAgLy8gYWxyZWFkeS11cGRhdGVkIHJlZi4gIFRoaXMgaXMgdGhlIFp1c3RhbmQgcGF0dGVybjogcmVmIGlzIHNvdXJjZSBvZlxuICAvLyB0cnV0aCwgUmVhY3Qgc3RhdGUgaXMgdGhlIHJlbmRlciBwcm9qZWN0aW9uLiAgV2l0aG91dCB0aGlzLCBwYXRoc1xuICAvLyB0aGF0IHF1ZXVlIGZ1bmN0aW9uYWwgdXBkYXRlcnMgdGhlbiBzeW5jaHJvbm91c2x5IHJlYWQgdGhlIHJlZlxuICAvLyAoZS5nLiBoYW5kbGVTcGVjdWxhdGlvbkFjY2VwdCDihpIgb25RdWVyeSkgc2VlIHN0YWxlIGRhdGEuXG4gIGNvbnN0IHNldE1lc3NhZ2VzID0gdXNlQ2FsbGJhY2soXG4gICAgKGFjdGlvbjogUmVhY3QuU2V0U3RhdGVBY3Rpb248TWVzc2FnZVR5cGVbXT4pID0+IHtcbiAgICAgIGNvbnN0IHByZXYgPSBtZXNzYWdlc1JlZi5jdXJyZW50XG4gICAgICBjb25zdCBuZXh0ID1cbiAgICAgICAgdHlwZW9mIGFjdGlvbiA9PT0gJ2Z1bmN0aW9uJyA/IGFjdGlvbihtZXNzYWdlc1JlZi5jdXJyZW50KSA6IGFjdGlvblxuICAgICAgbWVzc2FnZXNSZWYuY3VycmVudCA9IG5leHRcbiAgICAgIGlmIChuZXh0Lmxlbmd0aCA8IHVzZXJJbnB1dEJhc2VsaW5lUmVmLmN1cnJlbnQpIHtcbiAgICAgICAgLy8gU2hyYW5rIChjb21wYWN0L3Jld2luZC9jbGVhcikg4oCUIGNsYW1wIHNvIHBsYWNlaG9sZGVyVGV4dCdzIGxlbmd0aFxuICAgICAgICAvLyBjaGVjayBjYW4ndCBnbyBzdGFsZS5cbiAgICAgICAgdXNlcklucHV0QmFzZWxpbmVSZWYuY3VycmVudCA9IDBcbiAgICAgIH0gZWxzZSBpZiAobmV4dC5sZW5ndGggPiBwcmV2Lmxlbmd0aCAmJiB1c2VyTWVzc2FnZVBlbmRpbmdSZWYuY3VycmVudCkge1xuICAgICAgICAvLyBHcmV3IHdoaWxlIHRoZSBzdWJtaXR0ZWQgdXNlciBtZXNzYWdlIGhhc24ndCBsYW5kZWQgeWV0LiBJZiB0aGVcbiAgICAgICAgLy8gYWRkZWQgbWVzc2FnZXMgZG9uJ3QgaW5jbHVkZSBpdCAoYnJpZGdlIHN0YXR1cywgaG9vayByZXN1bHRzLFxuICAgICAgICAvLyBzY2hlZHVsZWQgdGFza3MgbGFuZGluZyBhc3luYyBkdXJpbmcgcHJvY2Vzc1VzZXJJbnB1dEJhc2UpLCBidW1wXG4gICAgICAgIC8vIGJhc2VsaW5lIHNvIHRoZSBwbGFjZWhvbGRlciBzdGF5cyB2aXNpYmxlLiBPbmNlIHRoZSB1c2VyIG1lc3NhZ2VcbiAgICAgICAgLy8gbGFuZHMsIHN0b3AgdHJhY2tpbmcg4oCUIGxhdGVyIGFkZGl0aW9ucyAoYXNzaXN0YW50IHN0cmVhbSkgc2hvdWxkXG4gICAgICAgIC8vIG5vdCByZS1zaG93IHRoZSBwbGFjZWhvbGRlci5cbiAgICAgICAgY29uc3QgZGVsdGEgPSBuZXh0Lmxlbmd0aCAtIHByZXYubGVuZ3RoXG4gICAgICAgIGNvbnN0IGFkZGVkID1cbiAgICAgICAgICBwcmV2Lmxlbmd0aCA9PT0gMCB8fCBuZXh0WzBdID09PSBwcmV2WzBdXG4gICAgICAgICAgICA/IG5leHQuc2xpY2UoLWRlbHRhKVxuICAgICAgICAgICAgOiBuZXh0LnNsaWNlKDAsIGRlbHRhKVxuICAgICAgICBpZiAoYWRkZWQuc29tZShpc0h1bWFuVHVybikpIHtcbiAgICAgICAgICB1c2VyTWVzc2FnZVBlbmRpbmdSZWYuY3VycmVudCA9IGZhbHNlXG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgdXNlcklucHV0QmFzZWxpbmVSZWYuY3VycmVudCA9IG5leHQubGVuZ3RoXG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIHJhd1NldE1lc3NhZ2VzKG5leHQpXG4gICAgfSxcbiAgICBbXSxcbiAgKVxuICAvLyBDYXB0dXJlIHRoZSBiYXNlbGluZSBtZXNzYWdlIGNvdW50IGFsb25nc2lkZSB0aGUgcGxhY2Vob2xkZXIgdGV4dCBzb1xuICAvLyB0aGUgcmVuZGVyIGNhbiBoaWRlIGl0IG9uY2UgZGlzcGxheWVkTWVzc2FnZXMgZ3Jvd3MgcGFzdCB0aGUgYmFzZWxpbmUuXG4gIGNvbnN0IHNldFVzZXJJbnB1dE9uUHJvY2Vzc2luZyA9IHVzZUNhbGxiYWNrKChpbnB1dDogc3RyaW5nIHwgdW5kZWZpbmVkKSA9PiB7XG4gICAgaWYgKGlucHV0ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgIHVzZXJJbnB1dEJhc2VsaW5lUmVmLmN1cnJlbnQgPSBtZXNzYWdlc1JlZi5jdXJyZW50Lmxlbmd0aFxuICAgICAgdXNlck1lc3NhZ2VQZW5kaW5nUmVmLmN1cnJlbnQgPSB0cnVlXG4gICAgfSBlbHNlIHtcbiAgICAgIHVzZXJNZXNzYWdlUGVuZGluZ1JlZi5jdXJyZW50ID0gZmFsc2VcbiAgICB9XG4gICAgc2V0VXNlcklucHV0T25Qcm9jZXNzaW5nUmF3KGlucHV0KVxuICB9LCBbXSlcbiAgLy8gRnVsbHNjcmVlbjogdHJhY2sgdGhlIHVuc2Vlbi1kaXZpZGVyIHBvc2l0aW9uLiBkaXZpZGVySW5kZXggY2hhbmdlc1xuICAvLyBvbmx5IH50d2ljZS9zY3JvbGwtc2Vzc2lvbiAoZmlyc3Qgc2Nyb2xsLWF3YXkgKyByZXBpbikuIHBpbGxWaXNpYmxlXG4gIC8vIGFuZCBzdGlja3lQcm9tcHQgbm93IGxpdmUgaW4gRnVsbHNjcmVlbkxheW91dCDigJQgdGhleSBzdWJzY3JpYmUgdG9cbiAgLy8gU2Nyb2xsQm94IGRpcmVjdGx5IHNvIHBlci1mcmFtZSBzY3JvbGwgbmV2ZXIgcmUtcmVuZGVycyBSRVBMLlxuICBjb25zdCB7XG4gICAgZGl2aWRlckluZGV4LFxuICAgIGRpdmlkZXJZUmVmLFxuICAgIG9uU2Nyb2xsQXdheSxcbiAgICBvblJlcGluLFxuICAgIGp1bXBUb05ldyxcbiAgICBzaGlmdERpdmlkZXIsXG4gIH0gPSB1c2VVbnNlZW5EaXZpZGVyKG1lc3NhZ2VzLmxlbmd0aClcbiAgaWYgKGZlYXR1cmUoJ0FXQVlfU1VNTUFSWScpKSB7XG4gICAgLy8gYmlvbWUtaWdub3JlIGxpbnQvY29ycmVjdG5lc3MvdXNlSG9va0F0VG9wTGV2ZWw6IGZlYXR1cmUoKSBpcyBhIGNvbXBpbGUtdGltZSBjb25zdGFudFxuICAgIHVzZUF3YXlTdW1tYXJ5KG1lc3NhZ2VzLCBzZXRNZXNzYWdlcywgaXNMb2FkaW5nKVxuICB9XG4gIGNvbnN0IFtjdXJzb3IsIHNldEN1cnNvcl0gPSB1c2VTdGF0ZTxNZXNzYWdlQWN0aW9uc1N0YXRlIHwgbnVsbD4obnVsbClcbiAgY29uc3QgY3Vyc29yTmF2UmVmID0gdXNlUmVmPE1lc3NhZ2VBY3Rpb25zTmF2IHwgbnVsbD4obnVsbClcbiAgLy8gTWVtb2l6ZWQgc28gTWVzc2FnZXMnIFJlYWN0Lm1lbW8gaG9sZHMuXG4gIGNvbnN0IHVuc2VlbkRpdmlkZXIgPSB1c2VNZW1vKFxuICAgICgpID0+IGNvbXB1dGVVbnNlZW5EaXZpZGVyKG1lc3NhZ2VzLCBkaXZpZGVySW5kZXgpLFxuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSByZWFjdC1ob29rcy9leGhhdXN0aXZlLWRlcHMgLS0gbGVuZ3RoIGNoYW5nZSBjb3ZlcnMgYXBwZW5kczsgdXNlVW5zZWVuRGl2aWRlcidzIGNvdW50LWRyb3AgZ3VhcmQgY2xlYXJzIGRpdmlkZXJJbmRleCBvbiByZXBsYWNlL3Jld2luZFxuICAgIFtkaXZpZGVySW5kZXgsIG1lc3NhZ2VzLmxlbmd0aF0sXG4gIClcbiAgLy8gUmUtcGluIHNjcm9sbCB0byBib3R0b20gYW5kIGNsZWFyIHRoZSB1bnNlZW4tbWVzc2FnZXMgYmFzZWxpbmUuIENhbGxlZFxuICAvLyBvbiBhbnkgdXNlci1kcml2ZW4gcmV0dXJuLXRvLWxpdmUgYWN0aW9uIChzdWJtaXQsIHR5cGUtaW50by1lbXB0eSxcbiAgLy8gb3ZlcmxheSBhcHBlYXIvZGlzbWlzcykuXG4gIGNvbnN0IHJlcGluU2Nyb2xsID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIHNjcm9sbFJlZi5jdXJyZW50Py5zY3JvbGxUb0JvdHRvbSgpXG4gICAgb25SZXBpbigpXG4gICAgc2V0Q3Vyc29yKG51bGwpXG4gIH0sIFtvblJlcGluLCBzZXRDdXJzb3JdKVxuICAvLyBCYWNrc3RvcCBmb3IgdGhlIHN1Ym1pdC1oYW5kbGVyIHJlcGluIGF0IG9uU3VibWl0LiBJZiBhIGJ1ZmZlcmVkIHN0ZGluXG4gIC8vIGV2ZW50ICh3aGVlbC9kcmFnKSByYWNlcyBiZXR3ZWVuIGhhbmRsZXItZmlyZSBhbmQgc3RhdGUtY29tbWl0LCB0aGVcbiAgLy8gaGFuZGxlcidzIHNjcm9sbFRvQm90dG9tIGNhbiBiZSB1bmRvbmUuIFRoaXMgZWZmZWN0IGZpcmVzIG9uIHRoZSByZW5kZXJcbiAgLy8gd2hlcmUgdGhlIHVzZXIncyBtZXNzYWdlIGFjdHVhbGx5IGxhbmRzIOKAlCB0aWVkIHRvIFJlYWN0J3MgY29tbWl0IGN5Y2xlLFxuICAvLyBzbyBpdCBjYW4ndCByYWNlIHdpdGggc3RkaW4uIEtleWVkIG9uIGxhc3RNc2cgaWRlbnRpdHkgKG5vdCBtZXNzYWdlcy5sZW5ndGgpXG4gIC8vIHNvIHVzZUFzc2lzdGFudEhpc3RvcnkncyBwcmVwZW5kcyBkb24ndCBzcHVyaW91c2x5IHJlcGluLlxuICBjb25zdCBsYXN0TXNnID0gbWVzc2FnZXMuYXQoLTEpXG4gIGNvbnN0IGxhc3RNc2dJc0h1bWFuID0gbGFzdE1zZyAhPSBudWxsICYmIGlzSHVtYW5UdXJuKGxhc3RNc2cpXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKGxhc3RNc2dJc0h1bWFuKSB7XG4gICAgICByZXBpblNjcm9sbCgpXG4gICAgfVxuICB9LCBbbGFzdE1zZ0lzSHVtYW4sIGxhc3RNc2csIHJlcGluU2Nyb2xsXSlcbiAgLy8gQXNzaXN0YW50LWNoYXQ6IGxhenktbG9hZCByZW1vdGUgaGlzdG9yeSBvbiBzY3JvbGwtdXAuIE5vLW9wIHVubGVzc1xuICAvLyBLQUlST1MgYnVpbGQgKyBjb25maWcudmlld2VyT25seS4gZmVhdHVyZSgpIGlzIGJ1aWxkLXRpbWUgY29uc3RhbnQgc29cbiAgLy8gdGhlIGJyYW5jaCBpcyBkZWFkLWNvZGUtZWxpbWluYXRlZCBpbiBub24tS0FJUk9TIGJ1aWxkcyAoc2FtZSBwYXR0ZXJuXG4gIC8vIGFzIHVzZVVuc2VlbkRpdmlkZXIgYWJvdmUpLlxuICBjb25zdCB7IG1heWJlTG9hZE9sZGVyIH0gPSBmZWF0dXJlKCdLQUlST1MnKVxuICAgID8gLy8gYmlvbWUtaWdub3JlIGxpbnQvY29ycmVjdG5lc3MvdXNlSG9va0F0VG9wTGV2ZWw6IGZlYXR1cmUoKSBpcyBhIGNvbXBpbGUtdGltZSBjb25zdGFudFxuICAgICAgdXNlQXNzaXN0YW50SGlzdG9yeSh7XG4gICAgICAgIGNvbmZpZzogcmVtb3RlU2Vzc2lvbkNvbmZpZyxcbiAgICAgICAgc2V0TWVzc2FnZXMsXG4gICAgICAgIHNjcm9sbFJlZixcbiAgICAgICAgb25QcmVwZW5kOiBzaGlmdERpdmlkZXIsXG4gICAgICB9KVxuICAgIDogSElTVE9SWV9TVFVCXG4gIC8vIENvbXBvc2UgdXNlVW5zZWVuRGl2aWRlcidzIGNhbGxiYWNrcyB3aXRoIHRoZSBsYXp5LWxvYWQgdHJpZ2dlci5cbiAgY29uc3QgY29tcG9zZWRPblNjcm9sbCA9IHVzZUNhbGxiYWNrKFxuICAgIChzdGlja3k6IGJvb2xlYW4sIGhhbmRsZTogU2Nyb2xsQm94SGFuZGxlKSA9PiB7XG4gICAgICBsYXN0VXNlclNjcm9sbFRzUmVmLmN1cnJlbnQgPSBEYXRlLm5vdygpXG4gICAgICBpZiAoc3RpY2t5KSB7XG4gICAgICAgIG9uUmVwaW4oKVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgb25TY3JvbGxBd2F5KGhhbmRsZSlcbiAgICAgICAgaWYgKGZlYXR1cmUoJ0tBSVJPUycpKSBtYXliZUxvYWRPbGRlcihoYW5kbGUpXG4gICAgICAgIC8vIERpc21pc3MgdGhlIGNvbXBhbmlvbiBidWJibGUgb24gc2Nyb2xsIOKAlCBpdCdzIGFic29sdXRlLXBvc2l0aW9uZWRcbiAgICAgICAgLy8gYXQgYm90dG9tLXJpZ2h0IGFuZCBjb3ZlcnMgdHJhbnNjcmlwdCBjb250ZW50LiBTY3JvbGxpbmcgPSB1c2VyIGlzXG4gICAgICAgIC8vIHRyeWluZyB0byByZWFkIHNvbWV0aGluZyB1bmRlciBpdC5cbiAgICAgICAgaWYgKGZlYXR1cmUoJ0JVRERZJykpIHtcbiAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+XG4gICAgICAgICAgICBwcmV2LmNvbXBhbmlvblJlYWN0aW9uID09PSB1bmRlZmluZWRcbiAgICAgICAgICAgICAgPyBwcmV2XG4gICAgICAgICAgICAgIDogeyAuLi5wcmV2LCBjb21wYW5pb25SZWFjdGlvbjogdW5kZWZpbmVkIH0sXG4gICAgICAgICAgKVxuICAgICAgICB9XG4gICAgICB9XG4gICAgfSxcbiAgICBbb25SZXBpbiwgb25TY3JvbGxBd2F5LCBtYXliZUxvYWRPbGRlciwgc2V0QXBwU3RhdGVdLFxuICApXG4gIC8vIERlZmVycmVkIFNlc3Npb25TdGFydCBob29rIG1lc3NhZ2VzIOKAlCBSRVBMIHJlbmRlcnMgaW1tZWRpYXRlbHkgYW5kXG4gIC8vIGhvb2sgbWVzc2FnZXMgYXJlIGluamVjdGVkIHdoZW4gdGhleSByZXNvbHZlLiBhd2FpdFBlbmRpbmdIb29rcygpXG4gIC8vIG11c3QgYmUgY2FsbGVkIGJlZm9yZSB0aGUgZmlyc3QgQVBJIGNhbGwgc28gdGhlIG1vZGVsIHNlZXMgaG9vayBjb250ZXh0LlxuICBjb25zdCBhd2FpdFBlbmRpbmdIb29rcyA9IHVzZURlZmVycmVkSG9va01lc3NhZ2VzKFxuICAgIHBlbmRpbmdIb29rTWVzc2FnZXMsXG4gICAgc2V0TWVzc2FnZXMsXG4gIClcblxuICAvLyBEZWZlcnJlZCBtZXNzYWdlcyBmb3IgdGhlIE1lc3NhZ2VzIGNvbXBvbmVudCDigJQgcmVuZGVycyBhdCB0cmFuc2l0aW9uXG4gIC8vIHByaW9yaXR5IHNvIHRoZSByZWNvbmNpbGVyIHlpZWxkcyBldmVyeSA1bXMsIGtlZXBpbmcgaW5wdXQgcmVzcG9uc2l2ZVxuICAvLyB3aGlsZSB0aGUgZXhwZW5zaXZlIG1lc3NhZ2UgcHJvY2Vzc2luZyBwaXBlbGluZSBydW5zLlxuICBjb25zdCBkZWZlcnJlZE1lc3NhZ2VzID0gdXNlRGVmZXJyZWRWYWx1ZShtZXNzYWdlcylcbiAgY29uc3QgZGVmZXJyZWRCZWhpbmQgPSBtZXNzYWdlcy5sZW5ndGggLSBkZWZlcnJlZE1lc3NhZ2VzLmxlbmd0aFxuICBpZiAoZGVmZXJyZWRCZWhpbmQgPiAwKSB7XG4gICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgYFt1c2VEZWZlcnJlZFZhbHVlXSBNZXNzYWdlcyBkZWZlcnJlZCBieSAke2RlZmVycmVkQmVoaW5kfSAoJHtkZWZlcnJlZE1lc3NhZ2VzLmxlbmd0aH3ihpIke21lc3NhZ2VzLmxlbmd0aH0pYCxcbiAgICApXG4gIH1cblxuICAvLyBGcm96ZW4gc3RhdGUgZm9yIHRyYW5zY3JpcHQgbW9kZSAtIHN0b3JlcyBsZW5ndGhzIGluc3RlYWQgb2YgY2xvbmluZyBhcnJheXMgZm9yIG1lbW9yeSBlZmZpY2llbmN5XG4gIGNvbnN0IFtmcm96ZW5UcmFuc2NyaXB0U3RhdGUsIHNldEZyb3plblRyYW5zY3JpcHRTdGF0ZV0gPSB1c2VTdGF0ZTx7XG4gICAgbWVzc2FnZXNMZW5ndGg6IG51bWJlclxuICAgIHN0cmVhbWluZ1Rvb2xVc2VzTGVuZ3RoOiBudW1iZXJcbiAgfSB8IG51bGw+KG51bGwpXG4gIC8vIEluaXRpYWxpemUgaW5wdXQgd2l0aCBhbnkgZWFybHkgaW5wdXQgdGhhdCB3YXMgY2FwdHVyZWQgYmVmb3JlIFJFUEwgd2FzIHJlYWR5LlxuICAvLyBVc2luZyBsYXp5IGluaXRpYWxpemF0aW9uIGVuc3VyZXMgY3Vyc29yIG9mZnNldCBpcyBzZXQgY29ycmVjdGx5IGluIFByb21wdElucHV0LlxuICBjb25zdCBbaW5wdXRWYWx1ZSwgc2V0SW5wdXRWYWx1ZVJhd10gPSB1c2VTdGF0ZSgoKSA9PiBjb25zdW1lRWFybHlJbnB1dCgpKVxuICBjb25zdCBpbnB1dFZhbHVlUmVmID0gdXNlUmVmKGlucHV0VmFsdWUpXG4gIGlucHV0VmFsdWVSZWYuY3VycmVudCA9IGlucHV0VmFsdWVcbiAgY29uc3QgaW5zZXJ0VGV4dFJlZiA9IHVzZVJlZjx7XG4gICAgaW5zZXJ0OiAodGV4dDogc3RyaW5nKSA9PiB2b2lkXG4gICAgc2V0SW5wdXRXaXRoQ3Vyc29yOiAodmFsdWU6IHN0cmluZywgY3Vyc29yOiBudW1iZXIpID0+IHZvaWRcbiAgICBjdXJzb3JPZmZzZXQ6IG51bWJlclxuICB9IHwgbnVsbD4obnVsbClcblxuICAvLyBXcmFwIHNldElucHV0VmFsdWUgdG8gY28tbG9jYXRlIHN1cHByZXNzaW9uIHN0YXRlIHVwZGF0ZXMuXG4gIC8vIEJvdGggc2V0U3RhdGUgY2FsbHMgaGFwcGVuIGluIHRoZSBzYW1lIHN5bmNocm9ub3VzIGNvbnRleHQgc28gUmVhY3RcbiAgLy8gYmF0Y2hlcyB0aGVtIGludG8gYSBzaW5nbGUgcmVuZGVyLCBlbGltaW5hdGluZyB0aGUgZXh0cmEgcmVuZGVyIHRoYXRcbiAgLy8gdGhlIHByZXZpb3VzIHVzZUVmZmVjdCDihpIgc2V0U3RhdGUgcGF0dGVybiBjYXVzZWQuXG4gIGNvbnN0IHNldElucHV0VmFsdWUgPSB1c2VDYWxsYmFjayhcbiAgICAodmFsdWU6IHN0cmluZykgPT4ge1xuICAgICAgaWYgKHRyeVN1Z2dlc3RCZ1BSSW50ZXJjZXB0KGlucHV0VmFsdWVSZWYuY3VycmVudCwgdmFsdWUpKSByZXR1cm5cbiAgICAgIC8vIEluIGZ1bGxzY3JlZW4gbW9kZSwgdHlwaW5nIGludG8gYW4gZW1wdHkgcHJvbXB0IHJlLXBpbnMgc2Nyb2xsIHRvXG4gICAgICAvLyBib3R0b20uIE9ubHkgZmlyZXMgb24gZW1wdHnihpJub24tZW1wdHkgc28gc2Nyb2xsaW5nIHVwIHRvIHJlZmVyZW5jZVxuICAgICAgLy8gc29tZXRoaW5nIHdoaWxlIGNvbXBvc2luZyBhIG1lc3NhZ2UgZG9lc24ndCB5YW5rIHRoZSB2aWV3IGJhY2sgb25cbiAgICAgIC8vIGV2ZXJ5IGtleXN0cm9rZS4gUmVzdG9yZXMgdGhlIHByZS1mdWxsc2NyZWVuIG11c2NsZSBtZW1vcnkgb2ZcbiAgICAgIC8vIHR5cGluZyB0byBzbmFwIGJhY2sgdG8gdGhlIGVuZCBvZiB0aGUgY29udmVyc2F0aW9uLlxuICAgICAgLy8gU2tpcHBlZCBpZiB0aGUgdXNlciBzY3JvbGxlZCB3aXRoaW4gdGhlIGxhc3QgM3Mg4oCUIHRoZXkncmUgYWN0aXZlbHlcbiAgICAgIC8vIHJlYWRpbmcsIG5vdCBsb3N0LiBsYXN0VXNlclNjcm9sbFRzUmVmIHN0YXJ0cyBhdCAwIHNvIHRoZSBmaXJzdC1cbiAgICAgIC8vIGV2ZXIga2V5cHJlc3MgKG5vIHNjcm9sbCB5ZXQpIGFsd2F5cyByZXBpbnMuXG4gICAgICBpZiAoXG4gICAgICAgIGlucHV0VmFsdWVSZWYuY3VycmVudCA9PT0gJycgJiZcbiAgICAgICAgdmFsdWUgIT09ICcnICYmXG4gICAgICAgIERhdGUubm93KCkgLSBsYXN0VXNlclNjcm9sbFRzUmVmLmN1cnJlbnQgPj1cbiAgICAgICAgICBSRUNFTlRfU0NST0xMX1JFUElOX1dJTkRPV19NU1xuICAgICAgKSB7XG4gICAgICAgIHJlcGluU2Nyb2xsKClcbiAgICAgIH1cbiAgICAgIC8vIFN5bmMgcmVmIGltbWVkaWF0ZWx5IChsaWtlIHNldE1lc3NhZ2VzKSBzbyBjYWxsZXJzIHRoYXQgcmVhZFxuICAgICAgLy8gaW5wdXRWYWx1ZVJlZiBiZWZvcmUgUmVhY3QgY29tbWl0cyDigJQgZS5nLiB0aGUgYXV0by1yZXN0b3JlIGZpbmFsbHlcbiAgICAgIC8vIGJsb2NrJ3MgYD09PSAnJ2AgZ3VhcmQg4oCUIHNlZSB0aGUgZnJlc2ggdmFsdWUsIG5vdCB0aGUgc3RhbGUgcmVuZGVyLlxuICAgICAgaW5wdXRWYWx1ZVJlZi5jdXJyZW50ID0gdmFsdWVcbiAgICAgIHNldElucHV0VmFsdWVSYXcodmFsdWUpXG4gICAgICBzZXRJc1Byb21wdElucHV0QWN0aXZlKHZhbHVlLnRyaW0oKS5sZW5ndGggPiAwKVxuICAgIH0sXG4gICAgW3NldElzUHJvbXB0SW5wdXRBY3RpdmUsIHJlcGluU2Nyb2xsLCB0cnlTdWdnZXN0QmdQUkludGVyY2VwdF0sXG4gIClcblxuICAvLyBTY2hlZHVsZSBhIHRpbWVvdXQgdG8gc3RvcCBzdXBwcmVzc2luZyBkaWFsb2dzIGFmdGVyIHRoZSB1c2VyIHN0b3BzIHR5cGluZy5cbiAgLy8gT25seSBtYW5hZ2VzIHRoZSB0aW1lb3V0IOKAlCB0aGUgaW1tZWRpYXRlIGFjdGl2YXRpb24gaXMgaGFuZGxlZCBieSBzZXRJbnB1dFZhbHVlIGFib3ZlLlxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmIChpbnB1dFZhbHVlLnRyaW0oKS5sZW5ndGggPT09IDApIHJldHVyblxuICAgIGNvbnN0IHRpbWVyID0gc2V0VGltZW91dChcbiAgICAgIHNldElzUHJvbXB0SW5wdXRBY3RpdmUsXG4gICAgICBQUk9NUFRfU1VQUFJFU1NJT05fTVMsXG4gICAgICBmYWxzZSxcbiAgICApXG4gICAgcmV0dXJuICgpID0+IGNsZWFyVGltZW91dCh0aW1lcilcbiAgfSwgW2lucHV0VmFsdWVdKVxuXG4gIGNvbnN0IFtpbnB1dE1vZGUsIHNldElucHV0TW9kZV0gPSB1c2VTdGF0ZTxQcm9tcHRJbnB1dE1vZGU+KCdwcm9tcHQnKVxuICBjb25zdCBbc3Rhc2hlZFByb21wdCwgc2V0U3Rhc2hlZFByb21wdF0gPSB1c2VTdGF0ZTxcbiAgICB8IHtcbiAgICAgICAgdGV4dDogc3RyaW5nXG4gICAgICAgIGN1cnNvck9mZnNldDogbnVtYmVyXG4gICAgICAgIHBhc3RlZENvbnRlbnRzOiBSZWNvcmQ8bnVtYmVyLCBQYXN0ZWRDb250ZW50PlxuICAgICAgfVxuICAgIHwgdW5kZWZpbmVkXG4gID4oKVxuXG4gIC8vIENhbGxiYWNrIHRvIGZpbHRlciBjb21tYW5kcyBiYXNlZCBvbiBDQ1IncyBhdmFpbGFibGUgc2xhc2ggY29tbWFuZHNcbiAgY29uc3QgaGFuZGxlUmVtb3RlSW5pdCA9IHVzZUNhbGxiYWNrKFxuICAgIChyZW1vdGVTbGFzaENvbW1hbmRzOiBzdHJpbmdbXSkgPT4ge1xuICAgICAgY29uc3QgcmVtb3RlQ29tbWFuZFNldCA9IG5ldyBTZXQocmVtb3RlU2xhc2hDb21tYW5kcylcbiAgICAgIC8vIEtlZXAgY29tbWFuZHMgdGhhdCBDQ1IgbGlzdHMgT1IgdGhhdCBhcmUgaW4gdGhlIGxvY2FsLXNhZmUgc2V0XG4gICAgICBzZXRMb2NhbENvbW1hbmRzKHByZXYgPT5cbiAgICAgICAgcHJldi5maWx0ZXIoXG4gICAgICAgICAgY21kID0+XG4gICAgICAgICAgICByZW1vdGVDb21tYW5kU2V0LmhhcyhjbWQubmFtZSkgfHwgUkVNT1RFX1NBRkVfQ09NTUFORFMuaGFzKGNtZCksXG4gICAgICAgICksXG4gICAgICApXG4gICAgfSxcbiAgICBbc2V0TG9jYWxDb21tYW5kc10sXG4gIClcblxuICBjb25zdCBbaW5Qcm9ncmVzc1Rvb2xVc2VJRHMsIHNldEluUHJvZ3Jlc3NUb29sVXNlSURzXSA9IHVzZVN0YXRlPFNldDxzdHJpbmc+PihcbiAgICBuZXcgU2V0KCksXG4gIClcbiAgY29uc3QgaGFzSW50ZXJydXB0aWJsZVRvb2xJblByb2dyZXNzUmVmID0gdXNlUmVmKGZhbHNlKVxuXG4gIC8vIFJlbW90ZSBzZXNzaW9uIGhvb2sgLSBtYW5hZ2VzIFdlYlNvY2tldCBjb25uZWN0aW9uIGFuZCBtZXNzYWdlIGhhbmRsaW5nIGZvciAtLXJlbW90ZSBtb2RlXG4gIGNvbnN0IHJlbW90ZVNlc3Npb24gPSB1c2VSZW1vdGVTZXNzaW9uKHtcbiAgICBjb25maWc6IHJlbW90ZVNlc3Npb25Db25maWcsXG4gICAgc2V0TWVzc2FnZXMsXG4gICAgc2V0SXNMb2FkaW5nOiBzZXRJc0V4dGVybmFsTG9hZGluZyxcbiAgICBvbkluaXQ6IGhhbmRsZVJlbW90ZUluaXQsXG4gICAgc2V0VG9vbFVzZUNvbmZpcm1RdWV1ZSxcbiAgICB0b29sczogY29tYmluZWRJbml0aWFsVG9vbHMsXG4gICAgc2V0U3RyZWFtaW5nVG9vbFVzZXMsXG4gICAgc2V0U3RyZWFtTW9kZSxcbiAgICBzZXRJblByb2dyZXNzVG9vbFVzZUlEcyxcbiAgfSlcblxuICAvLyBEaXJlY3QgY29ubmVjdCBob29rIC0gbWFuYWdlcyBXZWJTb2NrZXQgdG8gYSBjbGF1ZGUgc2VydmVyIGZvciBgY2xhdWRlIGNvbm5lY3RgIG1vZGVcbiAgY29uc3QgZGlyZWN0Q29ubmVjdCA9IHVzZURpcmVjdENvbm5lY3Qoe1xuICAgIGNvbmZpZzogZGlyZWN0Q29ubmVjdENvbmZpZyxcbiAgICBzZXRNZXNzYWdlcyxcbiAgICBzZXRJc0xvYWRpbmc6IHNldElzRXh0ZXJuYWxMb2FkaW5nLFxuICAgIHNldFRvb2xVc2VDb25maXJtUXVldWUsXG4gICAgdG9vbHM6IGNvbWJpbmVkSW5pdGlhbFRvb2xzLFxuICB9KVxuXG4gIC8vIFNTSCBzZXNzaW9uIGhvb2sgLSBtYW5hZ2VzIHNzaCBjaGlsZCBwcm9jZXNzIGZvciBgY2xhdWRlIHNzaGAgbW9kZS5cbiAgLy8gU2FtZSBjYWxsYmFjayBzaGFwZSBhcyB1c2VEaXJlY3RDb25uZWN0OyBvbmx5IHRoZSB0cmFuc3BvcnQgdW5kZXIgdGhlXG4gIC8vIGhvb2QgZGlmZmVycyAoQ2hpbGRQcm9jZXNzIHN0ZGluL3N0ZG91dCB2cyBXZWJTb2NrZXQpLlxuICBjb25zdCBzc2hSZW1vdGUgPSB1c2VTU0hTZXNzaW9uKHtcbiAgICBzZXNzaW9uOiBzc2hTZXNzaW9uLFxuICAgIHNldE1lc3NhZ2VzLFxuICAgIHNldElzTG9hZGluZzogc2V0SXNFeHRlcm5hbExvYWRpbmcsXG4gICAgc2V0VG9vbFVzZUNvbmZpcm1RdWV1ZSxcbiAgICB0b29sczogY29tYmluZWRJbml0aWFsVG9vbHMsXG4gIH0pXG5cbiAgLy8gVXNlIHdoaWNoZXZlciByZW1vdGUgbW9kZSBpcyBhY3RpdmVcbiAgY29uc3QgYWN0aXZlUmVtb3RlID0gc3NoUmVtb3RlLmlzUmVtb3RlTW9kZVxuICAgID8gc3NoUmVtb3RlXG4gICAgOiBkaXJlY3RDb25uZWN0LmlzUmVtb3RlTW9kZVxuICAgICAgPyBkaXJlY3RDb25uZWN0XG4gICAgICA6IHJlbW90ZVNlc3Npb25cblxuICBjb25zdCBbcGFzdGVkQ29udGVudHMsIHNldFBhc3RlZENvbnRlbnRzXSA9IHVzZVN0YXRlPFxuICAgIFJlY29yZDxudW1iZXIsIFBhc3RlZENvbnRlbnQ+XG4gID4oe30pXG4gIGNvbnN0IFtzdWJtaXRDb3VudCwgc2V0U3VibWl0Q291bnRdID0gdXNlU3RhdGUoMClcbiAgLy8gUmVmIGluc3RlYWQgb2Ygc3RhdGUgdG8gYXZvaWQgdHJpZ2dlcmluZyBSZWFjdCByZS1yZW5kZXJzIG9uIGV2ZXJ5XG4gIC8vIHN0cmVhbWluZyB0ZXh0X2RlbHRhLiBUaGUgc3Bpbm5lciByZWFkcyB0aGlzIHZpYSBpdHMgYW5pbWF0aW9uIHRpbWVyLlxuICBjb25zdCByZXNwb25zZUxlbmd0aFJlZiA9IHVzZVJlZigwKVxuICAvLyBBUEkgcGVyZm9ybWFuY2UgbWV0cmljcyByZWYgZm9yIGFudC1vbmx5IHNwaW5uZXIgZGlzcGxheSAoVFRGVC9PVFBTKS5cbiAgLy8gQWNjdW11bGF0ZXMgbWV0cmljcyBmcm9tIGFsbCBBUEkgcmVxdWVzdHMgaW4gYSB0dXJuIGZvciBQNTAgYWdncmVnYXRpb24uXG4gIGNvbnN0IGFwaU1ldHJpY3NSZWYgPSB1c2VSZWY8XG4gICAgQXJyYXk8e1xuICAgICAgdHRmdE1zOiBudW1iZXJcbiAgICAgIGZpcnN0VG9rZW5UaW1lOiBudW1iZXJcbiAgICAgIGxhc3RUb2tlblRpbWU6IG51bWJlclxuICAgICAgcmVzcG9uc2VMZW5ndGhCYXNlbGluZTogbnVtYmVyXG4gICAgICAvLyBUcmFja3MgcmVzcG9uc2VMZW5ndGhSZWYgYXQgdGhlIHRpbWUgb2YgdGhlIGxhc3QgY29udGVudCBhZGRpdGlvbi5cbiAgICAgIC8vIFVwZGF0ZWQgYnkgYm90aCBzdHJlYW1pbmcgZGVsdGFzIGFuZCBzdWJhZ2VudCBtZXNzYWdlIGNvbnRlbnQuXG4gICAgICAvLyBsYXN0VG9rZW5UaW1lIGlzIGFsc28gdXBkYXRlZCBhdCB0aGUgc2FtZSB0aW1lLCBzbyB0aGUgT1RQU1xuICAgICAgLy8gZGVub21pbmF0b3IgY29ycmVjdGx5IGluY2x1ZGVzIHN1YmFnZW50IHByb2Nlc3NpbmcgdGltZS5cbiAgICAgIGVuZFJlc3BvbnNlTGVuZ3RoOiBudW1iZXJcbiAgICB9PlxuICA+KFtdKVxuICBjb25zdCBzZXRSZXNwb25zZUxlbmd0aCA9IHVzZUNhbGxiYWNrKChmOiAocHJldjogbnVtYmVyKSA9PiBudW1iZXIpID0+IHtcbiAgICBjb25zdCBwcmV2ID0gcmVzcG9uc2VMZW5ndGhSZWYuY3VycmVudFxuICAgIHJlc3BvbnNlTGVuZ3RoUmVmLmN1cnJlbnQgPSBmKHByZXYpXG4gICAgLy8gV2hlbiBjb250ZW50IGlzIGFkZGVkIChub3QgYSBjb21wYWN0aW9uIHJlc2V0KSwgdXBkYXRlIHRoZSBsYXRlc3RcbiAgICAvLyBtZXRyaWNzIGVudHJ5IHNvIE9UUFMgcmVmbGVjdHMgYWxsIGNvbnRlbnQgZ2VuZXJhdGlvbiBhY3Rpdml0eS5cbiAgICAvLyBVcGRhdGluZyBsYXN0VG9rZW5UaW1lIGhlcmUgZW5zdXJlcyB0aGUgZGVub21pbmF0b3IgaW5jbHVkZXMgYm90aFxuICAgIC8vIHN0cmVhbWluZyB0aW1lIEFORCBzdWJhZ2VudCBleGVjdXRpb24gdGltZSwgcHJldmVudGluZyBpbmZsYXRpb24uXG4gICAgaWYgKHJlc3BvbnNlTGVuZ3RoUmVmLmN1cnJlbnQgPiBwcmV2KSB7XG4gICAgICBjb25zdCBlbnRyaWVzID0gYXBpTWV0cmljc1JlZi5jdXJyZW50XG4gICAgICBpZiAoZW50cmllcy5sZW5ndGggPiAwKSB7XG4gICAgICAgIGNvbnN0IGxhc3RFbnRyeSA9IGVudHJpZXMuYXQoLTEpIVxuICAgICAgICBsYXN0RW50cnkubGFzdFRva2VuVGltZSA9IERhdGUubm93KClcbiAgICAgICAgbGFzdEVudHJ5LmVuZFJlc3BvbnNlTGVuZ3RoID0gcmVzcG9uc2VMZW5ndGhSZWYuY3VycmVudFxuICAgICAgfVxuICAgIH1cbiAgfSwgW10pXG5cbiAgLy8gU3RyZWFtaW5nIHRleHQgZGlzcGxheTogc2V0IHN0YXRlIGRpcmVjdGx5IHBlciBkZWx0YSAoSW5rJ3MgMTZtcyByZW5kZXJcbiAgLy8gdGhyb3R0bGUgYmF0Y2hlcyByYXBpZCB1cGRhdGVzKS4gQ2xlYXJlZCBvbiBtZXNzYWdlIGFycml2YWwgKG1lc3NhZ2VzLnRzKVxuICAvLyBzbyBkaXNwbGF5ZWRNZXNzYWdlcyBzd2l0Y2hlcyBmcm9tIGRlZmVycmVkTWVzc2FnZXMgdG8gbWVzc2FnZXMgYXRvbWljYWxseS5cbiAgY29uc3QgW3N0cmVhbWluZ1RleHQsIHNldFN0cmVhbWluZ1RleHRdID0gdXNlU3RhdGU8c3RyaW5nIHwgbnVsbD4obnVsbClcbiAgY29uc3QgcmVkdWNlZE1vdGlvbiA9XG4gICAgdXNlQXBwU3RhdGUocyA9PiBzLnNldHRpbmdzLnByZWZlcnNSZWR1Y2VkTW90aW9uKSA/PyBmYWxzZVxuICBjb25zdCBzaG93U3RyZWFtaW5nVGV4dCA9ICFyZWR1Y2VkTW90aW9uICYmICFoYXNDdXJzb3JVcFZpZXdwb3J0WWFua0J1ZygpXG4gIGNvbnN0IG9uU3RyZWFtaW5nVGV4dCA9IHVzZUNhbGxiYWNrKFxuICAgIChmOiAoY3VycmVudDogc3RyaW5nIHwgbnVsbCkgPT4gc3RyaW5nIHwgbnVsbCkgPT4ge1xuICAgICAgaWYgKCFzaG93U3RyZWFtaW5nVGV4dCkgcmV0dXJuXG4gICAgICBzZXRTdHJlYW1pbmdUZXh0KGYpXG4gICAgfSxcbiAgICBbc2hvd1N0cmVhbWluZ1RleHRdLFxuICApXG5cbiAgLy8gSGlkZSB0aGUgaW4tcHJvZ3Jlc3Mgc291cmNlIGxpbmUgc28gdGV4dCBzdHJlYW1zIGxpbmUtYnktbGluZSwgbm90XG4gIC8vIGNoYXItYnktY2hhci4gbGFzdEluZGV4T2YgcmV0dXJucyAtMSB3aGVuIG5vIG5ld2xpbmUsIGdpdmluZyAnJyDihpIgbnVsbC5cbiAgLy8gR3VhcmQgb24gc2hvd1N0cmVhbWluZ1RleHQgc28gdG9nZ2xpbmcgcmVkdWNlZE1vdGlvbiBtaWQtc3RyZWFtXG4gIC8vIGltbWVkaWF0ZWx5IGhpZGVzIHRoZSBzdHJlYW1pbmcgcHJldmlldy5cbiAgY29uc3QgdmlzaWJsZVN0cmVhbWluZ1RleHQgPVxuICAgIHN0cmVhbWluZ1RleHQgJiYgc2hvd1N0cmVhbWluZ1RleHRcbiAgICAgID8gc3RyZWFtaW5nVGV4dC5zdWJzdHJpbmcoMCwgc3RyZWFtaW5nVGV4dC5sYXN0SW5kZXhPZignXFxuJykgKyAxKSB8fCBudWxsXG4gICAgICA6IG51bGxcblxuICBjb25zdCBbbGFzdFF1ZXJ5Q29tcGxldGlvblRpbWUsIHNldExhc3RRdWVyeUNvbXBsZXRpb25UaW1lXSA9IHVzZVN0YXRlKDApXG4gIGNvbnN0IFtzcGlubmVyTWVzc2FnZSwgc2V0U3Bpbm5lck1lc3NhZ2VdID0gdXNlU3RhdGU8c3RyaW5nIHwgbnVsbD4obnVsbClcbiAgY29uc3QgW3NwaW5uZXJDb2xvciwgc2V0U3Bpbm5lckNvbG9yXSA9IHVzZVN0YXRlPGtleW9mIFRoZW1lIHwgbnVsbD4obnVsbClcbiAgY29uc3QgW3NwaW5uZXJTaGltbWVyQ29sb3IsIHNldFNwaW5uZXJTaGltbWVyQ29sb3JdID0gdXNlU3RhdGU8XG4gICAga2V5b2YgVGhlbWUgfCBudWxsXG4gID4obnVsbClcbiAgY29uc3QgW2lzTWVzc2FnZVNlbGVjdG9yVmlzaWJsZSwgc2V0SXNNZXNzYWdlU2VsZWN0b3JWaXNpYmxlXSA9XG4gICAgdXNlU3RhdGUoZmFsc2UpXG4gIGNvbnN0IFttZXNzYWdlU2VsZWN0b3JQcmVzZWxlY3QsIHNldE1lc3NhZ2VTZWxlY3RvclByZXNlbGVjdF0gPSB1c2VTdGF0ZTxcbiAgICBVc2VyTWVzc2FnZSB8IHVuZGVmaW5lZFxuICA+KHVuZGVmaW5lZClcbiAgY29uc3QgW3Nob3dDb3N0RGlhbG9nLCBzZXRTaG93Q29zdERpYWxvZ10gPSB1c2VTdGF0ZShmYWxzZSlcbiAgY29uc3QgW2NvbnZlcnNhdGlvbklkLCBzZXRDb252ZXJzYXRpb25JZF0gPSB1c2VTdGF0ZShyYW5kb21VVUlEKCkpXG5cbiAgLy8gSWRsZS1yZXR1cm4gZGlhbG9nOiBzaG93biB3aGVuIHVzZXIgc3VibWl0cyBhZnRlciBhIGxvbmcgaWRsZSBnYXBcbiAgY29uc3QgW2lkbGVSZXR1cm5QZW5kaW5nLCBzZXRJZGxlUmV0dXJuUGVuZGluZ10gPSB1c2VTdGF0ZTx7XG4gICAgaW5wdXQ6IHN0cmluZ1xuICAgIGlkbGVNaW51dGVzOiBudW1iZXJcbiAgfSB8IG51bGw+KG51bGwpXG4gIGNvbnN0IHNraXBJZGxlQ2hlY2tSZWYgPSB1c2VSZWYoZmFsc2UpXG4gIGNvbnN0IGxhc3RRdWVyeUNvbXBsZXRpb25UaW1lUmVmID0gdXNlUmVmKGxhc3RRdWVyeUNvbXBsZXRpb25UaW1lKVxuICBsYXN0UXVlcnlDb21wbGV0aW9uVGltZVJlZi5jdXJyZW50ID0gbGFzdFF1ZXJ5Q29tcGxldGlvblRpbWVcblxuICAvLyBBZ2dyZWdhdGUgdG9vbCByZXN1bHQgYnVkZ2V0OiBwZXItY29udmVyc2F0aW9uIGRlY2lzaW9uIHRyYWNraW5nLlxuICAvLyBXaGVuIHRoZSBHcm93dGhCb29rIGZsYWcgaXMgb24sIHF1ZXJ5LnRzIGVuZm9yY2VzIHRoZSBidWRnZXQ7IHdoZW5cbiAgLy8gb2ZmICh1bmRlZmluZWQpLCBlbmZvcmNlbWVudCBpcyBza2lwcGVkIGVudGlyZWx5LiBTdGFsZSBlbnRyaWVzIGFmdGVyXG4gIC8vIC9jbGVhciwgcmV3aW5kLCBvciBjb21wYWN0IGFyZSBoYXJtbGVzcyAodG9vbF91c2VfaWRzIGFyZSBVVUlEcywgc3RhbGVcbiAgLy8ga2V5cyBhcmUgbmV2ZXIgbG9va2VkIHVwKS4gTWVtb3J5IGlzIGJvdW5kZWQgYnkgdG90YWwgcmVwbGFjZW1lbnQgY291bnRcbiAgLy8gw5cgfjJLQiBwcmV2aWV3IG92ZXIgdGhlIFJFUEwgbGlmZXRpbWUg4oCUIG5lZ2xpZ2libGUuXG4gIC8vXG4gIC8vIExhenkgaW5pdCB2aWEgdXNlU3RhdGUgaW5pdGlhbGl6ZXIg4oCUIHVzZVJlZihleHByKSBldmFsdWF0ZXMgZXhwciBvbiBldmVyeVxuICAvLyByZW5kZXIgKFJlYWN0IGlnbm9yZXMgaXQgYWZ0ZXIgZmlyc3QsIGJ1dCB0aGUgY29tcHV0YXRpb24gc3RpbGwgcnVucykuXG4gIC8vIEZvciBsYXJnZSByZXN1bWVkIHNlc3Npb25zLCByZWNvbnN0cnVjdGlvbiBkb2VzIE8obWVzc2FnZXMgw5cgYmxvY2tzKVxuICAvLyB3b3JrOyB3ZSBvbmx5IHdhbnQgdGhhdCBvbmNlLlxuICBjb25zdCBbY29udGVudFJlcGxhY2VtZW50U3RhdGVSZWZdID0gdXNlU3RhdGUoKCkgPT4gKHtcbiAgICBjdXJyZW50OiBwcm92aXNpb25Db250ZW50UmVwbGFjZW1lbnRTdGF0ZShcbiAgICAgIGluaXRpYWxNZXNzYWdlcyxcbiAgICAgIGluaXRpYWxDb250ZW50UmVwbGFjZW1lbnRzLFxuICAgICksXG4gIH0pKVxuXG4gIGNvbnN0IFtoYXZlU2hvd25Db3N0RGlhbG9nLCBzZXRIYXZlU2hvd25Db3N0RGlhbG9nXSA9IHVzZVN0YXRlKFxuICAgIGdldEdsb2JhbENvbmZpZygpLmhhc0Fja25vd2xlZGdlZENvc3RUaHJlc2hvbGQsXG4gIClcbiAgY29uc3QgW3ZpbU1vZGUsIHNldFZpbU1vZGVdID0gdXNlU3RhdGU8VmltTW9kZT4oJ0lOU0VSVCcpXG4gIGNvbnN0IFtzaG93QmFzaGVzRGlhbG9nLCBzZXRTaG93QmFzaGVzRGlhbG9nXSA9IHVzZVN0YXRlPHN0cmluZyB8IGJvb2xlYW4+KFxuICAgIGZhbHNlLFxuICApXG4gIGNvbnN0IFtpc1NlYXJjaGluZ0hpc3RvcnksIHNldElzU2VhcmNoaW5nSGlzdG9yeV0gPSB1c2VTdGF0ZShmYWxzZSlcbiAgY29uc3QgW2lzSGVscE9wZW4sIHNldElzSGVscE9wZW5dID0gdXNlU3RhdGUoZmFsc2UpXG5cbiAgLy8gc2hvd0Jhc2hlc0RpYWxvZyBpcyBSRVBMLWxldmVsIHNvIGl0IHN1cnZpdmVzIFByb21wdElucHV0IHVubW91bnRpbmcuXG4gIC8vIFdoZW4gdWx0cmFwbGFuIGFwcHJvdmFsIGZpcmVzIHdoaWxlIHRoZSBwaWxsIGRpYWxvZyBpcyBvcGVuLCBQcm9tcHRJbnB1dFxuICAvLyB1bm1vdW50cyAoZm9jdXNlZElucHV0RGlhbG9nIOKGkiAndWx0cmFwbGFuLWNob2ljZScpIGJ1dCB0aGlzIHN0YXlzIHRydWU7XG4gIC8vIGFmdGVyIGFjY2VwdGluZywgUHJvbXB0SW5wdXQgcmVtb3VudHMgaW50byBhbiBlbXB0eSBcIk5vIHRhc2tzXCIgZGlhbG9nXG4gIC8vICh0aGUgY29tcGxldGVkIHVsdHJhcGxhbiB0YXNrIGhhcyBiZWVuIGZpbHRlcmVkIG91dCkuIENsb3NlIGl0IGhlcmUuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKHVsdHJhcGxhblBlbmRpbmdDaG9pY2UgJiYgc2hvd0Jhc2hlc0RpYWxvZykge1xuICAgICAgc2V0U2hvd0Jhc2hlc0RpYWxvZyhmYWxzZSlcbiAgICB9XG4gIH0sIFt1bHRyYXBsYW5QZW5kaW5nQ2hvaWNlLCBzaG93QmFzaGVzRGlhbG9nXSlcblxuICBjb25zdCBpc1Rlcm1pbmFsRm9jdXNlZCA9IHVzZVRlcm1pbmFsRm9jdXMoKVxuICBjb25zdCB0ZXJtaW5hbEZvY3VzUmVmID0gdXNlUmVmKGlzVGVybWluYWxGb2N1c2VkKVxuICB0ZXJtaW5hbEZvY3VzUmVmLmN1cnJlbnQgPSBpc1Rlcm1pbmFsRm9jdXNlZFxuXG4gIGNvbnN0IFt0aGVtZV0gPSB1c2VUaGVtZSgpXG5cbiAgLy8gcmVzZXRMb2FkaW5nU3RhdGUgcnVucyB0d2ljZSBwZXIgdHVybiAob25RdWVyeUltcGwgdGFpbCArIG9uUXVlcnkgZmluYWxseSkuXG4gIC8vIFdpdGhvdXQgdGhpcyBndWFyZCwgYm90aCBjYWxscyBwaWNrIGEgdGlwIOKGkiB0d28gcmVjb3JkU2hvd25UaXAg4oaSIHR3b1xuICAvLyBzYXZlR2xvYmFsQ29uZmlnIHdyaXRlcyBiYWNrLXRvLWJhY2suIFJlc2V0IGF0IHN1Ym1pdCBpbiBvblN1Ym1pdC5cbiAgY29uc3QgdGlwUGlja2VkVGhpc1R1cm5SZWYgPSBSZWFjdC51c2VSZWYoZmFsc2UpXG4gIGNvbnN0IHBpY2tOZXdTcGlubmVyVGlwID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIGlmICh0aXBQaWNrZWRUaGlzVHVyblJlZi5jdXJyZW50KSByZXR1cm5cbiAgICB0aXBQaWNrZWRUaGlzVHVyblJlZi5jdXJyZW50ID0gdHJ1ZVxuICAgIGNvbnN0IG5ld01lc3NhZ2VzID0gbWVzc2FnZXNSZWYuY3VycmVudC5zbGljZShiYXNoVG9vbHNQcm9jZXNzZWRJZHguY3VycmVudClcbiAgICBmb3IgKGNvbnN0IHRvb2wgb2YgZXh0cmFjdEJhc2hUb29sc0Zyb21NZXNzYWdlcyhuZXdNZXNzYWdlcykpIHtcbiAgICAgIGJhc2hUb29scy5jdXJyZW50LmFkZCh0b29sKVxuICAgIH1cbiAgICBiYXNoVG9vbHNQcm9jZXNzZWRJZHguY3VycmVudCA9IG1lc3NhZ2VzUmVmLmN1cnJlbnQubGVuZ3RoXG4gICAgdm9pZCBnZXRUaXBUb1Nob3dPblNwaW5uZXIoe1xuICAgICAgdGhlbWUsXG4gICAgICByZWFkRmlsZVN0YXRlOiByZWFkRmlsZVN0YXRlLmN1cnJlbnQsXG4gICAgICBiYXNoVG9vbHM6IGJhc2hUb29scy5jdXJyZW50LFxuICAgIH0pLnRoZW4oYXN5bmMgdGlwID0+IHtcbiAgICAgIGlmICh0aXApIHtcbiAgICAgICAgY29uc3QgY29udGVudCA9IGF3YWl0IHRpcC5jb250ZW50KHsgdGhlbWUgfSlcbiAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiAoe1xuICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgc3Bpbm5lclRpcDogY29udGVudCxcbiAgICAgICAgfSkpXG4gICAgICAgIHJlY29yZFNob3duVGlwKHRpcClcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgICAgICAgIGlmIChwcmV2LnNwaW5uZXJUaXAgPT09IHVuZGVmaW5lZCkgcmV0dXJuIHByZXZcbiAgICAgICAgICByZXR1cm4geyAuLi5wcmV2LCBzcGlubmVyVGlwOiB1bmRlZmluZWQgfVxuICAgICAgICB9KVxuICAgICAgfVxuICAgIH0pXG4gIH0sIFtzZXRBcHBTdGF0ZSwgdGhlbWVdKVxuXG4gIC8vIFJlc2V0cyBVSSBsb2FkaW5nIHN0YXRlLiBEb2VzIE5PVCBjYWxsIG9uVHVybkNvbXBsZXRlIC0gdGhhdCBzaG91bGQgYmVcbiAgLy8gY2FsbGVkIGV4cGxpY2l0bHkgb25seSB3aGVuIGEgcXVlcnkgdHVybiBhY3R1YWxseSBjb21wbGV0ZXMuXG4gIGNvbnN0IHJlc2V0TG9hZGluZ1N0YXRlID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIC8vIGlzTG9hZGluZyBpcyBub3cgZGVyaXZlZCBmcm9tIHF1ZXJ5R3VhcmQg4oCUIG5vIHNldHRlciBjYWxsIG5lZWRlZC5cbiAgICAvLyBxdWVyeUd1YXJkLmVuZCgpIChvblF1ZXJ5IGZpbmFsbHkpIG9yIGNhbmNlbFJlc2VydmF0aW9uKCkgKGV4ZWN1dGVVc2VySW5wdXRcbiAgICAvLyBmaW5hbGx5KSBoYXZlIGFscmVhZHkgdHJhbnNpdGlvbmVkIHRoZSBndWFyZCB0byBpZGxlIGJ5IHRoZSB0aW1lIHRoaXMgcnVucy5cbiAgICAvLyBFeHRlcm5hbCBsb2FkaW5nIChyZW1vdGUvYmFja2dyb3VuZGluZykgaXMgcmVzZXQgc2VwYXJhdGVseSBieSB0aG9zZSBob29rcy5cbiAgICBzZXRJc0V4dGVybmFsTG9hZGluZyhmYWxzZSlcbiAgICBzZXRVc2VySW5wdXRPblByb2Nlc3NpbmcodW5kZWZpbmVkKVxuICAgIHJlc3BvbnNlTGVuZ3RoUmVmLmN1cnJlbnQgPSAwXG4gICAgYXBpTWV0cmljc1JlZi5jdXJyZW50ID0gW11cbiAgICBzZXRTdHJlYW1pbmdUZXh0KG51bGwpXG4gICAgc2V0U3RyZWFtaW5nVG9vbFVzZXMoW10pXG4gICAgc2V0U3Bpbm5lck1lc3NhZ2UobnVsbClcbiAgICBzZXRTcGlubmVyQ29sb3IobnVsbClcbiAgICBzZXRTcGlubmVyU2hpbW1lckNvbG9yKG51bGwpXG4gICAgcGlja05ld1NwaW5uZXJUaXAoKVxuICAgIGVuZEludGVyYWN0aW9uU3BhbigpXG4gICAgLy8gU3BlY3VsYXRpdmUgYmFzaCBjbGFzc2lmaWVyIGNoZWNrcyBhcmUgb25seSB2YWxpZCBmb3IgdGhlIGN1cnJlbnRcbiAgICAvLyB0dXJuJ3MgY29tbWFuZHMg4oCUIGNsZWFyIGFmdGVyIGVhY2ggdHVybiB0byBhdm9pZCBhY2N1bXVsYXRpbmdcbiAgICAvLyBQcm9taXNlIGNoYWlucyBmb3IgdW5jb25zdW1lZCBjaGVja3MgKGRlbmllZC9hYm9ydGVkIHBhdGhzKS5cbiAgICBjbGVhclNwZWN1bGF0aXZlQ2hlY2tzKClcbiAgfSwgW3BpY2tOZXdTcGlubmVyVGlwXSlcblxuICAvLyBTZXNzaW9uIGJhY2tncm91bmRpbmcg4oCUIGhvb2sgaXMgYmVsb3csIGFmdGVyIGdldFRvb2xVc2VDb250ZXh0XG5cbiAgY29uc3QgaGFzUnVubmluZ1RlYW1tYXRlcyA9IHVzZU1lbW8oXG4gICAgKCkgPT4gZ2V0QWxsSW5Qcm9jZXNzVGVhbW1hdGVUYXNrcyh0YXNrcykuc29tZSh0ID0+IHQuc3RhdHVzID09PSAncnVubmluZycpLFxuICAgIFt0YXNrc10sXG4gIClcblxuICAvLyBTaG93IGRlZmVycmVkIHR1cm4gZHVyYXRpb24gbWVzc2FnZSBvbmNlIGFsbCBzd2FybSB0ZWFtbWF0ZXMgZmluaXNoXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKCFoYXNSdW5uaW5nVGVhbW1hdGVzICYmIHN3YXJtU3RhcnRUaW1lUmVmLmN1cnJlbnQgIT09IG51bGwpIHtcbiAgICAgIGNvbnN0IHRvdGFsTXMgPSBEYXRlLm5vdygpIC0gc3dhcm1TdGFydFRpbWVSZWYuY3VycmVudFxuICAgICAgY29uc3QgZGVmZXJyZWRCdWRnZXQgPSBzd2FybUJ1ZGdldEluZm9SZWYuY3VycmVudFxuICAgICAgc3dhcm1TdGFydFRpbWVSZWYuY3VycmVudCA9IG51bGxcbiAgICAgIHN3YXJtQnVkZ2V0SW5mb1JlZi5jdXJyZW50ID0gdW5kZWZpbmVkXG4gICAgICBzZXRNZXNzYWdlcyhwcmV2ID0+IFtcbiAgICAgICAgLi4ucHJldixcbiAgICAgICAgY3JlYXRlVHVybkR1cmF0aW9uTWVzc2FnZShcbiAgICAgICAgICB0b3RhbE1zLFxuICAgICAgICAgIGRlZmVycmVkQnVkZ2V0LFxuICAgICAgICAgIC8vIENvdW50IG9ubHkgd2hhdCByZWNvcmRUcmFuc2NyaXB0IHdpbGwgcGVyc2lzdCDigJQgZXBoZW1lcmFsXG4gICAgICAgICAgLy8gcHJvZ3Jlc3MgdGlja3MgYW5kIG5vbi1hbnQgYXR0YWNobWVudHMgYXJlIGZpbHRlcmVkIGJ5XG4gICAgICAgICAgLy8gaXNMb2dnYWJsZU1lc3NhZ2UgYW5kIG5ldmVyIHJlYWNoIGRpc2suIFVzaW5nIHJhdyBwcmV2Lmxlbmd0aFxuICAgICAgICAgIC8vIHdvdWxkIG1ha2UgY2hlY2tSZXN1bWVDb25zaXN0ZW5jeSByZXBvcnQgZmFsc2UgZGVsdGE8MCBmb3JcbiAgICAgICAgICAvLyBldmVyeSB0dXJuIHRoYXQgcmFuIGEgcHJvZ3Jlc3MtZW1pdHRpbmcgdG9vbC5cbiAgICAgICAgICBjb3VudChwcmV2LCBpc0xvZ2dhYmxlTWVzc2FnZSksXG4gICAgICAgICksXG4gICAgICBdKVxuICAgIH1cbiAgfSwgW2hhc1J1bm5pbmdUZWFtbWF0ZXMsIHNldE1lc3NhZ2VzXSlcblxuICAvLyBTaG93IGF1dG8gcGVybWlzc2lvbnMgd2FybmluZyB3aGVuIGVudGVyaW5nIGF1dG8gbW9kZVxuICAvLyAoZWl0aGVyIHZpYSBTaGlmdCtUYWIgdG9nZ2xlIG9yIG9uIHN0YXJ0dXApLiBEZWJvdW5jZWQgdG8gYXZvaWRcbiAgLy8gZmxhc2hpbmcgd2hlbiB0aGUgdXNlciBpcyBjeWNsaW5nIHRocm91Z2ggbW9kZXMgcXVpY2tseS5cbiAgLy8gT25seSBzaG93biAzIHRpbWVzIHRvdGFsIGFjcm9zcyBzZXNzaW9ucy5cbiAgY29uc3Qgc2FmZVlvbG9NZXNzYWdlU2hvd25SZWYgPSB1c2VSZWYoZmFsc2UpXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKGZlYXR1cmUoJ1RSQU5TQ1JJUFRfQ0xBU1NJRklFUicpKSB7XG4gICAgICBpZiAodG9vbFBlcm1pc3Npb25Db250ZXh0Lm1vZGUgIT09ICdhdXRvJykge1xuICAgICAgICBzYWZlWW9sb01lc3NhZ2VTaG93blJlZi5jdXJyZW50ID0gZmFsc2VcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG4gICAgICBpZiAoc2FmZVlvbG9NZXNzYWdlU2hvd25SZWYuY3VycmVudCkgcmV0dXJuXG4gICAgICBjb25zdCBjb25maWcgPSBnZXRHbG9iYWxDb25maWcoKVxuICAgICAgY29uc3QgY291bnQgPSBjb25maWcuYXV0b1Blcm1pc3Npb25zTm90aWZpY2F0aW9uQ291bnQgPz8gMFxuICAgICAgaWYgKGNvdW50ID49IDMpIHJldHVyblxuICAgICAgY29uc3QgdGltZXIgPSBzZXRUaW1lb3V0KFxuICAgICAgICAocmVmLCBzZXRNZXNzYWdlcykgPT4ge1xuICAgICAgICAgIHJlZi5jdXJyZW50ID0gdHJ1ZVxuICAgICAgICAgIHNhdmVHbG9iYWxDb25maWcocHJldiA9PiB7XG4gICAgICAgICAgICBjb25zdCBwcmV2Q291bnQgPSBwcmV2LmF1dG9QZXJtaXNzaW9uc05vdGlmaWNhdGlvbkNvdW50ID8/IDBcbiAgICAgICAgICAgIGlmIChwcmV2Q291bnQgPj0gMykgcmV0dXJuIHByZXZcbiAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgIGF1dG9QZXJtaXNzaW9uc05vdGlmaWNhdGlvbkNvdW50OiBwcmV2Q291bnQgKyAxLFxuICAgICAgICAgICAgfVxuICAgICAgICAgIH0pXG4gICAgICAgICAgc2V0TWVzc2FnZXMocHJldiA9PiBbXG4gICAgICAgICAgICAuLi5wcmV2LFxuICAgICAgICAgICAgY3JlYXRlU3lzdGVtTWVzc2FnZShBVVRPX01PREVfREVTQ1JJUFRJT04sICd3YXJuaW5nJyksXG4gICAgICAgICAgXSlcbiAgICAgICAgfSxcbiAgICAgICAgODAwLFxuICAgICAgICBzYWZlWW9sb01lc3NhZ2VTaG93blJlZixcbiAgICAgICAgc2V0TWVzc2FnZXMsXG4gICAgICApXG4gICAgICByZXR1cm4gKCkgPT4gY2xlYXJUaW1lb3V0KHRpbWVyKVxuICAgIH1cbiAgfSwgW3Rvb2xQZXJtaXNzaW9uQ29udGV4dC5tb2RlLCBzZXRNZXNzYWdlc10pXG5cbiAgLy8gSWYgd29ya3RyZWUgY3JlYXRpb24gd2FzIHNsb3cgYW5kIHNwYXJzZS1jaGVja291dCBpc24ndCBjb25maWd1cmVkLFxuICAvLyBudWRnZSB0aGUgdXNlciB0b3dhcmQgc2V0dGluZ3Mud29ya3RyZWUuc3BhcnNlUGF0aHMuXG4gIGNvbnN0IHdvcmt0cmVlVGlwU2hvd25SZWYgPSB1c2VSZWYoZmFsc2UpXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKHdvcmt0cmVlVGlwU2hvd25SZWYuY3VycmVudCkgcmV0dXJuXG4gICAgY29uc3Qgd3QgPSBnZXRDdXJyZW50V29ya3RyZWVTZXNzaW9uKClcbiAgICBpZiAoIXd0Py5jcmVhdGlvbkR1cmF0aW9uTXMgfHwgd3QudXNlZFNwYXJzZVBhdGhzKSByZXR1cm5cbiAgICBpZiAod3QuY3JlYXRpb25EdXJhdGlvbk1zIDwgMTVfMDAwKSByZXR1cm5cbiAgICB3b3JrdHJlZVRpcFNob3duUmVmLmN1cnJlbnQgPSB0cnVlXG4gICAgY29uc3Qgc2VjcyA9IE1hdGgucm91bmQod3QuY3JlYXRpb25EdXJhdGlvbk1zIC8gMTAwMClcbiAgICBzZXRNZXNzYWdlcyhwcmV2ID0+IFtcbiAgICAgIC4uLnByZXYsXG4gICAgICBjcmVhdGVTeXN0ZW1NZXNzYWdlKFxuICAgICAgICBgV29ya3RyZWUgY3JlYXRpb24gdG9vayAke3NlY3N9cy4gRm9yIGxhcmdlIHJlcG9zLCBzZXQgXFxgd29ya3RyZWUuc3BhcnNlUGF0aHNcXGAgaW4gLmNsYXVkZS9zZXR0aW5ncy5qc29uIHRvIGNoZWNrIG91dCBvbmx5IHRoZSBkaXJlY3RvcmllcyB5b3UgbmVlZCDigJQgZS5nLiBcXGB7XCJ3b3JrdHJlZVwiOiB7XCJzcGFyc2VQYXRoc1wiOiBbXCJzcmNcIiwgXCJwYWNrYWdlcy9mb29cIl19fVxcYC5gLFxuICAgICAgICAnaW5mbycsXG4gICAgICApLFxuICAgIF0pXG4gIH0sIFtzZXRNZXNzYWdlc10pXG5cbiAgLy8gSGlkZSBzcGlubmVyIHdoZW4gdGhlIG9ubHkgaW4tcHJvZ3Jlc3MgdG9vbCBpcyBTbGVlcFxuICBjb25zdCBvbmx5U2xlZXBUb29sQWN0aXZlID0gdXNlTWVtbygoKSA9PiB7XG4gICAgY29uc3QgbGFzdEFzc2lzdGFudCA9IG1lc3NhZ2VzLmZpbmRMYXN0KG0gPT4gbS50eXBlID09PSAnYXNzaXN0YW50JylcbiAgICBpZiAobGFzdEFzc2lzdGFudD8udHlwZSAhPT0gJ2Fzc2lzdGFudCcpIHJldHVybiBmYWxzZVxuICAgIGNvbnN0IGluUHJvZ3Jlc3NUb29sVXNlcyA9IGxhc3RBc3Npc3RhbnQubWVzc2FnZS5jb250ZW50LmZpbHRlcihcbiAgICAgIGIgPT4gYi50eXBlID09PSAndG9vbF91c2UnICYmIGluUHJvZ3Jlc3NUb29sVXNlSURzLmhhcyhiLmlkKSxcbiAgICApXG4gICAgcmV0dXJuIChcbiAgICAgIGluUHJvZ3Jlc3NUb29sVXNlcy5sZW5ndGggPiAwICYmXG4gICAgICBpblByb2dyZXNzVG9vbFVzZXMuZXZlcnkoXG4gICAgICAgIGIgPT4gYi50eXBlID09PSAndG9vbF91c2UnICYmIGIubmFtZSA9PT0gU0xFRVBfVE9PTF9OQU1FLFxuICAgICAgKVxuICAgIClcbiAgfSwgW21lc3NhZ2VzLCBpblByb2dyZXNzVG9vbFVzZUlEc10pXG5cbiAgY29uc3Qge1xuICAgIG9uQmVmb3JlUXVlcnk6IG1yT25CZWZvcmVRdWVyeSxcbiAgICBvblR1cm5Db21wbGV0ZTogbXJPblR1cm5Db21wbGV0ZSxcbiAgICByZW5kZXI6IG1yUmVuZGVyLFxuICB9ID0gdXNlTW9yZVJpZ2h0KHtcbiAgICBlbmFibGVkOiBtb3JlUmlnaHRFbmFibGVkLFxuICAgIHNldE1lc3NhZ2VzLFxuICAgIGlucHV0VmFsdWUsXG4gICAgc2V0SW5wdXRWYWx1ZSxcbiAgICBzZXRUb29sSlNYLFxuICB9KVxuXG4gIGNvbnN0IHNob3dTcGlubmVyID1cbiAgICAoIXRvb2xKU1ggfHwgdG9vbEpTWC5zaG93U3Bpbm5lciA9PT0gdHJ1ZSkgJiZcbiAgICB0b29sVXNlQ29uZmlybVF1ZXVlLmxlbmd0aCA9PT0gMCAmJlxuICAgIHByb21wdFF1ZXVlLmxlbmd0aCA9PT0gMCAmJlxuICAgIC8vIFNob3cgc3Bpbm5lciBkdXJpbmcgaW5wdXQgcHJvY2Vzc2luZywgQVBJIGNhbGwsIHdoaWxlIHRlYW1tYXRlcyBhcmUgcnVubmluZyxcbiAgICAvLyBvciB3aGlsZSBwZW5kaW5nIHRhc2sgbm90aWZpY2F0aW9ucyBhcmUgcXVldWVkIChwcmV2ZW50cyBzcGlubmVyIGJvdW5jZSBiZXR3ZWVuIGNvbnNlY3V0aXZlIG5vdGlmaWNhdGlvbnMpXG4gICAgKGlzTG9hZGluZyB8fFxuICAgICAgdXNlcklucHV0T25Qcm9jZXNzaW5nIHx8XG4gICAgICBoYXNSdW5uaW5nVGVhbW1hdGVzIHx8XG4gICAgICAvLyBLZWVwIHNwaW5uZXIgdmlzaWJsZSB3aGlsZSB0YXNrIG5vdGlmaWNhdGlvbnMgYXJlIHF1ZXVlZCBmb3IgcHJvY2Vzc2luZy5cbiAgICAgIC8vIFdpdGhvdXQgdGhpcywgdGhlIHNwaW5uZXIgYnJpZWZseSBkaXNhcHBlYXJzIGJldHdlZW4gY29uc2VjdXRpdmUgbm90aWZpY2F0aW9uc1xuICAgICAgLy8gKGUuZy4sIG11bHRpcGxlIGJhY2tncm91bmQgYWdlbnRzIGNvbXBsZXRpbmcgaW4gcmFwaWQgc3VjY2Vzc2lvbikgYmVjYXVzZVxuICAgICAgLy8gaXNMb2FkaW5nIGdvZXMgZmFsc2UgbW9tZW50YXJpbHkgYmV0d2VlbiBwcm9jZXNzaW5nIGVhY2ggb25lLlxuICAgICAgZ2V0Q29tbWFuZFF1ZXVlTGVuZ3RoKCkgPiAwKSAmJlxuICAgIC8vIEhpZGUgc3Bpbm5lciB3aGVuIHdhaXRpbmcgZm9yIGxlYWRlciB0byBhcHByb3ZlIHBlcm1pc3Npb24gcmVxdWVzdFxuICAgICFwZW5kaW5nV29ya2VyUmVxdWVzdCAmJlxuICAgICFvbmx5U2xlZXBUb29sQWN0aXZlICYmXG4gICAgLy8gSGlkZSBzcGlubmVyIHdoZW4gc3RyZWFtaW5nIHRleHQgaXMgdmlzaWJsZSAodGhlIHRleHQgSVMgdGhlIGZlZWRiYWNrKSxcbiAgICAvLyBidXQga2VlcCBpdCB3aGVuIGlzQnJpZWZPbmx5IHN1cHByZXNzZXMgdGhlIHN0cmVhbWluZyB0ZXh0IGRpc3BsYXlcbiAgICAoIXZpc2libGVTdHJlYW1pbmdUZXh0IHx8IGlzQnJpZWZPbmx5KVxuXG4gIC8vIENoZWNrIGlmIGFueSBwZXJtaXNzaW9uIG9yIGFzayBxdWVzdGlvbiBwcm9tcHQgaXMgY3VycmVudGx5IHZpc2libGVcbiAgLy8gVGhpcyBpcyB1c2VkIHRvIHByZXZlbnQgdGhlIHN1cnZleSBmcm9tIG9wZW5pbmcgd2hpbGUgcHJvbXB0cyBhcmUgYWN0aXZlXG4gIGNvbnN0IGhhc0FjdGl2ZVByb21wdCA9XG4gICAgdG9vbFVzZUNvbmZpcm1RdWV1ZS5sZW5ndGggPiAwIHx8XG4gICAgcHJvbXB0UXVldWUubGVuZ3RoID4gMCB8fFxuICAgIHNhbmRib3hQZXJtaXNzaW9uUmVxdWVzdFF1ZXVlLmxlbmd0aCA+IDAgfHxcbiAgICBlbGljaXRhdGlvbi5xdWV1ZS5sZW5ndGggPiAwIHx8XG4gICAgd29ya2VyU2FuZGJveFBlcm1pc3Npb25zLnF1ZXVlLmxlbmd0aCA+IDBcblxuICBjb25zdCBmZWVkYmFja1N1cnZleU9yaWdpbmFsID0gdXNlRmVlZGJhY2tTdXJ2ZXkoXG4gICAgbWVzc2FnZXMsXG4gICAgaXNMb2FkaW5nLFxuICAgIHN1Ym1pdENvdW50LFxuICAgICdzZXNzaW9uJyxcbiAgICBoYXNBY3RpdmVQcm9tcHQsXG4gIClcblxuICBjb25zdCBza2lsbEltcHJvdmVtZW50U3VydmV5ID0gdXNlU2tpbGxJbXByb3ZlbWVudFN1cnZleShzZXRNZXNzYWdlcylcblxuICBjb25zdCBzaG93SXNzdWVGbGFnQmFubmVyID0gdXNlSXNzdWVGbGFnQmFubmVyKG1lc3NhZ2VzLCBzdWJtaXRDb3VudClcblxuICAvLyBXcmFwIGZlZWRiYWNrIHN1cnZleSBoYW5kbGVyIHRvIHRyaWdnZXIgYXV0by1ydW4gL2lzc3VlXG4gIGNvbnN0IGZlZWRiYWNrU3VydmV5ID0gdXNlTWVtbyhcbiAgICAoKSA9PiAoe1xuICAgICAgLi4uZmVlZGJhY2tTdXJ2ZXlPcmlnaW5hbCxcbiAgICAgIGhhbmRsZVNlbGVjdDogKHNlbGVjdGVkOiAnZGlzbWlzc2VkJyB8ICdiYWQnIHwgJ2ZpbmUnIHwgJ2dvb2QnKSA9PiB7XG4gICAgICAgIC8vIFJlc2V0IHRoZSByZWYgd2hlbiBhIG5ldyBzdXJ2ZXkgcmVzcG9uc2UgY29tZXMgaW5cbiAgICAgICAgZGlkQXV0b1J1bklzc3VlUmVmLmN1cnJlbnQgPSBmYWxzZVxuICAgICAgICBjb25zdCBzaG93ZWRUcmFuc2NyaXB0UHJvbXB0ID1cbiAgICAgICAgICBmZWVkYmFja1N1cnZleU9yaWdpbmFsLmhhbmRsZVNlbGVjdChzZWxlY3RlZClcbiAgICAgICAgLy8gQXV0by1ydW4gL2lzc3VlIGZvciBcImJhZFwiIGlmIHRyYW5zY3JpcHQgcHJvbXB0IHdhc24ndCBzaG93blxuICAgICAgICBpZiAoXG4gICAgICAgICAgc2VsZWN0ZWQgPT09ICdiYWQnICYmXG4gICAgICAgICAgIXNob3dlZFRyYW5zY3JpcHRQcm9tcHQgJiZcbiAgICAgICAgICBzaG91bGRBdXRvUnVuSXNzdWUoJ2ZlZWRiYWNrX3N1cnZleV9iYWQnKVxuICAgICAgICApIHtcbiAgICAgICAgICBzZXRBdXRvUnVuSXNzdWVSZWFzb24oJ2ZlZWRiYWNrX3N1cnZleV9iYWQnKVxuICAgICAgICAgIGRpZEF1dG9SdW5Jc3N1ZVJlZi5jdXJyZW50ID0gdHJ1ZVxuICAgICAgICB9XG4gICAgICB9LFxuICAgIH0pLFxuICAgIFtmZWVkYmFja1N1cnZleU9yaWdpbmFsXSxcbiAgKVxuXG4gIC8vIFBvc3QtY29tcGFjdCBzdXJ2ZXk6IHNob3duIGFmdGVyIGNvbXBhY3Rpb24gaWYgZmVhdHVyZSBnYXRlIGlzIGVuYWJsZWRcbiAgY29uc3QgcG9zdENvbXBhY3RTdXJ2ZXkgPSB1c2VQb3N0Q29tcGFjdFN1cnZleShcbiAgICBtZXNzYWdlcyxcbiAgICBpc0xvYWRpbmcsXG4gICAgaGFzQWN0aXZlUHJvbXB0LFxuICAgIHsgZW5hYmxlZDogIWlzUmVtb3RlU2Vzc2lvbiB9LFxuICApXG5cbiAgLy8gTWVtb3J5IHN1cnZleTogc2hvd24gd2hlbiB0aGUgYXNzaXN0YW50IG1lbnRpb25zIG1lbW9yeSBhbmQgYSBtZW1vcnkgZmlsZVxuICAvLyB3YXMgcmVhZCB0aGlzIGNvbnZlcnNhdGlvblxuICBjb25zdCBtZW1vcnlTdXJ2ZXkgPSB1c2VNZW1vcnlTdXJ2ZXkobWVzc2FnZXMsIGlzTG9hZGluZywgaGFzQWN0aXZlUHJvbXB0LCB7XG4gICAgZW5hYmxlZDogIWlzUmVtb3RlU2Vzc2lvbixcbiAgfSlcblxuICAvLyBGcnVzdHJhdGlvbiBkZXRlY3Rpb246IHNob3cgdHJhbnNjcmlwdCBzaGFyaW5nIHByb21wdCBhZnRlciBkZXRlY3RpbmcgZnJ1c3RyYXRlZCBtZXNzYWdlc1xuICBjb25zdCBmcnVzdHJhdGlvbkRldGVjdGlvbiA9IHVzZUZydXN0cmF0aW9uRGV0ZWN0aW9uKFxuICAgIG1lc3NhZ2VzLFxuICAgIGlzTG9hZGluZyxcbiAgICBoYXNBY3RpdmVQcm9tcHQsXG4gICAgZmVlZGJhY2tTdXJ2ZXkuc3RhdGUgIT09ICdjbG9zZWQnIHx8XG4gICAgICBwb3N0Q29tcGFjdFN1cnZleS5zdGF0ZSAhPT0gJ2Nsb3NlZCcgfHxcbiAgICAgIG1lbW9yeVN1cnZleS5zdGF0ZSAhPT0gJ2Nsb3NlZCcsXG4gIClcblxuICAvLyBJbml0aWFsaXplIElERSBpbnRlZ3JhdGlvblxuICB1c2VJREVJbnRlZ3JhdGlvbih7XG4gICAgYXV0b0Nvbm5lY3RJZGVGbGFnLFxuICAgIGlkZVRvSW5zdGFsbEV4dGVuc2lvbixcbiAgICBzZXREeW5hbWljTWNwQ29uZmlnLFxuICAgIHNldFNob3dJZGVPbmJvYXJkaW5nLFxuICAgIHNldElERUluc3RhbGxhdGlvblN0YXRlOiBzZXRJREVJbnN0YWxsYXRpb25TdGF0dXMsXG4gIH0pXG5cbiAgdXNlRmlsZUhpc3RvcnlTbmFwc2hvdEluaXQoXG4gICAgaW5pdGlhbEZpbGVIaXN0b3J5U25hcHNob3RzLFxuICAgIGZpbGVIaXN0b3J5LFxuICAgIGZpbGVIaXN0b3J5U3RhdGUgPT5cbiAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4gKHtcbiAgICAgICAgLi4ucHJldixcbiAgICAgICAgZmlsZUhpc3Rvcnk6IGZpbGVIaXN0b3J5U3RhdGUsXG4gICAgICB9KSksXG4gIClcblxuICBjb25zdCByZXN1bWUgPSB1c2VDYWxsYmFjayhcbiAgICBhc3luYyAoc2Vzc2lvbklkOiBVVUlELCBsb2c6IExvZ09wdGlvbiwgZW50cnlwb2ludDogUmVzdW1lRW50cnlwb2ludCkgPT4ge1xuICAgICAgY29uc3QgcmVzdW1lU3RhcnQgPSBwZXJmb3JtYW5jZS5ub3coKVxuICAgICAgdHJ5IHtcbiAgICAgICAgLy8gRGVzZXJpYWxpemUgbWVzc2FnZXMgdG8gcHJvcGVybHkgY2xlYW4gdXAgdGhlIGNvbnZlcnNhdGlvblxuICAgICAgICAvLyBUaGlzIGZpbHRlcnMgdW5yZXNvbHZlZCB0b29sIHVzZXMgYW5kIGFkZHMgYSBzeW50aGV0aWMgYXNzaXN0YW50IG1lc3NhZ2UgaWYgbmVlZGVkXG4gICAgICAgIGNvbnN0IG1lc3NhZ2VzID0gZGVzZXJpYWxpemVNZXNzYWdlcyhsb2cubWVzc2FnZXMpXG5cbiAgICAgICAgLy8gTWF0Y2ggY29vcmRpbmF0b3Ivbm9ybWFsIG1vZGUgdG8gdGhlIHJlc3VtZWQgc2Vzc2lvblxuICAgICAgICBpZiAoZmVhdHVyZSgnQ09PUkRJTkFUT1JfTU9ERScpKSB7XG4gICAgICAgICAgLyogZXNsaW50LWRpc2FibGUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXJlcXVpcmUtaW1wb3J0cyAqL1xuICAgICAgICAgIGNvbnN0IGNvb3JkaW5hdG9yTW9kdWxlID1cbiAgICAgICAgICAgIHJlcXVpcmUoJy4uL2Nvb3JkaW5hdG9yL2Nvb3JkaW5hdG9yTW9kZS5qcycpIGFzIHR5cGVvZiBpbXBvcnQoJy4uL2Nvb3JkaW5hdG9yL2Nvb3JkaW5hdG9yTW9kZS5qcycpXG4gICAgICAgICAgLyogZXNsaW50LWVuYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tcmVxdWlyZS1pbXBvcnRzICovXG4gICAgICAgICAgY29uc3Qgd2FybmluZyA9IGNvb3JkaW5hdG9yTW9kdWxlLm1hdGNoU2Vzc2lvbk1vZGUobG9nLm1vZGUpXG4gICAgICAgICAgaWYgKHdhcm5pbmcpIHtcbiAgICAgICAgICAgIC8vIFJlLWRlcml2ZSBhZ2VudCBkZWZpbml0aW9ucyBhZnRlciBtb2RlIHN3aXRjaCBzbyBidWlsdC1pbiBhZ2VudHNcbiAgICAgICAgICAgIC8vIHJlZmxlY3QgdGhlIG5ldyBjb29yZGluYXRvci9ub3JtYWwgbW9kZVxuICAgICAgICAgICAgLyogZXNsaW50LWRpc2FibGUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXJlcXVpcmUtaW1wb3J0cyAqL1xuICAgICAgICAgICAgY29uc3Qge1xuICAgICAgICAgICAgICBnZXRBZ2VudERlZmluaXRpb25zV2l0aE92ZXJyaWRlcyxcbiAgICAgICAgICAgICAgZ2V0QWN0aXZlQWdlbnRzRnJvbUxpc3QsXG4gICAgICAgICAgICB9ID1cbiAgICAgICAgICAgICAgcmVxdWlyZSgnLi4vdG9vbHMvQWdlbnRUb29sL2xvYWRBZ2VudHNEaXIuanMnKSBhcyB0eXBlb2YgaW1wb3J0KCcuLi90b29scy9BZ2VudFRvb2wvbG9hZEFnZW50c0Rpci5qcycpXG4gICAgICAgICAgICAvKiBlc2xpbnQtZW5hYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby1yZXF1aXJlLWltcG9ydHMgKi9cbiAgICAgICAgICAgIGdldEFnZW50RGVmaW5pdGlvbnNXaXRoT3ZlcnJpZGVzLmNhY2hlLmNsZWFyPy4oKVxuICAgICAgICAgICAgY29uc3QgZnJlc2hBZ2VudERlZnMgPSBhd2FpdCBnZXRBZ2VudERlZmluaXRpb25zV2l0aE92ZXJyaWRlcyhcbiAgICAgICAgICAgICAgZ2V0T3JpZ2luYWxDd2QoKSxcbiAgICAgICAgICAgIClcblxuICAgICAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiAoe1xuICAgICAgICAgICAgICAuLi5wcmV2LFxuICAgICAgICAgICAgICBhZ2VudERlZmluaXRpb25zOiB7XG4gICAgICAgICAgICAgICAgLi4uZnJlc2hBZ2VudERlZnMsXG4gICAgICAgICAgICAgICAgYWxsQWdlbnRzOiBmcmVzaEFnZW50RGVmcy5hbGxBZ2VudHMsXG4gICAgICAgICAgICAgICAgYWN0aXZlQWdlbnRzOiBnZXRBY3RpdmVBZ2VudHNGcm9tTGlzdChmcmVzaEFnZW50RGVmcy5hbGxBZ2VudHMpLFxuICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgfSkpXG4gICAgICAgICAgICBtZXNzYWdlcy5wdXNoKGNyZWF0ZVN5c3RlbU1lc3NhZ2Uod2FybmluZywgJ3dhcm5pbmcnKSlcbiAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICAvLyBGaXJlIFNlc3Npb25FbmQgaG9va3MgZm9yIHRoZSBjdXJyZW50IHNlc3Npb24gYmVmb3JlIHN0YXJ0aW5nIHRoZVxuICAgICAgICAvLyByZXN1bWVkIG9uZSwgbWlycm9yaW5nIHRoZSAvY2xlYXIgZmxvdyBpbiBjb252ZXJzYXRpb24udHMuXG4gICAgICAgIGNvbnN0IHNlc3Npb25FbmRUaW1lb3V0TXMgPSBnZXRTZXNzaW9uRW5kSG9va1RpbWVvdXRNcygpXG4gICAgICAgIGF3YWl0IGV4ZWN1dGVTZXNzaW9uRW5kSG9va3MoJ3Jlc3VtZScsIHtcbiAgICAgICAgICBnZXRBcHBTdGF0ZTogKCkgPT4gc3RvcmUuZ2V0U3RhdGUoKSxcbiAgICAgICAgICBzZXRBcHBTdGF0ZSxcbiAgICAgICAgICBzaWduYWw6IEFib3J0U2lnbmFsLnRpbWVvdXQoc2Vzc2lvbkVuZFRpbWVvdXRNcyksXG4gICAgICAgICAgdGltZW91dE1zOiBzZXNzaW9uRW5kVGltZW91dE1zLFxuICAgICAgICB9KVxuXG4gICAgICAgIC8vIFByb2Nlc3Mgc2Vzc2lvbiBzdGFydCBob29rcyBmb3IgcmVzdW1lXG4gICAgICAgIGNvbnN0IGhvb2tNZXNzYWdlcyA9IGF3YWl0IHByb2Nlc3NTZXNzaW9uU3RhcnRIb29rcygncmVzdW1lJywge1xuICAgICAgICAgIHNlc3Npb25JZCxcbiAgICAgICAgICBhZ2VudFR5cGU6IG1haW5UaHJlYWRBZ2VudERlZmluaXRpb24/LmFnZW50VHlwZSxcbiAgICAgICAgICBtb2RlbDogbWFpbkxvb3BNb2RlbCxcbiAgICAgICAgfSlcblxuICAgICAgICAvLyBBcHBlbmQgaG9vayBtZXNzYWdlcyB0byB0aGUgY29udmVyc2F0aW9uXG4gICAgICAgIG1lc3NhZ2VzLnB1c2goLi4uaG9va01lc3NhZ2VzKVxuICAgICAgICAvLyBGb3IgZm9ya3MsIGdlbmVyYXRlIGEgbmV3IHBsYW4gc2x1ZyBhbmQgY29weSB0aGUgcGxhbiBjb250ZW50IHNvIHRoZVxuICAgICAgICAvLyBvcmlnaW5hbCBhbmQgZm9ya2VkIHNlc3Npb25zIGRvbid0IGNsb2JiZXIgZWFjaCBvdGhlcidzIHBsYW4gZmlsZXMuXG4gICAgICAgIC8vIEZvciByZWd1bGFyIHJlc3VtZXMsIHJldXNlIHRoZSBvcmlnaW5hbCBzZXNzaW9uJ3MgcGxhbiBzbHVnLlxuICAgICAgICBpZiAoZW50cnlwb2ludCA9PT0gJ2ZvcmsnKSB7XG4gICAgICAgICAgdm9pZCBjb3B5UGxhbkZvckZvcmsobG9nLCBhc1Nlc3Npb25JZChzZXNzaW9uSWQpKVxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHZvaWQgY29weVBsYW5Gb3JSZXN1bWUobG9nLCBhc1Nlc3Npb25JZChzZXNzaW9uSWQpKVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gUmVzdG9yZSBmaWxlIGhpc3RvcnkgYW5kIGF0dHJpYnV0aW9uIHN0YXRlIGZyb20gdGhlIHJlc3VtZWQgY29udmVyc2F0aW9uXG4gICAgICAgIHJlc3RvcmVTZXNzaW9uU3RhdGVGcm9tTG9nKGxvZywgc2V0QXBwU3RhdGUpXG4gICAgICAgIGlmIChsb2cuZmlsZUhpc3RvcnlTbmFwc2hvdHMpIHtcbiAgICAgICAgICB2b2lkIGNvcHlGaWxlSGlzdG9yeUZvclJlc3VtZShsb2cpXG4gICAgICAgIH1cblxuICAgICAgICAvLyBSZXN0b3JlIGFnZW50IHNldHRpbmcgZnJvbSB0aGUgcmVzdW1lZCBjb252ZXJzYXRpb25cbiAgICAgICAgLy8gQWx3YXlzIHJlc2V0IHRvIHRoZSBuZXcgc2Vzc2lvbidzIHZhbHVlcyAob3IgY2xlYXIgaWYgbm9uZSksXG4gICAgICAgIC8vIG1hdGNoaW5nIHRoZSBzdGFuZGFsb25lQWdlbnRDb250ZXh0IHBhdHRlcm4gYmVsb3dcbiAgICAgICAgY29uc3QgeyBhZ2VudERlZmluaXRpb246IHJlc3RvcmVkQWdlbnQgfSA9IHJlc3RvcmVBZ2VudEZyb21TZXNzaW9uKFxuICAgICAgICAgIGxvZy5hZ2VudFNldHRpbmcsXG4gICAgICAgICAgaW5pdGlhbE1haW5UaHJlYWRBZ2VudERlZmluaXRpb24sXG4gICAgICAgICAgYWdlbnREZWZpbml0aW9ucyxcbiAgICAgICAgKVxuICAgICAgICBzZXRNYWluVGhyZWFkQWdlbnREZWZpbml0aW9uKHJlc3RvcmVkQWdlbnQpXG4gICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4gKHsgLi4ucHJldiwgYWdlbnQ6IHJlc3RvcmVkQWdlbnQ/LmFnZW50VHlwZSB9KSlcblxuICAgICAgICAvLyBSZXN0b3JlIHN0YW5kYWxvbmUgYWdlbnQgY29udGV4dCBmcm9tIHRoZSByZXN1bWVkIGNvbnZlcnNhdGlvblxuICAgICAgICAvLyBBbHdheXMgcmVzZXQgdG8gdGhlIG5ldyBzZXNzaW9uJ3MgdmFsdWVzIChvciBjbGVhciBpZiBub25lKVxuICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+ICh7XG4gICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICBzdGFuZGFsb25lQWdlbnRDb250ZXh0OiBjb21wdXRlU3RhbmRhbG9uZUFnZW50Q29udGV4dChcbiAgICAgICAgICAgIGxvZy5hZ2VudE5hbWUsXG4gICAgICAgICAgICBsb2cuYWdlbnRDb2xvcixcbiAgICAgICAgICApLFxuICAgICAgICB9KSlcbiAgICAgICAgdm9pZCB1cGRhdGVTZXNzaW9uTmFtZShsb2cuYWdlbnROYW1lKVxuXG4gICAgICAgIC8vIFJlc3RvcmUgcmVhZCBmaWxlIHN0YXRlIGZyb20gdGhlIG1lc3NhZ2UgaGlzdG9yeVxuICAgICAgICByZXN0b3JlUmVhZEZpbGVTdGF0ZShtZXNzYWdlcywgbG9nLnByb2plY3RQYXRoID8/IGdldE9yaWdpbmFsQ3dkKCkpXG5cbiAgICAgICAgLy8gQ2xlYXIgYW55IGFjdGl2ZSBsb2FkaW5nIHN0YXRlIChubyBxdWVyeUlkIHNpbmNlIHdlJ3JlIG5vdCBpbiBhIHF1ZXJ5KVxuICAgICAgICByZXNldExvYWRpbmdTdGF0ZSgpXG4gICAgICAgIHNldEFib3J0Q29udHJvbGxlcihudWxsKVxuXG4gICAgICAgIHNldENvbnZlcnNhdGlvbklkKHNlc3Npb25JZClcblxuICAgICAgICAvLyBHZXQgdGFyZ2V0IHNlc3Npb24ncyBjb3N0cyBCRUZPUkUgc2F2aW5nIGN1cnJlbnQgc2Vzc2lvblxuICAgICAgICAvLyAoc2F2ZUN1cnJlbnRTZXNzaW9uQ29zdHMgb3ZlcndyaXRlcyB0aGUgY29uZmlnLCBzbyB3ZSBuZWVkIHRvIHJlYWQgZmlyc3QpXG4gICAgICAgIGNvbnN0IHRhcmdldFNlc3Npb25Db3N0cyA9IGdldFN0b3JlZFNlc3Npb25Db3N0cyhzZXNzaW9uSWQpXG5cbiAgICAgICAgLy8gU2F2ZSBjdXJyZW50IHNlc3Npb24ncyBjb3N0cyBiZWZvcmUgc3dpdGNoaW5nIHRvIGF2b2lkIGxvc2luZyBhY2N1bXVsYXRlZCBjb3N0c1xuICAgICAgICBzYXZlQ3VycmVudFNlc3Npb25Db3N0cygpXG5cbiAgICAgICAgLy8gUmVzZXQgY29zdCBzdGF0ZSBmb3IgY2xlYW4gc2xhdGUgYmVmb3JlIHJlc3RvcmluZyB0YXJnZXQgc2Vzc2lvblxuICAgICAgICByZXNldENvc3RTdGF0ZSgpXG5cbiAgICAgICAgLy8gU3dpdGNoIHNlc3Npb24gKGlkICsgcHJvamVjdCBkaXIgYXRvbWljYWxseSkuIGZ1bGxQYXRoIG1heSBwb2ludCB0b1xuICAgICAgICAvLyBhIGRpZmZlcmVudCBwcm9qZWN0IChjcm9zcy13b3JrdHJlZSwgL2JyYW5jaCk7IG51bGwgZGVyaXZlcyBmcm9tXG4gICAgICAgIC8vIGN1cnJlbnQgb3JpZ2luYWxDd2QuXG4gICAgICAgIHN3aXRjaFNlc3Npb24oXG4gICAgICAgICAgYXNTZXNzaW9uSWQoc2Vzc2lvbklkKSxcbiAgICAgICAgICBsb2cuZnVsbFBhdGggPyBkaXJuYW1lKGxvZy5mdWxsUGF0aCkgOiBudWxsLFxuICAgICAgICApXG4gICAgICAgIC8vIFJlbmFtZSBhc2NpaWNhc3QgcmVjb3JkaW5nIHRvIG1hdGNoIHRoZSByZXN1bWVkIHNlc3Npb24gSURcbiAgICAgICAgY29uc3QgeyByZW5hbWVSZWNvcmRpbmdGb3JTZXNzaW9uIH0gPSBhd2FpdCBpbXBvcnQoXG4gICAgICAgICAgJy4uL3V0aWxzL2FzY2lpY2FzdC5qcydcbiAgICAgICAgKVxuICAgICAgICBhd2FpdCByZW5hbWVSZWNvcmRpbmdGb3JTZXNzaW9uKClcbiAgICAgICAgYXdhaXQgcmVzZXRTZXNzaW9uRmlsZVBvaW50ZXIoKVxuXG4gICAgICAgIC8vIENsZWFyIHRoZW4gcmVzdG9yZSBzZXNzaW9uIG1ldGFkYXRhIHNvIGl0J3MgcmUtYXBwZW5kZWQgb24gZXhpdCB2aWFcbiAgICAgICAgLy8gcmVBcHBlbmRTZXNzaW9uTWV0YWRhdGEuIGNsZWFyU2Vzc2lvbk1ldGFkYXRhIG11c3QgYmUgY2FsbGVkIGZpcnN0OlxuICAgICAgICAvLyByZXN0b3JlU2Vzc2lvbk1ldGFkYXRhIG9ubHkgc2V0cy1pZi10cnV0aHksIHNvIHdpdGhvdXQgdGhlIGNsZWFyLFxuICAgICAgICAvLyBhIHNlc3Npb24gd2l0aG91dCBhbiBhZ2VudCBuYW1lIHdvdWxkIGluaGVyaXQgdGhlIHByZXZpb3VzIHNlc3Npb24nc1xuICAgICAgICAvLyBjYWNoZWQgbmFtZSBhbmQgd3JpdGUgaXQgdG8gdGhlIHdyb25nIHRyYW5zY3JpcHQgb24gZmlyc3QgbWVzc2FnZS5cbiAgICAgICAgY2xlYXJTZXNzaW9uTWV0YWRhdGEoKVxuICAgICAgICByZXN0b3JlU2Vzc2lvbk1ldGFkYXRhKGxvZylcbiAgICAgICAgLy8gUmVzdW1lZCBzZXNzaW9ucyBzaG91bGRuJ3QgcmUtdGl0bGUgZnJvbSBtaWQtY29udmVyc2F0aW9uIGNvbnRleHRcbiAgICAgICAgLy8gKHNhbWUgcmVhc29uaW5nIGFzIHRoZSB1c2VSZWYgc2VlZCksIGFuZCB0aGUgcHJldmlvdXMgc2Vzc2lvbidzXG4gICAgICAgIC8vIEhhaWt1IHRpdGxlIHNob3VsZG4ndCBjYXJyeSBvdmVyLlxuICAgICAgICBoYWlrdVRpdGxlQXR0ZW1wdGVkUmVmLmN1cnJlbnQgPSB0cnVlXG4gICAgICAgIHNldEhhaWt1VGl0bGUodW5kZWZpbmVkKVxuXG4gICAgICAgIC8vIEV4aXQgYW55IHdvcmt0cmVlIGEgcHJpb3IgL3Jlc3VtZSBlbnRlcmVkLCB0aGVuIGNkIGludG8gdGhlIG9uZVxuICAgICAgICAvLyB0aGlzIHNlc3Npb24gd2FzIGluLiBXaXRob3V0IHRoZSBleGl0LCByZXN1bWluZyBmcm9tIHdvcmt0cmVlIEJcbiAgICAgICAgLy8gdG8gbm9uLXdvcmt0cmVlIEMgbGVhdmVzIGN3ZC9jdXJyZW50V29ya3RyZWVTZXNzaW9uIHN0YWxlO1xuICAgICAgICAvLyByZXN1bWluZyBC4oaSQyB3aGVyZSBDIGlzIGFsc28gYSB3b3JrdHJlZSBmYWlscyBlbnRpcmVseVxuICAgICAgICAvLyAoZ2V0Q3VycmVudFdvcmt0cmVlU2Vzc2lvbiBndWFyZCBibG9ja3MgdGhlIHN3aXRjaCkuXG4gICAgICAgIC8vXG4gICAgICAgIC8vIFNraXBwZWQgZm9yIC9icmFuY2g6IGZvcmtMb2cgZG9lc24ndCBjYXJyeSB3b3JrdHJlZVNlc3Npb24sIHNvXG4gICAgICAgIC8vIHRoaXMgd291bGQga2ljayB0aGUgdXNlciBvdXQgb2YgYSB3b3JrdHJlZSB0aGV5J3JlIHN0aWxsIHdvcmtpbmdcbiAgICAgICAgLy8gaW4uIFNhbWUgZm9yayBza2lwIGFzIHByb2Nlc3NSZXN1bWVkQ29udmVyc2F0aW9uIGZvciB0aGUgYWRvcHQg4oCUXG4gICAgICAgIC8vIGZvcmsgbWF0ZXJpYWxpemVzIGl0cyBvd24gZmlsZSB2aWEgcmVjb3JkVHJhbnNjcmlwdCBvbiBSRVBMIG1vdW50LlxuICAgICAgICBpZiAoZW50cnlwb2ludCAhPT0gJ2ZvcmsnKSB7XG4gICAgICAgICAgZXhpdFJlc3RvcmVkV29ya3RyZWUoKVxuICAgICAgICAgIHJlc3RvcmVXb3JrdHJlZUZvclJlc3VtZShsb2cud29ya3RyZWVTZXNzaW9uKVxuICAgICAgICAgIGFkb3B0UmVzdW1lZFNlc3Npb25GaWxlKClcbiAgICAgICAgICB2b2lkIHJlc3RvcmVSZW1vdGVBZ2VudFRhc2tzKHtcbiAgICAgICAgICAgIGFib3J0Q29udHJvbGxlcjogbmV3IEFib3J0Q29udHJvbGxlcigpLFxuICAgICAgICAgICAgZ2V0QXBwU3RhdGU6ICgpID0+IHN0b3JlLmdldFN0YXRlKCksXG4gICAgICAgICAgICBzZXRBcHBTdGF0ZSxcbiAgICAgICAgICB9KVxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIC8vIEZvcms6IHNhbWUgcmUtcGVyc2lzdCBhcyAvY2xlYXIgKGNvbnZlcnNhdGlvbi50cykuIFRoZSBjbGVhclxuICAgICAgICAgIC8vIGFib3ZlIHdpcGVkIGN1cnJlbnRTZXNzaW9uV29ya3RyZWUsIGZvcmtMb2cgZG9lc24ndCBjYXJyeSBpdCxcbiAgICAgICAgICAvLyBhbmQgdGhlIHByb2Nlc3MgaXMgc3RpbGwgaW4gdGhlIHNhbWUgd29ya3RyZWUuXG4gICAgICAgICAgY29uc3Qgd3MgPSBnZXRDdXJyZW50V29ya3RyZWVTZXNzaW9uKClcbiAgICAgICAgICBpZiAod3MpIHNhdmVXb3JrdHJlZVN0YXRlKHdzKVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gUGVyc2lzdCB0aGUgY3VycmVudCBtb2RlIHNvIGZ1dHVyZSByZXN1bWVzIGtub3cgd2hhdCBtb2RlIHRoaXMgc2Vzc2lvbiB3YXMgaW5cbiAgICAgICAgaWYgKGZlYXR1cmUoJ0NPT1JESU5BVE9SX01PREUnKSkge1xuICAgICAgICAgIC8qIGVzbGludC1kaXNhYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby1yZXF1aXJlLWltcG9ydHMgKi9cbiAgICAgICAgICBjb25zdCB7IHNhdmVNb2RlIH0gPSByZXF1aXJlKCcuLi91dGlscy9zZXNzaW9uU3RvcmFnZS5qcycpXG4gICAgICAgICAgY29uc3QgeyBpc0Nvb3JkaW5hdG9yTW9kZSB9ID1cbiAgICAgICAgICAgIHJlcXVpcmUoJy4uL2Nvb3JkaW5hdG9yL2Nvb3JkaW5hdG9yTW9kZS5qcycpIGFzIHR5cGVvZiBpbXBvcnQoJy4uL2Nvb3JkaW5hdG9yL2Nvb3JkaW5hdG9yTW9kZS5qcycpXG4gICAgICAgICAgLyogZXNsaW50LWVuYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tcmVxdWlyZS1pbXBvcnRzICovXG4gICAgICAgICAgc2F2ZU1vZGUoaXNDb29yZGluYXRvck1vZGUoKSA/ICdjb29yZGluYXRvcicgOiAnbm9ybWFsJylcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFJlc3RvcmUgdGFyZ2V0IHNlc3Npb24ncyBjb3N0cyBmcm9tIHRoZSBkYXRhIHdlIHJlYWQgZWFybGllclxuICAgICAgICBpZiAodGFyZ2V0U2Vzc2lvbkNvc3RzKSB7XG4gICAgICAgICAgc2V0Q29zdFN0YXRlRm9yUmVzdG9yZSh0YXJnZXRTZXNzaW9uQ29zdHMpXG4gICAgICAgIH1cblxuICAgICAgICAvLyBSZWNvbnN0cnVjdCByZXBsYWNlbWVudCBzdGF0ZSBmb3IgdGhlIHJlc3VtZWQgc2Vzc2lvbi4gUnVucyBhZnRlclxuICAgICAgICAvLyBzZXRTZXNzaW9uSWQgc28gYW55IE5FVyByZXBsYWNlbWVudHMgcG9zdC1yZXN1bWUgd3JpdGUgdG8gdGhlXG4gICAgICAgIC8vIHJlc3VtZWQgc2Vzc2lvbidzIHRvb2wtcmVzdWx0cyBkaXIuIEdhdGVkIG9uIHJlZi5jdXJyZW50OiB0aGVcbiAgICAgICAgLy8gaW5pdGlhbCBtb3VudCBhbHJlYWR5IHJlYWQgdGhlIGZlYXR1cmUgZmxhZywgc28gd2UgZG9uJ3QgcmUtcmVhZFxuICAgICAgICAvLyBpdCBoZXJlIChtaWQtc2Vzc2lvbiBmbGFnIGZsaXBzIHN0YXkgdW5vYnNlcnZhYmxlIGluIGJvdGhcbiAgICAgICAgLy8gZGlyZWN0aW9ucykuXG4gICAgICAgIC8vXG4gICAgICAgIC8vIFNraXBwZWQgZm9yIGluLXNlc3Npb24gL2JyYW5jaDogdGhlIGV4aXN0aW5nIHJlZiBpcyBhbHJlYWR5IGNvcnJlY3RcbiAgICAgICAgLy8gKGJyYW5jaCBwcmVzZXJ2ZXMgdG9vbF91c2VfaWRzKSwgc28gdGhlcmUncyBubyBuZWVkIHRvIHJlY29uc3RydWN0LlxuICAgICAgICAvLyBjcmVhdGVGb3JrKCkgZG9lcyB3cml0ZSBjb250ZW50LXJlcGxhY2VtZW50IGVudHJpZXMgdG8gdGhlIGZvcmtlZFxuICAgICAgICAvLyBKU09OTCB3aXRoIHRoZSBmb3JrJ3Mgc2Vzc2lvbklkLCBzbyBgY2xhdWRlIC1yIHtmb3JrSWR9YCBhbHNvIHdvcmtzLlxuICAgICAgICBpZiAoY29udGVudFJlcGxhY2VtZW50U3RhdGVSZWYuY3VycmVudCAmJiBlbnRyeXBvaW50ICE9PSAnZm9yaycpIHtcbiAgICAgICAgICBjb250ZW50UmVwbGFjZW1lbnRTdGF0ZVJlZi5jdXJyZW50ID1cbiAgICAgICAgICAgIHJlY29uc3RydWN0Q29udGVudFJlcGxhY2VtZW50U3RhdGUoXG4gICAgICAgICAgICAgIG1lc3NhZ2VzLFxuICAgICAgICAgICAgICBsb2cuY29udGVudFJlcGxhY2VtZW50cyA/PyBbXSxcbiAgICAgICAgICAgIClcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFJlc2V0IG1lc3NhZ2VzIHRvIHRoZSBwcm92aWRlZCBpbml0aWFsIG1lc3NhZ2VzXG4gICAgICAgIC8vIFVzZSBhIGNhbGxiYWNrIHRvIGVuc3VyZSB3ZSdyZSBub3QgZGVwZW5kZW50IG9uIHN0YWxlIHN0YXRlXG4gICAgICAgIHNldE1lc3NhZ2VzKCgpID0+IG1lc3NhZ2VzKVxuXG4gICAgICAgIC8vIENsZWFyIGFueSBhY3RpdmUgdG9vbCBKU1hcbiAgICAgICAgc2V0VG9vbEpTWChudWxsKVxuXG4gICAgICAgIC8vIENsZWFyIGlucHV0IHRvIGVuc3VyZSBubyByZXNpZHVhbCBzdGF0ZVxuICAgICAgICBzZXRJbnB1dFZhbHVlKCcnKVxuXG4gICAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9zZXNzaW9uX3Jlc3VtZWQnLCB7XG4gICAgICAgICAgZW50cnlwb2ludDpcbiAgICAgICAgICAgIGVudHJ5cG9pbnQgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgICAgICBzdWNjZXNzOiB0cnVlLFxuICAgICAgICAgIHJlc3VtZV9kdXJhdGlvbl9tczogTWF0aC5yb3VuZChwZXJmb3JtYW5jZS5ub3coKSAtIHJlc3VtZVN0YXJ0KSxcbiAgICAgICAgfSlcbiAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9zZXNzaW9uX3Jlc3VtZWQnLCB7XG4gICAgICAgICAgZW50cnlwb2ludDpcbiAgICAgICAgICAgIGVudHJ5cG9pbnQgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgICAgICBzdWNjZXNzOiBmYWxzZSxcbiAgICAgICAgfSlcbiAgICAgICAgdGhyb3cgZXJyb3JcbiAgICAgIH1cbiAgICB9LFxuICAgIFtyZXNldExvYWRpbmdTdGF0ZSwgc2V0QXBwU3RhdGVdLFxuICApXG5cbiAgLy8gTGF6eSBpbml0OiB1c2VSZWYoY3JlYXRlWCgpKSB3b3VsZCBjYWxsIGNyZWF0ZVggb24gZXZlcnkgcmVuZGVyIGFuZFxuICAvLyBkaXNjYXJkIHRoZSByZXN1bHQuIExSVUNhY2hlIGNvbnN0cnVjdGlvbiBpbnNpZGUgRmlsZVN0YXRlQ2FjaGUgaXNcbiAgLy8gZXhwZW5zaXZlICh+MTcwbXMpLCBzbyB3ZSB1c2UgdXNlU3RhdGUncyBsYXp5IGluaXRpYWxpemVyIHRvIGNyZWF0ZVxuICAvLyBpdCBleGFjdGx5IG9uY2UsIHRoZW4gZmVlZCB0aGF0IHN0YWJsZSByZWZlcmVuY2UgaW50byB1c2VSZWYuXG4gIGNvbnN0IFtpbml0aWFsUmVhZEZpbGVTdGF0ZV0gPSB1c2VTdGF0ZSgoKSA9PlxuICAgIGNyZWF0ZUZpbGVTdGF0ZUNhY2hlV2l0aFNpemVMaW1pdChSRUFEX0ZJTEVfU1RBVEVfQ0FDSEVfU0laRSksXG4gIClcbiAgY29uc3QgcmVhZEZpbGVTdGF0ZSA9IHVzZVJlZihpbml0aWFsUmVhZEZpbGVTdGF0ZSlcbiAgY29uc3QgYmFzaFRvb2xzID0gdXNlUmVmKG5ldyBTZXQ8c3RyaW5nPigpKVxuICBjb25zdCBiYXNoVG9vbHNQcm9jZXNzZWRJZHggPSB1c2VSZWYoMClcbiAgLy8gU2Vzc2lvbi1zY29wZWQgc2tpbGwgZGlzY292ZXJ5IHRyYWNraW5nIChmZWVkcyB3YXNfZGlzY292ZXJlZCBvblxuICAvLyB0ZW5ndV9za2lsbF90b29sX2ludm9jYXRpb24pLiBNdXN0IHBlcnNpc3QgYWNyb3NzIGdldFRvb2xVc2VDb250ZXh0XG4gIC8vIHJlYnVpbGRzIHdpdGhpbiBhIHNlc3Npb246IHR1cm4tMCBkaXNjb3Zlcnkgd3JpdGVzIHZpYSBwcm9jZXNzVXNlcklucHV0XG4gIC8vIGJlZm9yZSBvblF1ZXJ5IGJ1aWxkcyBpdHMgb3duIGNvbnRleHQsIGFuZCBkaXNjb3Zlcnkgb24gdHVybiBOIG11c3RcbiAgLy8gc3RpbGwgYXR0cmlidXRlIGEgU2tpbGxUb29sIGNhbGwgb24gdHVybiBOK2suIENsZWFyZWQgaW4gY2xlYXJDb252ZXJzYXRpb24uXG4gIGNvbnN0IGRpc2NvdmVyZWRTa2lsbE5hbWVzUmVmID0gdXNlUmVmKG5ldyBTZXQ8c3RyaW5nPigpKVxuICAvLyBTZXNzaW9uLWxldmVsIGRlZHVwIGZvciBuZXN0ZWRfbWVtb3J5IENMQVVERS5tZCBhdHRhY2htZW50cy5cbiAgLy8gcmVhZEZpbGVTdGF0ZSBpcyBhIDEwMC1lbnRyeSBMUlU7IG9uY2UgaXQgZXZpY3RzIGEgQ0xBVURFLm1kIHBhdGgsXG4gIC8vIHRoZSBuZXh0IGRpc2NvdmVyeSBjeWNsZSByZS1pbmplY3RzIGl0LiBDbGVhcmVkIGluIGNsZWFyQ29udmVyc2F0aW9uLlxuICBjb25zdCBsb2FkZWROZXN0ZWRNZW1vcnlQYXRoc1JlZiA9IHVzZVJlZihuZXcgU2V0PHN0cmluZz4oKSlcblxuICAvLyBIZWxwZXIgdG8gcmVzdG9yZSByZWFkIGZpbGUgc3RhdGUgZnJvbSBtZXNzYWdlcyAodXNlZCBmb3IgcmVzdW1lIGZsb3dzKVxuICAvLyBUaGlzIGFsbG93cyBDbGF1ZGUgdG8gZWRpdCBmaWxlcyB0aGF0IHdlcmUgcmVhZCBpbiBwcmV2aW91cyBzZXNzaW9uc1xuICBjb25zdCByZXN0b3JlUmVhZEZpbGVTdGF0ZSA9IHVzZUNhbGxiYWNrKFxuICAgIChtZXNzYWdlczogTWVzc2FnZVR5cGVbXSwgY3dkOiBzdHJpbmcpID0+IHtcbiAgICAgIGNvbnN0IGV4dHJhY3RlZCA9IGV4dHJhY3RSZWFkRmlsZXNGcm9tTWVzc2FnZXMoXG4gICAgICAgIG1lc3NhZ2VzLFxuICAgICAgICBjd2QsXG4gICAgICAgIFJFQURfRklMRV9TVEFURV9DQUNIRV9TSVpFLFxuICAgICAgKVxuICAgICAgcmVhZEZpbGVTdGF0ZS5jdXJyZW50ID0gbWVyZ2VGaWxlU3RhdGVDYWNoZXMoXG4gICAgICAgIHJlYWRGaWxlU3RhdGUuY3VycmVudCxcbiAgICAgICAgZXh0cmFjdGVkLFxuICAgICAgKVxuICAgICAgZm9yIChjb25zdCB0b29sIG9mIGV4dHJhY3RCYXNoVG9vbHNGcm9tTWVzc2FnZXMobWVzc2FnZXMpKSB7XG4gICAgICAgIGJhc2hUb29scy5jdXJyZW50LmFkZCh0b29sKVxuICAgICAgfVxuICAgIH0sXG4gICAgW10sXG4gIClcblxuICAvLyBFeHRyYWN0IHJlYWQgZmlsZSBzdGF0ZSBmcm9tIGluaXRpYWxNZXNzYWdlcyBvbiBtb3VudFxuICAvLyBUaGlzIGhhbmRsZXMgQ0xJIGZsYWcgcmVzdW1lICgtLXJlc3VtZS1zZXNzaW9uKSBhbmQgUmVzdW1lQ29udmVyc2F0aW9uIHNjcmVlblxuICAvLyB3aGVyZSBtZXNzYWdlcyBhcmUgcGFzc2VkIGFzIHByb3BzIHJhdGhlciB0aGFuIHRocm91Z2ggdGhlIHJlc3VtZSBjYWxsYmFja1xuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmIChpbml0aWFsTWVzc2FnZXMgJiYgaW5pdGlhbE1lc3NhZ2VzLmxlbmd0aCA+IDApIHtcbiAgICAgIHJlc3RvcmVSZWFkRmlsZVN0YXRlKGluaXRpYWxNZXNzYWdlcywgZ2V0T3JpZ2luYWxDd2QoKSlcbiAgICAgIHZvaWQgcmVzdG9yZVJlbW90ZUFnZW50VGFza3Moe1xuICAgICAgICBhYm9ydENvbnRyb2xsZXI6IG5ldyBBYm9ydENvbnRyb2xsZXIoKSxcbiAgICAgICAgZ2V0QXBwU3RhdGU6ICgpID0+IHN0b3JlLmdldFN0YXRlKCksXG4gICAgICAgIHNldEFwcFN0YXRlLFxuICAgICAgfSlcbiAgICB9XG4gICAgLy8gT25seSBydW4gb24gbW91bnQgLSBpbml0aWFsTWVzc2FnZXMgc2hvdWxkbid0IGNoYW5nZSBkdXJpbmcgY29tcG9uZW50IGxpZmV0aW1lXG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIHJlYWN0LWhvb2tzL2V4aGF1c3RpdmUtZGVwc1xuICB9LCBbXSlcblxuICBjb25zdCB7IHN0YXR1czogYXBpS2V5U3RhdHVzLCByZXZlcmlmeSB9ID0gdXNlQXBpS2V5VmVyaWZpY2F0aW9uKClcblxuICAvLyBBdXRvLXJ1biAvaXNzdWUgc3RhdGVcbiAgY29uc3QgW2F1dG9SdW5Jc3N1ZVJlYXNvbiwgc2V0QXV0b1J1bklzc3VlUmVhc29uXSA9XG4gICAgdXNlU3RhdGU8QXV0b1J1bklzc3VlUmVhc29uIHwgbnVsbD4obnVsbClcbiAgLy8gUmVmIHRvIHRyYWNrIGlmIGF1dG9SdW5Jc3N1ZSB3YXMgdHJpZ2dlcmVkIHRoaXMgc3VydmV5IGN5Y2xlLFxuICAvLyBzbyB3ZSBjYW4gc3VwcHJlc3MgdGhlIFsxXSBmb2xsb3ctdXAgcHJvbXB0IGV2ZW4gYWZ0ZXJcbiAgLy8gYXV0b1J1bklzc3VlUmVhc29uIGlzIGNsZWFyZWQuXG4gIGNvbnN0IGRpZEF1dG9SdW5Jc3N1ZVJlZiA9IHVzZVJlZihmYWxzZSlcblxuICAvLyBTdGF0ZSBmb3IgZXhpdCBmZWVkYmFjayBmbG93XG4gIGNvbnN0IFtleGl0Rmxvdywgc2V0RXhpdEZsb3ddID0gdXNlU3RhdGU8UmVhY3QuUmVhY3ROb2RlPihudWxsKVxuICBjb25zdCBbaXNFeGl0aW5nLCBzZXRJc0V4aXRpbmddID0gdXNlU3RhdGUoZmFsc2UpXG5cbiAgLy8gQ2FsY3VsYXRlIGlmIGNvc3QgZGlhbG9nIHNob3VsZCBiZSBzaG93blxuICBjb25zdCBzaG93aW5nQ29zdERpYWxvZyA9ICFpc0xvYWRpbmcgJiYgc2hvd0Nvc3REaWFsb2dcblxuICAvLyBEZXRlcm1pbmUgd2hpY2ggZGlhbG9nIHNob3VsZCBoYXZlIGZvY3VzIChpZiBhbnkpXG4gIC8vIFBlcm1pc3Npb24gYW5kIGludGVyYWN0aXZlIGRpYWxvZ3MgY2FuIHNob3cgZXZlbiB3aGVuIHRvb2xKU1ggaXMgc2V0LFxuICAvLyBhcyBsb25nIGFzIHNob3VsZENvbnRpbnVlQW5pbWF0aW9uIGlzIHRydWUuIFRoaXMgcHJldmVudHMgZGVhZGxvY2tzIHdoZW5cbiAgLy8gYWdlbnRzIHNldCBiYWNrZ3JvdW5kIGhpbnRzIHdoaWxlIHdhaXRpbmcgZm9yIHVzZXIgaW50ZXJhY3Rpb24uXG4gIGZ1bmN0aW9uIGdldEZvY3VzZWRJbnB1dERpYWxvZygpOlxuICAgIHwgJ21lc3NhZ2Utc2VsZWN0b3InXG4gICAgfCAnc2FuZGJveC1wZXJtaXNzaW9uJ1xuICAgIHwgJ3Rvb2wtcGVybWlzc2lvbidcbiAgICB8ICdwcm9tcHQnXG4gICAgfCAnd29ya2VyLXNhbmRib3gtcGVybWlzc2lvbidcbiAgICB8ICdlbGljaXRhdGlvbidcbiAgICB8ICdjb3N0J1xuICAgIHwgJ2lkbGUtcmV0dXJuJ1xuICAgIHwgJ2luaXQtb25ib2FyZGluZydcbiAgICB8ICdpZGUtb25ib2FyZGluZydcbiAgICB8ICdtb2RlbC1zd2l0Y2gnXG4gICAgfCAndW5kZXJjb3Zlci1jYWxsb3V0J1xuICAgIHwgJ2VmZm9ydC1jYWxsb3V0J1xuICAgIHwgJ3JlbW90ZS1jYWxsb3V0J1xuICAgIHwgJ2xzcC1yZWNvbW1lbmRhdGlvbidcbiAgICB8ICdwbHVnaW4taGludCdcbiAgICB8ICdkZXNrdG9wLXVwc2VsbCdcbiAgICB8ICd1bHRyYXBsYW4tY2hvaWNlJ1xuICAgIHwgJ3VsdHJhcGxhbi1sYXVuY2gnXG4gICAgfCB1bmRlZmluZWQge1xuICAgIC8vIEV4aXQgc3RhdGVzIGFsd2F5cyB0YWtlIHByZWNlZGVuY2VcbiAgICBpZiAoaXNFeGl0aW5nIHx8IGV4aXRGbG93KSByZXR1cm4gdW5kZWZpbmVkXG5cbiAgICAvLyBIaWdoIHByaW9yaXR5IGRpYWxvZ3MgKGFsd2F5cyBzaG93IHJlZ2FyZGxlc3Mgb2YgdHlwaW5nKVxuICAgIGlmIChpc01lc3NhZ2VTZWxlY3RvclZpc2libGUpIHJldHVybiAnbWVzc2FnZS1zZWxlY3RvcidcblxuICAgIC8vIFN1cHByZXNzIGludGVycnVwdCBkaWFsb2dzIHdoaWxlIHVzZXIgaXMgYWN0aXZlbHkgdHlwaW5nXG4gICAgaWYgKGlzUHJvbXB0SW5wdXRBY3RpdmUpIHJldHVybiB1bmRlZmluZWRcblxuICAgIGlmIChzYW5kYm94UGVybWlzc2lvblJlcXVlc3RRdWV1ZVswXSkgcmV0dXJuICdzYW5kYm94LXBlcm1pc3Npb24nXG5cbiAgICAvLyBQZXJtaXNzaW9uL2ludGVyYWN0aXZlIGRpYWxvZ3MgKHNob3cgdW5sZXNzIGJsb2NrZWQgYnkgdG9vbEpTWClcbiAgICBjb25zdCBhbGxvd0RpYWxvZ3NXaXRoQW5pbWF0aW9uID1cbiAgICAgICF0b29sSlNYIHx8IHRvb2xKU1guc2hvdWxkQ29udGludWVBbmltYXRpb25cblxuICAgIGlmIChhbGxvd0RpYWxvZ3NXaXRoQW5pbWF0aW9uICYmIHRvb2xVc2VDb25maXJtUXVldWVbMF0pXG4gICAgICByZXR1cm4gJ3Rvb2wtcGVybWlzc2lvbidcbiAgICBpZiAoYWxsb3dEaWFsb2dzV2l0aEFuaW1hdGlvbiAmJiBwcm9tcHRRdWV1ZVswXSkgcmV0dXJuICdwcm9tcHQnXG4gICAgLy8gV29ya2VyIHNhbmRib3ggcGVybWlzc2lvbiBwcm9tcHRzIChuZXR3b3JrIGFjY2VzcykgZnJvbSBzd2FybSB3b3JrZXJzXG4gICAgaWYgKGFsbG93RGlhbG9nc1dpdGhBbmltYXRpb24gJiYgd29ya2VyU2FuZGJveFBlcm1pc3Npb25zLnF1ZXVlWzBdKVxuICAgICAgcmV0dXJuICd3b3JrZXItc2FuZGJveC1wZXJtaXNzaW9uJ1xuICAgIGlmIChhbGxvd0RpYWxvZ3NXaXRoQW5pbWF0aW9uICYmIGVsaWNpdGF0aW9uLnF1ZXVlWzBdKSByZXR1cm4gJ2VsaWNpdGF0aW9uJ1xuICAgIGlmIChhbGxvd0RpYWxvZ3NXaXRoQW5pbWF0aW9uICYmIHNob3dpbmdDb3N0RGlhbG9nKSByZXR1cm4gJ2Nvc3QnXG4gICAgaWYgKGFsbG93RGlhbG9nc1dpdGhBbmltYXRpb24gJiYgaWRsZVJldHVyblBlbmRpbmcpIHJldHVybiAnaWRsZS1yZXR1cm4nXG5cbiAgICBpZiAoXG4gICAgICBmZWF0dXJlKCdVTFRSQVBMQU4nKSAmJlxuICAgICAgYWxsb3dEaWFsb2dzV2l0aEFuaW1hdGlvbiAmJlxuICAgICAgIWlzTG9hZGluZyAmJlxuICAgICAgdWx0cmFwbGFuUGVuZGluZ0Nob2ljZVxuICAgIClcbiAgICAgIHJldHVybiAndWx0cmFwbGFuLWNob2ljZSdcblxuICAgIGlmIChcbiAgICAgIGZlYXR1cmUoJ1VMVFJBUExBTicpICYmXG4gICAgICBhbGxvd0RpYWxvZ3NXaXRoQW5pbWF0aW9uICYmXG4gICAgICAhaXNMb2FkaW5nICYmXG4gICAgICB1bHRyYXBsYW5MYXVuY2hQZW5kaW5nXG4gICAgKVxuICAgICAgcmV0dXJuICd1bHRyYXBsYW4tbGF1bmNoJ1xuXG4gICAgLy8gT25ib2FyZGluZyBkaWFsb2dzIChzcGVjaWFsIGNvbmRpdGlvbnMpXG4gICAgaWYgKGFsbG93RGlhbG9nc1dpdGhBbmltYXRpb24gJiYgc2hvd0lkZU9uYm9hcmRpbmcpIHJldHVybiAnaWRlLW9uYm9hcmRpbmcnXG5cbiAgICAvLyBNb2RlbCBzd2l0Y2ggY2FsbG91dCAoYW50LW9ubHksIGVsaW1pbmF0ZWQgZnJvbSBleHRlcm5hbCBidWlsZHMpXG4gICAgaWYgKFxuICAgICAgXCJleHRlcm5hbFwiID09PSAnYW50JyAmJlxuICAgICAgYWxsb3dEaWFsb2dzV2l0aEFuaW1hdGlvbiAmJlxuICAgICAgc2hvd01vZGVsU3dpdGNoQ2FsbG91dFxuICAgIClcbiAgICAgIHJldHVybiAnbW9kZWwtc3dpdGNoJ1xuXG4gICAgLy8gVW5kZXJjb3ZlciBhdXRvLWVuYWJsZSBleHBsYWluZXIgKGFudC1vbmx5LCBlbGltaW5hdGVkIGZyb20gZXh0ZXJuYWwgYnVpbGRzKVxuICAgIGlmIChcbiAgICAgIFwiZXh0ZXJuYWxcIiA9PT0gJ2FudCcgJiZcbiAgICAgIGFsbG93RGlhbG9nc1dpdGhBbmltYXRpb24gJiZcbiAgICAgIHNob3dVbmRlcmNvdmVyQ2FsbG91dFxuICAgIClcbiAgICAgIHJldHVybiAndW5kZXJjb3Zlci1jYWxsb3V0J1xuXG4gICAgLy8gRWZmb3J0IGNhbGxvdXQgKHNob3duIG9uY2UgZm9yIE9wdXMgNC42IHVzZXJzIHdoZW4gZWZmb3J0IGlzIGVuYWJsZWQpXG4gICAgaWYgKGFsbG93RGlhbG9nc1dpdGhBbmltYXRpb24gJiYgc2hvd0VmZm9ydENhbGxvdXQpIHJldHVybiAnZWZmb3J0LWNhbGxvdXQnXG5cbiAgICAvLyBSZW1vdGUgY2FsbG91dCAoc2hvd24gb25jZSBiZWZvcmUgZmlyc3QgYnJpZGdlIGVuYWJsZSlcbiAgICBpZiAoYWxsb3dEaWFsb2dzV2l0aEFuaW1hdGlvbiAmJiBzaG93UmVtb3RlQ2FsbG91dCkgcmV0dXJuICdyZW1vdGUtY2FsbG91dCdcblxuICAgIC8vIExTUCBwbHVnaW4gcmVjb21tZW5kYXRpb24gKGxvd2VzdCBwcmlvcml0eSAtIG5vbi1ibG9ja2luZyBzdWdnZXN0aW9uKVxuICAgIGlmIChhbGxvd0RpYWxvZ3NXaXRoQW5pbWF0aW9uICYmIGxzcFJlY29tbWVuZGF0aW9uKVxuICAgICAgcmV0dXJuICdsc3AtcmVjb21tZW5kYXRpb24nXG5cbiAgICAvLyBQbHVnaW4gaGludCBmcm9tIENMSS9TREsgc3RkZXJyIChzYW1lIHByaW9yaXR5IGJhbmQgYXMgTFNQIHJlYylcbiAgICBpZiAoYWxsb3dEaWFsb2dzV2l0aEFuaW1hdGlvbiAmJiBoaW50UmVjb21tZW5kYXRpb24pIHJldHVybiAncGx1Z2luLWhpbnQnXG5cbiAgICAvLyBEZXNrdG9wIGFwcCB1cHNlbGwgKG1heCAzIGxhdW5jaGVzLCBsb3dlc3QgcHJpb3JpdHkpXG4gICAgaWYgKGFsbG93RGlhbG9nc1dpdGhBbmltYXRpb24gJiYgc2hvd0Rlc2t0b3BVcHNlbGxTdGFydHVwKVxuICAgICAgcmV0dXJuICdkZXNrdG9wLXVwc2VsbCdcblxuICAgIHJldHVybiB1bmRlZmluZWRcbiAgfVxuXG4gIGNvbnN0IGZvY3VzZWRJbnB1dERpYWxvZyA9IGdldEZvY3VzZWRJbnB1dERpYWxvZygpXG5cbiAgLy8gVHJ1ZSB3aGVuIHBlcm1pc3Npb24gcHJvbXB0cyBleGlzdCBidXQgYXJlIGhpZGRlbiBiZWNhdXNlIHRoZSB1c2VyIGlzIHR5cGluZ1xuICBjb25zdCBoYXNTdXBwcmVzc2VkRGlhbG9ncyA9XG4gICAgaXNQcm9tcHRJbnB1dEFjdGl2ZSAmJlxuICAgIChzYW5kYm94UGVybWlzc2lvblJlcXVlc3RRdWV1ZVswXSB8fFxuICAgICAgdG9vbFVzZUNvbmZpcm1RdWV1ZVswXSB8fFxuICAgICAgcHJvbXB0UXVldWVbMF0gfHxcbiAgICAgIHdvcmtlclNhbmRib3hQZXJtaXNzaW9ucy5xdWV1ZVswXSB8fFxuICAgICAgZWxpY2l0YXRpb24ucXVldWVbMF0gfHxcbiAgICAgIHNob3dpbmdDb3N0RGlhbG9nKVxuXG4gIC8vIEtlZXAgcmVmIGluIHN5bmMgc28gdGltZXIgY2FsbGJhY2tzIGNhbiByZWFkIHRoZSBjdXJyZW50IHZhbHVlXG4gIGZvY3VzZWRJbnB1dERpYWxvZ1JlZi5jdXJyZW50ID0gZm9jdXNlZElucHV0RGlhbG9nXG5cbiAgLy8gSW1tZWRpYXRlbHkgY2FwdHVyZSBwYXVzZS9yZXN1bWUgd2hlbiBmb2N1c2VkSW5wdXREaWFsb2cgY2hhbmdlc1xuICAvLyBUaGlzIGVuc3VyZXMgYWNjdXJhdGUgdGltaW5nIGV2ZW4gdW5kZXIgaGlnaCBzeXN0ZW0gbG9hZCwgcmF0aGVyIHRoYW5cbiAgLy8gcmVseWluZyBvbiB0aGUgMTAwbXMgcG9sbGluZyBpbnRlcnZhbCB0byBkZXRlY3Qgc3RhdGUgY2hhbmdlc1xuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmICghaXNMb2FkaW5nKSByZXR1cm5cblxuICAgIGNvbnN0IGlzUGF1c2VkID0gZm9jdXNlZElucHV0RGlhbG9nID09PSAndG9vbC1wZXJtaXNzaW9uJ1xuICAgIGNvbnN0IG5vdyA9IERhdGUubm93KClcblxuICAgIGlmIChpc1BhdXNlZCAmJiBwYXVzZVN0YXJ0VGltZVJlZi5jdXJyZW50ID09PSBudWxsKSB7XG4gICAgICAvLyBKdXN0IGVudGVyZWQgcGF1c2Ugc3RhdGUgLSByZWNvcmQgdGhlIGV4YWN0IG1vbWVudFxuICAgICAgcGF1c2VTdGFydFRpbWVSZWYuY3VycmVudCA9IG5vd1xuICAgIH0gZWxzZSBpZiAoIWlzUGF1c2VkICYmIHBhdXNlU3RhcnRUaW1lUmVmLmN1cnJlbnQgIT09IG51bGwpIHtcbiAgICAgIC8vIEp1c3QgZXhpdGVkIHBhdXNlIHN0YXRlIC0gYWNjdW11bGF0ZSBwYXVzZWQgdGltZSBpbW1lZGlhdGVseVxuICAgICAgdG90YWxQYXVzZWRNc1JlZi5jdXJyZW50ICs9IG5vdyAtIHBhdXNlU3RhcnRUaW1lUmVmLmN1cnJlbnRcbiAgICAgIHBhdXNlU3RhcnRUaW1lUmVmLmN1cnJlbnQgPSBudWxsXG4gICAgfVxuICB9LCBbZm9jdXNlZElucHV0RGlhbG9nLCBpc0xvYWRpbmddKVxuXG4gIC8vIFJlLXBpbiBzY3JvbGwgdG8gYm90dG9tIHdoZW5ldmVyIHRoZSBwZXJtaXNzaW9uIG92ZXJsYXkgYXBwZWFycyBvclxuICAvLyBkaXNtaXNzZXMuIE92ZXJsYXkgbm93IHJlbmRlcnMgYmVsb3cgbWVzc2FnZXMgaW5zaWRlIHRoZSBzYW1lXG4gIC8vIFNjcm9sbEJveCAobm8gcmVtb3VudCksIHNvIHdlIG5lZWQgYW4gZXhwbGljaXQgc2Nyb2xsVG9Cb3R0b20gZm9yOlxuICAvLyAgLSBhcHBlYXI6IHVzZXIgbWF5IGhhdmUgYmVlbiBzY3JvbGxlZCB1cCAoc3RpY2t5IGJyb2tlbikg4oCUIHRoZVxuICAvLyAgICBkaWFsb2cgaXMgYmxvY2tpbmcgYW5kIG11c3QgYmUgdmlzaWJsZVxuICAvLyAgLSBkaXNtaXNzOiB1c2VyIG1heSBoYXZlIHNjcm9sbGVkIHVwIHRvIHJlYWQgY29udGV4dCBkdXJpbmcgdGhlXG4gIC8vICAgIG92ZXJsYXksIGFuZCBvblNjcm9sbCB3YXMgc3VwcHJlc3NlZCBzbyB0aGUgcGlsbCBzdGF0ZSBpcyBzdGFsZVxuICAvLyB1c2VMYXlvdXRFZmZlY3Qgc28gdGhlIHJlLXBpbiBjb21taXRzIGJlZm9yZSB0aGUgSW5rIGZyYW1lIHJlbmRlcnMg4oCUXG4gIC8vIG5vIDEtZnJhbWUgZmxhc2ggb2YgdGhlIHdyb25nIHNjcm9sbCBwb3NpdGlvbi5cbiAgY29uc3QgcHJldkRpYWxvZ1JlZiA9IHVzZVJlZihmb2N1c2VkSW5wdXREaWFsb2cpXG4gIHVzZUxheW91dEVmZmVjdCgoKSA9PiB7XG4gICAgY29uc3Qgd2FzID0gcHJldkRpYWxvZ1JlZi5jdXJyZW50ID09PSAndG9vbC1wZXJtaXNzaW9uJ1xuICAgIGNvbnN0IG5vdyA9IGZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ3Rvb2wtcGVybWlzc2lvbidcbiAgICBpZiAod2FzICE9PSBub3cpIHJlcGluU2Nyb2xsKClcbiAgICBwcmV2RGlhbG9nUmVmLmN1cnJlbnQgPSBmb2N1c2VkSW5wdXREaWFsb2dcbiAgfSwgW2ZvY3VzZWRJbnB1dERpYWxvZywgcmVwaW5TY3JvbGxdKVxuXG4gIGZ1bmN0aW9uIG9uQ2FuY2VsKCkge1xuICAgIGlmIChmb2N1c2VkSW5wdXREaWFsb2cgPT09ICdlbGljaXRhdGlvbicpIHtcbiAgICAgIC8vIEVsaWNpdGF0aW9uIGRpYWxvZyBoYW5kbGVzIGl0cyBvd24gRXNjYXBlLCBhbmQgY2xvc2luZyBpdCBzaG91bGRuJ3QgYWZmZWN0IGFueSBsb2FkaW5nIHN0YXRlLlxuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgYFtvbkNhbmNlbF0gZm9jdXNlZElucHV0RGlhbG9nPSR7Zm9jdXNlZElucHV0RGlhbG9nfSBzdHJlYW1Nb2RlPSR7c3RyZWFtTW9kZX1gLFxuICAgIClcblxuICAgIC8vIFBhdXNlIHByb2FjdGl2ZSBtb2RlIHNvIHRoZSB1c2VyIGdldHMgY29udHJvbCBiYWNrLlxuICAgIC8vIEl0IHdpbGwgcmVzdW1lIHdoZW4gdGhleSBzdWJtaXQgdGhlaXIgbmV4dCBpbnB1dCAoc2VlIG9uU3VibWl0KS5cbiAgICBpZiAoZmVhdHVyZSgnUFJPQUNUSVZFJykgfHwgZmVhdHVyZSgnS0FJUk9TJykpIHtcbiAgICAgIHByb2FjdGl2ZU1vZHVsZT8ucGF1c2VQcm9hY3RpdmUoKVxuICAgIH1cblxuICAgIHF1ZXJ5R3VhcmQuZm9yY2VFbmQoKVxuICAgIHNraXBJZGxlQ2hlY2tSZWYuY3VycmVudCA9IGZhbHNlXG5cbiAgICAvLyBQcmVzZXJ2ZSBwYXJ0aWFsbHktc3RyZWFtZWQgdGV4dCBzbyB0aGUgdXNlciBjYW4gcmVhZCB3aGF0IHdhc1xuICAgIC8vIGdlbmVyYXRlZCBiZWZvcmUgcHJlc3NpbmcgRXNjLiBQdXNoZWQgYmVmb3JlIHJlc2V0TG9hZGluZ1N0YXRlIGNsZWFyc1xuICAgIC8vIHN0cmVhbWluZ1RleHQsIGFuZCBiZWZvcmUgcXVlcnkudHMgeWllbGRzIHRoZSBhc3luYyBpbnRlcnJ1cHQgbWFya2VyLFxuICAgIC8vIGdpdmluZyBmaW5hbCBvcmRlciBbdXNlciwgcGFydGlhbC1hc3Npc3RhbnQsIFtSZXF1ZXN0IGludGVycnVwdGVkIGJ5IHVzZXJdXS5cbiAgICBpZiAoc3RyZWFtaW5nVGV4dD8udHJpbSgpKSB7XG4gICAgICBzZXRNZXNzYWdlcyhwcmV2ID0+IFtcbiAgICAgICAgLi4ucHJldixcbiAgICAgICAgY3JlYXRlQXNzaXN0YW50TWVzc2FnZSh7IGNvbnRlbnQ6IHN0cmVhbWluZ1RleHQgfSksXG4gICAgICBdKVxuICAgIH1cblxuICAgIHJlc2V0TG9hZGluZ1N0YXRlKClcblxuICAgIC8vIENsZWFyIGFueSBhY3RpdmUgdG9rZW4gYnVkZ2V0IHNvIHRoZSBiYWNrc3RvcCBkb2Vzbid0IGZpcmUgb25cbiAgICAvLyBhIHN0YWxlIGJ1ZGdldCBpZiB0aGUgcXVlcnkgZ2VuZXJhdG9yIGhhc24ndCBleGl0ZWQgeWV0LlxuICAgIGlmIChmZWF0dXJlKCdUT0tFTl9CVURHRVQnKSkge1xuICAgICAgc25hcHNob3RPdXRwdXRUb2tlbnNGb3JUdXJuKG51bGwpXG4gICAgfVxuXG4gICAgaWYgKGZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ3Rvb2wtcGVybWlzc2lvbicpIHtcbiAgICAgIC8vIFRvb2wgdXNlIGNvbmZpcm0gaGFuZGxlcyB0aGUgYWJvcnQgc2lnbmFsIGl0c2VsZlxuICAgICAgdG9vbFVzZUNvbmZpcm1RdWV1ZVswXT8ub25BYm9ydCgpXG4gICAgICBzZXRUb29sVXNlQ29uZmlybVF1ZXVlKFtdKVxuICAgIH0gZWxzZSBpZiAoZm9jdXNlZElucHV0RGlhbG9nID09PSAncHJvbXB0Jykge1xuICAgICAgLy8gUmVqZWN0IGFsbCBwZW5kaW5nIHByb21wdHMgYW5kIGNsZWFyIHRoZSBxdWV1ZVxuICAgICAgZm9yIChjb25zdCBpdGVtIG9mIHByb21wdFF1ZXVlKSB7XG4gICAgICAgIGl0ZW0ucmVqZWN0KG5ldyBFcnJvcignUHJvbXB0IGNhbmNlbGxlZCBieSB1c2VyJykpXG4gICAgICB9XG4gICAgICBzZXRQcm9tcHRRdWV1ZShbXSlcbiAgICAgIGFib3J0Q29udHJvbGxlcj8uYWJvcnQoJ3VzZXItY2FuY2VsJylcbiAgICB9IGVsc2UgaWYgKGFjdGl2ZVJlbW90ZS5pc1JlbW90ZU1vZGUpIHtcbiAgICAgIC8vIFJlbW90ZSBtb2RlOiBzZW5kIGludGVycnVwdCBzaWduYWwgdG8gQ0NSXG4gICAgICBhY3RpdmVSZW1vdGUuY2FuY2VsUmVxdWVzdCgpXG4gICAgfSBlbHNlIHtcbiAgICAgIGFib3J0Q29udHJvbGxlcj8uYWJvcnQoJ3VzZXItY2FuY2VsJylcbiAgICB9XG5cbiAgICAvLyBDbGVhciB0aGUgY29udHJvbGxlciBzbyBzdWJzZXF1ZW50IEVzY2FwZSBwcmVzc2VzIGRvbid0IHNlZSBhIHN0YWxlXG4gICAgLy8gYWJvcnRlZCBzaWduYWwuIFdpdGhvdXQgdGhpcywgY2FuQ2FuY2VsUnVubmluZ1Rhc2sgaXMgZmFsc2UgKHNpZ25hbFxuICAgIC8vIGRlZmluZWQgYnV0IC5hYm9ydGVkID09PSB0cnVlKSwgc28gaXNBY3RpdmUgYmVjb21lcyBmYWxzZSBpZiBubyBvdGhlclxuICAgIC8vIGFjdGl2YXRpbmcgY29uZGl0aW9ucyBob2xkIOKAlCBsZWF2aW5nIHRoZSBFc2NhcGUga2V5YmluZGluZyBpbmFjdGl2ZS5cbiAgICBzZXRBYm9ydENvbnRyb2xsZXIobnVsbClcblxuICAgIC8vIGZvcmNlRW5kKCkgc2tpcHMgdGhlIGZpbmFsbHkgcGF0aCDigJQgZmlyZSBkaXJlY3RseSAoYWJvcnRlZD10cnVlKS5cbiAgICB2b2lkIG1yT25UdXJuQ29tcGxldGUobWVzc2FnZXNSZWYuY3VycmVudCwgdHJ1ZSlcbiAgfVxuXG4gIC8vIEZ1bmN0aW9uIHRvIGhhbmRsZSBxdWV1ZWQgY29tbWFuZCB3aGVuIGNhbmNlbGluZyBhIHBlcm1pc3Npb24gcmVxdWVzdFxuICBjb25zdCBoYW5kbGVRdWV1ZWRDb21tYW5kT25DYW5jZWwgPSB1c2VDYWxsYmFjaygoKSA9PiB7XG4gICAgY29uc3QgcmVzdWx0ID0gcG9wQWxsRWRpdGFibGUoaW5wdXRWYWx1ZSwgMClcbiAgICBpZiAoIXJlc3VsdCkgcmV0dXJuXG4gICAgc2V0SW5wdXRWYWx1ZShyZXN1bHQudGV4dClcbiAgICBzZXRJbnB1dE1vZGUoJ3Byb21wdCcpXG5cbiAgICAvLyBSZXN0b3JlIGltYWdlcyBmcm9tIHF1ZXVlZCBjb21tYW5kcyB0byBwYXN0ZWRDb250ZW50c1xuICAgIGlmIChyZXN1bHQuaW1hZ2VzLmxlbmd0aCA+IDApIHtcbiAgICAgIHNldFBhc3RlZENvbnRlbnRzKHByZXYgPT4ge1xuICAgICAgICBjb25zdCBuZXdDb250ZW50cyA9IHsgLi4ucHJldiB9XG4gICAgICAgIGZvciAoY29uc3QgaW1hZ2Ugb2YgcmVzdWx0LmltYWdlcykge1xuICAgICAgICAgIG5ld0NvbnRlbnRzW2ltYWdlLmlkXSA9IGltYWdlXG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIG5ld0NvbnRlbnRzXG4gICAgICB9KVxuICAgIH1cbiAgfSwgW3NldElucHV0VmFsdWUsIHNldElucHV0TW9kZSwgaW5wdXRWYWx1ZSwgc2V0UGFzdGVkQ29udGVudHNdKVxuXG4gIC8vIENhbmNlbFJlcXVlc3RIYW5kbGVyIHByb3BzIC0gcmVuZGVyZWQgaW5zaWRlIEtleWJpbmRpbmdTZXR1cFxuICBjb25zdCBjYW5jZWxSZXF1ZXN0UHJvcHMgPSB7XG4gICAgc2V0VG9vbFVzZUNvbmZpcm1RdWV1ZSxcbiAgICBvbkNhbmNlbCxcbiAgICBvbkFnZW50c0tpbGxlZDogKCkgPT5cbiAgICAgIHNldE1lc3NhZ2VzKHByZXYgPT4gWy4uLnByZXYsIGNyZWF0ZUFnZW50c0tpbGxlZE1lc3NhZ2UoKV0pLFxuICAgIGlzTWVzc2FnZVNlbGVjdG9yVmlzaWJsZTogaXNNZXNzYWdlU2VsZWN0b3JWaXNpYmxlIHx8ICEhc2hvd0Jhc2hlc0RpYWxvZyxcbiAgICBzY3JlZW4sXG4gICAgYWJvcnRTaWduYWw6IGFib3J0Q29udHJvbGxlcj8uc2lnbmFsLFxuICAgIHBvcENvbW1hbmRGcm9tUXVldWU6IGhhbmRsZVF1ZXVlZENvbW1hbmRPbkNhbmNlbCxcbiAgICB2aW1Nb2RlLFxuICAgIGlzTG9jYWxKU1hDb21tYW5kOiB0b29sSlNYPy5pc0xvY2FsSlNYQ29tbWFuZCxcbiAgICBpc1NlYXJjaGluZ0hpc3RvcnksXG4gICAgaXNIZWxwT3BlbixcbiAgICBpbnB1dE1vZGUsXG4gICAgaW5wdXRWYWx1ZSxcbiAgICBzdHJlYW1Nb2RlLFxuICB9XG5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBjb25zdCB0b3RhbENvc3QgPSBnZXRUb3RhbENvc3QoKVxuICAgIGlmICh0b3RhbENvc3QgPj0gNSAvKiAkNSAqLyAmJiAhc2hvd0Nvc3REaWFsb2cgJiYgIWhhdmVTaG93bkNvc3REaWFsb2cpIHtcbiAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9jb3N0X3RocmVzaG9sZF9yZWFjaGVkJywge30pXG4gICAgICAvLyBNYXJrIGFzIHNob3duIGV2ZW4gaWYgdGhlIGRpYWxvZyB3b24ndCByZW5kZXIgKG5vIGNvbnNvbGUgYmlsbGluZ1xuICAgICAgLy8gYWNjZXNzKS4gT3RoZXJ3aXNlIHRoaXMgZWZmZWN0IHJlLWZpcmVzIG9uIGV2ZXJ5IG1lc3NhZ2UgY2hhbmdlIGZvclxuICAgICAgLy8gdGhlIHJlc3Qgb2YgdGhlIHNlc3Npb24g4oCUIDIwMGsrIHNwdXJpb3VzIGV2ZW50cyBvYnNlcnZlZC5cbiAgICAgIHNldEhhdmVTaG93bkNvc3REaWFsb2codHJ1ZSlcbiAgICAgIGlmIChoYXNDb25zb2xlQmlsbGluZ0FjY2VzcygpKSB7XG4gICAgICAgIHNldFNob3dDb3N0RGlhbG9nKHRydWUpXG4gICAgICB9XG4gICAgfVxuICB9LCBbbWVzc2FnZXMsIHNob3dDb3N0RGlhbG9nLCBoYXZlU2hvd25Db3N0RGlhbG9nXSlcblxuICBjb25zdCBzYW5kYm94QXNrQ2FsbGJhY2s6IFNhbmRib3hBc2tDYWxsYmFjayA9IHVzZUNhbGxiYWNrKFxuICAgIGFzeW5jIChob3N0UGF0dGVybjogTmV0d29ya0hvc3RQYXR0ZXJuKSA9PiB7XG4gICAgICAvLyBJZiBydW5uaW5nIGFzIGEgc3dhcm0gd29ya2VyLCBmb3J3YXJkIHRoZSByZXF1ZXN0IHRvIHRoZSBsZWFkZXIgdmlhIG1haWxib3hcbiAgICAgIGlmIChpc0FnZW50U3dhcm1zRW5hYmxlZCgpICYmIGlzU3dhcm1Xb3JrZXIoKSkge1xuICAgICAgICBjb25zdCByZXF1ZXN0SWQgPSBnZW5lcmF0ZVNhbmRib3hSZXF1ZXN0SWQoKVxuXG4gICAgICAgIC8vIFNlbmQgdGhlIHJlcXVlc3QgdG8gdGhlIGxlYWRlciB2aWEgbWFpbGJveFxuICAgICAgICBjb25zdCBzZW50ID0gYXdhaXQgc2VuZFNhbmRib3hQZXJtaXNzaW9uUmVxdWVzdFZpYU1haWxib3goXG4gICAgICAgICAgaG9zdFBhdHRlcm4uaG9zdCxcbiAgICAgICAgICByZXF1ZXN0SWQsXG4gICAgICAgIClcblxuICAgICAgICByZXR1cm4gbmV3IFByb21pc2UocmVzb2x2ZVNob3VsZEFsbG93SG9zdCA9PiB7XG4gICAgICAgICAgaWYgKCFzZW50KSB7XG4gICAgICAgICAgICAvLyBJZiB3ZSBjb3VsZG4ndCBzZW5kIHZpYSBtYWlsYm94LCBmYWxsIGJhY2sgdG8gbG9jYWwgaGFuZGxpbmdcbiAgICAgICAgICAgIHNldFNhbmRib3hQZXJtaXNzaW9uUmVxdWVzdFF1ZXVlKHByZXYgPT4gW1xuICAgICAgICAgICAgICAuLi5wcmV2LFxuICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgaG9zdFBhdHRlcm4sXG4gICAgICAgICAgICAgICAgcmVzb2x2ZVByb21pc2U6IHJlc29sdmVTaG91bGRBbGxvd0hvc3QsXG4gICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICBdKVxuICAgICAgICAgICAgcmV0dXJuXG4gICAgICAgICAgfVxuXG4gICAgICAgICAgLy8gUmVnaXN0ZXIgdGhlIGNhbGxiYWNrIGZvciB3aGVuIHRoZSBsZWFkZXIgcmVzcG9uZHNcbiAgICAgICAgICByZWdpc3RlclNhbmRib3hQZXJtaXNzaW9uQ2FsbGJhY2soe1xuICAgICAgICAgICAgcmVxdWVzdElkLFxuICAgICAgICAgICAgaG9zdDogaG9zdFBhdHRlcm4uaG9zdCxcbiAgICAgICAgICAgIHJlc29sdmU6IHJlc29sdmVTaG91bGRBbGxvd0hvc3QsXG4gICAgICAgICAgfSlcblxuICAgICAgICAgIC8vIFVwZGF0ZSBBcHBTdGF0ZSB0byBzaG93IHBlbmRpbmcgaW5kaWNhdG9yXG4gICAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiAoe1xuICAgICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICAgIHBlbmRpbmdTYW5kYm94UmVxdWVzdDoge1xuICAgICAgICAgICAgICByZXF1ZXN0SWQsXG4gICAgICAgICAgICAgIGhvc3Q6IGhvc3RQYXR0ZXJuLmhvc3QsXG4gICAgICAgICAgICB9LFxuICAgICAgICAgIH0pKVxuICAgICAgICB9KVxuICAgICAgfVxuXG4gICAgICAvLyBOb3JtYWwgZmxvdyBmb3Igbm9uLXdvcmtlcnM6IHNob3cgbG9jYWwgVUkgYW5kIG9wdGlvbmFsbHkgcmFjZVxuICAgICAgLy8gYWdhaW5zdCB0aGUgUkVQTCBicmlkZ2UgKFJlbW90ZSBDb250cm9sKSBpZiBjb25uZWN0ZWQuXG4gICAgICByZXR1cm4gbmV3IFByb21pc2UocmVzb2x2ZVNob3VsZEFsbG93SG9zdCA9PiB7XG4gICAgICAgIGxldCByZXNvbHZlZCA9IGZhbHNlXG4gICAgICAgIGZ1bmN0aW9uIHJlc29sdmVPbmNlKGFsbG93OiBib29sZWFuKTogdm9pZCB7XG4gICAgICAgICAgaWYgKHJlc29sdmVkKSByZXR1cm5cbiAgICAgICAgICByZXNvbHZlZCA9IHRydWVcbiAgICAgICAgICByZXNvbHZlU2hvdWxkQWxsb3dIb3N0KGFsbG93KVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gUXVldWUgdGhlIGxvY2FsIHNhbmRib3ggcGVybWlzc2lvbiBkaWFsb2dcbiAgICAgICAgc2V0U2FuZGJveFBlcm1pc3Npb25SZXF1ZXN0UXVldWUocHJldiA9PiBbXG4gICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICB7XG4gICAgICAgICAgICBob3N0UGF0dGVybixcbiAgICAgICAgICAgIHJlc29sdmVQcm9taXNlOiByZXNvbHZlT25jZSxcbiAgICAgICAgICB9LFxuICAgICAgICBdKVxuXG4gICAgICAgIC8vIFdoZW4gdGhlIFJFUEwgYnJpZGdlIGlzIGNvbm5lY3RlZCwgYWxzbyBmb3J3YXJkIHRoZSBzYW5kYm94XG4gICAgICAgIC8vIHBlcm1pc3Npb24gcmVxdWVzdCBhcyBhIGNhbl91c2VfdG9vbCBjb250cm9sX3JlcXVlc3Qgc28gdGhlXG4gICAgICAgIC8vIHJlbW90ZSB1c2VyIChlLmcuIG9uIGNsYXVkZS5haSkgY2FuIGFwcHJvdmUgaXQgdG9vLlxuICAgICAgICBpZiAoZmVhdHVyZSgnQlJJREdFX01PREUnKSkge1xuICAgICAgICAgIGNvbnN0IGJyaWRnZUNhbGxiYWNrcyA9IHN0b3JlLmdldFN0YXRlKCkucmVwbEJyaWRnZVBlcm1pc3Npb25DYWxsYmFja3NcbiAgICAgICAgICBpZiAoYnJpZGdlQ2FsbGJhY2tzKSB7XG4gICAgICAgICAgICBjb25zdCBicmlkZ2VSZXF1ZXN0SWQgPSByYW5kb21VVUlEKClcbiAgICAgICAgICAgIGJyaWRnZUNhbGxiYWNrcy5zZW5kUmVxdWVzdChcbiAgICAgICAgICAgICAgYnJpZGdlUmVxdWVzdElkLFxuICAgICAgICAgICAgICBTQU5EQk9YX05FVFdPUktfQUNDRVNTX1RPT0xfTkFNRSxcbiAgICAgICAgICAgICAgeyBob3N0OiBob3N0UGF0dGVybi5ob3N0IH0sXG4gICAgICAgICAgICAgIHJhbmRvbVVVSUQoKSxcbiAgICAgICAgICAgICAgYEFsbG93IG5ldHdvcmsgY29ubmVjdGlvbiB0byAke2hvc3RQYXR0ZXJuLmhvc3R9P2AsXG4gICAgICAgICAgICApXG5cbiAgICAgICAgICAgIGNvbnN0IHVuc3Vic2NyaWJlID0gYnJpZGdlQ2FsbGJhY2tzLm9uUmVzcG9uc2UoXG4gICAgICAgICAgICAgIGJyaWRnZVJlcXVlc3RJZCxcbiAgICAgICAgICAgICAgcmVzcG9uc2UgPT4ge1xuICAgICAgICAgICAgICAgIHVuc3Vic2NyaWJlKClcbiAgICAgICAgICAgICAgICBjb25zdCBhbGxvdyA9IHJlc3BvbnNlLmJlaGF2aW9yID09PSAnYWxsb3cnXG4gICAgICAgICAgICAgICAgLy8gUmVzb2x2ZSBBTEwgcGVuZGluZyByZXF1ZXN0cyBmb3IgdGhlIHNhbWUgaG9zdCwgbm90IGp1c3RcbiAgICAgICAgICAgICAgICAvLyB0aGlzIG9uZSDigJQgbWlycm9ycyB0aGUgbG9jYWwgZGlhbG9nIGhhbmRsZXIgcGF0dGVybi5cbiAgICAgICAgICAgICAgICBzZXRTYW5kYm94UGVybWlzc2lvblJlcXVlc3RRdWV1ZShxdWV1ZSA9PiB7XG4gICAgICAgICAgICAgICAgICBxdWV1ZVxuICAgICAgICAgICAgICAgICAgICAuZmlsdGVyKGl0ZW0gPT4gaXRlbS5ob3N0UGF0dGVybi5ob3N0ID09PSBob3N0UGF0dGVybi5ob3N0KVxuICAgICAgICAgICAgICAgICAgICAuZm9yRWFjaChpdGVtID0+IGl0ZW0ucmVzb2x2ZVByb21pc2UoYWxsb3cpKVxuICAgICAgICAgICAgICAgICAgcmV0dXJuIHF1ZXVlLmZpbHRlcihcbiAgICAgICAgICAgICAgICAgICAgaXRlbSA9PiBpdGVtLmhvc3RQYXR0ZXJuLmhvc3QgIT09IGhvc3RQYXR0ZXJuLmhvc3QsXG4gICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgICAvLyBDbGVhbiB1cCBhbGwgc2libGluZyBicmlkZ2Ugc3Vic2NyaXB0aW9ucyBmb3IgdGhpcyBob3N0XG4gICAgICAgICAgICAgICAgLy8gKG90aGVyIGNvbmN1cnJlbnQgc2FtZS1ob3N0IHJlcXVlc3RzKSBiZWZvcmUgZGVsZXRpbmcuXG4gICAgICAgICAgICAgICAgY29uc3Qgc2libGluZ0NsZWFudXBzID0gc2FuZGJveEJyaWRnZUNsZWFudXBSZWYuY3VycmVudC5nZXQoXG4gICAgICAgICAgICAgICAgICBob3N0UGF0dGVybi5ob3N0LFxuICAgICAgICAgICAgICAgIClcbiAgICAgICAgICAgICAgICBpZiAoc2libGluZ0NsZWFudXBzKSB7XG4gICAgICAgICAgICAgICAgICBmb3IgKGNvbnN0IGZuIG9mIHNpYmxpbmdDbGVhbnVwcykge1xuICAgICAgICAgICAgICAgICAgICBmbigpXG4gICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICBzYW5kYm94QnJpZGdlQ2xlYW51cFJlZi5jdXJyZW50LmRlbGV0ZShob3N0UGF0dGVybi5ob3N0KVxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIClcblxuICAgICAgICAgICAgLy8gUmVnaXN0ZXIgY2xlYW51cCBzbyB0aGUgbG9jYWwgZGlhbG9nIGhhbmRsZXIgY2FuIGNhbmNlbFxuICAgICAgICAgICAgLy8gdGhlIHJlbW90ZSBwcm9tcHQgYW5kIHVuc3Vic2NyaWJlIHdoZW4gdGhlIGxvY2FsIHVzZXJcbiAgICAgICAgICAgIC8vIHJlc3BvbmRzIGZpcnN0LlxuICAgICAgICAgICAgY29uc3QgY2xlYW51cCA9ICgpID0+IHtcbiAgICAgICAgICAgICAgdW5zdWJzY3JpYmUoKVxuICAgICAgICAgICAgICBicmlkZ2VDYWxsYmFja3MuY2FuY2VsUmVxdWVzdChicmlkZ2VSZXF1ZXN0SWQpXG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBjb25zdCBleGlzdGluZyA9XG4gICAgICAgICAgICAgIHNhbmRib3hCcmlkZ2VDbGVhbnVwUmVmLmN1cnJlbnQuZ2V0KGhvc3RQYXR0ZXJuLmhvc3QpID8/IFtdXG4gICAgICAgICAgICBleGlzdGluZy5wdXNoKGNsZWFudXApXG4gICAgICAgICAgICBzYW5kYm94QnJpZGdlQ2xlYW51cFJlZi5jdXJyZW50LnNldChob3N0UGF0dGVybi5ob3N0LCBleGlzdGluZylcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH0pXG4gICAgfSxcbiAgICBbc2V0QXBwU3RhdGUsIHN0b3JlXSxcbiAgKVxuXG4gIC8vICMzNDA0NDogaWYgdXNlciBleHBsaWNpdGx5IHNldCBzYW5kYm94LmVuYWJsZWQ9dHJ1ZSBidXQgZGVwcyBhcmUgbWlzc2luZyxcbiAgLy8gaXNTYW5kYm94aW5nRW5hYmxlZCgpIHJldHVybnMgZmFsc2Ugc2lsZW50bHkuIFN1cmZhY2UgdGhlIHJlYXNvbiBvbmNlIGF0XG4gIC8vIG1vdW50IHNvIHVzZXJzIGtub3cgdGhlaXIgc2VjdXJpdHkgY29uZmlnIGlzbid0IGJlaW5nIGVuZm9yY2VkLiBGdWxsXG4gIC8vIHJlYXNvbiBnb2VzIHRvIGRlYnVnIGxvZzsgbm90aWZpY2F0aW9uIHBvaW50cyB0byAvc2FuZGJveCBmb3IgZGV0YWlscy5cbiAgLy8gYWRkTm90aWZpY2F0aW9uIGlzIHN0YWJsZSAodXNlQ2FsbGJhY2spIHNvIHRoZSBlZmZlY3QgZmlyZXMgb25jZS5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBjb25zdCByZWFzb24gPSBTYW5kYm94TWFuYWdlci5nZXRTYW5kYm94VW5hdmFpbGFibGVSZWFzb24oKVxuICAgIGlmICghcmVhc29uKSByZXR1cm5cbiAgICBpZiAoU2FuZGJveE1hbmFnZXIuaXNTYW5kYm94UmVxdWlyZWQoKSkge1xuICAgICAgcHJvY2Vzcy5zdGRlcnIud3JpdGUoXG4gICAgICAgIGBcXG5FcnJvcjogc2FuZGJveCByZXF1aXJlZCBidXQgdW5hdmFpbGFibGU6ICR7cmVhc29ufVxcbmAgK1xuICAgICAgICAgIGAgIHNhbmRib3guZmFpbElmVW5hdmFpbGFibGUgaXMgc2V0IOKAlCByZWZ1c2luZyB0byBzdGFydCB3aXRob3V0IGEgd29ya2luZyBzYW5kYm94LlxcblxcbmAsXG4gICAgICApXG4gICAgICBncmFjZWZ1bFNodXRkb3duU3luYygxLCAnb3RoZXInKVxuICAgICAgcmV0dXJuXG4gICAgfVxuICAgIGxvZ0ZvckRlYnVnZ2luZyhgc2FuZGJveCBkaXNhYmxlZDogJHtyZWFzb259YCwgeyBsZXZlbDogJ3dhcm4nIH0pXG4gICAgYWRkTm90aWZpY2F0aW9uKHtcbiAgICAgIGtleTogJ3NhbmRib3gtdW5hdmFpbGFibGUnLFxuICAgICAganN4OiAoXG4gICAgICAgIDw+XG4gICAgICAgICAgPFRleHQgY29sb3I9XCJ3YXJuaW5nXCI+c2FuZGJveCBkaXNhYmxlZDwvVGV4dD5cbiAgICAgICAgICA8VGV4dCBkaW1Db2xvcj4gwrcgL3NhbmRib3g8L1RleHQ+XG4gICAgICAgIDwvPlxuICAgICAgKSxcbiAgICAgIHByaW9yaXR5OiAnbWVkaXVtJyxcbiAgICB9KVxuICB9LCBbYWRkTm90aWZpY2F0aW9uXSlcblxuICBpZiAoU2FuZGJveE1hbmFnZXIuaXNTYW5kYm94aW5nRW5hYmxlZCgpKSB7XG4gICAgLy8gSWYgc2FuZGJveGluZyBpcyBlbmFibGVkIChzZXR0aW5nLnNhbmRib3ggaXMgZGVmaW5lZCwgaW5pdGlhbGlzZSB0aGUgbWFuYWdlcilcbiAgICBTYW5kYm94TWFuYWdlci5pbml0aWFsaXplKHNhbmRib3hBc2tDYWxsYmFjaykuY2F0Y2goZXJyID0+IHtcbiAgICAgIC8vIEluaXRpYWxpemF0aW9uL3ZhbGlkYXRpb24gZmFpbGVkIC0gZGlzcGxheSBlcnJvciBhbmQgZXhpdFxuICAgICAgcHJvY2Vzcy5zdGRlcnIud3JpdGUoYFxcbuKdjCBTYW5kYm94IEVycm9yOiAke2Vycm9yTWVzc2FnZShlcnIpfVxcbmApXG4gICAgICBncmFjZWZ1bFNodXRkb3duU3luYygxLCAnb3RoZXInKVxuICAgIH0pXG4gIH1cblxuICBjb25zdCBzZXRUb29sUGVybWlzc2lvbkNvbnRleHQgPSB1c2VDYWxsYmFjayhcbiAgICAoY29udGV4dDogVG9vbFBlcm1pc3Npb25Db250ZXh0LCBvcHRpb25zPzogeyBwcmVzZXJ2ZU1vZGU/OiBib29sZWFuIH0pID0+IHtcbiAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4gKHtcbiAgICAgICAgLi4ucHJldixcbiAgICAgICAgdG9vbFBlcm1pc3Npb25Db250ZXh0OiB7XG4gICAgICAgICAgLi4uY29udGV4dCxcbiAgICAgICAgICAvLyBQcmVzZXJ2ZSB0aGUgY29vcmRpbmF0b3IncyBtb2RlIG9ubHkgd2hlbiBleHBsaWNpdGx5IHJlcXVlc3RlZC5cbiAgICAgICAgICAvLyBXb3JrZXJzJyBnZXRBcHBTdGF0ZSgpIHJldHVybnMgYSB0cmFuc2Zvcm1lZCBjb250ZXh0IHdpdGggbW9kZVxuICAgICAgICAgIC8vICdhY2NlcHRFZGl0cycgdGhhdCBtdXN0IG5vdCBsZWFrIGludG8gdGhlIGNvb3JkaW5hdG9yJ3MgYWN0dWFsXG4gICAgICAgICAgLy8gc3RhdGUgdmlhIHBlcm1pc3Npb24tcnVsZSB1cGRhdGVzIOKAlCB0aG9zZSBjYWxsIHNpdGVzIHBhc3NcbiAgICAgICAgICAvLyB7IHByZXNlcnZlTW9kZTogdHJ1ZSB9LiBVc2VyLWluaXRpYXRlZCBtb2RlIGNoYW5nZXMgKGUuZy4sXG4gICAgICAgICAgLy8gc2VsZWN0aW5nIFwiYWxsb3cgYWxsIGVkaXRzXCIpIG11c3QgTk9UIGJlIG92ZXJyaWRkZW4uXG4gICAgICAgICAgbW9kZTogb3B0aW9ucz8ucHJlc2VydmVNb2RlXG4gICAgICAgICAgICA/IHByZXYudG9vbFBlcm1pc3Npb25Db250ZXh0Lm1vZGVcbiAgICAgICAgICAgIDogY29udGV4dC5tb2RlLFxuICAgICAgICB9LFxuICAgICAgfSkpXG5cbiAgICAgIC8vIFdoZW4gcGVybWlzc2lvbiBjb250ZXh0IGNoYW5nZXMsIHJlY2hlY2sgYWxsIHF1ZXVlZCBpdGVtc1xuICAgICAgLy8gVGhpcyBoYW5kbGVzIHRoZSBjYXNlIHdoZXJlIGFwcHJvdmluZyBpdGVtMSB3aXRoIFwiZG9uJ3QgYXNrIGFnYWluXCJcbiAgICAgIC8vIHNob3VsZCBhdXRvLWFwcHJvdmUgb3RoZXIgcXVldWVkIGl0ZW1zIHRoYXQgbm93IG1hdGNoIHRoZSB1cGRhdGVkIHJ1bGVzXG4gICAgICBzZXRJbW1lZGlhdGUoc2V0VG9vbFVzZUNvbmZpcm1RdWV1ZSA9PiB7XG4gICAgICAgIC8vIFVzZSBzZXRUb29sVXNlQ29uZmlybVF1ZXVlIGNhbGxiYWNrIHRvIGdldCBjdXJyZW50IHF1ZXVlIHN0YXRlXG4gICAgICAgIC8vIGluc3RlYWQgb2YgY2FwdHVyaW5nIGl0IGluIHRoZSBjbG9zdXJlLCB0byBhdm9pZCBzdGFsZSBjbG9zdXJlIGlzc3Vlc1xuICAgICAgICBzZXRUb29sVXNlQ29uZmlybVF1ZXVlKGN1cnJlbnRRdWV1ZSA9PiB7XG4gICAgICAgICAgY3VycmVudFF1ZXVlLmZvckVhY2goaXRlbSA9PiB7XG4gICAgICAgICAgICB2b2lkIGl0ZW0ucmVjaGVja1Blcm1pc3Npb24oKVxuICAgICAgICAgIH0pXG4gICAgICAgICAgcmV0dXJuIGN1cnJlbnRRdWV1ZVxuICAgICAgICB9KVxuICAgICAgfSwgc2V0VG9vbFVzZUNvbmZpcm1RdWV1ZSlcbiAgICB9LFxuICAgIFtzZXRBcHBTdGF0ZSwgc2V0VG9vbFVzZUNvbmZpcm1RdWV1ZV0sXG4gIClcblxuICAvLyBSZWdpc3RlciB0aGUgbGVhZGVyJ3Mgc2V0VG9vbFBlcm1pc3Npb25Db250ZXh0IGZvciBpbi1wcm9jZXNzIHRlYW1tYXRlc1xuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIHJlZ2lzdGVyTGVhZGVyU2V0VG9vbFBlcm1pc3Npb25Db250ZXh0KHNldFRvb2xQZXJtaXNzaW9uQ29udGV4dClcbiAgICByZXR1cm4gKCkgPT4gdW5yZWdpc3RlckxlYWRlclNldFRvb2xQZXJtaXNzaW9uQ29udGV4dCgpXG4gIH0sIFtzZXRUb29sUGVybWlzc2lvbkNvbnRleHRdKVxuXG4gIGNvbnN0IGNhblVzZVRvb2wgPSB1c2VDYW5Vc2VUb29sKFxuICAgIHNldFRvb2xVc2VDb25maXJtUXVldWUsXG4gICAgc2V0VG9vbFBlcm1pc3Npb25Db250ZXh0LFxuICApXG5cbiAgY29uc3QgcmVxdWVzdFByb21wdCA9IHVzZUNhbGxiYWNrKFxuICAgICh0aXRsZTogc3RyaW5nLCB0b29sSW5wdXRTdW1tYXJ5Pzogc3RyaW5nIHwgbnVsbCkgPT5cbiAgICAgIChyZXF1ZXN0OiBQcm9tcHRSZXF1ZXN0KTogUHJvbWlzZTxQcm9tcHRSZXNwb25zZT4gPT5cbiAgICAgICAgbmV3IFByb21pc2U8UHJvbXB0UmVzcG9uc2U+KChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICAgICAgICBzZXRQcm9tcHRRdWV1ZShwcmV2ID0+IFtcbiAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICB7IHJlcXVlc3QsIHRpdGxlLCB0b29sSW5wdXRTdW1tYXJ5LCByZXNvbHZlLCByZWplY3QgfSxcbiAgICAgICAgICBdKVxuICAgICAgICB9KSxcbiAgICBbXSxcbiAgKVxuXG4gIGNvbnN0IGdldFRvb2xVc2VDb250ZXh0ID0gdXNlQ2FsbGJhY2soXG4gICAgKFxuICAgICAgbWVzc2FnZXM6IE1lc3NhZ2VUeXBlW10sXG4gICAgICBuZXdNZXNzYWdlczogTWVzc2FnZVR5cGVbXSxcbiAgICAgIGFib3J0Q29udHJvbGxlcjogQWJvcnRDb250cm9sbGVyLFxuICAgICAgbWFpbkxvb3BNb2RlbDogc3RyaW5nLFxuICAgICk6IFByb2Nlc3NVc2VySW5wdXRDb250ZXh0ID0+IHtcbiAgICAgIC8vIFJlYWQgbXV0YWJsZSB2YWx1ZXMgZnJlc2ggZnJvbSB0aGUgc3RvcmUgcmF0aGVyIHRoYW4gY2xvc3VyZS1jYXB0dXJpbmdcbiAgICAgIC8vIHVzZUFwcFN0YXRlKCkgc25hcHNob3RzLiBTYW1lIHZhbHVlcyB0b2RheSAoY2xvc3VyZSBpcyByZWZyZXNoZWQgYnkgdGhlXG4gICAgICAvLyByZW5kZXIgYmV0d2VlbiB0dXJucyk7IGRlY291cGxlcyBmcmVzaG5lc3MgZnJvbSBSZWFjdCdzIHJlbmRlciBjeWNsZSBmb3JcbiAgICAgIC8vIGEgZnV0dXJlIGhlYWRsZXNzIGNvbnZlcnNhdGlvbiBsb29wLiBTYW1lIHBhdHRlcm4gcmVmcmVzaFRvb2xzKCkgdXNlcy5cbiAgICAgIGNvbnN0IHMgPSBzdG9yZS5nZXRTdGF0ZSgpXG5cbiAgICAgIC8vIENvbXB1dGUgdG9vbHMgZnJlc2ggZnJvbSBzdG9yZS5nZXRTdGF0ZSgpIHJhdGhlciB0aGFuIHRoZSBjbG9zdXJlLVxuICAgICAgLy8gY2FwdHVyZWQgYHRvb2xzYC4gdXNlTWFuYWdlTUNQQ29ubmVjdGlvbnMgcG9wdWxhdGVzIGFwcFN0YXRlLm1jcFxuICAgICAgLy8gYXN5bmMgYXMgc2VydmVycyBjb25uZWN0IOKAlCB0aGUgc3RvcmUgbWF5IGhhdmUgbmV3ZXIgTUNQIHN0YXRlIHRoYW5cbiAgICAgIC8vIHRoZSBjbG9zdXJlIGNhcHR1cmVkIGF0IHJlbmRlciB0aW1lLiBBbHNvIGRvdWJsZXMgYXMgcmVmcmVzaFRvb2xzKClcbiAgICAgIC8vIGZvciBtaWQtcXVlcnkgdG9vbCBsaXN0IHVwZGF0ZXMuXG4gICAgICBjb25zdCBjb21wdXRlVG9vbHMgPSAoKSA9PiB7XG4gICAgICAgIGNvbnN0IHN0YXRlID0gc3RvcmUuZ2V0U3RhdGUoKVxuICAgICAgICBjb25zdCBhc3NlbWJsZWQgPSBhc3NlbWJsZVRvb2xQb29sKFxuICAgICAgICAgIHN0YXRlLnRvb2xQZXJtaXNzaW9uQ29udGV4dCxcbiAgICAgICAgICBzdGF0ZS5tY3AudG9vbHMsXG4gICAgICAgIClcbiAgICAgICAgY29uc3QgbWVyZ2VkID0gbWVyZ2VBbmRGaWx0ZXJUb29scyhcbiAgICAgICAgICBjb21iaW5lZEluaXRpYWxUb29scyxcbiAgICAgICAgICBhc3NlbWJsZWQsXG4gICAgICAgICAgc3RhdGUudG9vbFBlcm1pc3Npb25Db250ZXh0Lm1vZGUsXG4gICAgICAgIClcbiAgICAgICAgaWYgKCFtYWluVGhyZWFkQWdlbnREZWZpbml0aW9uKSByZXR1cm4gbWVyZ2VkXG4gICAgICAgIHJldHVybiByZXNvbHZlQWdlbnRUb29scyhtYWluVGhyZWFkQWdlbnREZWZpbml0aW9uLCBtZXJnZWQsIGZhbHNlLCB0cnVlKVxuICAgICAgICAgIC5yZXNvbHZlZFRvb2xzXG4gICAgICB9XG5cbiAgICAgIHJldHVybiB7XG4gICAgICAgIGFib3J0Q29udHJvbGxlcixcbiAgICAgICAgb3B0aW9uczoge1xuICAgICAgICAgIGNvbW1hbmRzLFxuICAgICAgICAgIHRvb2xzOiBjb21wdXRlVG9vbHMoKSxcbiAgICAgICAgICBkZWJ1ZyxcbiAgICAgICAgICB2ZXJib3NlOiBzLnZlcmJvc2UsXG4gICAgICAgICAgbWFpbkxvb3BNb2RlbCxcbiAgICAgICAgICB0aGlua2luZ0NvbmZpZzpcbiAgICAgICAgICAgIHMudGhpbmtpbmdFbmFibGVkICE9PSBmYWxzZSA/IHRoaW5raW5nQ29uZmlnIDogeyB0eXBlOiAnZGlzYWJsZWQnIH0sXG4gICAgICAgICAgLy8gTWVyZ2UgZnJlc2ggZnJvbSBzdG9yZSByYXRoZXIgdGhhbiBjbG9zaW5nIG92ZXIgdXNlTWVyZ2VkQ2xpZW50cydcbiAgICAgICAgICAvLyBtZW1vaXplZCBvdXRwdXQuIGluaXRpYWxNY3BDbGllbnRzIGlzIGEgcHJvcCAoc2Vzc2lvbi1jb25zdGFudCkuXG4gICAgICAgICAgbWNwQ2xpZW50czogbWVyZ2VDbGllbnRzKGluaXRpYWxNY3BDbGllbnRzLCBzLm1jcC5jbGllbnRzKSxcbiAgICAgICAgICBtY3BSZXNvdXJjZXM6IHMubWNwLnJlc291cmNlcyxcbiAgICAgICAgICBpZGVJbnN0YWxsYXRpb25TdGF0dXM6IGlkZUluc3RhbGxhdGlvblN0YXR1cyxcbiAgICAgICAgICBpc05vbkludGVyYWN0aXZlU2Vzc2lvbjogZmFsc2UsXG4gICAgICAgICAgZHluYW1pY01jcENvbmZpZyxcbiAgICAgICAgICB0aGVtZSxcbiAgICAgICAgICBhZ2VudERlZmluaXRpb25zOiBhbGxvd2VkQWdlbnRUeXBlc1xuICAgICAgICAgICAgPyB7IC4uLnMuYWdlbnREZWZpbml0aW9ucywgYWxsb3dlZEFnZW50VHlwZXMgfVxuICAgICAgICAgICAgOiBzLmFnZW50RGVmaW5pdGlvbnMsXG4gICAgICAgICAgY3VzdG9tU3lzdGVtUHJvbXB0LFxuICAgICAgICAgIGFwcGVuZFN5c3RlbVByb21wdCxcbiAgICAgICAgICByZWZyZXNoVG9vbHM6IGNvbXB1dGVUb29scyxcbiAgICAgICAgfSxcbiAgICAgICAgZ2V0QXBwU3RhdGU6ICgpID0+IHN0b3JlLmdldFN0YXRlKCksXG4gICAgICAgIHNldEFwcFN0YXRlLFxuICAgICAgICBtZXNzYWdlcyxcbiAgICAgICAgc2V0TWVzc2FnZXMsXG4gICAgICAgIHVwZGF0ZUZpbGVIaXN0b3J5U3RhdGUoXG4gICAgICAgICAgdXBkYXRlcjogKHByZXY6IEZpbGVIaXN0b3J5U3RhdGUpID0+IEZpbGVIaXN0b3J5U3RhdGUsXG4gICAgICAgICkge1xuICAgICAgICAgIC8vIFBlcmY6IHNraXAgdGhlIHNldFN0YXRlIHdoZW4gdGhlIHVwZGF0ZXIgcmV0dXJucyB0aGUgc2FtZSByZWZlcmVuY2VcbiAgICAgICAgICAvLyAoZS5nLiBmaWxlSGlzdG9yeVRyYWNrRWRpdCByZXR1cm5zIGBzdGF0ZWAgd2hlbiB0aGUgZmlsZSBpcyBhbHJlYWR5XG4gICAgICAgICAgLy8gdHJhY2tlZCkuIE90aGVyd2lzZSBldmVyeSBuby1vcCBjYWxsIHdvdWxkIG5vdGlmeSBhbGwgc3RvcmUgbGlzdGVuZXJzLlxuICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgICAgICAgICAgY29uc3QgdXBkYXRlZCA9IHVwZGF0ZXIocHJldi5maWxlSGlzdG9yeSlcbiAgICAgICAgICAgIGlmICh1cGRhdGVkID09PSBwcmV2LmZpbGVIaXN0b3J5KSByZXR1cm4gcHJldlxuICAgICAgICAgICAgcmV0dXJuIHsgLi4ucHJldiwgZmlsZUhpc3Rvcnk6IHVwZGF0ZWQgfVxuICAgICAgICAgIH0pXG4gICAgICAgIH0sXG4gICAgICAgIHVwZGF0ZUF0dHJpYnV0aW9uU3RhdGUoXG4gICAgICAgICAgdXBkYXRlcjogKHByZXY6IEF0dHJpYnV0aW9uU3RhdGUpID0+IEF0dHJpYnV0aW9uU3RhdGUsXG4gICAgICAgICkge1xuICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgICAgICAgICAgY29uc3QgdXBkYXRlZCA9IHVwZGF0ZXIocHJldi5hdHRyaWJ1dGlvbilcbiAgICAgICAgICAgIGlmICh1cGRhdGVkID09PSBwcmV2LmF0dHJpYnV0aW9uKSByZXR1cm4gcHJldlxuICAgICAgICAgICAgcmV0dXJuIHsgLi4ucHJldiwgYXR0cmlidXRpb246IHVwZGF0ZWQgfVxuICAgICAgICAgIH0pXG4gICAgICAgIH0sXG4gICAgICAgIG9wZW5NZXNzYWdlU2VsZWN0b3I6ICgpID0+IHtcbiAgICAgICAgICBpZiAoIWRpc2FibGVkKSB7XG4gICAgICAgICAgICBzZXRJc01lc3NhZ2VTZWxlY3RvclZpc2libGUodHJ1ZSlcbiAgICAgICAgICB9XG4gICAgICAgIH0sXG4gICAgICAgIG9uQ2hhbmdlQVBJS2V5OiByZXZlcmlmeSxcbiAgICAgICAgcmVhZEZpbGVTdGF0ZTogcmVhZEZpbGVTdGF0ZS5jdXJyZW50LFxuICAgICAgICBzZXRUb29sSlNYLFxuICAgICAgICBhZGROb3RpZmljYXRpb24sXG4gICAgICAgIGFwcGVuZFN5c3RlbU1lc3NhZ2U6IG1zZyA9PiBzZXRNZXNzYWdlcyhwcmV2ID0+IFsuLi5wcmV2LCBtc2ddKSxcbiAgICAgICAgc2VuZE9TTm90aWZpY2F0aW9uOiBvcHRzID0+IHtcbiAgICAgICAgICB2b2lkIHNlbmROb3RpZmljYXRpb24ob3B0cywgdGVybWluYWwpXG4gICAgICAgIH0sXG4gICAgICAgIG9uQ2hhbmdlRHluYW1pY01jcENvbmZpZyxcbiAgICAgICAgb25JbnN0YWxsSURFRXh0ZW5zaW9uOiBzZXRJREVUb0luc3RhbGxFeHRlbnNpb24sXG4gICAgICAgIG5lc3RlZE1lbW9yeUF0dGFjaG1lbnRUcmlnZ2VyczogbmV3IFNldDxzdHJpbmc+KCksXG4gICAgICAgIGxvYWRlZE5lc3RlZE1lbW9yeVBhdGhzOiBsb2FkZWROZXN0ZWRNZW1vcnlQYXRoc1JlZi5jdXJyZW50LFxuICAgICAgICBkeW5hbWljU2tpbGxEaXJUcmlnZ2VyczogbmV3IFNldDxzdHJpbmc+KCksXG4gICAgICAgIGRpc2NvdmVyZWRTa2lsbE5hbWVzOiBkaXNjb3ZlcmVkU2tpbGxOYW1lc1JlZi5jdXJyZW50LFxuICAgICAgICBzZXRSZXNwb25zZUxlbmd0aCxcbiAgICAgICAgcHVzaEFwaU1ldHJpY3NFbnRyeTpcbiAgICAgICAgICBcImV4dGVybmFsXCIgPT09ICdhbnQnXG4gICAgICAgICAgICA/ICh0dGZ0TXM6IG51bWJlcikgPT4ge1xuICAgICAgICAgICAgICAgIGNvbnN0IG5vdyA9IERhdGUubm93KClcbiAgICAgICAgICAgICAgICBjb25zdCBiYXNlbGluZSA9IHJlc3BvbnNlTGVuZ3RoUmVmLmN1cnJlbnRcbiAgICAgICAgICAgICAgICBhcGlNZXRyaWNzUmVmLmN1cnJlbnQucHVzaCh7XG4gICAgICAgICAgICAgICAgICB0dGZ0TXMsXG4gICAgICAgICAgICAgICAgICBmaXJzdFRva2VuVGltZTogbm93LFxuICAgICAgICAgICAgICAgICAgbGFzdFRva2VuVGltZTogbm93LFxuICAgICAgICAgICAgICAgICAgcmVzcG9uc2VMZW5ndGhCYXNlbGluZTogYmFzZWxpbmUsXG4gICAgICAgICAgICAgICAgICBlbmRSZXNwb25zZUxlbmd0aDogYmFzZWxpbmUsXG4gICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgOiB1bmRlZmluZWQsXG4gICAgICAgIHNldFN0cmVhbU1vZGUsXG4gICAgICAgIG9uQ29tcGFjdFByb2dyZXNzOiBldmVudCA9PiB7XG4gICAgICAgICAgc3dpdGNoIChldmVudC50eXBlKSB7XG4gICAgICAgICAgICBjYXNlICdob29rc19zdGFydCc6XG4gICAgICAgICAgICAgIHNldFNwaW5uZXJDb2xvcignY2xhdWRlQmx1ZV9GT1JfU1lTVEVNX1NQSU5ORVInKVxuICAgICAgICAgICAgICBzZXRTcGlubmVyU2hpbW1lckNvbG9yKCdjbGF1ZGVCbHVlU2hpbW1lcl9GT1JfU1lTVEVNX1NQSU5ORVInKVxuICAgICAgICAgICAgICBzZXRTcGlubmVyTWVzc2FnZShcbiAgICAgICAgICAgICAgICBldmVudC5ob29rVHlwZSA9PT0gJ3ByZV9jb21wYWN0J1xuICAgICAgICAgICAgICAgICAgPyAnUnVubmluZyBQcmVDb21wYWN0IGhvb2tzXFx1MjAyNidcbiAgICAgICAgICAgICAgICAgIDogZXZlbnQuaG9va1R5cGUgPT09ICdwb3N0X2NvbXBhY3QnXG4gICAgICAgICAgICAgICAgICAgID8gJ1J1bm5pbmcgUG9zdENvbXBhY3QgaG9va3NcXHUyMDI2J1xuICAgICAgICAgICAgICAgICAgICA6ICdSdW5uaW5nIFNlc3Npb25TdGFydCBob29rc1xcdTIwMjYnLFxuICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgIGJyZWFrXG4gICAgICAgICAgICBjYXNlICdjb21wYWN0X3N0YXJ0JzpcbiAgICAgICAgICAgICAgc2V0U3Bpbm5lck1lc3NhZ2UoJ0NvbXBhY3RpbmcgY29udmVyc2F0aW9uJylcbiAgICAgICAgICAgICAgYnJlYWtcbiAgICAgICAgICAgIGNhc2UgJ2NvbXBhY3RfZW5kJzpcbiAgICAgICAgICAgICAgc2V0U3Bpbm5lck1lc3NhZ2UobnVsbClcbiAgICAgICAgICAgICAgc2V0U3Bpbm5lckNvbG9yKG51bGwpXG4gICAgICAgICAgICAgIHNldFNwaW5uZXJTaGltbWVyQ29sb3IobnVsbClcbiAgICAgICAgICAgICAgYnJlYWtcbiAgICAgICAgICB9XG4gICAgICAgIH0sXG4gICAgICAgIHNldEluUHJvZ3Jlc3NUb29sVXNlSURzLFxuICAgICAgICBzZXRIYXNJbnRlcnJ1cHRpYmxlVG9vbEluUHJvZ3Jlc3M6ICh2OiBib29sZWFuKSA9PiB7XG4gICAgICAgICAgaGFzSW50ZXJydXB0aWJsZVRvb2xJblByb2dyZXNzUmVmLmN1cnJlbnQgPSB2XG4gICAgICAgIH0sXG4gICAgICAgIHJlc3VtZSxcbiAgICAgICAgc2V0Q29udmVyc2F0aW9uSWQsXG4gICAgICAgIHJlcXVlc3RQcm9tcHQ6IGZlYXR1cmUoJ0hPT0tfUFJPTVBUUycpID8gcmVxdWVzdFByb21wdCA6IHVuZGVmaW5lZCxcbiAgICAgICAgY29udGVudFJlcGxhY2VtZW50U3RhdGU6IGNvbnRlbnRSZXBsYWNlbWVudFN0YXRlUmVmLmN1cnJlbnQsXG4gICAgICB9XG4gICAgfSxcbiAgICBbXG4gICAgICBjb21tYW5kcyxcbiAgICAgIGNvbWJpbmVkSW5pdGlhbFRvb2xzLFxuICAgICAgbWFpblRocmVhZEFnZW50RGVmaW5pdGlvbixcbiAgICAgIGRlYnVnLFxuICAgICAgaW5pdGlhbE1jcENsaWVudHMsXG4gICAgICBpZGVJbnN0YWxsYXRpb25TdGF0dXMsXG4gICAgICBkeW5hbWljTWNwQ29uZmlnLFxuICAgICAgdGhlbWUsXG4gICAgICBhbGxvd2VkQWdlbnRUeXBlcyxcbiAgICAgIHN0b3JlLFxuICAgICAgc2V0QXBwU3RhdGUsXG4gICAgICByZXZlcmlmeSxcbiAgICAgIGFkZE5vdGlmaWNhdGlvbixcbiAgICAgIHNldE1lc3NhZ2VzLFxuICAgICAgb25DaGFuZ2VEeW5hbWljTWNwQ29uZmlnLFxuICAgICAgcmVzdW1lLFxuICAgICAgcmVxdWVzdFByb21wdCxcbiAgICAgIGRpc2FibGVkLFxuICAgICAgY3VzdG9tU3lzdGVtUHJvbXB0LFxuICAgICAgYXBwZW5kU3lzdGVtUHJvbXB0LFxuICAgICAgc2V0Q29udmVyc2F0aW9uSWQsXG4gICAgXSxcbiAgKVxuXG4gIC8vIFNlc3Npb24gYmFja2dyb3VuZGluZyAoQ3RybCtCIHRvIGJhY2tncm91bmQvZm9yZWdyb3VuZClcbiAgY29uc3QgaGFuZGxlQmFja2dyb3VuZFF1ZXJ5ID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIC8vIFN0b3AgdGhlIGZvcmVncm91bmQgcXVlcnkgc28gdGhlIGJhY2tncm91bmQgb25lIHRha2VzIG92ZXJcbiAgICBhYm9ydENvbnRyb2xsZXI/LmFib3J0KCdiYWNrZ3JvdW5kJylcbiAgICAvLyBBYm9ydGluZyBzdWJhZ2VudHMgbWF5IHByb2R1Y2UgdGFzay1jb21wbGV0ZWQgbm90aWZpY2F0aW9ucy5cbiAgICAvLyBDbGVhciB0YXNrIG5vdGlmaWNhdGlvbnMgc28gdGhlIHF1ZXVlIHByb2Nlc3NvciBkb2Vzbid0IGltbWVkaWF0ZWx5XG4gICAgLy8gc3RhcnQgYSBuZXcgZm9yZWdyb3VuZCBxdWVyeTsgZm9yd2FyZCB0aGVtIHRvIHRoZSBiYWNrZ3JvdW5kIHNlc3Npb24uXG4gICAgY29uc3QgcmVtb3ZlZE5vdGlmaWNhdGlvbnMgPSByZW1vdmVCeUZpbHRlcihcbiAgICAgIGNtZCA9PiBjbWQubW9kZSA9PT0gJ3Rhc2stbm90aWZpY2F0aW9uJyxcbiAgICApXG5cbiAgICB2b2lkIChhc3luYyAoKSA9PiB7XG4gICAgICBjb25zdCB0b29sVXNlQ29udGV4dCA9IGdldFRvb2xVc2VDb250ZXh0KFxuICAgICAgICBtZXNzYWdlc1JlZi5jdXJyZW50LFxuICAgICAgICBbXSxcbiAgICAgICAgbmV3IEFib3J0Q29udHJvbGxlcigpLFxuICAgICAgICBtYWluTG9vcE1vZGVsLFxuICAgICAgKVxuXG4gICAgICBjb25zdCBbZGVmYXVsdFN5c3RlbVByb21wdCwgdXNlckNvbnRleHQsIHN5c3RlbUNvbnRleHRdID1cbiAgICAgICAgYXdhaXQgUHJvbWlzZS5hbGwoW1xuICAgICAgICAgIGdldFN5c3RlbVByb21wdChcbiAgICAgICAgICAgIHRvb2xVc2VDb250ZXh0Lm9wdGlvbnMudG9vbHMsXG4gICAgICAgICAgICBtYWluTG9vcE1vZGVsLFxuICAgICAgICAgICAgQXJyYXkuZnJvbShcbiAgICAgICAgICAgICAgdG9vbFBlcm1pc3Npb25Db250ZXh0LmFkZGl0aW9uYWxXb3JraW5nRGlyZWN0b3JpZXMua2V5cygpLFxuICAgICAgICAgICAgKSxcbiAgICAgICAgICAgIHRvb2xVc2VDb250ZXh0Lm9wdGlvbnMubWNwQ2xpZW50cyxcbiAgICAgICAgICApLFxuICAgICAgICAgIGdldFVzZXJDb250ZXh0KCksXG4gICAgICAgICAgZ2V0U3lzdGVtQ29udGV4dCgpLFxuICAgICAgICBdKVxuXG4gICAgICBjb25zdCBzeXN0ZW1Qcm9tcHQgPSBidWlsZEVmZmVjdGl2ZVN5c3RlbVByb21wdCh7XG4gICAgICAgIG1haW5UaHJlYWRBZ2VudERlZmluaXRpb24sXG4gICAgICAgIHRvb2xVc2VDb250ZXh0LFxuICAgICAgICBjdXN0b21TeXN0ZW1Qcm9tcHQsXG4gICAgICAgIGRlZmF1bHRTeXN0ZW1Qcm9tcHQsXG4gICAgICAgIGFwcGVuZFN5c3RlbVByb21wdCxcbiAgICAgIH0pXG4gICAgICB0b29sVXNlQ29udGV4dC5yZW5kZXJlZFN5c3RlbVByb21wdCA9IHN5c3RlbVByb21wdFxuXG4gICAgICBjb25zdCBub3RpZmljYXRpb25BdHRhY2htZW50cyA9IGF3YWl0IGdldFF1ZXVlZENvbW1hbmRBdHRhY2htZW50cyhcbiAgICAgICAgcmVtb3ZlZE5vdGlmaWNhdGlvbnMsXG4gICAgICApLmNhdGNoKCgpID0+IFtdKVxuICAgICAgY29uc3Qgbm90aWZpY2F0aW9uTWVzc2FnZXMgPSBub3RpZmljYXRpb25BdHRhY2htZW50cy5tYXAoXG4gICAgICAgIGNyZWF0ZUF0dGFjaG1lbnRNZXNzYWdlLFxuICAgICAgKVxuXG4gICAgICAvLyBEZWR1cGxpY2F0ZTogaWYgdGhlIHF1ZXJ5IGxvb3AgYWxyZWFkeSB5aWVsZGVkIGEgbm90aWZpY2F0aW9uIGludG9cbiAgICAgIC8vIG1lc3NhZ2VzUmVmIGJlZm9yZSB3ZSByZW1vdmVkIGl0IGZyb20gdGhlIHF1ZXVlLCBza2lwIGR1cGxpY2F0ZXMuXG4gICAgICAvLyBXZSB1c2UgcHJvbXB0IHRleHQgZm9yIGRlZHVwIGJlY2F1c2Ugc291cmNlX3V1aWQgaXMgbm90IHNldCBvblxuICAgICAgLy8gdGFzay1ub3RpZmljYXRpb24gUXVldWVkQ29tbWFuZHMgKGVucXVldWVQZW5kaW5nTm90aWZpY2F0aW9uIGNhbGxlcnNcbiAgICAgIC8vIGRvbid0IHBhc3MgdXVpZCksIHNvIGl0IHdvdWxkIGFsd2F5cyBiZSB1bmRlZmluZWQuXG4gICAgICBjb25zdCBleGlzdGluZ1Byb21wdHMgPSBuZXcgU2V0PHN0cmluZz4oKVxuICAgICAgZm9yIChjb25zdCBtIG9mIG1lc3NhZ2VzUmVmLmN1cnJlbnQpIHtcbiAgICAgICAgaWYgKFxuICAgICAgICAgIG0udHlwZSA9PT0gJ2F0dGFjaG1lbnQnICYmXG4gICAgICAgICAgbS5hdHRhY2htZW50LnR5cGUgPT09ICdxdWV1ZWRfY29tbWFuZCcgJiZcbiAgICAgICAgICBtLmF0dGFjaG1lbnQuY29tbWFuZE1vZGUgPT09ICd0YXNrLW5vdGlmaWNhdGlvbicgJiZcbiAgICAgICAgICB0eXBlb2YgbS5hdHRhY2htZW50LnByb21wdCA9PT0gJ3N0cmluZydcbiAgICAgICAgKSB7XG4gICAgICAgICAgZXhpc3RpbmdQcm9tcHRzLmFkZChtLmF0dGFjaG1lbnQucHJvbXB0KVxuICAgICAgICB9XG4gICAgICB9XG4gICAgICBjb25zdCB1bmlxdWVOb3RpZmljYXRpb25zID0gbm90aWZpY2F0aW9uTWVzc2FnZXMuZmlsdGVyKFxuICAgICAgICBtID0+XG4gICAgICAgICAgbS5hdHRhY2htZW50LnR5cGUgPT09ICdxdWV1ZWRfY29tbWFuZCcgJiZcbiAgICAgICAgICAodHlwZW9mIG0uYXR0YWNobWVudC5wcm9tcHQgIT09ICdzdHJpbmcnIHx8XG4gICAgICAgICAgICAhZXhpc3RpbmdQcm9tcHRzLmhhcyhtLmF0dGFjaG1lbnQucHJvbXB0KSksXG4gICAgICApXG5cbiAgICAgIHN0YXJ0QmFja2dyb3VuZFNlc3Npb24oe1xuICAgICAgICBtZXNzYWdlczogWy4uLm1lc3NhZ2VzUmVmLmN1cnJlbnQsIC4uLnVuaXF1ZU5vdGlmaWNhdGlvbnNdLFxuICAgICAgICBxdWVyeVBhcmFtczoge1xuICAgICAgICAgIHN5c3RlbVByb21wdCxcbiAgICAgICAgICB1c2VyQ29udGV4dCxcbiAgICAgICAgICBzeXN0ZW1Db250ZXh0LFxuICAgICAgICAgIGNhblVzZVRvb2wsXG4gICAgICAgICAgdG9vbFVzZUNvbnRleHQsXG4gICAgICAgICAgcXVlcnlTb3VyY2U6IGdldFF1ZXJ5U291cmNlRm9yUkVQTCgpLFxuICAgICAgICB9LFxuICAgICAgICBkZXNjcmlwdGlvbjogdGVybWluYWxUaXRsZSxcbiAgICAgICAgc2V0QXBwU3RhdGUsXG4gICAgICAgIGFnZW50RGVmaW5pdGlvbjogbWFpblRocmVhZEFnZW50RGVmaW5pdGlvbixcbiAgICAgIH0pXG4gICAgfSkoKVxuICB9LCBbXG4gICAgYWJvcnRDb250cm9sbGVyLFxuICAgIG1haW5Mb29wTW9kZWwsXG4gICAgdG9vbFBlcm1pc3Npb25Db250ZXh0LFxuICAgIG1haW5UaHJlYWRBZ2VudERlZmluaXRpb24sXG4gICAgZ2V0VG9vbFVzZUNvbnRleHQsXG4gICAgY3VzdG9tU3lzdGVtUHJvbXB0LFxuICAgIGFwcGVuZFN5c3RlbVByb21wdCxcbiAgICBjYW5Vc2VUb29sLFxuICAgIHNldEFwcFN0YXRlLFxuICBdKVxuXG4gIGNvbnN0IHsgaGFuZGxlQmFja2dyb3VuZFNlc3Npb24gfSA9IHVzZVNlc3Npb25CYWNrZ3JvdW5kaW5nKHtcbiAgICBzZXRNZXNzYWdlcyxcbiAgICBzZXRJc0xvYWRpbmc6IHNldElzRXh0ZXJuYWxMb2FkaW5nLFxuICAgIHJlc2V0TG9hZGluZ1N0YXRlLFxuICAgIHNldEFib3J0Q29udHJvbGxlcixcbiAgICBvbkJhY2tncm91bmRRdWVyeTogaGFuZGxlQmFja2dyb3VuZFF1ZXJ5LFxuICB9KVxuXG4gIGNvbnN0IG9uUXVlcnlFdmVudCA9IHVzZUNhbGxiYWNrKFxuICAgIChldmVudDogUGFyYW1ldGVyczx0eXBlb2YgaGFuZGxlTWVzc2FnZUZyb21TdHJlYW0+WzBdKSA9PiB7XG4gICAgICBoYW5kbGVNZXNzYWdlRnJvbVN0cmVhbShcbiAgICAgICAgZXZlbnQsXG4gICAgICAgIG5ld01lc3NhZ2UgPT4ge1xuICAgICAgICAgIGlmIChpc0NvbXBhY3RCb3VuZGFyeU1lc3NhZ2UobmV3TWVzc2FnZSkpIHtcbiAgICAgICAgICAgIC8vIEZ1bGxzY3JlZW46IGtlZXAgcHJlLWNvbXBhY3QgbWVzc2FnZXMgZm9yIHNjcm9sbGJhY2suIHF1ZXJ5LnRzXG4gICAgICAgICAgICAvLyBzbGljZXMgYXQgdGhlIGJvdW5kYXJ5IGZvciBBUEkgY2FsbHMsIE1lc3NhZ2VzLnRzeCBza2lwcyB0aGVcbiAgICAgICAgICAgIC8vIGJvdW5kYXJ5IGZpbHRlciBpbiBmdWxsc2NyZWVuLCBhbmQgdXNlTG9nTWVzc2FnZXMgdHJlYXRzIHRoaXNcbiAgICAgICAgICAgIC8vIGFzIGFuIGluY3JlbWVudGFsIGFwcGVuZCAoZmlyc3QgdXVpZCB1bmNoYW5nZWQpLiBDYXAgYXQgb25lXG4gICAgICAgICAgICAvLyBjb21wYWN0LWludGVydmFsIG9mIHNjcm9sbGJhY2sg4oCUIG5vcm1hbGl6ZU1lc3NhZ2VzL2FwcGx5R3JvdXBpbmdcbiAgICAgICAgICAgIC8vIGFyZSBPKG4pIHBlciByZW5kZXIsIHNvIGRyb3AgZXZlcnl0aGluZyBiZWZvcmUgdGhlIHByZXZpb3VzXG4gICAgICAgICAgICAvLyBib3VuZGFyeSB0byBrZWVwIG4gYm91bmRlZCBhY3Jvc3MgbXVsdGktZGF5IHNlc3Npb25zLlxuICAgICAgICAgICAgaWYgKGlzRnVsbHNjcmVlbkVudkVuYWJsZWQoKSkge1xuICAgICAgICAgICAgICBzZXRNZXNzYWdlcyhvbGQgPT4gW1xuICAgICAgICAgICAgICAgIC4uLmdldE1lc3NhZ2VzQWZ0ZXJDb21wYWN0Qm91bmRhcnkob2xkLCB7XG4gICAgICAgICAgICAgICAgICBpbmNsdWRlU25pcHBlZDogdHJ1ZSxcbiAgICAgICAgICAgICAgICB9KSxcbiAgICAgICAgICAgICAgICBuZXdNZXNzYWdlLFxuICAgICAgICAgICAgICBdKVxuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgc2V0TWVzc2FnZXMoKCkgPT4gW25ld01lc3NhZ2VdKVxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgLy8gQnVtcCBjb252ZXJzYXRpb25JZCBzbyBNZXNzYWdlcy50c3ggcm93IGtleXMgY2hhbmdlIGFuZFxuICAgICAgICAgICAgLy8gc3RhbGUgbWVtb2l6ZWQgcm93cyByZW1vdW50IHdpdGggcG9zdC1jb21wYWN0IGNvbnRlbnQuXG4gICAgICAgICAgICBzZXRDb252ZXJzYXRpb25JZChyYW5kb21VVUlEKCkpXG4gICAgICAgICAgICAvLyBDb21wYWN0aW9uIHN1Y2NlZWRlZCDigJQgY2xlYXIgdGhlIGNvbnRleHQtYmxvY2tlZCBmbGFnIHNvIHRpY2tzIHJlc3VtZVxuICAgICAgICAgICAgaWYgKGZlYXR1cmUoJ1BST0FDVElWRScpIHx8IGZlYXR1cmUoJ0tBSVJPUycpKSB7XG4gICAgICAgICAgICAgIHByb2FjdGl2ZU1vZHVsZT8uc2V0Q29udGV4dEJsb2NrZWQoZmFsc2UpXG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSBlbHNlIGlmIChcbiAgICAgICAgICAgIG5ld01lc3NhZ2UudHlwZSA9PT0gJ3Byb2dyZXNzJyAmJlxuICAgICAgICAgICAgaXNFcGhlbWVyYWxUb29sUHJvZ3Jlc3MobmV3TWVzc2FnZS5kYXRhLnR5cGUpXG4gICAgICAgICAgKSB7XG4gICAgICAgICAgICAvLyBSZXBsYWNlIHRoZSBwcmV2aW91cyBlcGhlbWVyYWwgcHJvZ3Jlc3MgdGljayBmb3IgdGhlIHNhbWUgdG9vbFxuICAgICAgICAgICAgLy8gY2FsbCBpbnN0ZWFkIG9mIGFwcGVuZGluZy4gU2xlZXAvQmFzaCBlbWl0IGEgdGljayBwZXIgc2Vjb25kIGFuZFxuICAgICAgICAgICAgLy8gb25seSB0aGUgbGFzdCBvbmUgaXMgcmVuZGVyZWQ7IGFwcGVuZGluZyBibG93cyB1cCB0aGUgbWVzc2FnZXNcbiAgICAgICAgICAgIC8vIGFycmF5ICgxM2srIG9ic2VydmVkKSBhbmQgdGhlIHRyYW5zY3JpcHQgKDEyME1CIG9mIHNsZWVwX3Byb2dyZXNzXG4gICAgICAgICAgICAvLyBsaW5lcykuIHVzZUxvZ01lc3NhZ2VzIHRyYWNrcyBsZW5ndGgsIHNvIHNhbWUtbGVuZ3RoIHJlcGxhY2VtZW50XG4gICAgICAgICAgICAvLyBhbHNvIHNraXBzIHRoZSB0cmFuc2NyaXB0IHdyaXRlLlxuICAgICAgICAgICAgLy8gYWdlbnRfcHJvZ3Jlc3MgLyBob29rX3Byb2dyZXNzIC8gc2tpbGxfcHJvZ3Jlc3MgYXJlIE5PVCBlcGhlbWVyYWxcbiAgICAgICAgICAgIC8vIOKAlCBlYWNoIGNhcnJpZXMgZGlzdGluY3Qgc3RhdGUgdGhlIFVJIG5lZWRzIChlLmcuIHN1YmFnZW50IHRvb2xcbiAgICAgICAgICAgIC8vIGhpc3RvcnkpLiBSZXBsYWNpbmcgdGhvc2UgbGVhdmVzIHRoZSBBZ2VudFRvb2wgVUkgc3R1Y2sgYXRcbiAgICAgICAgICAgIC8vIFwiSW5pdGlhbGl6aW5n4oCmXCIgYmVjYXVzZSBpdCByZW5kZXJzIHRoZSBmdWxsIHByb2dyZXNzIHRyYWlsLlxuICAgICAgICAgICAgc2V0TWVzc2FnZXMob2xkTWVzc2FnZXMgPT4ge1xuICAgICAgICAgICAgICBjb25zdCBsYXN0ID0gb2xkTWVzc2FnZXMuYXQoLTEpXG4gICAgICAgICAgICAgIGlmIChcbiAgICAgICAgICAgICAgICBsYXN0Py50eXBlID09PSAncHJvZ3Jlc3MnICYmXG4gICAgICAgICAgICAgICAgbGFzdC5wYXJlbnRUb29sVXNlSUQgPT09IG5ld01lc3NhZ2UucGFyZW50VG9vbFVzZUlEICYmXG4gICAgICAgICAgICAgICAgbGFzdC5kYXRhLnR5cGUgPT09IG5ld01lc3NhZ2UuZGF0YS50eXBlXG4gICAgICAgICAgICAgICkge1xuICAgICAgICAgICAgICAgIGNvbnN0IGNvcHkgPSBvbGRNZXNzYWdlcy5zbGljZSgpXG4gICAgICAgICAgICAgICAgY29weVtjb3B5Lmxlbmd0aCAtIDFdID0gbmV3TWVzc2FnZVxuICAgICAgICAgICAgICAgIHJldHVybiBjb3B5XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgcmV0dXJuIFsuLi5vbGRNZXNzYWdlcywgbmV3TWVzc2FnZV1cbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHNldE1lc3NhZ2VzKG9sZE1lc3NhZ2VzID0+IFsuLi5vbGRNZXNzYWdlcywgbmV3TWVzc2FnZV0pXG4gICAgICAgICAgfVxuICAgICAgICAgIC8vIEJsb2NrIHRpY2tzIG9uIEFQSSBlcnJvcnMgdG8gcHJldmVudCB0aWNrIOKGkiBlcnJvciDihpIgdGlja1xuICAgICAgICAgIC8vIHJ1bmF3YXkgbG9vcHMgKGUuZy4sIGF1dGggZmFpbHVyZSwgcmF0ZSBsaW1pdCwgYmxvY2tpbmcgbGltaXQpLlxuICAgICAgICAgIC8vIENsZWFyZWQgb24gY29tcGFjdCBib3VuZGFyeSAoYWJvdmUpIG9yIHN1Y2Nlc3NmdWwgcmVzcG9uc2UgKGJlbG93KS5cbiAgICAgICAgICBpZiAoZmVhdHVyZSgnUFJPQUNUSVZFJykgfHwgZmVhdHVyZSgnS0FJUk9TJykpIHtcbiAgICAgICAgICAgIGlmIChcbiAgICAgICAgICAgICAgbmV3TWVzc2FnZS50eXBlID09PSAnYXNzaXN0YW50JyAmJlxuICAgICAgICAgICAgICAnaXNBcGlFcnJvck1lc3NhZ2UnIGluIG5ld01lc3NhZ2UgJiZcbiAgICAgICAgICAgICAgbmV3TWVzc2FnZS5pc0FwaUVycm9yTWVzc2FnZVxuICAgICAgICAgICAgKSB7XG4gICAgICAgICAgICAgIHByb2FjdGl2ZU1vZHVsZT8uc2V0Q29udGV4dEJsb2NrZWQodHJ1ZSlcbiAgICAgICAgICAgIH0gZWxzZSBpZiAobmV3TWVzc2FnZS50eXBlID09PSAnYXNzaXN0YW50Jykge1xuICAgICAgICAgICAgICBwcm9hY3RpdmVNb2R1bGU/LnNldENvbnRleHRCbG9ja2VkKGZhbHNlKVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfSxcbiAgICAgICAgbmV3Q29udGVudCA9PiB7XG4gICAgICAgICAgLy8gc2V0UmVzcG9uc2VMZW5ndGggaGFuZGxlcyB1cGRhdGluZyBib3RoIHJlc3BvbnNlTGVuZ3RoUmVmIChmb3JcbiAgICAgICAgICAvLyBzcGlubmVyIGFuaW1hdGlvbikgYW5kIGFwaU1ldHJpY3NSZWYgKGVuZFJlc3BvbnNlTGVuZ3RoL2xhc3RUb2tlblRpbWVcbiAgICAgICAgICAvLyBmb3IgT1RQUykuIE5vIHNlcGFyYXRlIG1ldHJpY3MgdXBkYXRlIG5lZWRlZCBoZXJlLlxuICAgICAgICAgIHNldFJlc3BvbnNlTGVuZ3RoKGxlbmd0aCA9PiBsZW5ndGggKyBuZXdDb250ZW50Lmxlbmd0aClcbiAgICAgICAgfSxcbiAgICAgICAgc2V0U3RyZWFtTW9kZSxcbiAgICAgICAgc2V0U3RyZWFtaW5nVG9vbFVzZXMsXG4gICAgICAgIHRvbWJzdG9uZWRNZXNzYWdlID0+IHtcbiAgICAgICAgICBzZXRNZXNzYWdlcyhvbGRNZXNzYWdlcyA9PlxuICAgICAgICAgICAgb2xkTWVzc2FnZXMuZmlsdGVyKG0gPT4gbSAhPT0gdG9tYnN0b25lZE1lc3NhZ2UpLFxuICAgICAgICAgIClcbiAgICAgICAgICB2b2lkIHJlbW92ZVRyYW5zY3JpcHRNZXNzYWdlKHRvbWJzdG9uZWRNZXNzYWdlLnV1aWQpXG4gICAgICAgIH0sXG4gICAgICAgIHNldFN0cmVhbWluZ1RoaW5raW5nLFxuICAgICAgICBtZXRyaWNzID0+IHtcbiAgICAgICAgICBjb25zdCBub3cgPSBEYXRlLm5vdygpXG4gICAgICAgICAgY29uc3QgYmFzZWxpbmUgPSByZXNwb25zZUxlbmd0aFJlZi5jdXJyZW50XG4gICAgICAgICAgYXBpTWV0cmljc1JlZi5jdXJyZW50LnB1c2goe1xuICAgICAgICAgICAgLi4ubWV0cmljcyxcbiAgICAgICAgICAgIGZpcnN0VG9rZW5UaW1lOiBub3csXG4gICAgICAgICAgICBsYXN0VG9rZW5UaW1lOiBub3csXG4gICAgICAgICAgICByZXNwb25zZUxlbmd0aEJhc2VsaW5lOiBiYXNlbGluZSxcbiAgICAgICAgICAgIGVuZFJlc3BvbnNlTGVuZ3RoOiBiYXNlbGluZSxcbiAgICAgICAgICB9KVxuICAgICAgICB9LFxuICAgICAgICBvblN0cmVhbWluZ1RleHQsXG4gICAgICApXG4gICAgfSxcbiAgICBbXG4gICAgICBzZXRNZXNzYWdlcyxcbiAgICAgIHNldFJlc3BvbnNlTGVuZ3RoLFxuICAgICAgc2V0U3RyZWFtTW9kZSxcbiAgICAgIHNldFN0cmVhbWluZ1Rvb2xVc2VzLFxuICAgICAgc2V0U3RyZWFtaW5nVGhpbmtpbmcsXG4gICAgICBvblN0cmVhbWluZ1RleHQsXG4gICAgXSxcbiAgKVxuXG4gIGNvbnN0IG9uUXVlcnlJbXBsID0gdXNlQ2FsbGJhY2soXG4gICAgYXN5bmMgKFxuICAgICAgbWVzc2FnZXNJbmNsdWRpbmdOZXdNZXNzYWdlczogTWVzc2FnZVR5cGVbXSxcbiAgICAgIG5ld01lc3NhZ2VzOiBNZXNzYWdlVHlwZVtdLFxuICAgICAgYWJvcnRDb250cm9sbGVyOiBBYm9ydENvbnRyb2xsZXIsXG4gICAgICBzaG91bGRRdWVyeTogYm9vbGVhbixcbiAgICAgIGFkZGl0aW9uYWxBbGxvd2VkVG9vbHM6IHN0cmluZ1tdLFxuICAgICAgbWFpbkxvb3BNb2RlbFBhcmFtOiBzdHJpbmcsXG4gICAgICBlZmZvcnQ/OiBFZmZvcnRWYWx1ZSxcbiAgICApID0+IHtcbiAgICAgIC8vIFByZXBhcmUgSURFIGludGVncmF0aW9uIGZvciBuZXcgcHJvbXB0LiBSZWFkIG1jcENsaWVudHMgZnJlc2ggZnJvbVxuICAgICAgLy8gc3RvcmUg4oCUIHVzZU1hbmFnZU1DUENvbm5lY3Rpb25zIG1heSBoYXZlIHBvcHVsYXRlZCBpdCBzaW5jZSB0aGVcbiAgICAgIC8vIHJlbmRlciB0aGF0IGNhcHR1cmVkIHRoaXMgY2xvc3VyZSAoc2FtZSBwYXR0ZXJuIGFzIGNvbXB1dGVUb29scykuXG4gICAgICBpZiAoc2hvdWxkUXVlcnkpIHtcbiAgICAgICAgY29uc3QgZnJlc2hDbGllbnRzID0gbWVyZ2VDbGllbnRzKFxuICAgICAgICAgIGluaXRpYWxNY3BDbGllbnRzLFxuICAgICAgICAgIHN0b3JlLmdldFN0YXRlKCkubWNwLmNsaWVudHMsXG4gICAgICAgIClcbiAgICAgICAgdm9pZCBkaWFnbm9zdGljVHJhY2tlci5oYW5kbGVRdWVyeVN0YXJ0KGZyZXNoQ2xpZW50cylcbiAgICAgICAgY29uc3QgaWRlQ2xpZW50ID0gZ2V0Q29ubmVjdGVkSWRlQ2xpZW50KGZyZXNoQ2xpZW50cylcbiAgICAgICAgaWYgKGlkZUNsaWVudCkge1xuICAgICAgICAgIHZvaWQgY2xvc2VPcGVuRGlmZnMoaWRlQ2xpZW50KVxuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIE1hcmsgb25ib2FyZGluZyBhcyBjb21wbGV0ZSB3aGVuIGFueSB1c2VyIG1lc3NhZ2UgaXMgc2VudCB0byBDbGF1ZGVcbiAgICAgIHZvaWQgbWF5YmVNYXJrUHJvamVjdE9uYm9hcmRpbmdDb21wbGV0ZSgpXG5cbiAgICAgIC8vIEV4dHJhY3QgYSBzZXNzaW9uIHRpdGxlIGZyb20gdGhlIGZpcnN0IHJlYWwgdXNlciBtZXNzYWdlLiBPbmUtc2hvdFxuICAgICAgLy8gdmlhIHJlZiAod2FzIHRlbmd1X2JpcmNoX21pc3QgZXhwZXJpbWVudDogZmlyc3QtbWVzc2FnZS1vbmx5IHRvIHNhdmVcbiAgICAgIC8vIEhhaWt1IGNhbGxzKS4gVGhlIHJlZiByZXBsYWNlcyB0aGUgb2xkIGBtZXNzYWdlcy5sZW5ndGggPD0gMWAgY2hlY2ssXG4gICAgICAvLyB3aGljaCB3YXMgYnJva2VuIGJ5IFNlc3Npb25TdGFydCBob29rIG1lc3NhZ2VzIChwcmVwZW5kZWQgdmlhXG4gICAgICAvLyB1c2VEZWZlcnJlZEhvb2tNZXNzYWdlcykgYW5kIGF0dGFjaG1lbnQgbWVzc2FnZXMgKGFwcGVuZGVkIGJ5XG4gICAgICAvLyBwcm9jZXNzVGV4dFByb21wdCkg4oCUIGJvdGggcHVzaGVkIGxlbmd0aCBwYXN0IDEgb24gdHVybiBvbmUsIHNvIHRoZVxuICAgICAgLy8gdGl0bGUgc2lsZW50bHkgZmVsbCB0aHJvdWdoIHRvIHRoZSBcIkNsYXVkZSBDb2RlXCIgZGVmYXVsdC5cbiAgICAgIGlmIChcbiAgICAgICAgIXRpdGxlRGlzYWJsZWQgJiZcbiAgICAgICAgIXNlc3Npb25UaXRsZSAmJlxuICAgICAgICAhYWdlbnRUaXRsZSAmJlxuICAgICAgICAhaGFpa3VUaXRsZUF0dGVtcHRlZFJlZi5jdXJyZW50XG4gICAgICApIHtcbiAgICAgICAgY29uc3QgZmlyc3RVc2VyTWVzc2FnZSA9IG5ld01lc3NhZ2VzLmZpbmQoXG4gICAgICAgICAgbSA9PiBtLnR5cGUgPT09ICd1c2VyJyAmJiAhbS5pc01ldGEsXG4gICAgICAgIClcbiAgICAgICAgY29uc3QgdGV4dCA9XG4gICAgICAgICAgZmlyc3RVc2VyTWVzc2FnZT8udHlwZSA9PT0gJ3VzZXInXG4gICAgICAgICAgICA/IGdldENvbnRlbnRUZXh0KGZpcnN0VXNlck1lc3NhZ2UubWVzc2FnZS5jb250ZW50KVxuICAgICAgICAgICAgOiBudWxsXG4gICAgICAgIC8vIFNraXAgc3ludGhldGljIGJyZWFkY3J1bWJzIOKAlCBzbGFzaC1jb21tYW5kIG91dHB1dCwgcHJvbXB0LXNraWxsXG4gICAgICAgIC8vIGV4cGFuc2lvbnMgKC9jb21taXQg4oaSIDxjb21tYW5kLW1lc3NhZ2U+KSwgbG9jYWwtY29tbWFuZCBoZWFkZXJzXG4gICAgICAgIC8vICgvaGVscCDihpIgPGNvbW1hbmQtbmFtZT4pLCBhbmQgYmFzaC1tb2RlICghY21kIOKGkiA8YmFzaC1pbnB1dD4pLlxuICAgICAgICAvLyBOb25lIG9mIHRoZXNlIGFyZSB0aGUgdXNlcidzIHRvcGljOyB3YWl0IGZvciByZWFsIHByb3NlLlxuICAgICAgICBpZiAoXG4gICAgICAgICAgdGV4dCAmJlxuICAgICAgICAgICF0ZXh0LnN0YXJ0c1dpdGgoYDwke0xPQ0FMX0NPTU1BTkRfU1RET1VUX1RBR30+YCkgJiZcbiAgICAgICAgICAhdGV4dC5zdGFydHNXaXRoKGA8JHtDT01NQU5EX01FU1NBR0VfVEFHfT5gKSAmJlxuICAgICAgICAgICF0ZXh0LnN0YXJ0c1dpdGgoYDwke0NPTU1BTkRfTkFNRV9UQUd9PmApICYmXG4gICAgICAgICAgIXRleHQuc3RhcnRzV2l0aChgPCR7QkFTSF9JTlBVVF9UQUd9PmApXG4gICAgICAgICkge1xuICAgICAgICAgIGhhaWt1VGl0bGVBdHRlbXB0ZWRSZWYuY3VycmVudCA9IHRydWVcbiAgICAgICAgICB2b2lkIGdlbmVyYXRlU2Vzc2lvblRpdGxlKHRleHQsIG5ldyBBYm9ydENvbnRyb2xsZXIoKS5zaWduYWwpLnRoZW4oXG4gICAgICAgICAgICB0aXRsZSA9PiB7XG4gICAgICAgICAgICAgIGlmICh0aXRsZSkgc2V0SGFpa3VUaXRsZSh0aXRsZSlcbiAgICAgICAgICAgICAgZWxzZSBoYWlrdVRpdGxlQXR0ZW1wdGVkUmVmLmN1cnJlbnQgPSBmYWxzZVxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICgpID0+IHtcbiAgICAgICAgICAgICAgaGFpa3VUaXRsZUF0dGVtcHRlZFJlZi5jdXJyZW50ID0gZmFsc2VcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgKVxuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIEFwcGx5IHNsYXNoLWNvbW1hbmQtc2NvcGVkIGFsbG93ZWRUb29scyAoZnJvbSBza2lsbCBmcm9udG1hdHRlcikgdG8gdGhlXG4gICAgICAvLyBzdG9yZSBvbmNlIHBlciB0dXJuLiBUaGlzIGFsc28gY292ZXJzIHRoZSByZXNldDogdGhlIG5leHQgbm9uLXNraWxsIHR1cm5cbiAgICAgIC8vIHBhc3NlcyBbXSBhbmQgY2xlYXJzIGl0LiBNdXN0IHJ1biBiZWZvcmUgdGhlICFzaG91bGRRdWVyeSBnYXRlOiBmb3JrZWRcbiAgICAgIC8vIGNvbW1hbmRzIChleGVjdXRlRm9ya2VkU2xhc2hDb21tYW5kKSByZXR1cm4gc2hvdWxkUXVlcnk9ZmFsc2UsIGFuZFxuICAgICAgLy8gY3JlYXRlR2V0QXBwU3RhdGVXaXRoQWxsb3dlZFRvb2xzIGluIGZvcmtlZEFnZW50LnRzIHJlYWRzIHRoaXMgZmllbGQsIHNvXG4gICAgICAvLyBzdGFsZSBza2lsbCB0b29scyB3b3VsZCBvdGhlcndpc2UgbGVhayBpbnRvIGZvcmtlZCBhZ2VudCBwZXJtaXNzaW9ucy5cbiAgICAgIC8vIFByZXZpb3VzbHkgdGhpcyB3cml0ZSB3YXMgaGlkZGVuIGluc2lkZSBnZXRUb29sVXNlQ29udGV4dCdzIGdldEFwcFN0YXRlXG4gICAgICAvLyAofjg1IGNhbGxzL3R1cm4pOyBob2lzdGluZyBpdCBoZXJlIG1ha2VzIGdldEFwcFN0YXRlIGEgcHVyZSByZWFkIGFuZCBzdG9wc1xuICAgICAgLy8gZXBoZW1lcmFsIGNvbnRleHRzIChwZXJtaXNzaW9uIGRpYWxvZywgQmFja2dyb3VuZFRhc2tzRGlhbG9nKSBmcm9tXG4gICAgICAvLyBhY2NpZGVudGFsbHkgY2xlYXJpbmcgaXQgbWlkLXR1cm4uXG4gICAgICBzdG9yZS5zZXRTdGF0ZShwcmV2ID0+IHtcbiAgICAgICAgY29uc3QgY3VyID0gcHJldi50b29sUGVybWlzc2lvbkNvbnRleHQuYWx3YXlzQWxsb3dSdWxlcy5jb21tYW5kXG4gICAgICAgIGlmIChcbiAgICAgICAgICBjdXIgPT09IGFkZGl0aW9uYWxBbGxvd2VkVG9vbHMgfHxcbiAgICAgICAgICAoY3VyPy5sZW5ndGggPT09IGFkZGl0aW9uYWxBbGxvd2VkVG9vbHMubGVuZ3RoICYmXG4gICAgICAgICAgICBjdXIuZXZlcnkoKHYsIGkpID0+IHYgPT09IGFkZGl0aW9uYWxBbGxvd2VkVG9vbHNbaV0pKVxuICAgICAgICApIHtcbiAgICAgICAgICByZXR1cm4gcHJldlxuICAgICAgICB9XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICB0b29sUGVybWlzc2lvbkNvbnRleHQ6IHtcbiAgICAgICAgICAgIC4uLnByZXYudG9vbFBlcm1pc3Npb25Db250ZXh0LFxuICAgICAgICAgICAgYWx3YXlzQWxsb3dSdWxlczoge1xuICAgICAgICAgICAgICAuLi5wcmV2LnRvb2xQZXJtaXNzaW9uQ29udGV4dC5hbHdheXNBbGxvd1J1bGVzLFxuICAgICAgICAgICAgICBjb21tYW5kOiBhZGRpdGlvbmFsQWxsb3dlZFRvb2xzLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICB9LFxuICAgICAgICB9XG4gICAgICB9KVxuXG4gICAgICAvLyBUaGUgbGFzdCBtZXNzYWdlIGlzIGFuIGFzc2lzdGFudCBtZXNzYWdlIGlmIHRoZSB1c2VyIGlucHV0IHdhcyBhIGJhc2ggY29tbWFuZCxcbiAgICAgIC8vIG9yIGlmIHRoZSB1c2VyIGlucHV0IHdhcyBhbiBpbnZhbGlkIHNsYXNoIGNvbW1hbmQuXG4gICAgICBpZiAoIXNob3VsZFF1ZXJ5KSB7XG4gICAgICAgIC8vIE1hbnVhbCAvY29tcGFjdCBzZXRzIG1lc3NhZ2VzIGRpcmVjdGx5IChzaG91bGRRdWVyeT1mYWxzZSkgYnlwYXNzaW5nXG4gICAgICAgIC8vIGhhbmRsZU1lc3NhZ2VGcm9tU3RyZWFtLiBDbGVhciBjb250ZXh0LWJsb2NrZWQgaWYgYSBjb21wYWN0IGJvdW5kYXJ5XG4gICAgICAgIC8vIGlzIHByZXNlbnQgc28gcHJvYWN0aXZlIHRpY2tzIHJlc3VtZSBhZnRlciBjb21wYWN0aW9uLlxuICAgICAgICBpZiAobmV3TWVzc2FnZXMuc29tZShpc0NvbXBhY3RCb3VuZGFyeU1lc3NhZ2UpKSB7XG4gICAgICAgICAgLy8gQnVtcCBjb252ZXJzYXRpb25JZCBzbyBNZXNzYWdlcy50c3ggcm93IGtleXMgY2hhbmdlIGFuZFxuICAgICAgICAgIC8vIHN0YWxlIG1lbW9pemVkIHJvd3MgcmVtb3VudCB3aXRoIHBvc3QtY29tcGFjdCBjb250ZW50LlxuICAgICAgICAgIHNldENvbnZlcnNhdGlvbklkKHJhbmRvbVVVSUQoKSlcbiAgICAgICAgICBpZiAoZmVhdHVyZSgnUFJPQUNUSVZFJykgfHwgZmVhdHVyZSgnS0FJUk9TJykpIHtcbiAgICAgICAgICAgIHByb2FjdGl2ZU1vZHVsZT8uc2V0Q29udGV4dEJsb2NrZWQoZmFsc2UpXG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHJlc2V0TG9hZGluZ1N0YXRlKClcbiAgICAgICAgc2V0QWJvcnRDb250cm9sbGVyKG51bGwpXG4gICAgICAgIHJldHVyblxuICAgICAgfVxuXG4gICAgICBjb25zdCB0b29sVXNlQ29udGV4dCA9IGdldFRvb2xVc2VDb250ZXh0KFxuICAgICAgICBtZXNzYWdlc0luY2x1ZGluZ05ld01lc3NhZ2VzLFxuICAgICAgICBuZXdNZXNzYWdlcyxcbiAgICAgICAgYWJvcnRDb250cm9sbGVyLFxuICAgICAgICBtYWluTG9vcE1vZGVsUGFyYW0sXG4gICAgICApXG4gICAgICAvLyBnZXRUb29sVXNlQ29udGV4dCByZWFkcyB0b29scy9tY3BDbGllbnRzIGZyZXNoIGZyb20gc3RvcmUuZ2V0U3RhdGUoKVxuICAgICAgLy8gKHZpYSBjb21wdXRlVG9vbHMvbWVyZ2VDbGllbnRzKS4gVXNlIHRob3NlIHJhdGhlciB0aGFuIHRoZSBjbG9zdXJlLVxuICAgICAgLy8gY2FwdHVyZWQgYHRvb2xzYC9gbWNwQ2xpZW50c2Ag4oCUIHVzZU1hbmFnZU1DUENvbm5lY3Rpb25zIG1heSBoYXZlXG4gICAgICAvLyBmbHVzaGVkIG5ldyBNQ1Agc3RhdGUgYmV0d2VlbiB0aGUgcmVuZGVyIHRoYXQgY2FwdHVyZWQgdGhpcyBjbG9zdXJlXG4gICAgICAvLyBhbmQgbm93LiBUdXJuIDEgdmlhIHByb2Nlc3NJbml0aWFsTWVzc2FnZSBpcyB0aGUgbWFpbiBiZW5lZmljaWFyeS5cbiAgICAgIGNvbnN0IHsgdG9vbHM6IGZyZXNoVG9vbHMsIG1jcENsaWVudHM6IGZyZXNoTWNwQ2xpZW50cyB9ID1cbiAgICAgICAgdG9vbFVzZUNvbnRleHQub3B0aW9uc1xuXG4gICAgICAvLyBTY29wZSB0aGUgc2tpbGwncyBlZmZvcnQgb3ZlcnJpZGUgdG8gdGhpcyB0dXJuJ3MgY29udGV4dCBvbmx5IOKAlFxuICAgICAgLy8gd3JhcHBpbmcgZ2V0QXBwU3RhdGUga2VlcHMgdGhlIG92ZXJyaWRlIG91dCBvZiB0aGUgZ2xvYmFsIHN0b3JlIHNvXG4gICAgICAvLyBiYWNrZ3JvdW5kIGFnZW50cyBhbmQgVUkgc3Vic2NyaWJlcnMgKFNwaW5uZXIsIExvZ29WMikgbmV2ZXIgc2VlIGl0LlxuICAgICAgaWYgKGVmZm9ydCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgIGNvbnN0IHByZXZpb3VzR2V0QXBwU3RhdGUgPSB0b29sVXNlQ29udGV4dC5nZXRBcHBTdGF0ZVxuICAgICAgICB0b29sVXNlQ29udGV4dC5nZXRBcHBTdGF0ZSA9ICgpID0+ICh7XG4gICAgICAgICAgLi4ucHJldmlvdXNHZXRBcHBTdGF0ZSgpLFxuICAgICAgICAgIGVmZm9ydFZhbHVlOiBlZmZvcnQsXG4gICAgICAgIH0pXG4gICAgICB9XG5cbiAgICAgIHF1ZXJ5Q2hlY2twb2ludCgncXVlcnlfY29udGV4dF9sb2FkaW5nX3N0YXJ0JylcbiAgICAgIGNvbnN0IFssICwgZGVmYXVsdFN5c3RlbVByb21wdCwgYmFzZVVzZXJDb250ZXh0LCBzeXN0ZW1Db250ZXh0XSA9XG4gICAgICAgIGF3YWl0IFByb21pc2UuYWxsKFtcbiAgICAgICAgICAvLyBJTVBPUlRBTlQ6IGRvIHRoaXMgYWZ0ZXIgc2V0TWVzc2FnZXMoKSBhYm92ZSwgdG8gYXZvaWQgVUkgamFua1xuICAgICAgICAgIGNoZWNrQW5kRGlzYWJsZUJ5cGFzc1Blcm1pc3Npb25zSWZOZWVkZWQoXG4gICAgICAgICAgICB0b29sUGVybWlzc2lvbkNvbnRleHQsXG4gICAgICAgICAgICBzZXRBcHBTdGF0ZSxcbiAgICAgICAgICApLFxuICAgICAgICAgIC8vIEdhdGVkIG9uIFRSQU5TQ1JJUFRfQ0xBU1NJRklFUiBzbyBHcm93dGhCb29rIGtpbGwgc3dpdGNoIHJ1bnMgd2hlcmV2ZXIgYXV0byBtb2RlIGlzIGJ1aWx0IGluXG4gICAgICAgICAgZmVhdHVyZSgnVFJBTlNDUklQVF9DTEFTU0lGSUVSJylcbiAgICAgICAgICAgID8gY2hlY2tBbmREaXNhYmxlQXV0b01vZGVJZk5lZWRlZChcbiAgICAgICAgICAgICAgICB0b29sUGVybWlzc2lvbkNvbnRleHQsXG4gICAgICAgICAgICAgICAgc2V0QXBwU3RhdGUsXG4gICAgICAgICAgICAgICAgc3RvcmUuZ2V0U3RhdGUoKS5mYXN0TW9kZSxcbiAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgOiB1bmRlZmluZWQsXG4gICAgICAgICAgZ2V0U3lzdGVtUHJvbXB0KFxuICAgICAgICAgICAgZnJlc2hUb29scyxcbiAgICAgICAgICAgIG1haW5Mb29wTW9kZWxQYXJhbSxcbiAgICAgICAgICAgIEFycmF5LmZyb20oXG4gICAgICAgICAgICAgIHRvb2xQZXJtaXNzaW9uQ29udGV4dC5hZGRpdGlvbmFsV29ya2luZ0RpcmVjdG9yaWVzLmtleXMoKSxcbiAgICAgICAgICAgICksXG4gICAgICAgICAgICBmcmVzaE1jcENsaWVudHMsXG4gICAgICAgICAgKSxcbiAgICAgICAgICBnZXRVc2VyQ29udGV4dCgpLFxuICAgICAgICAgIGdldFN5c3RlbUNvbnRleHQoKSxcbiAgICAgICAgXSlcbiAgICAgIGNvbnN0IHVzZXJDb250ZXh0ID0ge1xuICAgICAgICAuLi5iYXNlVXNlckNvbnRleHQsXG4gICAgICAgIC4uLmdldENvb3JkaW5hdG9yVXNlckNvbnRleHQoXG4gICAgICAgICAgZnJlc2hNY3BDbGllbnRzLFxuICAgICAgICAgIGlzU2NyYXRjaHBhZEVuYWJsZWQoKSA/IGdldFNjcmF0Y2hwYWREaXIoKSA6IHVuZGVmaW5lZCxcbiAgICAgICAgKSxcbiAgICAgICAgLi4uKChmZWF0dXJlKCdQUk9BQ1RJVkUnKSB8fCBmZWF0dXJlKCdLQUlST1MnKSkgJiZcbiAgICAgICAgcHJvYWN0aXZlTW9kdWxlPy5pc1Byb2FjdGl2ZUFjdGl2ZSgpICYmXG4gICAgICAgICF0ZXJtaW5hbEZvY3VzUmVmLmN1cnJlbnRcbiAgICAgICAgICA/IHtcbiAgICAgICAgICAgICAgdGVybWluYWxGb2N1czpcbiAgICAgICAgICAgICAgICAnVGhlIHRlcm1pbmFsIGlzIHVuZm9jdXNlZCBcXHUyMDE0IHRoZSB1c2VyIGlzIG5vdCBhY3RpdmVseSB3YXRjaGluZy4nLFxuICAgICAgICAgICAgfVxuICAgICAgICAgIDoge30pLFxuICAgICAgfVxuICAgICAgcXVlcnlDaGVja3BvaW50KCdxdWVyeV9jb250ZXh0X2xvYWRpbmdfZW5kJylcblxuICAgICAgY29uc3Qgc3lzdGVtUHJvbXB0ID0gYnVpbGRFZmZlY3RpdmVTeXN0ZW1Qcm9tcHQoe1xuICAgICAgICBtYWluVGhyZWFkQWdlbnREZWZpbml0aW9uLFxuICAgICAgICB0b29sVXNlQ29udGV4dCxcbiAgICAgICAgY3VzdG9tU3lzdGVtUHJvbXB0LFxuICAgICAgICBkZWZhdWx0U3lzdGVtUHJvbXB0LFxuICAgICAgICBhcHBlbmRTeXN0ZW1Qcm9tcHQsXG4gICAgICB9KVxuICAgICAgdG9vbFVzZUNvbnRleHQucmVuZGVyZWRTeXN0ZW1Qcm9tcHQgPSBzeXN0ZW1Qcm9tcHRcblxuICAgICAgcXVlcnlDaGVja3BvaW50KCdxdWVyeV9xdWVyeV9zdGFydCcpXG4gICAgICByZXNldFR1cm5Ib29rRHVyYXRpb24oKVxuICAgICAgcmVzZXRUdXJuVG9vbER1cmF0aW9uKClcbiAgICAgIHJlc2V0VHVybkNsYXNzaWZpZXJEdXJhdGlvbigpXG5cbiAgICAgIGZvciBhd2FpdCAoY29uc3QgZXZlbnQgb2YgcXVlcnkoe1xuICAgICAgICBtZXNzYWdlczogbWVzc2FnZXNJbmNsdWRpbmdOZXdNZXNzYWdlcyxcbiAgICAgICAgc3lzdGVtUHJvbXB0LFxuICAgICAgICB1c2VyQ29udGV4dCxcbiAgICAgICAgc3lzdGVtQ29udGV4dCxcbiAgICAgICAgY2FuVXNlVG9vbCxcbiAgICAgICAgdG9vbFVzZUNvbnRleHQsXG4gICAgICAgIHF1ZXJ5U291cmNlOiBnZXRRdWVyeVNvdXJjZUZvclJFUEwoKSxcbiAgICAgIH0pKSB7XG4gICAgICAgIG9uUXVlcnlFdmVudChldmVudClcbiAgICAgIH1cblxuXG4gICAgICBpZiAoZmVhdHVyZSgnQlVERFknKSkge1xuICAgICAgICB2b2lkIGZpcmVDb21wYW5pb25PYnNlcnZlcihtZXNzYWdlc1JlZi5jdXJyZW50LCByZWFjdGlvbiA9PlxuICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT5cbiAgICAgICAgICAgIHByZXYuY29tcGFuaW9uUmVhY3Rpb24gPT09IHJlYWN0aW9uXG4gICAgICAgICAgICAgID8gcHJldlxuICAgICAgICAgICAgICA6IHsgLi4ucHJldiwgY29tcGFuaW9uUmVhY3Rpb246IHJlYWN0aW9uIH0sXG4gICAgICAgICAgKSxcbiAgICAgICAgKVxuICAgICAgfVxuXG4gICAgICBxdWVyeUNoZWNrcG9pbnQoJ3F1ZXJ5X2VuZCcpXG5cbiAgICAgIC8vIENhcHR1cmUgYW50LW9ubHkgQVBJIG1ldHJpY3MgYmVmb3JlIHJlc2V0TG9hZGluZ1N0YXRlIGNsZWFycyB0aGUgcmVmLlxuICAgICAgLy8gRm9yIG11bHRpLXJlcXVlc3QgdHVybnMgKHRvb2wgdXNlIGxvb3BzKSwgY29tcHV0ZSBQNTAgYWNyb3NzIGFsbCByZXF1ZXN0cy5cbiAgICAgIGlmIChcImV4dGVybmFsXCIgPT09ICdhbnQnICYmIGFwaU1ldHJpY3NSZWYuY3VycmVudC5sZW5ndGggPiAwKSB7XG4gICAgICAgIGNvbnN0IGVudHJpZXMgPSBhcGlNZXRyaWNzUmVmLmN1cnJlbnRcblxuICAgICAgICBjb25zdCB0dGZ0cyA9IGVudHJpZXMubWFwKGUgPT4gZS50dGZ0TXMpXG4gICAgICAgIC8vIENvbXB1dGUgcGVyLXJlcXVlc3QgT1RQUyB1c2luZyBvbmx5IGFjdGl2ZSBzdHJlYW1pbmcgdGltZSBhbmRcbiAgICAgICAgLy8gc3RyZWFtaW5nLW9ubHkgY29udGVudC4gZW5kUmVzcG9uc2VMZW5ndGggdHJhY2tzIGNvbnRlbnQgYWRkZWQgYnlcbiAgICAgICAgLy8gc3RyZWFtaW5nIGRlbHRhcyBvbmx5LCBleGNsdWRpbmcgc3ViYWdlbnQvY29tcGFjdGlvbiBpbmZsYXRpb24uXG4gICAgICAgIGNvbnN0IG90cHNWYWx1ZXMgPSBlbnRyaWVzLm1hcChlID0+IHtcbiAgICAgICAgICBjb25zdCBkZWx0YSA9IE1hdGgucm91bmQoXG4gICAgICAgICAgICAoZS5lbmRSZXNwb25zZUxlbmd0aCAtIGUucmVzcG9uc2VMZW5ndGhCYXNlbGluZSkgLyA0LFxuICAgICAgICAgIClcbiAgICAgICAgICBjb25zdCBzYW1wbGluZ01zID0gZS5sYXN0VG9rZW5UaW1lIC0gZS5maXJzdFRva2VuVGltZVxuICAgICAgICAgIHJldHVybiBzYW1wbGluZ01zID4gMCA/IE1hdGgucm91bmQoZGVsdGEgLyAoc2FtcGxpbmdNcyAvIDEwMDApKSA6IDBcbiAgICAgICAgfSlcblxuICAgICAgICBjb25zdCBpc011bHRpUmVxdWVzdCA9IGVudHJpZXMubGVuZ3RoID4gMVxuICAgICAgICBjb25zdCBob29rTXMgPSBnZXRUdXJuSG9va0R1cmF0aW9uTXMoKVxuICAgICAgICBjb25zdCBob29rQ291bnQgPSBnZXRUdXJuSG9va0NvdW50KClcbiAgICAgICAgY29uc3QgdG9vbE1zID0gZ2V0VHVyblRvb2xEdXJhdGlvbk1zKClcbiAgICAgICAgY29uc3QgdG9vbENvdW50ID0gZ2V0VHVyblRvb2xDb3VudCgpXG4gICAgICAgIGNvbnN0IGNsYXNzaWZpZXJNcyA9IGdldFR1cm5DbGFzc2lmaWVyRHVyYXRpb25NcygpXG4gICAgICAgIGNvbnN0IGNsYXNzaWZpZXJDb3VudCA9IGdldFR1cm5DbGFzc2lmaWVyQ291bnQoKVxuICAgICAgICBjb25zdCB0dXJuTXMgPSBEYXRlLm5vdygpIC0gbG9hZGluZ1N0YXJ0VGltZVJlZi5jdXJyZW50XG4gICAgICAgIHNldE1lc3NhZ2VzKHByZXYgPT4gW1xuICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgY3JlYXRlQXBpTWV0cmljc01lc3NhZ2Uoe1xuICAgICAgICAgICAgdHRmdE1zOiBpc011bHRpUmVxdWVzdCA/IG1lZGlhbih0dGZ0cykgOiB0dGZ0c1swXSEsXG4gICAgICAgICAgICBvdHBzOiBpc011bHRpUmVxdWVzdCA/IG1lZGlhbihvdHBzVmFsdWVzKSA6IG90cHNWYWx1ZXNbMF0hLFxuICAgICAgICAgICAgaXNQNTA6IGlzTXVsdGlSZXF1ZXN0LFxuICAgICAgICAgICAgaG9va0R1cmF0aW9uTXM6IGhvb2tNcyA+IDAgPyBob29rTXMgOiB1bmRlZmluZWQsXG4gICAgICAgICAgICBob29rQ291bnQ6IGhvb2tDb3VudCA+IDAgPyBob29rQ291bnQgOiB1bmRlZmluZWQsXG4gICAgICAgICAgICB0dXJuRHVyYXRpb25NczogdHVybk1zID4gMCA/IHR1cm5NcyA6IHVuZGVmaW5lZCxcbiAgICAgICAgICAgIHRvb2xEdXJhdGlvbk1zOiB0b29sTXMgPiAwID8gdG9vbE1zIDogdW5kZWZpbmVkLFxuICAgICAgICAgICAgdG9vbENvdW50OiB0b29sQ291bnQgPiAwID8gdG9vbENvdW50IDogdW5kZWZpbmVkLFxuICAgICAgICAgICAgY2xhc3NpZmllckR1cmF0aW9uTXM6IGNsYXNzaWZpZXJNcyA+IDAgPyBjbGFzc2lmaWVyTXMgOiB1bmRlZmluZWQsXG4gICAgICAgICAgICBjbGFzc2lmaWVyQ291bnQ6IGNsYXNzaWZpZXJDb3VudCA+IDAgPyBjbGFzc2lmaWVyQ291bnQgOiB1bmRlZmluZWQsXG4gICAgICAgICAgICBjb25maWdXcml0ZUNvdW50OiBnZXRHbG9iYWxDb25maWdXcml0ZUNvdW50KCksXG4gICAgICAgICAgfSksXG4gICAgICAgIF0pXG4gICAgICB9XG5cbiAgICAgIHJlc2V0TG9hZGluZ1N0YXRlKClcblxuICAgICAgLy8gTG9nIHF1ZXJ5IHByb2ZpbGluZyByZXBvcnQgaWYgZW5hYmxlZFxuICAgICAgbG9nUXVlcnlQcm9maWxlUmVwb3J0KClcblxuICAgICAgLy8gU2lnbmFsIHRoYXQgYSBxdWVyeSB0dXJuIGhhcyBjb21wbGV0ZWQgc3VjY2Vzc2Z1bGx5XG4gICAgICBhd2FpdCBvblR1cm5Db21wbGV0ZT8uKG1lc3NhZ2VzUmVmLmN1cnJlbnQpXG4gICAgfSxcbiAgICBbXG4gICAgICBpbml0aWFsTWNwQ2xpZW50cyxcbiAgICAgIHJlc2V0TG9hZGluZ1N0YXRlLFxuICAgICAgZ2V0VG9vbFVzZUNvbnRleHQsXG4gICAgICB0b29sUGVybWlzc2lvbkNvbnRleHQsXG4gICAgICBzZXRBcHBTdGF0ZSxcbiAgICAgIGN1c3RvbVN5c3RlbVByb21wdCxcbiAgICAgIG9uVHVybkNvbXBsZXRlLFxuICAgICAgYXBwZW5kU3lzdGVtUHJvbXB0LFxuICAgICAgY2FuVXNlVG9vbCxcbiAgICAgIG1haW5UaHJlYWRBZ2VudERlZmluaXRpb24sXG4gICAgICBvblF1ZXJ5RXZlbnQsXG4gICAgICBzZXNzaW9uVGl0bGUsXG4gICAgICB0aXRsZURpc2FibGVkLFxuICAgIF0sXG4gIClcblxuICBjb25zdCBvblF1ZXJ5ID0gdXNlQ2FsbGJhY2soXG4gICAgYXN5bmMgKFxuICAgICAgbmV3TWVzc2FnZXM6IE1lc3NhZ2VUeXBlW10sXG4gICAgICBhYm9ydENvbnRyb2xsZXI6IEFib3J0Q29udHJvbGxlcixcbiAgICAgIHNob3VsZFF1ZXJ5OiBib29sZWFuLFxuICAgICAgYWRkaXRpb25hbEFsbG93ZWRUb29sczogc3RyaW5nW10sXG4gICAgICBtYWluTG9vcE1vZGVsUGFyYW06IHN0cmluZyxcbiAgICAgIG9uQmVmb3JlUXVlcnlDYWxsYmFjaz86IChcbiAgICAgICAgaW5wdXQ6IHN0cmluZyxcbiAgICAgICAgbmV3TWVzc2FnZXM6IE1lc3NhZ2VUeXBlW10sXG4gICAgICApID0+IFByb21pc2U8Ym9vbGVhbj4sXG4gICAgICBpbnB1dD86IHN0cmluZyxcbiAgICAgIGVmZm9ydD86IEVmZm9ydFZhbHVlLFxuICAgICk6IFByb21pc2U8dm9pZD4gPT4ge1xuICAgICAgLy8gSWYgdGhpcyBpcyBhIHRlYW1tYXRlLCBtYXJrIHRoZW0gYXMgYWN0aXZlIHdoZW4gc3RhcnRpbmcgYSB0dXJuXG4gICAgICBpZiAoaXNBZ2VudFN3YXJtc0VuYWJsZWQoKSkge1xuICAgICAgICBjb25zdCB0ZWFtTmFtZSA9IGdldFRlYW1OYW1lKClcbiAgICAgICAgY29uc3QgYWdlbnROYW1lID0gZ2V0QWdlbnROYW1lKClcbiAgICAgICAgaWYgKHRlYW1OYW1lICYmIGFnZW50TmFtZSkge1xuICAgICAgICAgIC8vIEZpcmUgYW5kIGZvcmdldCAtIHR1cm4gc3RhcnRzIGltbWVkaWF0ZWx5LCB3cml0ZSBoYXBwZW5zIGluIGJhY2tncm91bmRcbiAgICAgICAgICB2b2lkIHNldE1lbWJlckFjdGl2ZSh0ZWFtTmFtZSwgYWdlbnROYW1lLCB0cnVlKVxuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIENvbmN1cnJlbnQgZ3VhcmQgdmlhIHN0YXRlIG1hY2hpbmUuIHRyeVN0YXJ0KCkgYXRvbWljYWxseSBjaGVja3NcbiAgICAgIC8vIGFuZCB0cmFuc2l0aW9ucyBpZGxl4oaScnVubmluZywgcmV0dXJuaW5nIHRoZSBnZW5lcmF0aW9uIG51bWJlci5cbiAgICAgIC8vIFJldHVybnMgbnVsbCBpZiBhbHJlYWR5IHJ1bm5pbmcg4oCUIG5vIHNlcGFyYXRlIGNoZWNrLXRoZW4tc2V0LlxuICAgICAgY29uc3QgdGhpc0dlbmVyYXRpb24gPSBxdWVyeUd1YXJkLnRyeVN0YXJ0KClcbiAgICAgIGlmICh0aGlzR2VuZXJhdGlvbiA9PT0gbnVsbCkge1xuICAgICAgICBsb2dFdmVudCgndGVuZ3VfY29uY3VycmVudF9vbnF1ZXJ5X2RldGVjdGVkJywge30pXG5cbiAgICAgICAgLy8gRXh0cmFjdCBhbmQgZW5xdWV1ZSB1c2VyIG1lc3NhZ2UgdGV4dCwgc2tpcHBpbmcgbWV0YSBtZXNzYWdlc1xuICAgICAgICAvLyAoZS5nLiBleHBhbmRlZCBza2lsbCBjb250ZW50LCB0aWNrIHByb21wdHMpIHRoYXQgc2hvdWxkIG5vdCBiZVxuICAgICAgICAvLyByZXBsYXllZCBhcyB1c2VyLXZpc2libGUgdGV4dC5cbiAgICAgICAgbmV3TWVzc2FnZXNcbiAgICAgICAgICAuZmlsdGVyKChtKTogbSBpcyBVc2VyTWVzc2FnZSA9PiBtLnR5cGUgPT09ICd1c2VyJyAmJiAhbS5pc01ldGEpXG4gICAgICAgICAgLm1hcChfID0+IGdldENvbnRlbnRUZXh0KF8ubWVzc2FnZS5jb250ZW50KSlcbiAgICAgICAgICAuZmlsdGVyKF8gPT4gXyAhPT0gbnVsbClcbiAgICAgICAgICAuZm9yRWFjaCgobXNnLCBpKSA9PiB7XG4gICAgICAgICAgICBlbnF1ZXVlKHsgdmFsdWU6IG1zZywgbW9kZTogJ3Byb21wdCcgfSlcbiAgICAgICAgICAgIGlmIChpID09PSAwKSB7XG4gICAgICAgICAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9jb25jdXJyZW50X29ucXVlcnlfZW5xdWV1ZWQnLCB7fSlcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9KVxuICAgICAgICByZXR1cm5cbiAgICAgIH1cblxuICAgICAgdHJ5IHtcbiAgICAgICAgLy8gaXNMb2FkaW5nIGlzIGRlcml2ZWQgZnJvbSBxdWVyeUd1YXJkIOKAlCB0cnlTdGFydCgpIGFib3ZlIGFscmVhZHlcbiAgICAgICAgLy8gdHJhbnNpdGlvbmVkIGRpc3BhdGNoaW5n4oaScnVubmluZywgc28gbm8gc2V0dGVyIGNhbGwgbmVlZGVkIGhlcmUuXG4gICAgICAgIHJlc2V0VGltaW5nUmVmcygpXG4gICAgICAgIHNldE1lc3NhZ2VzKG9sZE1lc3NhZ2VzID0+IFsuLi5vbGRNZXNzYWdlcywgLi4ubmV3TWVzc2FnZXNdKVxuICAgICAgICByZXNwb25zZUxlbmd0aFJlZi5jdXJyZW50ID0gMFxuICAgICAgICBpZiAoZmVhdHVyZSgnVE9LRU5fQlVER0VUJykpIHtcbiAgICAgICAgICBjb25zdCBwYXJzZWRCdWRnZXQgPSBpbnB1dCA/IHBhcnNlVG9rZW5CdWRnZXQoaW5wdXQpIDogbnVsbFxuICAgICAgICAgIHNuYXBzaG90T3V0cHV0VG9rZW5zRm9yVHVybihcbiAgICAgICAgICAgIHBhcnNlZEJ1ZGdldCA/PyBnZXRDdXJyZW50VHVyblRva2VuQnVkZ2V0KCksXG4gICAgICAgICAgKVxuICAgICAgICB9XG4gICAgICAgIGFwaU1ldHJpY3NSZWYuY3VycmVudCA9IFtdXG4gICAgICAgIHNldFN0cmVhbWluZ1Rvb2xVc2VzKFtdKVxuICAgICAgICBzZXRTdHJlYW1pbmdUZXh0KG51bGwpXG5cbiAgICAgICAgLy8gbWVzc2FnZXNSZWYgaXMgdXBkYXRlZCBzeW5jaHJvbm91c2x5IGJ5IHRoZSBzZXRNZXNzYWdlcyB3cmFwcGVyXG4gICAgICAgIC8vIGFib3ZlLCBzbyBpdCBhbHJlYWR5IGluY2x1ZGVzIG5ld01lc3NhZ2VzIGZyb20gdGhlIGFwcGVuZCBhdCB0aGVcbiAgICAgICAgLy8gdG9wIG9mIHRoaXMgdHJ5IGJsb2NrLiAgTm8gcmVjb25zdHJ1Y3Rpb24gbmVlZGVkLCBubyB3YWl0aW5nIGZvclxuICAgICAgICAvLyBSZWFjdCdzIHNjaGVkdWxlciAocHJldmlvdXNseSBjb3N0IDIwLTU2bXMgcGVyIHByb21wdDsgdGhlIDU2bXNcbiAgICAgICAgLy8gY2FzZSB3YXMgYSBHQyBwYXVzZSBjYXVnaHQgZHVyaW5nIHRoZSBhd2FpdCkuXG4gICAgICAgIGNvbnN0IGxhdGVzdE1lc3NhZ2VzID0gbWVzc2FnZXNSZWYuY3VycmVudFxuXG4gICAgICAgIGlmIChpbnB1dCkge1xuICAgICAgICAgIGF3YWl0IG1yT25CZWZvcmVRdWVyeShpbnB1dCwgbGF0ZXN0TWVzc2FnZXMsIG5ld01lc3NhZ2VzLmxlbmd0aClcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFBhc3MgZnVsbCBjb252ZXJzYXRpb24gaGlzdG9yeSB0byBjYWxsYmFja1xuICAgICAgICBpZiAob25CZWZvcmVRdWVyeUNhbGxiYWNrICYmIGlucHV0KSB7XG4gICAgICAgICAgY29uc3Qgc2hvdWxkUHJvY2VlZCA9IGF3YWl0IG9uQmVmb3JlUXVlcnlDYWxsYmFjayhcbiAgICAgICAgICAgIGlucHV0LFxuICAgICAgICAgICAgbGF0ZXN0TWVzc2FnZXMsXG4gICAgICAgICAgKVxuICAgICAgICAgIGlmICghc2hvdWxkUHJvY2VlZCkge1xuICAgICAgICAgICAgcmV0dXJuXG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgYXdhaXQgb25RdWVyeUltcGwoXG4gICAgICAgICAgbGF0ZXN0TWVzc2FnZXMsXG4gICAgICAgICAgbmV3TWVzc2FnZXMsXG4gICAgICAgICAgYWJvcnRDb250cm9sbGVyLFxuICAgICAgICAgIHNob3VsZFF1ZXJ5LFxuICAgICAgICAgIGFkZGl0aW9uYWxBbGxvd2VkVG9vbHMsXG4gICAgICAgICAgbWFpbkxvb3BNb2RlbFBhcmFtLFxuICAgICAgICAgIGVmZm9ydCxcbiAgICAgICAgKVxuICAgICAgfSBmaW5hbGx5IHtcbiAgICAgICAgLy8gcXVlcnlHdWFyZC5lbmQoKSBhdG9taWNhbGx5IGNoZWNrcyBnZW5lcmF0aW9uIGFuZCB0cmFuc2l0aW9uc1xuICAgICAgICAvLyBydW5uaW5n4oaSaWRsZS4gUmV0dXJucyBmYWxzZSBpZiBhIG5ld2VyIHF1ZXJ5IG93bnMgdGhlIGd1YXJkXG4gICAgICAgIC8vIChjYW5jZWwrcmVzdWJtaXQgcmFjZSB3aGVyZSB0aGUgc3RhbGUgZmluYWxseSBmaXJlcyBhcyBhIG1pY3JvdGFzaykuXG4gICAgICAgIGlmIChxdWVyeUd1YXJkLmVuZCh0aGlzR2VuZXJhdGlvbikpIHtcbiAgICAgICAgICBzZXRMYXN0UXVlcnlDb21wbGV0aW9uVGltZShEYXRlLm5vdygpKVxuICAgICAgICAgIHNraXBJZGxlQ2hlY2tSZWYuY3VycmVudCA9IGZhbHNlXG4gICAgICAgICAgLy8gQWx3YXlzIHJlc2V0IGxvYWRpbmcgc3RhdGUgaW4gZmluYWxseSAtIHRoaXMgZW5zdXJlcyBjbGVhbnVwIGV2ZW5cbiAgICAgICAgICAvLyBpZiBvblF1ZXJ5SW1wbCB0aHJvd3MuIG9uVHVybkNvbXBsZXRlIGlzIGNhbGxlZCBzZXBhcmF0ZWx5IGluXG4gICAgICAgICAgLy8gb25RdWVyeUltcGwgb25seSBvbiBzdWNjZXNzZnVsIGNvbXBsZXRpb24uXG4gICAgICAgICAgcmVzZXRMb2FkaW5nU3RhdGUoKVxuXG4gICAgICAgICAgYXdhaXQgbXJPblR1cm5Db21wbGV0ZShcbiAgICAgICAgICAgIG1lc3NhZ2VzUmVmLmN1cnJlbnQsXG4gICAgICAgICAgICBhYm9ydENvbnRyb2xsZXIuc2lnbmFsLmFib3J0ZWQsXG4gICAgICAgICAgKVxuXG4gICAgICAgICAgLy8gTm90aWZ5IGJyaWRnZSBjbGllbnRzIHRoYXQgdGhlIHR1cm4gaXMgY29tcGxldGUgc28gbW9iaWxlIGFwcHNcbiAgICAgICAgICAvLyBjYW4gc3RvcCB0aGUgc3BhcmsgYW5pbWF0aW9uIGFuZCBzaG93IHBvc3QtdHVybiBVSS5cbiAgICAgICAgICBzZW5kQnJpZGdlUmVzdWx0UmVmLmN1cnJlbnQoKVxuXG4gICAgICAgICAgLy8gQXV0by1oaWRlIHR1bmdzdGVuIHBhbmVsIGNvbnRlbnQgYXQgdHVybiBlbmQgKGFudC1vbmx5KSwgYnV0IGtlZXBcbiAgICAgICAgICAvLyB0dW5nc3RlbkFjdGl2ZVNlc3Npb24gc2V0IHNvIHRoZSBwaWxsIHN0YXlzIGluIHRoZSBmb290ZXIgYW5kIHRoZSB1c2VyXG4gICAgICAgICAgLy8gY2FuIHJlb3BlbiB0aGUgcGFuZWwuIEJhY2tncm91bmQgdG11eCB0YXNrcyAoZS5nLiAvaHVudGVyKSBydW4gZm9yXG4gICAgICAgICAgLy8gbWludXRlcyDigJQgd2lwaW5nIHRoZSBzZXNzaW9uIG1hZGUgdGhlIHBpbGwgZGlzYXBwZWFyIGVudGlyZWx5LCBmb3JjaW5nXG4gICAgICAgICAgLy8gdGhlIHVzZXIgdG8gcmUtaW52b2tlIFRtdXgganVzdCB0byBwZWVrLiBTa2lwIG9uIGFib3J0IHNvIHRoZSBwYW5lbFxuICAgICAgICAgIC8vIHN0YXlzIG9wZW4gZm9yIGluc3BlY3Rpb24gKG1hdGNoZXMgdGhlIHR1cm4tZHVyYXRpb24gZ3VhcmQgYmVsb3cpLlxuICAgICAgICAgIGlmIChcbiAgICAgICAgICAgIFwiZXh0ZXJuYWxcIiA9PT0gJ2FudCcgJiZcbiAgICAgICAgICAgICFhYm9ydENvbnRyb2xsZXIuc2lnbmFsLmFib3J0ZWRcbiAgICAgICAgICApIHtcbiAgICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgICAgICAgICAgICBpZiAocHJldi50dW5nc3RlbkFjdGl2ZVNlc3Npb24gPT09IHVuZGVmaW5lZCkgcmV0dXJuIHByZXZcbiAgICAgICAgICAgICAgaWYgKHByZXYudHVuZ3N0ZW5QYW5lbEF1dG9IaWRkZW4gPT09IHRydWUpIHJldHVybiBwcmV2XG4gICAgICAgICAgICAgIHJldHVybiB7IC4uLnByZXYsIHR1bmdzdGVuUGFuZWxBdXRvSGlkZGVuOiB0cnVlIH1cbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgfVxuXG4gICAgICAgICAgLy8gQ2FwdHVyZSBidWRnZXQgaW5mbyBiZWZvcmUgY2xlYXJpbmcgKGFudC1vbmx5KVxuICAgICAgICAgIGxldCBidWRnZXRJbmZvOlxuICAgICAgICAgICAgfCB7IHRva2VuczogbnVtYmVyOyBsaW1pdDogbnVtYmVyOyBudWRnZXM6IG51bWJlciB9XG4gICAgICAgICAgICB8IHVuZGVmaW5lZFxuICAgICAgICAgIGlmIChmZWF0dXJlKCdUT0tFTl9CVURHRVQnKSkge1xuICAgICAgICAgICAgaWYgKFxuICAgICAgICAgICAgICBnZXRDdXJyZW50VHVyblRva2VuQnVkZ2V0KCkgIT09IG51bGwgJiZcbiAgICAgICAgICAgICAgZ2V0Q3VycmVudFR1cm5Ub2tlbkJ1ZGdldCgpISA+IDAgJiZcbiAgICAgICAgICAgICAgIWFib3J0Q29udHJvbGxlci5zaWduYWwuYWJvcnRlZFxuICAgICAgICAgICAgKSB7XG4gICAgICAgICAgICAgIGJ1ZGdldEluZm8gPSB7XG4gICAgICAgICAgICAgICAgdG9rZW5zOiBnZXRUdXJuT3V0cHV0VG9rZW5zKCksXG4gICAgICAgICAgICAgICAgbGltaXQ6IGdldEN1cnJlbnRUdXJuVG9rZW5CdWRnZXQoKSEsXG4gICAgICAgICAgICAgICAgbnVkZ2VzOiBnZXRCdWRnZXRDb250aW51YXRpb25Db3VudCgpLFxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBzbmFwc2hvdE91dHB1dFRva2Vuc0ZvclR1cm4obnVsbClcbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyBBZGQgdHVybiBkdXJhdGlvbiBtZXNzYWdlIGZvciB0dXJucyBsb25nZXIgdGhhbiAzMHMgb3Igd2l0aCBhIGJ1ZGdldFxuICAgICAgICAgIC8vIFNraXAgaWYgdXNlciBhYm9ydGVkIG9yIGlmIGluIGxvb3AgbW9kZSAodG9vIG5vaXN5IGJldHdlZW4gdGlja3MpXG4gICAgICAgICAgLy8gRGVmZXIgaWYgc3dhcm0gdGVhbW1hdGVzIGFyZSBzdGlsbCBydW5uaW5nIChzaG93IHdoZW4gdGhleSBmaW5pc2gpXG4gICAgICAgICAgY29uc3QgdHVybkR1cmF0aW9uTXMgPVxuICAgICAgICAgICAgRGF0ZS5ub3coKSAtIGxvYWRpbmdTdGFydFRpbWVSZWYuY3VycmVudCAtIHRvdGFsUGF1c2VkTXNSZWYuY3VycmVudFxuICAgICAgICAgIGlmIChcbiAgICAgICAgICAgICh0dXJuRHVyYXRpb25NcyA+IDMwMDAwIHx8IGJ1ZGdldEluZm8gIT09IHVuZGVmaW5lZCkgJiZcbiAgICAgICAgICAgICFhYm9ydENvbnRyb2xsZXIuc2lnbmFsLmFib3J0ZWQgJiZcbiAgICAgICAgICAgICFwcm9hY3RpdmVBY3RpdmVcbiAgICAgICAgICApIHtcbiAgICAgICAgICAgIGNvbnN0IGhhc1J1bm5pbmdTd2FybUFnZW50cyA9IGdldEFsbEluUHJvY2Vzc1RlYW1tYXRlVGFza3MoXG4gICAgICAgICAgICAgIHN0b3JlLmdldFN0YXRlKCkudGFza3MsXG4gICAgICAgICAgICApLnNvbWUodCA9PiB0LnN0YXR1cyA9PT0gJ3J1bm5pbmcnKVxuICAgICAgICAgICAgaWYgKGhhc1J1bm5pbmdTd2FybUFnZW50cykge1xuICAgICAgICAgICAgICAvLyBPbmx5IHJlY29yZCBzdGFydCB0aW1lIG9uIHRoZSBmaXJzdCBkZWZlcnJlZCB0dXJuXG4gICAgICAgICAgICAgIGlmIChzd2FybVN0YXJ0VGltZVJlZi5jdXJyZW50ID09PSBudWxsKSB7XG4gICAgICAgICAgICAgICAgc3dhcm1TdGFydFRpbWVSZWYuY3VycmVudCA9IGxvYWRpbmdTdGFydFRpbWVSZWYuY3VycmVudFxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIC8vIEFsd2F5cyB1cGRhdGUgYnVkZ2V0IOKAlCBsYXRlciB0dXJucyBtYXkgY2FycnkgdGhlIGFjdHVhbCBidWRnZXRcbiAgICAgICAgICAgICAgaWYgKGJ1ZGdldEluZm8pIHtcbiAgICAgICAgICAgICAgICBzd2FybUJ1ZGdldEluZm9SZWYuY3VycmVudCA9IGJ1ZGdldEluZm9cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgc2V0TWVzc2FnZXMocHJldiA9PiBbXG4gICAgICAgICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICAgICAgICBjcmVhdGVUdXJuRHVyYXRpb25NZXNzYWdlKFxuICAgICAgICAgICAgICAgICAgdHVybkR1cmF0aW9uTXMsXG4gICAgICAgICAgICAgICAgICBidWRnZXRJbmZvLFxuICAgICAgICAgICAgICAgICAgY291bnQocHJldiwgaXNMb2dnYWJsZU1lc3NhZ2UpLFxuICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgIF0pXG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICAgIC8vIENsZWFyIHRoZSBjb250cm9sbGVyIHNvIENhbmNlbFJlcXVlc3RIYW5kbGVyJ3MgY2FuQ2FuY2VsUnVubmluZ1Rhc2tcbiAgICAgICAgICAvLyByZWFkcyBmYWxzZSBhdCB0aGUgaWRsZSBwcm9tcHQuIFdpdGhvdXQgdGhpcywgdGhlIHN0YWxlIG5vbi1hYm9ydGVkXG4gICAgICAgICAgLy8gY29udHJvbGxlciBtYWtlcyBjdHJsK2MgZmlyZSBvbkNhbmNlbCgpIChhYm9ydGluZyBub3RoaW5nKSBpbnN0ZWFkIG9mXG4gICAgICAgICAgLy8gcHJvcGFnYXRpbmcgdG8gdGhlIGRvdWJsZS1wcmVzcyBleGl0IGZsb3cuXG4gICAgICAgICAgc2V0QWJvcnRDb250cm9sbGVyKG51bGwpXG4gICAgICAgIH1cblxuICAgICAgICAvLyBBdXRvLXJlc3RvcmU6IGlmIHRoZSB1c2VyIGludGVycnVwdGVkIGJlZm9yZSBhbnkgbWVhbmluZ2Z1bCByZXNwb25zZVxuICAgICAgICAvLyBhcnJpdmVkLCByZXdpbmQgdGhlIGNvbnZlcnNhdGlvbiBhbmQgcmVzdG9yZSB0aGVpciBwcm9tcHQg4oCUIHNhbWUgYXNcbiAgICAgICAgLy8gb3BlbmluZyB0aGUgbWVzc2FnZSBzZWxlY3RvciBhbmQgcGlja2luZyB0aGUgbGFzdCBtZXNzYWdlLlxuICAgICAgICAvLyBUaGlzIHJ1bnMgT1VUU0lERSB0aGUgcXVlcnlHdWFyZC5lbmQoKSBjaGVjayBiZWNhdXNlIG9uQ2FuY2VsIGNhbGxzXG4gICAgICAgIC8vIGZvcmNlRW5kKCksIHdoaWNoIGJ1bXBzIHRoZSBnZW5lcmF0aW9uIHNvIGVuZCgpIHJldHVybnMgZmFsc2UgYWJvdmUuXG4gICAgICAgIC8vIEd1YXJkczogcmVhc29uID09PSAndXNlci1jYW5jZWwnIChvbkNhbmNlbC9Fc2M7IHByb2dyYW1tYXRpYyBhYm9ydHNcbiAgICAgICAgLy8gdXNlICdiYWNrZ3JvdW5kJy8naW50ZXJydXB0JyBhbmQgbXVzdCBub3QgcmV3aW5kIOKAlCBub3RlIGFib3J0KCkgd2l0aFxuICAgICAgICAvLyBubyBhcmdzIHNldHMgcmVhc29uIHRvIGEgRE9NRXhjZXB0aW9uLCBub3QgdW5kZWZpbmVkKSwgIWlzQWN0aXZlIChub1xuICAgICAgICAvLyBuZXdlciBxdWVyeSBzdGFydGVkIOKAlCBjYW5jZWwrcmVzdWJtaXQgcmFjZSksIGVtcHR5IGlucHV0IChkb24ndFxuICAgICAgICAvLyBjbG9iYmVyIHRleHQgdHlwZWQgZHVyaW5nIGxvYWRpbmcpLCBubyBxdWV1ZWQgY29tbWFuZHMgKHVzZXIgcXVldWVkXG4gICAgICAgIC8vIEIgd2hpbGUgQSB3YXMgbG9hZGluZyDihpIgdGhleSd2ZSBtb3ZlZCBvbiwgZG9uJ3QgcmVzdG9yZSBBOyBhbHNvXG4gICAgICAgIC8vIGF2b2lkcyByZW1vdmVMYXN0RnJvbUhpc3RvcnkgcmVtb3ZpbmcgQidzIGVudHJ5IGluc3RlYWQgb2YgQSdzKSxcbiAgICAgICAgLy8gbm90IHZpZXdpbmcgYSB0ZWFtbWF0ZSAobWVzc2FnZXNSZWYgaXMgdGhlIG1haW4gY29udmVyc2F0aW9uIOKAlCB0aGVcbiAgICAgICAgLy8gb2xkIFVwLWFycm93IHF1aWNrLXJlc3RvcmUgaGFkIHRoaXMgZ3VhcmQsIHByZXNlcnZlIGl0KS5cbiAgICAgICAgaWYgKFxuICAgICAgICAgIGFib3J0Q29udHJvbGxlci5zaWduYWwucmVhc29uID09PSAndXNlci1jYW5jZWwnICYmXG4gICAgICAgICAgIXF1ZXJ5R3VhcmQuaXNBY3RpdmUgJiZcbiAgICAgICAgICBpbnB1dFZhbHVlUmVmLmN1cnJlbnQgPT09ICcnICYmXG4gICAgICAgICAgZ2V0Q29tbWFuZFF1ZXVlTGVuZ3RoKCkgPT09IDAgJiZcbiAgICAgICAgICAhc3RvcmUuZ2V0U3RhdGUoKS52aWV3aW5nQWdlbnRUYXNrSWRcbiAgICAgICAgKSB7XG4gICAgICAgICAgY29uc3QgbXNncyA9IG1lc3NhZ2VzUmVmLmN1cnJlbnRcbiAgICAgICAgICBjb25zdCBsYXN0VXNlck1zZyA9IG1zZ3MuZmluZExhc3Qoc2VsZWN0YWJsZVVzZXJNZXNzYWdlc0ZpbHRlcilcbiAgICAgICAgICBpZiAobGFzdFVzZXJNc2cpIHtcbiAgICAgICAgICAgIGNvbnN0IGlkeCA9IG1zZ3MubGFzdEluZGV4T2YobGFzdFVzZXJNc2cpXG4gICAgICAgICAgICBpZiAobWVzc2FnZXNBZnRlckFyZU9ubHlTeW50aGV0aWMobXNncywgaWR4KSkge1xuICAgICAgICAgICAgICAvLyBUaGUgc3VibWl0IGlzIGJlaW5nIHVuZG9uZSDigJQgdW5kbyBpdHMgaGlzdG9yeSBlbnRyeSB0b28sXG4gICAgICAgICAgICAgIC8vIG90aGVyd2lzZSBVcC1hcnJvdyBzaG93cyB0aGUgcmVzdG9yZWQgdGV4dCB0d2ljZS5cbiAgICAgICAgICAgICAgcmVtb3ZlTGFzdEZyb21IaXN0b3J5KClcbiAgICAgICAgICAgICAgcmVzdG9yZU1lc3NhZ2VTeW5jUmVmLmN1cnJlbnQobGFzdFVzZXJNc2cpXG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9XG4gICAgfSxcbiAgICBbXG4gICAgICBvblF1ZXJ5SW1wbCxcbiAgICAgIHNldEFwcFN0YXRlLFxuICAgICAgcmVzZXRMb2FkaW5nU3RhdGUsXG4gICAgICBxdWVyeUd1YXJkLFxuICAgICAgbXJPbkJlZm9yZVF1ZXJ5LFxuICAgICAgbXJPblR1cm5Db21wbGV0ZSxcbiAgICBdLFxuICApXG5cbiAgLy8gSGFuZGxlIGluaXRpYWwgbWVzc2FnZSAoZnJvbSBDTEkgYXJncyBvciBwbGFuIG1vZGUgZXhpdCB3aXRoIGNvbnRleHQgY2xlYXIpXG4gIC8vIFRoaXMgZWZmZWN0IHJ1bnMgd2hlbiBpc0xvYWRpbmcgYmVjb21lcyBmYWxzZSBhbmQgdGhlcmUncyBhIHBlbmRpbmcgbWVzc2FnZVxuICBjb25zdCBpbml0aWFsTWVzc2FnZVJlZiA9IHVzZVJlZihmYWxzZSlcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBjb25zdCBwZW5kaW5nID0gaW5pdGlhbE1lc3NhZ2VcbiAgICBpZiAoIXBlbmRpbmcgfHwgaXNMb2FkaW5nIHx8IGluaXRpYWxNZXNzYWdlUmVmLmN1cnJlbnQpIHJldHVyblxuXG4gICAgLy8gTWFyayBhcyBwcm9jZXNzaW5nIHRvIHByZXZlbnQgcmUtZW50cnlcbiAgICBpbml0aWFsTWVzc2FnZVJlZi5jdXJyZW50ID0gdHJ1ZVxuXG4gICAgYXN5bmMgZnVuY3Rpb24gcHJvY2Vzc0luaXRpYWxNZXNzYWdlKFxuICAgICAgaW5pdGlhbE1zZzogTm9uTnVsbGFibGU8dHlwZW9mIHBlbmRpbmc+LFxuICAgICkge1xuICAgICAgLy8gQ2xlYXIgY29udGV4dCBpZiByZXF1ZXN0ZWQgKHBsYW4gbW9kZSBleGl0KVxuICAgICAgaWYgKGluaXRpYWxNc2cuY2xlYXJDb250ZXh0KSB7XG4gICAgICAgIC8vIFByZXNlcnZlIHRoZSBwbGFuIHNsdWcgYmVmb3JlIGNsZWFyaW5nIGNvbnRleHQsIHNvIHRoZSBuZXcgc2Vzc2lvblxuICAgICAgICAvLyBjYW4gYWNjZXNzIHRoZSBzYW1lIHBsYW4gZmlsZSBhZnRlciByZWdlbmVyYXRlU2Vzc2lvbklkKClcbiAgICAgICAgY29uc3Qgb2xkUGxhblNsdWcgPSBpbml0aWFsTXNnLm1lc3NhZ2UucGxhbkNvbnRlbnRcbiAgICAgICAgICA/IGdldFBsYW5TbHVnKClcbiAgICAgICAgICA6IHVuZGVmaW5lZFxuXG4gICAgICAgIGNvbnN0IHsgY2xlYXJDb252ZXJzYXRpb24gfSA9IGF3YWl0IGltcG9ydChcbiAgICAgICAgICAnLi4vY29tbWFuZHMvY2xlYXIvY29udmVyc2F0aW9uLmpzJ1xuICAgICAgICApXG4gICAgICAgIGF3YWl0IGNsZWFyQ29udmVyc2F0aW9uKHtcbiAgICAgICAgICBzZXRNZXNzYWdlcyxcbiAgICAgICAgICByZWFkRmlsZVN0YXRlOiByZWFkRmlsZVN0YXRlLmN1cnJlbnQsXG4gICAgICAgICAgZGlzY292ZXJlZFNraWxsTmFtZXM6IGRpc2NvdmVyZWRTa2lsbE5hbWVzUmVmLmN1cnJlbnQsXG4gICAgICAgICAgbG9hZGVkTmVzdGVkTWVtb3J5UGF0aHM6IGxvYWRlZE5lc3RlZE1lbW9yeVBhdGhzUmVmLmN1cnJlbnQsXG4gICAgICAgICAgZ2V0QXBwU3RhdGU6ICgpID0+IHN0b3JlLmdldFN0YXRlKCksXG4gICAgICAgICAgc2V0QXBwU3RhdGUsXG4gICAgICAgICAgc2V0Q29udmVyc2F0aW9uSWQsXG4gICAgICAgIH0pXG4gICAgICAgIGhhaWt1VGl0bGVBdHRlbXB0ZWRSZWYuY3VycmVudCA9IGZhbHNlXG4gICAgICAgIHNldEhhaWt1VGl0bGUodW5kZWZpbmVkKVxuICAgICAgICBiYXNoVG9vbHMuY3VycmVudC5jbGVhcigpXG4gICAgICAgIGJhc2hUb29sc1Byb2Nlc3NlZElkeC5jdXJyZW50ID0gMFxuXG4gICAgICAgIC8vIFJlc3RvcmUgdGhlIHBsYW4gc2x1ZyBmb3IgdGhlIG5ldyBzZXNzaW9uIHNvIGdldFBsYW4oKSBmaW5kcyB0aGUgZmlsZVxuICAgICAgICBpZiAob2xkUGxhblNsdWcpIHtcbiAgICAgICAgICBzZXRQbGFuU2x1ZyhnZXRTZXNzaW9uSWQoKSwgb2xkUGxhblNsdWcpXG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgLy8gQXRvbWljYWxseTogY2xlYXIgaW5pdGlhbCBtZXNzYWdlLCBzZXQgcGVybWlzc2lvbiBtb2RlIGFuZCBydWxlcywgYW5kIHN0b3JlIHBsYW4gZm9yIHZlcmlmaWNhdGlvblxuICAgICAgY29uc3Qgc2hvdWxkU3RvcmVQbGFuRm9yVmVyaWZpY2F0aW9uID1cbiAgICAgICAgaW5pdGlhbE1zZy5tZXNzYWdlLnBsYW5Db250ZW50ICYmXG4gICAgICAgIFwiZXh0ZXJuYWxcIiA9PT0gJ2FudCcgJiZcbiAgICAgICAgaXNFbnZUcnV0aHkodW5kZWZpbmVkKVxuXG4gICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+IHtcbiAgICAgICAgLy8gQnVpbGQgYW5kIGFwcGx5IHBlcm1pc3Npb24gdXBkYXRlcyAobW9kZSArIGFsbG93ZWRQcm9tcHRzIHJ1bGVzKVxuICAgICAgICBsZXQgdXBkYXRlZFRvb2xQZXJtaXNzaW9uQ29udGV4dCA9IGluaXRpYWxNc2cubW9kZVxuICAgICAgICAgID8gYXBwbHlQZXJtaXNzaW9uVXBkYXRlcyhcbiAgICAgICAgICAgICAgcHJldi50b29sUGVybWlzc2lvbkNvbnRleHQsXG4gICAgICAgICAgICAgIGJ1aWxkUGVybWlzc2lvblVwZGF0ZXMoXG4gICAgICAgICAgICAgICAgaW5pdGlhbE1zZy5tb2RlLFxuICAgICAgICAgICAgICAgIGluaXRpYWxNc2cuYWxsb3dlZFByb21wdHMsXG4gICAgICAgICAgICAgICksXG4gICAgICAgICAgICApXG4gICAgICAgICAgOiBwcmV2LnRvb2xQZXJtaXNzaW9uQ29udGV4dFxuICAgICAgICAvLyBGb3IgYXV0bywgb3ZlcnJpZGUgdGhlIG1vZGUgKGJ1aWxkUGVybWlzc2lvblVwZGF0ZXMgbWFwc1xuICAgICAgICAvLyBpdCB0byAnZGVmYXVsdCcgdmlhIHRvRXh0ZXJuYWxQZXJtaXNzaW9uTW9kZSkgYW5kIHN0cmlwIGRhbmdlcm91cyBydWxlc1xuICAgICAgICBpZiAoZmVhdHVyZSgnVFJBTlNDUklQVF9DTEFTU0lGSUVSJykgJiYgaW5pdGlhbE1zZy5tb2RlID09PSAnYXV0bycpIHtcbiAgICAgICAgICB1cGRhdGVkVG9vbFBlcm1pc3Npb25Db250ZXh0ID0gc3RyaXBEYW5nZXJvdXNQZXJtaXNzaW9uc0ZvckF1dG9Nb2RlKHtcbiAgICAgICAgICAgIC4uLnVwZGF0ZWRUb29sUGVybWlzc2lvbkNvbnRleHQsXG4gICAgICAgICAgICBtb2RlOiAnYXV0bycsXG4gICAgICAgICAgICBwcmVQbGFuTW9kZTogdW5kZWZpbmVkLFxuICAgICAgICAgIH0pXG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgaW5pdGlhbE1lc3NhZ2U6IG51bGwsXG4gICAgICAgICAgdG9vbFBlcm1pc3Npb25Db250ZXh0OiB1cGRhdGVkVG9vbFBlcm1pc3Npb25Db250ZXh0LFxuICAgICAgICAgIC4uLihzaG91bGRTdG9yZVBsYW5Gb3JWZXJpZmljYXRpb24gJiYge1xuICAgICAgICAgICAgcGVuZGluZ1BsYW5WZXJpZmljYXRpb246IHtcbiAgICAgICAgICAgICAgcGxhbjogaW5pdGlhbE1zZy5tZXNzYWdlLnBsYW5Db250ZW50ISxcbiAgICAgICAgICAgICAgdmVyaWZpY2F0aW9uU3RhcnRlZDogZmFsc2UsXG4gICAgICAgICAgICAgIHZlcmlmaWNhdGlvbkNvbXBsZXRlZDogZmFsc2UsXG4gICAgICAgICAgICB9LFxuICAgICAgICAgIH0pLFxuICAgICAgICB9XG4gICAgICB9KVxuXG4gICAgICAvLyBDcmVhdGUgZmlsZSBoaXN0b3J5IHNuYXBzaG90IGZvciBjb2RlIHJld2luZFxuICAgICAgaWYgKGZpbGVIaXN0b3J5RW5hYmxlZCgpKSB7XG4gICAgICAgIHZvaWQgZmlsZUhpc3RvcnlNYWtlU25hcHNob3QoXG4gICAgICAgICAgKHVwZGF0ZXI6IChwcmV2OiBGaWxlSGlzdG9yeVN0YXRlKSA9PiBGaWxlSGlzdG9yeVN0YXRlKSA9PiB7XG4gICAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+ICh7XG4gICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgIGZpbGVIaXN0b3J5OiB1cGRhdGVyKHByZXYuZmlsZUhpc3RvcnkpLFxuICAgICAgICAgICAgfSkpXG4gICAgICAgICAgfSxcbiAgICAgICAgICBpbml0aWFsTXNnLm1lc3NhZ2UudXVpZCxcbiAgICAgICAgKVxuICAgICAgfVxuXG4gICAgICAvLyBFbnN1cmUgU2Vzc2lvblN0YXJ0IGhvb2sgY29udGV4dCBpcyBhdmFpbGFibGUgYmVmb3JlIHRoZSBmaXJzdCBBUElcbiAgICAgIC8vIGNhbGwuIG9uU3VibWl0IGNhbGxzIHRoaXMgaW50ZXJuYWxseSBidXQgdGhlIG9uUXVlcnkgcGF0aCBiZWxvd1xuICAgICAgLy8gYnlwYXNzZXMgb25TdWJtaXQg4oCUIGhvaXN0IGhlcmUgc28gYm90aCBwYXRocyBzZWUgaG9vayBtZXNzYWdlcy5cbiAgICAgIGF3YWl0IGF3YWl0UGVuZGluZ0hvb2tzKClcblxuICAgICAgLy8gUm91dGUgYWxsIGluaXRpYWwgcHJvbXB0cyB0aHJvdWdoIG9uU3VibWl0IHRvIGVuc3VyZSBVc2VyUHJvbXB0U3VibWl0IGhvb2tzIGZpcmVcbiAgICAgIC8vIFRPRE86IFNpbXBsaWZ5IGJ5IGFsd2F5cyByb3V0aW5nIHRocm91Z2ggb25TdWJtaXQgb25jZSBpdCBzdXBwb3J0c1xuICAgICAgLy8gQ29udGVudEJsb2NrUGFyYW0gYXJyYXlzIChpbWFnZXMpIGFzIGlucHV0XG4gICAgICBjb25zdCBjb250ZW50ID0gaW5pdGlhbE1zZy5tZXNzYWdlLm1lc3NhZ2UuY29udGVudFxuXG4gICAgICAvLyBSb3V0ZSBhbGwgc3RyaW5nIGNvbnRlbnQgdGhyb3VnaCBvblN1Ym1pdCB0byBlbnN1cmUgaG9va3MgZmlyZVxuICAgICAgLy8gRm9yIGNvbXBsZXggY29udGVudCAoaW1hZ2VzLCBldGMuKSwgZmFsbCBiYWNrIHRvIGRpcmVjdCBvblF1ZXJ5XG4gICAgICAvLyBQbGFuIG1lc3NhZ2VzIGJ5cGFzcyBvblN1Ym1pdCB0byBwcmVzZXJ2ZSBwbGFuQ29udGVudCBtZXRhZGF0YSBmb3IgcmVuZGVyaW5nXG4gICAgICBpZiAodHlwZW9mIGNvbnRlbnQgPT09ICdzdHJpbmcnICYmICFpbml0aWFsTXNnLm1lc3NhZ2UucGxhbkNvbnRlbnQpIHtcbiAgICAgICAgLy8gUm91dGUgdGhyb3VnaCBvblN1Ym1pdCBmb3IgcHJvcGVyIHByb2Nlc3NpbmcgaW5jbHVkaW5nIFVzZXJQcm9tcHRTdWJtaXQgaG9va3NcbiAgICAgICAgdm9pZCBvblN1Ym1pdChjb250ZW50LCB7XG4gICAgICAgICAgc2V0Q3Vyc29yT2Zmc2V0OiAoKSA9PiB7fSxcbiAgICAgICAgICBjbGVhckJ1ZmZlcjogKCkgPT4ge30sXG4gICAgICAgICAgcmVzZXRIaXN0b3J5OiAoKSA9PiB7fSxcbiAgICAgICAgfSlcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIFBsYW4gbWVzc2FnZXMgb3IgY29tcGxleCBjb250ZW50IChpbWFnZXMsIGV0Yy4pIC0gc2VuZCBkaXJlY3RseSB0byBtb2RlbFxuICAgICAgICAvLyBQbGFuIG1lc3NhZ2VzIHVzZSBvblF1ZXJ5IHRvIHByZXNlcnZlIHBsYW5Db250ZW50IG1ldGFkYXRhIGZvciByZW5kZXJpbmdcbiAgICAgICAgLy8gVE9ETzogT25jZSBvblN1Ym1pdCBzdXBwb3J0cyBDb250ZW50QmxvY2tQYXJhbSBhcnJheXMsIHJlbW92ZSB0aGlzIGJyYW5jaFxuICAgICAgICBjb25zdCBuZXdBYm9ydENvbnRyb2xsZXIgPSBjcmVhdGVBYm9ydENvbnRyb2xsZXIoKVxuICAgICAgICBzZXRBYm9ydENvbnRyb2xsZXIobmV3QWJvcnRDb250cm9sbGVyKVxuXG4gICAgICAgIHZvaWQgb25RdWVyeShcbiAgICAgICAgICBbaW5pdGlhbE1zZy5tZXNzYWdlXSxcbiAgICAgICAgICBuZXdBYm9ydENvbnRyb2xsZXIsXG4gICAgICAgICAgdHJ1ZSwgLy8gc2hvdWxkUXVlcnlcbiAgICAgICAgICBbXSwgLy8gYWRkaXRpb25hbEFsbG93ZWRUb29sc1xuICAgICAgICAgIG1haW5Mb29wTW9kZWwsXG4gICAgICAgIClcbiAgICAgIH1cblxuICAgICAgLy8gUmVzZXQgcmVmIGFmdGVyIGEgZGVsYXkgdG8gYWxsb3cgbmV3IGluaXRpYWwgbWVzc2FnZXNcbiAgICAgIHNldFRpbWVvdXQoXG4gICAgICAgIHJlZiA9PiB7XG4gICAgICAgICAgcmVmLmN1cnJlbnQgPSBmYWxzZVxuICAgICAgICB9LFxuICAgICAgICAxMDAsXG4gICAgICAgIGluaXRpYWxNZXNzYWdlUmVmLFxuICAgICAgKVxuICAgIH1cblxuICAgIHZvaWQgcHJvY2Vzc0luaXRpYWxNZXNzYWdlKHBlbmRpbmcpXG4gIH0sIFtcbiAgICBpbml0aWFsTWVzc2FnZSxcbiAgICBpc0xvYWRpbmcsXG4gICAgc2V0TWVzc2FnZXMsXG4gICAgc2V0QXBwU3RhdGUsXG4gICAgb25RdWVyeSxcbiAgICBtYWluTG9vcE1vZGVsLFxuICAgIHRvb2xzLFxuICBdKVxuXG4gIGNvbnN0IG9uU3VibWl0ID0gdXNlQ2FsbGJhY2soXG4gICAgYXN5bmMgKFxuICAgICAgaW5wdXQ6IHN0cmluZyxcbiAgICAgIGhlbHBlcnM6IFByb21wdElucHV0SGVscGVycyxcbiAgICAgIHNwZWN1bGF0aW9uQWNjZXB0Pzoge1xuICAgICAgICBzdGF0ZTogQWN0aXZlU3BlY3VsYXRpb25TdGF0ZVxuICAgICAgICBzcGVjdWxhdGlvblNlc3Npb25UaW1lU2F2ZWRNczogbnVtYmVyXG4gICAgICAgIHNldEFwcFN0YXRlOiBTZXRBcHBTdGF0ZVxuICAgICAgfSxcbiAgICAgIG9wdGlvbnM/OiB7IGZyb21LZXliaW5kaW5nPzogYm9vbGVhbiB9LFxuICAgICkgPT4ge1xuICAgICAgLy8gUmUtcGluIHNjcm9sbCB0byBib3R0b20gb24gc3VibWl0IHNvIHRoZSB1c2VyIGFsd2F5cyBzZWVzIHRoZSBuZXdcbiAgICAgIC8vIGV4Y2hhbmdlIChtYXRjaGVzIE9wZW5Db2RlJ3MgYXV0by1zY3JvbGwgYmVoYXZpb3IpLlxuICAgICAgcmVwaW5TY3JvbGwoKVxuXG4gICAgICAvLyBSZXN1bWUgbG9vcCBtb2RlIGlmIHBhdXNlZFxuICAgICAgaWYgKGZlYXR1cmUoJ1BST0FDVElWRScpIHx8IGZlYXR1cmUoJ0tBSVJPUycpKSB7XG4gICAgICAgIHByb2FjdGl2ZU1vZHVsZT8ucmVzdW1lUHJvYWN0aXZlKClcbiAgICAgIH1cblxuICAgICAgLy8gSGFuZGxlIGltbWVkaWF0ZSBjb21tYW5kcyAtIHRoZXNlIGJ5cGFzcyB0aGUgcXVldWUgYW5kIGV4ZWN1dGUgcmlnaHQgYXdheVxuICAgICAgLy8gZXZlbiB3aGlsZSBDbGF1ZGUgaXMgcHJvY2Vzc2luZy4gQ29tbWFuZHMgb3B0LWluIHZpYSBgaW1tZWRpYXRlOiB0cnVlYC5cbiAgICAgIC8vIENvbW1hbmRzIHRyaWdnZXJlZCB2aWEga2V5YmluZGluZ3MgYXJlIGFsd2F5cyB0cmVhdGVkIGFzIGltbWVkaWF0ZS5cbiAgICAgIGlmICghc3BlY3VsYXRpb25BY2NlcHQgJiYgaW5wdXQudHJpbSgpLnN0YXJ0c1dpdGgoJy8nKSkge1xuICAgICAgICAvLyBFeHBhbmQgW1Bhc3RlZCB0ZXh0ICNOXSByZWZzIHNvIGltbWVkaWF0ZSBjb21tYW5kcyAoZS5nLiAvYnR3KSByZWNlaXZlXG4gICAgICAgIC8vIHRoZSBwYXN0ZWQgY29udGVudCwgbm90IHRoZSBwbGFjZWhvbGRlci4gVGhlIG5vbi1pbW1lZGlhdGUgcGF0aCBnZXRzXG4gICAgICAgIC8vIHRoaXMgZXhwYW5zaW9uIGxhdGVyIGluIGhhbmRsZVByb21wdFN1Ym1pdC5cbiAgICAgICAgY29uc3QgdHJpbW1lZElucHV0ID0gZXhwYW5kUGFzdGVkVGV4dFJlZnMoaW5wdXQsIHBhc3RlZENvbnRlbnRzKS50cmltKClcbiAgICAgICAgY29uc3Qgc3BhY2VJbmRleCA9IHRyaW1tZWRJbnB1dC5pbmRleE9mKCcgJylcbiAgICAgICAgY29uc3QgY29tbWFuZE5hbWUgPVxuICAgICAgICAgIHNwYWNlSW5kZXggPT09IC0xXG4gICAgICAgICAgICA/IHRyaW1tZWRJbnB1dC5zbGljZSgxKVxuICAgICAgICAgICAgOiB0cmltbWVkSW5wdXQuc2xpY2UoMSwgc3BhY2VJbmRleClcbiAgICAgICAgY29uc3QgY29tbWFuZEFyZ3MgPVxuICAgICAgICAgIHNwYWNlSW5kZXggPT09IC0xID8gJycgOiB0cmltbWVkSW5wdXQuc2xpY2Uoc3BhY2VJbmRleCArIDEpLnRyaW0oKVxuXG4gICAgICAgIC8vIEZpbmQgbWF0Y2hpbmcgY29tbWFuZCAtIHRyZWF0IGFzIGltbWVkaWF0ZSBpZjpcbiAgICAgICAgLy8gMS4gQ29tbWFuZCBoYXMgYGltbWVkaWF0ZTogdHJ1ZWAsIE9SXG4gICAgICAgIC8vIDIuIENvbW1hbmQgd2FzIHRyaWdnZXJlZCB2aWEga2V5YmluZGluZyAoZnJvbUtleWJpbmRpbmcgb3B0aW9uKVxuICAgICAgICBjb25zdCBtYXRjaGluZ0NvbW1hbmQgPSBjb21tYW5kcy5maW5kKFxuICAgICAgICAgIGNtZCA9PlxuICAgICAgICAgICAgaXNDb21tYW5kRW5hYmxlZChjbWQpICYmXG4gICAgICAgICAgICAoY21kLm5hbWUgPT09IGNvbW1hbmROYW1lIHx8XG4gICAgICAgICAgICAgIGNtZC5hbGlhc2VzPy5pbmNsdWRlcyhjb21tYW5kTmFtZSkgfHxcbiAgICAgICAgICAgICAgZ2V0Q29tbWFuZE5hbWUoY21kKSA9PT0gY29tbWFuZE5hbWUpLFxuICAgICAgICApXG4gICAgICAgIGlmIChtYXRjaGluZ0NvbW1hbmQ/Lm5hbWUgPT09ICdjbGVhcicgJiYgaWRsZUhpbnRTaG93blJlZi5jdXJyZW50KSB7XG4gICAgICAgICAgbG9nRXZlbnQoJ3Rlbmd1X2lkbGVfcmV0dXJuX2FjdGlvbicsIHtcbiAgICAgICAgICAgIGFjdGlvbjpcbiAgICAgICAgICAgICAgJ2hpbnRfY29udmVydGVkJyBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICAgICAgdmFyaWFudDpcbiAgICAgICAgICAgICAgaWRsZUhpbnRTaG93blJlZi5jdXJyZW50IGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgICAgICBpZGxlTWludXRlczogTWF0aC5yb3VuZChcbiAgICAgICAgICAgICAgKERhdGUubm93KCkgLSBsYXN0UXVlcnlDb21wbGV0aW9uVGltZVJlZi5jdXJyZW50KSAvIDYwXzAwMCxcbiAgICAgICAgICAgICksXG4gICAgICAgICAgICBtZXNzYWdlQ291bnQ6IG1lc3NhZ2VzUmVmLmN1cnJlbnQubGVuZ3RoLFxuICAgICAgICAgICAgdG90YWxJbnB1dFRva2VuczogZ2V0VG90YWxJbnB1dFRva2VucygpLFxuICAgICAgICAgIH0pXG4gICAgICAgICAgaWRsZUhpbnRTaG93blJlZi5jdXJyZW50ID0gZmFsc2VcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IHNob3VsZFRyZWF0QXNJbW1lZGlhdGUgPVxuICAgICAgICAgIHF1ZXJ5R3VhcmQuaXNBY3RpdmUgJiZcbiAgICAgICAgICAobWF0Y2hpbmdDb21tYW5kPy5pbW1lZGlhdGUgfHwgb3B0aW9ucz8uZnJvbUtleWJpbmRpbmcpXG5cbiAgICAgICAgaWYgKFxuICAgICAgICAgIG1hdGNoaW5nQ29tbWFuZCAmJlxuICAgICAgICAgIHNob3VsZFRyZWF0QXNJbW1lZGlhdGUgJiZcbiAgICAgICAgICBtYXRjaGluZ0NvbW1hbmQudHlwZSA9PT0gJ2xvY2FsLWpzeCdcbiAgICAgICAgKSB7XG4gICAgICAgICAgLy8gT25seSBjbGVhciBpbnB1dCBpZiB0aGUgc3VibWl0dGVkIHRleHQgbWF0Y2hlcyB3aGF0J3MgaW4gdGhlIHByb21wdC5cbiAgICAgICAgICAvLyBXaGVuIGEgY29tbWFuZCBrZXliaW5kaW5nIGZpcmVzLCBpbnB1dCBpcyBcIi88Y29tbWFuZD5cIiBidXQgdGhlIGFjdHVhbFxuICAgICAgICAgIC8vIGlucHV0IHZhbHVlIGlzIHRoZSB1c2VyJ3MgZXhpc3RpbmcgdGV4dCAtIGRvbid0IGNsZWFyIGl0IGluIHRoYXQgY2FzZS5cbiAgICAgICAgICBpZiAoaW5wdXQudHJpbSgpID09PSBpbnB1dFZhbHVlUmVmLmN1cnJlbnQudHJpbSgpKSB7XG4gICAgICAgICAgICBzZXRJbnB1dFZhbHVlKCcnKVxuICAgICAgICAgICAgaGVscGVycy5zZXRDdXJzb3JPZmZzZXQoMClcbiAgICAgICAgICAgIGhlbHBlcnMuY2xlYXJCdWZmZXIoKVxuICAgICAgICAgICAgc2V0UGFzdGVkQ29udGVudHMoe30pXG4gICAgICAgICAgfVxuXG4gICAgICAgICAgY29uc3QgcGFzdGVkVGV4dFJlZnMgPSBwYXJzZVJlZmVyZW5jZXMoaW5wdXQpLmZpbHRlcihcbiAgICAgICAgICAgIHIgPT4gcGFzdGVkQ29udGVudHNbci5pZF0/LnR5cGUgPT09ICd0ZXh0JyxcbiAgICAgICAgICApXG4gICAgICAgICAgY29uc3QgcGFzdGVkVGV4dENvdW50ID0gcGFzdGVkVGV4dFJlZnMubGVuZ3RoXG4gICAgICAgICAgY29uc3QgcGFzdGVkVGV4dEJ5dGVzID0gcGFzdGVkVGV4dFJlZnMucmVkdWNlKFxuICAgICAgICAgICAgKHN1bSwgcikgPT4gc3VtICsgKHBhc3RlZENvbnRlbnRzW3IuaWRdPy5jb250ZW50Lmxlbmd0aCA/PyAwKSxcbiAgICAgICAgICAgIDAsXG4gICAgICAgICAgKVxuICAgICAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9wYXN0ZV90ZXh0JywgeyBwYXN0ZWRUZXh0Q291bnQsIHBhc3RlZFRleHRCeXRlcyB9KVxuICAgICAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9pbW1lZGlhdGVfY29tbWFuZF9leGVjdXRlZCcsIHtcbiAgICAgICAgICAgIGNvbW1hbmROYW1lOlxuICAgICAgICAgICAgICBtYXRjaGluZ0NvbW1hbmQubmFtZSBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICAgICAgZnJvbUtleWJpbmRpbmc6IG9wdGlvbnM/LmZyb21LZXliaW5kaW5nID8/IGZhbHNlLFxuICAgICAgICAgIH0pXG5cbiAgICAgICAgICAvLyBFeGVjdXRlIHRoZSBjb21tYW5kIGRpcmVjdGx5XG4gICAgICAgICAgY29uc3QgZXhlY3V0ZUltbWVkaWF0ZUNvbW1hbmQgPSBhc3luYyAoKTogUHJvbWlzZTx2b2lkPiA9PiB7XG4gICAgICAgICAgICBsZXQgZG9uZVdhc0NhbGxlZCA9IGZhbHNlXG4gICAgICAgICAgICBjb25zdCBvbkRvbmUgPSAoXG4gICAgICAgICAgICAgIHJlc3VsdD86IHN0cmluZyxcbiAgICAgICAgICAgICAgZG9uZU9wdGlvbnM/OiB7XG4gICAgICAgICAgICAgICAgZGlzcGxheT86IENvbW1hbmRSZXN1bHREaXNwbGF5XG4gICAgICAgICAgICAgICAgbWV0YU1lc3NhZ2VzPzogc3RyaW5nW11cbiAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICk6IHZvaWQgPT4ge1xuICAgICAgICAgICAgICBkb25lV2FzQ2FsbGVkID0gdHJ1ZVxuICAgICAgICAgICAgICBzZXRUb29sSlNYKHtcbiAgICAgICAgICAgICAgICBqc3g6IG51bGwsXG4gICAgICAgICAgICAgICAgc2hvdWxkSGlkZVByb21wdElucHV0OiBmYWxzZSxcbiAgICAgICAgICAgICAgICBjbGVhckxvY2FsSlNYOiB0cnVlLFxuICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICBjb25zdCBuZXdNZXNzYWdlczogTWVzc2FnZVR5cGVbXSA9IFtdXG4gICAgICAgICAgICAgIGlmIChyZXN1bHQgJiYgZG9uZU9wdGlvbnM/LmRpc3BsYXkgIT09ICdza2lwJykge1xuICAgICAgICAgICAgICAgIGFkZE5vdGlmaWNhdGlvbih7XG4gICAgICAgICAgICAgICAgICBrZXk6IGBpbW1lZGlhdGUtJHttYXRjaGluZ0NvbW1hbmQubmFtZX1gLFxuICAgICAgICAgICAgICAgICAgdGV4dDogcmVzdWx0LFxuICAgICAgICAgICAgICAgICAgcHJpb3JpdHk6ICdpbW1lZGlhdGUnLFxuICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgLy8gSW4gZnVsbHNjcmVlbiB0aGUgY29tbWFuZCBqdXN0IHNob3dlZCBhcyBhIGNlbnRlcmVkIG1vZGFsXG4gICAgICAgICAgICAgICAgLy8gcGFuZSDigJQgdGhlIG5vdGlmaWNhdGlvbiBhYm92ZSBpcyBlbm91Z2ggZmVlZGJhY2suIEFkZGluZ1xuICAgICAgICAgICAgICAgIC8vIFwi4p2vIC9jb25maWdcIiArIFwi4o6/IGRpc21pc3NlZFwiIHRvIHRoZSB0cmFuc2NyaXB0IGlzIGNsdXR0ZXJcbiAgICAgICAgICAgICAgICAvLyAodGhvc2UgbWVzc2FnZXMgYXJlIHR5cGU6c3lzdGVtIHN1YnR5cGU6bG9jYWxfY29tbWFuZCDigJRcbiAgICAgICAgICAgICAgICAvLyB1c2VyLXZpc2libGUgYnV0IE5PVCBzZW50IHRvIHRoZSBtb2RlbCwgc28gc2tpcHBpbmcgdGhlbVxuICAgICAgICAgICAgICAgIC8vIGRvZXNuJ3QgY2hhbmdlIG1vZGVsIGNvbnRleHQpLiBPdXRzaWRlIGZ1bGxzY3JlZW4gdGhlXG4gICAgICAgICAgICAgICAgLy8gdHJhbnNjcmlwdCBlbnRyeSBzdGF5cyBzbyBzY3JvbGxiYWNrIHNob3dzIHdoYXQgcmFuLlxuICAgICAgICAgICAgICAgIGlmICghaXNGdWxsc2NyZWVuRW52RW5hYmxlZCgpKSB7XG4gICAgICAgICAgICAgICAgICBuZXdNZXNzYWdlcy5wdXNoKFxuICAgICAgICAgICAgICAgICAgICBjcmVhdGVDb21tYW5kSW5wdXRNZXNzYWdlKFxuICAgICAgICAgICAgICAgICAgICAgIGZvcm1hdENvbW1hbmRJbnB1dFRhZ3MoXG4gICAgICAgICAgICAgICAgICAgICAgICBnZXRDb21tYW5kTmFtZShtYXRjaGluZ0NvbW1hbmQpLFxuICAgICAgICAgICAgICAgICAgICAgICAgY29tbWFuZEFyZ3MsXG4gICAgICAgICAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgICAgICAgICAgY3JlYXRlQ29tbWFuZElucHV0TWVzc2FnZShcbiAgICAgICAgICAgICAgICAgICAgICBgPCR7TE9DQUxfQ09NTUFORF9TVERPVVRfVEFHfT4ke2VzY2FwZVhtbChyZXN1bHQpfTwvJHtMT0NBTF9DT01NQU5EX1NURE9VVF9UQUd9PmAsXG4gICAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIC8vIEluamVjdCBtZXRhIG1lc3NhZ2VzIChtb2RlbC12aXNpYmxlLCB1c2VyLWhpZGRlbikgaW50byB0aGUgdHJhbnNjcmlwdFxuICAgICAgICAgICAgICBpZiAoZG9uZU9wdGlvbnM/Lm1ldGFNZXNzYWdlcz8ubGVuZ3RoKSB7XG4gICAgICAgICAgICAgICAgbmV3TWVzc2FnZXMucHVzaChcbiAgICAgICAgICAgICAgICAgIC4uLmRvbmVPcHRpb25zLm1ldGFNZXNzYWdlcy5tYXAoY29udGVudCA9PlxuICAgICAgICAgICAgICAgICAgICBjcmVhdGVVc2VyTWVzc2FnZSh7IGNvbnRlbnQsIGlzTWV0YTogdHJ1ZSB9KSxcbiAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIGlmIChuZXdNZXNzYWdlcy5sZW5ndGgpIHtcbiAgICAgICAgICAgICAgICBzZXRNZXNzYWdlcyhwcmV2ID0+IFsuLi5wcmV2LCAuLi5uZXdNZXNzYWdlc10pXG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgLy8gUmVzdG9yZSBzdGFzaGVkIHByb21wdCBhZnRlciBsb2NhbC1qc3ggY29tbWFuZCBjb21wbGV0ZXMuXG4gICAgICAgICAgICAgIC8vIFRoZSBub3JtYWwgc3Rhc2ggcmVzdG9yYXRpb24gcGF0aCAoYmVsb3cpIGlzIHNraXBwZWQgYmVjYXVzZVxuICAgICAgICAgICAgICAvLyBsb2NhbC1qc3ggY29tbWFuZHMgcmV0dXJuIGVhcmx5IGZyb20gb25TdWJtaXQuXG4gICAgICAgICAgICAgIGlmIChzdGFzaGVkUHJvbXB0ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICBzZXRJbnB1dFZhbHVlKHN0YXNoZWRQcm9tcHQudGV4dClcbiAgICAgICAgICAgICAgICBoZWxwZXJzLnNldEN1cnNvck9mZnNldChzdGFzaGVkUHJvbXB0LmN1cnNvck9mZnNldClcbiAgICAgICAgICAgICAgICBzZXRQYXN0ZWRDb250ZW50cyhzdGFzaGVkUHJvbXB0LnBhc3RlZENvbnRlbnRzKVxuICAgICAgICAgICAgICAgIHNldFN0YXNoZWRQcm9tcHQodW5kZWZpbmVkKVxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIEJ1aWxkIGNvbnRleHQgZm9yIHRoZSBjb21tYW5kIChyZXVzZXMgZXhpc3RpbmcgZ2V0VG9vbFVzZUNvbnRleHQpLlxuICAgICAgICAgICAgLy8gUmVhZCBtZXNzYWdlcyB2aWEgcmVmIHRvIGtlZXAgb25TdWJtaXQgc3RhYmxlIGFjcm9zcyBtZXNzYWdlXG4gICAgICAgICAgICAvLyB1cGRhdGVzIOKAlCBtYXRjaGVzIHRoZSBwYXR0ZXJuIGF0IEwyMzg0L0wyNDAwL0wyNjYyIGFuZCBhdm9pZHNcbiAgICAgICAgICAgIC8vIHBpbm5pbmcgc3RhbGUgUkVQTCByZW5kZXIgc2NvcGVzIGluIGRvd25zdHJlYW0gY2xvc3VyZXMuXG4gICAgICAgICAgICBjb25zdCBjb250ZXh0ID0gZ2V0VG9vbFVzZUNvbnRleHQoXG4gICAgICAgICAgICAgIG1lc3NhZ2VzUmVmLmN1cnJlbnQsXG4gICAgICAgICAgICAgIFtdLFxuICAgICAgICAgICAgICBjcmVhdGVBYm9ydENvbnRyb2xsZXIoKSxcbiAgICAgICAgICAgICAgbWFpbkxvb3BNb2RlbCxcbiAgICAgICAgICAgIClcblxuICAgICAgICAgICAgY29uc3QgbW9kID0gYXdhaXQgbWF0Y2hpbmdDb21tYW5kLmxvYWQoKVxuICAgICAgICAgICAgY29uc3QganN4ID0gYXdhaXQgbW9kLmNhbGwob25Eb25lLCBjb250ZXh0LCBjb21tYW5kQXJncylcblxuICAgICAgICAgICAgLy8gU2tpcCBpZiBvbkRvbmUgYWxyZWFkeSBmaXJlZCDigJQgcHJldmVudHMgc3R1Y2sgaXNMb2NhbEpTWENvbW1hbmRcbiAgICAgICAgICAgIC8vIChzZWUgcHJvY2Vzc1NsYXNoQ29tbWFuZC50c3ggbG9jYWwtanN4IGNhc2UgZm9yIGZ1bGwgbWVjaGFuaXNtKS5cbiAgICAgICAgICAgIGlmIChqc3ggJiYgIWRvbmVXYXNDYWxsZWQpIHtcbiAgICAgICAgICAgICAgLy8gc2hvdWxkSGlkZVByb21wdElucHV0OiBmYWxzZSBrZWVwcyBOb3RpZmljYXRpb25zIG1vdW50ZWRcbiAgICAgICAgICAgICAgLy8gc28gdGhlIG9uRG9uZSByZXN1bHQgaXNuJ3QgbG9zdFxuICAgICAgICAgICAgICBzZXRUb29sSlNYKHtcbiAgICAgICAgICAgICAgICBqc3gsXG4gICAgICAgICAgICAgICAgc2hvdWxkSGlkZVByb21wdElucHV0OiBmYWxzZSxcbiAgICAgICAgICAgICAgICBpc0xvY2FsSlNYQ29tbWFuZDogdHJ1ZSxcbiAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgICAgdm9pZCBleGVjdXRlSW1tZWRpYXRlQ29tbWFuZCgpXG4gICAgICAgICAgcmV0dXJuIC8vIEFsd2F5cyByZXR1cm4gZWFybHkgLSBkb24ndCBhZGQgdG8gaGlzdG9yeSBvciBxdWV1ZVxuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIFJlbW90ZSBtb2RlOiBza2lwIGVtcHR5IGlucHV0IGVhcmx5IGJlZm9yZSBhbnkgc3RhdGUgbXV0YXRpb25zXG4gICAgICBpZiAoYWN0aXZlUmVtb3RlLmlzUmVtb3RlTW9kZSAmJiAhaW5wdXQudHJpbSgpKSB7XG4gICAgICAgIHJldHVyblxuICAgICAgfVxuXG4gICAgICAvLyBJZGxlLXJldHVybjogcHJvbXB0IHJldHVybmluZyB1c2VycyB0byBzdGFydCBmcmVzaCB3aGVuIHRoZVxuICAgICAgLy8gY29udmVyc2F0aW9uIGlzIGxhcmdlIGFuZCB0aGUgY2FjaGUgaXMgY29sZC4gdGVuZ3Vfd2lsbG93X21vZGVcbiAgICAgIC8vIGNvbnRyb2xzIHRyZWF0bWVudDogXCJkaWFsb2dcIiAoYmxvY2tpbmcpLCBcImhpbnRcIiAobm90aWZpY2F0aW9uKSwgXCJvZmZcIi5cbiAgICAgIHtcbiAgICAgICAgY29uc3Qgd2lsbG93TW9kZSA9IGdldEZlYXR1cmVWYWx1ZV9DQUNIRURfTUFZX0JFX1NUQUxFKFxuICAgICAgICAgICd0ZW5ndV93aWxsb3dfbW9kZScsXG4gICAgICAgICAgJ29mZicsXG4gICAgICAgIClcbiAgICAgICAgY29uc3QgaWRsZVRocmVzaG9sZE1pbiA9IE51bWJlcihcbiAgICAgICAgICBwcm9jZXNzLmVudi5DTEFVREVfQ09ERV9JRExFX1RIUkVTSE9MRF9NSU5VVEVTID8/IDc1LFxuICAgICAgICApXG4gICAgICAgIGNvbnN0IHRva2VuVGhyZXNob2xkID0gTnVtYmVyKFxuICAgICAgICAgIHByb2Nlc3MuZW52LkNMQVVERV9DT0RFX0lETEVfVE9LRU5fVEhSRVNIT0xEID8/IDEwMF8wMDAsXG4gICAgICAgIClcbiAgICAgICAgaWYgKFxuICAgICAgICAgIHdpbGxvd01vZGUgIT09ICdvZmYnICYmXG4gICAgICAgICAgIWdldEdsb2JhbENvbmZpZygpLmlkbGVSZXR1cm5EaXNtaXNzZWQgJiZcbiAgICAgICAgICAhc2tpcElkbGVDaGVja1JlZi5jdXJyZW50ICYmXG4gICAgICAgICAgIXNwZWN1bGF0aW9uQWNjZXB0ICYmXG4gICAgICAgICAgIWlucHV0LnRyaW0oKS5zdGFydHNXaXRoKCcvJykgJiZcbiAgICAgICAgICBsYXN0UXVlcnlDb21wbGV0aW9uVGltZVJlZi5jdXJyZW50ID4gMCAmJlxuICAgICAgICAgIGdldFRvdGFsSW5wdXRUb2tlbnMoKSA+PSB0b2tlblRocmVzaG9sZFxuICAgICAgICApIHtcbiAgICAgICAgICBjb25zdCBpZGxlTXMgPSBEYXRlLm5vdygpIC0gbGFzdFF1ZXJ5Q29tcGxldGlvblRpbWVSZWYuY3VycmVudFxuICAgICAgICAgIGNvbnN0IGlkbGVNaW51dGVzID0gaWRsZU1zIC8gNjBfMDAwXG4gICAgICAgICAgaWYgKGlkbGVNaW51dGVzID49IGlkbGVUaHJlc2hvbGRNaW4gJiYgd2lsbG93TW9kZSA9PT0gJ2RpYWxvZycpIHtcbiAgICAgICAgICAgIHNldElkbGVSZXR1cm5QZW5kaW5nKHsgaW5wdXQsIGlkbGVNaW51dGVzIH0pXG4gICAgICAgICAgICBzZXRJbnB1dFZhbHVlKCcnKVxuICAgICAgICAgICAgaGVscGVycy5zZXRDdXJzb3JPZmZzZXQoMClcbiAgICAgICAgICAgIGhlbHBlcnMuY2xlYXJCdWZmZXIoKVxuICAgICAgICAgICAgcmV0dXJuXG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIEFkZCB0byBoaXN0b3J5IGZvciBkaXJlY3QgdXNlciBzdWJtaXNzaW9ucy5cbiAgICAgIC8vIFF1ZXVlZCBjb21tYW5kIHByb2Nlc3NpbmcgKGV4ZWN1dGVRdWV1ZWRJbnB1dCkgZG9lc24ndCBjYWxsIG9uU3VibWl0LFxuICAgICAgLy8gc28gbm90aWZpY2F0aW9ucyBhbmQgYWxyZWFkeS1xdWV1ZWQgdXNlciBpbnB1dCB3b24ndCBiZSBhZGRlZCB0byBoaXN0b3J5IGhlcmUuXG4gICAgICAvLyBTa2lwIGhpc3RvcnkgZm9yIGtleWJpbmRpbmctdHJpZ2dlcmVkIGNvbW1hbmRzICh1c2VyIGRpZG4ndCB0eXBlIHRoZSBjb21tYW5kKS5cbiAgICAgIGlmICghb3B0aW9ucz8uZnJvbUtleWJpbmRpbmcpIHtcbiAgICAgICAgYWRkVG9IaXN0b3J5KHtcbiAgICAgICAgICBkaXNwbGF5OiBzcGVjdWxhdGlvbkFjY2VwdFxuICAgICAgICAgICAgPyBpbnB1dFxuICAgICAgICAgICAgOiBwcmVwZW5kTW9kZUNoYXJhY3RlclRvSW5wdXQoaW5wdXQsIGlucHV0TW9kZSksXG4gICAgICAgICAgcGFzdGVkQ29udGVudHM6IHNwZWN1bGF0aW9uQWNjZXB0ID8ge30gOiBwYXN0ZWRDb250ZW50cyxcbiAgICAgICAgfSlcbiAgICAgICAgLy8gQWRkIHRoZSBqdXN0LXN1Ym1pdHRlZCBjb21tYW5kIHRvIHRoZSBmcm9udCBvZiB0aGUgZ2hvc3QtdGV4dFxuICAgICAgICAvLyBjYWNoZSBzbyBpdCdzIHN1Z2dlc3RlZCBpbW1lZGlhdGVseSAobm90IGFmdGVyIHRoZSA2MHMgVFRMKS5cbiAgICAgICAgaWYgKGlucHV0TW9kZSA9PT0gJ2Jhc2gnKSB7XG4gICAgICAgICAgcHJlcGVuZFRvU2hlbGxIaXN0b3J5Q2FjaGUoaW5wdXQudHJpbSgpKVxuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIFJlc3RvcmUgc3Rhc2ggaWYgcHJlc2VudCwgYnV0IE5PVCBmb3Igc2xhc2ggY29tbWFuZHMgb3Igd2hlbiBsb2FkaW5nLlxuICAgICAgLy8gLSBTbGFzaCBjb21tYW5kcyAoZXNwZWNpYWxseSBpbnRlcmFjdGl2ZSBvbmVzIGxpa2UgL21vZGVsLCAvY29udGV4dCkgaGlkZVxuICAgICAgLy8gICB0aGUgcHJvbXB0IGFuZCBzaG93IGEgcGlja2VyIFVJLiBSZXN0b3JpbmcgdGhlIHN0YXNoIGR1cmluZyBhIGNvbW1hbmQgd291bGRcbiAgICAgIC8vICAgcGxhY2UgdGhlIHRleHQgaW4gYSBoaWRkZW4gaW5wdXQsIGFuZCB0aGUgdXNlciB3b3VsZCBsb3NlIGl0IGJ5IHR5cGluZyB0aGVcbiAgICAgIC8vICAgbmV4dCBjb21tYW5kLiBJbnN0ZWFkLCBwcmVzZXJ2ZSB0aGUgc3Rhc2ggc28gaXQgc3Vydml2ZXMgYWNyb3NzIGNvbW1hbmQgcnVucy5cbiAgICAgIC8vIC0gV2hlbiBsb2FkaW5nLCB0aGUgc3VibWl0dGVkIGlucHV0IHdpbGwgYmUgcXVldWVkIGFuZCBoYW5kbGVQcm9tcHRTdWJtaXRcbiAgICAgIC8vICAgd2lsbCBjbGVhciB0aGUgaW5wdXQgZmllbGQgKG9uSW5wdXRDaGFuZ2UoJycpKSwgd2hpY2ggd291bGQgY2xvYmJlciB0aGVcbiAgICAgIC8vICAgcmVzdG9yZWQgc3Rhc2guIERlZmVyIHJlc3RvcmF0aW9uIHRvIGFmdGVyIGhhbmRsZVByb21wdFN1Ym1pdCAoYmVsb3cpLlxuICAgICAgLy8gICBSZW1vdGUgbW9kZSBpcyBleGVtcHQ6IGl0IHNlbmRzIHZpYSBXZWJTb2NrZXQgYW5kIHJldHVybnMgZWFybHkgd2l0aG91dFxuICAgICAgLy8gICBjYWxsaW5nIGhhbmRsZVByb21wdFN1Ym1pdCwgc28gdGhlcmUncyBubyBjbG9iYmVyaW5nIHJpc2sg4oCUIHJlc3RvcmUgZWFnZXJseS5cbiAgICAgIC8vIEluIGJvdGggZGVmZXJyZWQgY2FzZXMsIHRoZSBzdGFzaCBpcyByZXN0b3JlZCBhZnRlciBhd2FpdCBoYW5kbGVQcm9tcHRTdWJtaXQuXG4gICAgICBjb25zdCBpc1NsYXNoQ29tbWFuZCA9ICFzcGVjdWxhdGlvbkFjY2VwdCAmJiBpbnB1dC50cmltKCkuc3RhcnRzV2l0aCgnLycpXG4gICAgICAvLyBTdWJtaXQgcnVucyBcIm5vd1wiIChub3QgcXVldWVkKSB3aGVuIG5vdCBhbHJlYWR5IGxvYWRpbmcsIG9yIHdoZW5cbiAgICAgIC8vIGFjY2VwdGluZyBzcGVjdWxhdGlvbiwgb3IgaW4gcmVtb3RlIG1vZGUgKHdoaWNoIHNlbmRzIHZpYSBXUyBhbmRcbiAgICAgIC8vIHJldHVybnMgZWFybHkgd2l0aG91dCBjYWxsaW5nIGhhbmRsZVByb21wdFN1Ym1pdCkuXG4gICAgICBjb25zdCBzdWJtaXRzTm93ID1cbiAgICAgICAgIWlzTG9hZGluZyB8fCBzcGVjdWxhdGlvbkFjY2VwdCB8fCBhY3RpdmVSZW1vdGUuaXNSZW1vdGVNb2RlXG4gICAgICBpZiAoc3Rhc2hlZFByb21wdCAhPT0gdW5kZWZpbmVkICYmICFpc1NsYXNoQ29tbWFuZCAmJiBzdWJtaXRzTm93KSB7XG4gICAgICAgIHNldElucHV0VmFsdWUoc3Rhc2hlZFByb21wdC50ZXh0KVxuICAgICAgICBoZWxwZXJzLnNldEN1cnNvck9mZnNldChzdGFzaGVkUHJvbXB0LmN1cnNvck9mZnNldClcbiAgICAgICAgc2V0UGFzdGVkQ29udGVudHMoc3Rhc2hlZFByb21wdC5wYXN0ZWRDb250ZW50cylcbiAgICAgICAgc2V0U3Rhc2hlZFByb21wdCh1bmRlZmluZWQpXG4gICAgICB9IGVsc2UgaWYgKHN1Ym1pdHNOb3cpIHtcbiAgICAgICAgaWYgKCFvcHRpb25zPy5mcm9tS2V5YmluZGluZykge1xuICAgICAgICAgIC8vIENsZWFyIGlucHV0IHdoZW4gbm90IGxvYWRpbmcgb3IgYWNjZXB0aW5nIHNwZWN1bGF0aW9uLlxuICAgICAgICAgIC8vIFByZXNlcnZlIGlucHV0IGZvciBrZXliaW5kaW5nLXRyaWdnZXJlZCBjb21tYW5kcy5cbiAgICAgICAgICBzZXRJbnB1dFZhbHVlKCcnKVxuICAgICAgICAgIGhlbHBlcnMuc2V0Q3Vyc29yT2Zmc2V0KDApXG4gICAgICAgIH1cbiAgICAgICAgc2V0UGFzdGVkQ29udGVudHMoe30pXG4gICAgICB9XG5cbiAgICAgIGlmIChzdWJtaXRzTm93KSB7XG4gICAgICAgIHNldElucHV0TW9kZSgncHJvbXB0JylcbiAgICAgICAgc2V0SURFU2VsZWN0aW9uKHVuZGVmaW5lZClcbiAgICAgICAgc2V0U3VibWl0Q291bnQoXyA9PiBfICsgMSlcbiAgICAgICAgaGVscGVycy5jbGVhckJ1ZmZlcigpXG4gICAgICAgIHRpcFBpY2tlZFRoaXNUdXJuUmVmLmN1cnJlbnQgPSBmYWxzZVxuXG4gICAgICAgIC8vIFNob3cgdGhlIHBsYWNlaG9sZGVyIGluIHRoZSBzYW1lIFJlYWN0IGJhdGNoIGFzIHNldElucHV0VmFsdWUoJycpLlxuICAgICAgICAvLyBTa2lwIGZvciBzbGFzaC9iYXNoICh0aGV5IGhhdmUgdGhlaXIgb3duIGVjaG8pLCBzcGVjdWxhdGlvbiBhbmQgcmVtb3RlXG4gICAgICAgIC8vIG1vZGUgKGJvdGggc2V0TWVzc2FnZXMgZGlyZWN0bHkgd2l0aCBubyBnYXAgdG8gYnJpZGdlKS5cbiAgICAgICAgaWYgKFxuICAgICAgICAgICFpc1NsYXNoQ29tbWFuZCAmJlxuICAgICAgICAgIGlucHV0TW9kZSA9PT0gJ3Byb21wdCcgJiZcbiAgICAgICAgICAhc3BlY3VsYXRpb25BY2NlcHQgJiZcbiAgICAgICAgICAhYWN0aXZlUmVtb3RlLmlzUmVtb3RlTW9kZVxuICAgICAgICApIHtcbiAgICAgICAgICBzZXRVc2VySW5wdXRPblByb2Nlc3NpbmcoaW5wdXQpXG4gICAgICAgICAgLy8gc2hvd1NwaW5uZXIgaW5jbHVkZXMgdXNlcklucHV0T25Qcm9jZXNzaW5nLCBzbyB0aGUgc3Bpbm5lciBhcHBlYXJzXG4gICAgICAgICAgLy8gb24gdGhpcyByZW5kZXIuIFJlc2V0IHRpbWluZyByZWZzIG5vdyAoYmVmb3JlIHF1ZXJ5R3VhcmQucmVzZXJ2ZSgpXG4gICAgICAgICAgLy8gd291bGQpIHNvIGVsYXBzZWQgdGltZSBkb2Vzbid0IHJlYWQgYXMgRGF0ZS5ub3coKSAtIDAuIFRoZVxuICAgICAgICAgIC8vIGlzUXVlcnlBY3RpdmUgdHJhbnNpdGlvbiBhYm92ZSBkb2VzIHRoZSBzYW1lIHJlc2V0IOKAlCBpZGVtcG90ZW50LlxuICAgICAgICAgIHJlc2V0VGltaW5nUmVmcygpXG4gICAgICAgIH1cblxuICAgICAgICAvLyBJbmNyZW1lbnQgcHJvbXB0IGNvdW50IGZvciBhdHRyaWJ1dGlvbiB0cmFja2luZyBhbmQgc2F2ZSBzbmFwc2hvdFxuICAgICAgICAvLyBUaGUgc25hcHNob3QgcGVyc2lzdHMgcHJvbXB0Q291bnQgc28gaXQgc3Vydml2ZXMgY29tcGFjdGlvblxuICAgICAgICBpZiAoZmVhdHVyZSgnQ09NTUlUX0FUVFJJQlVUSU9OJykpIHtcbiAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+ICh7XG4gICAgICAgICAgICAuLi5wcmV2LFxuICAgICAgICAgICAgYXR0cmlidXRpb246IGluY3JlbWVudFByb21wdENvdW50KHByZXYuYXR0cmlidXRpb24sIHNuYXBzaG90ID0+IHtcbiAgICAgICAgICAgICAgdm9pZCByZWNvcmRBdHRyaWJ1dGlvblNuYXBzaG90KHNuYXBzaG90KS5jYXRjaChlcnJvciA9PiB7XG4gICAgICAgICAgICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICAgICAgICAgICAgYEF0dHJpYnV0aW9uOiBGYWlsZWQgdG8gc2F2ZSBzbmFwc2hvdDogJHtlcnJvcn1gLFxuICAgICAgICAgICAgICAgIClcbiAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIH0pLFxuICAgICAgICAgIH0pKVxuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIEhhbmRsZSBzcGVjdWxhdGlvbiBhY2NlcHRhbmNlXG4gICAgICBpZiAoc3BlY3VsYXRpb25BY2NlcHQpIHtcbiAgICAgICAgY29uc3QgeyBxdWVyeVJlcXVpcmVkIH0gPSBhd2FpdCBoYW5kbGVTcGVjdWxhdGlvbkFjY2VwdChcbiAgICAgICAgICBzcGVjdWxhdGlvbkFjY2VwdC5zdGF0ZSxcbiAgICAgICAgICBzcGVjdWxhdGlvbkFjY2VwdC5zcGVjdWxhdGlvblNlc3Npb25UaW1lU2F2ZWRNcyxcbiAgICAgICAgICBzcGVjdWxhdGlvbkFjY2VwdC5zZXRBcHBTdGF0ZSxcbiAgICAgICAgICBpbnB1dCxcbiAgICAgICAgICB7XG4gICAgICAgICAgICBzZXRNZXNzYWdlcyxcbiAgICAgICAgICAgIHJlYWRGaWxlU3RhdGUsXG4gICAgICAgICAgICBjd2Q6IGdldE9yaWdpbmFsQ3dkKCksXG4gICAgICAgICAgfSxcbiAgICAgICAgKVxuICAgICAgICBpZiAocXVlcnlSZXF1aXJlZCkge1xuICAgICAgICAgIGNvbnN0IG5ld0Fib3J0Q29udHJvbGxlciA9IGNyZWF0ZUFib3J0Q29udHJvbGxlcigpXG4gICAgICAgICAgc2V0QWJvcnRDb250cm9sbGVyKG5ld0Fib3J0Q29udHJvbGxlcilcbiAgICAgICAgICB2b2lkIG9uUXVlcnkoW10sIG5ld0Fib3J0Q29udHJvbGxlciwgdHJ1ZSwgW10sIG1haW5Mb29wTW9kZWwpXG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG5cbiAgICAgIC8vIFJlbW90ZSBtb2RlOiBzZW5kIGlucHV0IHZpYSBzdHJlYW0tanNvbiBpbnN0ZWFkIG9mIGxvY2FsIHF1ZXJ5LlxuICAgICAgLy8gUGVybWlzc2lvbiByZXF1ZXN0cyBmcm9tIHRoZSByZW1vdGUgYXJlIGJyaWRnZWQgaW50byB0b29sVXNlQ29uZmlybVF1ZXVlXG4gICAgICAvLyBhbmQgcmVuZGVyZWQgdXNpbmcgdGhlIHN0YW5kYXJkIFBlcm1pc3Npb25SZXF1ZXN0IGNvbXBvbmVudC5cbiAgICAgIC8vXG4gICAgICAvLyBsb2NhbC1qc3ggc2xhc2ggY29tbWFuZHMgKGUuZy4gL2FnZW50cywgL2NvbmZpZykgcmVuZGVyIFVJIGluIFRISVNcbiAgICAgIC8vIHByb2Nlc3Mg4oCUIHRoZXkgaGF2ZSBubyByZW1vdGUgZXF1aXZhbGVudC4gTGV0IHRob3NlIGZhbGwgdGhyb3VnaCB0b1xuICAgICAgLy8gaGFuZGxlUHJvbXB0U3VibWl0IHNvIHRoZXkgZXhlY3V0ZSBsb2NhbGx5LiBQcm9tcHQgY29tbWFuZHMgYW5kXG4gICAgICAvLyBwbGFpbiB0ZXh0IGdvIHRvIHRoZSByZW1vdGUuXG4gICAgICBpZiAoXG4gICAgICAgIGFjdGl2ZVJlbW90ZS5pc1JlbW90ZU1vZGUgJiZcbiAgICAgICAgIShcbiAgICAgICAgICBpc1NsYXNoQ29tbWFuZCAmJlxuICAgICAgICAgIGNvbW1hbmRzLmZpbmQoYyA9PiB7XG4gICAgICAgICAgICBjb25zdCBuYW1lID0gaW5wdXQudHJpbSgpLnNsaWNlKDEpLnNwbGl0KC9cXHMvKVswXVxuICAgICAgICAgICAgcmV0dXJuIChcbiAgICAgICAgICAgICAgaXNDb21tYW5kRW5hYmxlZChjKSAmJlxuICAgICAgICAgICAgICAoYy5uYW1lID09PSBuYW1lIHx8XG4gICAgICAgICAgICAgICAgYy5hbGlhc2VzPy5pbmNsdWRlcyhuYW1lISkgfHxcbiAgICAgICAgICAgICAgICBnZXRDb21tYW5kTmFtZShjKSA9PT0gbmFtZSlcbiAgICAgICAgICAgIClcbiAgICAgICAgICB9KT8udHlwZSA9PT0gJ2xvY2FsLWpzeCdcbiAgICAgICAgKVxuICAgICAgKSB7XG4gICAgICAgIC8vIEJ1aWxkIGNvbnRlbnQgYmxvY2tzIHdoZW4gdGhlcmUgYXJlIHBhc3RlZCBhdHRhY2htZW50cyAoaW1hZ2VzKVxuICAgICAgICBjb25zdCBwYXN0ZWRWYWx1ZXMgPSBPYmplY3QudmFsdWVzKHBhc3RlZENvbnRlbnRzKVxuICAgICAgICBjb25zdCBpbWFnZUNvbnRlbnRzID0gcGFzdGVkVmFsdWVzLmZpbHRlcihjID0+IGMudHlwZSA9PT0gJ2ltYWdlJylcbiAgICAgICAgY29uc3QgaW1hZ2VQYXN0ZUlkcyA9XG4gICAgICAgICAgaW1hZ2VDb250ZW50cy5sZW5ndGggPiAwID8gaW1hZ2VDb250ZW50cy5tYXAoYyA9PiBjLmlkKSA6IHVuZGVmaW5lZFxuXG4gICAgICAgIGxldCBtZXNzYWdlQ29udGVudDogc3RyaW5nIHwgQ29udGVudEJsb2NrUGFyYW1bXSA9IGlucHV0LnRyaW0oKVxuICAgICAgICBsZXQgcmVtb3RlQ29udGVudDogUmVtb3RlTWVzc2FnZUNvbnRlbnQgPSBpbnB1dC50cmltKClcbiAgICAgICAgaWYgKHBhc3RlZFZhbHVlcy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgY29uc3QgY29udGVudEJsb2NrczogQ29udGVudEJsb2NrUGFyYW1bXSA9IFtdXG4gICAgICAgICAgY29uc3QgcmVtb3RlQmxvY2tzOiBBcnJheTx7IHR5cGU6IHN0cmluZzsgW2tleTogc3RyaW5nXTogdW5rbm93biB9PiA9XG4gICAgICAgICAgICBbXVxuXG4gICAgICAgICAgY29uc3QgdHJpbW1lZElucHV0ID0gaW5wdXQudHJpbSgpXG4gICAgICAgICAgaWYgKHRyaW1tZWRJbnB1dCkge1xuICAgICAgICAgICAgY29udGVudEJsb2Nrcy5wdXNoKHsgdHlwZTogJ3RleHQnLCB0ZXh0OiB0cmltbWVkSW5wdXQgfSlcbiAgICAgICAgICAgIHJlbW90ZUJsb2Nrcy5wdXNoKHsgdHlwZTogJ3RleHQnLCB0ZXh0OiB0cmltbWVkSW5wdXQgfSlcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBmb3IgKGNvbnN0IHBhc3RlZCBvZiBwYXN0ZWRWYWx1ZXMpIHtcbiAgICAgICAgICAgIGlmIChwYXN0ZWQudHlwZSA9PT0gJ2ltYWdlJykge1xuICAgICAgICAgICAgICBjb25zdCBzb3VyY2UgPSB7XG4gICAgICAgICAgICAgICAgdHlwZTogJ2Jhc2U2NCcgYXMgY29uc3QsXG4gICAgICAgICAgICAgICAgbWVkaWFfdHlwZTogKHBhc3RlZC5tZWRpYVR5cGUgPz8gJ2ltYWdlL3BuZycpIGFzXG4gICAgICAgICAgICAgICAgICB8ICdpbWFnZS9qcGVnJ1xuICAgICAgICAgICAgICAgICAgfCAnaW1hZ2UvcG5nJ1xuICAgICAgICAgICAgICAgICAgfCAnaW1hZ2UvZ2lmJ1xuICAgICAgICAgICAgICAgICAgfCAnaW1hZ2Uvd2VicCcsXG4gICAgICAgICAgICAgICAgZGF0YTogcGFzdGVkLmNvbnRlbnQsXG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgY29udGVudEJsb2Nrcy5wdXNoKHsgdHlwZTogJ2ltYWdlJywgc291cmNlIH0pXG4gICAgICAgICAgICAgIHJlbW90ZUJsb2Nrcy5wdXNoKHsgdHlwZTogJ2ltYWdlJywgc291cmNlIH0pXG4gICAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgICBjb250ZW50QmxvY2tzLnB1c2goeyB0eXBlOiAndGV4dCcsIHRleHQ6IHBhc3RlZC5jb250ZW50IH0pXG4gICAgICAgICAgICAgIHJlbW90ZUJsb2Nrcy5wdXNoKHsgdHlwZTogJ3RleHQnLCB0ZXh0OiBwYXN0ZWQuY29udGVudCB9KVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIG1lc3NhZ2VDb250ZW50ID0gY29udGVudEJsb2Nrc1xuICAgICAgICAgIHJlbW90ZUNvbnRlbnQgPSByZW1vdGVCbG9ja3NcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIENyZWF0ZSBhbmQgYWRkIHVzZXIgbWVzc2FnZSB0byBVSVxuICAgICAgICAvLyBOb3RlOiBlbXB0eSBpbnB1dCBhbHJlYWR5IGhhbmRsZWQgYnkgZWFybHkgcmV0dXJuIGFib3ZlXG4gICAgICAgIGNvbnN0IHVzZXJNZXNzYWdlID0gY3JlYXRlVXNlck1lc3NhZ2Uoe1xuICAgICAgICAgIGNvbnRlbnQ6IG1lc3NhZ2VDb250ZW50LFxuICAgICAgICAgIGltYWdlUGFzdGVJZHMsXG4gICAgICAgIH0pXG4gICAgICAgIHNldE1lc3NhZ2VzKHByZXYgPT4gWy4uLnByZXYsIHVzZXJNZXNzYWdlXSlcblxuICAgICAgICAvLyBTZW5kIHRvIHJlbW90ZSBzZXNzaW9uXG4gICAgICAgIGF3YWl0IGFjdGl2ZVJlbW90ZS5zZW5kTWVzc2FnZShyZW1vdGVDb250ZW50LCB7XG4gICAgICAgICAgdXVpZDogdXNlck1lc3NhZ2UudXVpZCxcbiAgICAgICAgfSlcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG5cbiAgICAgIC8vIEVuc3VyZSBTZXNzaW9uU3RhcnQgaG9vayBjb250ZXh0IGlzIGF2YWlsYWJsZSBiZWZvcmUgdGhlIGZpcnN0IEFQSSBjYWxsLlxuICAgICAgYXdhaXQgYXdhaXRQZW5kaW5nSG9va3MoKVxuXG4gICAgICBhd2FpdCBoYW5kbGVQcm9tcHRTdWJtaXQoe1xuICAgICAgICBpbnB1dCxcbiAgICAgICAgaGVscGVycyxcbiAgICAgICAgcXVlcnlHdWFyZCxcbiAgICAgICAgaXNFeHRlcm5hbExvYWRpbmcsXG4gICAgICAgIG1vZGU6IGlucHV0TW9kZSxcbiAgICAgICAgY29tbWFuZHMsXG4gICAgICAgIG9uSW5wdXRDaGFuZ2U6IHNldElucHV0VmFsdWUsXG4gICAgICAgIHNldFBhc3RlZENvbnRlbnRzLFxuICAgICAgICBzZXRUb29sSlNYLFxuICAgICAgICBnZXRUb29sVXNlQ29udGV4dCxcbiAgICAgICAgbWVzc2FnZXM6IG1lc3NhZ2VzUmVmLmN1cnJlbnQsXG4gICAgICAgIG1haW5Mb29wTW9kZWwsXG4gICAgICAgIHBhc3RlZENvbnRlbnRzLFxuICAgICAgICBpZGVTZWxlY3Rpb24sXG4gICAgICAgIHNldFVzZXJJbnB1dE9uUHJvY2Vzc2luZyxcbiAgICAgICAgc2V0QWJvcnRDb250cm9sbGVyLFxuICAgICAgICBhYm9ydENvbnRyb2xsZXIsXG4gICAgICAgIG9uUXVlcnksXG4gICAgICAgIHNldEFwcFN0YXRlLFxuICAgICAgICBxdWVyeVNvdXJjZTogZ2V0UXVlcnlTb3VyY2VGb3JSRVBMKCksXG4gICAgICAgIG9uQmVmb3JlUXVlcnksXG4gICAgICAgIGNhblVzZVRvb2wsXG4gICAgICAgIGFkZE5vdGlmaWNhdGlvbixcbiAgICAgICAgc2V0TWVzc2FnZXMsXG4gICAgICAgIC8vIFJlYWQgdmlhIHJlZiBzbyBzdHJlYW1Nb2RlIGNhbiBiZSBkcm9wcGVkIGZyb20gb25TdWJtaXQgZGVwcyDigJRcbiAgICAgICAgLy8gaGFuZGxlUHJvbXB0U3VibWl0IG9ubHkgdXNlcyBpdCBmb3IgZGVidWcgbG9nICsgdGVsZW1ldHJ5IGV2ZW50LlxuICAgICAgICBzdHJlYW1Nb2RlOiBzdHJlYW1Nb2RlUmVmLmN1cnJlbnQsXG4gICAgICAgIGhhc0ludGVycnVwdGlibGVUb29sSW5Qcm9ncmVzczpcbiAgICAgICAgICBoYXNJbnRlcnJ1cHRpYmxlVG9vbEluUHJvZ3Jlc3NSZWYuY3VycmVudCxcbiAgICAgIH0pXG5cbiAgICAgIC8vIFJlc3RvcmUgc3Rhc2ggdGhhdCB3YXMgZGVmZXJyZWQgYWJvdmUuIFR3byBjYXNlczpcbiAgICAgIC8vIC0gU2xhc2ggY29tbWFuZDogaGFuZGxlUHJvbXB0U3VibWl0IGF3YWl0ZWQgdGhlIGZ1bGwgY29tbWFuZCBleGVjdXRpb25cbiAgICAgIC8vICAgKGluY2x1ZGluZyBpbnRlcmFjdGl2ZSBwaWNrZXJzKS4gUmVzdG9yaW5nIG5vdyBwbGFjZXMgdGhlIHN0YXNoIGJhY2sgaW5cbiAgICAgIC8vICAgdGhlIHZpc2libGUgaW5wdXQuXG4gICAgICAvLyAtIExvYWRpbmcgKHF1ZXVlZCk6IGhhbmRsZVByb21wdFN1Ym1pdCBlbnF1ZXVlZCArIGNsZWFyZWQgaW5wdXQsIHRoZW5cbiAgICAgIC8vICAgcmV0dXJuZWQgcXVpY2tseS4gUmVzdG9yaW5nIG5vdyBwbGFjZXMgdGhlIHN0YXNoIGJhY2sgYWZ0ZXIgdGhlIGNsZWFyLlxuICAgICAgaWYgKChpc1NsYXNoQ29tbWFuZCB8fCBpc0xvYWRpbmcpICYmIHN0YXNoZWRQcm9tcHQgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICBzZXRJbnB1dFZhbHVlKHN0YXNoZWRQcm9tcHQudGV4dClcbiAgICAgICAgaGVscGVycy5zZXRDdXJzb3JPZmZzZXQoc3Rhc2hlZFByb21wdC5jdXJzb3JPZmZzZXQpXG4gICAgICAgIHNldFBhc3RlZENvbnRlbnRzKHN0YXNoZWRQcm9tcHQucGFzdGVkQ29udGVudHMpXG4gICAgICAgIHNldFN0YXNoZWRQcm9tcHQodW5kZWZpbmVkKVxuICAgICAgfVxuICAgIH0sXG4gICAgW1xuICAgICAgcXVlcnlHdWFyZCxcbiAgICAgIC8vIGlzTG9hZGluZyBpcyByZWFkIGF0IHRoZSAhaXNMb2FkaW5nIGNoZWNrcyBhYm92ZSBmb3IgaW5wdXQtY2xlYXJpbmdcbiAgICAgIC8vIGFuZCBzdWJtaXRDb3VudCBnYXRpbmcuIEl0J3MgZGVyaXZlZCBmcm9tIGlzUXVlcnlBY3RpdmUgfHwgaXNFeHRlcm5hbExvYWRpbmcsXG4gICAgICAvLyBzbyBpbmNsdWRpbmcgaXQgaGVyZSBlbnN1cmVzIHRoZSBjbG9zdXJlIGNhcHR1cmVzIHRoZSBmcmVzaCB2YWx1ZS5cbiAgICAgIGlzTG9hZGluZyxcbiAgICAgIGlzRXh0ZXJuYWxMb2FkaW5nLFxuICAgICAgaW5wdXRNb2RlLFxuICAgICAgY29tbWFuZHMsXG4gICAgICBzZXRJbnB1dFZhbHVlLFxuICAgICAgc2V0SW5wdXRNb2RlLFxuICAgICAgc2V0UGFzdGVkQ29udGVudHMsXG4gICAgICBzZXRTdWJtaXRDb3VudCxcbiAgICAgIHNldElERVNlbGVjdGlvbixcbiAgICAgIHNldFRvb2xKU1gsXG4gICAgICBnZXRUb29sVXNlQ29udGV4dCxcbiAgICAgIC8vIG1lc3NhZ2VzIGlzIHJlYWQgdmlhIG1lc3NhZ2VzUmVmLmN1cnJlbnQgaW5zaWRlIHRoZSBjYWxsYmFjayB0b1xuICAgICAgLy8ga2VlcCBvblN1Ym1pdCBzdGFibGUgYWNyb3NzIG1lc3NhZ2UgdXBkYXRlcyAoc2VlIEwyMzg0L0wyNDAwL0wyNjYyKS5cbiAgICAgIC8vIFdpdGhvdXQgdGhpcywgZWFjaCBzZXRNZXNzYWdlcyBjYWxsICh+MzDDlyBwZXIgdHVybikgcmVjcmVhdGVzXG4gICAgICAvLyBvblN1Ym1pdCwgcGlubmluZyB0aGUgUkVQTCByZW5kZXIgc2NvcGUgKDE3NzZCKSArIHRoYXQgcmVuZGVyJ3NcbiAgICAgIC8vIG1lc3NhZ2VzIGFycmF5IGluIGRvd25zdHJlYW0gY2xvc3VyZXMgKFByb21wdElucHV0LCBoYW5kbGVBdXRvUnVuSXNzdWUpLlxuICAgICAgLy8gSGVhcCBhbmFseXNpcyBzaG93ZWQgfjkgUkVQTCBzY29wZXMgYW5kIH4xNSBtZXNzYWdlcyBhcnJheSB2ZXJzaW9uc1xuICAgICAgLy8gYWNjdW11bGF0aW5nIGFmdGVyICMyMDE3NC8jMjAxNzUsIGFsbCB0cmFjZWQgdG8gdGhpcyBkZXAuXG4gICAgICBtYWluTG9vcE1vZGVsLFxuICAgICAgcGFzdGVkQ29udGVudHMsXG4gICAgICBpZGVTZWxlY3Rpb24sXG4gICAgICBzZXRVc2VySW5wdXRPblByb2Nlc3NpbmcsXG4gICAgICBzZXRBYm9ydENvbnRyb2xsZXIsXG4gICAgICBhZGROb3RpZmljYXRpb24sXG4gICAgICBvblF1ZXJ5LFxuICAgICAgc3Rhc2hlZFByb21wdCxcbiAgICAgIHNldFN0YXNoZWRQcm9tcHQsXG4gICAgICBzZXRBcHBTdGF0ZSxcbiAgICAgIG9uQmVmb3JlUXVlcnksXG4gICAgICBjYW5Vc2VUb29sLFxuICAgICAgcmVtb3RlU2Vzc2lvbixcbiAgICAgIHNldE1lc3NhZ2VzLFxuICAgICAgYXdhaXRQZW5kaW5nSG9va3MsXG4gICAgICByZXBpblNjcm9sbCxcbiAgICBdLFxuICApXG5cbiAgLy8gQ2FsbGJhY2sgZm9yIHdoZW4gdXNlciBzdWJtaXRzIGlucHV0IHdoaWxlIHZpZXdpbmcgYSB0ZWFtbWF0ZSdzIHRyYW5zY3JpcHRcbiAgY29uc3Qgb25BZ2VudFN1Ym1pdCA9IHVzZUNhbGxiYWNrKFxuICAgIGFzeW5jIChcbiAgICAgIGlucHV0OiBzdHJpbmcsXG4gICAgICB0YXNrOiBJblByb2Nlc3NUZWFtbWF0ZVRhc2tTdGF0ZSB8IExvY2FsQWdlbnRUYXNrU3RhdGUsXG4gICAgICBoZWxwZXJzOiBQcm9tcHRJbnB1dEhlbHBlcnMsXG4gICAgKSA9PiB7XG4gICAgICBpZiAoaXNMb2NhbEFnZW50VGFzayh0YXNrKSkge1xuICAgICAgICBhcHBlbmRNZXNzYWdlVG9Mb2NhbEFnZW50KFxuICAgICAgICAgIHRhc2suaWQsXG4gICAgICAgICAgY3JlYXRlVXNlck1lc3NhZ2UoeyBjb250ZW50OiBpbnB1dCB9KSxcbiAgICAgICAgICBzZXRBcHBTdGF0ZSxcbiAgICAgICAgKVxuICAgICAgICBpZiAodGFzay5zdGF0dXMgPT09ICdydW5uaW5nJykge1xuICAgICAgICAgIHF1ZXVlUGVuZGluZ01lc3NhZ2UodGFzay5pZCwgaW5wdXQsIHNldEFwcFN0YXRlKVxuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHZvaWQgcmVzdW1lQWdlbnRCYWNrZ3JvdW5kKHtcbiAgICAgICAgICAgIGFnZW50SWQ6IHRhc2suaWQsXG4gICAgICAgICAgICBwcm9tcHQ6IGlucHV0LFxuICAgICAgICAgICAgdG9vbFVzZUNvbnRleHQ6IGdldFRvb2xVc2VDb250ZXh0KFxuICAgICAgICAgICAgICBtZXNzYWdlc1JlZi5jdXJyZW50LFxuICAgICAgICAgICAgICBbXSxcbiAgICAgICAgICAgICAgbmV3IEFib3J0Q29udHJvbGxlcigpLFxuICAgICAgICAgICAgICBtYWluTG9vcE1vZGVsLFxuICAgICAgICAgICAgKSxcbiAgICAgICAgICAgIGNhblVzZVRvb2wsXG4gICAgICAgICAgfSkuY2F0Y2goZXJyID0+IHtcbiAgICAgICAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgICAgICAgYHJlc3VtZUFnZW50QmFja2dyb3VuZCBmYWlsZWQ6ICR7ZXJyb3JNZXNzYWdlKGVycil9YCxcbiAgICAgICAgICAgIClcbiAgICAgICAgICAgIGFkZE5vdGlmaWNhdGlvbih7XG4gICAgICAgICAgICAgIGtleTogYHJlc3VtZS1hZ2VudC1mYWlsZWQtJHt0YXNrLmlkfWAsXG4gICAgICAgICAgICAgIGpzeDogKFxuICAgICAgICAgICAgICAgIDxUZXh0IGNvbG9yPVwiZXJyb3JcIj5cbiAgICAgICAgICAgICAgICAgIEZhaWxlZCB0byByZXN1bWUgYWdlbnQ6IHtlcnJvck1lc3NhZ2UoZXJyKX1cbiAgICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgIHByaW9yaXR5OiAnbG93JyxcbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgfSlcbiAgICAgICAgfVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgaW5qZWN0VXNlck1lc3NhZ2VUb1RlYW1tYXRlKHRhc2suaWQsIGlucHV0LCBzZXRBcHBTdGF0ZSlcbiAgICAgIH1cbiAgICAgIHNldElucHV0VmFsdWUoJycpXG4gICAgICBoZWxwZXJzLnNldEN1cnNvck9mZnNldCgwKVxuICAgICAgaGVscGVycy5jbGVhckJ1ZmZlcigpXG4gICAgfSxcbiAgICBbXG4gICAgICBzZXRBcHBTdGF0ZSxcbiAgICAgIHNldElucHV0VmFsdWUsXG4gICAgICBnZXRUb29sVXNlQ29udGV4dCxcbiAgICAgIGNhblVzZVRvb2wsXG4gICAgICBtYWluTG9vcE1vZGVsLFxuICAgICAgYWRkTm90aWZpY2F0aW9uLFxuICAgIF0sXG4gIClcblxuICAvLyBIYW5kbGVycyBmb3IgYXV0by1ydW4gL2lzc3VlIG9yIC9nb29kLWNsYXVkZSAoZGVmaW5lZCBhZnRlciBvblN1Ym1pdClcbiAgY29uc3QgaGFuZGxlQXV0b1J1bklzc3VlID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIGNvbnN0IGNvbW1hbmQgPSBhdXRvUnVuSXNzdWVSZWFzb25cbiAgICAgID8gZ2V0QXV0b1J1bkNvbW1hbmQoYXV0b1J1bklzc3VlUmVhc29uKVxuICAgICAgOiAnL2lzc3VlJ1xuICAgIHNldEF1dG9SdW5Jc3N1ZVJlYXNvbihudWxsKSAvLyBDbGVhciB0aGUgc3RhdGVcbiAgICBvblN1Ym1pdChjb21tYW5kLCB7XG4gICAgICBzZXRDdXJzb3JPZmZzZXQ6ICgpID0+IHt9LFxuICAgICAgY2xlYXJCdWZmZXI6ICgpID0+IHt9LFxuICAgICAgcmVzZXRIaXN0b3J5OiAoKSA9PiB7fSxcbiAgICB9KS5jYXRjaChlcnIgPT4ge1xuICAgICAgbG9nRm9yRGVidWdnaW5nKGBBdXRvLXJ1biAke2NvbW1hbmR9IGZhaWxlZDogJHtlcnJvck1lc3NhZ2UoZXJyKX1gKVxuICAgIH0pXG4gIH0sIFtvblN1Ym1pdCwgYXV0b1J1bklzc3VlUmVhc29uXSlcblxuICBjb25zdCBoYW5kbGVDYW5jZWxBdXRvUnVuSXNzdWUgPSB1c2VDYWxsYmFjaygoKSA9PiB7XG4gICAgc2V0QXV0b1J1bklzc3VlUmVhc29uKG51bGwpXG4gIH0sIFtdKVxuXG4gIC8vIEhhbmRsZXIgZm9yIHdoZW4gdXNlciBwcmVzc2VzIDEgb24gc3VydmV5IHRoYW5rcyBzY3JlZW4gdG8gc2hhcmUgZGV0YWlsc1xuICBjb25zdCBoYW5kbGVTdXJ2ZXlSZXF1ZXN0RmVlZGJhY2sgPSB1c2VDYWxsYmFjaygoKSA9PiB7XG4gICAgY29uc3QgY29tbWFuZCA9IFwiZXh0ZXJuYWxcIiA9PT0gJ2FudCcgPyAnL2lzc3VlJyA6ICcvZmVlZGJhY2snXG4gICAgb25TdWJtaXQoY29tbWFuZCwge1xuICAgICAgc2V0Q3Vyc29yT2Zmc2V0OiAoKSA9PiB7fSxcbiAgICAgIGNsZWFyQnVmZmVyOiAoKSA9PiB7fSxcbiAgICAgIHJlc2V0SGlzdG9yeTogKCkgPT4ge30sXG4gICAgfSkuY2F0Y2goZXJyID0+IHtcbiAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgYFN1cnZleSBmZWVkYmFjayByZXF1ZXN0IGZhaWxlZDogJHtlcnIgaW5zdGFuY2VvZiBFcnJvciA/IGVyci5tZXNzYWdlIDogU3RyaW5nKGVycil9YCxcbiAgICAgIClcbiAgICB9KVxuICB9LCBbb25TdWJtaXRdKVxuXG4gIC8vIG9uU3VibWl0IGlzIHVuc3RhYmxlIChkZXBzIGluY2x1ZGUgYG1lc3NhZ2VzYCB3aGljaCBjaGFuZ2VzIGV2ZXJ5IHR1cm4pLlxuICAvLyBgaGFuZGxlT3BlblJhdGVMaW1pdE9wdGlvbnNgIGlzIHByb3AtZHJpbGxlZCB0byBldmVyeSBNZXNzYWdlUm93LCBhbmQgZWFjaFxuICAvLyBNZXNzYWdlUm93IGZpYmVyIHBpbnMgdGhlIGNsb3N1cmUgKGFuZCB0cmFuc2l0aXZlbHkgdGhlIGVudGlyZSBSRVBMIHJlbmRlclxuICAvLyBzY29wZSwgfjEuOEtCKSBhdCBtb3VudCB0aW1lLiBVc2luZyBhIHJlZiBrZWVwcyB0aGlzIGNhbGxiYWNrIHN0YWJsZSBzb1xuICAvLyBvbGQgUkVQTCBzY29wZXMgY2FuIGJlIEdDJ2Qg4oCUIHNhdmVzIH4zNU1CIG92ZXIgYSAxMDAwLXR1cm4gc2Vzc2lvbi5cbiAgY29uc3Qgb25TdWJtaXRSZWYgPSB1c2VSZWYob25TdWJtaXQpXG4gIG9uU3VibWl0UmVmLmN1cnJlbnQgPSBvblN1Ym1pdFxuICBjb25zdCBoYW5kbGVPcGVuUmF0ZUxpbWl0T3B0aW9ucyA9IHVzZUNhbGxiYWNrKCgpID0+IHtcbiAgICB2b2lkIG9uU3VibWl0UmVmLmN1cnJlbnQoJy9yYXRlLWxpbWl0LW9wdGlvbnMnLCB7XG4gICAgICBzZXRDdXJzb3JPZmZzZXQ6ICgpID0+IHt9LFxuICAgICAgY2xlYXJCdWZmZXI6ICgpID0+IHt9LFxuICAgICAgcmVzZXRIaXN0b3J5OiAoKSA9PiB7fSxcbiAgICB9KVxuICB9LCBbXSlcblxuICBjb25zdCBoYW5kbGVFeGl0ID0gdXNlQ2FsbGJhY2soYXN5bmMgKCkgPT4ge1xuICAgIHNldElzRXhpdGluZyh0cnVlKVxuICAgIC8vIEluIGJnIHNlc3Npb25zLCBhbHdheXMgZGV0YWNoIGluc3RlYWQgb2Yga2lsbCDigJQgZXZlbiB3aGVuIGEgd29ya3RyZWUgaXNcbiAgICAvLyBhY3RpdmUuIFdpdGhvdXQgdGhpcyBndWFyZCwgdGhlIHdvcmt0cmVlIGJyYW5jaCBiZWxvdyBzaG9ydC1jaXJjdWl0cyBpbnRvXG4gICAgLy8gRXhpdEZsb3cgKHdoaWNoIGNhbGxzIGdyYWNlZnVsU2h1dGRvd24pIGJlZm9yZSBleGl0LnRzeCBpcyBldmVyIGxvYWRlZC5cbiAgICBpZiAoZmVhdHVyZSgnQkdfU0VTU0lPTlMnKSAmJiBpc0JnU2Vzc2lvbigpKSB7XG4gICAgICBzcGF3blN5bmMoJ3RtdXgnLCBbJ2RldGFjaC1jbGllbnQnXSwgeyBzdGRpbzogJ2lnbm9yZScgfSlcbiAgICAgIHNldElzRXhpdGluZyhmYWxzZSlcbiAgICAgIHJldHVyblxuICAgIH1cbiAgICBjb25zdCBzaG93V29ya3RyZWUgPSBnZXRDdXJyZW50V29ya3RyZWVTZXNzaW9uKCkgIT09IG51bGxcbiAgICBpZiAoc2hvd1dvcmt0cmVlKSB7XG4gICAgICBzZXRFeGl0RmxvdyhcbiAgICAgICAgPEV4aXRGbG93XG4gICAgICAgICAgc2hvd1dvcmt0cmVlXG4gICAgICAgICAgb25Eb25lPXsoKSA9PiB7fX1cbiAgICAgICAgICBvbkNhbmNlbD17KCkgPT4ge1xuICAgICAgICAgICAgc2V0RXhpdEZsb3cobnVsbClcbiAgICAgICAgICAgIHNldElzRXhpdGluZyhmYWxzZSlcbiAgICAgICAgICB9fVxuICAgICAgICAvPixcbiAgICAgIClcbiAgICAgIHJldHVyblxuICAgIH1cbiAgICBjb25zdCBleGl0TW9kID0gYXdhaXQgZXhpdC5sb2FkKClcbiAgICBjb25zdCBleGl0Rmxvd1Jlc3VsdCA9IGF3YWl0IGV4aXRNb2QuY2FsbCgoKSA9PiB7fSlcbiAgICBzZXRFeGl0RmxvdyhleGl0Rmxvd1Jlc3VsdClcbiAgICAvLyBJZiBjYWxsKCkgcmV0dXJuZWQgd2l0aG91dCBraWxsaW5nIHRoZSBwcm9jZXNzIChiZyBzZXNzaW9uIGRldGFjaCksXG4gICAgLy8gY2xlYXIgaXNFeGl0aW5nIHNvIHRoZSBVSSBpcyB1c2FibGUgb24gcmVhdHRhY2guIE5vLW9wIG9uIHRoZSBub3JtYWxcbiAgICAvLyBwYXRoIOKAlCBncmFjZWZ1bFNodXRkb3duJ3MgcHJvY2Vzcy5leGl0KCkgbWVhbnMgd2UgbmV2ZXIgZ2V0IGhlcmUuXG4gICAgaWYgKGV4aXRGbG93UmVzdWx0ID09PSBudWxsKSB7XG4gICAgICBzZXRJc0V4aXRpbmcoZmFsc2UpXG4gICAgfVxuICB9LCBbXSlcblxuICBjb25zdCBoYW5kbGVTaG93TWVzc2FnZVNlbGVjdG9yID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIHNldElzTWVzc2FnZVNlbGVjdG9yVmlzaWJsZShwcmV2ID0+ICFwcmV2KVxuICB9LCBbXSlcblxuICAvLyBSZXdpbmQgY29udmVyc2F0aW9uIHN0YXRlIHRvIGp1c3QgYmVmb3JlIGBtZXNzYWdlYDogc2xpY2UgbWVzc2FnZXMsXG4gIC8vIHJlc2V0IGNvbnZlcnNhdGlvbiBJRCwgbWljcm9jb21wYWN0IHN0YXRlLCBwZXJtaXNzaW9uIG1vZGUsIHByb21wdCBzdWdnZXN0aW9uLlxuICAvLyBEb2VzIE5PVCB0b3VjaCB0aGUgcHJvbXB0IGlucHV0LiBJbmRleCBpcyBjb21wdXRlZCBmcm9tIG1lc3NhZ2VzUmVmIChhbHdheXNcbiAgLy8gZnJlc2ggdmlhIHRoZSBzZXRNZXNzYWdlcyB3cmFwcGVyKSBzbyBjYWxsZXJzIGRvbid0IG5lZWQgdG8gd29ycnkgYWJvdXRcbiAgLy8gc3RhbGUgY2xvc3VyZXMuXG4gIGNvbnN0IHJld2luZENvbnZlcnNhdGlvblRvID0gdXNlQ2FsbGJhY2soXG4gICAgKG1lc3NhZ2U6IFVzZXJNZXNzYWdlKSA9PiB7XG4gICAgICBjb25zdCBwcmV2ID0gbWVzc2FnZXNSZWYuY3VycmVudFxuICAgICAgY29uc3QgbWVzc2FnZUluZGV4ID0gcHJldi5sYXN0SW5kZXhPZihtZXNzYWdlKVxuICAgICAgaWYgKG1lc3NhZ2VJbmRleCA9PT0gLTEpIHJldHVyblxuXG4gICAgICBsb2dFdmVudCgndGVuZ3VfY29udmVyc2F0aW9uX3Jld2luZCcsIHtcbiAgICAgICAgcHJlUmV3aW5kTWVzc2FnZUNvdW50OiBwcmV2Lmxlbmd0aCxcbiAgICAgICAgcG9zdFJld2luZE1lc3NhZ2VDb3VudDogbWVzc2FnZUluZGV4LFxuICAgICAgICBtZXNzYWdlc1JlbW92ZWQ6IHByZXYubGVuZ3RoIC0gbWVzc2FnZUluZGV4LFxuICAgICAgICByZXdpbmRUb01lc3NhZ2VJbmRleDogbWVzc2FnZUluZGV4LFxuICAgICAgfSlcbiAgICAgIHNldE1lc3NhZ2VzKHByZXYuc2xpY2UoMCwgbWVzc2FnZUluZGV4KSlcbiAgICAgIC8vIENhcmVmdWwsIHRoaXMgaGFzIHRvIGhhcHBlbiBhZnRlciBzZXRNZXNzYWdlc1xuICAgICAgc2V0Q29udmVyc2F0aW9uSWQocmFuZG9tVVVJRCgpKVxuICAgICAgLy8gUmVzZXQgY2FjaGVkIG1pY3JvY29tcGFjdCBzdGF0ZSBzbyBzdGFsZSBwaW5uZWQgY2FjaGUgZWRpdHNcbiAgICAgIC8vIGRvbid0IHJlZmVyZW5jZSB0b29sX3VzZV9pZHMgZnJvbSB0cnVuY2F0ZWQgbWVzc2FnZXNcbiAgICAgIHJlc2V0TWljcm9jb21wYWN0U3RhdGUoKVxuICAgICAgaWYgKGZlYXR1cmUoJ0NPTlRFWFRfQ09MTEFQU0UnKSkge1xuICAgICAgICAvLyBSZXdpbmQgdHJ1bmNhdGVzIHRoZSBSRVBMIGFycmF5LiBDb21taXRzIHdob3NlIGFyY2hpdmVkIHNwYW5cbiAgICAgICAgLy8gd2FzIHBhc3QgdGhlIHJld2luZCBwb2ludCBjYW4ndCBiZSBwcm9qZWN0ZWQgYW55bW9yZVxuICAgICAgICAvLyAocHJvamVjdFZpZXcgc2lsZW50bHkgc2tpcHMgdGhlbSkgYnV0IHRoZSBzdGFnZWQgcXVldWUgYW5kIElEXG4gICAgICAgIC8vIG1hcHMgcmVmZXJlbmNlIHN0YWxlIHV1aWRzLiBTaW1wbGVzdCBzYWZlIHJlc2V0OiBkcm9wXG4gICAgICAgIC8vIGV2ZXJ5dGhpbmcuIFRoZSBjdHgtYWdlbnQgd2lsbCByZS1zdGFnZSBvbiB0aGUgbmV4dFxuICAgICAgICAvLyB0aHJlc2hvbGQgY3Jvc3NpbmcuXG4gICAgICAgIC8qIGVzbGludC1kaXNhYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby1yZXF1aXJlLWltcG9ydHMgKi9cbiAgICAgICAgOyhcbiAgICAgICAgICByZXF1aXJlKCcuLi9zZXJ2aWNlcy9jb250ZXh0Q29sbGFwc2UvaW5kZXguanMnKSBhcyB0eXBlb2YgaW1wb3J0KCcuLi9zZXJ2aWNlcy9jb250ZXh0Q29sbGFwc2UvaW5kZXguanMnKVxuICAgICAgICApLnJlc2V0Q29udGV4dENvbGxhcHNlKClcbiAgICAgICAgLyogZXNsaW50LWVuYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tcmVxdWlyZS1pbXBvcnRzICovXG4gICAgICB9XG5cbiAgICAgIC8vIFJlc3RvcmUgc3RhdGUgZnJvbSB0aGUgbWVzc2FnZSB3ZSdyZSByZXdpbmRpbmcgdG9cbiAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4gKHtcbiAgICAgICAgLi4ucHJldixcbiAgICAgICAgLy8gUmVzdG9yZSBwZXJtaXNzaW9uIG1vZGUgZnJvbSB0aGUgbWVzc2FnZVxuICAgICAgICB0b29sUGVybWlzc2lvbkNvbnRleHQ6XG4gICAgICAgICAgbWVzc2FnZS5wZXJtaXNzaW9uTW9kZSAmJlxuICAgICAgICAgIHByZXYudG9vbFBlcm1pc3Npb25Db250ZXh0Lm1vZGUgIT09IG1lc3NhZ2UucGVybWlzc2lvbk1vZGVcbiAgICAgICAgICAgID8ge1xuICAgICAgICAgICAgICAgIC4uLnByZXYudG9vbFBlcm1pc3Npb25Db250ZXh0LFxuICAgICAgICAgICAgICAgIG1vZGU6IG1lc3NhZ2UucGVybWlzc2lvbk1vZGUsXG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIDogcHJldi50b29sUGVybWlzc2lvbkNvbnRleHQsXG4gICAgICAgIC8vIENsZWFyIHN0YWxlIHByb21wdCBzdWdnZXN0aW9uIGZyb20gcHJldmlvdXMgY29udmVyc2F0aW9uIHN0YXRlXG4gICAgICAgIHByb21wdFN1Z2dlc3Rpb246IHtcbiAgICAgICAgICB0ZXh0OiBudWxsLFxuICAgICAgICAgIHByb21wdElkOiBudWxsLFxuICAgICAgICAgIHNob3duQXQ6IDAsXG4gICAgICAgICAgYWNjZXB0ZWRBdDogMCxcbiAgICAgICAgICBnZW5lcmF0aW9uUmVxdWVzdElkOiBudWxsLFxuICAgICAgICB9LFxuICAgICAgfSkpXG4gICAgfSxcbiAgICBbc2V0TWVzc2FnZXMsIHNldEFwcFN0YXRlXSxcbiAgKVxuXG4gIC8vIFN5bmNocm9ub3VzIHJld2luZCArIGlucHV0IHBvcHVsYXRpb24uIFVzZWQgZGlyZWN0bHkgYnkgYXV0by1yZXN0b3JlIG9uXG4gIC8vIGludGVycnVwdCAoc28gUmVhY3QgYmF0Y2hlcyB3aXRoIHRoZSBhYm9ydCdzIHNldE1lc3NhZ2VzIOKGkiBzaW5nbGUgcmVuZGVyLFxuICAvLyBubyBmbGlja2VyKS4gTWVzc2FnZVNlbGVjdG9yIHdyYXBzIHRoaXMgaW4gc2V0SW1tZWRpYXRlIHZpYSBoYW5kbGVSZXN0b3JlTWVzc2FnZS5cbiAgY29uc3QgcmVzdG9yZU1lc3NhZ2VTeW5jID0gdXNlQ2FsbGJhY2soXG4gICAgKG1lc3NhZ2U6IFVzZXJNZXNzYWdlKSA9PiB7XG4gICAgICByZXdpbmRDb252ZXJzYXRpb25UbyhtZXNzYWdlKVxuXG4gICAgICBjb25zdCByID0gdGV4dEZvclJlc3VibWl0KG1lc3NhZ2UpXG4gICAgICBpZiAocikge1xuICAgICAgICBzZXRJbnB1dFZhbHVlKHIudGV4dClcbiAgICAgICAgc2V0SW5wdXRNb2RlKHIubW9kZSlcbiAgICAgIH1cblxuICAgICAgLy8gUmVzdG9yZSBwYXN0ZWQgaW1hZ2VzXG4gICAgICBpZiAoXG4gICAgICAgIEFycmF5LmlzQXJyYXkobWVzc2FnZS5tZXNzYWdlLmNvbnRlbnQpICYmXG4gICAgICAgIG1lc3NhZ2UubWVzc2FnZS5jb250ZW50LnNvbWUoYmxvY2sgPT4gYmxvY2sudHlwZSA9PT0gJ2ltYWdlJylcbiAgICAgICkge1xuICAgICAgICBjb25zdCBpbWFnZUJsb2NrczogQXJyYXk8SW1hZ2VCbG9ja1BhcmFtPiA9XG4gICAgICAgICAgbWVzc2FnZS5tZXNzYWdlLmNvbnRlbnQuZmlsdGVyKGJsb2NrID0+IGJsb2NrLnR5cGUgPT09ICdpbWFnZScpXG4gICAgICAgIGlmIChpbWFnZUJsb2Nrcy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgY29uc3QgbmV3UGFzdGVkQ29udGVudHM6IFJlY29yZDxudW1iZXIsIFBhc3RlZENvbnRlbnQ+ID0ge31cbiAgICAgICAgICBpbWFnZUJsb2Nrcy5mb3JFYWNoKChibG9jaywgaW5kZXgpID0+IHtcbiAgICAgICAgICAgIGlmIChibG9jay5zb3VyY2UudHlwZSA9PT0gJ2Jhc2U2NCcpIHtcbiAgICAgICAgICAgICAgY29uc3QgaWQgPSBtZXNzYWdlLmltYWdlUGFzdGVJZHM/LltpbmRleF0gPz8gaW5kZXggKyAxXG4gICAgICAgICAgICAgIG5ld1Bhc3RlZENvbnRlbnRzW2lkXSA9IHtcbiAgICAgICAgICAgICAgICBpZCxcbiAgICAgICAgICAgICAgICB0eXBlOiAnaW1hZ2UnLFxuICAgICAgICAgICAgICAgIGNvbnRlbnQ6IGJsb2NrLnNvdXJjZS5kYXRhLFxuICAgICAgICAgICAgICAgIG1lZGlhVHlwZTogYmxvY2suc291cmNlLm1lZGlhX3R5cGUsXG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9KVxuICAgICAgICAgIHNldFBhc3RlZENvbnRlbnRzKG5ld1Bhc3RlZENvbnRlbnRzKVxuICAgICAgICB9XG4gICAgICB9XG4gICAgfSxcbiAgICBbcmV3aW5kQ29udmVyc2F0aW9uVG8sIHNldElucHV0VmFsdWVdLFxuICApXG4gIHJlc3RvcmVNZXNzYWdlU3luY1JlZi5jdXJyZW50ID0gcmVzdG9yZU1lc3NhZ2VTeW5jXG5cbiAgLy8gTWVzc2FnZVNlbGVjdG9yIHBhdGg6IGRlZmVyIHZpYSBzZXRJbW1lZGlhdGUgc28gdGhlIFwiSW50ZXJydXB0ZWRcIiBtZXNzYWdlXG4gIC8vIHJlbmRlcnMgdG8gc3RhdGljIG91dHB1dCBiZWZvcmUgcmV3aW5kIOKAlCBvdGhlcndpc2UgaXQgcmVtYWlucyB2ZXN0aWdpYWxcbiAgLy8gYXQgdGhlIHRvcCBvZiB0aGUgc2NyZWVuLlxuICBjb25zdCBoYW5kbGVSZXN0b3JlTWVzc2FnZSA9IHVzZUNhbGxiYWNrKFxuICAgIGFzeW5jIChtZXNzYWdlOiBVc2VyTWVzc2FnZSkgPT4ge1xuICAgICAgc2V0SW1tZWRpYXRlKFxuICAgICAgICAocmVzdG9yZSwgbWVzc2FnZSkgPT4gcmVzdG9yZShtZXNzYWdlKSxcbiAgICAgICAgcmVzdG9yZU1lc3NhZ2VTeW5jLFxuICAgICAgICBtZXNzYWdlLFxuICAgICAgKVxuICAgIH0sXG4gICAgW3Jlc3RvcmVNZXNzYWdlU3luY10sXG4gIClcblxuICAvLyBOb3QgbWVtb2l6ZWQg4oCUIGhvb2sgc3RvcmVzIGNhcHMgdmlhIHJlZiwgcmVhZHMgbGF0ZXN0IGNsb3N1cmUgYXQgZGlzcGF0Y2guXG4gIC8vIDI0LWNoYXIgcHJlZml4OiBkZXJpdmVVVUlEIHByZXNlcnZlcyBmaXJzdCAyNCwgcmVuZGVyYWJsZSB1dWlkIHByZWZpeC1tYXRjaGVzIHJhdyBzb3VyY2UuXG4gIGNvbnN0IGZpbmRSYXdJbmRleCA9ICh1dWlkOiBzdHJpbmcpID0+IHtcbiAgICBjb25zdCBwcmVmaXggPSB1dWlkLnNsaWNlKDAsIDI0KVxuICAgIHJldHVybiBtZXNzYWdlcy5maW5kSW5kZXgobSA9PiBtLnV1aWQuc2xpY2UoMCwgMjQpID09PSBwcmVmaXgpXG4gIH1cbiAgY29uc3QgbWVzc2FnZUFjdGlvbkNhcHM6IE1lc3NhZ2VBY3Rpb25DYXBzID0ge1xuICAgIGNvcHk6IHRleHQgPT5cbiAgICAgIC8vIHNldENsaXBib2FyZCBSRVRVUk5TIE9TQyA1MiDigJQgY2FsbGVyIG11c3Qgc3Rkb3V0LndyaXRlICh0bXV4IHNpZGUtZWZmZWN0cyBsb2FkLWJ1ZmZlciwgYnV0IHRoYXQncyB0bXV4LW9ubHkpLlxuICAgICAgdm9pZCBzZXRDbGlwYm9hcmQodGV4dCkudGhlbihyYXcgPT4ge1xuICAgICAgICBpZiAocmF3KSBwcm9jZXNzLnN0ZG91dC53cml0ZShyYXcpXG4gICAgICAgIGFkZE5vdGlmaWNhdGlvbih7XG4gICAgICAgICAgLy8gU2FtZSBrZXkgYXMgdGV4dC1zZWxlY3Rpb24gY29weSDigJQgcmVwZWF0ZWQgY29waWVzIHJlcGxhY2UgdG9hc3QsIGRvbid0IHF1ZXVlLlxuICAgICAgICAgIGtleTogJ3NlbGVjdGlvbi1jb3BpZWQnLFxuICAgICAgICAgIHRleHQ6ICdjb3BpZWQnLFxuICAgICAgICAgIGNvbG9yOiAnc3VjY2VzcycsXG4gICAgICAgICAgcHJpb3JpdHk6ICdpbW1lZGlhdGUnLFxuICAgICAgICAgIHRpbWVvdXRNczogMjAwMCxcbiAgICAgICAgfSlcbiAgICAgIH0pLFxuICAgIGVkaXQ6IGFzeW5jIG1zZyA9PiB7XG4gICAgICAvLyBTYW1lIHNraXAtY29uZmlybSBjaGVjayBhcyAvcmV3aW5kOiBsb3NzbGVzcyDihpIgZGlyZWN0LCBlbHNlIGNvbmZpcm0gZGlhbG9nLlxuICAgICAgY29uc3QgcmF3SWR4ID0gZmluZFJhd0luZGV4KG1zZy51dWlkKVxuICAgICAgY29uc3QgcmF3ID0gcmF3SWR4ID49IDAgPyBtZXNzYWdlc1tyYXdJZHhdIDogdW5kZWZpbmVkXG4gICAgICBpZiAoIXJhdyB8fCAhc2VsZWN0YWJsZVVzZXJNZXNzYWdlc0ZpbHRlcihyYXcpKSByZXR1cm5cbiAgICAgIGNvbnN0IG5vRmlsZUNoYW5nZXMgPSAhKGF3YWl0IGZpbGVIaXN0b3J5SGFzQW55Q2hhbmdlcyhcbiAgICAgICAgZmlsZUhpc3RvcnksXG4gICAgICAgIHJhdy51dWlkLFxuICAgICAgKSlcbiAgICAgIGNvbnN0IG9ubHlTeW50aGV0aWMgPSBtZXNzYWdlc0FmdGVyQXJlT25seVN5bnRoZXRpYyhtZXNzYWdlcywgcmF3SWR4KVxuICAgICAgaWYgKG5vRmlsZUNoYW5nZXMgJiYgb25seVN5bnRoZXRpYykge1xuICAgICAgICAvLyByZXdpbmRDb252ZXJzYXRpb25UbydzIHNldE1lc3NhZ2VzIHJhY2VzIHN0cmVhbSBhcHBlbmRzIOKAlCBjYW5jZWwgZmlyc3QgKGlkZW1wb3RlbnQpLlxuICAgICAgICBvbkNhbmNlbCgpXG4gICAgICAgIC8vIGhhbmRsZVJlc3RvcmVNZXNzYWdlIGFsc28gcmVzdG9yZXMgcGFzdGVkIGltYWdlcy5cbiAgICAgICAgdm9pZCBoYW5kbGVSZXN0b3JlTWVzc2FnZShyYXcpXG4gICAgICB9IGVsc2Uge1xuICAgICAgICAvLyBEaWFsb2cgcGF0aDogb25QcmVSZXN0b3JlICg9IG9uQ2FuY2VsKSBmaXJlcyB3aGVuIHVzZXIgQ09ORklSTVMsIG5vdCBvbiBuZXZlcm1pbmQuXG4gICAgICAgIHNldE1lc3NhZ2VTZWxlY3RvclByZXNlbGVjdChyYXcpXG4gICAgICAgIHNldElzTWVzc2FnZVNlbGVjdG9yVmlzaWJsZSh0cnVlKVxuICAgICAgfVxuICAgIH0sXG4gIH1cbiAgY29uc3QgeyBlbnRlcjogZW50ZXJNZXNzYWdlQWN0aW9ucywgaGFuZGxlcnM6IG1lc3NhZ2VBY3Rpb25IYW5kbGVycyB9ID1cbiAgICB1c2VNZXNzYWdlQWN0aW9ucyhjdXJzb3IsIHNldEN1cnNvciwgY3Vyc29yTmF2UmVmLCBtZXNzYWdlQWN0aW9uQ2FwcylcblxuICBhc3luYyBmdW5jdGlvbiBvbkluaXQoKSB7XG4gICAgLy8gQWx3YXlzIHZlcmlmeSBBUEkga2V5IG9uIHN0YXJ0dXAsIHNvIHdlIGNhbiBzaG93IHRoZSB1c2VyIGFuIGVycm9yIGluIHRoZVxuICAgIC8vIGJvdHRvbSByaWdodCBjb3JuZXIgb2YgdGhlIHNjcmVlbiBpZiB0aGUgQVBJIGtleSBpcyBpbnZhbGlkLlxuICAgIHZvaWQgcmV2ZXJpZnkoKVxuXG4gICAgLy8gUG9wdWxhdGUgcmVhZEZpbGVTdGF0ZSB3aXRoIENMQVVERS5tZCBmaWxlcyBhdCBzdGFydHVwXG4gICAgY29uc3QgbWVtb3J5RmlsZXMgPSBhd2FpdCBnZXRNZW1vcnlGaWxlcygpXG4gICAgaWYgKG1lbW9yeUZpbGVzLmxlbmd0aCA+IDApIHtcbiAgICAgIGNvbnN0IGZpbGVMaXN0ID0gbWVtb3J5RmlsZXNcbiAgICAgICAgLm1hcChcbiAgICAgICAgICBmID0+XG4gICAgICAgICAgICBgICBbJHtmLnR5cGV9XSAke2YucGF0aH0gKCR7Zi5jb250ZW50Lmxlbmd0aH0gY2hhcnMpJHtmLnBhcmVudCA/IGAgKGluY2x1ZGVkIGJ5ICR7Zi5wYXJlbnR9KWAgOiAnJ31gLFxuICAgICAgICApXG4gICAgICAgIC5qb2luKCdcXG4nKVxuICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICBgTG9hZGVkICR7bWVtb3J5RmlsZXMubGVuZ3RofSBDTEFVREUubWQvcnVsZXMgZmlsZXM6XFxuJHtmaWxlTGlzdH1gLFxuICAgICAgKVxuICAgIH0gZWxzZSB7XG4gICAgICBsb2dGb3JEZWJ1Z2dpbmcoJ05vIENMQVVERS5tZC9ydWxlcyBmaWxlcyBmb3VuZCcpXG4gICAgfVxuICAgIGZvciAoY29uc3QgZmlsZSBvZiBtZW1vcnlGaWxlcykge1xuICAgICAgLy8gV2hlbiB0aGUgaW5qZWN0ZWQgY29udGVudCBkb2Vzbid0IG1hdGNoIGRpc2sgKHN0cmlwcGVkIEhUTUwgY29tbWVudHMsXG4gICAgICAvLyBzdHJpcHBlZCBmcm9udG1hdHRlciwgTUVNT1JZLm1kIHRydW5jYXRpb24pLCBjYWNoZSB0aGUgUkFXIGRpc2sgYnl0ZXNcbiAgICAgIC8vIHdpdGggaXNQYXJ0aWFsVmlldyBzbyBFZGl0L1dyaXRlIHJlcXVpcmUgYSByZWFsIFJlYWQgZmlyc3Qgd2hpbGVcbiAgICAgIC8vIGdldENoYW5nZWRGaWxlcyArIG5lc3RlZF9tZW1vcnkgZGVkdXAgc3RpbGwgd29yay5cbiAgICAgIHJlYWRGaWxlU3RhdGUuY3VycmVudC5zZXQoZmlsZS5wYXRoLCB7XG4gICAgICAgIGNvbnRlbnQ6IGZpbGUuY29udGVudERpZmZlcnNGcm9tRGlza1xuICAgICAgICAgID8gKGZpbGUucmF3Q29udGVudCA/PyBmaWxlLmNvbnRlbnQpXG4gICAgICAgICAgOiBmaWxlLmNvbnRlbnQsXG4gICAgICAgIHRpbWVzdGFtcDogRGF0ZS5ub3coKSxcbiAgICAgICAgb2Zmc2V0OiB1bmRlZmluZWQsXG4gICAgICAgIGxpbWl0OiB1bmRlZmluZWQsXG4gICAgICAgIGlzUGFydGlhbFZpZXc6IGZpbGUuY29udGVudERpZmZlcnNGcm9tRGlzayxcbiAgICAgIH0pXG4gICAgfVxuXG4gICAgLy8gSW5pdGlhbCBtZXNzYWdlIGhhbmRsaW5nIGlzIGRvbmUgdmlhIHRoZSBpbml0aWFsTWVzc2FnZSBlZmZlY3RcbiAgfVxuXG4gIC8vIFJlZ2lzdGVyIGNvc3Qgc3VtbWFyeSB0cmFja2VyXG4gIHVzZUNvc3RTdW1tYXJ5KHVzZUZwc01ldHJpY3MoKSlcblxuICAvLyBSZWNvcmQgdHJhbnNjcmlwdHMgbG9jYWxseSwgZm9yIGRlYnVnZ2luZyBhbmQgY29udmVyc2F0aW9uIHJlY292ZXJ5XG4gIC8vIERvbid0IHJlY29yZCBjb252ZXJzYXRpb24gaWYgd2Ugb25seSBoYXZlIGluaXRpYWwgbWVzc2FnZXM7IG9wdGltaXplc1xuICAvLyB0aGUgY2FzZSB3aGVyZSB1c2VyIHJlc3VtZXMgYSBjb252ZXJzYXRpb24gdGhlbiBxdWl0ZXMgYmVmb3JlIGRvaW5nXG4gIC8vIGFueXRoaW5nIGVsc2VcbiAgdXNlTG9nTWVzc2FnZXMobWVzc2FnZXMsIG1lc3NhZ2VzLmxlbmd0aCA9PT0gaW5pdGlhbE1lc3NhZ2VzPy5sZW5ndGgpXG5cbiAgLy8gUkVQTCBCcmlkZ2U6IHJlcGxpY2F0ZSB1c2VyL2Fzc2lzdGFudCBtZXNzYWdlcyB0byB0aGUgYnJpZGdlIHNlc3Npb25cbiAgLy8gZm9yIHJlbW90ZSBhY2Nlc3MgdmlhIGNsYXVkZS5haS4gTm8tb3AgaW4gZXh0ZXJuYWwgYnVpbGRzIG9yIHdoZW4gbm90IGVuYWJsZWQuXG4gIGNvbnN0IHsgc2VuZEJyaWRnZVJlc3VsdCB9ID0gdXNlUmVwbEJyaWRnZShcbiAgICBtZXNzYWdlcyxcbiAgICBzZXRNZXNzYWdlcyxcbiAgICBhYm9ydENvbnRyb2xsZXJSZWYsXG4gICAgY29tbWFuZHMsXG4gICAgbWFpbkxvb3BNb2RlbCxcbiAgKVxuICBzZW5kQnJpZGdlUmVzdWx0UmVmLmN1cnJlbnQgPSBzZW5kQnJpZGdlUmVzdWx0XG5cbiAgdXNlQWZ0ZXJGaXJzdFJlbmRlcigpXG5cbiAgLy8gVHJhY2sgcHJvbXB0IHF1ZXVlIHVzYWdlIGZvciBhbmFseXRpY3MuIEZpcmUgb25jZSBwZXIgdHJhbnNpdGlvbiBmcm9tXG4gIC8vIGVtcHR5IHRvIG5vbi1lbXB0eSwgbm90IG9uIGV2ZXJ5IGxlbmd0aCBjaGFuZ2UgLS0gb3RoZXJ3aXNlIGEgcmVuZGVyIGxvb3BcbiAgLy8gKGNvbmN1cnJlbnQgb25RdWVyeSB0aHJhc2hpbmcsIGV0Yy4pIHNwYW1zIHNhdmVHbG9iYWxDb25maWcsIHdoaWNoIGhpdHNcbiAgLy8gRUxPQ0tFRCB1bmRlciBjb25jdXJyZW50IHNlc3Npb25zIGFuZCBmYWxscyBiYWNrIHRvIHVubG9ja2VkIHdyaXRlcy5cbiAgLy8gVGhhdCB3cml0ZSBzdG9ybSBpcyB0aGUgcHJpbWFyeSB0cmlnZ2VyIGZvciB+Ly5jbGF1ZGUuanNvbiBjb3JydXB0aW9uXG4gIC8vIChHSCAjMzExNykuXG4gIGNvbnN0IGhhc0NvdW50ZWRRdWV1ZVVzZVJlZiA9IHVzZVJlZihmYWxzZSlcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAocXVldWVkQ29tbWFuZHMubGVuZ3RoIDwgMSkge1xuICAgICAgaGFzQ291bnRlZFF1ZXVlVXNlUmVmLmN1cnJlbnQgPSBmYWxzZVxuICAgICAgcmV0dXJuXG4gICAgfVxuICAgIGlmIChoYXNDb3VudGVkUXVldWVVc2VSZWYuY3VycmVudCkgcmV0dXJuXG4gICAgaGFzQ291bnRlZFF1ZXVlVXNlUmVmLmN1cnJlbnQgPSB0cnVlXG4gICAgc2F2ZUdsb2JhbENvbmZpZyhjdXJyZW50ID0+ICh7XG4gICAgICAuLi5jdXJyZW50LFxuICAgICAgcHJvbXB0UXVldWVVc2VDb3VudDogKGN1cnJlbnQucHJvbXB0UXVldWVVc2VDb3VudCA/PyAwKSArIDEsXG4gICAgfSkpXG4gIH0sIFtxdWV1ZWRDb21tYW5kcy5sZW5ndGhdKVxuXG4gIC8vIFByb2Nlc3MgcXVldWVkIGNvbW1hbmRzIHdoZW4gcXVlcnkgY29tcGxldGVzIGFuZCBxdWV1ZSBoYXMgaXRlbXNcblxuICBjb25zdCBleGVjdXRlUXVldWVkSW5wdXQgPSB1c2VDYWxsYmFjayhcbiAgICBhc3luYyAocXVldWVkQ29tbWFuZHM6IFF1ZXVlZENvbW1hbmRbXSkgPT4ge1xuICAgICAgYXdhaXQgaGFuZGxlUHJvbXB0U3VibWl0KHtcbiAgICAgICAgaGVscGVyczoge1xuICAgICAgICAgIHNldEN1cnNvck9mZnNldDogKCkgPT4ge30sXG4gICAgICAgICAgY2xlYXJCdWZmZXI6ICgpID0+IHt9LFxuICAgICAgICAgIHJlc2V0SGlzdG9yeTogKCkgPT4ge30sXG4gICAgICAgIH0sXG4gICAgICAgIHF1ZXJ5R3VhcmQsXG4gICAgICAgIGNvbW1hbmRzLFxuICAgICAgICBvbklucHV0Q2hhbmdlOiAoKSA9PiB7fSxcbiAgICAgICAgc2V0UGFzdGVkQ29udGVudHM6ICgpID0+IHt9LFxuICAgICAgICBzZXRUb29sSlNYLFxuICAgICAgICBnZXRUb29sVXNlQ29udGV4dCxcbiAgICAgICAgbWVzc2FnZXMsXG4gICAgICAgIG1haW5Mb29wTW9kZWwsXG4gICAgICAgIGlkZVNlbGVjdGlvbixcbiAgICAgICAgc2V0VXNlcklucHV0T25Qcm9jZXNzaW5nLFxuICAgICAgICBzZXRBYm9ydENvbnRyb2xsZXIsXG4gICAgICAgIG9uUXVlcnksXG4gICAgICAgIHNldEFwcFN0YXRlLFxuICAgICAgICBxdWVyeVNvdXJjZTogZ2V0UXVlcnlTb3VyY2VGb3JSRVBMKCksXG4gICAgICAgIG9uQmVmb3JlUXVlcnksXG4gICAgICAgIGNhblVzZVRvb2wsXG4gICAgICAgIGFkZE5vdGlmaWNhdGlvbixcbiAgICAgICAgc2V0TWVzc2FnZXMsXG4gICAgICAgIHF1ZXVlZENvbW1hbmRzLFxuICAgICAgfSlcbiAgICB9LFxuICAgIFtcbiAgICAgIHF1ZXJ5R3VhcmQsXG4gICAgICBjb21tYW5kcyxcbiAgICAgIHNldFRvb2xKU1gsXG4gICAgICBnZXRUb29sVXNlQ29udGV4dCxcbiAgICAgIG1lc3NhZ2VzLFxuICAgICAgbWFpbkxvb3BNb2RlbCxcbiAgICAgIGlkZVNlbGVjdGlvbixcbiAgICAgIHNldFVzZXJJbnB1dE9uUHJvY2Vzc2luZyxcbiAgICAgIGNhblVzZVRvb2wsXG4gICAgICBzZXRBYm9ydENvbnRyb2xsZXIsXG4gICAgICBvblF1ZXJ5LFxuICAgICAgYWRkTm90aWZpY2F0aW9uLFxuICAgICAgc2V0QXBwU3RhdGUsXG4gICAgICBvbkJlZm9yZVF1ZXJ5LFxuICAgIF0sXG4gIClcblxuICB1c2VRdWV1ZVByb2Nlc3Nvcih7XG4gICAgZXhlY3V0ZVF1ZXVlZElucHV0LFxuICAgIGhhc0FjdGl2ZUxvY2FsSnN4VUk6IGlzU2hvd2luZ0xvY2FsSlNYQ29tbWFuZCxcbiAgICBxdWVyeUd1YXJkLFxuICB9KVxuXG4gIC8vIFdlJ2xsIHVzZSB0aGUgZ2xvYmFsIGxhc3RJbnRlcmFjdGlvblRpbWUgZnJvbSBzdGF0ZS50c1xuXG4gIC8vIFVwZGF0ZSBsYXN0IGludGVyYWN0aW9uIHRpbWUgd2hlbiBpbnB1dCBjaGFuZ2VzLlxuICAvLyBNdXN0IGJlIGltbWVkaWF0ZSBiZWNhdXNlIHVzZUVmZmVjdCBydW5zIGFmdGVyIHRoZSBJbmsgcmVuZGVyIGN5Y2xlIGZsdXNoLlxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGFjdGl2aXR5TWFuYWdlci5yZWNvcmRVc2VyQWN0aXZpdHkoKVxuICAgIHVwZGF0ZUxhc3RJbnRlcmFjdGlvblRpbWUodHJ1ZSlcbiAgfSwgW2lucHV0VmFsdWUsIHN1Ym1pdENvdW50XSlcblxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIGlmIChzdWJtaXRDb3VudCA9PT0gMSkge1xuICAgICAgc3RhcnRCYWNrZ3JvdW5kSG91c2VrZWVwaW5nKClcbiAgICB9XG4gIH0sIFtzdWJtaXRDb3VudF0pXG5cbiAgLy8gU2hvdyBub3RpZmljYXRpb24gd2hlbiBDbGF1ZGUgaXMgZG9uZSByZXNwb25kaW5nIGFuZCB1c2VyIGlzIGlkbGVcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICAvLyBEb24ndCBzZXQgdXAgbm90aWZpY2F0aW9uIGlmIENsYXVkZSBpcyBidXN5XG4gICAgaWYgKGlzTG9hZGluZykgcmV0dXJuXG5cbiAgICAvLyBPbmx5IGVuYWJsZSBub3RpZmljYXRpb25zIGFmdGVyIHRoZSBmaXJzdCBuZXcgaW50ZXJhY3Rpb24gaW4gdGhpcyBzZXNzaW9uXG4gICAgaWYgKHN1Ym1pdENvdW50ID09PSAwKSByZXR1cm5cblxuICAgIC8vIE5vIHF1ZXJ5IGhhcyBjb21wbGV0ZWQgeWV0XG4gICAgaWYgKGxhc3RRdWVyeUNvbXBsZXRpb25UaW1lID09PSAwKSByZXR1cm5cblxuICAgIC8vIFNldCB0aW1lb3V0IHRvIGNoZWNrIGlkbGUgc3RhdGVcbiAgICBjb25zdCB0aW1lciA9IHNldFRpbWVvdXQoXG4gICAgICAoXG4gICAgICAgIGxhc3RRdWVyeUNvbXBsZXRpb25UaW1lLFxuICAgICAgICBpc0xvYWRpbmcsXG4gICAgICAgIHRvb2xKU1gsXG4gICAgICAgIGZvY3VzZWRJbnB1dERpYWxvZ1JlZixcbiAgICAgICAgdGVybWluYWwsXG4gICAgICApID0+IHtcbiAgICAgICAgLy8gQ2hlY2sgaWYgdXNlciBoYXMgaW50ZXJhY3RlZCBzaW5jZSB0aGUgcmVzcG9uc2UgZW5kZWRcbiAgICAgICAgY29uc3QgbGFzdFVzZXJJbnRlcmFjdGlvbiA9IGdldExhc3RJbnRlcmFjdGlvblRpbWUoKVxuXG4gICAgICAgIGlmIChsYXN0VXNlckludGVyYWN0aW9uID4gbGFzdFF1ZXJ5Q29tcGxldGlvblRpbWUpIHtcbiAgICAgICAgICAvLyBVc2VyIGhhcyBpbnRlcmFjdGVkIHNpbmNlIENsYXVkZSBmaW5pc2hlZCAtIHRoZXkncmUgbm90IGlkbGUsIGRvbid0IG5vdGlmeVxuICAgICAgICAgIHJldHVyblxuICAgICAgICB9XG5cbiAgICAgICAgLy8gVXNlciBoYXNuJ3QgaW50ZXJhY3RlZCBzaW5jZSByZXNwb25zZSBlbmRlZCwgY2hlY2sgb3RoZXIgY29uZGl0aW9uc1xuICAgICAgICBjb25zdCBpZGxlVGltZVNpbmNlUmVzcG9uc2UgPSBEYXRlLm5vdygpIC0gbGFzdFF1ZXJ5Q29tcGxldGlvblRpbWVcbiAgICAgICAgaWYgKFxuICAgICAgICAgICFpc0xvYWRpbmcgJiZcbiAgICAgICAgICAhdG9vbEpTWCAmJlxuICAgICAgICAgIC8vIFVzZSByZWYgdG8gZ2V0IGN1cnJlbnQgZGlhbG9nIHN0YXRlLCBhdm9pZGluZyBzdGFsZSBjbG9zdXJlXG4gICAgICAgICAgZm9jdXNlZElucHV0RGlhbG9nUmVmLmN1cnJlbnQgPT09IHVuZGVmaW5lZCAmJlxuICAgICAgICAgIGlkbGVUaW1lU2luY2VSZXNwb25zZSA+PSBnZXRHbG9iYWxDb25maWcoKS5tZXNzYWdlSWRsZU5vdGlmVGhyZXNob2xkTXNcbiAgICAgICAgKSB7XG4gICAgICAgICAgdm9pZCBzZW5kTm90aWZpY2F0aW9uKFxuICAgICAgICAgICAge1xuICAgICAgICAgICAgICBtZXNzYWdlOiAnQ2xhdWRlIGlzIHdhaXRpbmcgZm9yIHlvdXIgaW5wdXQnLFxuICAgICAgICAgICAgICBub3RpZmljYXRpb25UeXBlOiAnaWRsZV9wcm9tcHQnLFxuICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIHRlcm1pbmFsLFxuICAgICAgICAgIClcbiAgICAgICAgfVxuICAgICAgfSxcbiAgICAgIGdldEdsb2JhbENvbmZpZygpLm1lc3NhZ2VJZGxlTm90aWZUaHJlc2hvbGRNcyxcbiAgICAgIGxhc3RRdWVyeUNvbXBsZXRpb25UaW1lLFxuICAgICAgaXNMb2FkaW5nLFxuICAgICAgdG9vbEpTWCxcbiAgICAgIGZvY3VzZWRJbnB1dERpYWxvZ1JlZixcbiAgICAgIHRlcm1pbmFsLFxuICAgIClcblxuICAgIHJldHVybiAoKSA9PiBjbGVhclRpbWVvdXQodGltZXIpXG4gIH0sIFtpc0xvYWRpbmcsIHRvb2xKU1gsIHN1Ym1pdENvdW50LCBsYXN0UXVlcnlDb21wbGV0aW9uVGltZSwgdGVybWluYWxdKVxuXG4gIC8vIElkbGUtcmV0dXJuIGhpbnQ6IHNob3cgbm90aWZpY2F0aW9uIHdoZW4gaWRsZSB0aHJlc2hvbGQgaXMgZXhjZWVkZWQuXG4gIC8vIFRpbWVyIGZpcmVzIGFmdGVyIHRoZSBjb25maWd1cmVkIGlkbGUgcGVyaW9kOyBub3RpZmljYXRpb24gcGVyc2lzdHMgdW50aWxcbiAgLy8gZGlzbWlzc2VkIG9yIHRoZSB1c2VyIHN1Ym1pdHMuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgaWYgKGxhc3RRdWVyeUNvbXBsZXRpb25UaW1lID09PSAwKSByZXR1cm5cbiAgICBpZiAoaXNMb2FkaW5nKSByZXR1cm5cbiAgICBjb25zdCB3aWxsb3dNb2RlOiBzdHJpbmcgPSBnZXRGZWF0dXJlVmFsdWVfQ0FDSEVEX01BWV9CRV9TVEFMRShcbiAgICAgICd0ZW5ndV93aWxsb3dfbW9kZScsXG4gICAgICAnb2ZmJyxcbiAgICApXG4gICAgaWYgKHdpbGxvd01vZGUgIT09ICdoaW50JyAmJiB3aWxsb3dNb2RlICE9PSAnaGludF92MicpIHJldHVyblxuICAgIGlmIChnZXRHbG9iYWxDb25maWcoKS5pZGxlUmV0dXJuRGlzbWlzc2VkKSByZXR1cm5cblxuICAgIGNvbnN0IHRva2VuVGhyZXNob2xkID0gTnVtYmVyKFxuICAgICAgcHJvY2Vzcy5lbnYuQ0xBVURFX0NPREVfSURMRV9UT0tFTl9USFJFU0hPTEQgPz8gMTAwXzAwMCxcbiAgICApXG4gICAgaWYgKGdldFRvdGFsSW5wdXRUb2tlbnMoKSA8IHRva2VuVGhyZXNob2xkKSByZXR1cm5cblxuICAgIGNvbnN0IGlkbGVUaHJlc2hvbGRNcyA9XG4gICAgICBOdW1iZXIocHJvY2Vzcy5lbnYuQ0xBVURFX0NPREVfSURMRV9USFJFU0hPTERfTUlOVVRFUyA/PyA3NSkgKiA2MF8wMDBcbiAgICBjb25zdCBlbGFwc2VkID0gRGF0ZS5ub3coKSAtIGxhc3RRdWVyeUNvbXBsZXRpb25UaW1lXG4gICAgY29uc3QgcmVtYWluaW5nID0gaWRsZVRocmVzaG9sZE1zIC0gZWxhcHNlZFxuXG4gICAgY29uc3QgdGltZXIgPSBzZXRUaW1lb3V0KFxuICAgICAgKGxxY3QsIGFkZE5vdGlmLCBtc2dzUmVmLCBtb2RlLCBoaW50UmVmKSA9PiB7XG4gICAgICAgIGlmIChtc2dzUmVmLmN1cnJlbnQubGVuZ3RoID09PSAwKSByZXR1cm5cbiAgICAgICAgY29uc3QgdG90YWxUb2tlbnMgPSBnZXRUb3RhbElucHV0VG9rZW5zKClcbiAgICAgICAgY29uc3QgZm9ybWF0dGVkVG9rZW5zID0gZm9ybWF0VG9rZW5zKHRvdGFsVG9rZW5zKVxuICAgICAgICBjb25zdCBpZGxlTWludXRlcyA9IChEYXRlLm5vdygpIC0gbHFjdCkgLyA2MF8wMDBcbiAgICAgICAgYWRkTm90aWYoe1xuICAgICAgICAgIGtleTogJ2lkbGUtcmV0dXJuLWhpbnQnLFxuICAgICAgICAgIGpzeDpcbiAgICAgICAgICAgIG1vZGUgPT09ICdoaW50X3YyJyA/IChcbiAgICAgICAgICAgICAgPD5cbiAgICAgICAgICAgICAgICA8VGV4dCBkaW1Db2xvcj5uZXcgdGFzaz8gPC9UZXh0PlxuICAgICAgICAgICAgICAgIDxUZXh0IGNvbG9yPVwic3VnZ2VzdGlvblwiPi9jbGVhcjwvVGV4dD5cbiAgICAgICAgICAgICAgICA8VGV4dCBkaW1Db2xvcj4gdG8gc2F2ZSA8L1RleHQ+XG4gICAgICAgICAgICAgICAgPFRleHQgY29sb3I9XCJzdWdnZXN0aW9uXCI+e2Zvcm1hdHRlZFRva2Vuc30gdG9rZW5zPC9UZXh0PlxuICAgICAgICAgICAgICA8Lz5cbiAgICAgICAgICAgICkgOiAoXG4gICAgICAgICAgICAgIDxUZXh0IGNvbG9yPVwid2FybmluZ1wiPlxuICAgICAgICAgICAgICAgIG5ldyB0YXNrPyAvY2xlYXIgdG8gc2F2ZSB7Zm9ybWF0dGVkVG9rZW5zfSB0b2tlbnNcbiAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgKSxcbiAgICAgICAgICBwcmlvcml0eTogJ21lZGl1bScsXG4gICAgICAgICAgLy8gUGVyc2lzdCB1bnRpbCBzdWJtaXQg4oCUIHRoZSBoaW50IGZpcmVzIGF0IFQrNzVtaW4gaWRsZSwgdXNlciBtYXlcbiAgICAgICAgICAvLyBub3QgcmV0dXJuIGZvciBob3Vycy4gcmVtb3ZlTm90aWZpY2F0aW9uIGluIHVzZUVmZmVjdCBjbGVhbnVwXG4gICAgICAgICAgLy8gaGFuZGxlcyBkaXNtaXNzYWwuIDB4N0ZGRkZGRkYgPSBzZXRUaW1lb3V0IG1heCAofjI0LjggZGF5cykuXG4gICAgICAgICAgdGltZW91dE1zOiAweDdmZmZmZmZmLFxuICAgICAgICB9KVxuICAgICAgICBoaW50UmVmLmN1cnJlbnQgPSBtb2RlXG4gICAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9pZGxlX3JldHVybl9hY3Rpb24nLCB7XG4gICAgICAgICAgYWN0aW9uOlxuICAgICAgICAgICAgJ2hpbnRfc2hvd24nIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgICAgdmFyaWFudDpcbiAgICAgICAgICAgIG1vZGUgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgICAgICBpZGxlTWludXRlczogTWF0aC5yb3VuZChpZGxlTWludXRlcyksXG4gICAgICAgICAgbWVzc2FnZUNvdW50OiBtc2dzUmVmLmN1cnJlbnQubGVuZ3RoLFxuICAgICAgICAgIHRvdGFsSW5wdXRUb2tlbnM6IHRvdGFsVG9rZW5zLFxuICAgICAgICB9KVxuICAgICAgfSxcbiAgICAgIE1hdGgubWF4KDAsIHJlbWFpbmluZyksXG4gICAgICBsYXN0UXVlcnlDb21wbGV0aW9uVGltZSxcbiAgICAgIGFkZE5vdGlmaWNhdGlvbixcbiAgICAgIG1lc3NhZ2VzUmVmLFxuICAgICAgd2lsbG93TW9kZSxcbiAgICAgIGlkbGVIaW50U2hvd25SZWYsXG4gICAgKVxuXG4gICAgcmV0dXJuICgpID0+IHtcbiAgICAgIGNsZWFyVGltZW91dCh0aW1lcilcbiAgICAgIHJlbW92ZU5vdGlmaWNhdGlvbignaWRsZS1yZXR1cm4taGludCcpXG4gICAgICBpZGxlSGludFNob3duUmVmLmN1cnJlbnQgPSBmYWxzZVxuICAgIH1cbiAgfSwgW2xhc3RRdWVyeUNvbXBsZXRpb25UaW1lLCBpc0xvYWRpbmcsIGFkZE5vdGlmaWNhdGlvbiwgcmVtb3ZlTm90aWZpY2F0aW9uXSlcblxuICAvLyBTdWJtaXRzIGluY29taW5nIHByb21wdHMgZnJvbSB0ZWFtbWF0ZSBtZXNzYWdlcyBvciB0YXNrcyBtb2RlIGFzIG5ldyB0dXJuc1xuICAvLyBSZXR1cm5zIHRydWUgaWYgc3VibWlzc2lvbiBzdWNjZWVkZWQsIGZhbHNlIGlmIGEgcXVlcnkgaXMgYWxyZWFkeSBydW5uaW5nXG4gIGNvbnN0IGhhbmRsZUluY29taW5nUHJvbXB0ID0gdXNlQ2FsbGJhY2soXG4gICAgKGNvbnRlbnQ6IHN0cmluZywgb3B0aW9ucz86IHsgaXNNZXRhPzogYm9vbGVhbiB9KTogYm9vbGVhbiA9PiB7XG4gICAgICBpZiAocXVlcnlHdWFyZC5pc0FjdGl2ZSkgcmV0dXJuIGZhbHNlXG5cbiAgICAgIC8vIERlZmVyIHRvIHVzZXItcXVldWVkIGNvbW1hbmRzIOKAlCB1c2VyIGlucHV0IGFsd2F5cyB0YWtlcyBwcmlvcml0eVxuICAgICAgLy8gb3ZlciBzeXN0ZW0gbWVzc2FnZXMgKHRlYW1tYXRlIG1lc3NhZ2VzLCB0YXNrIGxpc3QgaXRlbXMsIGV0Yy4pXG4gICAgICAvLyBSZWFkIGZyb20gdGhlIG1vZHVsZS1sZXZlbCBzdG9yZSBhdCBjYWxsIHRpbWUgKG5vdCB0aGUgcmVuZGVyLXRpbWVcbiAgICAgIC8vIHNuYXBzaG90KSB0byBhdm9pZCBhIHN0YWxlIGNsb3N1cmUg4oCUIHRoaXMgY2FsbGJhY2sncyBkZXBzIGRvbid0XG4gICAgICAvLyBpbmNsdWRlIHRoZSBxdWV1ZS5cbiAgICAgIGlmIChcbiAgICAgICAgZ2V0Q29tbWFuZFF1ZXVlKCkuc29tZShcbiAgICAgICAgICBjbWQgPT4gY21kLm1vZGUgPT09ICdwcm9tcHQnIHx8IGNtZC5tb2RlID09PSAnYmFzaCcsXG4gICAgICAgIClcbiAgICAgICkge1xuICAgICAgICByZXR1cm4gZmFsc2VcbiAgICAgIH1cblxuICAgICAgY29uc3QgbmV3QWJvcnRDb250cm9sbGVyID0gY3JlYXRlQWJvcnRDb250cm9sbGVyKClcbiAgICAgIHNldEFib3J0Q29udHJvbGxlcihuZXdBYm9ydENvbnRyb2xsZXIpXG5cbiAgICAgIC8vIENyZWF0ZSBhIHVzZXIgbWVzc2FnZSB3aXRoIHRoZSBmb3JtYXR0ZWQgY29udGVudCAoaW5jbHVkZXMgWE1MIHdyYXBwZXIpXG4gICAgICBjb25zdCB1c2VyTWVzc2FnZSA9IGNyZWF0ZVVzZXJNZXNzYWdlKHtcbiAgICAgICAgY29udGVudCxcbiAgICAgICAgaXNNZXRhOiBvcHRpb25zPy5pc01ldGEgPyB0cnVlIDogdW5kZWZpbmVkLFxuICAgICAgfSlcblxuICAgICAgdm9pZCBvblF1ZXJ5KFt1c2VyTWVzc2FnZV0sIG5ld0Fib3J0Q29udHJvbGxlciwgdHJ1ZSwgW10sIG1haW5Mb29wTW9kZWwpXG4gICAgICByZXR1cm4gdHJ1ZVxuICAgIH0sXG4gICAgW29uUXVlcnksIG1haW5Mb29wTW9kZWwsIHN0b3JlXSxcbiAgKVxuXG4gIC8vIFZvaWNlIGlucHV0IGludGVncmF0aW9uIChWT0lDRV9NT0RFIGJ1aWxkcyBvbmx5KVxuICBjb25zdCB2b2ljZSA9IGZlYXR1cmUoJ1ZPSUNFX01PREUnKVxuICAgID8gLy8gYmlvbWUtaWdub3JlIGxpbnQvY29ycmVjdG5lc3MvdXNlSG9va0F0VG9wTGV2ZWw6IGZlYXR1cmUoKSBpcyBhIGNvbXBpbGUtdGltZSBjb25zdGFudFxuICAgICAgdXNlVm9pY2VJbnRlZ3JhdGlvbih7IHNldElucHV0VmFsdWVSYXcsIGlucHV0VmFsdWVSZWYsIGluc2VydFRleHRSZWYgfSlcbiAgICA6IHtcbiAgICAgICAgc3RyaXBUcmFpbGluZzogKCkgPT4gMCxcbiAgICAgICAgaGFuZGxlS2V5RXZlbnQ6ICgpID0+IHt9LFxuICAgICAgICByZXNldEFuY2hvcjogKCkgPT4ge30sXG4gICAgICAgIGludGVyaW1SYW5nZTogbnVsbCxcbiAgICAgIH1cblxuICB1c2VJbmJveFBvbGxlcih7XG4gICAgZW5hYmxlZDogaXNBZ2VudFN3YXJtc0VuYWJsZWQoKSxcbiAgICBpc0xvYWRpbmcsXG4gICAgZm9jdXNlZElucHV0RGlhbG9nLFxuICAgIG9uU3VibWl0TWVzc2FnZTogaGFuZGxlSW5jb21pbmdQcm9tcHQsXG4gIH0pXG5cbiAgdXNlTWFpbGJveEJyaWRnZSh7IGlzTG9hZGluZywgb25TdWJtaXRNZXNzYWdlOiBoYW5kbGVJbmNvbWluZ1Byb21wdCB9KVxuXG4gIC8vIFNjaGVkdWxlZCB0YXNrcyBmcm9tIC5jbGF1ZGUvc2NoZWR1bGVkX3Rhc2tzLmpzb24gKENyb25DcmVhdGUvRGVsZXRlL0xpc3QpXG4gIGlmIChmZWF0dXJlKCdBR0VOVF9UUklHR0VSUycpKSB7XG4gICAgLy8gQXNzaXN0YW50IG1vZGUgYnlwYXNzZXMgdGhlIGlzTG9hZGluZyBnYXRlICh0aGUgcHJvYWN0aXZlIHRpY2sg4oaSXG4gICAgLy8gU2xlZXAg4oaSIHRpY2sgbG9vcCB3b3VsZCBvdGhlcndpc2Ugc3RhcnZlIHRoZSBzY2hlZHVsZXIpLlxuICAgIC8vIGthaXJvc0VuYWJsZWQgaXMgc2V0IG9uY2UgaW4gaW5pdGlhbFN0YXRlIChtYWluLnRzeCkgYW5kIG5ldmVyIG11dGF0ZWQg4oCUIG5vXG4gICAgLy8gc3Vic2NyaXB0aW9uIG5lZWRlZC4gVGhlIHRlbmd1X2thaXJvc19jcm9uIHJ1bnRpbWUgZ2F0ZSBpcyBjaGVja2VkIGluc2lkZVxuICAgIC8vIHVzZVNjaGVkdWxlZFRhc2tzJ3MgZWZmZWN0IChub3QgaGVyZSkgc2luY2Ugd3JhcHBpbmcgYSBob29rIGNhbGwgaW4gYSBkeW5hbWljXG4gICAgLy8gY29uZGl0aW9uIHdvdWxkIGJyZWFrIHJ1bGVzLW9mLWhvb2tzLlxuICAgIGNvbnN0IGFzc2lzdGFudE1vZGUgPSBzdG9yZS5nZXRTdGF0ZSgpLmthaXJvc0VuYWJsZWRcbiAgICAvLyBiaW9tZS1pZ25vcmUgbGludC9jb3JyZWN0bmVzcy91c2VIb29rQXRUb3BMZXZlbDogZmVhdHVyZSgpIGlzIGEgY29tcGlsZS10aW1lIGNvbnN0YW50XG4gICAgdXNlU2NoZWR1bGVkVGFza3MhKHsgaXNMb2FkaW5nLCBhc3Npc3RhbnRNb2RlLCBzZXRNZXNzYWdlcyB9KVxuICB9XG5cbiAgLy8gTm90ZTogUGVybWlzc2lvbiBwb2xsaW5nIGlzIG5vdyBoYW5kbGVkIGJ5IHVzZUluYm94UG9sbGVyXG4gIC8vIC0gV29ya2VycyByZWNlaXZlIHBlcm1pc3Npb24gcmVzcG9uc2VzIHZpYSBtYWlsYm94IG1lc3NhZ2VzXG4gIC8vIC0gTGVhZGVycyByZWNlaXZlIHBlcm1pc3Npb24gcmVxdWVzdHMgdmlhIG1haWxib3ggbWVzc2FnZXNcblxuICBpZiAoXCJleHRlcm5hbFwiID09PSAnYW50Jykge1xuICAgIC8vIFRhc2tzIG1vZGU6IHdhdGNoIGZvciB0YXNrcyBhbmQgYXV0by1wcm9jZXNzIHRoZW1cbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgcmVhY3QtaG9va3MvcnVsZXMtb2YtaG9va3NcbiAgICAvLyBiaW9tZS1pZ25vcmUgbGludC9jb3JyZWN0bmVzcy91c2VIb29rQXRUb3BMZXZlbDogY29uZGl0aW9uYWwgZm9yIGRlYWQgY29kZSBlbGltaW5hdGlvbiBpbiBleHRlcm5hbCBidWlsZHNcbiAgICB1c2VUYXNrTGlzdFdhdGNoZXIoe1xuICAgICAgdGFza0xpc3RJZCxcbiAgICAgIGlzTG9hZGluZyxcbiAgICAgIG9uU3VibWl0VGFzazogaGFuZGxlSW5jb21pbmdQcm9tcHQsXG4gICAgfSlcblxuICAgIC8vIExvb3AgbW9kZTogYXV0by10aWNrIHdoZW4gZW5hYmxlZCAodmlhIC9qb2IgY29tbWFuZClcbiAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgcmVhY3QtaG9va3MvcnVsZXMtb2YtaG9va3NcbiAgICAvLyBiaW9tZS1pZ25vcmUgbGludC9jb3JyZWN0bmVzcy91c2VIb29rQXRUb3BMZXZlbDogY29uZGl0aW9uYWwgZm9yIGRlYWQgY29kZSBlbGltaW5hdGlvbiBpbiBleHRlcm5hbCBidWlsZHNcbiAgICB1c2VQcm9hY3RpdmU/Lih7XG4gICAgICAvLyBTdXBwcmVzcyB0aWNrcyB3aGlsZSBhbiBpbml0aWFsIG1lc3NhZ2UgaXMgcGVuZGluZyDigJQgdGhlIGluaXRpYWxcbiAgICAgIC8vIG1lc3NhZ2Ugd2lsbCBiZSBwcm9jZXNzZWQgYXN5bmNocm9ub3VzbHkgYW5kIGEgcHJlbWF0dXJlIHRpY2sgd291bGRcbiAgICAgIC8vIHJhY2Ugd2l0aCBpdCwgY2F1c2luZyBjb25jdXJyZW50LXF1ZXJ5IGVucXVldWUgb2YgZXhwYW5kZWQgc2tpbGwgdGV4dC5cbiAgICAgIGlzTG9hZGluZzogaXNMb2FkaW5nIHx8IGluaXRpYWxNZXNzYWdlICE9PSBudWxsLFxuICAgICAgcXVldWVkQ29tbWFuZHNMZW5ndGg6IHF1ZXVlZENvbW1hbmRzLmxlbmd0aCxcbiAgICAgIGhhc0FjdGl2ZUxvY2FsSnN4VUk6IGlzU2hvd2luZ0xvY2FsSlNYQ29tbWFuZCxcbiAgICAgIGlzSW5QbGFuTW9kZTogdG9vbFBlcm1pc3Npb25Db250ZXh0Lm1vZGUgPT09ICdwbGFuJyxcbiAgICAgIG9uU3VibWl0VGljazogKHByb21wdDogc3RyaW5nKSA9PlxuICAgICAgICBoYW5kbGVJbmNvbWluZ1Byb21wdChwcm9tcHQsIHsgaXNNZXRhOiB0cnVlIH0pLFxuICAgICAgb25RdWV1ZVRpY2s6IChwcm9tcHQ6IHN0cmluZykgPT5cbiAgICAgICAgZW5xdWV1ZSh7IG1vZGU6ICdwcm9tcHQnLCB2YWx1ZTogcHJvbXB0LCBpc01ldGE6IHRydWUgfSksXG4gICAgfSlcbiAgfVxuXG4gIC8vIEFib3J0IHRoZSBjdXJyZW50IG9wZXJhdGlvbiB3aGVuIGEgJ25vdycgcHJpb3JpdHkgbWVzc2FnZSBhcnJpdmVzXG4gIC8vIChlLmcuIGZyb20gYSBjaGF0IFVJIGNsaWVudCB2aWEgVURTKS5cbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAocXVldWVkQ29tbWFuZHMuc29tZShjbWQgPT4gY21kLnByaW9yaXR5ID09PSAnbm93JykpIHtcbiAgICAgIGFib3J0Q29udHJvbGxlclJlZi5jdXJyZW50Py5hYm9ydCgnaW50ZXJydXB0JylcbiAgICB9XG4gIH0sIFtxdWV1ZWRDb21tYW5kc10pXG5cbiAgLy8gSW5pdGlhbCBsb2FkXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgdm9pZCBvbkluaXQoKVxuXG4gICAgLy8gQ2xlYW51cCBvbiB1bm1vdW50XG4gICAgcmV0dXJuICgpID0+IHtcbiAgICAgIHZvaWQgZGlhZ25vc3RpY1RyYWNrZXIuc2h1dGRvd24oKVxuICAgIH1cbiAgICAvLyBUT0RPOiBmaXggdGhpc1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSByZWFjdC1ob29rcy9leGhhdXN0aXZlLWRlcHNcbiAgfSwgW10pXG5cbiAgLy8gTGlzdGVuIGZvciBzdXNwZW5kL3Jlc3VtZSBldmVudHNcbiAgY29uc3QgeyBpbnRlcm5hbF9ldmVudEVtaXR0ZXIgfSA9IHVzZVN0ZGluKClcbiAgY29uc3QgW3JlbW91bnRLZXksIHNldFJlbW91bnRLZXldID0gdXNlU3RhdGUoMClcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBjb25zdCBoYW5kbGVTdXNwZW5kID0gKCkgPT4ge1xuICAgICAgLy8gUHJpbnQgc3VzcGVuc2lvbiBpbnN0cnVjdGlvbnNcbiAgICAgIHByb2Nlc3Muc3Rkb3V0LndyaXRlKFxuICAgICAgICBgXFxuQ2xhdWRlIENvZGUgaGFzIGJlZW4gc3VzcGVuZGVkLiBSdW4gXFxgZmdcXGAgdG8gYnJpbmcgQ2xhdWRlIENvZGUgYmFjay5cXG5Ob3RlOiBjdHJsICsgeiBub3cgc3VzcGVuZHMgQ2xhdWRlIENvZGUsIGN0cmwgKyBfIHVuZG9lcyBpbnB1dC5cXG5gLFxuICAgICAgKVxuICAgIH1cblxuICAgIGNvbnN0IGhhbmRsZVJlc3VtZSA9ICgpID0+IHtcbiAgICAgIC8vIEZvcmNlIGNvbXBsZXRlIGNvbXBvbmVudCB0cmVlIHJlcGxhY2VtZW50IGluc3RlYWQgb2YgdGVybWluYWwgY2xlYXJcbiAgICAgIC8vIEluayBub3cgaGFuZGxlcyBsaW5lIGNvdW50IHJlc2V0IGludGVybmFsbHkgb24gU0lHQ09OVFxuICAgICAgc2V0UmVtb3VudEtleShwcmV2ID0+IHByZXYgKyAxKVxuICAgIH1cblxuICAgIGludGVybmFsX2V2ZW50RW1pdHRlcj8ub24oJ3N1c3BlbmQnLCBoYW5kbGVTdXNwZW5kKVxuICAgIGludGVybmFsX2V2ZW50RW1pdHRlcj8ub24oJ3Jlc3VtZScsIGhhbmRsZVJlc3VtZSlcbiAgICByZXR1cm4gKCkgPT4ge1xuICAgICAgaW50ZXJuYWxfZXZlbnRFbWl0dGVyPy5vZmYoJ3N1c3BlbmQnLCBoYW5kbGVTdXNwZW5kKVxuICAgICAgaW50ZXJuYWxfZXZlbnRFbWl0dGVyPy5vZmYoJ3Jlc3VtZScsIGhhbmRsZVJlc3VtZSlcbiAgICB9XG4gIH0sIFtpbnRlcm5hbF9ldmVudEVtaXR0ZXJdKVxuXG4gIC8vIERlcml2ZSBzdG9wIGhvb2sgc3Bpbm5lciBzdWZmaXggZnJvbSBtZXNzYWdlcyBzdGF0ZVxuICBjb25zdCBzdG9wSG9va1NwaW5uZXJTdWZmaXggPSB1c2VNZW1vKCgpID0+IHtcbiAgICBpZiAoIWlzTG9hZGluZykgcmV0dXJuIG51bGxcblxuICAgIC8vIEZpbmQgc3RvcCBob29rIHByb2dyZXNzIG1lc3NhZ2VzXG4gICAgY29uc3QgcHJvZ3Jlc3NNc2dzID0gbWVzc2FnZXMuZmlsdGVyKFxuICAgICAgKG0pOiBtIGlzIFByb2dyZXNzTWVzc2FnZTxIb29rUHJvZ3Jlc3M+ID0+XG4gICAgICAgIG0udHlwZSA9PT0gJ3Byb2dyZXNzJyAmJlxuICAgICAgICBtLmRhdGEudHlwZSA9PT0gJ2hvb2tfcHJvZ3Jlc3MnICYmXG4gICAgICAgIChtLmRhdGEuaG9va0V2ZW50ID09PSAnU3RvcCcgfHwgbS5kYXRhLmhvb2tFdmVudCA9PT0gJ1N1YmFnZW50U3RvcCcpLFxuICAgIClcbiAgICBpZiAocHJvZ3Jlc3NNc2dzLmxlbmd0aCA9PT0gMCkgcmV0dXJuIG51bGxcblxuICAgIC8vIEdldCB0aGUgbW9zdCByZWNlbnQgc3RvcCBob29rIGV4ZWN1dGlvblxuICAgIGNvbnN0IGN1cnJlbnRUb29sVXNlSUQgPSBwcm9ncmVzc01zZ3MuYXQoLTEpPy50b29sVXNlSURcbiAgICBpZiAoIWN1cnJlbnRUb29sVXNlSUQpIHJldHVybiBudWxsXG5cbiAgICAvLyBDaGVjayBpZiB0aGVyZSdzIGFscmVhZHkgYSBzdW1tYXJ5IG1lc3NhZ2UgZm9yIHRoaXMgZXhlY3V0aW9uIChob29rcyBjb21wbGV0ZWQpXG4gICAgY29uc3QgaGFzU3VtbWFyeUZvckN1cnJlbnRFeGVjdXRpb24gPSBtZXNzYWdlcy5zb21lKFxuICAgICAgbSA9PlxuICAgICAgICBtLnR5cGUgPT09ICdzeXN0ZW0nICYmXG4gICAgICAgIG0uc3VidHlwZSA9PT0gJ3N0b3BfaG9va19zdW1tYXJ5JyAmJlxuICAgICAgICBtLnRvb2xVc2VJRCA9PT0gY3VycmVudFRvb2xVc2VJRCxcbiAgICApXG4gICAgaWYgKGhhc1N1bW1hcnlGb3JDdXJyZW50RXhlY3V0aW9uKSByZXR1cm4gbnVsbFxuXG4gICAgY29uc3QgY3VycmVudEhvb2tzID0gcHJvZ3Jlc3NNc2dzLmZpbHRlcihcbiAgICAgIHAgPT4gcC50b29sVXNlSUQgPT09IGN1cnJlbnRUb29sVXNlSUQsXG4gICAgKVxuICAgIGNvbnN0IHRvdGFsID0gY3VycmVudEhvb2tzLmxlbmd0aFxuXG4gICAgLy8gQ291bnQgY29tcGxldGVkIGhvb2tzXG4gICAgY29uc3QgY29tcGxldGVkQ291bnQgPSBjb3VudChtZXNzYWdlcywgbSA9PiB7XG4gICAgICBpZiAobS50eXBlICE9PSAnYXR0YWNobWVudCcpIHJldHVybiBmYWxzZVxuICAgICAgY29uc3QgYXR0YWNobWVudCA9IG0uYXR0YWNobWVudFxuICAgICAgcmV0dXJuIChcbiAgICAgICAgJ2hvb2tFdmVudCcgaW4gYXR0YWNobWVudCAmJlxuICAgICAgICAoYXR0YWNobWVudC5ob29rRXZlbnQgPT09ICdTdG9wJyB8fFxuICAgICAgICAgIGF0dGFjaG1lbnQuaG9va0V2ZW50ID09PSAnU3ViYWdlbnRTdG9wJykgJiZcbiAgICAgICAgJ3Rvb2xVc2VJRCcgaW4gYXR0YWNobWVudCAmJlxuICAgICAgICBhdHRhY2htZW50LnRvb2xVc2VJRCA9PT0gY3VycmVudFRvb2xVc2VJRFxuICAgICAgKVxuICAgIH0pXG5cbiAgICAvLyBDaGVjayBpZiBhbnkgaG9vayBoYXMgYSBjdXN0b20gc3RhdHVzIG1lc3NhZ2VcbiAgICBjb25zdCBjdXN0b21NZXNzYWdlID0gY3VycmVudEhvb2tzLmZpbmQocCA9PiBwLmRhdGEuc3RhdHVzTWVzc2FnZSk/LmRhdGFcbiAgICAgIC5zdGF0dXNNZXNzYWdlXG5cbiAgICBpZiAoY3VzdG9tTWVzc2FnZSkge1xuICAgICAgLy8gVXNlIGN1c3RvbSBtZXNzYWdlIHdpdGggcHJvZ3Jlc3MgY291bnRlciBpZiBtdWx0aXBsZSBob29rc1xuICAgICAgcmV0dXJuIHRvdGFsID09PSAxXG4gICAgICAgID8gYCR7Y3VzdG9tTWVzc2FnZX3igKZgXG4gICAgICAgIDogYCR7Y3VzdG9tTWVzc2FnZX3igKYgJHtjb21wbGV0ZWRDb3VudH0vJHt0b3RhbH1gXG4gICAgfVxuXG4gICAgLy8gRmFsbCBiYWNrIHRvIGRlZmF1bHQgYmVoYXZpb3JcbiAgICBjb25zdCBob29rVHlwZSA9XG4gICAgICBjdXJyZW50SG9va3NbMF0/LmRhdGEuaG9va0V2ZW50ID09PSAnU3ViYWdlbnRTdG9wJ1xuICAgICAgICA/ICdzdWJhZ2VudCBzdG9wJ1xuICAgICAgICA6ICdzdG9wJ1xuXG4gICAgaWYgKFwiZXh0ZXJuYWxcIiA9PT0gJ2FudCcpIHtcbiAgICAgIGNvbnN0IGNtZCA9IGN1cnJlbnRIb29rc1tjb21wbGV0ZWRDb3VudF0/LmRhdGEuY29tbWFuZFxuICAgICAgY29uc3QgbGFiZWwgPSBjbWQgPyBgICcke3RydW5jYXRlVG9XaWR0aChjbWQsIDQwKX0nYCA6ICcnXG4gICAgICByZXR1cm4gdG90YWwgPT09IDFcbiAgICAgICAgPyBgcnVubmluZyAke2hvb2tUeXBlfSBob29rJHtsYWJlbH1gXG4gICAgICAgIDogYHJ1bm5pbmcgJHtob29rVHlwZX0gaG9vayR7bGFiZWx9XFx1MjAyNiAke2NvbXBsZXRlZENvdW50fS8ke3RvdGFsfWBcbiAgICB9XG5cbiAgICByZXR1cm4gdG90YWwgPT09IDFcbiAgICAgID8gYHJ1bm5pbmcgJHtob29rVHlwZX0gaG9va2BcbiAgICAgIDogYHJ1bm5pbmcgc3RvcCBob29rc+KApiAke2NvbXBsZXRlZENvdW50fS8ke3RvdGFsfWBcbiAgfSwgW21lc3NhZ2VzLCBpc0xvYWRpbmddKVxuXG4gIC8vIENhbGxiYWNrIHRvIGNhcHR1cmUgZnJvemVuIHN0YXRlIHdoZW4gZW50ZXJpbmcgdHJhbnNjcmlwdCBtb2RlXG4gIGNvbnN0IGhhbmRsZUVudGVyVHJhbnNjcmlwdCA9IHVzZUNhbGxiYWNrKCgpID0+IHtcbiAgICBzZXRGcm96ZW5UcmFuc2NyaXB0U3RhdGUoe1xuICAgICAgbWVzc2FnZXNMZW5ndGg6IG1lc3NhZ2VzLmxlbmd0aCxcbiAgICAgIHN0cmVhbWluZ1Rvb2xVc2VzTGVuZ3RoOiBzdHJlYW1pbmdUb29sVXNlcy5sZW5ndGgsXG4gICAgfSlcbiAgfSwgW21lc3NhZ2VzLmxlbmd0aCwgc3RyZWFtaW5nVG9vbFVzZXMubGVuZ3RoXSlcblxuICAvLyBDYWxsYmFjayB0byBjbGVhciBmcm96ZW4gc3RhdGUgd2hlbiBleGl0aW5nIHRyYW5zY3JpcHQgbW9kZVxuICBjb25zdCBoYW5kbGVFeGl0VHJhbnNjcmlwdCA9IHVzZUNhbGxiYWNrKCgpID0+IHtcbiAgICBzZXRGcm96ZW5UcmFuc2NyaXB0U3RhdGUobnVsbClcbiAgfSwgW10pXG5cbiAgLy8gUHJvcHMgZm9yIEdsb2JhbEtleWJpbmRpbmdIYW5kbGVycyBjb21wb25lbnQgKHJlbmRlcmVkIGluc2lkZSBLZXliaW5kaW5nU2V0dXApXG4gIGNvbnN0IHZpcnR1YWxTY3JvbGxBY3RpdmUgPSBpc0Z1bGxzY3JlZW5FbnZFbmFibGVkKCkgJiYgIWRpc2FibGVWaXJ0dWFsU2Nyb2xsXG5cbiAgLy8gVHJhbnNjcmlwdCBzZWFyY2ggc3RhdGUuIEhvb2tzIG11c3QgYmUgdW5jb25kaXRpb25hbCBzbyB0aGV5IGxpdmUgaGVyZVxuICAvLyAobm90IGluc2lkZSB0aGUgYGlmIChzY3JlZW4gPT09ICd0cmFuc2NyaXB0JylgIGJyYW5jaCBiZWxvdyk7IGlzQWN0aXZlXG4gIC8vIGdhdGVzIHRoZSB1c2VJbnB1dC4gUXVlcnkgcGVyc2lzdHMgYWNyb3NzIGJhciBvcGVuL2Nsb3NlIHNvIG4vTiBrZWVwXG4gIC8vIHdvcmtpbmcgYWZ0ZXIgRW50ZXIgZGlzbWlzc2VzIHRoZSBiYXIgKGxlc3Mgc2VtYW50aWNzKS5cbiAgY29uc3QganVtcFJlZiA9IHVzZVJlZjxKdW1wSGFuZGxlIHwgbnVsbD4obnVsbClcbiAgY29uc3QgW3NlYXJjaE9wZW4sIHNldFNlYXJjaE9wZW5dID0gdXNlU3RhdGUoZmFsc2UpXG4gIGNvbnN0IFtzZWFyY2hRdWVyeSwgc2V0U2VhcmNoUXVlcnldID0gdXNlU3RhdGUoJycpXG4gIGNvbnN0IFtzZWFyY2hDb3VudCwgc2V0U2VhcmNoQ291bnRdID0gdXNlU3RhdGUoMClcbiAgY29uc3QgW3NlYXJjaEN1cnJlbnQsIHNldFNlYXJjaEN1cnJlbnRdID0gdXNlU3RhdGUoMClcbiAgY29uc3Qgb25TZWFyY2hNYXRjaGVzQ2hhbmdlID0gdXNlQ2FsbGJhY2soXG4gICAgKGNvdW50OiBudW1iZXIsIGN1cnJlbnQ6IG51bWJlcikgPT4ge1xuICAgICAgc2V0U2VhcmNoQ291bnQoY291bnQpXG4gICAgICBzZXRTZWFyY2hDdXJyZW50KGN1cnJlbnQpXG4gICAgfSxcbiAgICBbXSxcbiAgKVxuXG4gIHVzZUlucHV0KFxuICAgIChpbnB1dCwga2V5LCBldmVudCkgPT4ge1xuICAgICAgaWYgKGtleS5jdHJsIHx8IGtleS5tZXRhKSByZXR1cm5cbiAgICAgIC8vIE5vIEVzYyBoYW5kbGluZyBoZXJlIOKAlCBsZXNzIGhhcyBubyBuYXZpZ2F0aW5nIG1vZGUuIFNlYXJjaCBzdGF0ZVxuICAgICAgLy8gKGhpZ2hsaWdodHMsIG4vTikgaXMganVzdCBzdGF0ZS4gRXNjL3EvY3RybCtjIOKGkiB0cmFuc2NyaXB0OmV4aXRcbiAgICAgIC8vICh1bmdhdGVkKS4gSGlnaGxpZ2h0cyBjbGVhciBvbiBleGl0IHZpYSB0aGUgc2NyZWVuLWNoYW5nZSBlZmZlY3QuXG4gICAgICBpZiAoaW5wdXQgPT09ICcvJykge1xuICAgICAgICAvLyBDYXB0dXJlIHNjcm9sbFRvcCBOT1cg4oCUIHR5cGluZyBpcyBhIHByZXZpZXcsIDAtbWF0Y2hlcyBzbmFwc1xuICAgICAgICAvLyBiYWNrIGhlcmUuIFN5bmNocm9ub3VzIHJlZiB3cml0ZSwgZmlyZXMgYmVmb3JlIHRoZSBiYXInc1xuICAgICAgICAvLyBtb3VudC1lZmZlY3QgY2FsbHMgc2V0U2VhcmNoUXVlcnkuXG4gICAgICAgIGp1bXBSZWYuY3VycmVudD8uc2V0QW5jaG9yKClcbiAgICAgICAgc2V0U2VhcmNoT3Blbih0cnVlKVxuICAgICAgICBldmVudC5zdG9wSW1tZWRpYXRlUHJvcGFnYXRpb24oKVxuICAgICAgICByZXR1cm5cbiAgICAgIH1cbiAgICAgIC8vIEhlbGQta2V5IGJhdGNoaW5nOiB0b2tlbml6ZXIgY29hbGVzY2VzIHRvICdubm4nLiBTYW1lIHVuaWZvcm0tYmF0Y2hcbiAgICAgIC8vIHBhdHRlcm4gYXMgbW9kYWxQYWdlckFjdGlvbiBpbiBTY3JvbGxLZXliaW5kaW5nSGFuZGxlci50c3guIEVhY2hcbiAgICAgIC8vIHJlcGVhdCBpcyBhIHN0ZXAgKG4gaXNuJ3QgaWRlbXBvdGVudCBsaWtlIGcpLlxuICAgICAgY29uc3QgYyA9IGlucHV0WzBdXG4gICAgICBpZiAoXG4gICAgICAgIChjID09PSAnbicgfHwgYyA9PT0gJ04nKSAmJlxuICAgICAgICBpbnB1dCA9PT0gYy5yZXBlYXQoaW5wdXQubGVuZ3RoKSAmJlxuICAgICAgICBzZWFyY2hDb3VudCA+IDBcbiAgICAgICkge1xuICAgICAgICBjb25zdCBmbiA9XG4gICAgICAgICAgYyA9PT0gJ24nID8ganVtcFJlZi5jdXJyZW50Py5uZXh0TWF0Y2ggOiBqdW1wUmVmLmN1cnJlbnQ/LnByZXZNYXRjaFxuICAgICAgICBpZiAoZm4pIGZvciAobGV0IGkgPSAwOyBpIDwgaW5wdXQubGVuZ3RoOyBpKyspIGZuKClcbiAgICAgICAgZXZlbnQuc3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uKClcbiAgICAgIH1cbiAgICB9LFxuICAgIC8vIFNlYXJjaCBuZWVkcyB2aXJ0dWFsIHNjcm9sbCAoanVtcFJlZiBkcml2ZXMgVmlydHVhbE1lc3NhZ2VMaXN0KS4gW1xuICAgIC8vIGtpbGxzIGl0LCBzbyAhZHVtcE1vZGUg4oCUIGFmdGVyIFsgdGhlcmUncyBub3RoaW5nIHRvIGp1bXAgaW4uXG4gICAge1xuICAgICAgaXNBY3RpdmU6XG4gICAgICAgIHNjcmVlbiA9PT0gJ3RyYW5zY3JpcHQnICYmXG4gICAgICAgIHZpcnR1YWxTY3JvbGxBY3RpdmUgJiZcbiAgICAgICAgIXNlYXJjaE9wZW4gJiZcbiAgICAgICAgIWR1bXBNb2RlLFxuICAgIH0sXG4gIClcbiAgY29uc3Qge1xuICAgIHNldFF1ZXJ5OiBzZXRIaWdobGlnaHQsXG4gICAgc2NhbkVsZW1lbnQsXG4gICAgc2V0UG9zaXRpb25zLFxuICB9ID0gdXNlU2VhcmNoSGlnaGxpZ2h0KClcblxuICAvLyBSZXNpemUg4oaSIGFib3J0IHNlYXJjaC4gUG9zaXRpb25zIGFyZSAobXNnLCBxdWVyeSwgV0lEVEgpLWtleWVkIOKAlFxuICAvLyBjYWNoZWQgcG9zaXRpb25zIGFyZSBzdGFsZSBhZnRlciBhIHdpZHRoIGNoYW5nZSAobmV3IGxheW91dCwgbmV3XG4gIC8vIHdyYXBwaW5nKS4gQ2xlYXJpbmcgc2VhcmNoUXVlcnkgdHJpZ2dlcnMgVk1MJ3Mgc2V0U2VhcmNoUXVlcnkoJycpXG4gIC8vIHdoaWNoIGNsZWFycyBwb3NpdGlvbnNDYWNoZSArIHNldFBvc2l0aW9ucyhudWxsKS4gQmFyIGNsb3Nlcy5cbiAgLy8gVXNlciBoaXRzIC8gYWdhaW4g4oaSIGZyZXNoIGV2ZXJ5dGhpbmcuXG4gIGNvbnN0IHRyYW5zY3JpcHRDb2xzID0gdXNlVGVybWluYWxTaXplKCkuY29sdW1uc1xuICBjb25zdCBwcmV2Q29sc1JlZiA9IFJlYWN0LnVzZVJlZih0cmFuc2NyaXB0Q29scylcbiAgUmVhY3QudXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAocHJldkNvbHNSZWYuY3VycmVudCAhPT0gdHJhbnNjcmlwdENvbHMpIHtcbiAgICAgIHByZXZDb2xzUmVmLmN1cnJlbnQgPSB0cmFuc2NyaXB0Q29sc1xuICAgICAgaWYgKHNlYXJjaFF1ZXJ5IHx8IHNlYXJjaE9wZW4pIHtcbiAgICAgICAgc2V0U2VhcmNoT3BlbihmYWxzZSlcbiAgICAgICAgc2V0U2VhcmNoUXVlcnkoJycpXG4gICAgICAgIHNldFNlYXJjaENvdW50KDApXG4gICAgICAgIHNldFNlYXJjaEN1cnJlbnQoMClcbiAgICAgICAganVtcFJlZi5jdXJyZW50Py5kaXNhcm1TZWFyY2goKVxuICAgICAgICBzZXRIaWdobGlnaHQoJycpXG4gICAgICB9XG4gICAgfVxuICB9LCBbdHJhbnNjcmlwdENvbHMsIHNlYXJjaFF1ZXJ5LCBzZWFyY2hPcGVuLCBzZXRIaWdobGlnaHRdKVxuXG4gIC8vIFRyYW5zY3JpcHQgZXNjYXBlIGhhdGNoZXMuIEJhcmUgbGV0dGVycyBpbiBtb2RhbCBjb250ZXh0IChubyBwcm9tcHRcbiAgLy8gY29tcGV0aW5nIGZvciBpbnB1dCkg4oCUIHNhbWUgY2xhc3MgYXMgZy9HL2ovayBpbiBTY3JvbGxLZXliaW5kaW5nSGFuZGxlci5cbiAgdXNlSW5wdXQoXG4gICAgKGlucHV0LCBrZXksIGV2ZW50KSA9PiB7XG4gICAgICBpZiAoa2V5LmN0cmwgfHwga2V5Lm1ldGEpIHJldHVyblxuICAgICAgaWYgKGlucHV0ID09PSAncScpIHtcbiAgICAgICAgLy8gbGVzczogcSBxdWl0cyB0aGUgcGFnZXIuIGN0cmwrbyB0b2dnbGVzOyBxIGlzIHRoZSBsaW5lYWdlIGV4aXQuXG4gICAgICAgIGhhbmRsZUV4aXRUcmFuc2NyaXB0KClcbiAgICAgICAgZXZlbnQuc3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uKClcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG4gICAgICBpZiAoaW5wdXQgPT09ICdbJyAmJiAhZHVtcE1vZGUpIHtcbiAgICAgICAgLy8gRm9yY2UgZHVtcC10by1zY3JvbGxiYWNrLiBBbHNvIGV4cGFuZCArIHVuY2FwIOKAlCBubyBwb2ludCBkdW1waW5nXG4gICAgICAgIC8vIGEgc3Vic2V0LiBUZXJtaW5hbC90bXV4IGNtZC1GIGNhbiBub3cgZmluZCBhbnl0aGluZy4gR3VhcmQgaGVyZVxuICAgICAgICAvLyAobm90IGluIGlzQWN0aXZlKSBzbyB2IHN0aWxsIHdvcmtzIHBvc3QtWyDigJQgZHVtcC1tb2RlIGZvb3RlciBhdFxuICAgICAgICAvLyB+NDg5OCB3aXJlcyBlZGl0b3JTdGF0dXMsIGNvbmZpcm1pbmcgdiBpcyBtZWFudCB0byBzdGF5IGxpdmUuXG4gICAgICAgIHNldER1bXBNb2RlKHRydWUpXG4gICAgICAgIHNldFNob3dBbGxJblRyYW5zY3JpcHQodHJ1ZSlcbiAgICAgICAgZXZlbnQuc3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uKClcbiAgICAgIH0gZWxzZSBpZiAoaW5wdXQgPT09ICd2Jykge1xuICAgICAgICAvLyBsZXNzLXN0eWxlOiB2IG9wZW5zIHRoZSBmaWxlIGluICRWSVNVQUwvJEVESVRPUi4gUmVuZGVyIHRoZSBmdWxsXG4gICAgICAgIC8vIHRyYW5zY3JpcHQgKHNhbWUgcGF0aCAvZXhwb3J0IHVzZXMpLCB3cml0ZSB0byB0bXAsIGhhbmQgb2ZmLlxuICAgICAgICAvLyBvcGVuRmlsZUluRXh0ZXJuYWxFZGl0b3IgaGFuZGxlcyBhbHQtc2NyZWVuIHN1c3BlbmQvcmVzdW1lIGZvclxuICAgICAgICAvLyB0ZXJtaW5hbCBlZGl0b3JzOyBHVUkgZWRpdG9ycyBzcGF3biBkZXRhY2hlZC5cbiAgICAgICAgZXZlbnQuc3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uKClcbiAgICAgICAgLy8gRHJvcCBkb3VibGUtdGFwczogdGhlIHJlbmRlciBpcyBhc3luYyBhbmQgYSBzZWNvbmQgcHJlc3MgYmVmb3JlIGl0XG4gICAgICAgIC8vIGNvbXBsZXRlcyB3b3VsZCBydW4gYSBzZWNvbmQgcGFyYWxsZWwgcmVuZGVyIChkb3VibGUgbWVtb3J5LCB0d29cbiAgICAgICAgLy8gdGVtcGZpbGVzLCB0d28gZWRpdG9yIHNwYXducykuIGVkaXRvckdlblJlZiBvbmx5IGd1YXJkc1xuICAgICAgICAvLyB0cmFuc2NyaXB0LWV4aXQgc3RhbGVuZXNzLCBub3Qgc2FtZS1zZXNzaW9uIGNvbmN1cnJlbmN5LlxuICAgICAgICBpZiAoZWRpdG9yUmVuZGVyaW5nUmVmLmN1cnJlbnQpIHJldHVyblxuICAgICAgICBlZGl0b3JSZW5kZXJpbmdSZWYuY3VycmVudCA9IHRydWVcbiAgICAgICAgLy8gQ2FwdHVyZSBnZW5lcmF0aW9uICsgbWFrZSBhIHN0YWxlbmVzcy1hd2FyZSBzZXR0ZXIuIEVhY2ggd3JpdGVcbiAgICAgICAgLy8gY2hlY2tzIGdlbiAodHJhbnNjcmlwdCBleGl0IGJ1bXBzIGl0IOKGkiBsYXRlIHdyaXRlcyBmcm9tIHRoZVxuICAgICAgICAvLyBhc3luYyByZW5kZXIgZ28gc2lsZW50KS5cbiAgICAgICAgY29uc3QgZ2VuID0gZWRpdG9yR2VuUmVmLmN1cnJlbnRcbiAgICAgICAgY29uc3Qgc2V0U3RhdHVzID0gKHM6IHN0cmluZyk6IHZvaWQgPT4ge1xuICAgICAgICAgIGlmIChnZW4gIT09IGVkaXRvckdlblJlZi5jdXJyZW50KSByZXR1cm5cbiAgICAgICAgICBjbGVhclRpbWVvdXQoZWRpdG9yVGltZXJSZWYuY3VycmVudClcbiAgICAgICAgICBzZXRFZGl0b3JTdGF0dXMocylcbiAgICAgICAgfVxuICAgICAgICBzZXRTdGF0dXMoYHJlbmRlcmluZyAke2RlZmVycmVkTWVzc2FnZXMubGVuZ3RofSBtZXNzYWdlc+KApmApXG4gICAgICAgIHZvaWQgKGFzeW5jICgpID0+IHtcbiAgICAgICAgICB0cnkge1xuICAgICAgICAgICAgLy8gV2lkdGggPSB0ZXJtaW5hbCBtaW51cyB2aW0ncyBsaW5lLW51bWJlciBndXR0ZXIgKDQgZGlnaXRzICtcbiAgICAgICAgICAgIC8vIHNwYWNlICsgc2xhY2spLiBGbG9vciBhdCA4MC4gUGFzc1Rocm91Z2ggaGFzIG5vIC5jb2x1bW5zIHNvXG4gICAgICAgICAgICAvLyB3aXRob3V0IHRoaXMgSW5rIGRlZmF1bHRzIHRvIDgwLiBUcmFpbGluZy1zcGFjZSBzdHJpcDogcmlnaHQtXG4gICAgICAgICAgICAvLyBhbGlnbmVkIHRpbWVzdGFtcHMgc3RpbGwgbGVhdmUgYSBmbGV4Ym94IHNwYWNlciBydW4gYXQgRU9MLlxuICAgICAgICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIGN1c3RvbS1ydWxlcy9wcmVmZXItdXNlLXRlcm1pbmFsLXNpemUgLS0gb25lLXNob3QgYXQga2V5cHJlc3MgdGltZSwgbm90IGEgcmVhY3RpdmUgcmVuZGVyIGRlcFxuICAgICAgICAgICAgY29uc3QgdyA9IE1hdGgubWF4KDgwLCAocHJvY2Vzcy5zdGRvdXQuY29sdW1ucyA/PyA4MCkgLSA2KVxuICAgICAgICAgICAgY29uc3QgcmF3ID0gYXdhaXQgcmVuZGVyTWVzc2FnZXNUb1BsYWluVGV4dChcbiAgICAgICAgICAgICAgZGVmZXJyZWRNZXNzYWdlcyxcbiAgICAgICAgICAgICAgdG9vbHMsXG4gICAgICAgICAgICAgIHcsXG4gICAgICAgICAgICApXG4gICAgICAgICAgICBjb25zdCB0ZXh0ID0gcmF3LnJlcGxhY2UoL1sgXFx0XSskL2dtLCAnJylcbiAgICAgICAgICAgIGNvbnN0IHBhdGggPSBqb2luKHRtcGRpcigpLCBgY2MtdHJhbnNjcmlwdC0ke0RhdGUubm93KCl9LnR4dGApXG4gICAgICAgICAgICBhd2FpdCB3cml0ZUZpbGUocGF0aCwgdGV4dClcbiAgICAgICAgICAgIGNvbnN0IG9wZW5lZCA9IG9wZW5GaWxlSW5FeHRlcm5hbEVkaXRvcihwYXRoKVxuICAgICAgICAgICAgc2V0U3RhdHVzKFxuICAgICAgICAgICAgICBvcGVuZWRcbiAgICAgICAgICAgICAgICA/IGBvcGVuaW5nICR7cGF0aH1gXG4gICAgICAgICAgICAgICAgOiBgd3JvdGUgJHtwYXRofSDCtyBubyAkVklTVUFMLyRFRElUT1Igc2V0YCxcbiAgICAgICAgICAgIClcbiAgICAgICAgICB9IGNhdGNoIChlKSB7XG4gICAgICAgICAgICBzZXRTdGF0dXMoXG4gICAgICAgICAgICAgIGByZW5kZXIgZmFpbGVkOiAke2UgaW5zdGFuY2VvZiBFcnJvciA/IGUubWVzc2FnZSA6IFN0cmluZyhlKX1gLFxuICAgICAgICAgICAgKVxuICAgICAgICAgIH1cbiAgICAgICAgICBlZGl0b3JSZW5kZXJpbmdSZWYuY3VycmVudCA9IGZhbHNlXG4gICAgICAgICAgaWYgKGdlbiAhPT0gZWRpdG9yR2VuUmVmLmN1cnJlbnQpIHJldHVyblxuICAgICAgICAgIGVkaXRvclRpbWVyUmVmLmN1cnJlbnQgPSBzZXRUaW1lb3V0KHMgPT4gcygnJyksIDQwMDAsIHNldEVkaXRvclN0YXR1cylcbiAgICAgICAgfSkoKVxuICAgICAgfVxuICAgIH0sXG4gICAgLy8gIXNlYXJjaE9wZW46IHR5cGluZyAndicgb3IgJ1snIGluIHRoZSBzZWFyY2ggYmFyIGlzIHNlYXJjaCBpbnB1dCwgbm90XG4gICAgLy8gYSBjb21tYW5kLiBObyAhZHVtcE1vZGUgaGVyZSDigJQgdiBzaG91bGQgd29yayBhZnRlciBbICh0aGUgWyBoYW5kbGVyXG4gICAgLy8gZ3VhcmRzIGl0c2VsZiBpbmxpbmUpLlxuICAgIHsgaXNBY3RpdmU6IHNjcmVlbiA9PT0gJ3RyYW5zY3JpcHQnICYmIHZpcnR1YWxTY3JvbGxBY3RpdmUgJiYgIXNlYXJjaE9wZW4gfSxcbiAgKVxuXG4gIC8vIEZyZXNoIGBsZXNzYCBwZXIgdHJhbnNjcmlwdCBlbnRyeS4gUHJldmVudHMgc3RhbGUgaGlnaGxpZ2h0cyBtYXRjaGluZ1xuICAvLyB1bnJlbGF0ZWQgbm9ybWFsLW1vZGUgdGV4dCAob3ZlcmxheSBpcyBhbHQtc2NyZWVuLWdsb2JhbCkgYW5kIGF2b2lkc1xuICAvLyBzdXJwcmlzZSBuL04gb24gcmUtZW50cnkuIFNhbWUgZXhpdCByZXNldHMgWyBkdW1wIG1vZGUg4oCUIGVhY2ggY3RybCtvXG4gIC8vIGVudHJ5IGlzIGEgZnJlc2ggaW5zdGFuY2UuXG4gIGNvbnN0IGluVHJhbnNjcmlwdCA9IHNjcmVlbiA9PT0gJ3RyYW5zY3JpcHQnICYmIHZpcnR1YWxTY3JvbGxBY3RpdmVcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBpZiAoIWluVHJhbnNjcmlwdCkge1xuICAgICAgc2V0U2VhcmNoUXVlcnkoJycpXG4gICAgICBzZXRTZWFyY2hDb3VudCgwKVxuICAgICAgc2V0U2VhcmNoQ3VycmVudCgwKVxuICAgICAgc2V0U2VhcmNoT3BlbihmYWxzZSlcbiAgICAgIGVkaXRvckdlblJlZi5jdXJyZW50KytcbiAgICAgIGNsZWFyVGltZW91dChlZGl0b3JUaW1lclJlZi5jdXJyZW50KVxuICAgICAgc2V0RHVtcE1vZGUoZmFsc2UpXG4gICAgICBzZXRFZGl0b3JTdGF0dXMoJycpXG4gICAgfVxuICB9LCBbaW5UcmFuc2NyaXB0XSlcbiAgdXNlRWZmZWN0KCgpID0+IHtcbiAgICBzZXRIaWdobGlnaHQoaW5UcmFuc2NyaXB0ID8gc2VhcmNoUXVlcnkgOiAnJylcbiAgICAvLyBDbGVhciB0aGUgcG9zaXRpb24tYmFzZWQgQ1VSUkVOVCAoeWVsbG93KSBvdmVybGF5IHRvby4gc2V0SGlnaGxpZ2h0XG4gICAgLy8gb25seSBjbGVhcnMgdGhlIHNjYW4tYmFzZWQgaW52ZXJzZS4gV2l0aG91dCB0aGlzLCB0aGUgeWVsbG93IGJveFxuICAgIC8vIHBlcnNpc3RzIGF0IGl0cyBsYXN0IHNjcmVlbiBjb29yZHMgYWZ0ZXIgY3RybC1jIGV4aXRzIHRyYW5zY3JpcHQuXG4gICAgaWYgKCFpblRyYW5zY3JpcHQpIHNldFBvc2l0aW9ucyhudWxsKVxuICB9LCBbaW5UcmFuc2NyaXB0LCBzZWFyY2hRdWVyeSwgc2V0SGlnaGxpZ2h0LCBzZXRQb3NpdGlvbnNdKVxuXG4gIGNvbnN0IGdsb2JhbEtleWJpbmRpbmdQcm9wcyA9IHtcbiAgICBzY3JlZW4sXG4gICAgc2V0U2NyZWVuLFxuICAgIHNob3dBbGxJblRyYW5zY3JpcHQsXG4gICAgc2V0U2hvd0FsbEluVHJhbnNjcmlwdCxcbiAgICBtZXNzYWdlQ291bnQ6IG1lc3NhZ2VzLmxlbmd0aCxcbiAgICBvbkVudGVyVHJhbnNjcmlwdDogaGFuZGxlRW50ZXJUcmFuc2NyaXB0LFxuICAgIG9uRXhpdFRyYW5zY3JpcHQ6IGhhbmRsZUV4aXRUcmFuc2NyaXB0LFxuICAgIHZpcnR1YWxTY3JvbGxBY3RpdmUsXG4gICAgLy8gQmFyLW9wZW4gaXMgYSBtb2RlIChvd25zIGtleXN0cm9rZXMg4oCUIGovayB0eXBlLCBFc2MgY2FuY2VscykuXG4gICAgLy8gTmF2aWdhdGluZyAocXVlcnkgc2V0LCBiYXIgY2xvc2VkKSBpcyBOT1Qg4oCUIEVzYyBleGl0cyB0cmFuc2NyaXB0LFxuICAgIC8vIHNhbWUgYXMgbGVzcyBxIHdpdGggaGlnaGxpZ2h0cyBzdGlsbCB2aXNpYmxlLiB1c2VTZWFyY2hJbnB1dFxuICAgIC8vIGRvZXNuJ3Qgc3RvcFByb3BhZ2F0aW9uLCBzbyB3aXRob3V0IHRoaXMgZ2F0ZSB0cmFuc2NyaXB0OmV4aXRcbiAgICAvLyB3b3VsZCBmaXJlIG9uIHRoZSBzYW1lIEVzYyB0aGF0IGNhbmNlbHMgdGhlIGJhciAoY2hpbGQgcmVnaXN0ZXJzXG4gICAgLy8gZmlyc3QsIGZpcmVzIGZpcnN0LCBidWJibGVzKS5cbiAgICBzZWFyY2hCYXJPcGVuOiBzZWFyY2hPcGVuLFxuICB9XG5cbiAgLy8gVXNlIGZyb3plbiBsZW5ndGhzIHRvIHNsaWNlIGFycmF5cywgYXZvaWRpbmcgbWVtb3J5IG92ZXJoZWFkIG9mIGNsb25pbmdcbiAgY29uc3QgdHJhbnNjcmlwdE1lc3NhZ2VzID0gZnJvemVuVHJhbnNjcmlwdFN0YXRlXG4gICAgPyBkZWZlcnJlZE1lc3NhZ2VzLnNsaWNlKDAsIGZyb3plblRyYW5zY3JpcHRTdGF0ZS5tZXNzYWdlc0xlbmd0aClcbiAgICA6IGRlZmVycmVkTWVzc2FnZXNcbiAgY29uc3QgdHJhbnNjcmlwdFN0cmVhbWluZ1Rvb2xVc2VzID0gZnJvemVuVHJhbnNjcmlwdFN0YXRlXG4gICAgPyBzdHJlYW1pbmdUb29sVXNlcy5zbGljZSgwLCBmcm96ZW5UcmFuc2NyaXB0U3RhdGUuc3RyZWFtaW5nVG9vbFVzZXNMZW5ndGgpXG4gICAgOiBzdHJlYW1pbmdUb29sVXNlc1xuXG4gIC8vIEhhbmRsZSBzaGlmdCtkb3duIGZvciB0ZWFtbWF0ZSBuYXZpZ2F0aW9uIGFuZCBiYWNrZ3JvdW5kIHRhc2sgbWFuYWdlbWVudC5cbiAgLy8gR3VhcmQgb25PcGVuQmFja2dyb3VuZFRhc2tzIHdoZW4gYSBsb2NhbC1qc3ggZGlhbG9nIChlLmcuIC9tY3ApIGlzIG9wZW4g4oCUXG4gIC8vIG90aGVyd2lzZSBTaGlmdCtEb3duIHN0YWNrcyBCYWNrZ3JvdW5kVGFza3NEaWFsb2cgb24gdG9wIGFuZCBkZWFkbG9ja3MgaW5wdXQuXG4gIHVzZUJhY2tncm91bmRUYXNrTmF2aWdhdGlvbih7XG4gICAgb25PcGVuQmFja2dyb3VuZFRhc2tzOiBpc1Nob3dpbmdMb2NhbEpTWENvbW1hbmRcbiAgICAgID8gdW5kZWZpbmVkXG4gICAgICA6ICgpID0+IHNldFNob3dCYXNoZXNEaWFsb2codHJ1ZSksXG4gIH0pXG4gIC8vIEF1dG8tZXhpdCB2aWV3aW5nIG1vZGUgd2hlbiB0ZWFtbWF0ZSBjb21wbGV0ZXMgb3IgZXJyb3JzXG4gIHVzZVRlYW1tYXRlVmlld0F1dG9FeGl0KClcblxuICBpZiAoc2NyZWVuID09PSAndHJhbnNjcmlwdCcpIHtcbiAgICAvLyBWaXJ0dWFsIHNjcm9sbCByZXBsYWNlcyB0aGUgMzAtbWVzc2FnZSBjYXA6IGV2ZXJ5dGhpbmcgaXMgc2Nyb2xsYWJsZVxuICAgIC8vIGFuZCBtZW1vcnkgaXMgYm91bmRlZCBieSB0aGUgdmlld3BvcnQuIFdpdGhvdXQgaXQsIHdyYXBwaW5nIHRyYW5zY3JpcHRcbiAgICAvLyBpbiBhIFNjcm9sbEJveCB3b3VsZCBtb3VudCBhbGwgbWVzc2FnZXMgKH4yNTAgTUIgb24gbG9uZyBzZXNzaW9ucyDigJRcbiAgICAvLyB0aGUgZXhhY3QgcHJvYmxlbSksIHNvIHRoZSBraWxsIHN3aXRjaCBhbmQgbm9uLWZ1bGxzY3JlZW4gcGF0aHMgbXVzdFxuICAgIC8vIGZhbGwgdGhyb3VnaCB0byB0aGUgbGVnYWN5IHJlbmRlcjogbm8gYWx0IHNjcmVlbiwgZHVtcCB0byB0ZXJtaW5hbFxuICAgIC8vIHNjcm9sbGJhY2ssIDMwLWNhcCArIEN0cmwrRS4gUmV1c2luZyBzY3JvbGxSZWYgaXMgc2FmZSDigJQgbm9ybWFsLW1vZGVcbiAgICAvLyBhbmQgdHJhbnNjcmlwdC1tb2RlIGFyZSBtdXR1YWxseSBleGNsdXNpdmUgKHRoaXMgZWFybHkgcmV0dXJuKSwgc29cbiAgICAvLyBvbmx5IG9uZSBTY3JvbGxCb3ggaXMgZXZlciBtb3VudGVkIGF0IGEgdGltZS5cbiAgICBjb25zdCB0cmFuc2NyaXB0U2Nyb2xsUmVmID1cbiAgICAgIGlzRnVsbHNjcmVlbkVudkVuYWJsZWQoKSAmJiAhZGlzYWJsZVZpcnR1YWxTY3JvbGwgJiYgIWR1bXBNb2RlXG4gICAgICAgID8gc2Nyb2xsUmVmXG4gICAgICAgIDogdW5kZWZpbmVkXG4gICAgY29uc3QgdHJhbnNjcmlwdE1lc3NhZ2VzRWxlbWVudCA9IChcbiAgICAgIDxNZXNzYWdlc1xuICAgICAgICBtZXNzYWdlcz17dHJhbnNjcmlwdE1lc3NhZ2VzfVxuICAgICAgICB0b29scz17dG9vbHN9XG4gICAgICAgIGNvbW1hbmRzPXtjb21tYW5kc31cbiAgICAgICAgdmVyYm9zZT17dHJ1ZX1cbiAgICAgICAgdG9vbEpTWD17bnVsbH1cbiAgICAgICAgdG9vbFVzZUNvbmZpcm1RdWV1ZT17W119XG4gICAgICAgIGluUHJvZ3Jlc3NUb29sVXNlSURzPXtpblByb2dyZXNzVG9vbFVzZUlEc31cbiAgICAgICAgaXNNZXNzYWdlU2VsZWN0b3JWaXNpYmxlPXtmYWxzZX1cbiAgICAgICAgY29udmVyc2F0aW9uSWQ9e2NvbnZlcnNhdGlvbklkfVxuICAgICAgICBzY3JlZW49e3NjcmVlbn1cbiAgICAgICAgYWdlbnREZWZpbml0aW9ucz17YWdlbnREZWZpbml0aW9uc31cbiAgICAgICAgc3RyZWFtaW5nVG9vbFVzZXM9e3RyYW5zY3JpcHRTdHJlYW1pbmdUb29sVXNlc31cbiAgICAgICAgc2hvd0FsbEluVHJhbnNjcmlwdD17c2hvd0FsbEluVHJhbnNjcmlwdH1cbiAgICAgICAgb25PcGVuUmF0ZUxpbWl0T3B0aW9ucz17aGFuZGxlT3BlblJhdGVMaW1pdE9wdGlvbnN9XG4gICAgICAgIGlzTG9hZGluZz17aXNMb2FkaW5nfVxuICAgICAgICBoaWRlUGFzdFRoaW5raW5nPXt0cnVlfVxuICAgICAgICBzdHJlYW1pbmdUaGlua2luZz17c3RyZWFtaW5nVGhpbmtpbmd9XG4gICAgICAgIHNjcm9sbFJlZj17dHJhbnNjcmlwdFNjcm9sbFJlZn1cbiAgICAgICAganVtcFJlZj17anVtcFJlZn1cbiAgICAgICAgb25TZWFyY2hNYXRjaGVzQ2hhbmdlPXtvblNlYXJjaE1hdGNoZXNDaGFuZ2V9XG4gICAgICAgIHNjYW5FbGVtZW50PXtzY2FuRWxlbWVudH1cbiAgICAgICAgc2V0UG9zaXRpb25zPXtzZXRQb3NpdGlvbnN9XG4gICAgICAgIGRpc2FibGVSZW5kZXJDYXA9e2R1bXBNb2RlfVxuICAgICAgLz5cbiAgICApXG4gICAgY29uc3QgdHJhbnNjcmlwdFRvb2xKU1ggPSB0b29sSlNYICYmIChcbiAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIHdpZHRoPVwiMTAwJVwiPlxuICAgICAgICB7dG9vbEpTWC5qc3h9XG4gICAgICA8L0JveD5cbiAgICApXG4gICAgY29uc3QgdHJhbnNjcmlwdFJldHVybiA9IChcbiAgICAgIDxLZXliaW5kaW5nU2V0dXA+XG4gICAgICAgIDxBbmltYXRlZFRlcm1pbmFsVGl0bGVcbiAgICAgICAgICBpc0FuaW1hdGluZz17dGl0bGVJc0FuaW1hdGluZ31cbiAgICAgICAgICB0aXRsZT17dGVybWluYWxUaXRsZX1cbiAgICAgICAgICBkaXNhYmxlZD17dGl0bGVEaXNhYmxlZH1cbiAgICAgICAgICBub1ByZWZpeD17c2hvd1N0YXR1c0luVGVybWluYWxUYWJ9XG4gICAgICAgIC8+XG4gICAgICAgIDxHbG9iYWxLZXliaW5kaW5nSGFuZGxlcnMgey4uLmdsb2JhbEtleWJpbmRpbmdQcm9wc30gLz5cbiAgICAgICAge2ZlYXR1cmUoJ1ZPSUNFX01PREUnKSA/IChcbiAgICAgICAgICA8Vm9pY2VLZXliaW5kaW5nSGFuZGxlclxuICAgICAgICAgICAgdm9pY2VIYW5kbGVLZXlFdmVudD17dm9pY2UuaGFuZGxlS2V5RXZlbnR9XG4gICAgICAgICAgICBzdHJpcFRyYWlsaW5nPXt2b2ljZS5zdHJpcFRyYWlsaW5nfVxuICAgICAgICAgICAgcmVzZXRBbmNob3I9e3ZvaWNlLnJlc2V0QW5jaG9yfVxuICAgICAgICAgICAgaXNBY3RpdmU9eyF0b29sSlNYPy5pc0xvY2FsSlNYQ29tbWFuZH1cbiAgICAgICAgICAvPlxuICAgICAgICApIDogbnVsbH1cbiAgICAgICAgPENvbW1hbmRLZXliaW5kaW5nSGFuZGxlcnNcbiAgICAgICAgICBvblN1Ym1pdD17b25TdWJtaXR9XG4gICAgICAgICAgaXNBY3RpdmU9eyF0b29sSlNYPy5pc0xvY2FsSlNYQ29tbWFuZH1cbiAgICAgICAgLz5cbiAgICAgICAge3RyYW5zY3JpcHRTY3JvbGxSZWYgPyAoXG4gICAgICAgICAgLy8gU2Nyb2xsS2V5YmluZGluZ0hhbmRsZXIgbXVzdCBtb3VudCBiZWZvcmUgQ2FuY2VsUmVxdWVzdEhhbmRsZXIgc29cbiAgICAgICAgICAvLyBjdHJsK2Mtd2l0aC1zZWxlY3Rpb24gY29waWVzIGluc3RlYWQgb2YgY2FuY2VsbGluZyB0aGUgYWN0aXZlIHRhc2suXG4gICAgICAgICAgLy8gSXRzIHJhdyB1c2VJbnB1dCBoYW5kbGVyIG9ubHkgc3RvcHMgcHJvcGFnYXRpb24gd2hlbiBhIHNlbGVjdGlvblxuICAgICAgICAgIC8vIGV4aXN0cyDigJQgd2l0aG91dCBvbmUsIGN0cmwrYyBmYWxscyB0aHJvdWdoIHRvIENhbmNlbFJlcXVlc3RIYW5kbGVyLlxuICAgICAgICAgIDxTY3JvbGxLZXliaW5kaW5nSGFuZGxlclxuICAgICAgICAgICAgc2Nyb2xsUmVmPXtzY3JvbGxSZWZ9XG4gICAgICAgICAgICAvLyBZaWVsZCB3aGVlbC9jdHJsK3UvZCB0byBVbHRyYXBsYW5DaG9pY2VEaWFsb2cncyBvd24gc2Nyb2xsXG4gICAgICAgICAgICAvLyBoYW5kbGVyIHdoaWxlIHRoZSBtb2RhbCBpcyBzaG93aW5nLlxuICAgICAgICAgICAgaXNBY3RpdmU9e2ZvY3VzZWRJbnB1dERpYWxvZyAhPT0gJ3VsdHJhcGxhbi1jaG9pY2UnfVxuICAgICAgICAgICAgLy8gZy9HL2ovay9jdHJsK3UvY3RybCtkIHdvdWxkIGVhdCBrZXlzdHJva2VzIHRoZSBzZWFyY2ggYmFyXG4gICAgICAgICAgICAvLyB3YW50cy4gT2ZmIHdoaWxlIHNlYXJjaGluZy5cbiAgICAgICAgICAgIGlzTW9kYWw9eyFzZWFyY2hPcGVufVxuICAgICAgICAgICAgLy8gTWFudWFsIHNjcm9sbCBleGl0cyB0aGUgc2VhcmNoIGNvbnRleHQg4oCUIGNsZWFyIHRoZSB5ZWxsb3dcbiAgICAgICAgICAgIC8vIGN1cnJlbnQtbWF0Y2ggbWFya2VyLiBQb3NpdGlvbnMgYXJlIChtc2csIHJvd09mZnNldCkta2V5ZWQ7XG4gICAgICAgICAgICAvLyBqL2sgY2hhbmdlcyBzY3JvbGxUb3Agc28gcm93T2Zmc2V0IGlzIHN0YWxlIOKGkiB3cm9uZyByb3dcbiAgICAgICAgICAgIC8vIGdldHMgeWVsbG93LiBOZXh0IG4vTiByZS1lc3RhYmxpc2hlcyB2aWEgc3RlcCgp4oaSanVtcCgpLlxuICAgICAgICAgICAgb25TY3JvbGw9eygpID0+IGp1bXBSZWYuY3VycmVudD8uZGlzYXJtU2VhcmNoKCl9XG4gICAgICAgICAgLz5cbiAgICAgICAgKSA6IG51bGx9XG4gICAgICAgIDxDYW5jZWxSZXF1ZXN0SGFuZGxlciB7Li4uY2FuY2VsUmVxdWVzdFByb3BzfSAvPlxuICAgICAgICB7dHJhbnNjcmlwdFNjcm9sbFJlZiA/IChcbiAgICAgICAgICA8RnVsbHNjcmVlbkxheW91dFxuICAgICAgICAgICAgc2Nyb2xsUmVmPXtzY3JvbGxSZWZ9XG4gICAgICAgICAgICBzY3JvbGxhYmxlPXtcbiAgICAgICAgICAgICAgPD5cbiAgICAgICAgICAgICAgICB7dHJhbnNjcmlwdE1lc3NhZ2VzRWxlbWVudH1cbiAgICAgICAgICAgICAgICB7dHJhbnNjcmlwdFRvb2xKU1h9XG4gICAgICAgICAgICAgICAgPFNhbmRib3hWaW9sYXRpb25FeHBhbmRlZFZpZXcgLz5cbiAgICAgICAgICAgICAgPC8+XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBib3R0b209e1xuICAgICAgICAgICAgICBzZWFyY2hPcGVuID8gKFxuICAgICAgICAgICAgICAgIDxUcmFuc2NyaXB0U2VhcmNoQmFyXG4gICAgICAgICAgICAgICAgICBqdW1wUmVmPXtqdW1wUmVmfVxuICAgICAgICAgICAgICAgICAgLy8gU2VlZCB3YXMgdHJpZWQgKGMwMTU3OGM4KSDigJQgYnJva2UgL2hlbGxvIG11c2NsZVxuICAgICAgICAgICAgICAgICAgLy8gbWVtb3J5IChjdXJzb3IgbGFuZHMgYWZ0ZXIgJ2ZvbycsIC9oZWxsbyDihpIgZm9vaGVsbG8pLlxuICAgICAgICAgICAgICAgICAgLy8gQ2FuY2VsLXJlc3RvcmUgaGFuZGxlcyB0aGUgJ2Rvbid0IGxvc2UgcHJpb3Igc2VhcmNoJ1xuICAgICAgICAgICAgICAgICAgLy8gY29uY2VybiBkaWZmZXJlbnRseSAob25DYW5jZWwgcmUtYXBwbGllcyBzZWFyY2hRdWVyeSkuXG4gICAgICAgICAgICAgICAgICBpbml0aWFsUXVlcnk9XCJcIlxuICAgICAgICAgICAgICAgICAgY291bnQ9e3NlYXJjaENvdW50fVxuICAgICAgICAgICAgICAgICAgY3VycmVudD17c2VhcmNoQ3VycmVudH1cbiAgICAgICAgICAgICAgICAgIG9uQ2xvc2U9e3EgPT4ge1xuICAgICAgICAgICAgICAgICAgICAvLyBFbnRlciDigJQgY29tbWl0LiAwLW1hdGNoIGd1YXJkOiBqdW5rIHF1ZXJ5IHNob3VsZG4ndFxuICAgICAgICAgICAgICAgICAgICAvLyBwZXJzaXN0IChiYWRnZSBoaWRkZW4sIG4vTiBkZWFkIGFueXdheSkuXG4gICAgICAgICAgICAgICAgICAgIHNldFNlYXJjaFF1ZXJ5KHNlYXJjaENvdW50ID4gMCA/IHEgOiAnJylcbiAgICAgICAgICAgICAgICAgICAgc2V0U2VhcmNoT3BlbihmYWxzZSlcbiAgICAgICAgICAgICAgICAgICAgLy8gb25DYW5jZWwgcGF0aDogYmFyIHVubW91bnRzIGJlZm9yZSBpdHMgdXNlRWZmZWN0KFtxdWVyeV0pXG4gICAgICAgICAgICAgICAgICAgIC8vIGNhbiBmaXJlIHdpdGggJycuIFdpdGhvdXQgdGhpcywgc2VhcmNoQ291bnQgc3RheXMgc3RhbGVcbiAgICAgICAgICAgICAgICAgICAgLy8gKG4gZ3VhcmQgYXQgOjQ5NTYgcGFzc2VzKSBhbmQgVk1MJ3MgbWF0Y2hlc1tdIHRvb1xuICAgICAgICAgICAgICAgICAgICAvLyAobmV4dE1hdGNoIHdhbGtzIHRoZSBvbGQgYXJyYXkpLiBQaGFudG9tIG5hdiwgbm9cbiAgICAgICAgICAgICAgICAgICAgLy8gaGlnaGxpZ2h0LiBvbkV4aXQgKEVudGVyLCBxIG5vbi1lbXB0eSkgc3RpbGwgY29tbWl0cy5cbiAgICAgICAgICAgICAgICAgICAgaWYgKCFxKSB7XG4gICAgICAgICAgICAgICAgICAgICAgc2V0U2VhcmNoQ291bnQoMClcbiAgICAgICAgICAgICAgICAgICAgICBzZXRTZWFyY2hDdXJyZW50KDApXG4gICAgICAgICAgICAgICAgICAgICAganVtcFJlZi5jdXJyZW50Py5zZXRTZWFyY2hRdWVyeSgnJylcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgIG9uQ2FuY2VsPXsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgIC8vIEVzYy9jdHJsK2MvY3RybCtnIOKAlCB1bmRvLiBCYXIncyBlZmZlY3QgbGFzdCBmaXJlZFxuICAgICAgICAgICAgICAgICAgICAvLyB3aXRoIHdoYXRldmVyIHdhcyB0eXBlZC4gc2VhcmNoUXVlcnkgKFJFUEwgc3RhdGUpXG4gICAgICAgICAgICAgICAgICAgIC8vIGlzIHVuY2hhbmdlZCBzaW5jZSAvIChvbkNsb3NlID0gY29tbWl0LCBkaWRuJ3QgcnVuKS5cbiAgICAgICAgICAgICAgICAgICAgLy8gVHdvIFZNTCBjYWxsczogJycgcmVzdG9yZXMgYW5jaG9yICgwLW1hdGNoIGVsc2UtXG4gICAgICAgICAgICAgICAgICAgIC8vIGJyYW5jaCksIHRoZW4gc2VhcmNoUXVlcnkgcmUtc2NhbnMgZnJvbSBhbmNob3Inc1xuICAgICAgICAgICAgICAgICAgICAvLyBuZWFyZXN0LiBCb3RoIHN5bmNocm9ub3VzIOKAlCBvbmUgUmVhY3QgYmF0Y2guXG4gICAgICAgICAgICAgICAgICAgIC8vIHNldEhpZ2hsaWdodCBleHBsaWNpdDogUkVQTCdzIHN5bmMtZWZmZWN0IGRlcCBpc1xuICAgICAgICAgICAgICAgICAgICAvLyBzZWFyY2hRdWVyeSAodW5jaGFuZ2VkKSwgd291bGRuJ3QgcmUtZmlyZS5cbiAgICAgICAgICAgICAgICAgICAgc2V0U2VhcmNoT3BlbihmYWxzZSlcbiAgICAgICAgICAgICAgICAgICAganVtcFJlZi5jdXJyZW50Py5zZXRTZWFyY2hRdWVyeSgnJylcbiAgICAgICAgICAgICAgICAgICAganVtcFJlZi5jdXJyZW50Py5zZXRTZWFyY2hRdWVyeShzZWFyY2hRdWVyeSlcbiAgICAgICAgICAgICAgICAgICAgc2V0SGlnaGxpZ2h0KHNlYXJjaFF1ZXJ5KVxuICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgIHNldEhpZ2hsaWdodD17c2V0SGlnaGxpZ2h0fVxuICAgICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICAgICkgOiAoXG4gICAgICAgICAgICAgICAgPFRyYW5zY3JpcHRNb2RlRm9vdGVyXG4gICAgICAgICAgICAgICAgICBzaG93QWxsSW5UcmFuc2NyaXB0PXtzaG93QWxsSW5UcmFuc2NyaXB0fVxuICAgICAgICAgICAgICAgICAgdmlydHVhbFNjcm9sbD17dHJ1ZX1cbiAgICAgICAgICAgICAgICAgIHN0YXR1cz17ZWRpdG9yU3RhdHVzIHx8IHVuZGVmaW5lZH1cbiAgICAgICAgICAgICAgICAgIHNlYXJjaEJhZGdlPXtcbiAgICAgICAgICAgICAgICAgICAgc2VhcmNoUXVlcnkgJiYgc2VhcmNoQ291bnQgPiAwXG4gICAgICAgICAgICAgICAgICAgICAgPyB7IGN1cnJlbnQ6IHNlYXJjaEN1cnJlbnQsIGNvdW50OiBzZWFyY2hDb3VudCB9XG4gICAgICAgICAgICAgICAgICAgICAgOiB1bmRlZmluZWRcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICApXG4gICAgICAgICAgICB9XG4gICAgICAgICAgLz5cbiAgICAgICAgKSA6IChcbiAgICAgICAgICA8PlxuICAgICAgICAgICAge3RyYW5zY3JpcHRNZXNzYWdlc0VsZW1lbnR9XG4gICAgICAgICAgICB7dHJhbnNjcmlwdFRvb2xKU1h9XG4gICAgICAgICAgICA8U2FuZGJveFZpb2xhdGlvbkV4cGFuZGVkVmlldyAvPlxuICAgICAgICAgICAgPFRyYW5zY3JpcHRNb2RlRm9vdGVyXG4gICAgICAgICAgICAgIHNob3dBbGxJblRyYW5zY3JpcHQ9e3Nob3dBbGxJblRyYW5zY3JpcHR9XG4gICAgICAgICAgICAgIHZpcnR1YWxTY3JvbGw9e2ZhbHNlfVxuICAgICAgICAgICAgICBzdXBwcmVzc1Nob3dBbGw9e2R1bXBNb2RlfVxuICAgICAgICAgICAgICBzdGF0dXM9e2VkaXRvclN0YXR1cyB8fCB1bmRlZmluZWR9XG4gICAgICAgICAgICAvPlxuICAgICAgICAgIDwvPlxuICAgICAgICApfVxuICAgICAgPC9LZXliaW5kaW5nU2V0dXA+XG4gICAgKVxuICAgIC8vIFRoZSB2aXJ0dWFsLXNjcm9sbCBicmFuY2ggKEZ1bGxzY3JlZW5MYXlvdXQgYWJvdmUpIG5lZWRzXG4gICAgLy8gPEFsdGVybmF0ZVNjcmVlbj4ncyA8Qm94IGhlaWdodD17cm93c30+IGNvbnN0cmFpbnQg4oCUIHdpdGhvdXQgaXQsXG4gICAgLy8gU2Nyb2xsQm94J3MgZmxleEdyb3cgaGFzIG5vIGNlaWxpbmcsIHZpZXdwb3J0ID0gY29udGVudCBoZWlnaHQsXG4gICAgLy8gc2Nyb2xsVG9wIHBpbnMgYXQgMCwgYW5kIEluaydzIHNjcmVlbiBidWZmZXIgc2l6ZXMgdG8gdGhlIGZ1bGxcbiAgICAvLyBzcGFjZXIgKDIwMMOXNWsrIHJvd3Mgb24gbG9uZyBzZXNzaW9ucykuIFNhbWUgcm9vdCB0eXBlICsgcHJvcHMgYXNcbiAgICAvLyBub3JtYWwgbW9kZSdzIHdyYXAgYmVsb3cgc28gUmVhY3QgcmVjb25jaWxlcyBhbmQgdGhlIGFsdCBidWZmZXJcbiAgICAvLyBzdGF5cyBlbnRlcmVkIGFjcm9zcyB0b2dnbGUuIFRoZSAzMC1jYXAgZHVtcCBicmFuY2ggc3RheXNcbiAgICAvLyB1bndyYXBwZWQg4oCUIGl0IHdhbnRzIG5hdGl2ZSB0ZXJtaW5hbCBzY3JvbGxiYWNrLlxuICAgIGlmICh0cmFuc2NyaXB0U2Nyb2xsUmVmKSB7XG4gICAgICByZXR1cm4gKFxuICAgICAgICA8QWx0ZXJuYXRlU2NyZWVuIG1vdXNlVHJhY2tpbmc9e2lzTW91c2VUcmFja2luZ0VuYWJsZWQoKX0+XG4gICAgICAgICAge3RyYW5zY3JpcHRSZXR1cm59XG4gICAgICAgIDwvQWx0ZXJuYXRlU2NyZWVuPlxuICAgICAgKVxuICAgIH1cbiAgICByZXR1cm4gdHJhbnNjcmlwdFJldHVyblxuICB9XG5cbiAgLy8gR2V0IHZpZXdlZCBhZ2VudCB0YXNrIChpbmxpbmVkIGZyb20gc2VsZWN0b3JzIGZvciBleHBsaWNpdCBkYXRhIGZsb3cpLlxuICAvLyB2aWV3ZWRBZ2VudFRhc2s6IHRlYW1tYXRlIE9SIGxvY2FsX2FnZW50IOKAlCBkcml2ZXMgdGhlIGJvb2xlYW4gY2hlY2tzXG4gIC8vIGJlbG93LiB2aWV3ZWRUZWFtbWF0ZVRhc2s6IHRlYW1tYXRlLW9ubHkgbmFycm93ZWQsIGZvciB0ZWFtbWF0ZS1zcGVjaWZpY1xuICAvLyBmaWVsZCBhY2Nlc3MgKGluUHJvZ3Jlc3NUb29sVXNlSURzKS5cbiAgY29uc3Qgdmlld2VkVGFzayA9IHZpZXdpbmdBZ2VudFRhc2tJZCA/IHRhc2tzW3ZpZXdpbmdBZ2VudFRhc2tJZF0gOiB1bmRlZmluZWRcbiAgY29uc3Qgdmlld2VkVGVhbW1hdGVUYXNrID1cbiAgICB2aWV3ZWRUYXNrICYmIGlzSW5Qcm9jZXNzVGVhbW1hdGVUYXNrKHZpZXdlZFRhc2spID8gdmlld2VkVGFzayA6IHVuZGVmaW5lZFxuICBjb25zdCB2aWV3ZWRBZ2VudFRhc2sgPVxuICAgIHZpZXdlZFRlYW1tYXRlVGFzayA/P1xuICAgICh2aWV3ZWRUYXNrICYmIGlzTG9jYWxBZ2VudFRhc2sodmlld2VkVGFzaykgPyB2aWV3ZWRUYXNrIDogdW5kZWZpbmVkKVxuXG4gIC8vIEJ5cGFzcyB1c2VEZWZlcnJlZFZhbHVlIHdoZW4gc3RyZWFtaW5nIHRleHQgaXMgc2hvd2luZyBzbyBNZXNzYWdlcyByZW5kZXJzXG4gIC8vIHRoZSBmaW5hbCBtZXNzYWdlIGluIHRoZSBzYW1lIGZyYW1lIHN0cmVhbWluZyB0ZXh0IGNsZWFycy4gQWxzbyBieXBhc3Mgd2hlblxuICAvLyBub3QgbG9hZGluZyDigJQgZGVmZXJyZWRNZXNzYWdlcyBvbmx5IG1hdHRlcnMgZHVyaW5nIHN0cmVhbWluZyAoa2VlcHMgaW5wdXRcbiAgLy8gcmVzcG9uc2l2ZSk7IGFmdGVyIHRoZSB0dXJuIGVuZHMsIHNob3dpbmcgbWVzc2FnZXMgaW1tZWRpYXRlbHkgcHJldmVudHMgYVxuICAvLyBqaXR0ZXIgZ2FwIHdoZXJlIHRoZSBzcGlubmVyIGlzIGdvbmUgYnV0IHRoZSBhbnN3ZXIgaGFzbid0IGFwcGVhcmVkIHlldC5cbiAgLy8gT25seSByZWR1Y2VkTW90aW9uIHVzZXJzIGtlZXAgdGhlIGRlZmVycmVkIHBhdGggZHVyaW5nIGxvYWRpbmcuXG4gIGNvbnN0IHVzZXNTeW5jTWVzc2FnZXMgPSBzaG93U3RyZWFtaW5nVGV4dCB8fCAhaXNMb2FkaW5nXG4gIC8vIFdoZW4gdmlld2luZyBhbiBhZ2VudCwgbmV2ZXIgZmFsbCB0aHJvdWdoIHRvIGxlYWRlciDigJQgZW1wdHkgdW50aWxcbiAgLy8gYm9vdHN0cmFwL3N0cmVhbSBmaWxscy4gQ2xvc2VzIHRoZSBzZWUtbGVhZGVyLXR5cGUtYWdlbnQgZm9vdGd1bi5cbiAgY29uc3QgZGlzcGxheWVkTWVzc2FnZXMgPSB2aWV3ZWRBZ2VudFRhc2tcbiAgICA/ICh2aWV3ZWRBZ2VudFRhc2subWVzc2FnZXMgPz8gW10pXG4gICAgOiB1c2VzU3luY01lc3NhZ2VzXG4gICAgICA/IG1lc3NhZ2VzXG4gICAgICA6IGRlZmVycmVkTWVzc2FnZXNcbiAgLy8gU2hvdyB0aGUgcGxhY2Vob2xkZXIgdW50aWwgdGhlIHJlYWwgdXNlciBtZXNzYWdlIGFwcGVhcnMgaW5cbiAgLy8gZGlzcGxheWVkTWVzc2FnZXMuIHVzZXJJbnB1dE9uUHJvY2Vzc2luZyBzdGF5cyBzZXQgZm9yIHRoZSB3aG9sZSB0dXJuXG4gIC8vIChjbGVhcmVkIGluIHJlc2V0TG9hZGluZ1N0YXRlKTsgdGhpcyBsZW5ndGggY2hlY2sgaGlkZXMgaXQgb25jZVxuICAvLyBkaXNwbGF5ZWRNZXNzYWdlcyBncm93cyBwYXN0IHRoZSBiYXNlbGluZSBjYXB0dXJlZCBhdCBzdWJtaXQgdGltZS5cbiAgLy8gQ292ZXJzIGJvdGggZ2FwczogYmVmb3JlIHNldE1lc3NhZ2VzIGlzIGNhbGxlZCAocHJvY2Vzc1VzZXJJbnB1dCksIGFuZFxuICAvLyB3aGlsZSBkZWZlcnJlZE1lc3NhZ2VzIGxhZ3MgYmVoaW5kIG1lc3NhZ2VzLiBTdXBwcmVzc2VkIHdoZW4gdmlld2luZyBhblxuICAvLyBhZ2VudCDigJQgZGlzcGxheWVkTWVzc2FnZXMgaXMgYSBkaWZmZXJlbnQgYXJyYXkgdGhlcmUsIGFuZCBvbkFnZW50U3VibWl0XG4gIC8vIGRvZXNuJ3QgdXNlIHRoZSBwbGFjZWhvbGRlciBhbnl3YXkuXG4gIGNvbnN0IHBsYWNlaG9sZGVyVGV4dCA9XG4gICAgdXNlcklucHV0T25Qcm9jZXNzaW5nICYmXG4gICAgIXZpZXdlZEFnZW50VGFzayAmJlxuICAgIGRpc3BsYXllZE1lc3NhZ2VzLmxlbmd0aCA8PSB1c2VySW5wdXRCYXNlbGluZVJlZi5jdXJyZW50XG4gICAgICA/IHVzZXJJbnB1dE9uUHJvY2Vzc2luZ1xuICAgICAgOiB1bmRlZmluZWRcblxuICBjb25zdCB0b29sUGVybWlzc2lvbk92ZXJsYXkgPVxuICAgIGZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ3Rvb2wtcGVybWlzc2lvbicgPyAoXG4gICAgICA8UGVybWlzc2lvblJlcXVlc3RcbiAgICAgICAga2V5PXt0b29sVXNlQ29uZmlybVF1ZXVlWzBdPy50b29sVXNlSUR9XG4gICAgICAgIG9uRG9uZT17KCkgPT4gc2V0VG9vbFVzZUNvbmZpcm1RdWV1ZSgoW18sIC4uLnRhaWxdKSA9PiB0YWlsKX1cbiAgICAgICAgb25SZWplY3Q9e2hhbmRsZVF1ZXVlZENvbW1hbmRPbkNhbmNlbH1cbiAgICAgICAgdG9vbFVzZUNvbmZpcm09e3Rvb2xVc2VDb25maXJtUXVldWVbMF0hfVxuICAgICAgICB0b29sVXNlQ29udGV4dD17Z2V0VG9vbFVzZUNvbnRleHQoXG4gICAgICAgICAgbWVzc2FnZXMsXG4gICAgICAgICAgbWVzc2FnZXMsXG4gICAgICAgICAgYWJvcnRDb250cm9sbGVyID8/IGNyZWF0ZUFib3J0Q29udHJvbGxlcigpLFxuICAgICAgICAgIG1haW5Mb29wTW9kZWwsXG4gICAgICAgICl9XG4gICAgICAgIHZlcmJvc2U9e3ZlcmJvc2V9XG4gICAgICAgIHdvcmtlckJhZGdlPXt0b29sVXNlQ29uZmlybVF1ZXVlWzBdPy53b3JrZXJCYWRnZX1cbiAgICAgICAgc2V0U3RpY2t5Rm9vdGVyPXtcbiAgICAgICAgICBpc0Z1bGxzY3JlZW5FbnZFbmFibGVkKCkgPyBzZXRQZXJtaXNzaW9uU3RpY2t5Rm9vdGVyIDogdW5kZWZpbmVkXG4gICAgICAgIH1cbiAgICAgIC8+XG4gICAgKSA6IG51bGxcblxuICAvLyBOYXJyb3cgdGVybWluYWxzOiBjb21wYW5pb24gY29sbGFwc2VzIHRvIGEgb25lLWxpbmVyIHRoYXQgUkVQTCBzdGFja3NcbiAgLy8gb24gaXRzIG93biByb3cgKGFib3ZlIGlucHV0IGluIGZ1bGxzY3JlZW4sIGJlbG93IGluIHNjcm9sbGJhY2spIGluc3RlYWRcbiAgLy8gb2Ygcm93LWJlc2lkZS4gV2lkZSB0ZXJtaW5hbHMga2VlcCB0aGUgcm93IGxheW91dCB3aXRoIHNwcml0ZSBvbiB0aGUgcmlnaHQuXG4gIGNvbnN0IGNvbXBhbmlvbk5hcnJvdyA9IHRyYW5zY3JpcHRDb2xzIDwgTUlOX0NPTFNfRk9SX0ZVTExfU1BSSVRFXG4gIC8vIEhpZGUgdGhlIHNwcml0ZSB3aGVuIFByb21wdElucHV0IGVhcmx5LXJldHVybnMgQmFja2dyb3VuZFRhc2tzRGlhbG9nLlxuICAvLyBUaGUgc3ByaXRlIHNpdHMgYXMgYSByb3cgc2libGluZyBvZiBQcm9tcHRJbnB1dCwgc28gdGhlIGRpYWxvZydzIFBhbmVcbiAgLy8gZGl2aWRlciBkcmF3cyBhdCB1c2VUZXJtaW5hbFNpemUoKSB3aWR0aCBidXQgb25seSBnZXRzIHRlcm1pbmFsV2lkdGggLVxuICAvLyBzcHJpdGVXaWR0aCDigJQgZGl2aWRlciBzdG9wcyBzaG9ydCBhbmQgZGlhbG9nIHRleHQgd3JhcHMgZWFybHkuIERvbid0XG4gIC8vIGNoZWNrIGZvb3RlclNlbGVjdGlvbjogcGlsbCBGT0NVUyAoYXJyb3ctZG93biB0byB0YXNrcyBwaWxsKSBtdXN0IGtlZXBcbiAgLy8gdGhlIHNwcml0ZSB2aXNpYmxlIHNvIGFycm93LXJpZ2h0IGNhbiBuYXZpZ2F0ZSB0byBpdC5cbiAgY29uc3QgY29tcGFuaW9uVmlzaWJsZSA9XG4gICAgIXRvb2xKU1g/LnNob3VsZEhpZGVQcm9tcHRJbnB1dCAmJiAhZm9jdXNlZElucHV0RGlhbG9nICYmICFzaG93QmFzaGVzRGlhbG9nXG5cbiAgLy8gSW4gZnVsbHNjcmVlbiwgQUxMIGxvY2FsLWpzeCBzbGFzaCBjb21tYW5kcyBmbG9hdCBpbiB0aGUgbW9kYWwgc2xvdCDigJRcbiAgLy8gRnVsbHNjcmVlbkxheW91dCB3cmFwcyB0aGVtIGluIGFuIGFic29sdXRlLXBvc2l0aW9uZWQgYm90dG9tLWFuY2hvcmVkXG4gIC8vIHBhbmUgKOKWlCBkaXZpZGVyLCBNb2RhbENvbnRleHQpLiBQYW5lL0RpYWxvZyBpbnNpZGUgZGV0ZWN0IHRoZSBjb250ZXh0XG4gIC8vIGFuZCBza2lwIHRoZWlyIG93biB0b3AtbGV2ZWwgZnJhbWUuIE5vbi1mdWxsc2NyZWVuIGtlZXBzIHRoZSBpbmxpbmVcbiAgLy8gcmVuZGVyIHBhdGhzIGJlbG93LiBDb21tYW5kcyB0aGF0IHVzZWQgdG8gcm91dGUgdGhyb3VnaCBib3R0b21cbiAgLy8gKGltbWVkaWF0ZTogL21vZGVsLCAvbWNwLCAvYnR3LCAuLi4pIGFuZCBzY3JvbGxhYmxlIChub24taW1tZWRpYXRlOlxuICAvLyAvY29uZmlnLCAvdGhlbWUsIC9kaWZmLCAuLi4pIGJvdGggZ28gaGVyZSBub3cuXG4gIGNvbnN0IHRvb2xKc3hDZW50ZXJlZCA9XG4gICAgaXNGdWxsc2NyZWVuRW52RW5hYmxlZCgpICYmIHRvb2xKU1g/LmlzTG9jYWxKU1hDb21tYW5kID09PSB0cnVlXG4gIGNvbnN0IGNlbnRlcmVkTW9kYWw6IFJlYWN0LlJlYWN0Tm9kZSA9IHRvb2xKc3hDZW50ZXJlZCA/IHRvb2xKU1ghLmpzeCA6IG51bGxcblxuICAvLyA8QWx0ZXJuYXRlU2NyZWVuPiBhdCB0aGUgcm9vdDogZXZlcnl0aGluZyBiZWxvdyBpcyBpbnNpZGUgaXRzXG4gIC8vIDxCb3ggaGVpZ2h0PXtyb3dzfT4uIEhhbmRsZXJzL2NvbnRleHRzIGFyZSB6ZXJvLWhlaWdodCBzbyBTY3JvbGxCb3gnc1xuICAvLyBmbGV4R3JvdyBpbiBGdWxsc2NyZWVuTGF5b3V0IHJlc29sdmVzIGFnYWluc3QgdGhpcyBCb3guIFRoZSB0cmFuc2NyaXB0XG4gIC8vIGVhcmx5IHJldHVybiBhYm92ZSB3cmFwcyBpdHMgdmlydHVhbC1zY3JvbGwgYnJhbmNoIHRoZSBzYW1lIHdheTsgb25seVxuICAvLyB0aGUgMzAtY2FwIGR1bXAgYnJhbmNoIHN0YXlzIHVud3JhcHBlZCBmb3IgbmF0aXZlIHRlcm1pbmFsIHNjcm9sbGJhY2suXG4gIGNvbnN0IG1haW5SZXR1cm4gPSAoXG4gICAgPEtleWJpbmRpbmdTZXR1cD5cbiAgICAgIDxBbmltYXRlZFRlcm1pbmFsVGl0bGVcbiAgICAgICAgaXNBbmltYXRpbmc9e3RpdGxlSXNBbmltYXRpbmd9XG4gICAgICAgIHRpdGxlPXt0ZXJtaW5hbFRpdGxlfVxuICAgICAgICBkaXNhYmxlZD17dGl0bGVEaXNhYmxlZH1cbiAgICAgICAgbm9QcmVmaXg9e3Nob3dTdGF0dXNJblRlcm1pbmFsVGFifVxuICAgICAgLz5cbiAgICAgIDxHbG9iYWxLZXliaW5kaW5nSGFuZGxlcnMgey4uLmdsb2JhbEtleWJpbmRpbmdQcm9wc30gLz5cbiAgICAgIHtmZWF0dXJlKCdWT0lDRV9NT0RFJykgPyAoXG4gICAgICAgIDxWb2ljZUtleWJpbmRpbmdIYW5kbGVyXG4gICAgICAgICAgdm9pY2VIYW5kbGVLZXlFdmVudD17dm9pY2UuaGFuZGxlS2V5RXZlbnR9XG4gICAgICAgICAgc3RyaXBUcmFpbGluZz17dm9pY2Uuc3RyaXBUcmFpbGluZ31cbiAgICAgICAgICByZXNldEFuY2hvcj17dm9pY2UucmVzZXRBbmNob3J9XG4gICAgICAgICAgaXNBY3RpdmU9eyF0b29sSlNYPy5pc0xvY2FsSlNYQ29tbWFuZH1cbiAgICAgICAgLz5cbiAgICAgICkgOiBudWxsfVxuICAgICAgPENvbW1hbmRLZXliaW5kaW5nSGFuZGxlcnNcbiAgICAgICAgb25TdWJtaXQ9e29uU3VibWl0fVxuICAgICAgICBpc0FjdGl2ZT17IXRvb2xKU1g/LmlzTG9jYWxKU1hDb21tYW5kfVxuICAgICAgLz5cbiAgICAgIHsvKiBTY3JvbGxLZXliaW5kaW5nSGFuZGxlciBtdXN0IG1vdW50IGJlZm9yZSBDYW5jZWxSZXF1ZXN0SGFuZGxlciBzb1xuICAgICAgICAgIGN0cmwrYy13aXRoLXNlbGVjdGlvbiBjb3BpZXMgaW5zdGVhZCBvZiBjYW5jZWxsaW5nIHRoZSBhY3RpdmUgdGFzay5cbiAgICAgICAgICBJdHMgcmF3IHVzZUlucHV0IGhhbmRsZXIgb25seSBzdG9wcyBwcm9wYWdhdGlvbiB3aGVuIGEgc2VsZWN0aW9uXG4gICAgICAgICAgZXhpc3RzIOKAlCB3aXRob3V0IG9uZSwgY3RybCtjIGZhbGxzIHRocm91Z2ggdG8gQ2FuY2VsUmVxdWVzdEhhbmRsZXIuXG4gICAgICAgICAgUGdVcC9QZ0RuL3doZWVsIGFsd2F5cyBzY3JvbGwgdGhlIHRyYW5zY3JpcHQgYmVoaW5kIHRoZSBtb2RhbCDigJRcbiAgICAgICAgICB0aGUgbW9kYWwncyBpbm5lciBTY3JvbGxCb3ggaXMgbm90IGtleWJvYXJkLWRyaXZlbi4gb25TY3JvbGxcbiAgICAgICAgICBzdGF5cyBzdXBwcmVzc2VkIHdoaWxlIGEgbW9kYWwgaXMgc2hvd2luZyBzbyBzY3JvbGwgZG9lc24ndFxuICAgICAgICAgIHN0YW1wIGRpdmlkZXIvcGlsbCBzdGF0ZS4gKi99XG4gICAgICA8U2Nyb2xsS2V5YmluZGluZ0hhbmRsZXJcbiAgICAgICAgc2Nyb2xsUmVmPXtzY3JvbGxSZWZ9XG4gICAgICAgIGlzQWN0aXZlPXtcbiAgICAgICAgICBpc0Z1bGxzY3JlZW5FbnZFbmFibGVkKCkgJiZcbiAgICAgICAgICAoY2VudGVyZWRNb2RhbCAhPSBudWxsIHx8XG4gICAgICAgICAgICAhZm9jdXNlZElucHV0RGlhbG9nIHx8XG4gICAgICAgICAgICBmb2N1c2VkSW5wdXREaWFsb2cgPT09ICd0b29sLXBlcm1pc3Npb24nKVxuICAgICAgICB9XG4gICAgICAgIG9uU2Nyb2xsPXtcbiAgICAgICAgICBjZW50ZXJlZE1vZGFsIHx8IHRvb2xQZXJtaXNzaW9uT3ZlcmxheSB8fCB2aWV3ZWRBZ2VudFRhc2tcbiAgICAgICAgICAgID8gdW5kZWZpbmVkXG4gICAgICAgICAgICA6IGNvbXBvc2VkT25TY3JvbGxcbiAgICAgICAgfVxuICAgICAgLz5cbiAgICAgIHtmZWF0dXJlKCdNRVNTQUdFX0FDVElPTlMnKSAmJlxuICAgICAgaXNGdWxsc2NyZWVuRW52RW5hYmxlZCgpICYmXG4gICAgICAhZGlzYWJsZU1lc3NhZ2VBY3Rpb25zID8gKFxuICAgICAgICA8TWVzc2FnZUFjdGlvbnNLZXliaW5kaW5nc1xuICAgICAgICAgIGhhbmRsZXJzPXttZXNzYWdlQWN0aW9uSGFuZGxlcnN9XG4gICAgICAgICAgaXNBY3RpdmU9e2N1cnNvciAhPT0gbnVsbH1cbiAgICAgICAgLz5cbiAgICAgICkgOiBudWxsfVxuICAgICAgPENhbmNlbFJlcXVlc3RIYW5kbGVyIHsuLi5jYW5jZWxSZXF1ZXN0UHJvcHN9IC8+XG4gICAgICA8TUNQQ29ubmVjdGlvbk1hbmFnZXJcbiAgICAgICAga2V5PXtyZW1vdW50S2V5fVxuICAgICAgICBkeW5hbWljTWNwQ29uZmlnPXtkeW5hbWljTWNwQ29uZmlnfVxuICAgICAgICBpc1N0cmljdE1jcENvbmZpZz17c3RyaWN0TWNwQ29uZmlnfVxuICAgICAgPlxuICAgICAgICA8RnVsbHNjcmVlbkxheW91dFxuICAgICAgICAgIHNjcm9sbFJlZj17c2Nyb2xsUmVmfVxuICAgICAgICAgIG92ZXJsYXk9e3Rvb2xQZXJtaXNzaW9uT3ZlcmxheX1cbiAgICAgICAgICBib3R0b21GbG9hdD17XG4gICAgICAgICAgICBmZWF0dXJlKCdCVUREWScpICYmIGNvbXBhbmlvblZpc2libGUgJiYgIWNvbXBhbmlvbk5hcnJvdyA/IChcbiAgICAgICAgICAgICAgPENvbXBhbmlvbkZsb2F0aW5nQnViYmxlIC8+XG4gICAgICAgICAgICApIDogdW5kZWZpbmVkXG4gICAgICAgICAgfVxuICAgICAgICAgIG1vZGFsPXtjZW50ZXJlZE1vZGFsfVxuICAgICAgICAgIG1vZGFsU2Nyb2xsUmVmPXttb2RhbFNjcm9sbFJlZn1cbiAgICAgICAgICBkaXZpZGVyWVJlZj17ZGl2aWRlcllSZWZ9XG4gICAgICAgICAgaGlkZVBpbGw9eyEhdmlld2VkQWdlbnRUYXNrfVxuICAgICAgICAgIGhpZGVTdGlja3k9eyEhdmlld2VkVGVhbW1hdGVUYXNrfVxuICAgICAgICAgIG5ld01lc3NhZ2VDb3VudD17dW5zZWVuRGl2aWRlcj8uY291bnQgPz8gMH1cbiAgICAgICAgICBvblBpbGxDbGljaz17KCkgPT4ge1xuICAgICAgICAgICAgc2V0Q3Vyc29yKG51bGwpXG4gICAgICAgICAgICBqdW1wVG9OZXcoc2Nyb2xsUmVmLmN1cnJlbnQpXG4gICAgICAgICAgfX1cbiAgICAgICAgICBzY3JvbGxhYmxlPXtcbiAgICAgICAgICAgIDw+XG4gICAgICAgICAgICAgIDxUZWFtbWF0ZVZpZXdIZWFkZXIgLz5cbiAgICAgICAgICAgICAgPE1lc3NhZ2VzXG4gICAgICAgICAgICAgICAgbWVzc2FnZXM9e2Rpc3BsYXllZE1lc3NhZ2VzfVxuICAgICAgICAgICAgICAgIHRvb2xzPXt0b29sc31cbiAgICAgICAgICAgICAgICBjb21tYW5kcz17Y29tbWFuZHN9XG4gICAgICAgICAgICAgICAgdmVyYm9zZT17dmVyYm9zZX1cbiAgICAgICAgICAgICAgICB0b29sSlNYPXt0b29sSlNYfVxuICAgICAgICAgICAgICAgIHRvb2xVc2VDb25maXJtUXVldWU9e3Rvb2xVc2VDb25maXJtUXVldWV9XG4gICAgICAgICAgICAgICAgaW5Qcm9ncmVzc1Rvb2xVc2VJRHM9e1xuICAgICAgICAgICAgICAgICAgdmlld2VkVGVhbW1hdGVUYXNrXG4gICAgICAgICAgICAgICAgICAgID8gKHZpZXdlZFRlYW1tYXRlVGFzay5pblByb2dyZXNzVG9vbFVzZUlEcyA/PyBuZXcgU2V0KCkpXG4gICAgICAgICAgICAgICAgICAgIDogaW5Qcm9ncmVzc1Rvb2xVc2VJRHNcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgaXNNZXNzYWdlU2VsZWN0b3JWaXNpYmxlPXtpc01lc3NhZ2VTZWxlY3RvclZpc2libGV9XG4gICAgICAgICAgICAgICAgY29udmVyc2F0aW9uSWQ9e2NvbnZlcnNhdGlvbklkfVxuICAgICAgICAgICAgICAgIHNjcmVlbj17c2NyZWVufVxuICAgICAgICAgICAgICAgIHN0cmVhbWluZ1Rvb2xVc2VzPXtzdHJlYW1pbmdUb29sVXNlc31cbiAgICAgICAgICAgICAgICBzaG93QWxsSW5UcmFuc2NyaXB0PXtzaG93QWxsSW5UcmFuc2NyaXB0fVxuICAgICAgICAgICAgICAgIGFnZW50RGVmaW5pdGlvbnM9e2FnZW50RGVmaW5pdGlvbnN9XG4gICAgICAgICAgICAgICAgb25PcGVuUmF0ZUxpbWl0T3B0aW9ucz17aGFuZGxlT3BlblJhdGVMaW1pdE9wdGlvbnN9XG4gICAgICAgICAgICAgICAgaXNMb2FkaW5nPXtpc0xvYWRpbmd9XG4gICAgICAgICAgICAgICAgc3RyZWFtaW5nVGV4dD17XG4gICAgICAgICAgICAgICAgICBpc0xvYWRpbmcgJiYgIXZpZXdlZEFnZW50VGFzayA/IHZpc2libGVTdHJlYW1pbmdUZXh0IDogbnVsbFxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBpc0JyaWVmT25seT17dmlld2VkQWdlbnRUYXNrID8gZmFsc2UgOiBpc0JyaWVmT25seX1cbiAgICAgICAgICAgICAgICB1bnNlZW5EaXZpZGVyPXt2aWV3ZWRBZ2VudFRhc2sgPyB1bmRlZmluZWQgOiB1bnNlZW5EaXZpZGVyfVxuICAgICAgICAgICAgICAgIHNjcm9sbFJlZj17aXNGdWxsc2NyZWVuRW52RW5hYmxlZCgpID8gc2Nyb2xsUmVmIDogdW5kZWZpbmVkfVxuICAgICAgICAgICAgICAgIHRyYWNrU3RpY2t5UHJvbXB0PXtpc0Z1bGxzY3JlZW5FbnZFbmFibGVkKCkgPyB0cnVlIDogdW5kZWZpbmVkfVxuICAgICAgICAgICAgICAgIGN1cnNvcj17Y3Vyc29yfVxuICAgICAgICAgICAgICAgIHNldEN1cnNvcj17c2V0Q3Vyc29yfVxuICAgICAgICAgICAgICAgIGN1cnNvck5hdlJlZj17Y3Vyc29yTmF2UmVmfVxuICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICA8QXdzQXV0aFN0YXR1c0JveCAvPlxuICAgICAgICAgICAgICB7LyogSGlkZSB0aGUgcHJvY2Vzc2luZyBwbGFjZWhvbGRlciB3aGlsZSBhIG1vZGFsIGlzIHNob3dpbmcg4oCUXG4gICAgICAgICAgICAgICAgICBpdCB3b3VsZCBzaXQgYXQgdGhlIGxhc3QgdmlzaWJsZSB0cmFuc2NyaXB0IHJvdyByaWdodCBhYm92ZVxuICAgICAgICAgICAgICAgICAgdGhlIOKWlCBkaXZpZGVyLCBzaG93aW5nIFwi4p2vIC9jb25maWdcIiBhcyByZWR1bmRhbnQgY2x1dHRlclxuICAgICAgICAgICAgICAgICAgKHRoZSBtb2RhbCBJUyB0aGUgL2NvbmZpZyBVSSkuIE91dHNpZGUgbW9kYWxzIGl0IHN0YXlzIHNvXG4gICAgICAgICAgICAgICAgICB0aGUgdXNlciBzZWVzIHRoZWlyIGlucHV0IGVjaG9lZCB3aGlsZSBDbGF1ZGUgcHJvY2Vzc2VzLiAqL31cbiAgICAgICAgICAgICAgeyFkaXNhYmxlZCAmJiBwbGFjZWhvbGRlclRleHQgJiYgIWNlbnRlcmVkTW9kYWwgJiYgKFxuICAgICAgICAgICAgICAgIDxVc2VyVGV4dE1lc3NhZ2VcbiAgICAgICAgICAgICAgICAgIHBhcmFtPXt7IHRleHQ6IHBsYWNlaG9sZGVyVGV4dCwgdHlwZTogJ3RleHQnIH19XG4gICAgICAgICAgICAgICAgICBhZGRNYXJnaW49e3RydWV9XG4gICAgICAgICAgICAgICAgICB2ZXJib3NlPXt2ZXJib3NlfVxuICAgICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgIHt0b29sSlNYICYmXG4gICAgICAgICAgICAgICAgISh0b29sSlNYLmlzTG9jYWxKU1hDb21tYW5kICYmIHRvb2xKU1guaXNJbW1lZGlhdGUpICYmXG4gICAgICAgICAgICAgICAgIXRvb2xKc3hDZW50ZXJlZCAmJiAoXG4gICAgICAgICAgICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiB3aWR0aD1cIjEwMCVcIj5cbiAgICAgICAgICAgICAgICAgICAge3Rvb2xKU1guanN4fVxuICAgICAgICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAge1wiZXh0ZXJuYWxcIiA9PT0gJ2FudCcgJiYgPFR1bmdzdGVuTGl2ZU1vbml0b3IgLz59XG4gICAgICAgICAgICAgIHtmZWF0dXJlKCdXRUJfQlJPV1NFUl9UT09MJylcbiAgICAgICAgICAgICAgICA/IFdlYkJyb3dzZXJQYW5lbE1vZHVsZSAmJiAoXG4gICAgICAgICAgICAgICAgICAgIDxXZWJCcm93c2VyUGFuZWxNb2R1bGUuV2ViQnJvd3NlclBhbmVsIC8+XG4gICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgOiBudWxsfVxuICAgICAgICAgICAgICA8Qm94IGZsZXhHcm93PXsxfSAvPlxuICAgICAgICAgICAgICB7c2hvd1NwaW5uZXIgJiYgKFxuICAgICAgICAgICAgICAgIDxTcGlubmVyV2l0aFZlcmJcbiAgICAgICAgICAgICAgICAgIG1vZGU9e3N0cmVhbU1vZGV9XG4gICAgICAgICAgICAgICAgICBzcGlubmVyVGlwPXtzcGlubmVyVGlwfVxuICAgICAgICAgICAgICAgICAgcmVzcG9uc2VMZW5ndGhSZWY9e3Jlc3BvbnNlTGVuZ3RoUmVmfVxuICAgICAgICAgICAgICAgICAgYXBpTWV0cmljc1JlZj17YXBpTWV0cmljc1JlZn1cbiAgICAgICAgICAgICAgICAgIG92ZXJyaWRlTWVzc2FnZT17c3Bpbm5lck1lc3NhZ2V9XG4gICAgICAgICAgICAgICAgICBzcGlubmVyU3VmZml4PXtzdG9wSG9va1NwaW5uZXJTdWZmaXh9XG4gICAgICAgICAgICAgICAgICB2ZXJib3NlPXt2ZXJib3NlfVxuICAgICAgICAgICAgICAgICAgbG9hZGluZ1N0YXJ0VGltZVJlZj17bG9hZGluZ1N0YXJ0VGltZVJlZn1cbiAgICAgICAgICAgICAgICAgIHRvdGFsUGF1c2VkTXNSZWY9e3RvdGFsUGF1c2VkTXNSZWZ9XG4gICAgICAgICAgICAgICAgICBwYXVzZVN0YXJ0VGltZVJlZj17cGF1c2VTdGFydFRpbWVSZWZ9XG4gICAgICAgICAgICAgICAgICBvdmVycmlkZUNvbG9yPXtzcGlubmVyQ29sb3J9XG4gICAgICAgICAgICAgICAgICBvdmVycmlkZVNoaW1tZXJDb2xvcj17c3Bpbm5lclNoaW1tZXJDb2xvcn1cbiAgICAgICAgICAgICAgICAgIGhhc0FjdGl2ZVRvb2xzPXtpblByb2dyZXNzVG9vbFVzZUlEcy5zaXplID4gMH1cbiAgICAgICAgICAgICAgICAgIGxlYWRlcklzSWRsZT17IWlzTG9hZGluZ31cbiAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICB7IXNob3dTcGlubmVyICYmXG4gICAgICAgICAgICAgICAgIWlzTG9hZGluZyAmJlxuICAgICAgICAgICAgICAgICF1c2VySW5wdXRPblByb2Nlc3NpbmcgJiZcbiAgICAgICAgICAgICAgICAhaGFzUnVubmluZ1RlYW1tYXRlcyAmJlxuICAgICAgICAgICAgICAgIGlzQnJpZWZPbmx5ICYmXG4gICAgICAgICAgICAgICAgIXZpZXdlZEFnZW50VGFzayAmJiA8QnJpZWZJZGxlU3RhdHVzIC8+fVxuICAgICAgICAgICAgICB7aXNGdWxsc2NyZWVuRW52RW5hYmxlZCgpICYmIDxQcm9tcHRJbnB1dFF1ZXVlZENvbW1hbmRzIC8+fVxuICAgICAgICAgICAgPC8+XG4gICAgICAgICAgfVxuICAgICAgICAgIGJvdHRvbT17XG4gICAgICAgICAgICA8Qm94XG4gICAgICAgICAgICAgIGZsZXhEaXJlY3Rpb249e1xuICAgICAgICAgICAgICAgIGZlYXR1cmUoJ0JVRERZJykgJiYgY29tcGFuaW9uTmFycm93ID8gJ2NvbHVtbicgOiAncm93J1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIHdpZHRoPVwiMTAwJVwiXG4gICAgICAgICAgICAgIGFsaWduSXRlbXM9e1xuICAgICAgICAgICAgICAgIGZlYXR1cmUoJ0JVRERZJykgJiYgY29tcGFuaW9uTmFycm93ID8gdW5kZWZpbmVkIDogJ2ZsZXgtZW5kJ1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIHtmZWF0dXJlKCdCVUREWScpICYmXG4gICAgICAgICAgICAgIGNvbXBhbmlvbk5hcnJvdyAmJlxuICAgICAgICAgICAgICBpc0Z1bGxzY3JlZW5FbnZFbmFibGVkKCkgJiZcbiAgICAgICAgICAgICAgY29tcGFuaW9uVmlzaWJsZSA/IChcbiAgICAgICAgICAgICAgICA8Q29tcGFuaW9uU3ByaXRlIC8+XG4gICAgICAgICAgICAgICkgOiBudWxsfVxuICAgICAgICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBmbGV4R3Jvdz17MX0+XG4gICAgICAgICAgICAgICAge3Blcm1pc3Npb25TdGlja3lGb290ZXJ9XG4gICAgICAgICAgICAgICAgey8qIEltbWVkaWF0ZSBsb2NhbC1qc3ggY29tbWFuZHMgKC9idHcsIC9zYW5kYm94LCAvYXNzaXN0YW50LFxuICAgICAgICAgICAgICAgICAgL2lzc3VlKSByZW5kZXIgaGVyZSwgTk9UIGluc2lkZSBzY3JvbGxhYmxlLiBUaGV5IHN0YXkgbW91bnRlZFxuICAgICAgICAgICAgICAgICAgd2hpbGUgdGhlIG1haW4gY29udmVyc2F0aW9uIHN0cmVhbXMgYmVoaW5kIHRoZW0sIHNvIFNjcm9sbEJveFxuICAgICAgICAgICAgICAgICAgcmVsYXlvdXRzIG9uIGVhY2ggbmV3IG1lc3NhZ2Ugd291bGQgZHJhZyB0aGVtIGFyb3VuZC4gYm90dG9tXG4gICAgICAgICAgICAgICAgICBpcyBmbGV4U2hyaW5rPXswfSBvdXRzaWRlIHRoZSBTY3JvbGxCb3gg4oCUIGl0IG5ldmVyIG1vdmVzLlxuICAgICAgICAgICAgICAgICAgTm9uLWltbWVkaWF0ZSBsb2NhbC1qc3ggKC9kaWZmLCAvc3RhdHVzLCAvdGhlbWUsIH40MCBvdGhlcnMpXG4gICAgICAgICAgICAgICAgICBzdGF5cyBpbiBzY3JvbGxhYmxlOiB0aGUgbWFpbiBsb29wIGlzIHBhdXNlZCBzbyBubyBqaWdnbGUsXG4gICAgICAgICAgICAgICAgICBhbmQgdGhlaXIgdGFsbCBjb250ZW50IChEaWZmRGV0YWlsVmlldyByZW5kZXJzIHVwIHRvIDQwMFxuICAgICAgICAgICAgICAgICAgbGluZXMgd2l0aCBubyBpbnRlcm5hbCBzY3JvbGwpIG5lZWRzIHRoZSBvdXRlciBTY3JvbGxCb3guICovfVxuICAgICAgICAgICAgICAgIHt0b29sSlNYPy5pc0xvY2FsSlNYQ29tbWFuZCAmJlxuICAgICAgICAgICAgICAgICAgdG9vbEpTWC5pc0ltbWVkaWF0ZSAmJlxuICAgICAgICAgICAgICAgICAgIXRvb2xKc3hDZW50ZXJlZCAmJiAoXG4gICAgICAgICAgICAgICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIHdpZHRoPVwiMTAwJVwiPlxuICAgICAgICAgICAgICAgICAgICAgIHt0b29sSlNYLmpzeH1cbiAgICAgICAgICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICAgIHshc2hvd1NwaW5uZXIgJiZcbiAgICAgICAgICAgICAgICAgICF0b29sSlNYPy5pc0xvY2FsSlNYQ29tbWFuZCAmJlxuICAgICAgICAgICAgICAgICAgc2hvd0V4cGFuZGVkVG9kb3MgJiZcbiAgICAgICAgICAgICAgICAgIHRhc2tzVjIgJiZcbiAgICAgICAgICAgICAgICAgIHRhc2tzVjIubGVuZ3RoID4gMCAmJiAoXG4gICAgICAgICAgICAgICAgICAgIDxCb3ggd2lkdGg9XCIxMDAlXCIgZmxleERpcmVjdGlvbj1cImNvbHVtblwiPlxuICAgICAgICAgICAgICAgICAgICAgIDxUYXNrTGlzdFYyIHRhc2tzPXt0YXNrc1YyfSBpc1N0YW5kYWxvbmU9e3RydWV9IC8+XG4gICAgICAgICAgICAgICAgICAgIDwvQm94PlxuICAgICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAgICB7Zm9jdXNlZElucHV0RGlhbG9nID09PSAnc2FuZGJveC1wZXJtaXNzaW9uJyAmJiAoXG4gICAgICAgICAgICAgICAgICA8U2FuZGJveFBlcm1pc3Npb25SZXF1ZXN0XG4gICAgICAgICAgICAgICAgICAgIGtleT17c2FuZGJveFBlcm1pc3Npb25SZXF1ZXN0UXVldWVbMF0hLmhvc3RQYXR0ZXJuLmhvc3R9XG4gICAgICAgICAgICAgICAgICAgIGhvc3RQYXR0ZXJuPXtzYW5kYm94UGVybWlzc2lvblJlcXVlc3RRdWV1ZVswXSEuaG9zdFBhdHRlcm59XG4gICAgICAgICAgICAgICAgICAgIG9uVXNlclJlc3BvbnNlPXsocmVzcG9uc2U6IHtcbiAgICAgICAgICAgICAgICAgICAgICBhbGxvdzogYm9vbGVhblxuICAgICAgICAgICAgICAgICAgICAgIHBlcnNpc3RUb1NldHRpbmdzOiBib29sZWFuXG4gICAgICAgICAgICAgICAgICAgIH0pID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCB7IGFsbG93LCBwZXJzaXN0VG9TZXR0aW5ncyB9ID0gcmVzcG9uc2VcbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjdXJyZW50UmVxdWVzdCA9IHNhbmRib3hQZXJtaXNzaW9uUmVxdWVzdFF1ZXVlWzBdXG4gICAgICAgICAgICAgICAgICAgICAgaWYgKCFjdXJyZW50UmVxdWVzdCkgcmV0dXJuXG5cbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBhcHByb3ZlZEhvc3QgPSBjdXJyZW50UmVxdWVzdC5ob3N0UGF0dGVybi5ob3N0XG5cbiAgICAgICAgICAgICAgICAgICAgICBpZiAocGVyc2lzdFRvU2V0dGluZ3MpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHVwZGF0ZSA9IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgdHlwZTogJ2FkZFJ1bGVzJyBhcyBjb25zdCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgcnVsZXM6IFtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0b29sTmFtZTogV0VCX0ZFVENIX1RPT0xfTkFNRSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJ1bGVDb250ZW50OiBgZG9tYWluOiR7YXBwcm92ZWRIb3N0fWAsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgXSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgYmVoYXZpb3I6IChhbGxvdyA/ICdhbGxvdycgOiAnZGVueScpIGFzXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfCAnYWxsb3cnXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgfCAnZGVueScsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGRlc3RpbmF0aW9uOiAnbG9jYWxTZXR0aW5ncycgYXMgY29uc3QsXG4gICAgICAgICAgICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4gKHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICAgICAgICAgICAgICAgICAgdG9vbFBlcm1pc3Npb25Db250ZXh0OiBhcHBseVBlcm1pc3Npb25VcGRhdGUoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgcHJldi50b29sUGVybWlzc2lvbkNvbnRleHQsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdXBkYXRlLFxuICAgICAgICAgICAgICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICAgICAgICAgICAgfSkpXG5cbiAgICAgICAgICAgICAgICAgICAgICAgIHBlcnNpc3RQZXJtaXNzaW9uVXBkYXRlKHVwZGF0ZSlcblxuICAgICAgICAgICAgICAgICAgICAgICAgLy8gSW1tZWRpYXRlbHkgdXBkYXRlIHNhbmRib3ggaW4tbWVtb3J5IGNvbmZpZyB0byBwcmV2ZW50IHJhY2UgY29uZGl0aW9uc1xuICAgICAgICAgICAgICAgICAgICAgICAgLy8gd2hlcmUgcGVuZGluZyByZXF1ZXN0cyBzbGlwIHRocm91Z2ggYmVmb3JlIHNldHRpbmdzIGNoYW5nZSBpcyBkZXRlY3RlZFxuICAgICAgICAgICAgICAgICAgICAgICAgU2FuZGJveE1hbmFnZXIucmVmcmVzaENvbmZpZygpXG4gICAgICAgICAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgICAgICAgICAgLy8gUmVzb2x2ZSBBTEwgcGVuZGluZyByZXF1ZXN0cyBmb3IgdGhlIHNhbWUgaG9zdCAobm90IGp1c3QgdGhlIGZpcnN0IG9uZSlcbiAgICAgICAgICAgICAgICAgICAgICAvLyBUaGlzIGhhbmRsZXMgdGhlIGNhc2Ugd2hlcmUgbXVsdGlwbGUgcGFyYWxsZWwgcmVxdWVzdHMgY2FtZSBpbiBmb3IgdGhlIHNhbWUgZG9tYWluXG4gICAgICAgICAgICAgICAgICAgICAgc2V0U2FuZGJveFBlcm1pc3Npb25SZXF1ZXN0UXVldWUocXVldWUgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgcXVldWVcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLmZpbHRlcihcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBpdGVtID0+IGl0ZW0uaG9zdFBhdHRlcm4uaG9zdCA9PT0gYXBwcm92ZWRIb3N0LFxuICAgICAgICAgICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgICAgICAgICAgIC5mb3JFYWNoKGl0ZW0gPT4gaXRlbS5yZXNvbHZlUHJvbWlzZShhbGxvdykpXG4gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm4gcXVldWUuZmlsdGVyKFxuICAgICAgICAgICAgICAgICAgICAgICAgICBpdGVtID0+IGl0ZW0uaG9zdFBhdHRlcm4uaG9zdCAhPT0gYXBwcm92ZWRIb3N0LFxuICAgICAgICAgICAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgICAgICAgICAgIH0pXG5cbiAgICAgICAgICAgICAgICAgICAgICAvLyBDbGVhbiB1cCBicmlkZ2Ugc3Vic2NyaXB0aW9ucyBhbmQgY2FuY2VsIHJlbW90ZSBwcm9tcHRzXG4gICAgICAgICAgICAgICAgICAgICAgLy8gZm9yIHRoaXMgaG9zdCBzaW5jZSB0aGUgbG9jYWwgdXNlciBhbHJlYWR5IHJlc3BvbmRlZC5cbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjbGVhbnVwcyA9XG4gICAgICAgICAgICAgICAgICAgICAgICBzYW5kYm94QnJpZGdlQ2xlYW51cFJlZi5jdXJyZW50LmdldChhcHByb3ZlZEhvc3QpXG4gICAgICAgICAgICAgICAgICAgICAgaWYgKGNsZWFudXBzKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBmb3IgKGNvbnN0IGZuIG9mIGNsZWFudXBzKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIGZuKClcbiAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgIHNhbmRib3hCcmlkZ2VDbGVhbnVwUmVmLmN1cnJlbnQuZGVsZXRlKGFwcHJvdmVkSG9zdClcbiAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAge2ZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ3Byb21wdCcgJiYgKFxuICAgICAgICAgICAgICAgICAgPFByb21wdERpYWxvZ1xuICAgICAgICAgICAgICAgICAgICBrZXk9e3Byb21wdFF1ZXVlWzBdIS5yZXF1ZXN0LnByb21wdH1cbiAgICAgICAgICAgICAgICAgICAgdGl0bGU9e3Byb21wdFF1ZXVlWzBdIS50aXRsZX1cbiAgICAgICAgICAgICAgICAgICAgdG9vbElucHV0U3VtbWFyeT17cHJvbXB0UXVldWVbMF0hLnRvb2xJbnB1dFN1bW1hcnl9XG4gICAgICAgICAgICAgICAgICAgIHJlcXVlc3Q9e3Byb21wdFF1ZXVlWzBdIS5yZXF1ZXN0fVxuICAgICAgICAgICAgICAgICAgICBvblJlc3BvbmQ9e3NlbGVjdGVkS2V5ID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBpdGVtID0gcHJvbXB0UXVldWVbMF1cbiAgICAgICAgICAgICAgICAgICAgICBpZiAoIWl0ZW0pIHJldHVyblxuICAgICAgICAgICAgICAgICAgICAgIGl0ZW0ucmVzb2x2ZSh7XG4gICAgICAgICAgICAgICAgICAgICAgICBwcm9tcHRfcmVzcG9uc2U6IGl0ZW0ucmVxdWVzdC5wcm9tcHQsXG4gICAgICAgICAgICAgICAgICAgICAgICBzZWxlY3RlZDogc2VsZWN0ZWRLZXksXG4gICAgICAgICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgICAgICAgICBzZXRQcm9tcHRRdWV1ZSgoWywgLi4udGFpbF0pID0+IHRhaWwpXG4gICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAgIG9uQWJvcnQ9eygpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBpdGVtID0gcHJvbXB0UXVldWVbMF1cbiAgICAgICAgICAgICAgICAgICAgICBpZiAoIWl0ZW0pIHJldHVyblxuICAgICAgICAgICAgICAgICAgICAgIGl0ZW0ucmVqZWN0KG5ldyBFcnJvcignUHJvbXB0IGNhbmNlbGxlZCBieSB1c2VyJykpXG4gICAgICAgICAgICAgICAgICAgICAgc2V0UHJvbXB0UXVldWUoKFssIC4uLnRhaWxdKSA9PiB0YWlsKVxuICAgICAgICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICAgIHsvKiBTaG93IHBlbmRpbmcgaW5kaWNhdG9yIG9uIHdvcmtlciB3aGlsZSB3YWl0aW5nIGZvciBsZWFkZXIgYXBwcm92YWwgKi99XG4gICAgICAgICAgICAgICAge3BlbmRpbmdXb3JrZXJSZXF1ZXN0ICYmIChcbiAgICAgICAgICAgICAgICAgIDxXb3JrZXJQZW5kaW5nUGVybWlzc2lvblxuICAgICAgICAgICAgICAgICAgICB0b29sTmFtZT17cGVuZGluZ1dvcmtlclJlcXVlc3QudG9vbE5hbWV9XG4gICAgICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uPXtwZW5kaW5nV29ya2VyUmVxdWVzdC5kZXNjcmlwdGlvbn1cbiAgICAgICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAgICB7LyogU2hvdyBwZW5kaW5nIGluZGljYXRvciBmb3Igc2FuZGJveCBwZXJtaXNzaW9uIG9uIHdvcmtlciBzaWRlICovfVxuICAgICAgICAgICAgICAgIHtwZW5kaW5nU2FuZGJveFJlcXVlc3QgJiYgKFxuICAgICAgICAgICAgICAgICAgPFdvcmtlclBlbmRpbmdQZXJtaXNzaW9uXG4gICAgICAgICAgICAgICAgICAgIHRvb2xOYW1lPVwiTmV0d29yayBBY2Nlc3NcIlxuICAgICAgICAgICAgICAgICAgICBkZXNjcmlwdGlvbj17YFdhaXRpbmcgZm9yIGxlYWRlciB0byBhcHByb3ZlIG5ldHdvcmsgYWNjZXNzIHRvICR7cGVuZGluZ1NhbmRib3hSZXF1ZXN0Lmhvc3R9YH1cbiAgICAgICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAgICB7LyogV29ya2VyIHNhbmRib3ggcGVybWlzc2lvbiByZXF1ZXN0cyBmcm9tIHN3YXJtIHdvcmtlcnMgKi99XG4gICAgICAgICAgICAgICAge2ZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ3dvcmtlci1zYW5kYm94LXBlcm1pc3Npb24nICYmIChcbiAgICAgICAgICAgICAgICAgIDxTYW5kYm94UGVybWlzc2lvblJlcXVlc3RcbiAgICAgICAgICAgICAgICAgICAga2V5PXt3b3JrZXJTYW5kYm94UGVybWlzc2lvbnMucXVldWVbMF0hLnJlcXVlc3RJZH1cbiAgICAgICAgICAgICAgICAgICAgaG9zdFBhdHRlcm49e1xuICAgICAgICAgICAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGhvc3Q6IHdvcmtlclNhbmRib3hQZXJtaXNzaW9ucy5xdWV1ZVswXSEuaG9zdCxcbiAgICAgICAgICAgICAgICAgICAgICAgIHBvcnQ6IHVuZGVmaW5lZCxcbiAgICAgICAgICAgICAgICAgICAgICB9IGFzIE5ldHdvcmtIb3N0UGF0dGVyblxuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgIG9uVXNlclJlc3BvbnNlPXsocmVzcG9uc2U6IHtcbiAgICAgICAgICAgICAgICAgICAgICBhbGxvdzogYm9vbGVhblxuICAgICAgICAgICAgICAgICAgICAgIHBlcnNpc3RUb1NldHRpbmdzOiBib29sZWFuXG4gICAgICAgICAgICAgICAgICAgIH0pID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCB7IGFsbG93LCBwZXJzaXN0VG9TZXR0aW5ncyB9ID0gcmVzcG9uc2VcbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBjdXJyZW50UmVxdWVzdCA9IHdvcmtlclNhbmRib3hQZXJtaXNzaW9ucy5xdWV1ZVswXVxuICAgICAgICAgICAgICAgICAgICAgIGlmICghY3VycmVudFJlcXVlc3QpIHJldHVyblxuXG4gICAgICAgICAgICAgICAgICAgICAgY29uc3QgYXBwcm92ZWRIb3N0ID0gY3VycmVudFJlcXVlc3QuaG9zdFxuXG4gICAgICAgICAgICAgICAgICAgICAgLy8gU2VuZCByZXNwb25zZSB2aWEgbWFpbGJveCB0byB0aGUgd29ya2VyXG4gICAgICAgICAgICAgICAgICAgICAgdm9pZCBzZW5kU2FuZGJveFBlcm1pc3Npb25SZXNwb25zZVZpYU1haWxib3goXG4gICAgICAgICAgICAgICAgICAgICAgICBjdXJyZW50UmVxdWVzdC53b3JrZXJOYW1lLFxuICAgICAgICAgICAgICAgICAgICAgICAgY3VycmVudFJlcXVlc3QucmVxdWVzdElkLFxuICAgICAgICAgICAgICAgICAgICAgICAgYXBwcm92ZWRIb3N0LFxuICAgICAgICAgICAgICAgICAgICAgICAgYWxsb3csXG4gICAgICAgICAgICAgICAgICAgICAgICB0ZWFtQ29udGV4dD8udGVhbU5hbWUsXG4gICAgICAgICAgICAgICAgICAgICAgKVxuXG4gICAgICAgICAgICAgICAgICAgICAgaWYgKHBlcnNpc3RUb1NldHRpbmdzICYmIGFsbG93KSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCB1cGRhdGUgPSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIHR5cGU6ICdhZGRSdWxlcycgYXMgY29uc3QsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIHJ1bGVzOiBbXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgdG9vbE5hbWU6IFdFQl9GRVRDSF9UT09MX05BTUUsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBydWxlQ29udGVudDogYGRvbWFpbjoke2FwcHJvdmVkSG9zdH1gLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAgICAgICAgICAgIF0sXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGJlaGF2aW9yOiAnYWxsb3cnIGFzIGNvbnN0LFxuICAgICAgICAgICAgICAgICAgICAgICAgICBkZXN0aW5hdGlvbjogJ2xvY2FsU2V0dGluZ3MnIGFzIGNvbnN0LFxuICAgICAgICAgICAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgICAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+ICh7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIHRvb2xQZXJtaXNzaW9uQ29udGV4dDogYXBwbHlQZXJtaXNzaW9uVXBkYXRlKFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHByZXYudG9vbFBlcm1pc3Npb25Db250ZXh0LFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHVwZGF0ZSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgICAgICAgICAgICAgIH0pKVxuXG4gICAgICAgICAgICAgICAgICAgICAgICBwZXJzaXN0UGVybWlzc2lvblVwZGF0ZSh1cGRhdGUpXG4gICAgICAgICAgICAgICAgICAgICAgICBTYW5kYm94TWFuYWdlci5yZWZyZXNoQ29uZmlnKClcbiAgICAgICAgICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgICAgICAgICAvLyBSZW1vdmUgZnJvbSBxdWV1ZVxuICAgICAgICAgICAgICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4gKHtcbiAgICAgICAgICAgICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgICAgICAgICAgICB3b3JrZXJTYW5kYm94UGVybWlzc2lvbnM6IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLi4ucHJldi53b3JrZXJTYW5kYm94UGVybWlzc2lvbnMsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIHF1ZXVlOiBwcmV2LndvcmtlclNhbmRib3hQZXJtaXNzaW9ucy5xdWV1ZS5zbGljZSgxKSxcbiAgICAgICAgICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAgICAgICAgfSkpXG4gICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAge2ZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ2VsaWNpdGF0aW9uJyAmJiAoXG4gICAgICAgICAgICAgICAgICA8RWxpY2l0YXRpb25EaWFsb2dcbiAgICAgICAgICAgICAgICAgICAga2V5PXtcbiAgICAgICAgICAgICAgICAgICAgICBlbGljaXRhdGlvbi5xdWV1ZVswXSEuc2VydmVyTmFtZSArXG4gICAgICAgICAgICAgICAgICAgICAgJzonICtcbiAgICAgICAgICAgICAgICAgICAgICBTdHJpbmcoZWxpY2l0YXRpb24ucXVldWVbMF0hLnJlcXVlc3RJZClcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICBldmVudD17ZWxpY2l0YXRpb24ucXVldWVbMF0hfVxuICAgICAgICAgICAgICAgICAgICBvblJlc3BvbnNlPXsoYWN0aW9uLCBjb250ZW50KSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgY29uc3QgY3VycmVudFJlcXVlc3QgPSBlbGljaXRhdGlvbi5xdWV1ZVswXVxuICAgICAgICAgICAgICAgICAgICAgIGlmICghY3VycmVudFJlcXVlc3QpIHJldHVyblxuICAgICAgICAgICAgICAgICAgICAgIC8vIENhbGwgcmVzcG9uZCBjYWxsYmFjayB0byByZXNvbHZlIFByb21pc2VcbiAgICAgICAgICAgICAgICAgICAgICBjdXJyZW50UmVxdWVzdC5yZXNwb25kKHsgYWN0aW9uLCBjb250ZW50IH0pXG4gICAgICAgICAgICAgICAgICAgICAgLy8gRm9yIFVSTCBhY2NlcHQsIGtlZXAgaW4gcXVldWUgZm9yIHBoYXNlIDJcbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBpc1VybEFjY2VwdCA9XG4gICAgICAgICAgICAgICAgICAgICAgICBjdXJyZW50UmVxdWVzdC5wYXJhbXMubW9kZSA9PT0gJ3VybCcgJiZcbiAgICAgICAgICAgICAgICAgICAgICAgIGFjdGlvbiA9PT0gJ2FjY2VwdCdcbiAgICAgICAgICAgICAgICAgICAgICBpZiAoIWlzVXJsQWNjZXB0KSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+ICh7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGVsaWNpdGF0aW9uOiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgcXVldWU6IHByZXYuZWxpY2l0YXRpb24ucXVldWUuc2xpY2UoMSksXG4gICAgICAgICAgICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAgICAgICAgICB9KSlcbiAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAgIG9uV2FpdGluZ0Rpc21pc3M9e2FjdGlvbiA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgY29uc3QgY3VycmVudFJlcXVlc3QgPSBlbGljaXRhdGlvbi5xdWV1ZVswXVxuICAgICAgICAgICAgICAgICAgICAgIC8vIFJlbW92ZSBmcm9tIHF1ZXVlXG4gICAgICAgICAgICAgICAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiAoe1xuICAgICAgICAgICAgICAgICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICAgICAgICAgICAgICAgIGVsaWNpdGF0aW9uOiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIHF1ZXVlOiBwcmV2LmVsaWNpdGF0aW9uLnF1ZXVlLnNsaWNlKDEpLFxuICAgICAgICAgICAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgICAgICAgICB9KSlcbiAgICAgICAgICAgICAgICAgICAgICBjdXJyZW50UmVxdWVzdD8ub25XYWl0aW5nRGlzbWlzcz8uKGFjdGlvbilcbiAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAgICB7Zm9jdXNlZElucHV0RGlhbG9nID09PSAnY29zdCcgJiYgKFxuICAgICAgICAgICAgICAgICAgPENvc3RUaHJlc2hvbGREaWFsb2dcbiAgICAgICAgICAgICAgICAgICAgb25Eb25lPXsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgc2V0U2hvd0Nvc3REaWFsb2coZmFsc2UpXG4gICAgICAgICAgICAgICAgICAgICAgc2V0SGF2ZVNob3duQ29zdERpYWxvZyh0cnVlKVxuICAgICAgICAgICAgICAgICAgICAgIHNhdmVHbG9iYWxDb25maWcoY3VycmVudCA9PiAoe1xuICAgICAgICAgICAgICAgICAgICAgICAgLi4uY3VycmVudCxcbiAgICAgICAgICAgICAgICAgICAgICAgIGhhc0Fja25vd2xlZGdlZENvc3RUaHJlc2hvbGQ6IHRydWUsXG4gICAgICAgICAgICAgICAgICAgICAgfSkpXG4gICAgICAgICAgICAgICAgICAgICAgbG9nRXZlbnQoJ3Rlbmd1X2Nvc3RfdGhyZXNob2xkX2Fja25vd2xlZGdlZCcsIHt9KVxuICAgICAgICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICAgIHtmb2N1c2VkSW5wdXREaWFsb2cgPT09ICdpZGxlLXJldHVybicgJiYgaWRsZVJldHVyblBlbmRpbmcgJiYgKFxuICAgICAgICAgICAgICAgICAgPElkbGVSZXR1cm5EaWFsb2dcbiAgICAgICAgICAgICAgICAgICAgaWRsZU1pbnV0ZXM9e2lkbGVSZXR1cm5QZW5kaW5nLmlkbGVNaW51dGVzfVxuICAgICAgICAgICAgICAgICAgICB0b3RhbElucHV0VG9rZW5zPXtnZXRUb3RhbElucHV0VG9rZW5zKCl9XG4gICAgICAgICAgICAgICAgICAgIG9uRG9uZT17YXN5bmMgYWN0aW9uID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBwZW5kaW5nID0gaWRsZVJldHVyblBlbmRpbmdcbiAgICAgICAgICAgICAgICAgICAgICBzZXRJZGxlUmV0dXJuUGVuZGluZyhudWxsKVxuICAgICAgICAgICAgICAgICAgICAgIGxvZ0V2ZW50KCd0ZW5ndV9pZGxlX3JldHVybl9hY3Rpb24nLCB7XG4gICAgICAgICAgICAgICAgICAgICAgICBhY3Rpb246XG4gICAgICAgICAgICAgICAgICAgICAgICAgIGFjdGlvbiBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICAgICAgICAgICAgICAgICAgaWRsZU1pbnV0ZXM6IE1hdGgucm91bmQocGVuZGluZy5pZGxlTWludXRlcyksXG4gICAgICAgICAgICAgICAgICAgICAgICBtZXNzYWdlQ291bnQ6IG1lc3NhZ2VzUmVmLmN1cnJlbnQubGVuZ3RoLFxuICAgICAgICAgICAgICAgICAgICAgICAgdG90YWxJbnB1dFRva2VuczogZ2V0VG90YWxJbnB1dFRva2VucygpLFxuICAgICAgICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgICAgICAgaWYgKGFjdGlvbiA9PT0gJ2Rpc21pc3MnKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBzZXRJbnB1dFZhbHVlKHBlbmRpbmcuaW5wdXQpXG4gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm5cbiAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgaWYgKGFjdGlvbiA9PT0gJ25ldmVyJykge1xuICAgICAgICAgICAgICAgICAgICAgICAgc2F2ZUdsb2JhbENvbmZpZyhjdXJyZW50ID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgKGN1cnJlbnQuaWRsZVJldHVybkRpc21pc3NlZCkgcmV0dXJuIGN1cnJlbnRcbiAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIHsgLi4uY3VycmVudCwgaWRsZVJldHVybkRpc21pc3NlZDogdHJ1ZSB9XG4gICAgICAgICAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICBpZiAoYWN0aW9uID09PSAnY2xlYXInKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBjb25zdCB7IGNsZWFyQ29udmVyc2F0aW9uIH0gPSBhd2FpdCBpbXBvcnQoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICcuLi9jb21tYW5kcy9jbGVhci9jb252ZXJzYXRpb24uanMnXG4gICAgICAgICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgICAgICAgICBhd2FpdCBjbGVhckNvbnZlcnNhdGlvbih7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIHNldE1lc3NhZ2VzLFxuICAgICAgICAgICAgICAgICAgICAgICAgICByZWFkRmlsZVN0YXRlOiByZWFkRmlsZVN0YXRlLmN1cnJlbnQsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGRpc2NvdmVyZWRTa2lsbE5hbWVzOiBkaXNjb3ZlcmVkU2tpbGxOYW1lc1JlZi5jdXJyZW50LFxuICAgICAgICAgICAgICAgICAgICAgICAgICBsb2FkZWROZXN0ZWRNZW1vcnlQYXRoczpcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBsb2FkZWROZXN0ZWRNZW1vcnlQYXRoc1JlZi5jdXJyZW50LFxuICAgICAgICAgICAgICAgICAgICAgICAgICBnZXRBcHBTdGF0ZTogKCkgPT4gc3RvcmUuZ2V0U3RhdGUoKSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgc2V0QXBwU3RhdGUsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIHNldENvbnZlcnNhdGlvbklkLFxuICAgICAgICAgICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgICAgICAgICAgIGhhaWt1VGl0bGVBdHRlbXB0ZWRSZWYuY3VycmVudCA9IGZhbHNlXG4gICAgICAgICAgICAgICAgICAgICAgICBzZXRIYWlrdVRpdGxlKHVuZGVmaW5lZClcbiAgICAgICAgICAgICAgICAgICAgICAgIGJhc2hUb29scy5jdXJyZW50LmNsZWFyKClcbiAgICAgICAgICAgICAgICAgICAgICAgIGJhc2hUb29sc1Byb2Nlc3NlZElkeC5jdXJyZW50ID0gMFxuICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICBza2lwSWRsZUNoZWNrUmVmLmN1cnJlbnQgPSB0cnVlXG4gICAgICAgICAgICAgICAgICAgICAgdm9pZCBvblN1Ym1pdFJlZi5jdXJyZW50KHBlbmRpbmcuaW5wdXQsIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIHNldEN1cnNvck9mZnNldDogKCkgPT4ge30sXG4gICAgICAgICAgICAgICAgICAgICAgICBjbGVhckJ1ZmZlcjogKCkgPT4ge30sXG4gICAgICAgICAgICAgICAgICAgICAgICByZXNldEhpc3Rvcnk6ICgpID0+IHt9LFxuICAgICAgICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAge2ZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ2lkZS1vbmJvYXJkaW5nJyAmJiAoXG4gICAgICAgICAgICAgICAgICA8SWRlT25ib2FyZGluZ0RpYWxvZ1xuICAgICAgICAgICAgICAgICAgICBvbkRvbmU9eygpID0+IHNldFNob3dJZGVPbmJvYXJkaW5nKGZhbHNlKX1cbiAgICAgICAgICAgICAgICAgICAgaW5zdGFsbGF0aW9uU3RhdHVzPXtpZGVJbnN0YWxsYXRpb25TdGF0dXN9XG4gICAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAge1wiZXh0ZXJuYWxcIiA9PT0gJ2FudCcgJiZcbiAgICAgICAgICAgICAgICAgIGZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ21vZGVsLXN3aXRjaCcgJiZcbiAgICAgICAgICAgICAgICAgIEFudE1vZGVsU3dpdGNoQ2FsbG91dCAmJiAoXG4gICAgICAgICAgICAgICAgICAgIDxBbnRNb2RlbFN3aXRjaENhbGxvdXRcbiAgICAgICAgICAgICAgICAgICAgICBvbkRvbmU9eyhzZWxlY3Rpb246IHN0cmluZywgbW9kZWxBbGlhcz86IHN0cmluZykgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgc2V0U2hvd01vZGVsU3dpdGNoQ2FsbG91dChmYWxzZSlcbiAgICAgICAgICAgICAgICAgICAgICAgIGlmIChzZWxlY3Rpb24gPT09ICdzd2l0Y2gnICYmIG1vZGVsQWxpYXMpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiAoe1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgbWFpbkxvb3BNb2RlbDogbW9kZWxBbGlhcyxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBtYWluTG9vcE1vZGVsRm9yU2Vzc2lvbjogbnVsbCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgfSkpXG4gICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAge1wiZXh0ZXJuYWxcIiA9PT0gJ2FudCcgJiZcbiAgICAgICAgICAgICAgICAgIGZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ3VuZGVyY292ZXItY2FsbG91dCcgJiZcbiAgICAgICAgICAgICAgICAgIFVuZGVyY292ZXJBdXRvQ2FsbG91dCAmJiAoXG4gICAgICAgICAgICAgICAgICAgIDxVbmRlcmNvdmVyQXV0b0NhbGxvdXRcbiAgICAgICAgICAgICAgICAgICAgICBvbkRvbmU9eygpID0+IHNldFNob3dVbmRlcmNvdmVyQ2FsbG91dChmYWxzZSl9XG4gICAgICAgICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICAgIHtmb2N1c2VkSW5wdXREaWFsb2cgPT09ICdlZmZvcnQtY2FsbG91dCcgJiYgKFxuICAgICAgICAgICAgICAgICAgPEVmZm9ydENhbGxvdXRcbiAgICAgICAgICAgICAgICAgICAgbW9kZWw9e21haW5Mb29wTW9kZWx9XG4gICAgICAgICAgICAgICAgICAgIG9uRG9uZT17c2VsZWN0aW9uID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICBzZXRTaG93RWZmb3J0Q2FsbG91dChmYWxzZSlcbiAgICAgICAgICAgICAgICAgICAgICBpZiAoc2VsZWN0aW9uICE9PSAnZGlzbWlzcycpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4gKHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICAgICAgICAgICAgICAgICAgZWZmb3J0VmFsdWU6IHNlbGVjdGlvbixcbiAgICAgICAgICAgICAgICAgICAgICAgIH0pKVxuICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAgICB7Zm9jdXNlZElucHV0RGlhbG9nID09PSAncmVtb3RlLWNhbGxvdXQnICYmIChcbiAgICAgICAgICAgICAgICAgIDxSZW1vdGVDYWxsb3V0XG4gICAgICAgICAgICAgICAgICAgIG9uRG9uZT17c2VsZWN0aW9uID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGlmICghcHJldi5zaG93UmVtb3RlQ2FsbG91dCkgcmV0dXJuIHByZXZcbiAgICAgICAgICAgICAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIHNob3dSZW1vdGVDYWxsb3V0OiBmYWxzZSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLi4uKHNlbGVjdGlvbiA9PT0gJ2VuYWJsZScgJiYge1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VFbmFibGVkOiB0cnVlLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VFeHBsaWNpdDogdHJ1ZSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXBsQnJpZGdlT3V0Ym91bmRPbmx5OiBmYWxzZSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgfSksXG4gICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICAgICAgKX1cblxuICAgICAgICAgICAgICAgIHtleGl0Rmxvd31cblxuICAgICAgICAgICAgICAgIHtmb2N1c2VkSW5wdXREaWFsb2cgPT09ICdwbHVnaW4taGludCcgJiYgaGludFJlY29tbWVuZGF0aW9uICYmIChcbiAgICAgICAgICAgICAgICAgIDxQbHVnaW5IaW50TWVudVxuICAgICAgICAgICAgICAgICAgICBwbHVnaW5OYW1lPXtoaW50UmVjb21tZW5kYXRpb24ucGx1Z2luTmFtZX1cbiAgICAgICAgICAgICAgICAgICAgcGx1Z2luRGVzY3JpcHRpb249e2hpbnRSZWNvbW1lbmRhdGlvbi5wbHVnaW5EZXNjcmlwdGlvbn1cbiAgICAgICAgICAgICAgICAgICAgbWFya2V0cGxhY2VOYW1lPXtoaW50UmVjb21tZW5kYXRpb24ubWFya2V0cGxhY2VOYW1lfVxuICAgICAgICAgICAgICAgICAgICBzb3VyY2VDb21tYW5kPXtoaW50UmVjb21tZW5kYXRpb24uc291cmNlQ29tbWFuZH1cbiAgICAgICAgICAgICAgICAgICAgb25SZXNwb25zZT17aGFuZGxlSGludFJlc3BvbnNlfVxuICAgICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgICApfVxuXG4gICAgICAgICAgICAgICAge2ZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ2xzcC1yZWNvbW1lbmRhdGlvbicgJiZcbiAgICAgICAgICAgICAgICAgIGxzcFJlY29tbWVuZGF0aW9uICYmIChcbiAgICAgICAgICAgICAgICAgICAgPExzcFJlY29tbWVuZGF0aW9uTWVudVxuICAgICAgICAgICAgICAgICAgICAgIHBsdWdpbk5hbWU9e2xzcFJlY29tbWVuZGF0aW9uLnBsdWdpbk5hbWV9XG4gICAgICAgICAgICAgICAgICAgICAgcGx1Z2luRGVzY3JpcHRpb249e2xzcFJlY29tbWVuZGF0aW9uLnBsdWdpbkRlc2NyaXB0aW9ufVxuICAgICAgICAgICAgICAgICAgICAgIGZpbGVFeHRlbnNpb249e2xzcFJlY29tbWVuZGF0aW9uLmZpbGVFeHRlbnNpb259XG4gICAgICAgICAgICAgICAgICAgICAgb25SZXNwb25zZT17aGFuZGxlTHNwUmVzcG9uc2V9XG4gICAgICAgICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICAgICAgICApfVxuXG4gICAgICAgICAgICAgICAge2ZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ2Rlc2t0b3AtdXBzZWxsJyAmJiAoXG4gICAgICAgICAgICAgICAgICA8RGVza3RvcFVwc2VsbFN0YXJ0dXBcbiAgICAgICAgICAgICAgICAgICAgb25Eb25lPXsoKSA9PiBzZXRTaG93RGVza3RvcFVwc2VsbFN0YXJ0dXAoZmFsc2UpfVxuICAgICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgICApfVxuXG4gICAgICAgICAgICAgICAge2ZlYXR1cmUoJ1VMVFJBUExBTicpXG4gICAgICAgICAgICAgICAgICA/IGZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ3VsdHJhcGxhbi1jaG9pY2UnICYmXG4gICAgICAgICAgICAgICAgICAgIHVsdHJhcGxhblBlbmRpbmdDaG9pY2UgJiYgKFxuICAgICAgICAgICAgICAgICAgICAgIDxVbHRyYXBsYW5DaG9pY2VEaWFsb2dcbiAgICAgICAgICAgICAgICAgICAgICAgIHBsYW49e3VsdHJhcGxhblBlbmRpbmdDaG9pY2UucGxhbn1cbiAgICAgICAgICAgICAgICAgICAgICAgIHNlc3Npb25JZD17dWx0cmFwbGFuUGVuZGluZ0Nob2ljZS5zZXNzaW9uSWR9XG4gICAgICAgICAgICAgICAgICAgICAgICB0YXNrSWQ9e3VsdHJhcGxhblBlbmRpbmdDaG9pY2UudGFza0lkfVxuICAgICAgICAgICAgICAgICAgICAgICAgc2V0TWVzc2FnZXM9e3NldE1lc3NhZ2VzfVxuICAgICAgICAgICAgICAgICAgICAgICAgcmVhZEZpbGVTdGF0ZT17cmVhZEZpbGVTdGF0ZS5jdXJyZW50fVxuICAgICAgICAgICAgICAgICAgICAgICAgZ2V0QXBwU3RhdGU9eygpID0+IHN0b3JlLmdldFN0YXRlKCl9XG4gICAgICAgICAgICAgICAgICAgICAgICBzZXRDb252ZXJzYXRpb25JZD17c2V0Q29udmVyc2F0aW9uSWR9XG4gICAgICAgICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgICAgICAgOiBudWxsfVxuXG4gICAgICAgICAgICAgICAge2ZlYXR1cmUoJ1VMVFJBUExBTicpXG4gICAgICAgICAgICAgICAgICA/IGZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ3VsdHJhcGxhbi1sYXVuY2gnICYmXG4gICAgICAgICAgICAgICAgICAgIHVsdHJhcGxhbkxhdW5jaFBlbmRpbmcgJiYgKFxuICAgICAgICAgICAgICAgICAgICAgIDxVbHRyYXBsYW5MYXVuY2hEaWFsb2dcbiAgICAgICAgICAgICAgICAgICAgICAgIG9uQ2hvaWNlPXsoY2hvaWNlLCBvcHRzKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGJsdXJiID0gdWx0cmFwbGFuTGF1bmNoUGVuZGluZy5ibHVyYlxuICAgICAgICAgICAgICAgICAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgcHJldi51bHRyYXBsYW5MYXVuY2hQZW5kaW5nXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA/IHsgLi4ucHJldiwgdWx0cmFwbGFuTGF1bmNoUGVuZGluZzogdW5kZWZpbmVkIH1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDogcHJldixcbiAgICAgICAgICAgICAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoY2hvaWNlID09PSAnY2FuY2VsJykgcmV0dXJuXG4gICAgICAgICAgICAgICAgICAgICAgICAgIC8vIENvbW1hbmQncyBvbkRvbmUgdXNlZCBkaXNwbGF5Oidza2lwJywgc28gYWRkIHRoZVxuICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBlY2hvIGhlcmUg4oCUIGdpdmVzIGltbWVkaWF0ZSBmZWVkYmFjayBiZWZvcmUgdGhlXG4gICAgICAgICAgICAgICAgICAgICAgICAgIC8vIH41cyB0ZWxlcG9ydFRvUmVtb3RlIHJlc29sdmVzLlxuICAgICAgICAgICAgICAgICAgICAgICAgICBzZXRNZXNzYWdlcyhwcmV2ID0+IFtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAuLi5wcmV2LFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNyZWF0ZUNvbW1hbmRJbnB1dE1lc3NhZ2UoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBmb3JtYXRDb21tYW5kSW5wdXRUYWdzKCd1bHRyYXBsYW4nLCBibHVyYiksXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgXSlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgYXBwZW5kU3Rkb3V0ID0gKG1zZzogc3RyaW5nKSA9PlxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNldE1lc3NhZ2VzKHByZXYgPT4gW1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNyZWF0ZUNvbW1hbmRJbnB1dE1lc3NhZ2UoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGA8JHtMT0NBTF9DT01NQU5EX1NURE9VVF9UQUd9PiR7ZXNjYXBlWG1sKG1zZyl9PC8ke0xPQ0FMX0NPTU1BTkRfU1RET1VUX1RBR30+YCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgXSlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gRGVmZXIgdGhlIHNlY29uZCBtZXNzYWdlIGlmIGEgcXVlcnkgaXMgbWlkLXR1cm5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gc28gaXQgbGFuZHMgYWZ0ZXIgdGhlIGFzc2lzdGFudCByZXBseSwgbm90XG4gICAgICAgICAgICAgICAgICAgICAgICAgIC8vIGJldHdlZW4gdGhlIHVzZXIncyBwcm9tcHQgYW5kIHRoZSByZXBseS5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgYXBwZW5kV2hlbklkbGUgPSAobXNnOiBzdHJpbmcpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoIXF1ZXJ5R3VhcmQuaXNBY3RpdmUpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGFwcGVuZFN0ZG91dChtc2cpXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXR1cm5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgY29uc3QgdW5zdWIgPSBxdWVyeUd1YXJkLnN1YnNjcmliZSgoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAocXVlcnlHdWFyZC5pc0FjdGl2ZSkgcmV0dXJuXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICB1bnN1YigpXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBTa2lwIGlmIHRoZSB1c2VyIHN0b3BwZWQgdWx0cmFwbGFuIHdoaWxlIHdlXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyB3ZXJlIHdhaXRpbmcg4oCUIGF2b2lkcyBhIHN0YWxlIFwiTW9uaXRvcmluZ1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gPHVybD5cIiBtZXNzYWdlIGZvciBhIHNlc3Npb24gdGhhdCdzIGdvbmUuXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBpZiAoIXN0b3JlLmdldFN0YXRlKCkudWx0cmFwbGFuU2Vzc2lvblVybCkgcmV0dXJuXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBhcHBlbmRTdGRvdXQobXNnKVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgdm9pZCBsYXVuY2hVbHRyYXBsYW4oe1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJsdXJiLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGdldEFwcFN0YXRlOiAoKSA9PiBzdG9yZS5nZXRTdGF0ZSgpLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNldEFwcFN0YXRlLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNpZ25hbDogY3JlYXRlQWJvcnRDb250cm9sbGVyKCkuc2lnbmFsLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRpc2Nvbm5lY3RlZEJyaWRnZTogb3B0cz8uZGlzY29ubmVjdGVkQnJpZGdlLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIG9uU2Vzc2lvblJlYWR5OiBhcHBlbmRXaGVuSWRsZSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAudGhlbihhcHBlbmRTdGRvdXQpXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgLmNhdGNoKGxvZ0Vycm9yKVxuICAgICAgICAgICAgICAgICAgICAgICAgfX1cbiAgICAgICAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgICA6IG51bGx9XG5cbiAgICAgICAgICAgICAgICB7bXJSZW5kZXIoKX1cblxuICAgICAgICAgICAgICAgIHshdG9vbEpTWD8uc2hvdWxkSGlkZVByb21wdElucHV0ICYmXG4gICAgICAgICAgICAgICAgICAhZm9jdXNlZElucHV0RGlhbG9nICYmXG4gICAgICAgICAgICAgICAgICAhaXNFeGl0aW5nICYmXG4gICAgICAgICAgICAgICAgICAhZGlzYWJsZWQgJiZcbiAgICAgICAgICAgICAgICAgICFjdXJzb3IgJiYgKFxuICAgICAgICAgICAgICAgICAgICA8PlxuICAgICAgICAgICAgICAgICAgICAgIHthdXRvUnVuSXNzdWVSZWFzb24gJiYgKFxuICAgICAgICAgICAgICAgICAgICAgICAgPEF1dG9SdW5Jc3N1ZU5vdGlmaWNhdGlvblxuICAgICAgICAgICAgICAgICAgICAgICAgICBvblJ1bj17aGFuZGxlQXV0b1J1bklzc3VlfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBvbkNhbmNlbD17aGFuZGxlQ2FuY2VsQXV0b1J1bklzc3VlfVxuICAgICAgICAgICAgICAgICAgICAgICAgICByZWFzb249e2dldEF1dG9SdW5Jc3N1ZVJlYXNvblRleHQoYXV0b1J1bklzc3VlUmVhc29uKX1cbiAgICAgICAgICAgICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAgICAgICAgICB7cG9zdENvbXBhY3RTdXJ2ZXkuc3RhdGUgIT09ICdjbG9zZWQnID8gKFxuICAgICAgICAgICAgICAgICAgICAgICAgPEZlZWRiYWNrU3VydmV5XG4gICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXRlPXtwb3N0Q29tcGFjdFN1cnZleS5zdGF0ZX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgbGFzdFJlc3BvbnNlPXtwb3N0Q29tcGFjdFN1cnZleS5sYXN0UmVzcG9uc2V9XG4gICAgICAgICAgICAgICAgICAgICAgICAgIGhhbmRsZVNlbGVjdD17cG9zdENvbXBhY3RTdXJ2ZXkuaGFuZGxlU2VsZWN0fVxuICAgICAgICAgICAgICAgICAgICAgICAgICBpbnB1dFZhbHVlPXtpbnB1dFZhbHVlfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBzZXRJbnB1dFZhbHVlPXtzZXRJbnB1dFZhbHVlfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBvblJlcXVlc3RGZWVkYmFjaz17aGFuZGxlU3VydmV5UmVxdWVzdEZlZWRiYWNrfVxuICAgICAgICAgICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgICAgICAgICApIDogbWVtb3J5U3VydmV5LnN0YXRlICE9PSAnY2xvc2VkJyA/IChcbiAgICAgICAgICAgICAgICAgICAgICAgIDxGZWVkYmFja1N1cnZleVxuICAgICAgICAgICAgICAgICAgICAgICAgICBzdGF0ZT17bWVtb3J5U3VydmV5LnN0YXRlfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBsYXN0UmVzcG9uc2U9e21lbW9yeVN1cnZleS5sYXN0UmVzcG9uc2V9XG4gICAgICAgICAgICAgICAgICAgICAgICAgIGhhbmRsZVNlbGVjdD17bWVtb3J5U3VydmV5LmhhbmRsZVNlbGVjdH1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgaGFuZGxlVHJhbnNjcmlwdFNlbGVjdD17XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgbWVtb3J5U3VydmV5LmhhbmRsZVRyYW5zY3JpcHRTZWxlY3RcbiAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBpbnB1dFZhbHVlPXtpbnB1dFZhbHVlfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBzZXRJbnB1dFZhbHVlPXtzZXRJbnB1dFZhbHVlfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBvblJlcXVlc3RGZWVkYmFjaz17aGFuZGxlU3VydmV5UmVxdWVzdEZlZWRiYWNrfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBtZXNzYWdlPVwiSG93IHdlbGwgZGlkIENsYXVkZSB1c2UgaXRzIG1lbW9yeT8gKG9wdGlvbmFsKVwiXG4gICAgICAgICAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICAgICAgICAgICkgOiAoXG4gICAgICAgICAgICAgICAgICAgICAgICA8RmVlZGJhY2tTdXJ2ZXlcbiAgICAgICAgICAgICAgICAgICAgICAgICAgc3RhdGU9e2ZlZWRiYWNrU3VydmV5LnN0YXRlfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBsYXN0UmVzcG9uc2U9e2ZlZWRiYWNrU3VydmV5Lmxhc3RSZXNwb25zZX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgaGFuZGxlU2VsZWN0PXtmZWVkYmFja1N1cnZleS5oYW5kbGVTZWxlY3R9XG4gICAgICAgICAgICAgICAgICAgICAgICAgIGhhbmRsZVRyYW5zY3JpcHRTZWxlY3Q9e1xuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZlZWRiYWNrU3VydmV5LmhhbmRsZVRyYW5zY3JpcHRTZWxlY3RcbiAgICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBpbnB1dFZhbHVlPXtpbnB1dFZhbHVlfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBzZXRJbnB1dFZhbHVlPXtzZXRJbnB1dFZhbHVlfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBvblJlcXVlc3RGZWVkYmFjaz17XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgZGlkQXV0b1J1bklzc3VlUmVmLmN1cnJlbnRcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgID8gdW5kZWZpbmVkXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICA6IGhhbmRsZVN1cnZleVJlcXVlc3RGZWVkYmFja1xuICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAgICAgICAgey8qIEZydXN0cmF0aW9uLXRyaWdnZXJlZCB0cmFuc2NyaXB0IHNoYXJpbmcgcHJvbXB0ICovfVxuICAgICAgICAgICAgICAgICAgICAgIHtmcnVzdHJhdGlvbkRldGVjdGlvbi5zdGF0ZSAhPT0gJ2Nsb3NlZCcgJiYgKFxuICAgICAgICAgICAgICAgICAgICAgICAgPEZlZWRiYWNrU3VydmV5XG4gICAgICAgICAgICAgICAgICAgICAgICAgIHN0YXRlPXtmcnVzdHJhdGlvbkRldGVjdGlvbi5zdGF0ZX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgbGFzdFJlc3BvbnNlPXtudWxsfVxuICAgICAgICAgICAgICAgICAgICAgICAgICBoYW5kbGVTZWxlY3Q9eygpID0+IHt9fVxuICAgICAgICAgICAgICAgICAgICAgICAgICBoYW5kbGVUcmFuc2NyaXB0U2VsZWN0PXtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBmcnVzdHJhdGlvbkRldGVjdGlvbi5oYW5kbGVUcmFuc2NyaXB0U2VsZWN0XG4gICAgICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgaW5wdXRWYWx1ZT17aW5wdXRWYWx1ZX1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgc2V0SW5wdXRWYWx1ZT17c2V0SW5wdXRWYWx1ZX1cbiAgICAgICAgICAgICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICAgICAgICAgICAgKX1cbiAgICAgICAgICAgICAgICAgICAgICB7LyogU2tpbGwgaW1wcm92ZW1lbnQgc3VydmV5IC0gYXBwZWFycyB3aGVuIGltcHJvdmVtZW50cyBkZXRlY3RlZCAoYW50LW9ubHkpICovfVxuICAgICAgICAgICAgICAgICAgICAgIHtcImV4dGVybmFsXCIgPT09ICdhbnQnICYmXG4gICAgICAgICAgICAgICAgICAgICAgICBza2lsbEltcHJvdmVtZW50U3VydmV5LnN1Z2dlc3Rpb24gJiYgKFxuICAgICAgICAgICAgICAgICAgICAgICAgICA8U2tpbGxJbXByb3ZlbWVudFN1cnZleVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGlzT3Blbj17c2tpbGxJbXByb3ZlbWVudFN1cnZleS5pc09wZW59XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgc2tpbGxOYW1lPXtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNraWxsSW1wcm92ZW1lbnRTdXJ2ZXkuc3VnZ2VzdGlvbi5za2lsbE5hbWVcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgdXBkYXRlcz17c2tpbGxJbXByb3ZlbWVudFN1cnZleS5zdWdnZXN0aW9uLnVwZGF0ZXN9XG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgaGFuZGxlU2VsZWN0PXtza2lsbEltcHJvdmVtZW50U3VydmV5LmhhbmRsZVNlbGVjdH1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBpbnB1dFZhbHVlPXtpbnB1dFZhbHVlfVxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNldElucHV0VmFsdWU9e3NldElucHV0VmFsdWV9XG4gICAgICAgICAgICAgICAgICAgICAgICAgIC8+XG4gICAgICAgICAgICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICAgICAgICAgIHtzaG93SXNzdWVGbGFnQmFubmVyICYmIDxJc3N1ZUZsYWdCYW5uZXIgLz59XG4gICAgICAgICAgICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgICAgICA8UHJvbXB0SW5wdXRcbiAgICAgICAgICAgICAgICAgICAgICAgIGRlYnVnPXtkZWJ1Z31cbiAgICAgICAgICAgICAgICAgICAgICAgIGlkZVNlbGVjdGlvbj17aWRlU2VsZWN0aW9ufVxuICAgICAgICAgICAgICAgICAgICAgICAgaGFzU3VwcHJlc3NlZERpYWxvZ3M9eyEhaGFzU3VwcHJlc3NlZERpYWxvZ3N9XG4gICAgICAgICAgICAgICAgICAgICAgICBpc0xvY2FsSlNYQ29tbWFuZEFjdGl2ZT17aXNTaG93aW5nTG9jYWxKU1hDb21tYW5kfVxuICAgICAgICAgICAgICAgICAgICAgICAgZ2V0VG9vbFVzZUNvbnRleHQ9e2dldFRvb2xVc2VDb250ZXh0fVxuICAgICAgICAgICAgICAgICAgICAgICAgdG9vbFBlcm1pc3Npb25Db250ZXh0PXt0b29sUGVybWlzc2lvbkNvbnRleHR9XG4gICAgICAgICAgICAgICAgICAgICAgICBzZXRUb29sUGVybWlzc2lvbkNvbnRleHQ9e3NldFRvb2xQZXJtaXNzaW9uQ29udGV4dH1cbiAgICAgICAgICAgICAgICAgICAgICAgIGFwaUtleVN0YXR1cz17YXBpS2V5U3RhdHVzfVxuICAgICAgICAgICAgICAgICAgICAgICAgY29tbWFuZHM9e2NvbW1hbmRzfVxuICAgICAgICAgICAgICAgICAgICAgICAgYWdlbnRzPXthZ2VudERlZmluaXRpb25zLmFjdGl2ZUFnZW50c31cbiAgICAgICAgICAgICAgICAgICAgICAgIGlzTG9hZGluZz17aXNMb2FkaW5nfVxuICAgICAgICAgICAgICAgICAgICAgICAgb25FeGl0PXtoYW5kbGVFeGl0fVxuICAgICAgICAgICAgICAgICAgICAgICAgdmVyYm9zZT17dmVyYm9zZX1cbiAgICAgICAgICAgICAgICAgICAgICAgIG1lc3NhZ2VzPXttZXNzYWdlc31cbiAgICAgICAgICAgICAgICAgICAgICAgIG9uQXV0b1VwZGF0ZXJSZXN1bHQ9e3NldEF1dG9VcGRhdGVyUmVzdWx0fVxuICAgICAgICAgICAgICAgICAgICAgICAgYXV0b1VwZGF0ZXJSZXN1bHQ9e2F1dG9VcGRhdGVyUmVzdWx0fVxuICAgICAgICAgICAgICAgICAgICAgICAgaW5wdXQ9e2lucHV0VmFsdWV9XG4gICAgICAgICAgICAgICAgICAgICAgICBvbklucHV0Q2hhbmdlPXtzZXRJbnB1dFZhbHVlfVxuICAgICAgICAgICAgICAgICAgICAgICAgbW9kZT17aW5wdXRNb2RlfVxuICAgICAgICAgICAgICAgICAgICAgICAgb25Nb2RlQ2hhbmdlPXtzZXRJbnB1dE1vZGV9XG4gICAgICAgICAgICAgICAgICAgICAgICBzdGFzaGVkUHJvbXB0PXtzdGFzaGVkUHJvbXB0fVxuICAgICAgICAgICAgICAgICAgICAgICAgc2V0U3Rhc2hlZFByb21wdD17c2V0U3Rhc2hlZFByb21wdH1cbiAgICAgICAgICAgICAgICAgICAgICAgIHN1Ym1pdENvdW50PXtzdWJtaXRDb3VudH1cbiAgICAgICAgICAgICAgICAgICAgICAgIG9uU2hvd01lc3NhZ2VTZWxlY3Rvcj17aGFuZGxlU2hvd01lc3NhZ2VTZWxlY3Rvcn1cbiAgICAgICAgICAgICAgICAgICAgICAgIG9uTWVzc2FnZUFjdGlvbnNFbnRlcj17XG4gICAgICAgICAgICAgICAgICAgICAgICAgIC8vIFdvcmtzIGR1cmluZyBpc0xvYWRpbmcg4oCUIGVkaXQgY2FuY2VscyBmaXJzdDsgdXVpZCBzZWxlY3Rpb24gc3Vydml2ZXMgYXBwZW5kcy5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgZmVhdHVyZSgnTUVTU0FHRV9BQ1RJT05TJykgJiZcbiAgICAgICAgICAgICAgICAgICAgICAgICAgaXNGdWxsc2NyZWVuRW52RW5hYmxlZCgpICYmXG4gICAgICAgICAgICAgICAgICAgICAgICAgICFkaXNhYmxlTWVzc2FnZUFjdGlvbnNcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA/IGVudGVyTWVzc2FnZUFjdGlvbnNcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICA6IHVuZGVmaW5lZFxuICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgbWNwQ2xpZW50cz17bWNwQ2xpZW50c31cbiAgICAgICAgICAgICAgICAgICAgICAgIHBhc3RlZENvbnRlbnRzPXtwYXN0ZWRDb250ZW50c31cbiAgICAgICAgICAgICAgICAgICAgICAgIHNldFBhc3RlZENvbnRlbnRzPXtzZXRQYXN0ZWRDb250ZW50c31cbiAgICAgICAgICAgICAgICAgICAgICAgIHZpbU1vZGU9e3ZpbU1vZGV9XG4gICAgICAgICAgICAgICAgICAgICAgICBzZXRWaW1Nb2RlPXtzZXRWaW1Nb2RlfVxuICAgICAgICAgICAgICAgICAgICAgICAgc2hvd0Jhc2hlc0RpYWxvZz17c2hvd0Jhc2hlc0RpYWxvZ31cbiAgICAgICAgICAgICAgICAgICAgICAgIHNldFNob3dCYXNoZXNEaWFsb2c9e3NldFNob3dCYXNoZXNEaWFsb2d9XG4gICAgICAgICAgICAgICAgICAgICAgICBvblN1Ym1pdD17b25TdWJtaXR9XG4gICAgICAgICAgICAgICAgICAgICAgICBvbkFnZW50U3VibWl0PXtvbkFnZW50U3VibWl0fVxuICAgICAgICAgICAgICAgICAgICAgICAgaXNTZWFyY2hpbmdIaXN0b3J5PXtpc1NlYXJjaGluZ0hpc3Rvcnl9XG4gICAgICAgICAgICAgICAgICAgICAgICBzZXRJc1NlYXJjaGluZ0hpc3Rvcnk9e3NldElzU2VhcmNoaW5nSGlzdG9yeX1cbiAgICAgICAgICAgICAgICAgICAgICAgIGhlbHBPcGVuPXtpc0hlbHBPcGVufVxuICAgICAgICAgICAgICAgICAgICAgICAgc2V0SGVscE9wZW49e3NldElzSGVscE9wZW59XG4gICAgICAgICAgICAgICAgICAgICAgICBpbnNlcnRUZXh0UmVmPXtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgZmVhdHVyZSgnVk9JQ0VfTU9ERScpID8gaW5zZXJ0VGV4dFJlZiA6IHVuZGVmaW5lZFxuICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgICAgdm9pY2VJbnRlcmltUmFuZ2U9e3ZvaWNlLmludGVyaW1SYW5nZX1cbiAgICAgICAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICAgICAgICAgIDxTZXNzaW9uQmFja2dyb3VuZEhpbnRcbiAgICAgICAgICAgICAgICAgICAgICAgIG9uQmFja2dyb3VuZFNlc3Npb249e2hhbmRsZUJhY2tncm91bmRTZXNzaW9ufVxuICAgICAgICAgICAgICAgICAgICAgICAgaXNMb2FkaW5nPXtpc0xvYWRpbmd9XG4gICAgICAgICAgICAgICAgICAgICAgLz5cbiAgICAgICAgICAgICAgICAgICAgPC8+XG4gICAgICAgICAgICAgICAgICApfVxuICAgICAgICAgICAgICAgIHtjdXJzb3IgJiYgKFxuICAgICAgICAgICAgICAgICAgLy8gaW5wdXRWYWx1ZSBpcyBSRVBMIHN0YXRlOyB0eXBlZCB0ZXh0IHN1cnZpdmVzIHRoZSByb3VuZC10cmlwLlxuICAgICAgICAgICAgICAgICAgPE1lc3NhZ2VBY3Rpb25zQmFyIGN1cnNvcj17Y3Vyc29yfSAvPlxuICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAge2ZvY3VzZWRJbnB1dERpYWxvZyA9PT0gJ21lc3NhZ2Utc2VsZWN0b3InICYmIChcbiAgICAgICAgICAgICAgICAgIDxNZXNzYWdlU2VsZWN0b3JcbiAgICAgICAgICAgICAgICAgICAgbWVzc2FnZXM9e21lc3NhZ2VzfVxuICAgICAgICAgICAgICAgICAgICBwcmVzZWxlY3RlZE1lc3NhZ2U9e21lc3NhZ2VTZWxlY3RvclByZXNlbGVjdH1cbiAgICAgICAgICAgICAgICAgICAgb25QcmVSZXN0b3JlPXtvbkNhbmNlbH1cbiAgICAgICAgICAgICAgICAgICAgb25SZXN0b3JlQ29kZT17YXN5bmMgKG1lc3NhZ2U6IFVzZXJNZXNzYWdlKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgYXdhaXQgZmlsZUhpc3RvcnlSZXdpbmQoXG4gICAgICAgICAgICAgICAgICAgICAgICAoXG4gICAgICAgICAgICAgICAgICAgICAgICAgIHVwZGF0ZXI6IChwcmV2OiBGaWxlSGlzdG9yeVN0YXRlKSA9PiBGaWxlSGlzdG9yeVN0YXRlLFxuICAgICAgICAgICAgICAgICAgICAgICAgKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4gKHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAuLi5wcmV2LFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZpbGVIaXN0b3J5OiB1cGRhdGVyKHByZXYuZmlsZUhpc3RvcnkpLFxuICAgICAgICAgICAgICAgICAgICAgICAgICB9KSlcbiAgICAgICAgICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAgICAgICAgICBtZXNzYWdlLnV1aWQsXG4gICAgICAgICAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgICAgICAgICBvblN1bW1hcml6ZT17YXN5bmMgKFxuICAgICAgICAgICAgICAgICAgICAgIG1lc3NhZ2U6IFVzZXJNZXNzYWdlLFxuICAgICAgICAgICAgICAgICAgICAgIGZlZWRiYWNrPzogc3RyaW5nLFxuICAgICAgICAgICAgICAgICAgICAgIGRpcmVjdGlvbjogUGFydGlhbENvbXBhY3REaXJlY3Rpb24gPSAnZnJvbScsXG4gICAgICAgICAgICAgICAgICAgICkgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgIC8vIFByb2plY3Qgc25pcHBlZCBtZXNzYWdlcyBzbyB0aGUgY29tcGFjdCBtb2RlbFxuICAgICAgICAgICAgICAgICAgICAgIC8vIGRvZXNuJ3Qgc3VtbWFyaXplIGNvbnRlbnQgdGhhdCB3YXMgaW50ZW50aW9uYWxseSByZW1vdmVkLlxuICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbXBhY3RNZXNzYWdlcyA9XG4gICAgICAgICAgICAgICAgICAgICAgICBnZXRNZXNzYWdlc0FmdGVyQ29tcGFjdEJvdW5kYXJ5KG1lc3NhZ2VzKVxuXG4gICAgICAgICAgICAgICAgICAgICAgY29uc3QgbWVzc2FnZUluZGV4ID0gY29tcGFjdE1lc3NhZ2VzLmluZGV4T2YobWVzc2FnZSlcbiAgICAgICAgICAgICAgICAgICAgICBpZiAobWVzc2FnZUluZGV4ID09PSAtMSkge1xuICAgICAgICAgICAgICAgICAgICAgICAgLy8gU2VsZWN0ZWQgYSBzbmlwcGVkIG9yIHByZS1jb21wYWN0IG1lc3NhZ2UgdGhhdCB0aGVcbiAgICAgICAgICAgICAgICAgICAgICAgIC8vIHNlbGVjdG9yIHN0aWxsIHNob3dzIChSRVBMIGtlZXBzIGZ1bGwgaGlzdG9yeSBmb3JcbiAgICAgICAgICAgICAgICAgICAgICAgIC8vIHNjcm9sbGJhY2spLiBTdXJmYWNlIHdoeSBub3RoaW5nIGhhcHBlbmVkIGluc3RlYWRcbiAgICAgICAgICAgICAgICAgICAgICAgIC8vIG9mIHNpbGVudGx5IG5vLW9waW5nLlxuICAgICAgICAgICAgICAgICAgICAgICAgc2V0TWVzc2FnZXMocHJldiA9PiBbXG4gICAgICAgICAgICAgICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGNyZWF0ZVN5c3RlbU1lc3NhZ2UoXG4gICAgICAgICAgICAgICAgICAgICAgICAgICAgJ1RoYXQgbWVzc2FnZSBpcyBubyBsb25nZXIgaW4gdGhlIGFjdGl2ZSBjb250ZXh0IChzbmlwcGVkIG9yIHByZS1jb21wYWN0KS4gQ2hvb3NlIGEgbW9yZSByZWNlbnQgbWVzc2FnZS4nLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgICd3YXJuaW5nJyxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgICAgICAgICAgICAgIF0pXG4gICAgICAgICAgICAgICAgICAgICAgICByZXR1cm5cbiAgICAgICAgICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBuZXdBYm9ydENvbnRyb2xsZXIgPSBjcmVhdGVBYm9ydENvbnRyb2xsZXIoKVxuICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGNvbnRleHQgPSBnZXRUb29sVXNlQ29udGV4dChcbiAgICAgICAgICAgICAgICAgICAgICAgIGNvbXBhY3RNZXNzYWdlcyxcbiAgICAgICAgICAgICAgICAgICAgICAgIFtdLFxuICAgICAgICAgICAgICAgICAgICAgICAgbmV3QWJvcnRDb250cm9sbGVyLFxuICAgICAgICAgICAgICAgICAgICAgICAgbWFpbkxvb3BNb2RlbCxcbiAgICAgICAgICAgICAgICAgICAgICApXG5cbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBhcHBTdGF0ZSA9IGNvbnRleHQuZ2V0QXBwU3RhdGUoKVxuICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IGRlZmF1bHRTeXNQcm9tcHQgPSBhd2FpdCBnZXRTeXN0ZW1Qcm9tcHQoXG4gICAgICAgICAgICAgICAgICAgICAgICBjb250ZXh0Lm9wdGlvbnMudG9vbHMsXG4gICAgICAgICAgICAgICAgICAgICAgICBjb250ZXh0Lm9wdGlvbnMubWFpbkxvb3BNb2RlbCxcbiAgICAgICAgICAgICAgICAgICAgICAgIEFycmF5LmZyb20oXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGFwcFN0YXRlLnRvb2xQZXJtaXNzaW9uQ29udGV4dC5hZGRpdGlvbmFsV29ya2luZ0RpcmVjdG9yaWVzLmtleXMoKSxcbiAgICAgICAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgICAgICAgICBjb250ZXh0Lm9wdGlvbnMubWNwQ2xpZW50cyxcbiAgICAgICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgICAgICAgY29uc3Qgc3lzdGVtUHJvbXB0ID0gYnVpbGRFZmZlY3RpdmVTeXN0ZW1Qcm9tcHQoe1xuICAgICAgICAgICAgICAgICAgICAgICAgbWFpblRocmVhZEFnZW50RGVmaW5pdGlvbjogdW5kZWZpbmVkLFxuICAgICAgICAgICAgICAgICAgICAgICAgdG9vbFVzZUNvbnRleHQ6IGNvbnRleHQsXG4gICAgICAgICAgICAgICAgICAgICAgICBjdXN0b21TeXN0ZW1Qcm9tcHQ6IGNvbnRleHQub3B0aW9ucy5jdXN0b21TeXN0ZW1Qcm9tcHQsXG4gICAgICAgICAgICAgICAgICAgICAgICBkZWZhdWx0U3lzdGVtUHJvbXB0OiBkZWZhdWx0U3lzUHJvbXB0LFxuICAgICAgICAgICAgICAgICAgICAgICAgYXBwZW5kU3lzdGVtUHJvbXB0OiBjb250ZXh0Lm9wdGlvbnMuYXBwZW5kU3lzdGVtUHJvbXB0LFxuICAgICAgICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgICAgICAgY29uc3QgW3VzZXJDb250ZXh0LCBzeXN0ZW1Db250ZXh0XSA9IGF3YWl0IFByb21pc2UuYWxsKFtcbiAgICAgICAgICAgICAgICAgICAgICAgIGdldFVzZXJDb250ZXh0KCksXG4gICAgICAgICAgICAgICAgICAgICAgICBnZXRTeXN0ZW1Db250ZXh0KCksXG4gICAgICAgICAgICAgICAgICAgICAgXSlcblxuICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHBhcnRpYWxDb21wYWN0Q29udmVyc2F0aW9uKFxuICAgICAgICAgICAgICAgICAgICAgICAgY29tcGFjdE1lc3NhZ2VzLFxuICAgICAgICAgICAgICAgICAgICAgICAgbWVzc2FnZUluZGV4LFxuICAgICAgICAgICAgICAgICAgICAgICAgY29udGV4dCxcbiAgICAgICAgICAgICAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgc3lzdGVtUHJvbXB0LFxuICAgICAgICAgICAgICAgICAgICAgICAgICB1c2VyQ29udGV4dCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgc3lzdGVtQ29udGV4dCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgdG9vbFVzZUNvbnRleHQ6IGNvbnRleHQsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGZvcmtDb250ZXh0TWVzc2FnZXM6IGNvbXBhY3RNZXNzYWdlcyxcbiAgICAgICAgICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAgICAgICAgICBmZWVkYmFjayxcbiAgICAgICAgICAgICAgICAgICAgICAgIGRpcmVjdGlvbixcbiAgICAgICAgICAgICAgICAgICAgICApXG5cbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBrZXB0ID0gcmVzdWx0Lm1lc3NhZ2VzVG9LZWVwID8/IFtdXG4gICAgICAgICAgICAgICAgICAgICAgY29uc3Qgb3JkZXJlZCA9XG4gICAgICAgICAgICAgICAgICAgICAgICBkaXJlY3Rpb24gPT09ICd1cF90bydcbiAgICAgICAgICAgICAgICAgICAgICAgICAgPyBbLi4ucmVzdWx0LnN1bW1hcnlNZXNzYWdlcywgLi4ua2VwdF1cbiAgICAgICAgICAgICAgICAgICAgICAgICAgOiBbLi4ua2VwdCwgLi4ucmVzdWx0LnN1bW1hcnlNZXNzYWdlc11cbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBwb3N0Q29tcGFjdCA9IFtcbiAgICAgICAgICAgICAgICAgICAgICAgIHJlc3VsdC5ib3VuZGFyeU1hcmtlcixcbiAgICAgICAgICAgICAgICAgICAgICAgIC4uLm9yZGVyZWQsXG4gICAgICAgICAgICAgICAgICAgICAgICAuLi5yZXN1bHQuYXR0YWNobWVudHMsXG4gICAgICAgICAgICAgICAgICAgICAgICAuLi5yZXN1bHQuaG9va1Jlc3VsdHMsXG4gICAgICAgICAgICAgICAgICAgICAgXVxuICAgICAgICAgICAgICAgICAgICAgIC8vIEZ1bGxzY3JlZW4gJ2Zyb20nIGtlZXBzIHNjcm9sbGJhY2s7ICd1cF90bycgbXVzdCBub3RcbiAgICAgICAgICAgICAgICAgICAgICAvLyAob2xkWzBdIHVuY2hhbmdlZCArIGdyb3duIGFycmF5IG1lYW5zIGluY3JlbWVudGFsXG4gICAgICAgICAgICAgICAgICAgICAgLy8gdXNlTG9nTWVzc2FnZXMgcGF0aCwgc28gYm91bmRhcnkgbmV2ZXIgcGVyc2lzdGVkKS5cbiAgICAgICAgICAgICAgICAgICAgICAvLyBGaW5kIGJ5IHV1aWQgc2luY2Ugb2xkIGlzIHJhdyBSRVBMIGhpc3RvcnkgYW5kIHNuaXBwZWRcbiAgICAgICAgICAgICAgICAgICAgICAvLyBlbnRyaWVzIGNhbiBzaGlmdCB0aGUgcHJvamVjdGVkIG1lc3NhZ2VJbmRleC5cbiAgICAgICAgICAgICAgICAgICAgICBpZiAoaXNGdWxsc2NyZWVuRW52RW5hYmxlZCgpICYmIGRpcmVjdGlvbiA9PT0gJ2Zyb20nKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBzZXRNZXNzYWdlcyhvbGQgPT4ge1xuICAgICAgICAgICAgICAgICAgICAgICAgICBjb25zdCByYXdJZHggPSBvbGQuZmluZEluZGV4KFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIG0gPT4gbS51dWlkID09PSBtZXNzYWdlLnV1aWQsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIClcbiAgICAgICAgICAgICAgICAgICAgICAgICAgcmV0dXJuIFtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAuLi5vbGQuc2xpY2UoMCwgcmF3SWR4ID09PSAtMSA/IDAgOiByYXdJZHgpLFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIC4uLnBvc3RDb21wYWN0LFxuICAgICAgICAgICAgICAgICAgICAgICAgICBdXG4gICAgICAgICAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgICAgICAgICAgICBzZXRNZXNzYWdlcyhwb3N0Q29tcGFjdClcbiAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgLy8gUGFydGlhbCBjb21wYWN0IGJ5cGFzc2VzIGhhbmRsZU1lc3NhZ2VGcm9tU3RyZWFtIOKAlCBjbGVhclxuICAgICAgICAgICAgICAgICAgICAgIC8vIHRoZSBjb250ZXh0LWJsb2NrZWQgZmxhZyBzbyBwcm9hY3RpdmUgdGlja3MgcmVzdW1lLlxuICAgICAgICAgICAgICAgICAgICAgIGlmIChmZWF0dXJlKCdQUk9BQ1RJVkUnKSB8fCBmZWF0dXJlKCdLQUlST1MnKSkge1xuICAgICAgICAgICAgICAgICAgICAgICAgcHJvYWN0aXZlTW9kdWxlPy5zZXRDb250ZXh0QmxvY2tlZChmYWxzZSlcbiAgICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICAgICAgc2V0Q29udmVyc2F0aW9uSWQocmFuZG9tVVVJRCgpKVxuICAgICAgICAgICAgICAgICAgICAgIHJ1blBvc3RDb21wYWN0Q2xlYW51cChjb250ZXh0Lm9wdGlvbnMucXVlcnlTb3VyY2UpXG5cbiAgICAgICAgICAgICAgICAgICAgICBpZiAoZGlyZWN0aW9uID09PSAnZnJvbScpIHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGNvbnN0IHIgPSB0ZXh0Rm9yUmVzdWJtaXQobWVzc2FnZSlcbiAgICAgICAgICAgICAgICAgICAgICAgIGlmIChyKSB7XG4gICAgICAgICAgICAgICAgICAgICAgICAgIHNldElucHV0VmFsdWUoci50ZXh0KVxuICAgICAgICAgICAgICAgICAgICAgICAgICBzZXRJbnB1dE1vZGUoci5tb2RlKVxuICAgICAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICAgICAgICAgIC8vIFNob3cgbm90aWZpY2F0aW9uIHdpdGggY3RybCtvIGhpbnRcbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBoaXN0b3J5U2hvcnRjdXQgPSBnZXRTaG9ydGN1dERpc3BsYXkoXG4gICAgICAgICAgICAgICAgICAgICAgICAnYXBwOnRvZ2dsZVRyYW5zY3JpcHQnLFxuICAgICAgICAgICAgICAgICAgICAgICAgJ0dsb2JhbCcsXG4gICAgICAgICAgICAgICAgICAgICAgICAnY3RybCtvJyxcbiAgICAgICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgICAgICAgYWRkTm90aWZpY2F0aW9uKHtcbiAgICAgICAgICAgICAgICAgICAgICAgIGtleTogJ3N1bW1hcml6ZS1jdHJsLW8taGludCcsXG4gICAgICAgICAgICAgICAgICAgICAgICB0ZXh0OiBgQ29udmVyc2F0aW9uIHN1bW1hcml6ZWQgKCR7aGlzdG9yeVNob3J0Y3V0fSBmb3IgaGlzdG9yeSlgLFxuICAgICAgICAgICAgICAgICAgICAgICAgcHJpb3JpdHk6ICdtZWRpdW0nLFxuICAgICAgICAgICAgICAgICAgICAgICAgdGltZW91dE1zOiA4MDAwLFxuICAgICAgICAgICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAgIG9uUmVzdG9yZU1lc3NhZ2U9e2hhbmRsZVJlc3RvcmVNZXNzYWdlfVxuICAgICAgICAgICAgICAgICAgICBvbkNsb3NlPXsoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgICAgc2V0SXNNZXNzYWdlU2VsZWN0b3JWaXNpYmxlKGZhbHNlKVxuICAgICAgICAgICAgICAgICAgICAgIHNldE1lc3NhZ2VTZWxlY3RvclByZXNlbGVjdCh1bmRlZmluZWQpXG4gICAgICAgICAgICAgICAgICAgIH19XG4gICAgICAgICAgICAgICAgICAvPlxuICAgICAgICAgICAgICAgICl9XG4gICAgICAgICAgICAgICAge1wiZXh0ZXJuYWxcIiA9PT0gJ2FudCcgJiYgPERldkJhciAvPn1cbiAgICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgICAgIHtmZWF0dXJlKCdCVUREWScpICYmXG4gICAgICAgICAgICAgICEoY29tcGFuaW9uTmFycm93ICYmIGlzRnVsbHNjcmVlbkVudkVuYWJsZWQoKSkgJiZcbiAgICAgICAgICAgICAgY29tcGFuaW9uVmlzaWJsZSA/IChcbiAgICAgICAgICAgICAgICA8Q29tcGFuaW9uU3ByaXRlIC8+XG4gICAgICAgICAgICAgICkgOiBudWxsfVxuICAgICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgfVxuICAgICAgICAvPlxuICAgICAgPC9NQ1BDb25uZWN0aW9uTWFuYWdlcj5cbiAgICA8L0tleWJpbmRpbmdTZXR1cD5cbiAgKVxuICBpZiAoaXNGdWxsc2NyZWVuRW52RW5hYmxlZCgpKSB7XG4gICAgcmV0dXJuIChcbiAgICAgIDxBbHRlcm5hdGVTY3JlZW4gbW91c2VUcmFja2luZz17aXNNb3VzZVRyYWNraW5nRW5hYmxlZCgpfT5cbiAgICAgICAge21haW5SZXR1cm59XG4gICAgICA8L0FsdGVybmF0ZVNjcmVlbj5cbiAgICApXG4gIH1cbiAgcmV0dXJuIG1haW5SZXR1cm5cbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBO0FBQ0EsU0FBU0EsT0FBTyxRQUFRLFlBQVk7QUFDcEMsU0FBU0MsU0FBUyxRQUFRLGVBQWU7QUFDekMsU0FDRUMsMkJBQTJCLEVBQzNCQyx5QkFBeUIsRUFDekJDLG1CQUFtQixFQUNuQkMsMEJBQTBCLEVBQzFCQyxtQkFBbUIsUUFDZCx1QkFBdUI7QUFDOUIsU0FBU0MsZ0JBQWdCLFFBQVEseUJBQXlCO0FBQzFELFNBQVNDLEtBQUssUUFBUSxtQkFBbUI7QUFDekMs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