/ hooks / useReplBridge.tsx
useReplBridge.tsx
  1  import { feature } from 'bun:bundle';
  2  import React, { useCallback, useEffect, useRef } from 'react';
  3  import { setMainLoopModelOverride } from '../bootstrap/state.js';
  4  import { type BridgePermissionCallbacks, type BridgePermissionResponse, isBridgePermissionResponse } from '../bridge/bridgePermissionCallbacks.js';
  5  import { buildBridgeConnectUrl } from '../bridge/bridgeStatusUtil.js';
  6  import { extractInboundMessageFields } from '../bridge/inboundMessages.js';
  7  import type { BridgeState, ReplBridgeHandle } from '../bridge/replBridge.js';
  8  import { setReplBridgeHandle } from '../bridge/replBridgeHandle.js';
  9  import type { Command } from '../commands.js';
 10  import { getSlashCommandToolSkills, isBridgeSafeCommand } from '../commands.js';
 11  import { getRemoteSessionUrl } from '../constants/product.js';
 12  import { useNotifications } from '../context/notifications.js';
 13  import type { PermissionMode, SDKMessage } from '../entrypoints/agentSdkTypes.js';
 14  import type { SDKControlResponse } from '../entrypoints/sdk/controlTypes.js';
 15  import { Text } from '../ink.js';
 16  import { getFeatureValue_CACHED_MAY_BE_STALE } from '../services/analytics/growthbook.js';
 17  import { useAppState, useAppStateStore, useSetAppState } from '../state/AppState.js';
 18  import type { Message } from '../types/message.js';
 19  import { getCwd } from '../utils/cwd.js';
 20  import { logForDebugging } from '../utils/debug.js';
 21  import { errorMessage } from '../utils/errors.js';
 22  import { enqueue } from '../utils/messageQueueManager.js';
 23  import { buildSystemInitMessage } from '../utils/messages/systemInit.js';
 24  import { createBridgeStatusMessage, createSystemMessage } from '../utils/messages.js';
 25  import { getAutoModeUnavailableNotification, getAutoModeUnavailableReason, isAutoModeGateEnabled, isBypassPermissionsModeDisabled, transitionPermissionMode } from '../utils/permissions/permissionSetup.js';
 26  import { getLeaderToolUseConfirmQueue } from '../utils/swarm/leaderPermissionBridge.js';
 27  
 28  /** How long after a failure before replBridgeEnabled is auto-cleared (stops retries). */
 29  export const BRIDGE_FAILURE_DISMISS_MS = 10_000;
 30  
 31  /**
 32   * Max consecutive initReplBridge failures before the hook stops re-attempting
 33   * for the session lifetime. Guards against paths that flip replBridgeEnabled
 34   * back on after auto-disable (settings sync, /remote-control, config tool)
 35   * when the underlying OAuth is unrecoverable — each re-attempt is another
 36   * guaranteed 401 against POST /v1/environments/bridge. Datadog 2026-03-08:
 37   * top stuck client generated 2,879 × 401/day alone (17% of all 401s on the
 38   * route).
 39   */
 40  const MAX_CONSECUTIVE_INIT_FAILURES = 3;
 41  
 42  /**
 43   * Hook that initializes an always-on bridge connection in the background
 44   * and writes new user/assistant messages to the bridge session.
 45   *
 46   * Silently skips if bridge is not enabled or user is not OAuth-authenticated.
 47   *
 48   * Watches AppState.replBridgeEnabled — when toggled off (via /config or footer),
 49   * the bridge is torn down. When toggled back on, it re-initializes.
 50   *
 51   * Inbound messages from claude.ai are injected into the REPL via queuedCommands.
 52   */
 53  export function useReplBridge(messages: Message[], setMessages: (action: React.SetStateAction<Message[]>) => void, abortControllerRef: React.RefObject<AbortController | null>, commands: readonly Command[], mainLoopModel: string): {
 54    sendBridgeResult: () => void;
 55  } {
 56    const handleRef = useRef<ReplBridgeHandle | null>(null);
 57    const teardownPromiseRef = useRef<Promise<void> | undefined>(undefined);
 58    const lastWrittenIndexRef = useRef(0);
 59    // Tracks UUIDs already flushed as initial messages. Persists across
 60    // bridge reconnections so Bridge #2+ only sends new messages — sending
 61    // duplicate UUIDs causes the server to kill the WebSocket.
 62    const flushedUUIDsRef = useRef(new Set<string>());
 63    const failureTimeoutRef = useRef<ReturnType<typeof setTimeout> | undefined>(undefined);
 64    // Persists across effect re-runs (unlike the effect's local state). Reset
 65    // only on successful init. Hits MAX_CONSECUTIVE_INIT_FAILURES → fuse blown
 66    // for the session, regardless of replBridgeEnabled re-toggling.
 67    const consecutiveFailuresRef = useRef(0);
 68    const setAppState = useSetAppState();
 69    const commandsRef = useRef(commands);
 70    commandsRef.current = commands;
 71    const mainLoopModelRef = useRef(mainLoopModel);
 72    mainLoopModelRef.current = mainLoopModel;
 73    const messagesRef = useRef(messages);
 74    messagesRef.current = messages;
 75    const store = useAppStateStore();
 76    const {
 77      addNotification
 78    } = useNotifications();
 79    const replBridgeEnabled = feature('BRIDGE_MODE') ?
 80    // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
 81    useAppState(s => s.replBridgeEnabled) : false;
 82    const replBridgeConnected = feature('BRIDGE_MODE') ?
 83    // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
 84    useAppState(s_0 => s_0.replBridgeConnected) : false;
 85    const replBridgeOutboundOnly = feature('BRIDGE_MODE') ?
 86    // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
 87    useAppState(s_1 => s_1.replBridgeOutboundOnly) : false;
 88    const replBridgeInitialName = feature('BRIDGE_MODE') ?
 89    // biome-ignore lint/correctness/useHookAtTopLevel: feature() is a compile-time constant
 90    useAppState(s_2 => s_2.replBridgeInitialName) : undefined;
 91  
 92    // Initialize/teardown bridge when enabled state changes.
 93    // Passes current messages as initialMessages so the remote session
 94    // starts with the existing conversation context (e.g. from /bridge).
 95    useEffect(() => {
 96      // feature() check must use positive pattern for dead code elimination —
 97      // negative pattern (if (!feature(...)) return) does NOT eliminate
 98      // dynamic imports below.
 99      if (feature('BRIDGE_MODE')) {
100        if (!replBridgeEnabled) return;
101        const outboundOnly = replBridgeOutboundOnly;
102        function notifyBridgeFailed(detail?: string): void {
103          if (outboundOnly) return;
104          addNotification({
105            key: 'bridge-failed',
106            jsx: <>
107                <Text color="error">Remote Control failed</Text>
108                {detail && <Text dimColor> · {detail}</Text>}
109              </>,
110            priority: 'immediate'
111          });
112        }
113        if (consecutiveFailuresRef.current >= MAX_CONSECUTIVE_INIT_FAILURES) {
114          logForDebugging(`[bridge:repl] Hook: ${consecutiveFailuresRef.current} consecutive init failures, not retrying this session`);
115          // Clear replBridgeEnabled so /remote-control doesn't mistakenly show
116          // BridgeDisconnectDialog for a bridge that never connected.
117          const fuseHint = 'disabled after repeated failures · restart to retry';
118          notifyBridgeFailed(fuseHint);
119          setAppState(prev => {
120            if (prev.replBridgeError === fuseHint && !prev.replBridgeEnabled) return prev;
121            return {
122              ...prev,
123              replBridgeError: fuseHint,
124              replBridgeEnabled: false
125            };
126          });
127          return;
128        }
129        let cancelled = false;
130        // Capture messages.length now so we don't re-send initial messages
131        // through writeMessages after the bridge connects.
132        const initialMessageCount = messages.length;
133        void (async () => {
134          try {
135            // Wait for any in-progress teardown to complete before registering
136            // a new environment. Without this, the deregister HTTP call from
137            // the previous teardown races with the new register call, and the
138            // server may tear down the freshly-created environment.
139            if (teardownPromiseRef.current) {
140              logForDebugging('[bridge:repl] Hook: waiting for previous teardown to complete before re-init');
141              await teardownPromiseRef.current;
142              teardownPromiseRef.current = undefined;
143              logForDebugging('[bridge:repl] Hook: previous teardown complete, proceeding with re-init');
144            }
145            if (cancelled) return;
146  
147            // Dynamic import so the module is tree-shaken in external builds
148            const {
149              initReplBridge
150            } = await import('../bridge/initReplBridge.js');
151            const {
152              shouldShowAppUpgradeMessage
153            } = await import('../bridge/envLessBridgeConfig.js');
154  
155            // Assistant mode: perpetual bridge session — claude.ai shows one
156            // continuous conversation across CLI restarts instead of a new
157            // session per invocation. initBridgeCore reads bridge-pointer.json
158            // (the same crash-recovery file #20735 added) and reuses its
159            // {environmentId, sessionId} via reuseEnvironmentId +
160            // api.reconnectSession(). Teardown skips archive/deregister/
161            // pointer-clear so the session survives clean exits, not just
162            // crashes. Non-assistant bridges clear the pointer on teardown
163            // (crash-recovery only).
164            let perpetual = false;
165            if (feature('KAIROS')) {
166              const {
167                isAssistantMode
168              } = await import('../assistant/index.js');
169              perpetual = isAssistantMode();
170            }
171  
172            // When a user message arrives from claude.ai, inject it into the REPL.
173            // Preserves the original UUID so that when the message is forwarded
174            // back to CCR, it matches the original — avoiding duplicate messages.
175            //
176            // Async because file_attachments (if present) need a network fetch +
177            // disk write before we enqueue with the @path prefix. Caller doesn't
178            // await — messages with attachments just land in the queue slightly
179            // later, which is fine (web messages aren't rapid-fire).
180            async function handleInboundMessage(msg: SDKMessage): Promise<void> {
181              try {
182                const fields = extractInboundMessageFields(msg);
183                if (!fields) return;
184                const {
185                  uuid
186                } = fields;
187  
188                // Dynamic import keeps the bridge code out of non-BRIDGE_MODE builds.
189                const {
190                  resolveAndPrepend
191                } = await import('../bridge/inboundAttachments.js');
192                let sanitized = fields.content;
193                if (feature('KAIROS_GITHUB_WEBHOOKS')) {
194                  /* eslint-disable @typescript-eslint/no-require-imports */
195                  const {
196                    sanitizeInboundWebhookContent
197                  } = require('../bridge/webhookSanitizer.js') as typeof import('../bridge/webhookSanitizer.js');
198                  /* eslint-enable @typescript-eslint/no-require-imports */
199                  sanitized = sanitizeInboundWebhookContent(fields.content);
200                }
201                const content = await resolveAndPrepend(msg, sanitized);
202                const preview = typeof content === 'string' ? content.slice(0, 80) : `[${content.length} content blocks]`;
203                logForDebugging(`[bridge:repl] Injecting inbound user message: ${preview}${uuid ? ` uuid=${uuid}` : ''}`);
204                enqueue({
205                  value: content,
206                  mode: 'prompt' as const,
207                  uuid,
208                  // skipSlashCommands stays true as defense-in-depth —
209                  // processUserInputBase overrides it internally when bridgeOrigin
210                  // is set AND the resolved command passes isBridgeSafeCommand.
211                  // This keeps exit-word suppression and immediate-command blocks
212                  // intact for any code path that checks skipSlashCommands directly.
213                  skipSlashCommands: true,
214                  bridgeOrigin: true
215                });
216              } catch (e) {
217                logForDebugging(`[bridge:repl] handleInboundMessage failed: ${e}`, {
218                  level: 'error'
219                });
220              }
221            }
222  
223            // State change callback — maps bridge lifecycle events to AppState.
224            function handleStateChange(state: BridgeState, detail_0?: string): void {
225              if (cancelled) return;
226              if (outboundOnly) {
227                logForDebugging(`[bridge:repl] Mirror state=${state}${detail_0 ? ` detail=${detail_0}` : ''}`);
228                // Sync replBridgeConnected so the forwarding effect starts/stops
229                // writing as the transport comes up or dies.
230                if (state === 'failed') {
231                  setAppState(prev_3 => {
232                    if (!prev_3.replBridgeConnected) return prev_3;
233                    return {
234                      ...prev_3,
235                      replBridgeConnected: false
236                    };
237                  });
238                } else if (state === 'ready' || state === 'connected') {
239                  setAppState(prev_4 => {
240                    if (prev_4.replBridgeConnected) return prev_4;
241                    return {
242                      ...prev_4,
243                      replBridgeConnected: true
244                    };
245                  });
246                }
247                return;
248              }
249              const handle = handleRef.current;
250              switch (state) {
251                case 'ready':
252                  setAppState(prev_9 => {
253                    const connectUrl = handle && handle.environmentId !== '' ? buildBridgeConnectUrl(handle.environmentId, handle.sessionIngressUrl) : prev_9.replBridgeConnectUrl;
254                    const sessionUrl = handle ? getRemoteSessionUrl(handle.bridgeSessionId, handle.sessionIngressUrl) : prev_9.replBridgeSessionUrl;
255                    const envId = handle?.environmentId;
256                    const sessionId = handle?.bridgeSessionId;
257                    if (prev_9.replBridgeConnected && !prev_9.replBridgeSessionActive && !prev_9.replBridgeReconnecting && prev_9.replBridgeConnectUrl === connectUrl && prev_9.replBridgeSessionUrl === sessionUrl && prev_9.replBridgeEnvironmentId === envId && prev_9.replBridgeSessionId === sessionId) {
258                      return prev_9;
259                    }
260                    return {
261                      ...prev_9,
262                      replBridgeConnected: true,
263                      replBridgeSessionActive: false,
264                      replBridgeReconnecting: false,
265                      replBridgeConnectUrl: connectUrl,
266                      replBridgeSessionUrl: sessionUrl,
267                      replBridgeEnvironmentId: envId,
268                      replBridgeSessionId: sessionId,
269                      replBridgeError: undefined
270                    };
271                  });
272                  break;
273                case 'connected':
274                  {
275                    setAppState(prev_8 => {
276                      if (prev_8.replBridgeSessionActive) return prev_8;
277                      return {
278                        ...prev_8,
279                        replBridgeConnected: true,
280                        replBridgeSessionActive: true,
281                        replBridgeReconnecting: false,
282                        replBridgeError: undefined
283                      };
284                    });
285                    // Send system/init so remote clients (web/iOS/Android) get
286                    // session metadata. REPL uses query() directly — never hits
287                    // QueryEngine's SDKMessage layer — so this is the only path
288                    // to put system/init on the REPL-bridge wire. Skills load is
289                    // async (memoized, cheap after REPL startup); fire-and-forget
290                    // so the connected-state transition isn't blocked.
291                    if (getFeatureValue_CACHED_MAY_BE_STALE('tengu_bridge_system_init', false)) {
292                      void (async () => {
293                        try {
294                          const skills = await getSlashCommandToolSkills(getCwd());
295                          if (cancelled) return;
296                          const state_0 = store.getState();
297                          handleRef.current?.writeSdkMessages([buildSystemInitMessage({
298                            // tools/mcpClients/plugins redacted for REPL-bridge:
299                            // MCP-prefixed tool names and server names leak which
300                            // integrations the user has wired up; plugin paths leak
301                            // raw filesystem paths (username, project structure).
302                            // CCR v2 persists SDK messages to Spanner — users who
303                            // tap "Connect from phone" may not expect these on
304                            // Anthropic's servers. QueryEngine (SDK) still emits
305                            // full lists — SDK consumers expect full telemetry.
306                            tools: [],
307                            mcpClients: [],
308                            model: mainLoopModelRef.current,
309                            permissionMode: state_0.toolPermissionContext.mode as PermissionMode,
310                            // TODO: avoid the cast
311                            // Remote clients can only invoke bridge-safe commands —
312                            // advertising unsafe ones (local-jsx, unallowed local)
313                            // would let mobile/web attempt them and hit errors.
314                            commands: commandsRef.current.filter(isBridgeSafeCommand),
315                            agents: state_0.agentDefinitions.activeAgents,
316                            skills,
317                            plugins: [],
318                            fastMode: state_0.fastMode
319                          })]);
320                        } catch (err_0) {
321                          logForDebugging(`[bridge:repl] Failed to send system/init: ${errorMessage(err_0)}`, {
322                            level: 'error'
323                          });
324                        }
325                      })();
326                    }
327                    break;
328                  }
329                case 'reconnecting':
330                  setAppState(prev_7 => {
331                    if (prev_7.replBridgeReconnecting) return prev_7;
332                    return {
333                      ...prev_7,
334                      replBridgeReconnecting: true,
335                      replBridgeSessionActive: false
336                    };
337                  });
338                  break;
339                case 'failed':
340                  // Clear any previous failure dismiss timer
341                  clearTimeout(failureTimeoutRef.current);
342                  notifyBridgeFailed(detail_0);
343                  setAppState(prev_5 => ({
344                    ...prev_5,
345                    replBridgeError: detail_0,
346                    replBridgeReconnecting: false,
347                    replBridgeSessionActive: false,
348                    replBridgeConnected: false
349                  }));
350                  // Auto-disable after timeout so the hook stops retrying.
351                  failureTimeoutRef.current = setTimeout(() => {
352                    if (cancelled) return;
353                    failureTimeoutRef.current = undefined;
354                    setAppState(prev_6 => {
355                      if (!prev_6.replBridgeError) return prev_6;
356                      return {
357                        ...prev_6,
358                        replBridgeEnabled: false,
359                        replBridgeError: undefined
360                      };
361                    });
362                  }, BRIDGE_FAILURE_DISMISS_MS);
363                  break;
364              }
365            }
366  
367            // Map of pending bridge permission response handlers, keyed by request_id.
368            // Each entry is an onResponse handler waiting for CCR to reply.
369            const pendingPermissionHandlers = new Map<string, (response: BridgePermissionResponse) => void>();
370  
371            // Dispatch incoming control_response messages to registered handlers
372            function handlePermissionResponse(msg_0: SDKControlResponse): void {
373              const requestId = msg_0.response?.request_id;
374              if (!requestId) return;
375              const handler = pendingPermissionHandlers.get(requestId);
376              if (!handler) {
377                logForDebugging(`[bridge:repl] No handler for control_response request_id=${requestId}`);
378                return;
379              }
380              pendingPermissionHandlers.delete(requestId);
381              // Extract the permission decision from the control_response payload
382              const inner = msg_0.response;
383              if (inner.subtype === 'success' && inner.response && isBridgePermissionResponse(inner.response)) {
384                handler(inner.response);
385              }
386            }
387            const handle_0 = await initReplBridge({
388              outboundOnly,
389              tags: outboundOnly ? ['ccr-mirror'] : undefined,
390              onInboundMessage: handleInboundMessage,
391              onPermissionResponse: handlePermissionResponse,
392              onInterrupt() {
393                abortControllerRef.current?.abort();
394              },
395              onSetModel(model) {
396                const resolved = model === 'default' ? null : model ?? null;
397                setMainLoopModelOverride(resolved);
398                setAppState(prev_10 => {
399                  if (prev_10.mainLoopModelForSession === resolved) return prev_10;
400                  return {
401                    ...prev_10,
402                    mainLoopModelForSession: resolved
403                  };
404                });
405              },
406              onSetMaxThinkingTokens(maxTokens) {
407                const enabled = maxTokens !== null;
408                setAppState(prev_11 => {
409                  if (prev_11.thinkingEnabled === enabled) return prev_11;
410                  return {
411                    ...prev_11,
412                    thinkingEnabled: enabled
413                  };
414                });
415              },
416              onSetPermissionMode(mode) {
417                // Policy guards MUST fire before transitionPermissionMode —
418                // its internal auto-gate check is a defensive throw (with a
419                // setAutoModeActive(true) side-effect BEFORE the throw) rather
420                // than a graceful reject. Letting that throw escape would:
421                // (1) leave STATE.autoModeActive=true while the mode is
422                //     unchanged (3-way invariant violation per src/CLAUDE.md)
423                // (2) fail to send a control_response → server kills WS
424                // These mirror print.ts handleSetPermissionMode; the bridge
425                // can't import the checks directly (bootstrap-isolation), so
426                // it relies on this verdict to emit the error response.
427                if (mode === 'bypassPermissions') {
428                  if (isBypassPermissionsModeDisabled()) {
429                    return {
430                      ok: false,
431                      error: 'Cannot set permission mode to bypassPermissions because it is disabled by settings or configuration'
432                    };
433                  }
434                  if (!store.getState().toolPermissionContext.isBypassPermissionsModeAvailable) {
435                    return {
436                      ok: false,
437                      error: 'Cannot set permission mode to bypassPermissions because the session was not launched with --dangerously-skip-permissions'
438                    };
439                  }
440                }
441                if (feature('TRANSCRIPT_CLASSIFIER') && mode === 'auto' && !isAutoModeGateEnabled()) {
442                  const reason = getAutoModeUnavailableReason();
443                  return {
444                    ok: false,
445                    error: reason ? `Cannot set permission mode to auto: ${getAutoModeUnavailableNotification(reason)}` : 'Cannot set permission mode to auto'
446                  };
447                }
448                // Guards passed — apply via the centralized transition so
449                // prePlanMode stashing and auto-mode state sync all fire.
450                setAppState(prev_12 => {
451                  const current = prev_12.toolPermissionContext.mode;
452                  if (current === mode) return prev_12;
453                  const next = transitionPermissionMode(current, mode, prev_12.toolPermissionContext);
454                  return {
455                    ...prev_12,
456                    toolPermissionContext: {
457                      ...next,
458                      mode
459                    }
460                  };
461                });
462                // Recheck queued permission prompts now that mode changed.
463                setImmediate(() => {
464                  getLeaderToolUseConfirmQueue()?.(currentQueue => {
465                    currentQueue.forEach(item => {
466                      void item.recheckPermission();
467                    });
468                    return currentQueue;
469                  });
470                });
471                return {
472                  ok: true
473                };
474              },
475              onStateChange: handleStateChange,
476              initialMessages: messages.length > 0 ? messages : undefined,
477              getMessages: () => messagesRef.current,
478              previouslyFlushedUUIDs: flushedUUIDsRef.current,
479              initialName: replBridgeInitialName,
480              perpetual
481            });
482            if (cancelled) {
483              // Effect was cancelled while initReplBridge was in flight.
484              // Tear down the handle to avoid leaking resources (poll loop,
485              // WebSocket, registered environment, cleanup callback).
486              logForDebugging(`[bridge:repl] Hook: init cancelled during flight, tearing down${handle_0 ? ` env=${handle_0.environmentId}` : ''}`);
487              if (handle_0) {
488                void handle_0.teardown();
489              }
490              return;
491            }
492            if (!handle_0) {
493              // initReplBridge returned null — a precondition failed. For most
494              // cases (no_oauth, policy_denied, etc.) onStateChange('failed')
495              // already fired with a specific hint. The GrowthBook-gate-off case
496              // is intentionally silent — not a failure, just not rolled out.
497              consecutiveFailuresRef.current++;
498              logForDebugging(`[bridge:repl] Init returned null (precondition or session creation failed); consecutive failures: ${consecutiveFailuresRef.current}`);
499              clearTimeout(failureTimeoutRef.current);
500              setAppState(prev_13 => ({
501                ...prev_13,
502                replBridgeError: prev_13.replBridgeError ?? 'check debug logs for details'
503              }));
504              failureTimeoutRef.current = setTimeout(() => {
505                if (cancelled) return;
506                failureTimeoutRef.current = undefined;
507                setAppState(prev_14 => {
508                  if (!prev_14.replBridgeError) return prev_14;
509                  return {
510                    ...prev_14,
511                    replBridgeEnabled: false,
512                    replBridgeError: undefined
513                  };
514                });
515              }, BRIDGE_FAILURE_DISMISS_MS);
516              return;
517            }
518            handleRef.current = handle_0;
519            setReplBridgeHandle(handle_0);
520            consecutiveFailuresRef.current = 0;
521            // Skip initial messages in the forwarding effect — they were
522            // already loaded as session events during creation.
523            lastWrittenIndexRef.current = initialMessageCount;
524            if (outboundOnly) {
525              setAppState(prev_15 => {
526                if (prev_15.replBridgeConnected && prev_15.replBridgeSessionId === handle_0.bridgeSessionId) return prev_15;
527                return {
528                  ...prev_15,
529                  replBridgeConnected: true,
530                  replBridgeSessionId: handle_0.bridgeSessionId,
531                  replBridgeSessionUrl: undefined,
532                  replBridgeConnectUrl: undefined,
533                  replBridgeError: undefined
534                };
535              });
536              logForDebugging(`[bridge:repl] Mirror initialized, session=${handle_0.bridgeSessionId}`);
537            } else {
538              // Build bridge permission callbacks so the interactive permission
539              // handler can race bridge responses against local user interaction.
540              const permissionCallbacks: BridgePermissionCallbacks = {
541                sendRequest(requestId_0, toolName, input, toolUseId, description, permissionSuggestions, blockedPath) {
542                  handle_0.sendControlRequest({
543                    type: 'control_request',
544                    request_id: requestId_0,
545                    request: {
546                      subtype: 'can_use_tool',
547                      tool_name: toolName,
548                      input,
549                      tool_use_id: toolUseId,
550                      description,
551                      ...(permissionSuggestions ? {
552                        permission_suggestions: permissionSuggestions
553                      } : {}),
554                      ...(blockedPath ? {
555                        blocked_path: blockedPath
556                      } : {})
557                    }
558                  });
559                },
560                sendResponse(requestId_1, response) {
561                  const payload: Record<string, unknown> = {
562                    ...response
563                  };
564                  handle_0.sendControlResponse({
565                    type: 'control_response',
566                    response: {
567                      subtype: 'success',
568                      request_id: requestId_1,
569                      response: payload
570                    }
571                  });
572                },
573                cancelRequest(requestId_2) {
574                  handle_0.sendControlCancelRequest(requestId_2);
575                },
576                onResponse(requestId_3, handler_0) {
577                  pendingPermissionHandlers.set(requestId_3, handler_0);
578                  return () => {
579                    pendingPermissionHandlers.delete(requestId_3);
580                  };
581                }
582              };
583              setAppState(prev_16 => ({
584                ...prev_16,
585                replBridgePermissionCallbacks: permissionCallbacks
586              }));
587              const url = getRemoteSessionUrl(handle_0.bridgeSessionId, handle_0.sessionIngressUrl);
588              // environmentId === '' signals the v2 env-less path. buildBridgeConnectUrl
589              // builds an env-specific connect URL, which doesn't exist without an env.
590              const hasEnv = handle_0.environmentId !== '';
591              const connectUrl_0 = hasEnv ? buildBridgeConnectUrl(handle_0.environmentId, handle_0.sessionIngressUrl) : undefined;
592              setAppState(prev_17 => {
593                if (prev_17.replBridgeConnected && prev_17.replBridgeSessionUrl === url) {
594                  return prev_17;
595                }
596                return {
597                  ...prev_17,
598                  replBridgeConnected: true,
599                  replBridgeSessionUrl: url,
600                  replBridgeConnectUrl: connectUrl_0 ?? prev_17.replBridgeConnectUrl,
601                  replBridgeEnvironmentId: handle_0.environmentId,
602                  replBridgeSessionId: handle_0.bridgeSessionId,
603                  replBridgeError: undefined
604                };
605              });
606  
607              // Show bridge status with URL in the transcript. perpetual (KAIROS
608              // assistant mode) falls back to v1 at initReplBridge.ts — skip the
609              // v2-only upgrade nudge for them. Own try/catch so a cosmetic
610              // GrowthBook hiccup doesn't hit the outer init-failure handler.
611              const upgradeNudge = !perpetual ? await shouldShowAppUpgradeMessage().catch(() => false) : false;
612              if (cancelled) return;
613              setMessages(prev_18 => [...prev_18, createBridgeStatusMessage(url, upgradeNudge ? 'Please upgrade to the latest version of the Claude mobile app to see your Remote Control sessions.' : undefined)]);
614              logForDebugging(`[bridge:repl] Hook initialized, session=${handle_0.bridgeSessionId}`);
615            }
616          } catch (err) {
617            // Never crash the REPL — surface the error in the UI.
618            // Check cancelled first (symmetry with the !handle path at line ~386):
619            // if initReplBridge threw during rapid toggle-off (in-flight network
620            // error), don't count that toward the fuse or spam a stale error
621            // into the UI. Also fixes pre-existing spurious setAppState/
622            // setMessages on cancelled throws.
623            if (cancelled) return;
624            consecutiveFailuresRef.current++;
625            const errMsg = errorMessage(err);
626            logForDebugging(`[bridge:repl] Init failed: ${errMsg}; consecutive failures: ${consecutiveFailuresRef.current}`);
627            clearTimeout(failureTimeoutRef.current);
628            notifyBridgeFailed(errMsg);
629            setAppState(prev_0 => ({
630              ...prev_0,
631              replBridgeError: errMsg
632            }));
633            failureTimeoutRef.current = setTimeout(() => {
634              if (cancelled) return;
635              failureTimeoutRef.current = undefined;
636              setAppState(prev_1 => {
637                if (!prev_1.replBridgeError) return prev_1;
638                return {
639                  ...prev_1,
640                  replBridgeEnabled: false,
641                  replBridgeError: undefined
642                };
643              });
644            }, BRIDGE_FAILURE_DISMISS_MS);
645            if (!outboundOnly) {
646              setMessages(prev_2 => [...prev_2, createSystemMessage(`Remote Control failed to connect: ${errMsg}`, 'warning')]);
647            }
648          }
649        })();
650        return () => {
651          cancelled = true;
652          clearTimeout(failureTimeoutRef.current);
653          failureTimeoutRef.current = undefined;
654          if (handleRef.current) {
655            logForDebugging(`[bridge:repl] Hook cleanup: starting teardown for env=${handleRef.current.environmentId} session=${handleRef.current.bridgeSessionId}`);
656            teardownPromiseRef.current = handleRef.current.teardown();
657            handleRef.current = null;
658            setReplBridgeHandle(null);
659          }
660          setAppState(prev_19 => {
661            if (!prev_19.replBridgeConnected && !prev_19.replBridgeSessionActive && !prev_19.replBridgeError) {
662              return prev_19;
663            }
664            return {
665              ...prev_19,
666              replBridgeConnected: false,
667              replBridgeSessionActive: false,
668              replBridgeReconnecting: false,
669              replBridgeConnectUrl: undefined,
670              replBridgeSessionUrl: undefined,
671              replBridgeEnvironmentId: undefined,
672              replBridgeSessionId: undefined,
673              replBridgeError: undefined,
674              replBridgePermissionCallbacks: undefined
675            };
676          });
677          lastWrittenIndexRef.current = 0;
678        };
679      }
680    }, [replBridgeEnabled, replBridgeOutboundOnly, setAppState, setMessages, addNotification]);
681  
682    // Write new messages as they appear.
683    // Also re-runs when replBridgeConnected changes (bridge finishes init),
684    // so any messages that arrived before the bridge was ready get written.
685    useEffect(() => {
686      // Positive feature() guard — see first useEffect comment
687      if (feature('BRIDGE_MODE')) {
688        if (!replBridgeConnected) return;
689        const handle_1 = handleRef.current;
690        if (!handle_1) return;
691  
692        // Clamp the index in case messages were compacted (array shortened).
693        // After compaction the ref could exceed messages.length, and without
694        // clamping no new messages would be forwarded.
695        if (lastWrittenIndexRef.current > messages.length) {
696          logForDebugging(`[bridge:repl] Compaction detected: lastWrittenIndex=${lastWrittenIndexRef.current} > messages.length=${messages.length}, clamping`);
697        }
698        const startIndex = Math.min(lastWrittenIndexRef.current, messages.length);
699  
700        // Collect new messages since last write
701        const newMessages: Message[] = [];
702        for (let i = startIndex; i < messages.length; i++) {
703          const msg_1 = messages[i];
704          if (msg_1 && (msg_1.type === 'user' || msg_1.type === 'assistant' || msg_1.type === 'system' && msg_1.subtype === 'local_command')) {
705            newMessages.push(msg_1);
706          }
707        }
708        lastWrittenIndexRef.current = messages.length;
709        if (newMessages.length > 0) {
710          handle_1.writeMessages(newMessages);
711        }
712      }
713    }, [messages, replBridgeConnected]);
714    const sendBridgeResult = useCallback(() => {
715      if (feature('BRIDGE_MODE')) {
716        handleRef.current?.sendResult();
717      }
718    }, []);
719    return {
720      sendBridgeResult
721    };
722  }
723  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmZWF0dXJlIiwiUmVhY3QiLCJ1c2VDYWxsYmFjayIsInVzZUVmZmVjdCIsInVzZVJlZiIsInNldE1haW5Mb29wTW9kZWxPdmVycmlkZSIsIkJyaWRnZVBlcm1pc3Npb25DYWxsYmFja3MiLCJCcmlkZ2VQZXJtaXNzaW9uUmVzcG9uc2UiLCJpc0JyaWRnZVBlcm1pc3Npb25SZXNwb25zZSIsImJ1aWxkQnJpZGdlQ29ubmVjdFVybCIsImV4dHJhY3RJbmJvdW5kTWVzc2FnZUZpZWxkcyIsIkJyaWRnZVN0YXRlIiwiUmVwbEJyaWRnZUhhbmRsZSIsInNldFJlcGxCcmlkZ2VIYW5kbGUiLCJDb21tYW5kIiwiZ2V0U2xhc2hDb21tYW5kVG9vbFNraWxscyIsImlzQnJpZGdlU2FmZUNvbW1hbmQiLCJnZXRSZW1vdGVTZXNzaW9uVXJsIiwidXNlTm90aWZpY2F0aW9ucyIsIlBlcm1pc3Npb25Nb2RlIiwiU0RLTWVzc2FnZSIsIlNES0NvbnRyb2xSZXNwb25zZSIsIlRleHQiLCJnZXRGZWF0dXJlVmFsdWVfQ0FDSEVEX01BWV9CRV9TVEFMRSIsInVzZUFwcFN0YXRlIiwidXNlQXBwU3RhdGVTdG9yZSIsInVzZVNldEFwcFN0YXRlIiwiTWVzc2FnZSIsImdldEN3ZCIsImxvZ0ZvckRlYnVnZ2luZyIsImVycm9yTWVzc2FnZSIsImVucXVldWUiLCJidWlsZFN5c3RlbUluaXRNZXNzYWdlIiwiY3JlYXRlQnJpZGdlU3RhdHVzTWVzc2FnZSIsImNyZWF0ZVN5c3RlbU1lc3NhZ2UiLCJnZXRBdXRvTW9kZVVuYXZhaWxhYmxlTm90aWZpY2F0aW9uIiwiZ2V0QXV0b01vZGVVbmF2YWlsYWJsZVJlYXNvbiIsImlzQXV0b01vZGVHYXRlRW5hYmxlZCIsImlzQnlwYXNzUGVybWlzc2lvbnNNb2RlRGlzYWJsZWQiLCJ0cmFuc2l0aW9uUGVybWlzc2lvbk1vZGUiLCJnZXRMZWFkZXJUb29sVXNlQ29uZmlybVF1ZXVlIiwiQlJJREdFX0ZBSUxVUkVfRElTTUlTU19NUyIsIk1BWF9DT05TRUNVVElWRV9JTklUX0ZBSUxVUkVTIiwidXNlUmVwbEJyaWRnZSIsIm1lc3NhZ2VzIiwic2V0TWVzc2FnZXMiLCJhY3Rpb24iLCJTZXRTdGF0ZUFjdGlvbiIsImFib3J0Q29udHJvbGxlclJlZiIsIlJlZk9iamVjdCIsIkFib3J0Q29udHJvbGxlciIsImNvbW1hbmRzIiwibWFpbkxvb3BNb2RlbCIsInNlbmRCcmlkZ2VSZXN1bHQiLCJoYW5kbGVSZWYiLCJ0ZWFyZG93blByb21pc2VSZWYiLCJQcm9taXNlIiwidW5kZWZpbmVkIiwibGFzdFdyaXR0ZW5JbmRleFJlZiIsImZsdXNoZWRVVUlEc1JlZiIsIlNldCIsImZhaWx1cmVUaW1lb3V0UmVmIiwiUmV0dXJuVHlwZSIsInNldFRpbWVvdXQiLCJjb25zZWN1dGl2ZUZhaWx1cmVzUmVmIiwic2V0QXBwU3RhdGUiLCJjb21tYW5kc1JlZiIsImN1cnJlbnQiLCJtYWluTG9vcE1vZGVsUmVmIiwibWVzc2FnZXNSZWYiLCJzdG9yZSIsImFkZE5vdGlmaWNhdGlvbiIsInJlcGxCcmlkZ2VFbmFibGVkIiwicyIsInJlcGxCcmlkZ2VDb25uZWN0ZWQiLCJyZXBsQnJpZGdlT3V0Ym91bmRPbmx5IiwicmVwbEJyaWRnZUluaXRpYWxOYW1lIiwib3V0Ym91bmRPbmx5Iiwibm90aWZ5QnJpZGdlRmFpbGVkIiwiZGV0YWlsIiwia2V5IiwianN4IiwicHJpb3JpdHkiLCJmdXNlSGludCIsInByZXYiLCJyZXBsQnJpZGdlRXJyb3IiLCJjYW5jZWxsZWQiLCJpbml0aWFsTWVzc2FnZUNvdW50IiwibGVuZ3RoIiwiaW5pdFJlcGxCcmlkZ2UiLCJzaG91bGRTaG93QXBwVXBncmFkZU1lc3NhZ2UiLCJwZXJwZXR1YWwiLCJpc0Fzc2lzdGFudE1vZGUiLCJoYW5kbGVJbmJvdW5kTWVzc2FnZSIsIm1zZyIsImZpZWxkcyIsInV1aWQiLCJyZXNvbHZlQW5kUHJlcGVuZCIsInNhbml0aXplZCIsImNvbnRlbnQiLCJzYW5pdGl6ZUluYm91bmRXZWJob29rQ29udGVudCIsInJlcXVpcmUiLCJwcmV2aWV3Iiwic2xpY2UiLCJ2YWx1ZSIsIm1vZGUiLCJjb25zdCIsInNraXBTbGFzaENvbW1hbmRzIiwiYnJpZGdlT3JpZ2luIiwiZSIsImxldmVsIiwiaGFuZGxlU3RhdGVDaGFuZ2UiLCJzdGF0ZSIsImhhbmRsZSIsImNvbm5lY3RVcmwiLCJlbnZpcm9ubWVudElkIiwic2Vzc2lvbkluZ3Jlc3NVcmwiLCJyZXBsQnJpZGdlQ29ubmVjdFVybCIsInNlc3Npb25VcmwiLCJicmlkZ2VTZXNzaW9uSWQiLCJyZXBsQnJpZGdlU2Vzc2lvblVybCIsImVudklkIiwic2Vzc2lvbklkIiwicmVwbEJyaWRnZVNlc3Npb25BY3RpdmUiLCJyZXBsQnJpZGdlUmVjb25uZWN0aW5nIiwicmVwbEJyaWRnZUVudmlyb25tZW50SWQiLCJyZXBsQnJpZGdlU2Vzc2lvbklkIiwic2tpbGxzIiwiZ2V0U3RhdGUiLCJ3cml0ZVNka01lc3NhZ2VzIiwidG9vbHMiLCJtY3BDbGllbnRzIiwibW9kZWwiLCJwZXJtaXNzaW9uTW9kZSIsInRvb2xQZXJtaXNzaW9uQ29udGV4dCIsImZpbHRlciIsImFnZW50cyIsImFnZW50RGVmaW5pdGlvbnMiLCJhY3RpdmVBZ2VudHMiLCJwbHVnaW5zIiwiZmFzdE1vZGUiLCJlcnIiLCJjbGVhclRpbWVvdXQiLCJwZW5kaW5nUGVybWlzc2lvbkhhbmRsZXJzIiwiTWFwIiwicmVzcG9uc2UiLCJoYW5kbGVQZXJtaXNzaW9uUmVzcG9uc2UiLCJyZXF1ZXN0SWQiLCJyZXF1ZXN0X2lkIiwiaGFuZGxlciIsImdldCIsImRlbGV0ZSIsImlubmVyIiwic3VidHlwZSIsInRhZ3MiLCJvbkluYm91bmRNZXNzYWdlIiwib25QZXJtaXNzaW9uUmVzcG9uc2UiLCJvbkludGVycnVwdCIsImFib3J0Iiwib25TZXRNb2RlbCIsInJlc29sdmVkIiwibWFpbkxvb3BNb2RlbEZvclNlc3Npb24iLCJvblNldE1heFRoaW5raW5nVG9rZW5zIiwibWF4VG9rZW5zIiwiZW5hYmxlZCIsInRoaW5raW5nRW5hYmxlZCIsIm9uU2V0UGVybWlzc2lvbk1vZGUiLCJvayIsImVycm9yIiwiaXNCeXBhc3NQZXJtaXNzaW9uc01vZGVBdmFpbGFibGUiLCJyZWFzb24iLCJuZXh0Iiwic2V0SW1tZWRpYXRlIiwiY3VycmVudFF1ZXVlIiwiZm9yRWFjaCIsIml0ZW0iLCJyZWNoZWNrUGVybWlzc2lvbiIsIm9uU3RhdGVDaGFuZ2UiLCJpbml0aWFsTWVzc2FnZXMiLCJnZXRNZXNzYWdlcyIsInByZXZpb3VzbHlGbHVzaGVkVVVJRHMiLCJpbml0aWFsTmFtZSIsInRlYXJkb3duIiwicGVybWlzc2lvbkNhbGxiYWNrcyIsInNlbmRSZXF1ZXN0IiwidG9vbE5hbWUiLCJpbnB1dCIsInRvb2xVc2VJZCIsImRlc2NyaXB0aW9uIiwicGVybWlzc2lvblN1Z2dlc3Rpb25zIiwiYmxvY2tlZFBhdGgiLCJzZW5kQ29udHJvbFJlcXVlc3QiLCJ0eXBlIiwicmVxdWVzdCIsInRvb2xfbmFtZSIsInRvb2xfdXNlX2lkIiwicGVybWlzc2lvbl9zdWdnZXN0aW9ucyIsImJsb2NrZWRfcGF0aCIsInNlbmRSZXNwb25zZSIsInBheWxvYWQiLCJSZWNvcmQiLCJzZW5kQ29udHJvbFJlc3BvbnNlIiwiY2FuY2VsUmVxdWVzdCIsInNlbmRDb250cm9sQ2FuY2VsUmVxdWVzdCIsIm9uUmVzcG9uc2UiLCJzZXQiLCJyZXBsQnJpZGdlUGVybWlzc2lvbkNhbGxiYWNrcyIsInVybCIsImhhc0VudiIsInVwZ3JhZGVOdWRnZSIsImNhdGNoIiwiZXJyTXNnIiwic3RhcnRJbmRleCIsIk1hdGgiLCJtaW4iLCJuZXdNZXNzYWdlcyIsImkiLCJwdXNoIiwid3JpdGVNZXNzYWdlcyIsInNlbmRSZXN1bHQiXSwic291cmNlcyI6WyJ1c2VSZXBsQnJpZGdlLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBmZWF0dXJlIH0gZnJvbSAnYnVuOmJ1bmRsZSdcbmltcG9ydCBSZWFjdCwgeyB1c2VDYWxsYmFjaywgdXNlRWZmZWN0LCB1c2VSZWYgfSBmcm9tICdyZWFjdCdcbmltcG9ydCB7IHNldE1haW5Mb29wTW9kZWxPdmVycmlkZSB9IGZyb20gJy4uL2Jvb3RzdHJhcC9zdGF0ZS5qcydcbmltcG9ydCB7XG4gIHR5cGUgQnJpZGdlUGVybWlzc2lvbkNhbGxiYWNrcyxcbiAgdHlwZSBCcmlkZ2VQZXJtaXNzaW9uUmVzcG9uc2UsXG4gIGlzQnJpZGdlUGVybWlzc2lvblJlc3BvbnNlLFxufSBmcm9tICcuLi9icmlkZ2UvYnJpZGdlUGVybWlzc2lvbkNhbGxiYWNrcy5qcydcbmltcG9ydCB7IGJ1aWxkQnJpZGdlQ29ubmVjdFVybCB9IGZyb20gJy4uL2JyaWRnZS9icmlkZ2VTdGF0dXNVdGlsLmpzJ1xuaW1wb3J0IHsgZXh0cmFjdEluYm91bmRNZXNzYWdlRmllbGRzIH0gZnJvbSAnLi4vYnJpZGdlL2luYm91bmRNZXNzYWdlcy5qcydcbmltcG9ydCB0eXBlIHsgQnJpZGdlU3RhdGUsIFJlcGxCcmlkZ2VIYW5kbGUgfSBmcm9tICcuLi9icmlkZ2UvcmVwbEJyaWRnZS5qcydcbmltcG9ydCB7IHNldFJlcGxCcmlkZ2VIYW5kbGUgfSBmcm9tICcuLi9icmlkZ2UvcmVwbEJyaWRnZUhhbmRsZS5qcydcbmltcG9ydCB0eXBlIHsgQ29tbWFuZCB9IGZyb20gJy4uL2NvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgZ2V0U2xhc2hDb21tYW5kVG9vbFNraWxscywgaXNCcmlkZ2VTYWZlQ29tbWFuZCB9IGZyb20gJy4uL2NvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgZ2V0UmVtb3RlU2Vzc2lvblVybCB9IGZyb20gJy4uL2NvbnN0YW50cy9wcm9kdWN0LmpzJ1xuaW1wb3J0IHsgdXNlTm90aWZpY2F0aW9ucyB9IGZyb20gJy4uL2NvbnRleHQvbm90aWZpY2F0aW9ucy5qcydcbmltcG9ydCB0eXBlIHtcbiAgUGVybWlzc2lvbk1vZGUsXG4gIFNES01lc3NhZ2UsXG59IGZyb20gJy4uL2VudHJ5cG9pbnRzL2FnZW50U2RrVHlwZXMuanMnXG5pbXBvcnQgdHlwZSB7IFNES0NvbnRyb2xSZXNwb25zZSB9IGZyb20gJy4uL2VudHJ5cG9pbnRzL3Nkay9jb250cm9sVHlwZXMuanMnXG5pbXBvcnQgeyBUZXh0IH0gZnJvbSAnLi4vaW5rLmpzJ1xuaW1wb3J0IHsgZ2V0RmVhdHVyZVZhbHVlX0NBQ0hFRF9NQVlfQkVfU1RBTEUgfSBmcm9tICcuLi9zZXJ2aWNlcy9hbmFseXRpY3MvZ3Jvd3RoYm9vay5qcydcbmltcG9ydCB7XG4gIHVzZUFwcFN0YXRlLFxuICB1c2VBcHBTdGF0ZVN0b3JlLFxuICB1c2VTZXRBcHBTdGF0ZSxcbn0gZnJvbSAnLi4vc3RhdGUvQXBwU3RhdGUuanMnXG5pbXBvcnQgdHlwZSB7IE1lc3NhZ2UgfSBmcm9tICcuLi90eXBlcy9tZXNzYWdlLmpzJ1xuaW1wb3J0IHsgZ2V0Q3dkIH0gZnJvbSAnLi4vdXRpbHMvY3dkLmpzJ1xuaW1wb3J0IHsgbG9nRm9yRGVidWdnaW5nIH0gZnJvbSAnLi4vdXRpbHMvZGVidWcuanMnXG5pbXBvcnQgeyBlcnJvck1lc3NhZ2UgfSBmcm9tICcuLi91dGlscy9lcnJvcnMuanMnXG5pbXBvcnQgeyBlbnF1ZXVlIH0gZnJvbSAnLi4vdXRpbHMvbWVzc2FnZVF1ZXVlTWFuYWdlci5qcydcbmltcG9ydCB7IGJ1aWxkU3lzdGVtSW5pdE1lc3NhZ2UgfSBmcm9tICcuLi91dGlscy9tZXNzYWdlcy9zeXN0ZW1Jbml0LmpzJ1xuaW1wb3J0IHtcbiAgY3JlYXRlQnJpZGdlU3RhdHVzTWVzc2FnZSxcbiAgY3JlYXRlU3lzdGVtTWVzc2FnZSxcbn0gZnJvbSAnLi4vdXRpbHMvbWVzc2FnZXMuanMnXG5pbXBvcnQge1xuICBnZXRBdXRvTW9kZVVuYXZhaWxhYmxlTm90aWZpY2F0aW9uLFxuICBnZXRBdXRvTW9kZVVuYXZhaWxhYmxlUmVhc29uLFxuICBpc0F1dG9Nb2RlR2F0ZUVuYWJsZWQsXG4gIGlzQnlwYXNzUGVybWlzc2lvbnNNb2RlRGlzYWJsZWQsXG4gIHRyYW5zaXRpb25QZXJtaXNzaW9uTW9kZSxcbn0gZnJvbSAnLi4vdXRpbHMvcGVybWlzc2lvbnMvcGVybWlzc2lvblNldHVwLmpzJ1xuaW1wb3J0IHsgZ2V0TGVhZGVyVG9vbFVzZUNvbmZpcm1RdWV1ZSB9IGZyb20gJy4uL3V0aWxzL3N3YXJtL2xlYWRlclBlcm1pc3Npb25CcmlkZ2UuanMnXG5cbi8qKiBIb3cgbG9uZyBhZnRlciBhIGZhaWx1cmUgYmVmb3JlIHJlcGxCcmlkZ2VFbmFibGVkIGlzIGF1dG8tY2xlYXJlZCAoc3RvcHMgcmV0cmllcykuICovXG5leHBvcnQgY29uc3QgQlJJREdFX0ZBSUxVUkVfRElTTUlTU19NUyA9IDEwXzAwMFxuXG4vKipcbiAqIE1heCBjb25zZWN1dGl2ZSBpbml0UmVwbEJyaWRnZSBmYWlsdXJlcyBiZWZvcmUgdGhlIGhvb2sgc3RvcHMgcmUtYXR0ZW1wdGluZ1xuICogZm9yIHRoZSBzZXNzaW9uIGxpZmV0aW1lLiBHdWFyZHMgYWdhaW5zdCBwYXRocyB0aGF0IGZsaXAgcmVwbEJyaWRnZUVuYWJsZWRcbiAqIGJhY2sgb24gYWZ0ZXIgYXV0by1kaXNhYmxlIChzZXR0aW5ncyBzeW5jLCAvcmVtb3RlLWNvbnRyb2wsIGNvbmZpZyB0b29sKVxuICogd2hlbiB0aGUgdW5kZXJseWluZyBPQXV0aCBpcyB1bnJlY292ZXJhYmxlIOKAlCBlYWNoIHJlLWF0dGVtcHQgaXMgYW5vdGhlclxuICogZ3VhcmFudGVlZCA0MDEgYWdhaW5zdCBQT1NUIC92MS9lbnZpcm9ubWVudHMvYnJpZGdlLiBEYXRhZG9nIDIwMjYtMDMtMDg6XG4gKiB0b3Agc3R1Y2sgY2xpZW50IGdlbmVyYXRlZCAyLDg3OSDDlyA0MDEvZGF5IGFsb25lICgxNyUgb2YgYWxsIDQwMXMgb24gdGhlXG4gKiByb3V0ZSkuXG4gKi9cbmNvbnN0IE1BWF9DT05TRUNVVElWRV9JTklUX0ZBSUxVUkVTID0gM1xuXG4vKipcbiAqIEhvb2sgdGhhdCBpbml0aWFsaXplcyBhbiBhbHdheXMtb24gYnJpZGdlIGNvbm5lY3Rpb24gaW4gdGhlIGJhY2tncm91bmRcbiAqIGFuZCB3cml0ZXMgbmV3IHVzZXIvYXNzaXN0YW50IG1lc3NhZ2VzIHRvIHRoZSBicmlkZ2Ugc2Vzc2lvbi5cbiAqXG4gKiBTaWxlbnRseSBza2lwcyBpZiBicmlkZ2UgaXMgbm90IGVuYWJsZWQgb3IgdXNlciBpcyBub3QgT0F1dGgtYXV0aGVudGljYXRlZC5cbiAqXG4gKiBXYXRjaGVzIEFwcFN0YXRlLnJlcGxCcmlkZ2VFbmFibGVkIOKAlCB3aGVuIHRvZ2dsZWQgb2ZmICh2aWEgL2NvbmZpZyBvciBmb290ZXIpLFxuICogdGhlIGJyaWRnZSBpcyB0b3JuIGRvd24uIFdoZW4gdG9nZ2xlZCBiYWNrIG9uLCBpdCByZS1pbml0aWFsaXplcy5cbiAqXG4gKiBJbmJvdW5kIG1lc3NhZ2VzIGZyb20gY2xhdWRlLmFpIGFyZSBpbmplY3RlZCBpbnRvIHRoZSBSRVBMIHZpYSBxdWV1ZWRDb21tYW5kcy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHVzZVJlcGxCcmlkZ2UoXG4gIG1lc3NhZ2VzOiBNZXNzYWdlW10sXG4gIHNldE1lc3NhZ2VzOiAoYWN0aW9uOiBSZWFjdC5TZXRTdGF0ZUFjdGlvbjxNZXNzYWdlW10+KSA9PiB2b2lkLFxuICBhYm9ydENvbnRyb2xsZXJSZWY6IFJlYWN0LlJlZk9iamVjdDxBYm9ydENvbnRyb2xsZXIgfCBudWxsPixcbiAgY29tbWFuZHM6IHJlYWRvbmx5IENvbW1hbmRbXSxcbiAgbWFpbkxvb3BNb2RlbDogc3RyaW5nLFxuKTogeyBzZW5kQnJpZGdlUmVzdWx0OiAoKSA9PiB2b2lkIH0ge1xuICBjb25zdCBoYW5kbGVSZWYgPSB1c2VSZWY8UmVwbEJyaWRnZUhhbmRsZSB8IG51bGw+KG51bGwpXG4gIGNvbnN0IHRlYXJkb3duUHJvbWlzZVJlZiA9IHVzZVJlZjxQcm9taXNlPHZvaWQ+IHwgdW5kZWZpbmVkPih1bmRlZmluZWQpXG4gIGNvbnN0IGxhc3RXcml0dGVuSW5kZXhSZWYgPSB1c2VSZWYoMClcbiAgLy8gVHJhY2tzIFVVSURzIGFscmVhZHkgZmx1c2hlZCBhcyBpbml0aWFsIG1lc3NhZ2VzLiBQZXJzaXN0cyBhY3Jvc3NcbiAgLy8gYnJpZGdlIHJlY29ubmVjdGlvbnMgc28gQnJpZGdlICMyKyBvbmx5IHNlbmRzIG5ldyBtZXNzYWdlcyDigJQgc2VuZGluZ1xuICAvLyBkdXBsaWNhdGUgVVVJRHMgY2F1c2VzIHRoZSBzZXJ2ZXIgdG8ga2lsbCB0aGUgV2ViU29ja2V0LlxuICBjb25zdCBmbHVzaGVkVVVJRHNSZWYgPSB1c2VSZWYobmV3IFNldDxzdHJpbmc+KCkpXG4gIGNvbnN0IGZhaWx1cmVUaW1lb3V0UmVmID0gdXNlUmVmPFJldHVyblR5cGU8dHlwZW9mIHNldFRpbWVvdXQ+IHwgdW5kZWZpbmVkPihcbiAgICB1bmRlZmluZWQsXG4gIClcbiAgLy8gUGVyc2lzdHMgYWNyb3NzIGVmZmVjdCByZS1ydW5zICh1bmxpa2UgdGhlIGVmZmVjdCdzIGxvY2FsIHN0YXRlKS4gUmVzZXRcbiAgLy8gb25seSBvbiBzdWNjZXNzZnVsIGluaXQuIEhpdHMgTUFYX0NPTlNFQ1VUSVZFX0lOSVRfRkFJTFVSRVMg4oaSIGZ1c2UgYmxvd25cbiAgLy8gZm9yIHRoZSBzZXNzaW9uLCByZWdhcmRsZXNzIG9mIHJlcGxCcmlkZ2VFbmFibGVkIHJlLXRvZ2dsaW5nLlxuICBjb25zdCBjb25zZWN1dGl2ZUZhaWx1cmVzUmVmID0gdXNlUmVmKDApXG4gIGNvbnN0IHNldEFwcFN0YXRlID0gdXNlU2V0QXBwU3RhdGUoKVxuICBjb25zdCBjb21tYW5kc1JlZiA9IHVzZVJlZihjb21tYW5kcylcbiAgY29tbWFuZHNSZWYuY3VycmVudCA9IGNvbW1hbmRzXG4gIGNvbnN0IG1haW5Mb29wTW9kZWxSZWYgPSB1c2VSZWYobWFpbkxvb3BNb2RlbClcbiAgbWFpbkxvb3BNb2RlbFJlZi5jdXJyZW50ID0gbWFpbkxvb3BNb2RlbFxuICBjb25zdCBtZXNzYWdlc1JlZiA9IHVzZVJlZihtZXNzYWdlcylcbiAgbWVzc2FnZXNSZWYuY3VycmVudCA9IG1lc3NhZ2VzXG4gIGNvbnN0IHN0b3JlID0gdXNlQXBwU3RhdGVTdG9yZSgpXG4gIGNvbnN0IHsgYWRkTm90aWZpY2F0aW9uIH0gPSB1c2VOb3RpZmljYXRpb25zKClcbiAgY29uc3QgcmVwbEJyaWRnZUVuYWJsZWQgPSBmZWF0dXJlKCdCUklER0VfTU9ERScpXG4gICAgPyAvLyBiaW9tZS1pZ25vcmUgbGludC9jb3JyZWN0bmVzcy91c2VIb29rQXRUb3BMZXZlbDogZmVhdHVyZSgpIGlzIGEgY29tcGlsZS10aW1lIGNvbnN0YW50XG4gICAgICB1c2VBcHBTdGF0ZShzID0+IHMucmVwbEJyaWRnZUVuYWJsZWQpXG4gICAgOiBmYWxzZVxuICBjb25zdCByZXBsQnJpZGdlQ29ubmVjdGVkID0gZmVhdHVyZSgnQlJJREdFX01PREUnKVxuICAgID8gLy8gYmlvbWUtaWdub3JlIGxpbnQvY29ycmVjdG5lc3MvdXNlSG9va0F0VG9wTGV2ZWw6IGZlYXR1cmUoKSBpcyBhIGNvbXBpbGUtdGltZSBjb25zdGFudFxuICAgICAgdXNlQXBwU3RhdGUocyA9PiBzLnJlcGxCcmlkZ2VDb25uZWN0ZWQpXG4gICAgOiBmYWxzZVxuICBjb25zdCByZXBsQnJpZGdlT3V0Ym91bmRPbmx5ID0gZmVhdHVyZSgnQlJJREdFX01PREUnKVxuICAgID8gLy8gYmlvbWUtaWdub3JlIGxpbnQvY29ycmVjdG5lc3MvdXNlSG9va0F0VG9wTGV2ZWw6IGZlYXR1cmUoKSBpcyBhIGNvbXBpbGUtdGltZSBjb25zdGFudFxuICAgICAgdXNlQXBwU3RhdGUocyA9PiBzLnJlcGxCcmlkZ2VPdXRib3VuZE9ubHkpXG4gICAgOiBmYWxzZVxuICBjb25zdCByZXBsQnJpZGdlSW5pdGlhbE5hbWUgPSBmZWF0dXJlKCdCUklER0VfTU9ERScpXG4gICAgPyAvLyBiaW9tZS1pZ25vcmUgbGludC9jb3JyZWN0bmVzcy91c2VIb29rQXRUb3BMZXZlbDogZmVhdHVyZSgpIGlzIGEgY29tcGlsZS10aW1lIGNvbnN0YW50XG4gICAgICB1c2VBcHBTdGF0ZShzID0+IHMucmVwbEJyaWRnZUluaXRpYWxOYW1lKVxuICAgIDogdW5kZWZpbmVkXG5cbiAgLy8gSW5pdGlhbGl6ZS90ZWFyZG93biBicmlkZ2Ugd2hlbiBlbmFibGVkIHN0YXRlIGNoYW5nZXMuXG4gIC8vIFBhc3NlcyBjdXJyZW50IG1lc3NhZ2VzIGFzIGluaXRpYWxNZXNzYWdlcyBzbyB0aGUgcmVtb3RlIHNlc3Npb25cbiAgLy8gc3RhcnRzIHdpdGggdGhlIGV4aXN0aW5nIGNvbnZlcnNhdGlvbiBjb250ZXh0IChlLmcuIGZyb20gL2JyaWRnZSkuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgLy8gZmVhdHVyZSgpIGNoZWNrIG11c3QgdXNlIHBvc2l0aXZlIHBhdHRlcm4gZm9yIGRlYWQgY29kZSBlbGltaW5hdGlvbiDigJRcbiAgICAvLyBuZWdhdGl2ZSBwYXR0ZXJuIChpZiAoIWZlYXR1cmUoLi4uKSkgcmV0dXJuKSBkb2VzIE5PVCBlbGltaW5hdGVcbiAgICAvLyBkeW5hbWljIGltcG9ydHMgYmVsb3cuXG4gICAgaWYgKGZlYXR1cmUoJ0JSSURHRV9NT0RFJykpIHtcbiAgICAgIGlmICghcmVwbEJyaWRnZUVuYWJsZWQpIHJldHVyblxuXG4gICAgICBjb25zdCBvdXRib3VuZE9ubHkgPSByZXBsQnJpZGdlT3V0Ym91bmRPbmx5XG4gICAgICBmdW5jdGlvbiBub3RpZnlCcmlkZ2VGYWlsZWQoZGV0YWlsPzogc3RyaW5nKTogdm9pZCB7XG4gICAgICAgIGlmIChvdXRib3VuZE9ubHkpIHJldHVyblxuICAgICAgICBhZGROb3RpZmljYXRpb24oe1xuICAgICAgICAgIGtleTogJ2JyaWRnZS1mYWlsZWQnLFxuICAgICAgICAgIGpzeDogKFxuICAgICAgICAgICAgPD5cbiAgICAgICAgICAgICAgPFRleHQgY29sb3I9XCJlcnJvclwiPlJlbW90ZSBDb250cm9sIGZhaWxlZDwvVGV4dD5cbiAgICAgICAgICAgICAge2RldGFpbCAmJiA8VGV4dCBkaW1Db2xvcj4gwrcge2RldGFpbH08L1RleHQ+fVxuICAgICAgICAgICAgPC8+XG4gICAgICAgICAgKSxcbiAgICAgICAgICBwcmlvcml0eTogJ2ltbWVkaWF0ZScsXG4gICAgICAgIH0pXG4gICAgICB9XG5cbiAgICAgIGlmIChjb25zZWN1dGl2ZUZhaWx1cmVzUmVmLmN1cnJlbnQgPj0gTUFYX0NPTlNFQ1VUSVZFX0lOSVRfRkFJTFVSRVMpIHtcbiAgICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICAgIGBbYnJpZGdlOnJlcGxdIEhvb2s6ICR7Y29uc2VjdXRpdmVGYWlsdXJlc1JlZi5jdXJyZW50fSBjb25zZWN1dGl2ZSBpbml0IGZhaWx1cmVzLCBub3QgcmV0cnlpbmcgdGhpcyBzZXNzaW9uYCxcbiAgICAgICAgKVxuICAgICAgICAvLyBDbGVhciByZXBsQnJpZGdlRW5hYmxlZCBzbyAvcmVtb3RlLWNvbnRyb2wgZG9lc24ndCBtaXN0YWtlbmx5IHNob3dcbiAgICAgICAgLy8gQnJpZGdlRGlzY29ubmVjdERpYWxvZyBmb3IgYSBicmlkZ2UgdGhhdCBuZXZlciBjb25uZWN0ZWQuXG4gICAgICAgIGNvbnN0IGZ1c2VIaW50ID0gJ2Rpc2FibGVkIGFmdGVyIHJlcGVhdGVkIGZhaWx1cmVzIMK3IHJlc3RhcnQgdG8gcmV0cnknXG4gICAgICAgIG5vdGlmeUJyaWRnZUZhaWxlZChmdXNlSGludClcbiAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiB7XG4gICAgICAgICAgaWYgKHByZXYucmVwbEJyaWRnZUVycm9yID09PSBmdXNlSGludCAmJiAhcHJldi5yZXBsQnJpZGdlRW5hYmxlZClcbiAgICAgICAgICAgIHJldHVybiBwcmV2XG4gICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICByZXBsQnJpZGdlRXJyb3I6IGZ1c2VIaW50LFxuICAgICAgICAgICAgcmVwbEJyaWRnZUVuYWJsZWQ6IGZhbHNlLFxuICAgICAgICAgIH1cbiAgICAgICAgfSlcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG5cbiAgICAgIGxldCBjYW5jZWxsZWQgPSBmYWxzZVxuICAgICAgLy8gQ2FwdHVyZSBtZXNzYWdlcy5sZW5ndGggbm93IHNvIHdlIGRvbid0IHJlLXNlbmQgaW5pdGlhbCBtZXNzYWdlc1xuICAgICAgLy8gdGhyb3VnaCB3cml0ZU1lc3NhZ2VzIGFmdGVyIHRoZSBicmlkZ2UgY29ubmVjdHMuXG4gICAgICBjb25zdCBpbml0aWFsTWVzc2FnZUNvdW50ID0gbWVzc2FnZXMubGVuZ3RoXG5cbiAgICAgIHZvaWQgKGFzeW5jICgpID0+IHtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICAvLyBXYWl0IGZvciBhbnkgaW4tcHJvZ3Jlc3MgdGVhcmRvd24gdG8gY29tcGxldGUgYmVmb3JlIHJlZ2lzdGVyaW5nXG4gICAgICAgICAgLy8gYSBuZXcgZW52aXJvbm1lbnQuIFdpdGhvdXQgdGhpcywgdGhlIGRlcmVnaXN0ZXIgSFRUUCBjYWxsIGZyb21cbiAgICAgICAgICAvLyB0aGUgcHJldmlvdXMgdGVhcmRvd24gcmFjZXMgd2l0aCB0aGUgbmV3IHJlZ2lzdGVyIGNhbGwsIGFuZCB0aGVcbiAgICAgICAgICAvLyBzZXJ2ZXIgbWF5IHRlYXIgZG93biB0aGUgZnJlc2hseS1jcmVhdGVkIGVudmlyb25tZW50LlxuICAgICAgICAgIGlmICh0ZWFyZG93blByb21pc2VSZWYuY3VycmVudCkge1xuICAgICAgICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICAgICAgICAnW2JyaWRnZTpyZXBsXSBIb29rOiB3YWl0aW5nIGZvciBwcmV2aW91cyB0ZWFyZG93biB0byBjb21wbGV0ZSBiZWZvcmUgcmUtaW5pdCcsXG4gICAgICAgICAgICApXG4gICAgICAgICAgICBhd2FpdCB0ZWFyZG93blByb21pc2VSZWYuY3VycmVudFxuICAgICAgICAgICAgdGVhcmRvd25Qcm9taXNlUmVmLmN1cnJlbnQgPSB1bmRlZmluZWRcbiAgICAgICAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgICAgICAgJ1ticmlkZ2U6cmVwbF0gSG9vazogcHJldmlvdXMgdGVhcmRvd24gY29tcGxldGUsIHByb2NlZWRpbmcgd2l0aCByZS1pbml0JyxcbiAgICAgICAgICAgIClcbiAgICAgICAgICB9XG4gICAgICAgICAgaWYgKGNhbmNlbGxlZCkgcmV0dXJuXG5cbiAgICAgICAgICAvLyBEeW5hbWljIGltcG9ydCBzbyB0aGUgbW9kdWxlIGlzIHRyZWUtc2hha2VuIGluIGV4dGVybmFsIGJ1aWxkc1xuICAgICAgICAgIGNvbnN0IHsgaW5pdFJlcGxCcmlkZ2UgfSA9IGF3YWl0IGltcG9ydCgnLi4vYnJpZGdlL2luaXRSZXBsQnJpZGdlLmpzJylcbiAgICAgICAgICBjb25zdCB7IHNob3VsZFNob3dBcHBVcGdyYWRlTWVzc2FnZSB9ID0gYXdhaXQgaW1wb3J0KFxuICAgICAgICAgICAgJy4uL2JyaWRnZS9lbnZMZXNzQnJpZGdlQ29uZmlnLmpzJ1xuICAgICAgICAgIClcblxuICAgICAgICAgIC8vIEFzc2lzdGFudCBtb2RlOiBwZXJwZXR1YWwgYnJpZGdlIHNlc3Npb24g4oCUIGNsYXVkZS5haSBzaG93cyBvbmVcbiAgICAgICAgICAvLyBjb250aW51b3VzIGNvbnZlcnNhdGlvbiBhY3Jvc3MgQ0xJIHJlc3RhcnRzIGluc3RlYWQgb2YgYSBuZXdcbiAgICAgICAgICAvLyBzZXNzaW9uIHBlciBpbnZvY2F0aW9uLiBpbml0QnJpZGdlQ29yZSByZWFkcyBicmlkZ2UtcG9pbnRlci5qc29uXG4gICAgICAgICAgLy8gKHRoZSBzYW1lIGNyYXNoLXJlY292ZXJ5IGZpbGUgIzIwNzM1IGFkZGVkKSBhbmQgcmV1c2VzIGl0c1xuICAgICAgICAgIC8vIHtlbnZpcm9ubWVudElkLCBzZXNzaW9uSWR9IHZpYSByZXVzZUVudmlyb25tZW50SWQgK1xuICAgICAgICAgIC8vIGFwaS5yZWNvbm5lY3RTZXNzaW9uKCkuIFRlYXJkb3duIHNraXBzIGFyY2hpdmUvZGVyZWdpc3Rlci9cbiAgICAgICAgICAvLyBwb2ludGVyLWNsZWFyIHNvIHRoZSBzZXNzaW9uIHN1cnZpdmVzIGNsZWFuIGV4aXRzLCBub3QganVzdFxuICAgICAgICAgIC8vIGNyYXNoZXMuIE5vbi1hc3Npc3RhbnQgYnJpZGdlcyBjbGVhciB0aGUgcG9pbnRlciBvbiB0ZWFyZG93blxuICAgICAgICAgIC8vIChjcmFzaC1yZWNvdmVyeSBvbmx5KS5cbiAgICAgICAgICBsZXQgcGVycGV0dWFsID0gZmFsc2VcbiAgICAgICAgICBpZiAoZmVhdHVyZSgnS0FJUk9TJykpIHtcbiAgICAgICAgICAgIGNvbnN0IHsgaXNBc3Npc3RhbnRNb2RlIH0gPSBhd2FpdCBpbXBvcnQoJy4uL2Fzc2lzdGFudC9pbmRleC5qcycpXG4gICAgICAgICAgICBwZXJwZXR1YWwgPSBpc0Fzc2lzdGFudE1vZGUoKVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIFdoZW4gYSB1c2VyIG1lc3NhZ2UgYXJyaXZlcyBmcm9tIGNsYXVkZS5haSwgaW5qZWN0IGl0IGludG8gdGhlIFJFUEwuXG4gICAgICAgICAgLy8gUHJlc2VydmVzIHRoZSBvcmlnaW5hbCBVVUlEIHNvIHRoYXQgd2hlbiB0aGUgbWVzc2FnZSBpcyBmb3J3YXJkZWRcbiAgICAgICAgICAvLyBiYWNrIHRvIENDUiwgaXQgbWF0Y2hlcyB0aGUgb3JpZ2luYWwg4oCUIGF2b2lkaW5nIGR1cGxpY2F0ZSBtZXNzYWdlcy5cbiAgICAgICAgICAvL1xuICAgICAgICAgIC8vIEFzeW5jIGJlY2F1c2UgZmlsZV9hdHRhY2htZW50cyAoaWYgcHJlc2VudCkgbmVlZCBhIG5ldHdvcmsgZmV0Y2ggK1xuICAgICAgICAgIC8vIGRpc2sgd3JpdGUgYmVmb3JlIHdlIGVucXVldWUgd2l0aCB0aGUgQHBhdGggcHJlZml4LiBDYWxsZXIgZG9lc24ndFxuICAgICAgICAgIC8vIGF3YWl0IOKAlCBtZXNzYWdlcyB3aXRoIGF0dGFjaG1lbnRzIGp1c3QgbGFuZCBpbiB0aGUgcXVldWUgc2xpZ2h0bHlcbiAgICAgICAgICAvLyBsYXRlciwgd2hpY2ggaXMgZmluZSAod2ViIG1lc3NhZ2VzIGFyZW4ndCByYXBpZC1maXJlKS5cbiAgICAgICAgICBhc3luYyBmdW5jdGlvbiBoYW5kbGVJbmJvdW5kTWVzc2FnZShtc2c6IFNES01lc3NhZ2UpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICAgIGNvbnN0IGZpZWxkcyA9IGV4dHJhY3RJbmJvdW5kTWVzc2FnZUZpZWxkcyhtc2cpXG4gICAgICAgICAgICAgIGlmICghZmllbGRzKSByZXR1cm5cblxuICAgICAgICAgICAgICBjb25zdCB7IHV1aWQgfSA9IGZpZWxkc1xuXG4gICAgICAgICAgICAgIC8vIER5bmFtaWMgaW1wb3J0IGtlZXBzIHRoZSBicmlkZ2UgY29kZSBvdXQgb2Ygbm9uLUJSSURHRV9NT0RFIGJ1aWxkcy5cbiAgICAgICAgICAgICAgY29uc3QgeyByZXNvbHZlQW5kUHJlcGVuZCB9ID0gYXdhaXQgaW1wb3J0KFxuICAgICAgICAgICAgICAgICcuLi9icmlkZ2UvaW5ib3VuZEF0dGFjaG1lbnRzLmpzJ1xuICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgIGxldCBzYW5pdGl6ZWQgPSBmaWVsZHMuY29udGVudFxuICAgICAgICAgICAgICBpZiAoZmVhdHVyZSgnS0FJUk9TX0dJVEhVQl9XRUJIT09LUycpKSB7XG4gICAgICAgICAgICAgICAgLyogZXNsaW50LWRpc2FibGUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXJlcXVpcmUtaW1wb3J0cyAqL1xuICAgICAgICAgICAgICAgIGNvbnN0IHsgc2FuaXRpemVJbmJvdW5kV2ViaG9va0NvbnRlbnQgfSA9XG4gICAgICAgICAgICAgICAgICByZXF1aXJlKCcuLi9icmlkZ2Uvd2ViaG9va1Nhbml0aXplci5qcycpIGFzIHR5cGVvZiBpbXBvcnQoJy4uL2JyaWRnZS93ZWJob29rU2FuaXRpemVyLmpzJylcbiAgICAgICAgICAgICAgICAvKiBlc2xpbnQtZW5hYmxlIEB0eXBlc2NyaXB0LWVzbGludC9uby1yZXF1aXJlLWltcG9ydHMgKi9cbiAgICAgICAgICAgICAgICBzYW5pdGl6ZWQgPSBzYW5pdGl6ZUluYm91bmRXZWJob29rQ29udGVudChmaWVsZHMuY29udGVudClcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICBjb25zdCBjb250ZW50ID0gYXdhaXQgcmVzb2x2ZUFuZFByZXBlbmQobXNnLCBzYW5pdGl6ZWQpXG5cbiAgICAgICAgICAgICAgY29uc3QgcHJldmlldyA9XG4gICAgICAgICAgICAgICAgdHlwZW9mIGNvbnRlbnQgPT09ICdzdHJpbmcnXG4gICAgICAgICAgICAgICAgICA/IGNvbnRlbnQuc2xpY2UoMCwgODApXG4gICAgICAgICAgICAgICAgICA6IGBbJHtjb250ZW50Lmxlbmd0aH0gY29udGVudCBibG9ja3NdYFxuICAgICAgICAgICAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICAgICAgICAgICAgYFticmlkZ2U6cmVwbF0gSW5qZWN0aW5nIGluYm91bmQgdXNlciBtZXNzYWdlOiAke3ByZXZpZXd9JHt1dWlkID8gYCB1dWlkPSR7dXVpZH1gIDogJyd9YCxcbiAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgICBlbnF1ZXVlKHtcbiAgICAgICAgICAgICAgICB2YWx1ZTogY29udGVudCxcbiAgICAgICAgICAgICAgICBtb2RlOiAncHJvbXB0JyBhcyBjb25zdCxcbiAgICAgICAgICAgICAgICB1dWlkLFxuICAgICAgICAgICAgICAgIC8vIHNraXBTbGFzaENvbW1hbmRzIHN0YXlzIHRydWUgYXMgZGVmZW5zZS1pbi1kZXB0aCDigJRcbiAgICAgICAgICAgICAgICAvLyBwcm9jZXNzVXNlcklucHV0QmFzZSBvdmVycmlkZXMgaXQgaW50ZXJuYWxseSB3aGVuIGJyaWRnZU9yaWdpblxuICAgICAgICAgICAgICAgIC8vIGlzIHNldCBBTkQgdGhlIHJlc29sdmVkIGNvbW1hbmQgcGFzc2VzIGlzQnJpZGdlU2FmZUNvbW1hbmQuXG4gICAgICAgICAgICAgICAgLy8gVGhpcyBrZWVwcyBleGl0LXdvcmQgc3VwcHJlc3Npb24gYW5kIGltbWVkaWF0ZS1jb21tYW5kIGJsb2Nrc1xuICAgICAgICAgICAgICAgIC8vIGludGFjdCBmb3IgYW55IGNvZGUgcGF0aCB0aGF0IGNoZWNrcyBza2lwU2xhc2hDb21tYW5kcyBkaXJlY3RseS5cbiAgICAgICAgICAgICAgICBza2lwU2xhc2hDb21tYW5kczogdHJ1ZSxcbiAgICAgICAgICAgICAgICBicmlkZ2VPcmlnaW46IHRydWUsXG4gICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICB9IGNhdGNoIChlKSB7XG4gICAgICAgICAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgICAgICAgICBgW2JyaWRnZTpyZXBsXSBoYW5kbGVJbmJvdW5kTWVzc2FnZSBmYWlsZWQ6ICR7ZX1gLFxuICAgICAgICAgICAgICAgIHsgbGV2ZWw6ICdlcnJvcicgfSxcbiAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIFN0YXRlIGNoYW5nZSBjYWxsYmFjayDigJQgbWFwcyBicmlkZ2UgbGlmZWN5Y2xlIGV2ZW50cyB0byBBcHBTdGF0ZS5cbiAgICAgICAgICBmdW5jdGlvbiBoYW5kbGVTdGF0ZUNoYW5nZShcbiAgICAgICAgICAgIHN0YXRlOiBCcmlkZ2VTdGF0ZSxcbiAgICAgICAgICAgIGRldGFpbD86IHN0cmluZyxcbiAgICAgICAgICApOiB2b2lkIHtcbiAgICAgICAgICAgIGlmIChjYW5jZWxsZWQpIHJldHVyblxuICAgICAgICAgICAgaWYgKG91dGJvdW5kT25seSkge1xuICAgICAgICAgICAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICAgICAgICAgICAgYFticmlkZ2U6cmVwbF0gTWlycm9yIHN0YXRlPSR7c3RhdGV9JHtkZXRhaWwgPyBgIGRldGFpbD0ke2RldGFpbH1gIDogJyd9YCxcbiAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgICAvLyBTeW5jIHJlcGxCcmlkZ2VDb25uZWN0ZWQgc28gdGhlIGZvcndhcmRpbmcgZWZmZWN0IHN0YXJ0cy9zdG9wc1xuICAgICAgICAgICAgICAvLyB3cml0aW5nIGFzIHRoZSB0cmFuc3BvcnQgY29tZXMgdXAgb3IgZGllcy5cbiAgICAgICAgICAgICAgaWYgKHN0YXRlID09PSAnZmFpbGVkJykge1xuICAgICAgICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgICAgICAgICAgICAgICAgaWYgKCFwcmV2LnJlcGxCcmlkZ2VDb25uZWN0ZWQpIHJldHVybiBwcmV2XG4gICAgICAgICAgICAgICAgICByZXR1cm4geyAuLi5wcmV2LCByZXBsQnJpZGdlQ29ubmVjdGVkOiBmYWxzZSB9XG4gICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgfSBlbHNlIGlmIChzdGF0ZSA9PT0gJ3JlYWR5JyB8fCBzdGF0ZSA9PT0gJ2Nvbm5lY3RlZCcpIHtcbiAgICAgICAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+IHtcbiAgICAgICAgICAgICAgICAgIGlmIChwcmV2LnJlcGxCcmlkZ2VDb25uZWN0ZWQpIHJldHVybiBwcmV2XG4gICAgICAgICAgICAgICAgICByZXR1cm4geyAuLi5wcmV2LCByZXBsQnJpZGdlQ29ubmVjdGVkOiB0cnVlIH1cbiAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIHJldHVyblxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgY29uc3QgaGFuZGxlID0gaGFuZGxlUmVmLmN1cnJlbnRcbiAgICAgICAgICAgIHN3aXRjaCAoc3RhdGUpIHtcbiAgICAgICAgICAgICAgY2FzZSAncmVhZHknOlxuICAgICAgICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgICAgICAgICAgICAgICAgY29uc3QgY29ubmVjdFVybCA9XG4gICAgICAgICAgICAgICAgICAgIGhhbmRsZSAmJiBoYW5kbGUuZW52aXJvbm1lbnRJZCAhPT0gJydcbiAgICAgICAgICAgICAgICAgICAgICA/IGJ1aWxkQnJpZGdlQ29ubmVjdFVybChcbiAgICAgICAgICAgICAgICAgICAgICAgICAgaGFuZGxlLmVudmlyb25tZW50SWQsXG4gICAgICAgICAgICAgICAgICAgICAgICAgIGhhbmRsZS5zZXNzaW9uSW5ncmVzc1VybCxcbiAgICAgICAgICAgICAgICAgICAgICAgIClcbiAgICAgICAgICAgICAgICAgICAgICA6IHByZXYucmVwbEJyaWRnZUNvbm5lY3RVcmxcbiAgICAgICAgICAgICAgICAgIGNvbnN0IHNlc3Npb25VcmwgPSBoYW5kbGVcbiAgICAgICAgICAgICAgICAgICAgPyBnZXRSZW1vdGVTZXNzaW9uVXJsKFxuICAgICAgICAgICAgICAgICAgICAgICAgaGFuZGxlLmJyaWRnZVNlc3Npb25JZCxcbiAgICAgICAgICAgICAgICAgICAgICAgIGhhbmRsZS5zZXNzaW9uSW5ncmVzc1VybCxcbiAgICAgICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgICAgIDogcHJldi5yZXBsQnJpZGdlU2Vzc2lvblVybFxuICAgICAgICAgICAgICAgICAgY29uc3QgZW52SWQgPSBoYW5kbGU/LmVudmlyb25tZW50SWRcbiAgICAgICAgICAgICAgICAgIGNvbnN0IHNlc3Npb25JZCA9IGhhbmRsZT8uYnJpZGdlU2Vzc2lvbklkXG4gICAgICAgICAgICAgICAgICBpZiAoXG4gICAgICAgICAgICAgICAgICAgIHByZXYucmVwbEJyaWRnZUNvbm5lY3RlZCAmJlxuICAgICAgICAgICAgICAgICAgICAhcHJldi5yZXBsQnJpZGdlU2Vzc2lvbkFjdGl2ZSAmJlxuICAgICAgICAgICAgICAgICAgICAhcHJldi5yZXBsQnJpZGdlUmVjb25uZWN0aW5nICYmXG4gICAgICAgICAgICAgICAgICAgIHByZXYucmVwbEJyaWRnZUNvbm5lY3RVcmwgPT09IGNvbm5lY3RVcmwgJiZcbiAgICAgICAgICAgICAgICAgICAgcHJldi5yZXBsQnJpZGdlU2Vzc2lvblVybCA9PT0gc2Vzc2lvblVybCAmJlxuICAgICAgICAgICAgICAgICAgICBwcmV2LnJlcGxCcmlkZ2VFbnZpcm9ubWVudElkID09PSBlbnZJZCAmJlxuICAgICAgICAgICAgICAgICAgICBwcmV2LnJlcGxCcmlkZ2VTZXNzaW9uSWQgPT09IHNlc3Npb25JZFxuICAgICAgICAgICAgICAgICAgKSB7XG4gICAgICAgICAgICAgICAgICAgIHJldHVybiBwcmV2XG4gICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAgICAgICAuLi5wcmV2LFxuICAgICAgICAgICAgICAgICAgICByZXBsQnJpZGdlQ29ubmVjdGVkOiB0cnVlLFxuICAgICAgICAgICAgICAgICAgICByZXBsQnJpZGdlU2Vzc2lvbkFjdGl2ZTogZmFsc2UsXG4gICAgICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VSZWNvbm5lY3Rpbmc6IGZhbHNlLFxuICAgICAgICAgICAgICAgICAgICByZXBsQnJpZGdlQ29ubmVjdFVybDogY29ubmVjdFVybCxcbiAgICAgICAgICAgICAgICAgICAgcmVwbEJyaWRnZVNlc3Npb25Vcmw6IHNlc3Npb25VcmwsXG4gICAgICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VFbnZpcm9ubWVudElkOiBlbnZJZCxcbiAgICAgICAgICAgICAgICAgICAgcmVwbEJyaWRnZVNlc3Npb25JZDogc2Vzc2lvbklkLFxuICAgICAgICAgICAgICAgICAgICByZXBsQnJpZGdlRXJyb3I6IHVuZGVmaW5lZCxcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAgIGJyZWFrXG4gICAgICAgICAgICAgIGNhc2UgJ2Nvbm5lY3RlZCc6IHtcbiAgICAgICAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+IHtcbiAgICAgICAgICAgICAgICAgIGlmIChwcmV2LnJlcGxCcmlkZ2VTZXNzaW9uQWN0aXZlKSByZXR1cm4gcHJldlxuICAgICAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICAgICAgICAgICAgcmVwbEJyaWRnZUNvbm5lY3RlZDogdHJ1ZSxcbiAgICAgICAgICAgICAgICAgICAgcmVwbEJyaWRnZVNlc3Npb25BY3RpdmU6IHRydWUsXG4gICAgICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VSZWNvbm5lY3Rpbmc6IGZhbHNlLFxuICAgICAgICAgICAgICAgICAgICByZXBsQnJpZGdlRXJyb3I6IHVuZGVmaW5lZCxcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAgIC8vIFNlbmQgc3lzdGVtL2luaXQgc28gcmVtb3RlIGNsaWVudHMgKHdlYi9pT1MvQW5kcm9pZCkgZ2V0XG4gICAgICAgICAgICAgICAgLy8gc2Vzc2lvbiBtZXRhZGF0YS4gUkVQTCB1c2VzIHF1ZXJ5KCkgZGlyZWN0bHkg4oCUIG5ldmVyIGhpdHNcbiAgICAgICAgICAgICAgICAvLyBRdWVyeUVuZ2luZSdzIFNES01lc3NhZ2UgbGF5ZXIg4oCUIHNvIHRoaXMgaXMgdGhlIG9ubHkgcGF0aFxuICAgICAgICAgICAgICAgIC8vIHRvIHB1dCBzeXN0ZW0vaW5pdCBvbiB0aGUgUkVQTC1icmlkZ2Ugd2lyZS4gU2tpbGxzIGxvYWQgaXNcbiAgICAgICAgICAgICAgICAvLyBhc3luYyAobWVtb2l6ZWQsIGNoZWFwIGFmdGVyIFJFUEwgc3RhcnR1cCk7IGZpcmUtYW5kLWZvcmdldFxuICAgICAgICAgICAgICAgIC8vIHNvIHRoZSBjb25uZWN0ZWQtc3RhdGUgdHJhbnNpdGlvbiBpc24ndCBibG9ja2VkLlxuICAgICAgICAgICAgICAgIGlmIChcbiAgICAgICAgICAgICAgICAgIGdldEZlYXR1cmVWYWx1ZV9DQUNIRURfTUFZX0JFX1NUQUxFKFxuICAgICAgICAgICAgICAgICAgICAndGVuZ3VfYnJpZGdlX3N5c3RlbV9pbml0JyxcbiAgICAgICAgICAgICAgICAgICAgZmFsc2UsXG4gICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgKSB7XG4gICAgICAgICAgICAgICAgICB2b2lkIChhc3luYyAoKSA9PiB7XG4gICAgICAgICAgICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICAgICAgICAgICAgY29uc3Qgc2tpbGxzID0gYXdhaXQgZ2V0U2xhc2hDb21tYW5kVG9vbFNraWxscyhnZXRDd2QoKSlcbiAgICAgICAgICAgICAgICAgICAgICBpZiAoY2FuY2VsbGVkKSByZXR1cm5cbiAgICAgICAgICAgICAgICAgICAgICBjb25zdCBzdGF0ZSA9IHN0b3JlLmdldFN0YXRlKClcbiAgICAgICAgICAgICAgICAgICAgICBoYW5kbGVSZWYuY3VycmVudD8ud3JpdGVTZGtNZXNzYWdlcyhbXG4gICAgICAgICAgICAgICAgICAgICAgICBidWlsZFN5c3RlbUluaXRNZXNzYWdlKHtcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gdG9vbHMvbWNwQ2xpZW50cy9wbHVnaW5zIHJlZGFjdGVkIGZvciBSRVBMLWJyaWRnZTpcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gTUNQLXByZWZpeGVkIHRvb2wgbmFtZXMgYW5kIHNlcnZlciBuYW1lcyBsZWFrIHdoaWNoXG4gICAgICAgICAgICAgICAgICAgICAgICAgIC8vIGludGVncmF0aW9ucyB0aGUgdXNlciBoYXMgd2lyZWQgdXA7IHBsdWdpbiBwYXRocyBsZWFrXG4gICAgICAgICAgICAgICAgICAgICAgICAgIC8vIHJhdyBmaWxlc3lzdGVtIHBhdGhzICh1c2VybmFtZSwgcHJvamVjdCBzdHJ1Y3R1cmUpLlxuICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBDQ1IgdjIgcGVyc2lzdHMgU0RLIG1lc3NhZ2VzIHRvIFNwYW5uZXIg4oCUIHVzZXJzIHdob1xuICAgICAgICAgICAgICAgICAgICAgICAgICAvLyB0YXAgXCJDb25uZWN0IGZyb20gcGhvbmVcIiBtYXkgbm90IGV4cGVjdCB0aGVzZSBvblxuICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBBbnRocm9waWMncyBzZXJ2ZXJzLiBRdWVyeUVuZ2luZSAoU0RLKSBzdGlsbCBlbWl0c1xuICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBmdWxsIGxpc3RzIOKAlCBTREsgY29uc3VtZXJzIGV4cGVjdCBmdWxsIHRlbGVtZXRyeS5cbiAgICAgICAgICAgICAgICAgICAgICAgICAgdG9vbHM6IFtdLFxuICAgICAgICAgICAgICAgICAgICAgICAgICBtY3BDbGllbnRzOiBbXSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgbW9kZWw6IG1haW5Mb29wTW9kZWxSZWYuY3VycmVudCxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgcGVybWlzc2lvbk1vZGU6IHN0YXRlLnRvb2xQZXJtaXNzaW9uQ29udGV4dFxuICAgICAgICAgICAgICAgICAgICAgICAgICAgIC5tb2RlIGFzIFBlcm1pc3Npb25Nb2RlLCAvLyBUT0RPOiBhdm9pZCB0aGUgY2FzdFxuICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBSZW1vdGUgY2xpZW50cyBjYW4gb25seSBpbnZva2UgYnJpZGdlLXNhZmUgY29tbWFuZHMg4oCUXG4gICAgICAgICAgICAgICAgICAgICAgICAgIC8vIGFkdmVydGlzaW5nIHVuc2FmZSBvbmVzIChsb2NhbC1qc3gsIHVuYWxsb3dlZCBsb2NhbClcbiAgICAgICAgICAgICAgICAgICAgICAgICAgLy8gd291bGQgbGV0IG1vYmlsZS93ZWIgYXR0ZW1wdCB0aGVtIGFuZCBoaXQgZXJyb3JzLlxuICAgICAgICAgICAgICAgICAgICAgICAgICBjb21tYW5kczpcbiAgICAgICAgICAgICAgICAgICAgICAgICAgICBjb21tYW5kc1JlZi5jdXJyZW50LmZpbHRlcihpc0JyaWRnZVNhZmVDb21tYW5kKSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgYWdlbnRzOiBzdGF0ZS5hZ2VudERlZmluaXRpb25zLmFjdGl2ZUFnZW50cyxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgc2tpbGxzLFxuICAgICAgICAgICAgICAgICAgICAgICAgICBwbHVnaW5zOiBbXSxcbiAgICAgICAgICAgICAgICAgICAgICAgICAgZmFzdE1vZGU6IHN0YXRlLmZhc3RNb2RlLFxuICAgICAgICAgICAgICAgICAgICAgICAgfSksXG4gICAgICAgICAgICAgICAgICAgICAgXSlcbiAgICAgICAgICAgICAgICAgICAgfSBjYXRjaCAoZXJyKSB7XG4gICAgICAgICAgICAgICAgICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICAgICAgICAgICAgICAgICAgYFticmlkZ2U6cmVwbF0gRmFpbGVkIHRvIHNlbmQgc3lzdGVtL2luaXQ6ICR7ZXJyb3JNZXNzYWdlKGVycil9YCxcbiAgICAgICAgICAgICAgICAgICAgICAgIHsgbGV2ZWw6ICdlcnJvcicgfSxcbiAgICAgICAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgIH0pKClcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgYnJlYWtcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICBjYXNlICdyZWNvbm5lY3RpbmcnOlxuICAgICAgICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgICAgICAgICAgICAgICAgaWYgKHByZXYucmVwbEJyaWRnZVJlY29ubmVjdGluZykgcmV0dXJuIHByZXZcbiAgICAgICAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VSZWNvbm5lY3Rpbmc6IHRydWUsXG4gICAgICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VTZXNzaW9uQWN0aXZlOiBmYWxzZSxcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAgIGJyZWFrXG4gICAgICAgICAgICAgIGNhc2UgJ2ZhaWxlZCc6XG4gICAgICAgICAgICAgICAgLy8gQ2xlYXIgYW55IHByZXZpb3VzIGZhaWx1cmUgZGlzbWlzcyB0aW1lclxuICAgICAgICAgICAgICAgIGNsZWFyVGltZW91dChmYWlsdXJlVGltZW91dFJlZi5jdXJyZW50KVxuICAgICAgICAgICAgICAgIG5vdGlmeUJyaWRnZUZhaWxlZChkZXRhaWwpXG4gICAgICAgICAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiAoe1xuICAgICAgICAgICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VFcnJvcjogZGV0YWlsLFxuICAgICAgICAgICAgICAgICAgcmVwbEJyaWRnZVJlY29ubmVjdGluZzogZmFsc2UsXG4gICAgICAgICAgICAgICAgICByZXBsQnJpZGdlU2Vzc2lvbkFjdGl2ZTogZmFsc2UsXG4gICAgICAgICAgICAgICAgICByZXBsQnJpZGdlQ29ubmVjdGVkOiBmYWxzZSxcbiAgICAgICAgICAgICAgICB9KSlcbiAgICAgICAgICAgICAgICAvLyBBdXRvLWRpc2FibGUgYWZ0ZXIgdGltZW91dCBzbyB0aGUgaG9vayBzdG9wcyByZXRyeWluZy5cbiAgICAgICAgICAgICAgICBmYWlsdXJlVGltZW91dFJlZi5jdXJyZW50ID0gc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICAgICAgICAgICAgICBpZiAoY2FuY2VsbGVkKSByZXR1cm5cbiAgICAgICAgICAgICAgICAgIGZhaWx1cmVUaW1lb3V0UmVmLmN1cnJlbnQgPSB1bmRlZmluZWRcbiAgICAgICAgICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgICAgICAgICAgICAgICAgICBpZiAoIXByZXYucmVwbEJyaWRnZUVycm9yKSByZXR1cm4gcHJldlxuICAgICAgICAgICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgICAgICAgICAgcmVwbEJyaWRnZUVuYWJsZWQ6IGZhbHNlLFxuICAgICAgICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VFcnJvcjogdW5kZWZpbmVkLFxuICAgICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICAgIH0sIEJSSURHRV9GQUlMVVJFX0RJU01JU1NfTVMpXG4gICAgICAgICAgICAgICAgYnJlYWtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyBNYXAgb2YgcGVuZGluZyBicmlkZ2UgcGVybWlzc2lvbiByZXNwb25zZSBoYW5kbGVycywga2V5ZWQgYnkgcmVxdWVzdF9pZC5cbiAgICAgICAgICAvLyBFYWNoIGVudHJ5IGlzIGFuIG9uUmVzcG9uc2UgaGFuZGxlciB3YWl0aW5nIGZvciBDQ1IgdG8gcmVwbHkuXG4gICAgICAgICAgY29uc3QgcGVuZGluZ1Blcm1pc3Npb25IYW5kbGVycyA9IG5ldyBNYXA8XG4gICAgICAgICAgICBzdHJpbmcsXG4gICAgICAgICAgICAocmVzcG9uc2U6IEJyaWRnZVBlcm1pc3Npb25SZXNwb25zZSkgPT4gdm9pZFxuICAgICAgICAgID4oKVxuXG4gICAgICAgICAgLy8gRGlzcGF0Y2ggaW5jb21pbmcgY29udHJvbF9yZXNwb25zZSBtZXNzYWdlcyB0byByZWdpc3RlcmVkIGhhbmRsZXJzXG4gICAgICAgICAgZnVuY3Rpb24gaGFuZGxlUGVybWlzc2lvblJlc3BvbnNlKG1zZzogU0RLQ29udHJvbFJlc3BvbnNlKTogdm9pZCB7XG4gICAgICAgICAgICBjb25zdCByZXF1ZXN0SWQgPSBtc2cucmVzcG9uc2U/LnJlcXVlc3RfaWRcbiAgICAgICAgICAgIGlmICghcmVxdWVzdElkKSByZXR1cm5cbiAgICAgICAgICAgIGNvbnN0IGhhbmRsZXIgPSBwZW5kaW5nUGVybWlzc2lvbkhhbmRsZXJzLmdldChyZXF1ZXN0SWQpXG4gICAgICAgICAgICBpZiAoIWhhbmRsZXIpIHtcbiAgICAgICAgICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICAgICAgICAgIGBbYnJpZGdlOnJlcGxdIE5vIGhhbmRsZXIgZm9yIGNvbnRyb2xfcmVzcG9uc2UgcmVxdWVzdF9pZD0ke3JlcXVlc3RJZH1gLFxuICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgIHJldHVyblxuICAgICAgICAgICAgfVxuICAgICAgICAgICAgcGVuZGluZ1Blcm1pc3Npb25IYW5kbGVycy5kZWxldGUocmVxdWVzdElkKVxuICAgICAgICAgICAgLy8gRXh0cmFjdCB0aGUgcGVybWlzc2lvbiBkZWNpc2lvbiBmcm9tIHRoZSBjb250cm9sX3Jlc3BvbnNlIHBheWxvYWRcbiAgICAgICAgICAgIGNvbnN0IGlubmVyID0gbXNnLnJlc3BvbnNlXG4gICAgICAgICAgICBpZiAoXG4gICAgICAgICAgICAgIGlubmVyLnN1YnR5cGUgPT09ICdzdWNjZXNzJyAmJlxuICAgICAgICAgICAgICBpbm5lci5yZXNwb25zZSAmJlxuICAgICAgICAgICAgICBpc0JyaWRnZVBlcm1pc3Npb25SZXNwb25zZShpbm5lci5yZXNwb25zZSlcbiAgICAgICAgICAgICkge1xuICAgICAgICAgICAgICBoYW5kbGVyKGlubmVyLnJlc3BvbnNlKVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIGNvbnN0IGhhbmRsZSA9IGF3YWl0IGluaXRSZXBsQnJpZGdlKHtcbiAgICAgICAgICAgIG91dGJvdW5kT25seSxcbiAgICAgICAgICAgIHRhZ3M6IG91dGJvdW5kT25seSA/IFsnY2NyLW1pcnJvciddIDogdW5kZWZpbmVkLFxuICAgICAgICAgICAgb25JbmJvdW5kTWVzc2FnZTogaGFuZGxlSW5ib3VuZE1lc3NhZ2UsXG4gICAgICAgICAgICBvblBlcm1pc3Npb25SZXNwb25zZTogaGFuZGxlUGVybWlzc2lvblJlc3BvbnNlLFxuICAgICAgICAgICAgb25JbnRlcnJ1cHQoKSB7XG4gICAgICAgICAgICAgIGFib3J0Q29udHJvbGxlclJlZi5jdXJyZW50Py5hYm9ydCgpXG4gICAgICAgICAgICB9LFxuICAgICAgICAgICAgb25TZXRNb2RlbChtb2RlbCkge1xuICAgICAgICAgICAgICBjb25zdCByZXNvbHZlZCA9IG1vZGVsID09PSAnZGVmYXVsdCcgPyBudWxsIDogKG1vZGVsID8/IG51bGwpXG4gICAgICAgICAgICAgIHNldE1haW5Mb29wTW9kZWxPdmVycmlkZShyZXNvbHZlZClcbiAgICAgICAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiB7XG4gICAgICAgICAgICAgICAgaWYgKHByZXYubWFpbkxvb3BNb2RlbEZvclNlc3Npb24gPT09IHJlc29sdmVkKSByZXR1cm4gcHJldlxuICAgICAgICAgICAgICAgIHJldHVybiB7IC4uLnByZXYsIG1haW5Mb29wTW9kZWxGb3JTZXNzaW9uOiByZXNvbHZlZCB9XG4gICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICB9LFxuICAgICAgICAgICAgb25TZXRNYXhUaGlua2luZ1Rva2VucyhtYXhUb2tlbnMpIHtcbiAgICAgICAgICAgICAgY29uc3QgZW5hYmxlZCA9IG1heFRva2VucyAhPT0gbnVsbFxuICAgICAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+IHtcbiAgICAgICAgICAgICAgICBpZiAocHJldi50aGlua2luZ0VuYWJsZWQgPT09IGVuYWJsZWQpIHJldHVybiBwcmV2XG4gICAgICAgICAgICAgICAgcmV0dXJuIHsgLi4ucHJldiwgdGhpbmtpbmdFbmFibGVkOiBlbmFibGVkIH1cbiAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgICBvblNldFBlcm1pc3Npb25Nb2RlKG1vZGUpIHtcbiAgICAgICAgICAgICAgLy8gUG9saWN5IGd1YXJkcyBNVVNUIGZpcmUgYmVmb3JlIHRyYW5zaXRpb25QZXJtaXNzaW9uTW9kZSDigJRcbiAgICAgICAgICAgICAgLy8gaXRzIGludGVybmFsIGF1dG8tZ2F0ZSBjaGVjayBpcyBhIGRlZmVuc2l2ZSB0aHJvdyAod2l0aCBhXG4gICAgICAgICAgICAgIC8vIHNldEF1dG9Nb2RlQWN0aXZlKHRydWUpIHNpZGUtZWZmZWN0IEJFRk9SRSB0aGUgdGhyb3cpIHJhdGhlclxuICAgICAgICAgICAgICAvLyB0aGFuIGEgZ3JhY2VmdWwgcmVqZWN0LiBMZXR0aW5nIHRoYXQgdGhyb3cgZXNjYXBlIHdvdWxkOlxuICAgICAgICAgICAgICAvLyAoMSkgbGVhdmUgU1RBVEUuYXV0b01vZGVBY3RpdmU9dHJ1ZSB3aGlsZSB0aGUgbW9kZSBpc1xuICAgICAgICAgICAgICAvLyAgICAgdW5jaGFuZ2VkICgzLXdheSBpbnZhcmlhbnQgdmlvbGF0aW9uIHBlciBzcmMvQ0xBVURFLm1kKVxuICAgICAgICAgICAgICAvLyAoMikgZmFpbCB0byBzZW5kIGEgY29udHJvbF9yZXNwb25zZSDihpIgc2VydmVyIGtpbGxzIFdTXG4gICAgICAgICAgICAgIC8vIFRoZXNlIG1pcnJvciBwcmludC50cyBoYW5kbGVTZXRQZXJtaXNzaW9uTW9kZTsgdGhlIGJyaWRnZVxuICAgICAgICAgICAgICAvLyBjYW4ndCBpbXBvcnQgdGhlIGNoZWNrcyBkaXJlY3RseSAoYm9vdHN0cmFwLWlzb2xhdGlvbiksIHNvXG4gICAgICAgICAgICAgIC8vIGl0IHJlbGllcyBvbiB0aGlzIHZlcmRpY3QgdG8gZW1pdCB0aGUgZXJyb3IgcmVzcG9uc2UuXG4gICAgICAgICAgICAgIGlmIChtb2RlID09PSAnYnlwYXNzUGVybWlzc2lvbnMnKSB7XG4gICAgICAgICAgICAgICAgaWYgKGlzQnlwYXNzUGVybWlzc2lvbnNNb2RlRGlzYWJsZWQoKSkge1xuICAgICAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICAgICAgb2s6IGZhbHNlLFxuICAgICAgICAgICAgICAgICAgICBlcnJvcjpcbiAgICAgICAgICAgICAgICAgICAgICAnQ2Fubm90IHNldCBwZXJtaXNzaW9uIG1vZGUgdG8gYnlwYXNzUGVybWlzc2lvbnMgYmVjYXVzZSBpdCBpcyBkaXNhYmxlZCBieSBzZXR0aW5ncyBvciBjb25maWd1cmF0aW9uJyxcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgaWYgKFxuICAgICAgICAgICAgICAgICAgIXN0b3JlLmdldFN0YXRlKCkudG9vbFBlcm1pc3Npb25Db250ZXh0XG4gICAgICAgICAgICAgICAgICAgIC5pc0J5cGFzc1Blcm1pc3Npb25zTW9kZUF2YWlsYWJsZVxuICAgICAgICAgICAgICAgICkge1xuICAgICAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICAgICAgb2s6IGZhbHNlLFxuICAgICAgICAgICAgICAgICAgICBlcnJvcjpcbiAgICAgICAgICAgICAgICAgICAgICAnQ2Fubm90IHNldCBwZXJtaXNzaW9uIG1vZGUgdG8gYnlwYXNzUGVybWlzc2lvbnMgYmVjYXVzZSB0aGUgc2Vzc2lvbiB3YXMgbm90IGxhdW5jaGVkIHdpdGggLS1kYW5nZXJvdXNseS1za2lwLXBlcm1pc3Npb25zJyxcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgaWYgKFxuICAgICAgICAgICAgICAgIGZlYXR1cmUoJ1RSQU5TQ1JJUFRfQ0xBU1NJRklFUicpICYmXG4gICAgICAgICAgICAgICAgbW9kZSA9PT0gJ2F1dG8nICYmXG4gICAgICAgICAgICAgICAgIWlzQXV0b01vZGVHYXRlRW5hYmxlZCgpXG4gICAgICAgICAgICAgICkge1xuICAgICAgICAgICAgICAgIGNvbnN0IHJlYXNvbiA9IGdldEF1dG9Nb2RlVW5hdmFpbGFibGVSZWFzb24oKVxuICAgICAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgICAgICBvazogZmFsc2UsXG4gICAgICAgICAgICAgICAgICBlcnJvcjogcmVhc29uXG4gICAgICAgICAgICAgICAgICAgID8gYENhbm5vdCBzZXQgcGVybWlzc2lvbiBtb2RlIHRvIGF1dG86ICR7Z2V0QXV0b01vZGVVbmF2YWlsYWJsZU5vdGlmaWNhdGlvbihyZWFzb24pfWBcbiAgICAgICAgICAgICAgICAgICAgOiAnQ2Fubm90IHNldCBwZXJtaXNzaW9uIG1vZGUgdG8gYXV0bycsXG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIC8vIEd1YXJkcyBwYXNzZWQg4oCUIGFwcGx5IHZpYSB0aGUgY2VudHJhbGl6ZWQgdHJhbnNpdGlvbiBzb1xuICAgICAgICAgICAgICAvLyBwcmVQbGFuTW9kZSBzdGFzaGluZyBhbmQgYXV0by1tb2RlIHN0YXRlIHN5bmMgYWxsIGZpcmUuXG4gICAgICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgICAgICAgICAgICAgIGNvbnN0IGN1cnJlbnQgPSBwcmV2LnRvb2xQZXJtaXNzaW9uQ29udGV4dC5tb2RlXG4gICAgICAgICAgICAgICAgaWYgKGN1cnJlbnQgPT09IG1vZGUpIHJldHVybiBwcmV2XG4gICAgICAgICAgICAgICAgY29uc3QgbmV4dCA9IHRyYW5zaXRpb25QZXJtaXNzaW9uTW9kZShcbiAgICAgICAgICAgICAgICAgIGN1cnJlbnQsXG4gICAgICAgICAgICAgICAgICBtb2RlLFxuICAgICAgICAgICAgICAgICAgcHJldi50b29sUGVybWlzc2lvbkNvbnRleHQsXG4gICAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgICAgICAuLi5wcmV2LFxuICAgICAgICAgICAgICAgICAgdG9vbFBlcm1pc3Npb25Db250ZXh0OiB7IC4uLm5leHQsIG1vZGUgfSxcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICAgIC8vIFJlY2hlY2sgcXVldWVkIHBlcm1pc3Npb24gcHJvbXB0cyBub3cgdGhhdCBtb2RlIGNoYW5nZWQuXG4gICAgICAgICAgICAgIHNldEltbWVkaWF0ZSgoKSA9PiB7XG4gICAgICAgICAgICAgICAgZ2V0TGVhZGVyVG9vbFVzZUNvbmZpcm1RdWV1ZSgpPy4oY3VycmVudFF1ZXVlID0+IHtcbiAgICAgICAgICAgICAgICAgIGN1cnJlbnRRdWV1ZS5mb3JFYWNoKGl0ZW0gPT4ge1xuICAgICAgICAgICAgICAgICAgICB2b2lkIGl0ZW0ucmVjaGVja1Blcm1pc3Npb24oKVxuICAgICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgICAgIHJldHVybiBjdXJyZW50UXVldWVcbiAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICByZXR1cm4geyBvazogdHJ1ZSB9XG4gICAgICAgICAgICB9LFxuICAgICAgICAgICAgb25TdGF0ZUNoYW5nZTogaGFuZGxlU3RhdGVDaGFuZ2UsXG4gICAgICAgICAgICBpbml0aWFsTWVzc2FnZXM6IG1lc3NhZ2VzLmxlbmd0aCA+IDAgPyBtZXNzYWdlcyA6IHVuZGVmaW5lZCxcbiAgICAgICAgICAgIGdldE1lc3NhZ2VzOiAoKSA9PiBtZXNzYWdlc1JlZi5jdXJyZW50LFxuICAgICAgICAgICAgcHJldmlvdXNseUZsdXNoZWRVVUlEczogZmx1c2hlZFVVSURzUmVmLmN1cnJlbnQsXG4gICAgICAgICAgICBpbml0aWFsTmFtZTogcmVwbEJyaWRnZUluaXRpYWxOYW1lLFxuICAgICAgICAgICAgcGVycGV0dWFsLFxuICAgICAgICAgIH0pXG4gICAgICAgICAgaWYgKGNhbmNlbGxlZCkge1xuICAgICAgICAgICAgLy8gRWZmZWN0IHdhcyBjYW5jZWxsZWQgd2hpbGUgaW5pdFJlcGxCcmlkZ2Ugd2FzIGluIGZsaWdodC5cbiAgICAgICAgICAgIC8vIFRlYXIgZG93biB0aGUgaGFuZGxlIHRvIGF2b2lkIGxlYWtpbmcgcmVzb3VyY2VzIChwb2xsIGxvb3AsXG4gICAgICAgICAgICAvLyBXZWJTb2NrZXQsIHJlZ2lzdGVyZWQgZW52aXJvbm1lbnQsIGNsZWFudXAgY2FsbGJhY2spLlxuICAgICAgICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICAgICAgICBgW2JyaWRnZTpyZXBsXSBIb29rOiBpbml0IGNhbmNlbGxlZCBkdXJpbmcgZmxpZ2h0LCB0ZWFyaW5nIGRvd24ke2hhbmRsZSA/IGAgZW52PSR7aGFuZGxlLmVudmlyb25tZW50SWR9YCA6ICcnfWAsXG4gICAgICAgICAgICApXG4gICAgICAgICAgICBpZiAoaGFuZGxlKSB7XG4gICAgICAgICAgICAgIHZvaWQgaGFuZGxlLnRlYXJkb3duKClcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIHJldHVyblxuICAgICAgICAgIH1cbiAgICAgICAgICBpZiAoIWhhbmRsZSkge1xuICAgICAgICAgICAgLy8gaW5pdFJlcGxCcmlkZ2UgcmV0dXJuZWQgbnVsbCDigJQgYSBwcmVjb25kaXRpb24gZmFpbGVkLiBGb3IgbW9zdFxuICAgICAgICAgICAgLy8gY2FzZXMgKG5vX29hdXRoLCBwb2xpY3lfZGVuaWVkLCBldGMuKSBvblN0YXRlQ2hhbmdlKCdmYWlsZWQnKVxuICAgICAgICAgICAgLy8gYWxyZWFkeSBmaXJlZCB3aXRoIGEgc3BlY2lmaWMgaGludC4gVGhlIEdyb3d0aEJvb2stZ2F0ZS1vZmYgY2FzZVxuICAgICAgICAgICAgLy8gaXMgaW50ZW50aW9uYWxseSBzaWxlbnQg4oCUIG5vdCBhIGZhaWx1cmUsIGp1c3Qgbm90IHJvbGxlZCBvdXQuXG4gICAgICAgICAgICBjb25zZWN1dGl2ZUZhaWx1cmVzUmVmLmN1cnJlbnQrK1xuICAgICAgICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICAgICAgICBgW2JyaWRnZTpyZXBsXSBJbml0IHJldHVybmVkIG51bGwgKHByZWNvbmRpdGlvbiBvciBzZXNzaW9uIGNyZWF0aW9uIGZhaWxlZCk7IGNvbnNlY3V0aXZlIGZhaWx1cmVzOiAke2NvbnNlY3V0aXZlRmFpbHVyZXNSZWYuY3VycmVudH1gLFxuICAgICAgICAgICAgKVxuICAgICAgICAgICAgY2xlYXJUaW1lb3V0KGZhaWx1cmVUaW1lb3V0UmVmLmN1cnJlbnQpXG4gICAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+ICh7XG4gICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgIHJlcGxCcmlkZ2VFcnJvcjpcbiAgICAgICAgICAgICAgICBwcmV2LnJlcGxCcmlkZ2VFcnJvciA/PyAnY2hlY2sgZGVidWcgbG9ncyBmb3IgZGV0YWlscycsXG4gICAgICAgICAgICB9KSlcbiAgICAgICAgICAgIGZhaWx1cmVUaW1lb3V0UmVmLmN1cnJlbnQgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICAgICAgaWYgKGNhbmNlbGxlZCkgcmV0dXJuXG4gICAgICAgICAgICAgIGZhaWx1cmVUaW1lb3V0UmVmLmN1cnJlbnQgPSB1bmRlZmluZWRcbiAgICAgICAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiB7XG4gICAgICAgICAgICAgICAgaWYgKCFwcmV2LnJlcGxCcmlkZ2VFcnJvcikgcmV0dXJuIHByZXZcbiAgICAgICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VFbmFibGVkOiBmYWxzZSxcbiAgICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VFcnJvcjogdW5kZWZpbmVkLFxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIH0sIEJSSURHRV9GQUlMVVJFX0RJU01JU1NfTVMpXG4gICAgICAgICAgICByZXR1cm5cbiAgICAgICAgICB9XG4gICAgICAgICAgaGFuZGxlUmVmLmN1cnJlbnQgPSBoYW5kbGVcbiAgICAgICAgICBzZXRSZXBsQnJpZGdlSGFuZGxlKGhhbmRsZSlcbiAgICAgICAgICBjb25zZWN1dGl2ZUZhaWx1cmVzUmVmLmN1cnJlbnQgPSAwXG4gICAgICAgICAgLy8gU2tpcCBpbml0aWFsIG1lc3NhZ2VzIGluIHRoZSBmb3J3YXJkaW5nIGVmZmVjdCDigJQgdGhleSB3ZXJlXG4gICAgICAgICAgLy8gYWxyZWFkeSBsb2FkZWQgYXMgc2Vzc2lvbiBldmVudHMgZHVyaW5nIGNyZWF0aW9uLlxuICAgICAgICAgIGxhc3RXcml0dGVuSW5kZXhSZWYuY3VycmVudCA9IGluaXRpYWxNZXNzYWdlQ291bnRcblxuICAgICAgICAgIGlmIChvdXRib3VuZE9ubHkpIHtcbiAgICAgICAgICAgIHNldEFwcFN0YXRlKHByZXYgPT4ge1xuICAgICAgICAgICAgICBpZiAoXG4gICAgICAgICAgICAgICAgcHJldi5yZXBsQnJpZGdlQ29ubmVjdGVkICYmXG4gICAgICAgICAgICAgICAgcHJldi5yZXBsQnJpZGdlU2Vzc2lvbklkID09PSBoYW5kbGUuYnJpZGdlU2Vzc2lvbklkXG4gICAgICAgICAgICAgIClcbiAgICAgICAgICAgICAgICByZXR1cm4gcHJldlxuICAgICAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgICAgcmVwbEJyaWRnZUNvbm5lY3RlZDogdHJ1ZSxcbiAgICAgICAgICAgICAgICByZXBsQnJpZGdlU2Vzc2lvbklkOiBoYW5kbGUuYnJpZGdlU2Vzc2lvbklkLFxuICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VTZXNzaW9uVXJsOiB1bmRlZmluZWQsXG4gICAgICAgICAgICAgICAgcmVwbEJyaWRnZUNvbm5lY3RVcmw6IHVuZGVmaW5lZCxcbiAgICAgICAgICAgICAgICByZXBsQnJpZGdlRXJyb3I6IHVuZGVmaW5lZCxcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgICAgICAgYFticmlkZ2U6cmVwbF0gTWlycm9yIGluaXRpYWxpemVkLCBzZXNzaW9uPSR7aGFuZGxlLmJyaWRnZVNlc3Npb25JZH1gLFxuICAgICAgICAgICAgKVxuICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAvLyBCdWlsZCBicmlkZ2UgcGVybWlzc2lvbiBjYWxsYmFja3Mgc28gdGhlIGludGVyYWN0aXZlIHBlcm1pc3Npb25cbiAgICAgICAgICAgIC8vIGhhbmRsZXIgY2FuIHJhY2UgYnJpZGdlIHJlc3BvbnNlcyBhZ2FpbnN0IGxvY2FsIHVzZXIgaW50ZXJhY3Rpb24uXG4gICAgICAgICAgICBjb25zdCBwZXJtaXNzaW9uQ2FsbGJhY2tzOiBCcmlkZ2VQZXJtaXNzaW9uQ2FsbGJhY2tzID0ge1xuICAgICAgICAgICAgICBzZW5kUmVxdWVzdChcbiAgICAgICAgICAgICAgICByZXF1ZXN0SWQsXG4gICAgICAgICAgICAgICAgdG9vbE5hbWUsXG4gICAgICAgICAgICAgICAgaW5wdXQsXG4gICAgICAgICAgICAgICAgdG9vbFVzZUlkLFxuICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgICAgICAgICAgIHBlcm1pc3Npb25TdWdnZXN0aW9ucyxcbiAgICAgICAgICAgICAgICBibG9ja2VkUGF0aCxcbiAgICAgICAgICAgICAgKSB7XG4gICAgICAgICAgICAgICAgaGFuZGxlLnNlbmRDb250cm9sUmVxdWVzdCh7XG4gICAgICAgICAgICAgICAgICB0eXBlOiAnY29udHJvbF9yZXF1ZXN0JyxcbiAgICAgICAgICAgICAgICAgIHJlcXVlc3RfaWQ6IHJlcXVlc3RJZCxcbiAgICAgICAgICAgICAgICAgIHJlcXVlc3Q6IHtcbiAgICAgICAgICAgICAgICAgICAgc3VidHlwZTogJ2Nhbl91c2VfdG9vbCcsXG4gICAgICAgICAgICAgICAgICAgIHRvb2xfbmFtZTogdG9vbE5hbWUsXG4gICAgICAgICAgICAgICAgICAgIGlucHV0LFxuICAgICAgICAgICAgICAgICAgICB0b29sX3VzZV9pZDogdG9vbFVzZUlkLFxuICAgICAgICAgICAgICAgICAgICBkZXNjcmlwdGlvbixcbiAgICAgICAgICAgICAgICAgICAgLi4uKHBlcm1pc3Npb25TdWdnZXN0aW9uc1xuICAgICAgICAgICAgICAgICAgICAgID8geyBwZXJtaXNzaW9uX3N1Z2dlc3Rpb25zOiBwZXJtaXNzaW9uU3VnZ2VzdGlvbnMgfVxuICAgICAgICAgICAgICAgICAgICAgIDoge30pLFxuICAgICAgICAgICAgICAgICAgICAuLi4oYmxvY2tlZFBhdGggPyB7IGJsb2NrZWRfcGF0aDogYmxvY2tlZFBhdGggfSA6IHt9KSxcbiAgICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgc2VuZFJlc3BvbnNlKHJlcXVlc3RJZCwgcmVzcG9uc2UpIHtcbiAgICAgICAgICAgICAgICBjb25zdCBwYXlsb2FkOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IHsgLi4ucmVzcG9uc2UgfVxuICAgICAgICAgICAgICAgIGhhbmRsZS5zZW5kQ29udHJvbFJlc3BvbnNlKHtcbiAgICAgICAgICAgICAgICAgIHR5cGU6ICdjb250cm9sX3Jlc3BvbnNlJyxcbiAgICAgICAgICAgICAgICAgIHJlc3BvbnNlOiB7XG4gICAgICAgICAgICAgICAgICAgIHN1YnR5cGU6ICdzdWNjZXNzJyxcbiAgICAgICAgICAgICAgICAgICAgcmVxdWVzdF9pZDogcmVxdWVzdElkLFxuICAgICAgICAgICAgICAgICAgICByZXNwb25zZTogcGF5bG9hZCxcbiAgICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgY2FuY2VsUmVxdWVzdChyZXF1ZXN0SWQpIHtcbiAgICAgICAgICAgICAgICBoYW5kbGUuc2VuZENvbnRyb2xDYW5jZWxSZXF1ZXN0KHJlcXVlc3RJZClcbiAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgb25SZXNwb25zZShyZXF1ZXN0SWQsIGhhbmRsZXIpIHtcbiAgICAgICAgICAgICAgICBwZW5kaW5nUGVybWlzc2lvbkhhbmRsZXJzLnNldChyZXF1ZXN0SWQsIGhhbmRsZXIpXG4gICAgICAgICAgICAgICAgcmV0dXJuICgpID0+IHtcbiAgICAgICAgICAgICAgICAgIHBlbmRpbmdQZXJtaXNzaW9uSGFuZGxlcnMuZGVsZXRlKHJlcXVlc3RJZClcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+ICh7XG4gICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgIHJlcGxCcmlkZ2VQZXJtaXNzaW9uQ2FsbGJhY2tzOiBwZXJtaXNzaW9uQ2FsbGJhY2tzLFxuICAgICAgICAgICAgfSkpXG4gICAgICAgICAgICBjb25zdCB1cmwgPSBnZXRSZW1vdGVTZXNzaW9uVXJsKFxuICAgICAgICAgICAgICBoYW5kbGUuYnJpZGdlU2Vzc2lvbklkLFxuICAgICAgICAgICAgICBoYW5kbGUuc2Vzc2lvbkluZ3Jlc3NVcmwsXG4gICAgICAgICAgICApXG4gICAgICAgICAgICAvLyBlbnZpcm9ubWVudElkID09PSAnJyBzaWduYWxzIHRoZSB2MiBlbnYtbGVzcyBwYXRoLiBidWlsZEJyaWRnZUNvbm5lY3RVcmxcbiAgICAgICAgICAgIC8vIGJ1aWxkcyBhbiBlbnYtc3BlY2lmaWMgY29ubmVjdCBVUkwsIHdoaWNoIGRvZXNuJ3QgZXhpc3Qgd2l0aG91dCBhbiBlbnYuXG4gICAgICAgICAgICBjb25zdCBoYXNFbnYgPSBoYW5kbGUuZW52aXJvbm1lbnRJZCAhPT0gJydcbiAgICAgICAgICAgIGNvbnN0IGNvbm5lY3RVcmwgPSBoYXNFbnZcbiAgICAgICAgICAgICAgPyBidWlsZEJyaWRnZUNvbm5lY3RVcmwoXG4gICAgICAgICAgICAgICAgICBoYW5kbGUuZW52aXJvbm1lbnRJZCxcbiAgICAgICAgICAgICAgICAgIGhhbmRsZS5zZXNzaW9uSW5ncmVzc1VybCxcbiAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgIDogdW5kZWZpbmVkXG4gICAgICAgICAgICBzZXRBcHBTdGF0ZShwcmV2ID0+IHtcbiAgICAgICAgICAgICAgaWYgKFxuICAgICAgICAgICAgICAgIHByZXYucmVwbEJyaWRnZUNvbm5lY3RlZCAmJlxuICAgICAgICAgICAgICAgIHByZXYucmVwbEJyaWRnZVNlc3Npb25VcmwgPT09IHVybFxuICAgICAgICAgICAgICApIHtcbiAgICAgICAgICAgICAgICByZXR1cm4gcHJldlxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICAgICAgICByZXBsQnJpZGdlQ29ubmVjdGVkOiB0cnVlLFxuICAgICAgICAgICAgICAgIHJlcGxCcmlkZ2VTZXNzaW9uVXJsOiB1cmwsXG4gICAgICAgICAgICAgICAgcmVwbEJyaWRnZUNvbm5lY3RVcmw6IGNvbm5lY3RVcmwgPz8gcHJldi5yZXBsQnJpZGdlQ29ubmVjdFVybCxcbiAgICAgICAgICAgICAgICByZXBsQnJpZGdlRW52aXJvbm1lbnRJZDogaGFuZGxlLmVudmlyb25tZW50SWQsXG4gICAgICAgICAgICAgICAgcmVwbEJyaWRnZVNlc3Npb25JZDogaGFuZGxlLmJyaWRnZVNlc3Npb25JZCxcbiAgICAgICAgICAgICAgICByZXBsQnJpZGdlRXJyb3I6IHVuZGVmaW5lZCxcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfSlcblxuICAgICAgICAgICAgLy8gU2hvdyBicmlkZ2Ugc3RhdHVzIHdpdGggVVJMIGluIHRoZSB0cmFuc2NyaXB0LiBwZXJwZXR1YWwgKEtBSVJPU1xuICAgICAgICAgICAgLy8gYXNzaXN0YW50IG1vZGUpIGZhbGxzIGJhY2sgdG8gdjEgYXQgaW5pdFJlcGxCcmlkZ2UudHMg4oCUIHNraXAgdGhlXG4gICAgICAgICAgICAvLyB2Mi1vbmx5IHVwZ3JhZGUgbnVkZ2UgZm9yIHRoZW0uIE93biB0cnkvY2F0Y2ggc28gYSBjb3NtZXRpY1xuICAgICAgICAgICAgLy8gR3Jvd3RoQm9vayBoaWNjdXAgZG9lc24ndCBoaXQgdGhlIG91dGVyIGluaXQtZmFpbHVyZSBoYW5kbGVyLlxuICAgICAgICAgICAgY29uc3QgdXBncmFkZU51ZGdlID0gIXBlcnBldHVhbFxuICAgICAgICAgICAgICA/IGF3YWl0IHNob3VsZFNob3dBcHBVcGdyYWRlTWVzc2FnZSgpLmNhdGNoKCgpID0+IGZhbHNlKVxuICAgICAgICAgICAgICA6IGZhbHNlXG4gICAgICAgICAgICBpZiAoY2FuY2VsbGVkKSByZXR1cm5cbiAgICAgICAgICAgIHNldE1lc3NhZ2VzKHByZXYgPT4gW1xuICAgICAgICAgICAgICAuLi5wcmV2LFxuICAgICAgICAgICAgICBjcmVhdGVCcmlkZ2VTdGF0dXNNZXNzYWdlKFxuICAgICAgICAgICAgICAgIHVybCxcbiAgICAgICAgICAgICAgICB1cGdyYWRlTnVkZ2VcbiAgICAgICAgICAgICAgICAgID8gJ1BsZWFzZSB1cGdyYWRlIHRvIHRoZSBsYXRlc3QgdmVyc2lvbiBvZiB0aGUgQ2xhdWRlIG1vYmlsZSBhcHAgdG8gc2VlIHlvdXIgUmVtb3RlIENvbnRyb2wgc2Vzc2lvbnMuJ1xuICAgICAgICAgICAgICAgICAgOiB1bmRlZmluZWQsXG4gICAgICAgICAgICAgICksXG4gICAgICAgICAgICBdKVxuXG4gICAgICAgICAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICAgICAgICAgIGBbYnJpZGdlOnJlcGxdIEhvb2sgaW5pdGlhbGl6ZWQsIHNlc3Npb249JHtoYW5kbGUuYnJpZGdlU2Vzc2lvbklkfWAsXG4gICAgICAgICAgICApXG4gICAgICAgICAgfVxuICAgICAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgICAgICAvLyBOZXZlciBjcmFzaCB0aGUgUkVQTCDigJQgc3VyZmFjZSB0aGUgZXJyb3IgaW4gdGhlIFVJLlxuICAgICAgICAgIC8vIENoZWNrIGNhbmNlbGxlZCBmaXJzdCAoc3ltbWV0cnkgd2l0aCB0aGUgIWhhbmRsZSBwYXRoIGF0IGxpbmUgfjM4Nik6XG4gICAgICAgICAgLy8gaWYgaW5pdFJlcGxCcmlkZ2UgdGhyZXcgZHVyaW5nIHJhcGlkIHRvZ2dsZS1vZmYgKGluLWZsaWdodCBuZXR3b3JrXG4gICAgICAgICAgLy8gZXJyb3IpLCBkb24ndCBjb3VudCB0aGF0IHRvd2FyZCB0aGUgZnVzZSBvciBzcGFtIGEgc3RhbGUgZXJyb3JcbiAgICAgICAgICAvLyBpbnRvIHRoZSBVSS4gQWxzbyBmaXhlcyBwcmUtZXhpc3Rpbmcgc3B1cmlvdXMgc2V0QXBwU3RhdGUvXG4gICAgICAgICAgLy8gc2V0TWVzc2FnZXMgb24gY2FuY2VsbGVkIHRocm93cy5cbiAgICAgICAgICBpZiAoY2FuY2VsbGVkKSByZXR1cm5cbiAgICAgICAgICBjb25zZWN1dGl2ZUZhaWx1cmVzUmVmLmN1cnJlbnQrK1xuICAgICAgICAgIGNvbnN0IGVyck1zZyA9IGVycm9yTWVzc2FnZShlcnIpXG4gICAgICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICAgICAgYFticmlkZ2U6cmVwbF0gSW5pdCBmYWlsZWQ6ICR7ZXJyTXNnfTsgY29uc2VjdXRpdmUgZmFpbHVyZXM6ICR7Y29uc2VjdXRpdmVGYWlsdXJlc1JlZi5jdXJyZW50fWAsXG4gICAgICAgICAgKVxuICAgICAgICAgIGNsZWFyVGltZW91dChmYWlsdXJlVGltZW91dFJlZi5jdXJyZW50KVxuICAgICAgICAgIG5vdGlmeUJyaWRnZUZhaWxlZChlcnJNc2cpXG4gICAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiAoe1xuICAgICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICAgIHJlcGxCcmlkZ2VFcnJvcjogZXJyTXNnLFxuICAgICAgICAgIH0pKVxuICAgICAgICAgIGZhaWx1cmVUaW1lb3V0UmVmLmN1cnJlbnQgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICAgIGlmIChjYW5jZWxsZWQpIHJldHVyblxuICAgICAgICAgICAgZmFpbHVyZVRpbWVvdXRSZWYuY3VycmVudCA9IHVuZGVmaW5lZFxuICAgICAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiB7XG4gICAgICAgICAgICAgIGlmICghcHJldi5yZXBsQnJpZGdlRXJyb3IpIHJldHVybiBwcmV2XG4gICAgICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICAgICAgLi4ucHJldixcbiAgICAgICAgICAgICAgICByZXBsQnJpZGdlRW5hYmxlZDogZmFsc2UsXG4gICAgICAgICAgICAgICAgcmVwbEJyaWRnZUVycm9yOiB1bmRlZmluZWQsXG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgfSwgQlJJREdFX0ZBSUxVUkVfRElTTUlTU19NUylcbiAgICAgICAgICBpZiAoIW91dGJvdW5kT25seSkge1xuICAgICAgICAgICAgc2V0TWVzc2FnZXMocHJldiA9PiBbXG4gICAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICAgIGNyZWF0ZVN5c3RlbU1lc3NhZ2UoXG4gICAgICAgICAgICAgICAgYFJlbW90ZSBDb250cm9sIGZhaWxlZCB0byBjb25uZWN0OiAke2Vyck1zZ31gLFxuICAgICAgICAgICAgICAgICd3YXJuaW5nJyxcbiAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgIF0pXG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9KSgpXG5cbiAgICAgIHJldHVybiAoKSA9PiB7XG4gICAgICAgIGNhbmNlbGxlZCA9IHRydWVcbiAgICAgICAgY2xlYXJUaW1lb3V0KGZhaWx1cmVUaW1lb3V0UmVmLmN1cnJlbnQpXG4gICAgICAgIGZhaWx1cmVUaW1lb3V0UmVmLmN1cnJlbnQgPSB1bmRlZmluZWRcbiAgICAgICAgaWYgKGhhbmRsZVJlZi5jdXJyZW50KSB7XG4gICAgICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICAgICAgYFticmlkZ2U6cmVwbF0gSG9vayBjbGVhbnVwOiBzdGFydGluZyB0ZWFyZG93biBmb3IgZW52PSR7aGFuZGxlUmVmLmN1cnJlbnQuZW52aXJvbm1lbnRJZH0gc2Vzc2lvbj0ke2hhbmRsZVJlZi5jdXJyZW50LmJyaWRnZVNlc3Npb25JZH1gLFxuICAgICAgICAgIClcbiAgICAgICAgICB0ZWFyZG93blByb21pc2VSZWYuY3VycmVudCA9IGhhbmRsZVJlZi5jdXJyZW50LnRlYXJkb3duKClcbiAgICAgICAgICBoYW5kbGVSZWYuY3VycmVudCA9IG51bGxcbiAgICAgICAgICBzZXRSZXBsQnJpZGdlSGFuZGxlKG51bGwpXG4gICAgICAgIH1cbiAgICAgICAgc2V0QXBwU3RhdGUocHJldiA9PiB7XG4gICAgICAgICAgaWYgKFxuICAgICAgICAgICAgIXByZXYucmVwbEJyaWRnZUNvbm5lY3RlZCAmJlxuICAgICAgICAgICAgIXByZXYucmVwbEJyaWRnZVNlc3Npb25BY3RpdmUgJiZcbiAgICAgICAgICAgICFwcmV2LnJlcGxCcmlkZ2VFcnJvclxuICAgICAgICAgICkge1xuICAgICAgICAgICAgcmV0dXJuIHByZXZcbiAgICAgICAgICB9XG4gICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgIC4uLnByZXYsXG4gICAgICAgICAgICByZXBsQnJpZGdlQ29ubmVjdGVkOiBmYWxzZSxcbiAgICAgICAgICAgIHJlcGxCcmlkZ2VTZXNzaW9uQWN0aXZlOiBmYWxzZSxcbiAgICAgICAgICAgIHJlcGxCcmlkZ2VSZWNvbm5lY3Rpbmc6IGZhbHNlLFxuICAgICAgICAgICAgcmVwbEJyaWRnZUNvbm5lY3RVcmw6IHVuZGVmaW5lZCxcbiAgICAgICAgICAgIHJlcGxCcmlkZ2VTZXNzaW9uVXJsOiB1bmRlZmluZWQsXG4gICAgICAgICAgICByZXBsQnJpZGdlRW52aXJvbm1lbnRJZDogdW5kZWZpbmVkLFxuICAgICAgICAgICAgcmVwbEJyaWRnZVNlc3Npb25JZDogdW5kZWZpbmVkLFxuICAgICAgICAgICAgcmVwbEJyaWRnZUVycm9yOiB1bmRlZmluZWQsXG4gICAgICAgICAgICByZXBsQnJpZGdlUGVybWlzc2lvbkNhbGxiYWNrczogdW5kZWZpbmVkLFxuICAgICAgICAgIH1cbiAgICAgICAgfSlcbiAgICAgICAgbGFzdFdyaXR0ZW5JbmRleFJlZi5jdXJyZW50ID0gMFxuICAgICAgfVxuICAgIH1cbiAgfSwgW1xuICAgIHJlcGxCcmlkZ2VFbmFibGVkLFxuICAgIHJlcGxCcmlkZ2VPdXRib3VuZE9ubHksXG4gICAgc2V0QXBwU3RhdGUsXG4gICAgc2V0TWVzc2FnZXMsXG4gICAgYWRkTm90aWZpY2F0aW9uLFxuICBdKVxuXG4gIC8vIFdyaXRlIG5ldyBtZXNzYWdlcyBhcyB0aGV5IGFwcGVhci5cbiAgLy8gQWxzbyByZS1ydW5zIHdoZW4gcmVwbEJyaWRnZUNvbm5lY3RlZCBjaGFuZ2VzIChicmlkZ2UgZmluaXNoZXMgaW5pdCksXG4gIC8vIHNvIGFueSBtZXNzYWdlcyB0aGF0IGFycml2ZWQgYmVmb3JlIHRoZSBicmlkZ2Ugd2FzIHJlYWR5IGdldCB3cml0dGVuLlxuICB1c2VFZmZlY3QoKCkgPT4ge1xuICAgIC8vIFBvc2l0aXZlIGZlYXR1cmUoKSBndWFyZCDigJQgc2VlIGZpcnN0IHVzZUVmZmVjdCBjb21tZW50XG4gICAgaWYgKGZlYXR1cmUoJ0JSSURHRV9NT0RFJykpIHtcbiAgICAgIGlmICghcmVwbEJyaWRnZUNvbm5lY3RlZCkgcmV0dXJuXG5cbiAgICAgIGNvbnN0IGhhbmRsZSA9IGhhbmRsZVJlZi5jdXJyZW50XG4gICAgICBpZiAoIWhhbmRsZSkgcmV0dXJuXG5cbiAgICAgIC8vIENsYW1wIHRoZSBpbmRleCBpbiBjYXNlIG1lc3NhZ2VzIHdlcmUgY29tcGFjdGVkIChhcnJheSBzaG9ydGVuZWQpLlxuICAgICAgLy8gQWZ0ZXIgY29tcGFjdGlvbiB0aGUgcmVmIGNvdWxkIGV4Y2VlZCBtZXNzYWdlcy5sZW5ndGgsIGFuZCB3aXRob3V0XG4gICAgICAvLyBjbGFtcGluZyBubyBuZXcgbWVzc2FnZXMgd291bGQgYmUgZm9yd2FyZGVkLlxuICAgICAgaWYgKGxhc3RXcml0dGVuSW5kZXhSZWYuY3VycmVudCA+IG1lc3NhZ2VzLmxlbmd0aCkge1xuICAgICAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICAgICAgYFticmlkZ2U6cmVwbF0gQ29tcGFjdGlvbiBkZXRlY3RlZDogbGFzdFdyaXR0ZW5JbmRleD0ke2xhc3RXcml0dGVuSW5kZXhSZWYuY3VycmVudH0gPiBtZXNzYWdlcy5sZW5ndGg9JHttZXNzYWdlcy5sZW5ndGh9LCBjbGFtcGluZ2AsXG4gICAgICAgIClcbiAgICAgIH1cbiAgICAgIGNvbnN0IHN0YXJ0SW5kZXggPSBNYXRoLm1pbihsYXN0V3JpdHRlbkluZGV4UmVmLmN1cnJlbnQsIG1lc3NhZ2VzLmxlbmd0aClcblxuICAgICAgLy8gQ29sbGVjdCBuZXcgbWVzc2FnZXMgc2luY2UgbGFzdCB3cml0ZVxuICAgICAgY29uc3QgbmV3TWVzc2FnZXM6IE1lc3NhZ2VbXSA9IFtdXG4gICAgICBmb3IgKGxldCBpID0gc3RhcnRJbmRleDsgaSA8IG1lc3NhZ2VzLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgIGNvbnN0IG1zZyA9IG1lc3NhZ2VzW2ldXG4gICAgICAgIGlmIChcbiAgICAgICAgICBtc2cgJiZcbiAgICAgICAgICAobXNnLnR5cGUgPT09ICd1c2VyJyB8fFxuICAgICAgICAgICAgbXNnLnR5cGUgPT09ICdhc3Npc3RhbnQnIHx8XG4gICAgICAgICAgICAobXNnLnR5cGUgPT09ICdzeXN0ZW0nICYmIG1zZy5zdWJ0eXBlID09PSAnbG9jYWxfY29tbWFuZCcpKVxuICAgICAgICApIHtcbiAgICAgICAgICBuZXdNZXNzYWdlcy5wdXNoKG1zZylcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgbGFzdFdyaXR0ZW5JbmRleFJlZi5jdXJyZW50ID0gbWVzc2FnZXMubGVuZ3RoXG5cbiAgICAgIGlmIChuZXdNZXNzYWdlcy5sZW5ndGggPiAwKSB7XG4gICAgICAgIGhhbmRsZS53cml0ZU1lc3NhZ2VzKG5ld01lc3NhZ2VzKVxuICAgICAgfVxuICAgIH1cbiAgfSwgW21lc3NhZ2VzLCByZXBsQnJpZGdlQ29ubmVjdGVkXSlcblxuICBjb25zdCBzZW5kQnJpZGdlUmVzdWx0ID0gdXNlQ2FsbGJhY2soKCkgPT4ge1xuICAgIGlmIChmZWF0dXJlKCdCUklER0VfTU9ERScpKSB7XG4gICAgICBoYW5kbGVSZWYuY3VycmVudD8uc2VuZFJlc3VsdCgpXG4gICAgfVxuICB9LCBbXSlcblxuICByZXR1cm4geyBzZW5kQnJpZGdlUmVzdWx0IH1cbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsU0FBU0EsT0FBTyxRQUFRLFlBQVk7QUFDcEMsT0FBT0Ms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