/ keybindings / KeybindingProviderSetup.tsx
KeybindingProviderSetup.tsx
  1  import { c as _c } from "react/compiler-runtime";
  2  /**
  3   * Setup utilities for integrating KeybindingProvider into the app.
  4   *
  5   * This file provides the bindings and a composed provider that can be
  6   * added to the app's component tree. It loads both default bindings and
  7   * user-defined bindings from ~/.claude/keybindings.json, with hot-reload
  8   * support when the file changes.
  9   */
 10  import React, { useCallback, useEffect, useRef, useState } from 'react';
 11  import { useNotifications } from '../context/notifications.js';
 12  import type { InputEvent } from '../ink/events/input-event.js';
 13  // ChordInterceptor intentionally uses useInput to intercept all keystrokes before
 14  // other handlers process them - this is required for chord sequence support
 15  // eslint-disable-next-line custom-rules/prefer-use-keybindings
 16  import { type Key, useInput } from '../ink.js';
 17  import { count } from '../utils/array.js';
 18  import { logForDebugging } from '../utils/debug.js';
 19  import { plural } from '../utils/stringUtils.js';
 20  import { KeybindingProvider } from './KeybindingContext.js';
 21  import { initializeKeybindingWatcher, type KeybindingsLoadResult, loadKeybindingsSyncWithWarnings, subscribeToKeybindingChanges } from './loadUserBindings.js';
 22  import { resolveKeyWithChordState } from './resolver.js';
 23  import type { KeybindingContextName, ParsedBinding, ParsedKeystroke } from './types.js';
 24  import type { KeybindingWarning } from './validate.js';
 25  
 26  /**
 27   * Timeout for chord sequences in milliseconds.
 28   * If the user doesn't complete the chord within this time, it's cancelled.
 29   */
 30  const CHORD_TIMEOUT_MS = 1000;
 31  type Props = {
 32    children: React.ReactNode;
 33  };
 34  
 35  /**
 36   * Keybinding provider with default + user bindings and hot-reload support.
 37   *
 38   * Usage: Wrap your app with this provider to enable keybinding support.
 39   *
 40   * ```tsx
 41   * <AppStateProvider>
 42   *   <KeybindingSetup>
 43   *     <REPL ... />
 44   *   </KeybindingSetup>
 45   * </AppStateProvider>
 46   * ```
 47   *
 48   * Features:
 49   * - Loads default bindings from code
 50   * - Merges with user bindings from ~/.claude/keybindings.json
 51   * - Watches for file changes and reloads automatically (hot-reload)
 52   * - User bindings override defaults (later entries win)
 53   * - Chord support with automatic timeout
 54   */
 55  /**
 56   * Display keybinding warnings to the user via notifications.
 57   * Shows a brief message pointing to /doctor for details.
 58   */
 59  function useKeybindingWarnings(warnings, isReload) {
 60    const $ = _c(9);
 61    const {
 62      addNotification,
 63      removeNotification
 64    } = useNotifications();
 65    let t0;
 66    if ($[0] !== addNotification || $[1] !== removeNotification || $[2] !== warnings) {
 67      t0 = () => {
 68        if (warnings.length === 0) {
 69          removeNotification("keybinding-config-warning");
 70          return;
 71        }
 72        const errorCount = count(warnings, _temp);
 73        const warnCount = count(warnings, _temp2);
 74        let message;
 75        if (errorCount > 0 && warnCount > 0) {
 76          message = `Found ${errorCount} keybinding ${plural(errorCount, "error")} and ${warnCount} ${plural(warnCount, "warning")}`;
 77        } else {
 78          if (errorCount > 0) {
 79            message = `Found ${errorCount} keybinding ${plural(errorCount, "error")}`;
 80          } else {
 81            message = `Found ${warnCount} keybinding ${plural(warnCount, "warning")}`;
 82          }
 83        }
 84        message = message + " \xB7 /doctor for details";
 85        addNotification({
 86          key: "keybinding-config-warning",
 87          text: message,
 88          color: errorCount > 0 ? "error" : "warning",
 89          priority: errorCount > 0 ? "immediate" : "high",
 90          timeoutMs: 60000
 91        });
 92      };
 93      $[0] = addNotification;
 94      $[1] = removeNotification;
 95      $[2] = warnings;
 96      $[3] = t0;
 97    } else {
 98      t0 = $[3];
 99    }
100    let t1;
101    if ($[4] !== addNotification || $[5] !== isReload || $[6] !== removeNotification || $[7] !== warnings) {
102      t1 = [warnings, isReload, addNotification, removeNotification];
103      $[4] = addNotification;
104      $[5] = isReload;
105      $[6] = removeNotification;
106      $[7] = warnings;
107      $[8] = t1;
108    } else {
109      t1 = $[8];
110    }
111    useEffect(t0, t1);
112  }
113  function _temp2(w_0) {
114    return w_0.severity === "warning";
115  }
116  function _temp(w) {
117    return w.severity === "error";
118  }
119  export function KeybindingSetup({
120    children
121  }: Props): React.ReactNode {
122    // Load bindings synchronously for initial render
123    const [{
124      bindings,
125      warnings
126    }, setLoadResult] = useState<KeybindingsLoadResult>(() => {
127      const result = loadKeybindingsSyncWithWarnings();
128      logForDebugging(`[keybindings] KeybindingSetup initialized with ${result.bindings.length} bindings, ${result.warnings.length} warnings`);
129      return result;
130    });
131  
132    // Track if this is a reload (not initial load)
133    const [isReload, setIsReload] = useState(false);
134  
135    // Display warnings via notifications
136    useKeybindingWarnings(warnings, isReload);
137  
138    // Chord state management - use ref for immediate access, state for re-renders
139    // The ref is used by resolve() to get the current value without waiting for re-render
140    // The state is used to trigger re-renders when needed (e.g., for UI updates)
141    const pendingChordRef = useRef<ParsedKeystroke[] | null>(null);
142    const [pendingChord, setPendingChordState] = useState<ParsedKeystroke[] | null>(null);
143    const chordTimeoutRef = useRef<NodeJS.Timeout | null>(null);
144  
145    // Handler registry for action callbacks (used by ChordInterceptor to invoke handlers)
146    const handlerRegistryRef = useRef(new Map<string, Set<{
147      action: string;
148      context: KeybindingContextName;
149      handler: () => void;
150    }>>());
151  
152    // Active context tracking for keybinding priority resolution
153    // Using a ref instead of state for synchronous updates - input handlers need
154    // to see the current value immediately, not after a React render cycle.
155    const activeContextsRef = useRef<Set<KeybindingContextName>>(new Set());
156    const registerActiveContext = useCallback((context: KeybindingContextName) => {
157      activeContextsRef.current.add(context);
158    }, []);
159    const unregisterActiveContext = useCallback((context_0: KeybindingContextName) => {
160      activeContextsRef.current.delete(context_0);
161    }, []);
162  
163    // Clear chord timeout when component unmounts or chord changes
164    const clearChordTimeout = useCallback(() => {
165      if (chordTimeoutRef.current) {
166        clearTimeout(chordTimeoutRef.current);
167        chordTimeoutRef.current = null;
168      }
169    }, []);
170  
171    // Wrapper for setPendingChord that manages timeout and syncs ref+state
172    const setPendingChord = useCallback((pending: ParsedKeystroke[] | null) => {
173      clearChordTimeout();
174      if (pending !== null) {
175        // Set timeout to cancel chord if not completed
176        chordTimeoutRef.current = setTimeout((pendingChordRef_0, setPendingChordState_0) => {
177          logForDebugging('[keybindings] Chord timeout - cancelling');
178          pendingChordRef_0.current = null;
179          setPendingChordState_0(null);
180        }, CHORD_TIMEOUT_MS, pendingChordRef, setPendingChordState);
181      }
182  
183      // Update ref immediately for synchronous access in resolve()
184      pendingChordRef.current = pending;
185      // Update state to trigger re-renders for UI updates
186      setPendingChordState(pending);
187    }, [clearChordTimeout]);
188    useEffect(() => {
189      // Initialize file watcher (idempotent - only runs once)
190      void initializeKeybindingWatcher();
191  
192      // Subscribe to changes
193      const unsubscribe = subscribeToKeybindingChanges(result_0 => {
194        // Any callback invocation is a reload since initial load happens
195        // synchronously in useState, not via this subscription
196        setIsReload(true);
197        setLoadResult(result_0);
198        logForDebugging(`[keybindings] Reloaded: ${result_0.bindings.length} bindings, ${result_0.warnings.length} warnings`);
199      });
200      return () => {
201        unsubscribe();
202        clearChordTimeout();
203      };
204    }, [clearChordTimeout]);
205    return <KeybindingProvider bindings={bindings} pendingChordRef={pendingChordRef} pendingChord={pendingChord} setPendingChord={setPendingChord} activeContexts={activeContextsRef.current} registerActiveContext={registerActiveContext} unregisterActiveContext={unregisterActiveContext} handlerRegistryRef={handlerRegistryRef}>
206        <ChordInterceptor bindings={bindings} pendingChordRef={pendingChordRef} setPendingChord={setPendingChord} activeContexts={activeContextsRef.current} handlerRegistryRef={handlerRegistryRef} />
207        {children}
208      </KeybindingProvider>;
209  }
210  
211  /**
212   * Global chord interceptor that registers useInput FIRST (before children).
213   *
214   * This component intercepts keystrokes that are part of chord sequences and
215   * stops propagation before other handlers (like PromptInput) can see them.
216   *
217   * Without this, the second key of a chord (e.g., 'r' in "ctrl+c r") would be
218   * captured by PromptInput and added to the input field before the keybinding
219   * system could recognize it as completing a chord.
220   */
221  type HandlerRegistration = {
222    action: string;
223    context: KeybindingContextName;
224    handler: () => void;
225  };
226  function ChordInterceptor(t0) {
227    const $ = _c(6);
228    const {
229      bindings,
230      pendingChordRef,
231      setPendingChord,
232      activeContexts,
233      handlerRegistryRef
234    } = t0;
235    let t1;
236    if ($[0] !== activeContexts || $[1] !== bindings || $[2] !== handlerRegistryRef || $[3] !== pendingChordRef || $[4] !== setPendingChord) {
237      t1 = (input, key, event) => {
238        if ((key.wheelUp || key.wheelDown) && pendingChordRef.current === null) {
239          return;
240        }
241        const registry = handlerRegistryRef.current;
242        const handlerContexts = new Set();
243        if (registry) {
244          for (const handlers of registry.values()) {
245            for (const registration of handlers) {
246              handlerContexts.add(registration.context);
247            }
248          }
249        }
250        const contexts = [...handlerContexts, ...activeContexts, "Global"];
251        const wasInChord = pendingChordRef.current !== null;
252        const result = resolveKeyWithChordState(input, key, contexts, bindings, pendingChordRef.current);
253        bb23: switch (result.type) {
254          case "chord_started":
255            {
256              setPendingChord(result.pending);
257              event.stopImmediatePropagation();
258              break bb23;
259            }
260          case "match":
261            {
262              setPendingChord(null);
263              if (wasInChord) {
264                const contextsSet = new Set(contexts);
265                if (registry) {
266                  const handlers_0 = registry.get(result.action);
267                  if (handlers_0 && handlers_0.size > 0) {
268                    for (const registration_0 of handlers_0) {
269                      if (contextsSet.has(registration_0.context)) {
270                        registration_0.handler();
271                        event.stopImmediatePropagation();
272                        break;
273                      }
274                    }
275                  }
276                }
277              }
278              break bb23;
279            }
280          case "chord_cancelled":
281            {
282              setPendingChord(null);
283              event.stopImmediatePropagation();
284              break bb23;
285            }
286          case "unbound":
287            {
288              setPendingChord(null);
289              event.stopImmediatePropagation();
290              break bb23;
291            }
292          case "none":
293        }
294      };
295      $[0] = activeContexts;
296      $[1] = bindings;
297      $[2] = handlerRegistryRef;
298      $[3] = pendingChordRef;
299      $[4] = setPendingChord;
300      $[5] = t1;
301    } else {
302      t1 = $[5];
303    }
304    const handleInput = t1;
305    useInput(handleInput);
306    return null;
307  }
308  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsInVzZUNhbGxiYWNrIiwidXNlRWZmZWN0IiwidXNlUmVmIiwidXNlU3RhdGUiLCJ1c2VOb3RpZmljYXRpb25zIiwiSW5wdXRFdmVudCIsIktleSIsInVzZUlucHV0IiwiY291bnQiLCJsb2dGb3JEZWJ1Z2dpbmciLCJwbHVyYWwiLCJLZXliaW5kaW5nUHJvdmlkZXIiLCJpbml0aWFsaXplS2V5YmluZGluZ1dhdGNoZXIiLCJLZXliaW5kaW5nc0xvYWRSZXN1bHQiLCJsb2FkS2V5YmluZGluZ3NTeW5jV2l0aFdhcm5pbmdzIiwic3Vic2NyaWJlVG9LZXliaW5kaW5nQ2hhbmdlcyIsInJlc29sdmVLZXlXaXRoQ2hvcmRTdGF0ZSIsIktleWJpbmRpbmdDb250ZXh0TmFtZSIsIlBhcnNlZEJpbmRpbmciLCJQYXJzZWRLZXlzdHJva2UiLCJLZXliaW5kaW5nV2FybmluZyIsIkNIT1JEX1RJTUVPVVRfTVMiLCJQcm9wcyIsImNoaWxkcmVuIiwiUmVhY3ROb2RlIiwidXNlS2V5YmluZGluZ1dhcm5pbmdzIiwid2FybmluZ3MiLCJpc1JlbG9hZCIsIiQiLCJfYyIsImFkZE5vdGlmaWNhdGlvbiIsInJlbW92ZU5vdGlmaWNhdGlvbiIsInQwIiwibGVuZ3RoIiwiZXJyb3JDb3VudCIsIl90ZW1wIiwid2FybkNvdW50IiwiX3RlbXAyIiwibWVzc2FnZSIsImtleSIsInRleHQiLCJjb2xvciIsInByaW9yaXR5IiwidGltZW91dE1zIiwidDEiLCJ3XzAiLCJ3Iiwic2V2ZXJpdHkiLCJLZXliaW5kaW5nU2V0dXAiLCJiaW5kaW5ncyIsInNldExvYWRSZXN1bHQiLCJyZXN1bHQiLCJzZXRJc1JlbG9hZCIsInBlbmRpbmdDaG9yZFJlZiIsInBlbmRpbmdDaG9yZCIsInNldFBlbmRpbmdDaG9yZFN0YXRlIiwiY2hvcmRUaW1lb3V0UmVmIiwiTm9kZUpTIiwiVGltZW91dCIsImhhbmRsZXJSZWdpc3RyeVJlZiIsIk1hcCIsIlNldCIsImFjdGlvbiIsImNvbnRleHQiLCJoYW5kbGVyIiwiYWN0aXZlQ29udGV4dHNSZWYiLCJyZWdpc3RlckFjdGl2ZUNvbnRleHQiLCJjdXJyZW50IiwiYWRkIiwidW5yZWdpc3RlckFjdGl2ZUNvbnRleHQiLCJkZWxldGUiLCJjbGVhckNob3JkVGltZW91dCIsImNsZWFyVGltZW91dCIsInNldFBlbmRpbmdDaG9yZCIsInBlbmRpbmciLCJzZXRUaW1lb3V0IiwidW5zdWJzY3JpYmUiLCJIYW5kbGVyUmVnaXN0cmF0aW9uIiwiQ2hvcmRJbnRlcmNlcHRvciIsImFjdGl2ZUNvbnRleHRzIiwiaW5wdXQiLCJldmVudCIsIndoZWVsVXAiLCJ3aGVlbERvd24iLCJyZWdpc3RyeSIsImhhbmRsZXJDb250ZXh0cyIsImhhbmRsZXJzIiwidmFsdWVzIiwicmVnaXN0cmF0aW9uIiwiY29udGV4dHMiLCJ3YXNJbkNob3JkIiwiYmIyMyIsInR5cGUiLCJzdG9wSW1tZWRpYXRlUHJvcGFnYXRpb24iLCJjb250ZXh0c1NldCIsImhhbmRsZXJzXzAiLCJnZXQiLCJzaXplIiwicmVnaXN0cmF0aW9uXzAiLCJoYXMiLCJoYW5kbGVJbnB1dCJdLCJzb3VyY2VzIjpbIktleWJpbmRpbmdQcm92aWRlclNldHVwLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFNldHVwIHV0aWxpdGllcyBmb3IgaW50ZWdyYXRpbmcgS2V5YmluZGluZ1Byb3ZpZGVyIGludG8gdGhlIGFwcC5cbiAqXG4gKiBUaGlzIGZpbGUgcHJvdmlkZXMgdGhlIGJpbmRpbmdzIGFuZCBhIGNvbXBvc2VkIHByb3ZpZGVyIHRoYXQgY2FuIGJlXG4gKiBhZGRlZCB0byB0aGUgYXBwJ3MgY29tcG9uZW50IHRyZWUuIEl0IGxvYWRzIGJvdGggZGVmYXVsdCBiaW5kaW5ncyBhbmRcbiAqIHVzZXItZGVmaW5lZCBiaW5kaW5ncyBmcm9tIH4vLmNsYXVkZS9rZXliaW5kaW5ncy5qc29uLCB3aXRoIGhvdC1yZWxvYWRcbiAqIHN1cHBvcnQgd2hlbiB0aGUgZmlsZSBjaGFuZ2VzLlxuICovXG5pbXBvcnQgUmVhY3QsIHsgdXNlQ2FsbGJhY2ssIHVzZUVmZmVjdCwgdXNlUmVmLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHsgdXNlTm90aWZpY2F0aW9ucyB9IGZyb20gJy4uL2NvbnRleHQvbm90aWZpY2F0aW9ucy5qcydcbmltcG9ydCB0eXBlIHsgSW5wdXRFdmVudCB9IGZyb20gJy4uL2luay9ldmVudHMvaW5wdXQtZXZlbnQuanMnXG4vLyBDaG9yZEludGVyY2VwdG9yIGludGVudGlvbmFsbHkgdXNlcyB1c2VJbnB1dCB0byBpbnRlcmNlcHQgYWxsIGtleXN0cm9rZXMgYmVmb3JlXG4vLyBvdGhlciBoYW5kbGVycyBwcm9jZXNzIHRoZW0gLSB0aGlzIGlzIHJlcXVpcmVkIGZvciBjaG9yZCBzZXF1ZW5jZSBzdXBwb3J0XG4vLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgY3VzdG9tLXJ1bGVzL3ByZWZlci11c2Uta2V5YmluZGluZ3NcbmltcG9ydCB7IHR5cGUgS2V5LCB1c2VJbnB1dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB7IGNvdW50IH0gZnJvbSAnLi4vdXRpbHMvYXJyYXkuanMnXG5pbXBvcnQgeyBsb2dGb3JEZWJ1Z2dpbmcgfSBmcm9tICcuLi91dGlscy9kZWJ1Zy5qcydcbmltcG9ydCB7IHBsdXJhbCB9IGZyb20gJy4uL3V0aWxzL3N0cmluZ1V0aWxzLmpzJ1xuaW1wb3J0IHsgS2V5YmluZGluZ1Byb3ZpZGVyIH0gZnJvbSAnLi9LZXliaW5kaW5nQ29udGV4dC5qcydcbmltcG9ydCB7XG4gIGluaXRpYWxpemVLZXliaW5kaW5nV2F0Y2hlcixcbiAgdHlwZSBLZXliaW5kaW5nc0xvYWRSZXN1bHQsXG4gIGxvYWRLZXliaW5kaW5nc1N5bmNXaXRoV2FybmluZ3MsXG4gIHN1YnNjcmliZVRvS2V5YmluZGluZ0NoYW5nZXMsXG59IGZyb20gJy4vbG9hZFVzZXJCaW5kaW5ncy5qcydcbmltcG9ydCB7IHJlc29sdmVLZXlXaXRoQ2hvcmRTdGF0ZSB9IGZyb20gJy4vcmVzb2x2ZXIuanMnXG5pbXBvcnQgdHlwZSB7XG4gIEtleWJpbmRpbmdDb250ZXh0TmFtZSxcbiAgUGFyc2VkQmluZGluZyxcbiAgUGFyc2VkS2V5c3Ryb2tlLFxufSBmcm9tICcuL3R5cGVzLmpzJ1xuaW1wb3J0IHR5cGUgeyBLZXliaW5kaW5nV2FybmluZyB9IGZyb20gJy4vdmFsaWRhdGUuanMnXG5cbi8qKlxuICogVGltZW91dCBmb3IgY2hvcmQgc2VxdWVuY2VzIGluIG1pbGxpc2Vjb25kcy5cbiAqIElmIHRoZSB1c2VyIGRvZXNuJ3QgY29tcGxldGUgdGhlIGNob3JkIHdpdGhpbiB0aGlzIHRpbWUsIGl0J3MgY2FuY2VsbGVkLlxuICovXG5jb25zdCBDSE9SRF9USU1FT1VUX01TID0gMTAwMFxuXG50eXBlIFByb3BzID0ge1xuICBjaGlsZHJlbjogUmVhY3QuUmVhY3ROb2RlXG59XG5cbi8qKlxuICogS2V5YmluZGluZyBwcm92aWRlciB3aXRoIGRlZmF1bHQgKyB1c2VyIGJpbmRpbmdzIGFuZCBob3QtcmVsb2FkIHN1cHBvcnQuXG4gKlxuICogVXNhZ2U6IFdyYXAgeW91ciBhcHAgd2l0aCB0aGlzIHByb3ZpZGVyIHRvIGVuYWJsZSBrZXliaW5kaW5nIHN1cHBvcnQuXG4gKlxuICogYGBgdHN4XG4gKiA8QXBwU3RhdGVQcm92aWRlcj5cbiAqICAgPEtleWJpbmRpbmdTZXR1cD5cbiAqICAgICA8UkVQTCAuLi4gLz5cbiAqICAgPC9LZXliaW5kaW5nU2V0dXA+XG4gKiA8L0FwcFN0YXRlUHJvdmlkZXI+XG4gKiBgYGBcbiAqXG4gKiBGZWF0dXJlczpcbiAqIC0gTG9hZHMgZGVmYXVsdCBiaW5kaW5ncyBmcm9tIGNvZGVcbiAqIC0gTWVyZ2VzIHdpdGggdXNlciBiaW5kaW5ncyBmcm9tIH4vLmNsYXVkZS9rZXliaW5kaW5ncy5qc29uXG4gKiAtIFdhdGNoZXMgZm9yIGZpbGUgY2hhbmdlcyBhbmQgcmVsb2FkcyBhdXRvbWF0aWNhbGx5IChob3QtcmVsb2FkKVxuICogLSBVc2VyIGJpbmRpbmdzIG92ZXJyaWRlIGRlZmF1bHRzIChsYXRlciBlbnRyaWVzIHdpbilcbiAqIC0gQ2hvcmQgc3VwcG9ydCB3aXRoIGF1dG9tYXRpYyB0aW1lb3V0XG4gKi9cbi8qKlxuICogRGlzcGxheSBrZXliaW5kaW5nIHdhcm5pbmdzIHRvIHRoZSB1c2VyIHZpYSBub3RpZmljYXRpb25zLlxuICogU2hvd3MgYSBicmllZiBtZXNzYWdlIHBvaW50aW5nIHRvIC9kb2N0b3IgZm9yIGRldGFpbHMuXG4gKi9cbmZ1bmN0aW9uIHVzZUtleWJpbmRpbmdXYXJuaW5ncyhcbiAgd2FybmluZ3M6IEtleWJpbmRpbmdXYXJuaW5nW10sXG4gIGlzUmVsb2FkOiBib29sZWFuLFxuKTogdm9pZCB7XG4gIGNvbnN0IHsgYWRkTm90aWZpY2F0aW9uLCByZW1vdmVOb3RpZmljYXRpb24gfSA9IHVzZU5vdGlmaWNhdGlvbnMoKVxuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgY29uc3Qgbm90aWZpY2F0aW9uS2V5ID0gJ2tleWJpbmRpbmctY29uZmlnLXdhcm5pbmcnXG5cbiAgICBpZiAod2FybmluZ3MubGVuZ3RoID09PSAwKSB7XG4gICAgICByZW1vdmVOb3RpZmljYXRpb24obm90aWZpY2F0aW9uS2V5KVxuICAgICAgcmV0dXJuXG4gICAgfVxuXG4gICAgY29uc3QgZXJyb3JDb3VudCA9IGNvdW50KHdhcm5pbmdzLCB3ID0+IHcuc2V2ZXJpdHkgPT09ICdlcnJvcicpXG4gICAgY29uc3Qgd2FybkNvdW50ID0gY291bnQod2FybmluZ3MsIHcgPT4gdy5zZXZlcml0eSA9PT0gJ3dhcm5pbmcnKVxuXG4gICAgbGV0IG1lc3NhZ2U6IHN0cmluZ1xuICAgIGlmIChlcnJvckNvdW50ID4gMCAmJiB3YXJuQ291bnQgPiAwKSB7XG4gICAgICBtZXNzYWdlID0gYEZvdW5kICR7ZXJyb3JDb3VudH0ga2V5YmluZGluZyAke3BsdXJhbChlcnJvckNvdW50LCAnZXJyb3InKX0gYW5kICR7d2FybkNvdW50fSAke3BsdXJhbCh3YXJuQ291bnQsICd3YXJuaW5nJyl9YFxuICAgIH0gZWxzZSBpZiAoZXJyb3JDb3VudCA+IDApIHtcbiAgICAgIG1lc3NhZ2UgPSBgRm91bmQgJHtlcnJvckNvdW50fSBrZXliaW5kaW5nICR7cGx1cmFsKGVycm9yQ291bnQsICdlcnJvcicpfWBcbiAgICB9IGVsc2Uge1xuICAgICAgbWVzc2FnZSA9IGBGb3VuZCAke3dhcm5Db3VudH0ga2V5YmluZGluZyAke3BsdXJhbCh3YXJuQ291bnQsICd3YXJuaW5nJyl9YFxuICAgIH1cbiAgICBtZXNzYWdlICs9ICcgwrcgL2RvY3RvciBmb3IgZGV0YWlscydcblxuICAgIGFkZE5vdGlmaWNhdGlvbih7XG4gICAgICBrZXk6IG5vdGlmaWNhdGlvbktleSxcbiAgICAgIHRleHQ6IG1lc3NhZ2UsXG4gICAgICBjb2xvcjogZXJyb3JDb3VudCA+IDAgPyAnZXJyb3InIDogJ3dhcm5pbmcnLFxuICAgICAgcHJpb3JpdHk6IGVycm9yQ291bnQgPiAwID8gJ2ltbWVkaWF0ZScgOiAnaGlnaCcsXG4gICAgICAvLyBLZWVwIHZpc2libGUgZm9yIDYwIHNlY29uZHMgbGlrZSBzZXR0aW5ncyBlcnJvcnNcbiAgICAgIHRpbWVvdXRNczogNjAwMDAsXG4gICAgfSlcbiAgfSwgW3dhcm5pbmdzLCBpc1JlbG9hZCwgYWRkTm90aWZpY2F0aW9uLCByZW1vdmVOb3RpZmljYXRpb25dKVxufVxuXG5leHBvcnQgZnVuY3Rpb24gS2V5YmluZGluZ1NldHVwKHsgY2hpbGRyZW4gfTogUHJvcHMpOiBSZWFjdC5SZWFjdE5vZGUge1xuICAvLyBMb2FkIGJpbmRpbmdzIHN5bmNocm9ub3VzbHkgZm9yIGluaXRpYWwgcmVuZGVyXG4gIGNvbnN0IFt7IGJpbmRpbmdzLCB3YXJuaW5ncyB9LCBzZXRMb2FkUmVzdWx0XSA9XG4gICAgdXNlU3RhdGU8S2V5YmluZGluZ3NMb2FkUmVzdWx0PigoKSA9PiB7XG4gICAgICBjb25zdCByZXN1bHQgPSBsb2FkS2V5YmluZGluZ3NTeW5jV2l0aFdhcm5pbmdzKClcbiAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgYFtrZXliaW5kaW5nc10gS2V5YmluZGluZ1NldHVwIGluaXRpYWxpemVkIHdpdGggJHtyZXN1bHQuYmluZGluZ3MubGVuZ3RofSBiaW5kaW5ncywgJHtyZXN1bHQud2FybmluZ3MubGVuZ3RofSB3YXJuaW5nc2AsXG4gICAgICApXG4gICAgICByZXR1cm4gcmVzdWx0XG4gICAgfSlcblxuICAvLyBUcmFjayBpZiB0aGlzIGlzIGEgcmVsb2FkIChub3QgaW5pdGlhbCBsb2FkKVxuICBjb25zdCBbaXNSZWxvYWQsIHNldElzUmVsb2FkXSA9IHVzZVN0YXRlKGZhbHNlKVxuXG4gIC8vIERpc3BsYXkgd2FybmluZ3MgdmlhIG5vdGlmaWNhdGlvbnNcbiAgdXNlS2V5YmluZGluZ1dhcm5pbmdzKHdhcm5pbmdzLCBpc1JlbG9hZClcblxuICAvLyBDaG9yZCBzdGF0ZSBtYW5hZ2VtZW50IC0gdXNlIHJlZiBmb3IgaW1tZWRpYXRlIGFjY2Vzcywgc3RhdGUgZm9yIHJlLXJlbmRlcnNcbiAgLy8gVGhlIHJlZiBpcyB1c2VkIGJ5IHJlc29sdmUoKSB0byBnZXQgdGhlIGN1cnJlbnQgdmFsdWUgd2l0aG91dCB3YWl0aW5nIGZvciByZS1yZW5kZXJcbiAgLy8gVGhlIHN0YXRlIGlzIHVzZWQgdG8gdHJpZ2dlciByZS1yZW5kZXJzIHdoZW4gbmVlZGVkIChlLmcuLCBmb3IgVUkgdXBkYXRlcylcbiAgY29uc3QgcGVuZGluZ0Nob3JkUmVmID0gdXNlUmVmPFBhcnNlZEtleXN0cm9rZVtdIHwgbnVsbD4obnVsbClcbiAgY29uc3QgW3BlbmRpbmdDaG9yZCwgc2V0UGVuZGluZ0Nob3JkU3RhdGVdID0gdXNlU3RhdGU8XG4gICAgUGFyc2VkS2V5c3Ryb2tlW10gfCBudWxsXG4gID4obnVsbClcbiAgY29uc3QgY2hvcmRUaW1lb3V0UmVmID0gdXNlUmVmPE5vZGVKUy5UaW1lb3V0IHwgbnVsbD4obnVsbClcblxuICAvLyBIYW5kbGVyIHJlZ2lzdHJ5IGZvciBhY3Rpb24gY2FsbGJhY2tzICh1c2VkIGJ5IENob3JkSW50ZXJjZXB0b3IgdG8gaW52b2tlIGhhbmRsZXJzKVxuICBjb25zdCBoYW5kbGVyUmVnaXN0cnlSZWYgPSB1c2VSZWYoXG4gICAgbmV3IE1hcDxcbiAgICAgIHN0cmluZyxcbiAgICAgIFNldDx7XG4gICAgICAgIGFjdGlvbjogc3RyaW5nXG4gICAgICAgIGNvbnRleHQ6IEtleWJpbmRpbmdDb250ZXh0TmFtZVxuICAgICAgICBoYW5kbGVyOiAoKSA9PiB2b2lkXG4gICAgICB9PlxuICAgID4oKSxcbiAgKVxuXG4gIC8vIEFjdGl2ZSBjb250ZXh0IHRyYWNraW5nIGZvciBrZXliaW5kaW5nIHByaW9yaXR5IHJlc29sdXRpb25cbiAgLy8gVXNpbmcgYSByZWYgaW5zdGVhZCBvZiBzdGF0ZSBmb3Igc3luY2hyb25vdXMgdXBkYXRlcyAtIGlucHV0IGhhbmRsZXJzIG5lZWRcbiAgLy8gdG8gc2VlIHRoZSBjdXJyZW50IHZhbHVlIGltbWVkaWF0ZWx5LCBub3QgYWZ0ZXIgYSBSZWFjdCByZW5kZXIgY3ljbGUuXG4gIGNvbnN0IGFjdGl2ZUNvbnRleHRzUmVmID0gdXNlUmVmPFNldDxLZXliaW5kaW5nQ29udGV4dE5hbWU+PihuZXcgU2V0KCkpXG5cbiAgY29uc3QgcmVnaXN0ZXJBY3RpdmVDb250ZXh0ID0gdXNlQ2FsbGJhY2soXG4gICAgKGNvbnRleHQ6IEtleWJpbmRpbmdDb250ZXh0TmFtZSkgPT4ge1xuICAgICAgYWN0aXZlQ29udGV4dHNSZWYuY3VycmVudC5hZGQoY29udGV4dClcbiAgICB9LFxuICAgIFtdLFxuICApXG5cbiAgY29uc3QgdW5yZWdpc3RlckFjdGl2ZUNvbnRleHQgPSB1c2VDYWxsYmFjayhcbiAgICAoY29udGV4dDogS2V5YmluZGluZ0NvbnRleHROYW1lKSA9PiB7XG4gICAgICBhY3RpdmVDb250ZXh0c1JlZi5jdXJyZW50LmRlbGV0ZShjb250ZXh0KVxuICAgIH0sXG4gICAgW10sXG4gIClcblxuICAvLyBDbGVhciBjaG9yZCB0aW1lb3V0IHdoZW4gY29tcG9uZW50IHVubW91bnRzIG9yIGNob3JkIGNoYW5nZXNcbiAgY29uc3QgY2xlYXJDaG9yZFRpbWVvdXQgPSB1c2VDYWxsYmFjaygoKSA9PiB7XG4gICAgaWYgKGNob3JkVGltZW91dFJlZi5jdXJyZW50KSB7XG4gICAgICBjbGVhclRpbWVvdXQoY2hvcmRUaW1lb3V0UmVmLmN1cnJlbnQpXG4gICAgICBjaG9yZFRpbWVvdXRSZWYuY3VycmVudCA9IG51bGxcbiAgICB9XG4gIH0sIFtdKVxuXG4gIC8vIFdyYXBwZXIgZm9yIHNldFBlbmRpbmdDaG9yZCB0aGF0IG1hbmFnZXMgdGltZW91dCBhbmQgc3luY3MgcmVmK3N0YXRlXG4gIGNvbnN0IHNldFBlbmRpbmdDaG9yZCA9IHVzZUNhbGxiYWNrKFxuICAgIChwZW5kaW5nOiBQYXJzZWRLZXlzdHJva2VbXSB8IG51bGwpID0+IHtcbiAgICAgIGNsZWFyQ2hvcmRUaW1lb3V0KClcblxuICAgICAgaWYgKHBlbmRpbmcgIT09IG51bGwpIHtcbiAgICAgICAgLy8gU2V0IHRpbWVvdXQgdG8gY2FuY2VsIGNob3JkIGlmIG5vdCBjb21wbGV0ZWRcbiAgICAgICAgY2hvcmRUaW1lb3V0UmVmLmN1cnJlbnQgPSBzZXRUaW1lb3V0KFxuICAgICAgICAgIChwZW5kaW5nQ2hvcmRSZWYsIHNldFBlbmRpbmdDaG9yZFN0YXRlKSA9PiB7XG4gICAgICAgICAgICBsb2dGb3JEZWJ1Z2dpbmcoJ1trZXliaW5kaW5nc10gQ2hvcmQgdGltZW91dCAtIGNhbmNlbGxpbmcnKVxuICAgICAgICAgICAgcGVuZGluZ0Nob3JkUmVmLmN1cnJlbnQgPSBudWxsXG4gICAgICAgICAgICBzZXRQZW5kaW5nQ2hvcmRTdGF0ZShudWxsKVxuICAgICAgICAgIH0sXG4gICAgICAgICAgQ0hPUkRfVElNRU9VVF9NUyxcbiAgICAgICAgICBwZW5kaW5nQ2hvcmRSZWYsXG4gICAgICAgICAgc2V0UGVuZGluZ0Nob3JkU3RhdGUsXG4gICAgICAgIClcbiAgICAgIH1cblxuICAgICAgLy8gVXBkYXRlIHJlZiBpbW1lZGlhdGVseSBmb3Igc3luY2hyb25vdXMgYWNjZXNzIGluIHJlc29sdmUoKVxuICAgICAgcGVuZGluZ0Nob3JkUmVmLmN1cnJlbnQgPSBwZW5kaW5nXG4gICAgICAvLyBVcGRhdGUgc3RhdGUgdG8gdHJpZ2dlciByZS1yZW5kZXJzIGZvciBVSSB1cGRhdGVzXG4gICAgICBzZXRQZW5kaW5nQ2hvcmRTdGF0ZShwZW5kaW5nKVxuICAgIH0sXG4gICAgW2NsZWFyQ2hvcmRUaW1lb3V0XSxcbiAgKVxuXG4gIHVzZUVmZmVjdCgoKSA9PiB7XG4gICAgLy8gSW5pdGlhbGl6ZSBmaWxlIHdhdGNoZXIgKGlkZW1wb3RlbnQgLSBvbmx5IHJ1bnMgb25jZSlcbiAgICB2b2lkIGluaXRpYWxpemVLZXliaW5kaW5nV2F0Y2hlcigpXG5cbiAgICAvLyBTdWJzY3JpYmUgdG8gY2hhbmdlc1xuICAgIGNvbnN0IHVuc3Vic2NyaWJlID0gc3Vic2NyaWJlVG9LZXliaW5kaW5nQ2hhbmdlcyhyZXN1bHQgPT4ge1xuICAgICAgLy8gQW55IGNhbGxiYWNrIGludm9jYXRpb24gaXMgYSByZWxvYWQgc2luY2UgaW5pdGlhbCBsb2FkIGhhcHBlbnNcbiAgICAgIC8vIHN5bmNocm9ub3VzbHkgaW4gdXNlU3RhdGUsIG5vdCB2aWEgdGhpcyBzdWJzY3JpcHRpb25cbiAgICAgIHNldElzUmVsb2FkKHRydWUpXG5cbiAgICAgIHNldExvYWRSZXN1bHQocmVzdWx0KVxuICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICBgW2tleWJpbmRpbmdzXSBSZWxvYWRlZDogJHtyZXN1bHQuYmluZGluZ3MubGVuZ3RofSBiaW5kaW5ncywgJHtyZXN1bHQud2FybmluZ3MubGVuZ3RofSB3YXJuaW5nc2AsXG4gICAgICApXG4gICAgfSlcblxuICAgIHJldHVybiAoKSA9PiB7XG4gICAgICB1bnN1YnNjcmliZSgpXG4gICAgICBjbGVhckNob3JkVGltZW91dCgpXG4gICAgfVxuICB9LCBbY2xlYXJDaG9yZFRpbWVvdXRdKVxuXG4gIHJldHVybiAoXG4gICAgPEtleWJpbmRpbmdQcm92aWRlclxuICAgICAgYmluZGluZ3M9e2JpbmRpbmdzfVxuICAgICAgcGVuZGluZ0Nob3JkUmVmPXtwZW5kaW5nQ2hvcmRSZWZ9XG4gICAgICBwZW5kaW5nQ2hvcmQ9e3BlbmRpbmdDaG9yZH1cbiAgICAgIHNldFBlbmRpbmdDaG9yZD17c2V0UGVuZGluZ0Nob3JkfVxuICAgICAgYWN0aXZlQ29udGV4dHM9e2FjdGl2ZUNvbnRleHRzUmVmLmN1cnJlbnR9XG4gICAgICByZWdpc3RlckFjdGl2ZUNvbnRleHQ9e3JlZ2lzdGVyQWN0aXZlQ29udGV4dH1cbiAgICAgIHVucmVnaXN0ZXJBY3RpdmVDb250ZXh0PXt1bnJlZ2lzdGVyQWN0aXZlQ29udGV4dH1cbiAgICAgIGhhbmRsZXJSZWdpc3RyeVJlZj17aGFuZGxlclJlZ2lzdHJ5UmVmfVxuICAgID5cbiAgICAgIDxDaG9yZEludGVyY2VwdG9yXG4gICAgICAgIGJpbmRpbmdzPXtiaW5kaW5nc31cbiAgICAgICAgcGVuZGluZ0Nob3JkUmVmPXtwZW5kaW5nQ2hvcmRSZWZ9XG4gICAgICAgIHNldFBlbmRpbmdDaG9yZD17c2V0UGVuZGluZ0Nob3JkfVxuICAgICAgICBhY3RpdmVDb250ZXh0cz17YWN0aXZlQ29udGV4dHNSZWYuY3VycmVudH1cbiAgICAgICAgaGFuZGxlclJlZ2lzdHJ5UmVmPXtoYW5kbGVyUmVnaXN0cnlSZWZ9XG4gICAgICAvPlxuICAgICAge2NoaWxkcmVufVxuICAgIDwvS2V5YmluZGluZ1Byb3ZpZGVyPlxuICApXG59XG5cbi8qKlxuICogR2xvYmFsIGNob3JkIGludGVyY2VwdG9yIHRoYXQgcmVnaXN0ZXJzIHVzZUlucHV0IEZJUlNUIChiZWZvcmUgY2hpbGRyZW4pLlxuICpcbiAqIFRoaXMgY29tcG9uZW50IGludGVyY2VwdHMga2V5c3Ryb2tlcyB0aGF0IGFyZSBwYXJ0IG9mIGNob3JkIHNlcXVlbmNlcyBhbmRcbiAqIHN0b3BzIHByb3BhZ2F0aW9uIGJlZm9yZSBvdGhlciBoYW5kbGVycyAobGlrZSBQcm9tcHRJbnB1dCkgY2FuIHNlZSB0aGVtLlxuICpcbiAqIFdpdGhvdXQgdGhpcywgdGhlIHNlY29uZCBrZXkgb2YgYSBjaG9yZCAoZS5nLiwgJ3InIGluIFwiY3RybCtjIHJcIikgd291bGQgYmVcbiAqIGNhcHR1cmVkIGJ5IFByb21wdElucHV0IGFuZCBhZGRlZCB0byB0aGUgaW5wdXQgZmllbGQgYmVmb3JlIHRoZSBrZXliaW5kaW5nXG4gKiBzeXN0ZW0gY291bGQgcmVjb2duaXplIGl0IGFzIGNvbXBsZXRpbmcgYSBjaG9yZC5cbiAqL1xudHlwZSBIYW5kbGVyUmVnaXN0cmF0aW9uID0ge1xuICBhY3Rpb246IHN0cmluZ1xuICBjb250ZXh0OiBLZXliaW5kaW5nQ29udGV4dE5hbWVcbiAgaGFuZGxlcjogKCkgPT4gdm9pZFxufVxuXG5mdW5jdGlvbiBDaG9yZEludGVyY2VwdG9yKHtcbiAgYmluZGluZ3MsXG4gIHBlbmRpbmdDaG9yZFJlZixcbiAgc2V0UGVuZGluZ0Nob3JkLFxuICBhY3RpdmVDb250ZXh0cyxcbiAgaGFuZGxlclJlZ2lzdHJ5UmVmLFxufToge1xuICBiaW5kaW5nczogUGFyc2VkQmluZGluZ1tdXG4gIHBlbmRpbmdDaG9yZFJlZjogUmVhY3QuUmVmT2JqZWN0PFBhcnNlZEtleXN0cm9rZVtdIHwgbnVsbD5cbiAgc2V0UGVuZGluZ0Nob3JkOiAocGVuZGluZzogUGFyc2VkS2V5c3Ryb2tlW10gfCBudWxsKSA9PiB2b2lkXG4gIGFjdGl2ZUNvbnRleHRzOiBTZXQ8S2V5YmluZGluZ0NvbnRleHROYW1lPlxuICBoYW5kbGVyUmVnaXN0cnlSZWY6IFJlYWN0LlJlZk9iamVjdDxNYXA8c3RyaW5nLCBTZXQ8SGFuZGxlclJlZ2lzdHJhdGlvbj4+PlxufSk6IG51bGwge1xuICBjb25zdCBoYW5kbGVJbnB1dCA9IHVzZUNhbGxiYWNrKFxuICAgIChpbnB1dDogc3RyaW5nLCBrZXk6IEtleSwgZXZlbnQ6IElucHV0RXZlbnQpID0+IHtcbiAgICAgIC8vIFdoZWVsIGV2ZW50cyBjYW4gbmV2ZXIgc3RhcnQgY2hvcmQgc2VxdWVuY2VzIOKAlCBzY3JvbGw6bGluZVVwL0Rvd24gYXJlXG4gICAgICAvLyBzaW5nbGUta2V5IGJpbmRpbmdzIGhhbmRsZWQgYnkgcGVyLWNvbXBvbmVudCB1c2VLZXliaW5kaW5ncyBob29rcywgbm90XG4gICAgICAvLyBoZXJlLiBTa2lwIHRoZSByZWdpc3RyeSBzY2FuLiBNaWQtY2hvcmQgd2hlZWwgc3RpbGwgZmFsbHMgdGhyb3VnaCBzb1xuICAgICAgLy8gc2Nyb2xsaW5nIGNhbmNlbHMgdGhlIHBlbmRpbmcgY2hvcmQgbGlrZSBhbnkgb3RoZXIgbm9uLW1hdGNoaW5nIGtleS5cbiAgICAgIGlmICgoa2V5LndoZWVsVXAgfHwga2V5LndoZWVsRG93bikgJiYgcGVuZGluZ0Nob3JkUmVmLmN1cnJlbnQgPT09IG51bGwpIHtcbiAgICAgICAgcmV0dXJuXG4gICAgICB9XG5cbiAgICAgIC8vIEJ1aWxkIGNvbnRleHQgbGlzdCBmcm9tIHJlZ2lzdGVyZWQgaGFuZGxlcnMgKyBhY3RpdmVDb250ZXh0cyArIEdsb2JhbFxuICAgICAgLy8gVGhpcyBlbnN1cmVzIHdlIGNhbiByZXNvbHZlIGNob3JkcyBmb3IgYWxsIGNvbnRleHRzIHRoYXQgaGF2ZSBoYW5kbGVyc1xuICAgICAgY29uc3QgcmVnaXN0cnkgPSBoYW5kbGVyUmVnaXN0cnlSZWYuY3VycmVudFxuICAgICAgY29uc3QgaGFuZGxlckNvbnRleHRzID0gbmV3IFNldDxLZXliaW5kaW5nQ29udGV4dE5hbWU+KClcbiAgICAgIGlmIChyZWdpc3RyeSkge1xuICAgICAgICBmb3IgKGNvbnN0IGhhbmRsZXJzIG9mIHJlZ2lzdHJ5LnZhbHVlcygpKSB7XG4gICAgICAgICAgZm9yIChjb25zdCByZWdpc3RyYXRpb24gb2YgaGFuZGxlcnMpIHtcbiAgICAgICAgICAgIGhhbmRsZXJDb250ZXh0cy5hZGQocmVnaXN0cmF0aW9uLmNvbnRleHQpXG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9XG4gICAgICBjb25zdCBjb250ZXh0czogS2V5YmluZGluZ0NvbnRleHROYW1lW10gPSBbXG4gICAgICAgIC4uLmhhbmRsZXJDb250ZXh0cyxcbiAgICAgICAgLi4uYWN0aXZlQ29udGV4dHMsXG4gICAgICAgICdHbG9iYWwnLFxuICAgICAgXVxuXG4gICAgICAvLyBUcmFjayB3aGV0aGVyIHdlJ3JlIGNvbXBsZXRpbmcgYSBjaG9yZCAocGVuZGluZyB3YXMgbm9uLW51bGwpXG4gICAgICBjb25zdCB3YXNJbkNob3JkID0gcGVuZGluZ0Nob3JkUmVmLmN1cnJlbnQgIT09IG51bGxcblxuICAgICAgLy8gQ2hlY2sgaWYgdGhpcyBrZXlzdHJva2UgaXMgcGFydCBvZiBhIGNob3JkIHNlcXVlbmNlXG4gICAgICBjb25zdCByZXN1bHQgPSByZXNvbHZlS2V5V2l0aENob3JkU3RhdGUoXG4gICAgICAgIGlucHV0LFxuICAgICAgICBrZXksXG4gICAgICAgIGNvbnRleHRzLFxuICAgICAgICBiaW5kaW5ncyxcbiAgICAgICAgcGVuZGluZ0Nob3JkUmVmLmN1cnJlbnQsXG4gICAgICApXG5cbiAgICAgIHN3aXRjaCAocmVzdWx0LnR5cGUpIHtcbiAgICAgICAgY2FzZSAnY2hvcmRfc3RhcnRlZCc6XG4gICAgICAgICAgLy8gVGhpcyBrZXkgc3RhcnRzIGEgY2hvcmQgLSBzdG9yZSBwZW5kaW5nIHN0YXRlIGFuZCBzdG9wIHByb3BhZ2F0aW9uXG4gICAgICAgICAgc2V0UGVuZGluZ0Nob3JkKHJlc3VsdC5wZW5kaW5nKVxuICAgICAgICAgIGV2ZW50LnN0b3BJbW1lZGlhdGVQcm9wYWdhdGlvbigpXG4gICAgICAgICAgYnJlYWtcblxuICAgICAgICBjYXNlICdtYXRjaCc6IHtcbiAgICAgICAgICAvLyBDbGVhciBwZW5kaW5nIHN0YXRlXG4gICAgICAgICAgc2V0UGVuZGluZ0Nob3JkKG51bGwpXG5cbiAgICAgICAgICAvLyBPbmx5IGludm9rZSBoYW5kbGVycyBhbmQgc3RvcCBwcm9wYWdhdGlvbiBmb3IgY2hvcmQgY29tcGxldGlvbnNcbiAgICAgICAgICAvLyAobXVsdGkta2V5c3Ryb2tlIHNlcXVlbmNlcykuIFNpbmdsZS1rZXlzdHJva2UgbWF0Y2hlcyBzaG91bGQgcHJvcGFnYXRlXG4gICAgICAgICAgLy8gdG8gcGVyLWhvb2sgaGFuZGxlcnMgdG8gYXZvaWQgaW50ZXJmZXJpbmcgd2l0aCBvdGhlciBpbnB1dCBoYW5kbGluZ1xuICAgICAgICAgIC8vIChlLmcuLCBFbnRlciBuZWVkcyB0byByZWFjaCB1c2VUeXBlYWhlYWQgZm9yIGF1dG9jb21wbGV0ZSBhY2NlcHRhbmNlXG4gICAgICAgICAgLy8gYmVmb3JlIHRoZSBzdWJtaXQgaGFuZGxlciBmaXJlcykuXG4gICAgICAgICAgaWYgKHdhc0luQ2hvcmQpIHtcbiAgICAgICAgICAgIC8vIEZpbmQgYW5kIGludm9rZSB0aGUgaGFuZGxlciBmb3IgdGhpcyBhY3Rpb25cbiAgICAgICAgICAgIC8vIFdlIG5lZWQgdG8gY2hlY2sgdGhhdCB0aGUgaGFuZGxlcidzIGNvbnRleHQgaXMgaW4gb3VyIHJlc29sdmVkIGNvbnRleHRzXG4gICAgICAgICAgICAvLyAod2hpY2ggaW5jbHVkZXMgaGFuZGxlckNvbnRleHRzICsgYWN0aXZlQ29udGV4dHMgKyBHbG9iYWwpXG4gICAgICAgICAgICBjb25zdCBjb250ZXh0c1NldCA9IG5ldyBTZXQoY29udGV4dHMpXG4gICAgICAgICAgICBpZiAocmVnaXN0cnkpIHtcbiAgICAgICAgICAgICAgY29uc3QgaGFuZGxlcnMgPSByZWdpc3RyeS5nZXQocmVzdWx0LmFjdGlvbilcbiAgICAgICAgICAgICAgaWYgKGhhbmRsZXJzICYmIGhhbmRsZXJzLnNpemUgPiAwKSB7XG4gICAgICAgICAgICAgICAgLy8gRmluZCBoYW5kbGVycyB3aG9zZSBjb250ZXh0IGlzIGluIG91ciByZXNvbHZlZCBjb250ZXh0c1xuICAgICAgICAgICAgICAgIGZvciAoY29uc3QgcmVnaXN0cmF0aW9uIG9mIGhhbmRsZXJzKSB7XG4gICAgICAgICAgICAgICAgICBpZiAoY29udGV4dHNTZXQuaGFzKHJlZ2lzdHJhdGlvbi5jb250ZXh0KSkge1xuICAgICAgICAgICAgICAgICAgICByZWdpc3RyYXRpb24uaGFuZGxlcigpXG4gICAgICAgICAgICAgICAgICAgIGV2ZW50LnN0b3BJbW1lZGlhdGVQcm9wYWdhdGlvbigpXG4gICAgICAgICAgICAgICAgICAgIGJyZWFrIC8vIE9ubHkgaW52b2tlIHRoZSBmaXJzdCBtYXRjaGluZyBoYW5kbGVyXG4gICAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICAgIGJyZWFrXG4gICAgICAgIH1cblxuICAgICAgICBjYXNlICdjaG9yZF9jYW5jZWxsZWQnOlxuICAgICAgICAgIC8vIEludmFsaWQga2V5IGR1cmluZyBjaG9yZCAtIGNsZWFyIHBlbmRpbmcgc3RhdGUgYW5kIHN3YWxsb3cgdGhlXG4gICAgICAgICAgLy8ga2V5c3Ryb2tlIHNvIGl0IGRvZXNuJ3QgcHJvcGFnYXRlIGFzIGEgc3RhbmRhbG9uZSBhY3Rpb25cbiAgICAgICAgICAvLyAoZS5nLiwgY3RybCt4IGN0cmwrYyBzaG91bGQgbm90IGZpcmUgYXBwOmludGVycnVwdCkuXG4gICAgICAgICAgc2V0UGVuZGluZ0Nob3JkKG51bGwpXG4gICAgICAgICAgZXZlbnQuc3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uKClcbiAgICAgICAgICBicmVha1xuXG4gICAgICAgIGNhc2UgJ3VuYm91bmQnOlxuICAgICAgICAgIC8vIEtleSBpcyBleHBsaWNpdGx5IHVuYm91bmQgLSBjbGVhciBwZW5kaW5nIHN0YXRlIGFuZCBzd2FsbG93XG4gICAgICAgICAgLy8gdGhlIGtleXN0cm9rZSAoaXQgd2FzIHBhcnQgb2YgYSBjaG9yZCBzZXF1ZW5jZSkuXG4gICAgICAgICAgc2V0UGVuZGluZ0Nob3JkKG51bGwpXG4gICAgICAgICAgZXZlbnQuc3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uKClcbiAgICAgICAgICBicmVha1xuXG4gICAgICAgIGNhc2UgJ25vbmUnOlxuICAgICAgICAgIC8vIE5vIGNob3JkIGludm9sdmVtZW50IC0gbGV0IG90aGVyIGhhbmRsZXJzIHByb2Nlc3NcbiAgICAgICAgICBicmVha1xuICAgICAgfVxuICAgIH0sXG4gICAgW1xuICAgICAgYmluZGluZ3MsXG4gICAgICBwZW5kaW5nQ2hvcmRSZWYsXG4gICAgICBzZXRQZW5kaW5nQ2hvcmQsXG4gICAgICBhY3RpdmVDb250ZXh0cyxcbiAgICAgIGhhbmRsZXJSZWdpc3RyeVJlZixcbiAgICBdLFxuICApXG5cbiAgdXNlSW5wdXQoaGFuZGxlSW5wdXQpXG5cbiAgcmV0dXJuIG51bGxcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxPQUFPQSxLQUFLLElBQUlDLFdBQVcsRUFBRUMsU0FBUyxFQUFFQyxNQUFNLEVBQUVDLFFBQVEsUUFBUSxPQUFPO0FBQ3ZFLFNBQVNDLGdCQUFnQixRQUFRLDZCQUE2QjtBQUM5RCxjQUFjQyxVQUFVLFFBQVEsOEJBQThCO0FBQzlEO0FBQ0E7QUFDQTtBQUNBLFNBQVMsS0FBS0MsR0FBRyxFQUFFQyxRQUFRLFFBQVEsV0FBVztBQUM5QyxTQUFTQyxLQUFLLFFBQVEsbUJBQW1CO0FBQ3pDLFNBQVNDLGVBQWUsUUFBUSxtQkFBbUI7QUFDbkQsU0FBU0MsTUFBTSxRQUFRLHlCQUF5QjtBQUNoRCxTQUFTQyxrQkFBa0IsUUFBUSx3QkFBd0I7QUFDM0QsU0FDRUMsMkJBQTJCLEVBQzNCLEtBQUtDLHFCQUFxQixFQUMxQkMsK0JBQStCLEVBQy9CQyw0QkFBNEIsUUFDdkIsdUJBQXVCO0FBQzlCLFNBQVNDLHdCQUF3QixRQUFRLGVBQWU7QUFDeEQsY0FDRUMscUJBQXFCLEVBQ3JCQyxhQUFhLEVBQ2JDLGVBQWUsUUFDVixZQUFZO0FBQ25CLGNBQWNDLGlCQUFpQixRQUFRLGVBQWU7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