/ utils / teleport.tsx
teleport.tsx
   1  import axios from 'axios';
   2  import chalk from 'chalk';
   3  import { randomUUID } from 'crypto';
   4  import React from 'react';
   5  import { getOriginalCwd, getSessionId } from 'src/bootstrap/state.js';
   6  import { checkGate_CACHED_OR_BLOCKING } from 'src/services/analytics/growthbook.js';
   7  import { type AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS, logEvent } from 'src/services/analytics/index.js';
   8  import { isPolicyAllowed } from 'src/services/policyLimits/index.js';
   9  import { z } from 'zod/v4';
  10  import { getTeleportErrors, TeleportError, type TeleportLocalErrorType } from '../components/TeleportError.js';
  11  import { getOauthConfig } from '../constants/oauth.js';
  12  import type { SDKMessage } from '../entrypoints/agentSdkTypes.js';
  13  import type { Root } from '../ink.js';
  14  import { KeybindingSetup } from '../keybindings/KeybindingProviderSetup.js';
  15  import { queryHaiku } from '../services/api/claude.js';
  16  import { getSessionLogsViaOAuth, getTeleportEvents } from '../services/api/sessionIngress.js';
  17  import { getOrganizationUUID } from '../services/oauth/client.js';
  18  import { AppStateProvider } from '../state/AppState.js';
  19  import type { Message, SystemMessage } from '../types/message.js';
  20  import type { PermissionMode } from '../types/permissions.js';
  21  import { checkAndRefreshOAuthTokenIfNeeded, getClaudeAIOAuthTokens } from './auth.js';
  22  import { checkGithubAppInstalled } from './background/remote/preconditions.js';
  23  import { deserializeMessages, type TeleportRemoteResponse } from './conversationRecovery.js';
  24  import { getCwd } from './cwd.js';
  25  import { logForDebugging } from './debug.js';
  26  import { detectCurrentRepositoryWithHost, parseGitHubRepository, parseGitRemote } from './detectRepository.js';
  27  import { isEnvTruthy } from './envUtils.js';
  28  import { TeleportOperationError, toError } from './errors.js';
  29  import { execFileNoThrow } from './execFileNoThrow.js';
  30  import { truncateToWidth } from './format.js';
  31  import { findGitRoot, getDefaultBranch, getIsClean, gitExe } from './git.js';
  32  import { safeParseJSON } from './json.js';
  33  import { logError } from './log.js';
  34  import { createSystemMessage, createUserMessage } from './messages.js';
  35  import { getMainLoopModel } from './model/model.js';
  36  import { isTranscriptMessage } from './sessionStorage.js';
  37  import { getSettings_DEPRECATED } from './settings/settings.js';
  38  import { jsonStringify } from './slowOperations.js';
  39  import { asSystemPrompt } from './systemPromptType.js';
  40  import { fetchSession, type GitRepositoryOutcome, type GitSource, getBranchFromSession, getOAuthHeaders, type SessionResource } from './teleport/api.js';
  41  import { fetchEnvironments } from './teleport/environments.js';
  42  import { createAndUploadGitBundle } from './teleport/gitBundle.js';
  43  export type TeleportResult = {
  44    messages: Message[];
  45    branchName: string;
  46  };
  47  export type TeleportProgressStep = 'validating' | 'fetching_logs' | 'fetching_branch' | 'checking_out' | 'done';
  48  export type TeleportProgressCallback = (step: TeleportProgressStep) => void;
  49  
  50  /**
  51   * Creates a system message to inform about teleport session resume
  52   * @returns SystemMessage indicating session was resumed from another machine
  53   */
  54  function createTeleportResumeSystemMessage(branchError: Error | null): SystemMessage {
  55    if (branchError === null) {
  56      return createSystemMessage('Session resumed', 'suggestion');
  57    }
  58    const formattedError = branchError instanceof TeleportOperationError ? branchError.formattedMessage : branchError.message;
  59    return createSystemMessage(`Session resumed without branch: ${formattedError}`, 'warning');
  60  }
  61  
  62  /**
  63   * Creates a user message to inform the model about teleport session resume
  64   * @returns User message indicating session was resumed from another machine
  65   */
  66  function createTeleportResumeUserMessage() {
  67    return createUserMessage({
  68      content: `This session is being continued from another machine. Application state may have changed. The updated working directory is ${getOriginalCwd()}`,
  69      isMeta: true
  70    });
  71  }
  72  type TeleportToRemoteResponse = {
  73    id: string;
  74    title: string;
  75  };
  76  const SESSION_TITLE_AND_BRANCH_PROMPT = `You are coming up with a succinct title and git branch name for a coding session based on the provided description. The title should be clear, concise, and accurately reflect the content of the coding task.
  77  You should keep it short and simple, ideally no more than 6 words. Avoid using jargon or overly technical terms unless absolutely necessary. The title should be easy to understand for anyone reading it.
  78  Use sentence case for the title (capitalize only the first word and proper nouns), not Title Case.
  79  
  80  The branch name should be clear, concise, and accurately reflect the content of the coding task.
  81  You should keep it short and simple, ideally no more than 4 words. The branch should always start with "claude/" and should be all lower case, with words separated by dashes.
  82  
  83  Return a JSON object with "title" and "branch" fields.
  84  
  85  Example 1: {"title": "Fix login button not working on mobile", "branch": "claude/fix-mobile-login-button"}
  86  Example 2: {"title": "Update README with installation instructions", "branch": "claude/update-readme"}
  87  Example 3: {"title": "Improve performance of data processing script", "branch": "claude/improve-data-processing"}
  88  
  89  Here is the session description:
  90  <description>{description}</description>
  91  Please generate a title and branch name for this session.`;
  92  type TitleAndBranch = {
  93    title: string;
  94    branchName: string;
  95  };
  96  
  97  /**
  98   * Generates a title and branch name for a coding session using Claude Haiku
  99   * @param description The description/prompt for the session
 100   * @returns Promise<TitleAndBranch> The generated title and branch name
 101   */
 102  async function generateTitleAndBranch(description: string, signal: AbortSignal): Promise<TitleAndBranch> {
 103    const fallbackTitle = truncateToWidth(description, 75);
 104    const fallbackBranch = 'claude/task';
 105    try {
 106      const userPrompt = SESSION_TITLE_AND_BRANCH_PROMPT.replace('{description}', description);
 107      const response = await queryHaiku({
 108        systemPrompt: asSystemPrompt([]),
 109        userPrompt,
 110        outputFormat: {
 111          type: 'json_schema',
 112          schema: {
 113            type: 'object',
 114            properties: {
 115              title: {
 116                type: 'string'
 117              },
 118              branch: {
 119                type: 'string'
 120              }
 121            },
 122            required: ['title', 'branch'],
 123            additionalProperties: false
 124          }
 125        },
 126        signal,
 127        options: {
 128          querySource: 'teleport_generate_title',
 129          agents: [],
 130          isNonInteractiveSession: false,
 131          hasAppendSystemPrompt: false,
 132          mcpTools: []
 133        }
 134      });
 135  
 136      // Extract text from the response
 137      const firstBlock = response.message.content[0];
 138      if (firstBlock?.type !== 'text') {
 139        return {
 140          title: fallbackTitle,
 141          branchName: fallbackBranch
 142        };
 143      }
 144      const parsed = safeParseJSON(firstBlock.text.trim());
 145      const parseResult = z.object({
 146        title: z.string(),
 147        branch: z.string()
 148      }).safeParse(parsed);
 149      if (parseResult.success) {
 150        return {
 151          title: parseResult.data.title || fallbackTitle,
 152          branchName: parseResult.data.branch || fallbackBranch
 153        };
 154      }
 155      return {
 156        title: fallbackTitle,
 157        branchName: fallbackBranch
 158      };
 159    } catch (error) {
 160      logError(new Error(`Error generating title and branch: ${error}`));
 161      return {
 162        title: fallbackTitle,
 163        branchName: fallbackBranch
 164      };
 165    }
 166  }
 167  
 168  /**
 169   * Validates that the git working directory is clean (ignoring untracked files)
 170   * Untracked files are ignored because they won't be lost during branch switching
 171   */
 172  export async function validateGitState(): Promise<void> {
 173    const isClean = await getIsClean({
 174      ignoreUntracked: true
 175    });
 176    if (!isClean) {
 177      logEvent('tengu_teleport_error_git_not_clean', {});
 178      const error = new TeleportOperationError('Git working directory is not clean. Please commit or stash your changes before using --teleport.', chalk.red('Error: Git working directory is not clean. Please commit or stash your changes before using --teleport.\n'));
 179      throw error;
 180    }
 181  }
 182  
 183  /**
 184   * Fetches a specific branch from remote origin
 185   * @param branch The branch to fetch. If not specified, fetches all branches.
 186   */
 187  async function fetchFromOrigin(branch?: string): Promise<void> {
 188    const fetchArgs = branch ? ['fetch', 'origin', `${branch}:${branch}`] : ['fetch', 'origin'];
 189    const {
 190      code: fetchCode,
 191      stderr: fetchStderr
 192    } = await execFileNoThrow(gitExe(), fetchArgs);
 193    if (fetchCode !== 0) {
 194      // If fetching a specific branch fails, it might not exist locally yet
 195      // Try fetching just the ref without mapping to local branch
 196      if (branch && fetchStderr.includes('refspec')) {
 197        logForDebugging(`Specific branch fetch failed, trying to fetch ref: ${branch}`);
 198        const {
 199          code: refFetchCode,
 200          stderr: refFetchStderr
 201        } = await execFileNoThrow(gitExe(), ['fetch', 'origin', branch]);
 202        if (refFetchCode !== 0) {
 203          logError(new Error(`Failed to fetch from remote origin: ${refFetchStderr}`));
 204        }
 205      } else {
 206        logError(new Error(`Failed to fetch from remote origin: ${fetchStderr}`));
 207      }
 208    }
 209  }
 210  
 211  /**
 212   * Ensures that the current branch has an upstream set
 213   * If not, sets it to origin/<branchName> if that remote branch exists
 214   */
 215  async function ensureUpstreamIsSet(branchName: string): Promise<void> {
 216    // Check if upstream is already set
 217    const {
 218      code: upstreamCheckCode
 219    } = await execFileNoThrow(gitExe(), ['rev-parse', '--abbrev-ref', `${branchName}@{upstream}`]);
 220    if (upstreamCheckCode === 0) {
 221      // Upstream is already set
 222      logForDebugging(`Branch '${branchName}' already has upstream set`);
 223      return;
 224    }
 225  
 226    // Check if origin/<branchName> exists
 227    const {
 228      code: remoteCheckCode
 229    } = await execFileNoThrow(gitExe(), ['rev-parse', '--verify', `origin/${branchName}`]);
 230    if (remoteCheckCode === 0) {
 231      // Remote branch exists, set upstream
 232      logForDebugging(`Setting upstream for '${branchName}' to 'origin/${branchName}'`);
 233      const {
 234        code: setUpstreamCode,
 235        stderr: setUpstreamStderr
 236      } = await execFileNoThrow(gitExe(), ['branch', '--set-upstream-to', `origin/${branchName}`, branchName]);
 237      if (setUpstreamCode !== 0) {
 238        logForDebugging(`Failed to set upstream for '${branchName}': ${setUpstreamStderr}`);
 239        // Don't throw, just log - this is not critical
 240      } else {
 241        logForDebugging(`Successfully set upstream for '${branchName}'`);
 242      }
 243    } else {
 244      logForDebugging(`Remote branch 'origin/${branchName}' does not exist, skipping upstream setup`);
 245    }
 246  }
 247  
 248  /**
 249   * Checks out a specific branch
 250   */
 251  async function checkoutBranch(branchName: string): Promise<void> {
 252    // First try to checkout the branch as-is (might be local)
 253    let {
 254      code: checkoutCode,
 255      stderr: checkoutStderr
 256    } = await execFileNoThrow(gitExe(), ['checkout', branchName]);
 257  
 258    // If that fails, try to checkout from origin
 259    if (checkoutCode !== 0) {
 260      logForDebugging(`Local checkout failed, trying to checkout from origin: ${checkoutStderr}`);
 261  
 262      // Try to checkout the remote branch and create a local tracking branch
 263      const result = await execFileNoThrow(gitExe(), ['checkout', '-b', branchName, '--track', `origin/${branchName}`]);
 264      checkoutCode = result.code;
 265      checkoutStderr = result.stderr;
 266  
 267      // If that also fails, try without -b in case the branch exists but isn't checked out
 268      if (checkoutCode !== 0) {
 269        logForDebugging(`Remote checkout with -b failed, trying without -b: ${checkoutStderr}`);
 270        const finalResult = await execFileNoThrow(gitExe(), ['checkout', '--track', `origin/${branchName}`]);
 271        checkoutCode = finalResult.code;
 272        checkoutStderr = finalResult.stderr;
 273      }
 274    }
 275    if (checkoutCode !== 0) {
 276      logEvent('tengu_teleport_error_branch_checkout_failed', {});
 277      throw new TeleportOperationError(`Failed to checkout branch '${branchName}': ${checkoutStderr}`, chalk.red(`Failed to checkout branch '${branchName}'\n`));
 278    }
 279  
 280    // After successful checkout, ensure upstream is set
 281    await ensureUpstreamIsSet(branchName);
 282  }
 283  
 284  /**
 285   * Gets the current branch name
 286   */
 287  async function getCurrentBranch(): Promise<string> {
 288    const {
 289      stdout: currentBranch
 290    } = await execFileNoThrow(gitExe(), ['branch', '--show-current']);
 291    return currentBranch.trim();
 292  }
 293  
 294  /**
 295   * Processes messages for teleport resume, removing incomplete tool_use blocks
 296   * and adding teleport notice messages
 297   * @param messages The conversation messages
 298   * @param error Optional error from branch checkout
 299   * @returns Processed messages ready for resume
 300   */
 301  export function processMessagesForTeleportResume(messages: Message[], error: Error | null): Message[] {
 302    // Shared logic with resume for handling interruped session transcripts
 303    const deserializedMessages = deserializeMessages(messages);
 304  
 305    // Add user message about teleport resume (visible to model)
 306    const messagesWithTeleportNotice = [...deserializedMessages, createTeleportResumeUserMessage(), createTeleportResumeSystemMessage(error)];
 307    return messagesWithTeleportNotice;
 308  }
 309  
 310  /**
 311   * Checks out the specified branch for a teleported session
 312   * @param branch Optional branch to checkout
 313   * @returns The current branch name and any error that occurred
 314   */
 315  export async function checkOutTeleportedSessionBranch(branch?: string): Promise<{
 316    branchName: string;
 317    branchError: Error | null;
 318  }> {
 319    try {
 320      const currentBranch = await getCurrentBranch();
 321      logForDebugging(`Current branch before teleport: '${currentBranch}'`);
 322      if (branch) {
 323        logForDebugging(`Switching to branch '${branch}'...`);
 324        await fetchFromOrigin(branch);
 325        await checkoutBranch(branch);
 326        const newBranch = await getCurrentBranch();
 327        logForDebugging(`Branch after checkout: '${newBranch}'`);
 328      } else {
 329        logForDebugging('No branch specified, staying on current branch');
 330      }
 331      const branchName = await getCurrentBranch();
 332      return {
 333        branchName,
 334        branchError: null
 335      };
 336    } catch (error) {
 337      const branchName = await getCurrentBranch();
 338      const branchError = toError(error);
 339      return {
 340        branchName,
 341        branchError
 342      };
 343    }
 344  }
 345  
 346  /**
 347   * Result of repository validation for teleport
 348   */
 349  export type RepoValidationResult = {
 350    status: 'match' | 'mismatch' | 'not_in_repo' | 'no_repo_required' | 'error';
 351    sessionRepo?: string;
 352    currentRepo?: string | null;
 353    /** Host of the session repo (e.g. "github.com" or "ghe.corp.com") — for display only */
 354    sessionHost?: string;
 355    /** Host of the current repo (e.g. "github.com" or "ghe.corp.com") — for display only */
 356    currentHost?: string;
 357    errorMessage?: string;
 358  };
 359  
 360  /**
 361   * Validates that the current repository matches the session's repository.
 362   * Returns a result object instead of throwing, allowing the caller to handle mismatches.
 363   *
 364   * @param sessionData The session resource to validate against
 365   * @returns Validation result with status and repo information
 366   */
 367  export async function validateSessionRepository(sessionData: SessionResource): Promise<RepoValidationResult> {
 368    const currentParsed = await detectCurrentRepositoryWithHost();
 369    const currentRepo = currentParsed ? `${currentParsed.owner}/${currentParsed.name}` : null;
 370    const gitSource = sessionData.session_context.sources.find((source): source is GitSource => source.type === 'git_repository');
 371    if (!gitSource?.url) {
 372      // Session has no repo requirement
 373      logForDebugging(currentRepo ? 'Session has no associated repository, proceeding without validation' : 'Session has no repo requirement and not in git directory, proceeding');
 374      return {
 375        status: 'no_repo_required'
 376      };
 377    }
 378    const sessionParsed = parseGitRemote(gitSource.url);
 379    const sessionRepo = sessionParsed ? `${sessionParsed.owner}/${sessionParsed.name}` : parseGitHubRepository(gitSource.url);
 380    if (!sessionRepo) {
 381      return {
 382        status: 'no_repo_required'
 383      };
 384    }
 385    logForDebugging(`Session is for repository: ${sessionRepo}, current repo: ${currentRepo ?? 'none'}`);
 386    if (!currentRepo) {
 387      // Not in a git repo, but session requires one
 388      return {
 389        status: 'not_in_repo',
 390        sessionRepo,
 391        sessionHost: sessionParsed?.host,
 392        currentRepo: null
 393      };
 394    }
 395  
 396    // Compare both owner/repo and host to avoid cross-instance mismatches.
 397    // Strip ports before comparing hosts — SSH remotes omit the port while
 398    // HTTPS remotes may include a non-standard port (e.g. ghe.corp.com:8443),
 399    // which would cause a false mismatch.
 400    const stripPort = (host: string): string => host.replace(/:\d+$/, '');
 401    const repoMatch = currentRepo.toLowerCase() === sessionRepo.toLowerCase();
 402    const hostMatch = !currentParsed || !sessionParsed || stripPort(currentParsed.host.toLowerCase()) === stripPort(sessionParsed.host.toLowerCase());
 403    if (repoMatch && hostMatch) {
 404      return {
 405        status: 'match',
 406        sessionRepo,
 407        currentRepo
 408      };
 409    }
 410  
 411    // Repo mismatch — keep sessionRepo/currentRepo as plain "owner/repo" so
 412    // downstream consumers (e.g. getKnownPathsForRepo) can use them as lookup keys.
 413    // Include host information in separate fields for display purposes.
 414    return {
 415      status: 'mismatch',
 416      sessionRepo,
 417      currentRepo,
 418      sessionHost: sessionParsed?.host,
 419      currentHost: currentParsed?.host
 420    };
 421  }
 422  
 423  /**
 424   * Handles teleporting from a code session ID.
 425   * Fetches session logs and validates repo.
 426   * @param sessionId The session ID to resume
 427   * @param onProgress Optional callback for progress updates
 428   * @returns The raw session log and branch name
 429   */
 430  export async function teleportResumeCodeSession(sessionId: string, onProgress?: TeleportProgressCallback): Promise<TeleportRemoteResponse> {
 431    if (!isPolicyAllowed('allow_remote_sessions')) {
 432      throw new Error("Remote sessions are disabled by your organization's policy.");
 433    }
 434    logForDebugging(`Resuming code session ID: ${sessionId}`);
 435    try {
 436      const accessToken = getClaudeAIOAuthTokens()?.accessToken;
 437      if (!accessToken) {
 438        logEvent('tengu_teleport_resume_error', {
 439          error_type: 'no_access_token' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
 440        });
 441        throw new Error('Claude Code web sessions require authentication with a Claude.ai account. API key authentication is not sufficient. Please run /login to authenticate, or check your authentication status with /status.');
 442      }
 443  
 444      // Get organization UUID
 445      const orgUUID = await getOrganizationUUID();
 446      if (!orgUUID) {
 447        logEvent('tengu_teleport_resume_error', {
 448          error_type: 'no_org_uuid' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
 449        });
 450        throw new Error('Unable to get organization UUID for constructing session URL');
 451      }
 452  
 453      // Fetch and validate repository matches before resuming
 454      onProgress?.('validating');
 455      const sessionData = await fetchSession(sessionId);
 456      const repoValidation = await validateSessionRepository(sessionData);
 457      switch (repoValidation.status) {
 458        case 'match':
 459        case 'no_repo_required':
 460          // Proceed with teleport
 461          break;
 462        case 'not_in_repo':
 463          {
 464            logEvent('tengu_teleport_error_repo_not_in_git_dir_sessions_api', {
 465              sessionId: sessionId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
 466            });
 467            // Include host for GHE users so they know which instance the repo is on
 468            const notInRepoDisplay = repoValidation.sessionHost && repoValidation.sessionHost.toLowerCase() !== 'github.com' ? `${repoValidation.sessionHost}/${repoValidation.sessionRepo}` : repoValidation.sessionRepo;
 469            throw new TeleportOperationError(`You must run claude --teleport ${sessionId} from a checkout of ${notInRepoDisplay}.`, chalk.red(`You must run claude --teleport ${sessionId} from a checkout of ${chalk.bold(notInRepoDisplay)}.\n`));
 470          }
 471        case 'mismatch':
 472          {
 473            logEvent('tengu_teleport_error_repo_mismatch_sessions_api', {
 474              sessionId: sessionId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
 475            });
 476            // Only include host prefix when hosts actually differ to disambiguate
 477            // cross-instance mismatches; for same-host mismatches the host is noise.
 478            const hostsDiffer = repoValidation.sessionHost && repoValidation.currentHost && repoValidation.sessionHost.replace(/:\d+$/, '').toLowerCase() !== repoValidation.currentHost.replace(/:\d+$/, '').toLowerCase();
 479            const sessionDisplay = hostsDiffer ? `${repoValidation.sessionHost}/${repoValidation.sessionRepo}` : repoValidation.sessionRepo;
 480            const currentDisplay = hostsDiffer ? `${repoValidation.currentHost}/${repoValidation.currentRepo}` : repoValidation.currentRepo;
 481            throw new TeleportOperationError(`You must run claude --teleport ${sessionId} from a checkout of ${sessionDisplay}.\nThis repo is ${currentDisplay}.`, chalk.red(`You must run claude --teleport ${sessionId} from a checkout of ${chalk.bold(sessionDisplay)}.\nThis repo is ${chalk.bold(currentDisplay)}.\n`));
 482          }
 483        case 'error':
 484          throw new TeleportOperationError(repoValidation.errorMessage || 'Failed to validate session repository', chalk.red(`Error: ${repoValidation.errorMessage || 'Failed to validate session repository'}\n`));
 485        default:
 486          {
 487            const _exhaustive: never = repoValidation.status;
 488            throw new Error(`Unhandled repo validation status: ${_exhaustive}`);
 489          }
 490      }
 491      return await teleportFromSessionsAPI(sessionId, orgUUID, accessToken, onProgress, sessionData);
 492    } catch (error) {
 493      if (error instanceof TeleportOperationError) {
 494        throw error;
 495      }
 496      const err = toError(error);
 497      logError(err);
 498      logEvent('tengu_teleport_resume_error', {
 499        error_type: 'resume_session_id_catch' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
 500      });
 501      throw new TeleportOperationError(err.message, chalk.red(`Error: ${err.message}\n`));
 502    }
 503  }
 504  
 505  /**
 506   * Helper function to handle teleport prerequisites (authentication and git state)
 507   * Shows TeleportError dialog rendered into the existing root if needed
 508   */
 509  async function handleTeleportPrerequisites(root: Root, errorsToIgnore?: Set<TeleportLocalErrorType>): Promise<void> {
 510    const errors = await getTeleportErrors();
 511    if (errors.size > 0) {
 512      // Log teleport errors detected
 513      logEvent('tengu_teleport_errors_detected', {
 514        error_types: Array.from(errors).join(',') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
 515        errors_ignored: Array.from(errorsToIgnore || []).join(',') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
 516      });
 517  
 518      // Show TeleportError dialog for user interaction
 519      await new Promise<void>(resolve => {
 520        root.render(<AppStateProvider>
 521            <KeybindingSetup>
 522              <TeleportError errorsToIgnore={errorsToIgnore} onComplete={() => {
 523              // Log when errors are resolved
 524              logEvent('tengu_teleport_errors_resolved', {
 525                error_types: Array.from(errors).join(',') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
 526              });
 527              void resolve();
 528            }} />
 529            </KeybindingSetup>
 530          </AppStateProvider>);
 531      });
 532    }
 533  }
 534  
 535  /**
 536   * Creates a remote Claude.ai session with error handling and UI feedback.
 537   * Shows prerequisite error dialog in the existing root if needed.
 538   * @param root The existing Ink root to render dialogs into
 539   * @param description The description/prompt for the new session (null for no initial prompt)
 540   * @param signal AbortSignal for cancellation
 541   * @param branchName Optional branch name for the remote session to use
 542   * @returns Promise<TeleportToRemoteResponse | null> The created session or null if creation fails
 543   */
 544  export async function teleportToRemoteWithErrorHandling(root: Root, description: string | null, signal: AbortSignal, branchName?: string): Promise<TeleportToRemoteResponse | null> {
 545    const errorsToIgnore = new Set<TeleportLocalErrorType>(['needsGitStash']);
 546    await handleTeleportPrerequisites(root, errorsToIgnore);
 547    return teleportToRemote({
 548      initialMessage: description,
 549      signal,
 550      branchName,
 551      onBundleFail: msg => process.stderr.write(`\n${msg}\n`)
 552    });
 553  }
 554  
 555  /**
 556   * Fetches session data from the session ingress API (/v1/session_ingress/)
 557   * Uses session logs instead of SDK events to get the correct message structure
 558   * @param sessionId The session ID to fetch
 559   * @param orgUUID The organization UUID
 560   * @param accessToken The OAuth access token
 561   * @param onProgress Optional callback for progress updates
 562   * @param sessionData Optional session data (used to extract branch info)
 563   * @returns TeleportRemoteResponse with session logs as Message[]
 564   */
 565  export async function teleportFromSessionsAPI(sessionId: string, orgUUID: string, accessToken: string, onProgress?: TeleportProgressCallback, sessionData?: SessionResource): Promise<TeleportRemoteResponse> {
 566    const startTime = Date.now();
 567    try {
 568      // Fetch session logs via session ingress
 569      logForDebugging(`[teleport] Starting fetch for session: ${sessionId}`);
 570      onProgress?.('fetching_logs');
 571      const logsStartTime = Date.now();
 572      // Try CCR v2 first (GetTeleportEvents — server dispatches Spanner/
 573      // threadstore). Fall back to session-ingress if it returns null
 574      // (endpoint not yet deployed, or transient error). Once session-ingress
 575      // is gone, the fallback becomes a no-op — getSessionLogsViaOAuth will
 576      // return null too and we fail with "Failed to fetch session logs".
 577      let logs = await getTeleportEvents(sessionId, accessToken, orgUUID);
 578      if (logs === null) {
 579        logForDebugging('[teleport] v2 endpoint returned null, trying session-ingress');
 580        logs = await getSessionLogsViaOAuth(sessionId, accessToken, orgUUID);
 581      }
 582      logForDebugging(`[teleport] Session logs fetched in ${Date.now() - logsStartTime}ms`);
 583      if (logs === null) {
 584        throw new Error('Failed to fetch session logs');
 585      }
 586  
 587      // Filter to get only transcript messages, excluding sidechain messages
 588      const filterStartTime = Date.now();
 589      const messages = logs.filter(entry => isTranscriptMessage(entry) && !entry.isSidechain) as Message[];
 590      logForDebugging(`[teleport] Filtered ${logs.length} entries to ${messages.length} messages in ${Date.now() - filterStartTime}ms`);
 591  
 592      // Extract branch info from session data
 593      onProgress?.('fetching_branch');
 594      const branch = sessionData ? getBranchFromSession(sessionData) : undefined;
 595      if (branch) {
 596        logForDebugging(`[teleport] Found branch: ${branch}`);
 597      }
 598      logForDebugging(`[teleport] Total teleportFromSessionsAPI time: ${Date.now() - startTime}ms`);
 599      return {
 600        log: messages,
 601        branch
 602      };
 603    } catch (error) {
 604      const err = toError(error);
 605  
 606      // Handle 404 specifically
 607      if (axios.isAxiosError(error) && error.response?.status === 404) {
 608        logEvent('tengu_teleport_error_session_not_found_404', {
 609          sessionId: sessionId as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
 610        });
 611        throw new TeleportOperationError(`${sessionId} not found.`, `${sessionId} not found.\n${chalk.dim('Run /status in Claude Code to check your account.')}`);
 612      }
 613      logError(err);
 614      throw new Error(`Failed to fetch session from Sessions API: ${err.message}`);
 615    }
 616  }
 617  
 618  /**
 619   * Response type for polling remote session events (uses SDK events format)
 620   */
 621  export type PollRemoteSessionResponse = {
 622    newEvents: SDKMessage[];
 623    lastEventId: string | null;
 624    branch?: string;
 625    sessionStatus?: 'idle' | 'running' | 'requires_action' | 'archived';
 626  };
 627  
 628  /**
 629   * Polls remote session events. Pass the previous response's `lastEventId`
 630   * as `afterId` to fetch only the delta. Set `skipMetadata` to avoid the
 631   * per-call GET /v1/sessions/{id} when branch/status aren't needed.
 632   */
 633  export async function pollRemoteSessionEvents(sessionId: string, afterId: string | null = null, opts?: {
 634    skipMetadata?: boolean;
 635  }): Promise<PollRemoteSessionResponse> {
 636    const accessToken = getClaudeAIOAuthTokens()?.accessToken;
 637    if (!accessToken) {
 638      throw new Error('No access token for polling');
 639    }
 640    const orgUUID = await getOrganizationUUID();
 641    if (!orgUUID) {
 642      throw new Error('No org UUID for polling');
 643    }
 644    const headers = {
 645      ...getOAuthHeaders(accessToken),
 646      'anthropic-beta': 'ccr-byoc-2025-07-29',
 647      'x-organization-uuid': orgUUID
 648    };
 649    const eventsUrl = `${getOauthConfig().BASE_API_URL}/v1/sessions/${sessionId}/events`;
 650    type EventsResponse = {
 651      data: unknown[];
 652      has_more: boolean;
 653      first_id: string | null;
 654      last_id: string | null;
 655    };
 656  
 657    // Cap is a safety valve against stuck cursors; steady-state is 0–1 pages.
 658    const MAX_EVENT_PAGES = 50;
 659    const sdkMessages: SDKMessage[] = [];
 660    let cursor = afterId;
 661    for (let page = 0; page < MAX_EVENT_PAGES; page++) {
 662      const eventsResponse = await axios.get(eventsUrl, {
 663        headers,
 664        params: cursor ? {
 665          after_id: cursor
 666        } : undefined,
 667        timeout: 30000
 668      });
 669      if (eventsResponse.status !== 200) {
 670        throw new Error(`Failed to fetch session events: ${eventsResponse.statusText}`);
 671      }
 672      const eventsData: EventsResponse = eventsResponse.data;
 673      if (!eventsData?.data || !Array.isArray(eventsData.data)) {
 674        throw new Error('Invalid events response');
 675      }
 676      for (const event of eventsData.data) {
 677        if (event && typeof event === 'object' && 'type' in event) {
 678          if (event.type === 'env_manager_log' || event.type === 'control_response') {
 679            continue;
 680          }
 681          if ('session_id' in event) {
 682            sdkMessages.push(event as SDKMessage);
 683          }
 684        }
 685      }
 686      if (!eventsData.last_id) break;
 687      cursor = eventsData.last_id;
 688      if (!eventsData.has_more) break;
 689    }
 690    if (opts?.skipMetadata) {
 691      return {
 692        newEvents: sdkMessages,
 693        lastEventId: cursor
 694      };
 695    }
 696  
 697    // Fetch session metadata (branch, status)
 698    let branch: string | undefined;
 699    let sessionStatus: PollRemoteSessionResponse['sessionStatus'];
 700    try {
 701      const sessionData = await fetchSession(sessionId);
 702      branch = getBranchFromSession(sessionData);
 703      sessionStatus = sessionData.session_status as PollRemoteSessionResponse['sessionStatus'];
 704    } catch (e) {
 705      logForDebugging(`teleport: failed to fetch session ${sessionId} metadata: ${e}`, {
 706        level: 'debug'
 707      });
 708    }
 709    return {
 710      newEvents: sdkMessages,
 711      lastEventId: cursor,
 712      branch,
 713      sessionStatus
 714    };
 715  }
 716  
 717  /**
 718   * Creates a remote Claude.ai session using the Sessions API.
 719   *
 720   * Two source modes:
 721   * - GitHub (default): backend clones from the repo's origin URL. Requires a
 722   *   GitHub remote + CCR-side GitHub connection. 43% of CLI sessions have an
 723   *   origin remote; far fewer pass the full precondition chain.
 724   * - Bundle (CCR_FORCE_BUNDLE=1): CLI creates `git bundle --all`, uploads via Files
 725   *   API, passes file_id as seed_bundle_file_id on the session context. CCR
 726   *   downloads it and clones from the bundle. No GitHub dependency — works for
 727   *   local-only repos. Reach: 54% of CLI sessions (anything with .git/).
 728   *   Backend: anthropic#303856.
 729   */
 730  export async function teleportToRemote(options: {
 731    initialMessage: string | null;
 732    branchName?: string;
 733    title?: string;
 734    /**
 735     * The description of the session. This is used to generate the title and
 736     * session branch name (unless they are explicitly provided).
 737     */
 738    description?: string;
 739    model?: string;
 740    permissionMode?: PermissionMode;
 741    ultraplan?: boolean;
 742    signal: AbortSignal;
 743    useDefaultEnvironment?: boolean;
 744    /**
 745     * Explicit environment_id (e.g. the code_review synthetic env). Bypasses
 746     * fetchEnvironments; the usual repo-detection → git source still runs so
 747     * the container gets the repo checked out (orchestrator reads --repo-dir
 748     * from pwd, it doesn't clone).
 749     */
 750    environmentId?: string;
 751    /**
 752     * Per-session env vars merged into session_context.environment_variables.
 753     * Write-only at the API layer (stripped from Get/List responses). When
 754     * environmentId is set, CLAUDE_CODE_OAUTH_TOKEN is auto-injected from the
 755     * caller's accessToken so the container's hook can hit inference (the
 756     * server only passes through what the caller sends; bughunter.go mints
 757     * its own, user sessions don't get one automatically).
 758     */
 759    environmentVariables?: Record<string, string>;
 760    /**
 761     * When set with environmentId, creates and uploads a git bundle of the
 762     * local working tree (createAndUploadGitBundle handles the stash-create
 763     * for uncommitted changes) and passes it as seed_bundle_file_id. Backend
 764     * clones from the bundle instead of GitHub — container gets the caller's
 765     * exact local state. Needs .git/ only, not a GitHub remote.
 766     */
 767    useBundle?: boolean;
 768    /**
 769     * Called with a user-facing message when the bundle path is attempted but
 770     * fails. The wrapper stderr.writes it (pre-REPL). Remote-agent callers
 771     * capture it to include in their throw (in-REPL, Ink-rendered).
 772     */
 773    onBundleFail?: (message: string) => void;
 774    /**
 775     * When true, disables the git-bundle fallback entirely. Use for flows like
 776     * autofix where CCR must push to GitHub — a bundle can't do that.
 777     */
 778    skipBundle?: boolean;
 779    /**
 780     * When set, reuses this branch as the outcome branch instead of generating
 781     * a new claude/ branch. Sets allow_unrestricted_git_push on the source and
 782     * reuse_outcome_branches on the session context so the remote pushes to the
 783     * caller's branch directly.
 784     */
 785    reuseOutcomeBranch?: string;
 786    /**
 787     * GitHub PR to attach to the session context. Backend uses this to
 788     * identify the PR associated with this session.
 789     */
 790    githubPr?: {
 791      owner: string;
 792      repo: string;
 793      number: number;
 794    };
 795  }): Promise<TeleportToRemoteResponse | null> {
 796    const {
 797      initialMessage,
 798      signal
 799    } = options;
 800    try {
 801      // Check authentication
 802      await checkAndRefreshOAuthTokenIfNeeded();
 803      const accessToken = getClaudeAIOAuthTokens()?.accessToken;
 804      if (!accessToken) {
 805        logError(new Error('No access token found for remote session creation'));
 806        return null;
 807      }
 808  
 809      // Get organization UUID
 810      const orgUUID = await getOrganizationUUID();
 811      if (!orgUUID) {
 812        logError(new Error('Unable to get organization UUID for remote session creation'));
 813        return null;
 814      }
 815  
 816      // Explicit environmentId short-circuits Haiku title-gen + env selection.
 817      // Still runs repo detection so the container gets a working directory —
 818      // the code_review orchestrator reads --repo-dir $(pwd), it doesn't clone
 819      // (bughunter.go:520 sets a git source too; env-manager does the checkout
 820      // before the SessionStart hook fires).
 821      if (options.environmentId) {
 822        const url = `${getOauthConfig().BASE_API_URL}/v1/sessions`;
 823        const headers = {
 824          ...getOAuthHeaders(accessToken),
 825          'anthropic-beta': 'ccr-byoc-2025-07-29',
 826          'x-organization-uuid': orgUUID
 827        };
 828        const envVars = {
 829          CLAUDE_CODE_OAUTH_TOKEN: accessToken,
 830          ...(options.environmentVariables ?? {})
 831        };
 832  
 833        // Bundle mode: upload local working tree (uncommitted changes via
 834        // refs/seed/stash), container clones from the bundle. No GitHub.
 835        // Otherwise: github.com source — caller checked eligibility.
 836        let gitSource: GitSource | null = null;
 837        let seedBundleFileId: string | null = null;
 838        if (options.useBundle) {
 839          const bundle = await createAndUploadGitBundle({
 840            oauthToken: accessToken,
 841            sessionId: getSessionId(),
 842            baseUrl: getOauthConfig().BASE_API_URL
 843          }, {
 844            signal
 845          });
 846          if (!bundle.success) {
 847            logError(new Error(`Bundle upload failed: ${bundle.error}`));
 848            return null;
 849          }
 850          seedBundleFileId = bundle.fileId;
 851          logEvent('tengu_teleport_bundle_mode', {
 852            size_bytes: bundle.bundleSizeBytes,
 853            scope: bundle.scope as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
 854            has_wip: bundle.hasWip,
 855            reason: 'explicit_env_bundle' as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
 856          });
 857        } else {
 858          const repoInfo = await detectCurrentRepositoryWithHost();
 859          if (repoInfo) {
 860            gitSource = {
 861              type: 'git_repository',
 862              url: `https://${repoInfo.host}/${repoInfo.owner}/${repoInfo.name}`,
 863              revision: options.branchName
 864            };
 865          }
 866        }
 867        const requestBody = {
 868          title: options.title || options.description || 'Remote task',
 869          events: [],
 870          session_context: {
 871            sources: gitSource ? [gitSource] : [],
 872            ...(seedBundleFileId && {
 873              seed_bundle_file_id: seedBundleFileId
 874            }),
 875            outcomes: [],
 876            environment_variables: envVars
 877          },
 878          environment_id: options.environmentId
 879        };
 880        logForDebugging(`[teleportToRemote] explicit env ${options.environmentId}, ${Object.keys(envVars).length} env vars, ${seedBundleFileId ? `bundle=${seedBundleFileId}` : `source=${gitSource?.url ?? 'none'}@${options.branchName ?? 'default'}`}`);
 881        const response = await axios.post(url, requestBody, {
 882          headers,
 883          signal
 884        });
 885        if (response.status !== 200 && response.status !== 201) {
 886          logError(new Error(`CreateSession ${response.status}: ${jsonStringify(response.data)}`));
 887          return null;
 888        }
 889        const sessionData = response.data as SessionResource;
 890        if (!sessionData || typeof sessionData.id !== 'string') {
 891          logError(new Error(`No session id in response: ${jsonStringify(response.data)}`));
 892          return null;
 893        }
 894        return {
 895          id: sessionData.id,
 896          title: sessionData.title || requestBody.title
 897        };
 898      }
 899      let gitSource: GitSource | null = null;
 900      let gitOutcome: GitRepositoryOutcome | null = null;
 901      let seedBundleFileId: string | null = null;
 902  
 903      // Source selection ladder: GitHub clone (if CCR can actually pull it) →
 904      // bundle fallback (if .git exists) → empty sandbox.
 905      //
 906      // The preflight is the same code path the container's git-proxy clone
 907      // will hit (get_github_client_with_user_auth → no_sync_user_token_found).
 908      // 50% of users who reach the "install GitHub App" step never finish it;
 909      // without the preflight, every one of them gets a container that 401s
 910      // on clone. With it, they silently fall back to bundle.
 911      //
 912      // CCR_FORCE_BUNDLE=1 skips the preflight entirely — useful for testing
 913      // or when you know your GitHub auth is busted. Read here (not in the
 914      // caller) so it works for remote-agent too, not just --remote.
 915  
 916      const repoInfo = await detectCurrentRepositoryWithHost();
 917  
 918      // Generate title and branch name for the session. Skip the Haiku call
 919      // when both title and outcome branch are explicitly provided.
 920      let sessionTitle: string;
 921      let sessionBranch: string;
 922      if (options.title && options.reuseOutcomeBranch) {
 923        sessionTitle = options.title;
 924        sessionBranch = options.reuseOutcomeBranch;
 925      } else {
 926        const generated = await generateTitleAndBranch(options.description || initialMessage || 'Background task', signal);
 927        sessionTitle = options.title || generated.title;
 928        sessionBranch = options.reuseOutcomeBranch || generated.branchName;
 929      }
 930  
 931      // Preflight: does CCR have a token that can clone this repo?
 932      // Only checked for github.com — GHES needs ghe_configuration_id which
 933      // we don't have, and GHES users are power users who probably finished
 934      // setup. For them (and for non-GitHub hosts that parseGitRemote
 935      // somehow accepted), fall through optimistically; if the backend
 936      // rejects the host, bundle next time.
 937      let ghViable = false;
 938      let sourceReason: 'github_preflight_ok' | 'ghes_optimistic' | 'github_preflight_failed' | 'no_github_remote' | 'forced_bundle' | 'no_git_at_all' = 'no_git_at_all';
 939  
 940      // gitRoot gates both bundle creation and the gate check itself — no
 941      // point awaiting GrowthBook when there's nothing to bundle.
 942      const gitRoot = findGitRoot(getCwd());
 943      const forceBundle = !options.skipBundle && isEnvTruthy(process.env.CCR_FORCE_BUNDLE);
 944      const bundleSeedGateOn = !options.skipBundle && gitRoot !== null && (isEnvTruthy(process.env.CCR_ENABLE_BUNDLE) || (await checkGate_CACHED_OR_BLOCKING('tengu_ccr_bundle_seed_enabled')));
 945      if (repoInfo && !forceBundle) {
 946        if (repoInfo.host === 'github.com') {
 947          ghViable = await checkGithubAppInstalled(repoInfo.owner, repoInfo.name, signal);
 948          sourceReason = ghViable ? 'github_preflight_ok' : 'github_preflight_failed';
 949        } else {
 950          ghViable = true;
 951          sourceReason = 'ghes_optimistic';
 952        }
 953      } else if (forceBundle) {
 954        sourceReason = 'forced_bundle';
 955      } else if (gitRoot) {
 956        sourceReason = 'no_github_remote';
 957      }
 958  
 959      // Preflight failed but bundle is off — fall through optimistically like
 960      // pre-preflight behavior. Backend reports the real auth error.
 961      if (!ghViable && !bundleSeedGateOn && repoInfo) {
 962        ghViable = true;
 963      }
 964      if (ghViable && repoInfo) {
 965        const {
 966          host,
 967          owner,
 968          name
 969        } = repoInfo;
 970        // Resolve the base branch: prefer explicit branchName, fall back to default branch
 971        const revision = options.branchName ?? (await getDefaultBranch()) ?? undefined;
 972        logForDebugging(`[teleportToRemote] Git source: ${host}/${owner}/${name}, revision: ${revision ?? 'none'}`);
 973        gitSource = {
 974          type: 'git_repository',
 975          url: `https://${host}/${owner}/${name}`,
 976          // The revision specifies which ref to checkout as the base branch
 977          revision,
 978          ...(options.reuseOutcomeBranch && {
 979            allow_unrestricted_git_push: true
 980          })
 981        };
 982        // type: 'github' is used for all GitHub-compatible hosts (github.com and GHE).
 983        // The CLI can't distinguish GHE from non-GitHub hosts (GitLab, Bitbucket)
 984        // client-side — the backend validates the URL against configured GHE instances
 985        // and ignores git_info for unrecognized hosts.
 986        gitOutcome = {
 987          type: 'git_repository',
 988          git_info: {
 989            type: 'github',
 990            repo: `${owner}/${name}`,
 991            branches: [sessionBranch]
 992          }
 993        };
 994      }
 995  
 996      // Bundle fallback. Only try bundle if GitHub wasn't viable, the gate is
 997      // on, and there's a .git/ to bundle from. Reaching here with
 998      // ghViable=false and repoInfo non-null means the preflight failed —
 999      // .git definitely exists (detectCurrentRepositoryWithHost read the
1000      // remote from it).
1001      if (!gitSource && bundleSeedGateOn) {
1002        logForDebugging(`[teleportToRemote] Bundling (reason: ${sourceReason})`);
1003        const bundle = await createAndUploadGitBundle({
1004          oauthToken: accessToken,
1005          sessionId: getSessionId(),
1006          baseUrl: getOauthConfig().BASE_API_URL
1007        }, {
1008          signal
1009        });
1010        if (!bundle.success) {
1011          logError(new Error(`Bundle upload failed: ${bundle.error}`));
1012          // Only steer users to GitHub setup when there's a remote to clone from.
1013          const setup = repoInfo ? '. Please setup GitHub on https://claude.ai/code' : '';
1014          let msg: string;
1015          switch (bundle.failReason) {
1016            case 'empty_repo':
1017              msg = 'Repository has no commits — run `git add . && git commit -m "initial"` then retry';
1018              break;
1019            case 'too_large':
1020              msg = `Repo is too large to teleport${setup}`;
1021              break;
1022            case 'git_error':
1023              msg = `Failed to create git bundle (${bundle.error})${setup}`;
1024              break;
1025            case undefined:
1026              msg = `Bundle upload failed: ${bundle.error}${setup}`;
1027              break;
1028            default:
1029              {
1030                const _exhaustive: never = bundle.failReason;
1031                void _exhaustive;
1032                msg = `Bundle upload failed: ${bundle.error}`;
1033              }
1034          }
1035          options.onBundleFail?.(msg);
1036          return null;
1037        }
1038        seedBundleFileId = bundle.fileId;
1039        logEvent('tengu_teleport_bundle_mode', {
1040          size_bytes: bundle.bundleSizeBytes,
1041          scope: bundle.scope as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1042          has_wip: bundle.hasWip,
1043          reason: sourceReason as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
1044        });
1045      }
1046      logEvent('tengu_teleport_source_decision', {
1047        reason: sourceReason as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS,
1048        path: (gitSource ? 'github' : seedBundleFileId ? 'bundle' : 'empty') as AnalyticsMetadata_I_VERIFIED_THIS_IS_NOT_CODE_OR_FILEPATHS
1049      });
1050      if (!gitSource && !seedBundleFileId) {
1051        logForDebugging('[teleportToRemote] No repository detected — session will have an empty sandbox');
1052      }
1053  
1054      // Fetch available environments
1055      let environments = await fetchEnvironments();
1056      if (!environments || environments.length === 0) {
1057        logError(new Error('No environments available for session creation'));
1058        return null;
1059      }
1060      logForDebugging(`Available environments: ${environments.map(e => `${e.environment_id} (${e.name}, ${e.kind})`).join(', ')}`);
1061  
1062      // Select environment based on settings, then anthropic_cloud preference, then first available.
1063      // Prefer anthropic_cloud environments over byoc: anthropic_cloud environments (e.g. "Default")
1064      // are the standard compute environments with full repo access, whereas byoc environments
1065      // (e.g. "monorepo") are user-owned compute that may not support the current repository.
1066      const settings = getSettings_DEPRECATED();
1067      const defaultEnvironmentId = options.useDefaultEnvironment ? undefined : settings?.remote?.defaultEnvironmentId;
1068      let cloudEnv = environments.find(env => env.kind === 'anthropic_cloud');
1069      // When the caller opts out of their configured default, do not fall
1070      // through to a BYOC env that may not support the current repo or the
1071      // requested permission mode. Retry once for eventual consistency,
1072      // then fail loudly.
1073      if (options.useDefaultEnvironment && !cloudEnv) {
1074        logForDebugging(`No anthropic_cloud in env list (${environments.length} envs); retrying fetchEnvironments`);
1075        const retried = await fetchEnvironments();
1076        cloudEnv = retried?.find(env => env.kind === 'anthropic_cloud');
1077        if (!cloudEnv) {
1078          logError(new Error(`No anthropic_cloud environment available after retry (got: ${(retried ?? environments).map(e => `${e.name} (${e.kind})`).join(', ')}). Silent byoc fallthrough would launch into a dead env — fail fast instead.`));
1079          return null;
1080        }
1081        if (retried) environments = retried;
1082      }
1083      const selectedEnvironment = defaultEnvironmentId && environments.find(env => env.environment_id === defaultEnvironmentId) || cloudEnv || environments.find(env => env.kind !== 'bridge') || environments[0];
1084      if (!selectedEnvironment) {
1085        logError(new Error('No environments available for session creation'));
1086        return null;
1087      }
1088      if (defaultEnvironmentId) {
1089        const matchedDefault = selectedEnvironment.environment_id === defaultEnvironmentId;
1090        logForDebugging(matchedDefault ? `Using configured default environment: ${defaultEnvironmentId}` : `Configured default environment ${defaultEnvironmentId} not found, using first available`);
1091      }
1092      const environmentId = selectedEnvironment.environment_id;
1093      logForDebugging(`Selected environment: ${environmentId} (${selectedEnvironment.name}, ${selectedEnvironment.kind})`);
1094  
1095      // Prepare API request for Sessions API
1096      const url = `${getOauthConfig().BASE_API_URL}/v1/sessions`;
1097      const headers = {
1098        ...getOAuthHeaders(accessToken),
1099        'anthropic-beta': 'ccr-byoc-2025-07-29',
1100        'x-organization-uuid': orgUUID
1101      };
1102      const sessionContext = {
1103        sources: gitSource ? [gitSource] : [],
1104        ...(seedBundleFileId && {
1105          seed_bundle_file_id: seedBundleFileId
1106        }),
1107        outcomes: gitOutcome ? [gitOutcome] : [],
1108        model: options.model ?? getMainLoopModel(),
1109        ...(options.reuseOutcomeBranch && {
1110          reuse_outcome_branches: true
1111        }),
1112        ...(options.githubPr && {
1113          github_pr: options.githubPr
1114        })
1115      };
1116  
1117      // CreateCCRSessionPayload has no permission_mode field — a top-level
1118      // body entry is silently dropped by the proto parser server-side.
1119      // Instead prepend a set_permission_mode control_request event. Initial
1120      // events are written to threadstore before the container connects, so
1121      // the CLI applies the mode before the first user turn — no readiness race.
1122      const events: Array<{
1123        type: 'event';
1124        data: Record<string, unknown>;
1125      }> = [];
1126      if (options.permissionMode) {
1127        events.push({
1128          type: 'event',
1129          data: {
1130            type: 'control_request',
1131            request_id: `set-mode-${randomUUID()}`,
1132            request: {
1133              subtype: 'set_permission_mode',
1134              mode: options.permissionMode,
1135              ultraplan: options.ultraplan
1136            }
1137          }
1138        });
1139      }
1140      if (initialMessage) {
1141        events.push({
1142          type: 'event',
1143          data: {
1144            uuid: randomUUID(),
1145            session_id: '',
1146            type: 'user',
1147            parent_tool_use_id: null,
1148            message: {
1149              role: 'user',
1150              content: initialMessage
1151            }
1152          }
1153        });
1154      }
1155      const requestBody = {
1156        title: options.ultraplan ? `ultraplan: ${sessionTitle}` : sessionTitle,
1157        events,
1158        session_context: sessionContext,
1159        environment_id: environmentId
1160      };
1161      logForDebugging(`Creating session with payload: ${jsonStringify(requestBody, null, 2)}`);
1162  
1163      // Make API call
1164      const response = await axios.post(url, requestBody, {
1165        headers,
1166        signal
1167      });
1168      const isSuccess = response.status === 200 || response.status === 201;
1169      if (!isSuccess) {
1170        logError(new Error(`API request failed with status ${response.status}: ${response.statusText}\n\nResponse data: ${jsonStringify(response.data, null, 2)}`));
1171        return null;
1172      }
1173  
1174      // Parse response as SessionResource
1175      const sessionData = response.data as SessionResource;
1176      if (!sessionData || typeof sessionData.id !== 'string') {
1177        logError(new Error(`Cannot determine session ID from API response: ${jsonStringify(response.data)}`));
1178        return null;
1179      }
1180      logForDebugging(`Successfully created remote session: ${sessionData.id}`);
1181      return {
1182        id: sessionData.id,
1183        title: sessionData.title || requestBody.title
1184      };
1185    } catch (error) {
1186      const err = toError(error);
1187      logError(err);
1188      return null;
1189    }
1190  }
1191  
1192  /**
1193   * Best-effort session archive. POST /v1/sessions/{id}/archive has no
1194   * running-status check (unlike DELETE which 409s on RUNNING), so it works
1195   * mid-implementation. Archived sessions reject new events (send_events.go),
1196   * so the remote stops on its next write. 409 (already archived) treated as
1197   * success. Fire-and-forget; failure leaks a visible session until the
1198   * reaper collects it.
1199   */
1200  export async function archiveRemoteSession(sessionId: string): Promise<void> {
1201    const accessToken = getClaudeAIOAuthTokens()?.accessToken;
1202    if (!accessToken) return;
1203    const orgUUID = await getOrganizationUUID();
1204    if (!orgUUID) return;
1205    const headers = {
1206      ...getOAuthHeaders(accessToken),
1207      'anthropic-beta': 'ccr-byoc-2025-07-29',
1208      'x-organization-uuid': orgUUID
1209    };
1210    const url = `${getOauthConfig().BASE_API_URL}/v1/sessions/${sessionId}/archive`;
1211    try {
1212      const resp = await axios.post(url, {}, {
1213        headers,
1214        timeout: 10000,
1215        validateStatus: s => s < 500
1216      });
1217      if (resp.status === 200 || resp.status === 409) {
1218        logForDebugging(`[archiveRemoteSession] archived ${sessionId}`);
1219      } else {
1220        logForDebugging(`[archiveRemoteSession] ${sessionId} failed ${resp.status}: ${jsonStringify(resp.data)}`);
1221      }
1222    } catch (err) {
1223      logError(err);
1224    }
1225  }
1226  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJheGlvcyIsImNoYWxrIiwicmFuZG9tVVVJRCIsIlJlYWN0IiwiZ2V0T3JpZ2luYWxDd2QiLCJnZXRTZXNzaW9uSWQiLCJjaGVja0dhdGVfQ0FDSEVEX09SX0JMT0NLSU5HIiwiQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyIsImxvZ0V2ZW50IiwiaXNQb2xpY3lBbGxvd2VkIiwieiIsImdldFRlbGVwb3J0RXJyb3JzIiwiVGVsZXBvcnRFcnJvciIsIlRlbGVwb3J0TG9jYWxFcnJvclR5cGUiLCJnZXRPYXV0aENvbmZpZyIsIlNES01lc3NhZ2UiLCJSb290IiwiS2V5YmluZGluZ1NldHVwIiwicXVlcnlIYWlrdSIsImdldFNlc3Npb25Mb2dzVmlhT0F1dGgiLCJnZXRUZWxlcG9ydEV2ZW50cyIsImdldE9yZ2FuaXphdGlvblVVSUQiLCJBcHBTdGF0ZVByb3ZpZGVyIiwiTWVzc2FnZSIsIlN5c3RlbU1lc3NhZ2UiLCJQZXJtaXNzaW9uTW9kZSIsImNoZWNrQW5kUmVmcmVzaE9BdXRoVG9rZW5JZk5lZWRlZCIsImdldENsYXVkZUFJT0F1dGhUb2tlbnMiLCJjaGVja0dpdGh1YkFwcEluc3RhbGxlZCIsImRlc2VyaWFsaXplTWVzc2FnZXMiLCJUZWxlcG9ydFJlbW90ZVJlc3BvbnNlIiwiZ2V0Q3dkIiwibG9nRm9yRGVidWdnaW5nIiwiZGV0ZWN0Q3VycmVudFJlcG9zaXRvcnlXaXRoSG9zdCIsInBhcnNlR2l0SHViUmVwb3NpdG9yeSIsInBhcnNlR2l0UmVtb3RlIiwiaXNFbnZUcnV0aHkiLCJUZWxlcG9ydE9wZXJhdGlvbkVycm9yIiwidG9FcnJvciIsImV4ZWNGaWxlTm9UaHJvdyIsInRydW5jYXRlVG9XaWR0aCIsImZpbmRHaXRSb290IiwiZ2V0RGVmYXVsdEJyYW5jaCIsImdldElzQ2xlYW4iLCJnaXRFeGUiLCJzYWZlUGFyc2VKU09OIiwibG9nRXJyb3IiLCJjcmVhdGVTeXN0ZW1NZXNzYWdlIiwiY3JlYXRlVXNlck1lc3NhZ2UiLCJnZXRNYWluTG9vcE1vZGVsIiwiaXNUcmFuc2NyaXB0TWVzc2FnZSIsImdldFNldHRpbmdzX0RFUFJFQ0FURUQiLCJqc29uU3RyaW5naWZ5IiwiYXNTeXN0ZW1Qcm9tcHQiLCJmZXRjaFNlc3Npb24iLCJHaXRSZXBvc2l0b3J5T3V0Y29tZSIsIkdpdFNvdXJjZSIsImdldEJyYW5jaEZyb21TZXNzaW9uIiwiZ2V0T0F1dGhIZWFkZXJzIiwiU2Vzc2lvblJlc291cmNlIiwiZmV0Y2hFbnZpcm9ubWVudHMiLCJjcmVhdGVBbmRVcGxvYWRHaXRCdW5kbGUiLCJUZWxlcG9ydFJlc3VsdCIsIm1lc3NhZ2VzIiwiYnJhbmNoTmFtZSIsIlRlbGVwb3J0UHJvZ3Jlc3NTdGVwIiwiVGVsZXBvcnRQcm9ncmVzc0NhbGxiYWNrIiwic3RlcCIsImNyZWF0ZVRlbGVwb3J0UmVzdW1lU3lzdGVtTWVzc2FnZSIsImJyYW5jaEVycm9yIiwiRXJyb3IiLCJmb3JtYXR0ZWRFcnJvciIsImZvcm1hdHRlZE1lc3NhZ2UiLCJtZXNzYWdlIiwiY3JlYXRlVGVsZXBvcnRSZXN1bWVVc2VyTWVzc2FnZSIsImNvbnRlbnQiLCJpc01ldGEiLCJUZWxlcG9ydFRvUmVtb3RlUmVzcG9uc2UiLCJpZCIsInRpdGxlIiwiU0VTU0lPTl9USVRMRV9BTkRfQlJBTkNIX1BST01QVCIsIlRpdGxlQW5kQnJhbmNoIiwiZ2VuZXJhdGVUaXRsZUFuZEJyYW5jaCIsImRlc2NyaXB0aW9uIiwic2lnbmFsIiwiQWJvcnRTaWduYWwiLCJQcm9taXNlIiwiZmFsbGJhY2tUaXRsZSIsImZhbGxiYWNrQnJhbmNoIiwidXNlclByb21wdCIsInJlcGxhY2UiLCJyZXNwb25zZSIsInN5c3RlbVByb21wdCIsIm91dHB1dEZvcm1hdCIsInR5cGUiLCJzY2hlbWEiLCJwcm9wZXJ0aWVzIiwiYnJhbmNoIiwicmVxdWlyZWQiLCJhZGRpdGlvbmFsUHJvcGVydGllcyIsIm9wdGlvbnMiLCJxdWVyeVNvdXJjZSIsImFnZW50cyIsImlzTm9uSW50ZXJhY3RpdmVTZXNzaW9uIiwiaGFzQXBwZW5kU3lzdGVtUHJvbXB0IiwibWNwVG9vbHMiLCJmaXJzdEJsb2NrIiwicGFyc2VkIiwidGV4dCIsInRyaW0iLCJwYXJzZVJlc3VsdCIsIm9iamVjdCIsInN0cmluZyIsInNhZmVQYXJzZSIsInN1Y2Nlc3MiLCJkYXRhIiwiZXJyb3IiLCJ2YWxpZGF0ZUdpdFN0YXRlIiwiaXNDbGVhbiIsImlnbm9yZVVudHJhY2tlZCIsInJlZCIsImZldGNoRnJvbU9yaWdpbiIsImZldGNoQXJncyIsImNvZGUiLCJmZXRjaENvZGUiLCJzdGRlcnIiLCJmZXRjaFN0ZGVyciIsImluY2x1ZGVzIiwicmVmRmV0Y2hDb2RlIiwicmVmRmV0Y2hTdGRlcnIiLCJlbnN1cmVVcHN0cmVhbUlzU2V0IiwidXBzdHJlYW1DaGVja0NvZGUiLCJyZW1vdGVDaGVja0NvZGUiLCJzZXRVcHN0cmVhbUNvZGUiLCJzZXRVcHN0cmVhbVN0ZGVyciIsImNoZWNrb3V0QnJhbmNoIiwiY2hlY2tvdXRDb2RlIiwiY2hlY2tvdXRTdGRlcnIiLCJyZXN1bHQiLCJmaW5hbFJlc3VsdCIsImdldEN1cnJlbnRCcmFuY2giLCJzdGRvdXQiLCJjdXJyZW50QnJhbmNoIiwicHJvY2Vzc01lc3NhZ2VzRm9yVGVsZXBvcnRSZXN1bWUiLCJkZXNlcmlhbGl6ZWRNZXNzYWdlcyIsIm1lc3NhZ2VzV2l0aFRlbGVwb3J0Tm90aWNlIiwiY2hlY2tPdXRUZWxlcG9ydGVkU2Vzc2lvbkJyYW5jaCIsIm5ld0JyYW5jaCIsIlJlcG9WYWxpZGF0aW9uUmVzdWx0Iiwic3RhdHVzIiwic2Vzc2lvblJlcG8iLCJjdXJyZW50UmVwbyIsInNlc3Npb25Ib3N0IiwiY3VycmVudEhvc3QiLCJlcnJvck1lc3NhZ2UiLCJ2YWxpZGF0ZVNlc3Npb25SZXBvc2l0b3J5Iiwic2Vzc2lvbkRhdGEiLCJjdXJyZW50UGFyc2VkIiwib3duZXIiLCJuYW1lIiwiZ2l0U291cmNlIiwic2Vzc2lvbl9jb250ZXh0Iiwic291cmNlcyIsImZpbmQiLCJzb3VyY2UiLCJ1cmwiLCJzZXNzaW9uUGFyc2VkIiwiaG9zdCIsInN0cmlwUG9ydCIsInJlcG9NYXRjaCIsInRvTG93ZXJDYXNlIiwiaG9zdE1hdGNoIiwidGVsZXBvcnRSZXN1bWVDb2RlU2Vzc2lvbiIsInNlc3Npb25JZCIsIm9uUHJvZ3Jlc3MiLCJhY2Nlc3NUb2tlbiIsImVycm9yX3R5cGUiLCJvcmdVVUlEIiwicmVwb1ZhbGlkYXRpb24iLCJub3RJblJlcG9EaXNwbGF5IiwiYm9sZCIsImhvc3RzRGlmZmVyIiwic2Vzc2lvbkRpc3BsYXkiLCJjdXJyZW50RGlzcGxheSIsIl9leGhhdXN0aXZlIiwidGVsZXBvcnRGcm9tU2Vzc2lvbnNBUEkiLCJlcnIiLCJoYW5kbGVUZWxlcG9ydFByZXJlcXVpc2l0ZXMiLCJyb290IiwiZXJyb3JzVG9JZ25vcmUiLCJTZXQiLCJlcnJvcnMiLCJzaXplIiwiZXJyb3JfdHlwZXMiLCJBcnJheSIsImZyb20iLCJqb2luIiwiZXJyb3JzX2lnbm9yZWQiLCJyZXNvbHZlIiwicmVuZGVyIiwidGVsZXBvcnRUb1JlbW90ZVdpdGhFcnJvckhhbmRsaW5nIiwidGVsZXBvcnRUb1JlbW90ZSIsImluaXRpYWxNZXNzYWdlIiwib25CdW5kbGVGYWlsIiwibXNnIiwicHJvY2VzcyIsIndyaXRlIiwic3RhcnRUaW1lIiwiRGF0ZSIsIm5vdyIsImxvZ3NTdGFydFRpbWUiLCJsb2dzIiwiZmlsdGVyU3RhcnRUaW1lIiwiZmlsdGVyIiwiZW50cnkiLCJpc1NpZGVjaGFpbiIsImxlbmd0aCIsInVuZGVmaW5lZCIsImxvZyIsImlzQXhpb3NFcnJvciIsImRpbSIsIlBvbGxSZW1vdGVTZXNzaW9uUmVzcG9uc2UiLCJuZXdFdmVudHMiLCJsYXN0RXZlbnRJZCIsInNlc3Npb25TdGF0dXMiLCJwb2xsUmVtb3RlU2Vzc2lvbkV2ZW50cyIsImFmdGVySWQiLCJvcHRzIiwic2tpcE1ldGFkYXRhIiwiaGVhZGVycyIsImV2ZW50c1VybCIsIkJBU0VfQVBJX1VSTCIsIkV2ZW50c1Jlc3BvbnNlIiwiaGFzX21vcmUiLCJmaXJzdF9pZCIsImxhc3RfaWQiLCJNQVhfRVZFTlRfUEFHRVMiLCJzZGtNZXNzYWdlcyIsImN1cnNvciIsInBhZ2UiLCJldmVudHNSZXNwb25zZSIsImdldCIsInBhcmFtcyIsImFmdGVyX2lkIiwidGltZW91dCIsInN0YXR1c1RleHQiLCJldmVudHNEYXRhIiwiaXNBcnJheSIsImV2ZW50IiwicHVzaCIsInNlc3Npb25fc3RhdHVzIiwiZSIsImxldmVsIiwibW9kZWwiLCJwZXJtaXNzaW9uTW9kZSIsInVsdHJhcGxhbiIsInVzZURlZmF1bHRFbnZpcm9ubWVudCIsImVudmlyb25tZW50SWQiLCJlbnZpcm9ubWVudFZhcmlhYmxlcyIsIlJlY29yZCIsInVzZUJ1bmRsZSIsInNraXBCdW5kbGUiLCJyZXVzZU91dGNvbWVCcmFuY2giLCJnaXRodWJQciIsInJlcG8iLCJudW1iZXIiLCJlbnZWYXJzIiwiQ0xBVURFX0NPREVfT0FVVEhfVE9LRU4iLCJzZWVkQnVuZGxlRmlsZUlkIiwiYnVuZGxlIiwib2F1dGhUb2tlbiIsImJhc2VVcmwiLCJmaWxlSWQiLCJzaXplX2J5dGVzIiwiYnVuZGxlU2l6ZUJ5dGVzIiwic2NvcGUiLCJoYXNfd2lwIiwiaGFzV2lwIiwicmVhc29uIiwicmVwb0luZm8iLCJyZXZpc2lvbiIsInJlcXVlc3RCb2R5IiwiZXZlbnRzIiwic2VlZF9idW5kbGVfZmlsZV9pZCIsIm91dGNvbWVzIiwiZW52aXJvbm1lbnRfdmFyaWFibGVzIiwiZW52aXJvbm1lbnRfaWQiLCJPYmplY3QiLCJrZXlzIiwicG9zdCIsImdpdE91dGNvbWUiLCJzZXNzaW9uVGl0bGUiLCJzZXNzaW9uQnJhbmNoIiwiZ2VuZXJhdGVkIiwiZ2hWaWFibGUiLCJzb3VyY2VSZWFzb24iLCJnaXRSb290IiwiZm9yY2VCdW5kbGUiLCJlbnYiLCJDQ1JfRk9SQ0VfQlVORExFIiwiYnVuZGxlU2VlZEdhdGVPbiIsIkNDUl9FTkFCTEVfQlVORExFIiwiYWxsb3dfdW5yZXN0cmljdGVkX2dpdF9wdXNoIiwiZ2l0X2luZm8iLCJicmFuY2hlcyIsInNldHVwIiwiZmFpbFJlYXNvbiIsInBhdGgiLCJlbnZpcm9ubWVudHMiLCJtYXAiLCJraW5kIiwic2V0dGluZ3MiLCJkZWZhdWx0RW52aXJvbm1lbnRJZCIsInJlbW90ZSIsImNsb3VkRW52IiwicmV0cmllZCIsInNlbGVjdGVkRW52aXJvbm1lbnQiLCJtYXRjaGVkRGVmYXVsdCIsInNlc3Npb25Db250ZXh0IiwicmV1c2Vfb3V0Y29tZV9icmFuY2hlcyIsImdpdGh1Yl9wciIsInJlcXVlc3RfaWQiLCJyZXF1ZXN0Iiwic3VidHlwZSIsIm1vZGUiLCJ1dWlkIiwic2Vzc2lvbl9pZCIsInBhcmVudF90b29sX3VzZV9pZCIsInJvbGUiLCJpc1N1Y2Nlc3MiLCJhcmNoaXZlUmVtb3RlU2Vzc2lvbiIsInJlc3AiLCJ2YWxpZGF0ZVN0YXR1cyIsInMiXSwic291cmNlcyI6WyJ0ZWxlcG9ydC50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGF4aW9zIGZyb20gJ2F4aW9zJ1xuaW1wb3J0IGNoYWxrIGZyb20gJ2NoYWxrJ1xuaW1wb3J0IHsgcmFuZG9tVVVJRCB9IGZyb20gJ2NyeXB0bydcbmltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IGdldE9yaWdpbmFsQ3dkLCBnZXRTZXNzaW9uSWQgfSBmcm9tICdzcmMvYm9vdHN0cmFwL3N0YXRlLmpzJ1xuaW1wb3J0IHsgY2hlY2tHYXRlX0NBQ0hFRF9PUl9CTE9DS0lORyB9IGZyb20gJ3NyYy9zZXJ2aWNlcy9hbmFseXRpY3MvZ3Jvd3RoYm9vay5qcydcbmltcG9ydCB7XG4gIHR5cGUgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgbG9nRXZlbnQsXG59IGZyb20gJ3NyYy9zZXJ2aWNlcy9hbmFseXRpY3MvaW5kZXguanMnXG5pbXBvcnQgeyBpc1BvbGljeUFsbG93ZWQgfSBmcm9tICdzcmMvc2VydmljZXMvcG9saWN5TGltaXRzL2luZGV4LmpzJ1xuaW1wb3J0IHsgeiB9IGZyb20gJ3pvZC92NCdcbmltcG9ydCB7XG4gIGdldFRlbGVwb3J0RXJyb3JzLFxuICBUZWxlcG9ydEVycm9yLFxuICB0eXBlIFRlbGVwb3J0TG9jYWxFcnJvclR5cGUsXG59IGZyb20gJy4uL2NvbXBvbmVudHMvVGVsZXBvcnRFcnJvci5qcydcbmltcG9ydCB7IGdldE9hdXRoQ29uZmlnIH0gZnJvbSAnLi4vY29uc3RhbnRzL29hdXRoLmpzJ1xuaW1wb3J0IHR5cGUgeyBTREtNZXNzYWdlIH0gZnJvbSAnLi4vZW50cnlwb2ludHMvYWdlbnRTZGtUeXBlcy5qcydcbmltcG9ydCB0eXBlIHsgUm9vdCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCB7IEtleWJpbmRpbmdTZXR1cCB9IGZyb20gJy4uL2tleWJpbmRpbmdzL0tleWJpbmRpbmdQcm92aWRlclNldHVwLmpzJ1xuaW1wb3J0IHsgcXVlcnlIYWlrdSB9IGZyb20gJy4uL3NlcnZpY2VzL2FwaS9jbGF1ZGUuanMnXG5pbXBvcnQge1xuICBnZXRTZXNzaW9uTG9nc1ZpYU9BdXRoLFxuICBnZXRUZWxlcG9ydEV2ZW50cyxcbn0gZnJvbSAnLi4vc2VydmljZXMvYXBpL3Nlc3Npb25JbmdyZXNzLmpzJ1xuaW1wb3J0IHsgZ2V0T3JnYW5pemF0aW9uVVVJRCB9IGZyb20gJy4uL3NlcnZpY2VzL29hdXRoL2NsaWVudC5qcydcbmltcG9ydCB7IEFwcFN0YXRlUHJvdmlkZXIgfSBmcm9tICcuLi9zdGF0ZS9BcHBTdGF0ZS5qcydcbmltcG9ydCB0eXBlIHsgTWVzc2FnZSwgU3lzdGVtTWVzc2FnZSB9IGZyb20gJy4uL3R5cGVzL21lc3NhZ2UuanMnXG5pbXBvcnQgdHlwZSB7IFBlcm1pc3Npb25Nb2RlIH0gZnJvbSAnLi4vdHlwZXMvcGVybWlzc2lvbnMuanMnXG5pbXBvcnQge1xuICBjaGVja0FuZFJlZnJlc2hPQXV0aFRva2VuSWZOZWVkZWQsXG4gIGdldENsYXVkZUFJT0F1dGhUb2tlbnMsXG59IGZyb20gJy4vYXV0aC5qcydcbmltcG9ydCB7IGNoZWNrR2l0aHViQXBwSW5zdGFsbGVkIH0gZnJvbSAnLi9iYWNrZ3JvdW5kL3JlbW90ZS9wcmVjb25kaXRpb25zLmpzJ1xuaW1wb3J0IHtcbiAgZGVzZXJpYWxpemVNZXNzYWdlcyxcbiAgdHlwZSBUZWxlcG9ydFJlbW90ZVJlc3BvbnNlLFxufSBmcm9tICcuL2NvbnZlcnNhdGlvblJlY292ZXJ5LmpzJ1xuaW1wb3J0IHsgZ2V0Q3dkIH0gZnJvbSAnLi9jd2QuanMnXG5pbXBvcnQgeyBsb2dGb3JEZWJ1Z2dpbmcgfSBmcm9tICcuL2RlYnVnLmpzJ1xuaW1wb3J0IHtcbiAgZGV0ZWN0Q3VycmVudFJlcG9zaXRvcnlXaXRoSG9zdCxcbiAgcGFyc2VHaXRIdWJSZXBvc2l0b3J5LFxuICBwYXJzZUdpdFJlbW90ZSxcbn0gZnJvbSAnLi9kZXRlY3RSZXBvc2l0b3J5LmpzJ1xuaW1wb3J0IHsgaXNFbnZUcnV0aHkgfSBmcm9tICcuL2VudlV0aWxzLmpzJ1xuaW1wb3J0IHsgVGVsZXBvcnRPcGVyYXRpb25FcnJvciwgdG9FcnJvciB9IGZyb20gJy4vZXJyb3JzLmpzJ1xuaW1wb3J0IHsgZXhlY0ZpbGVOb1Rocm93IH0gZnJvbSAnLi9leGVjRmlsZU5vVGhyb3cuanMnXG5pbXBvcnQgeyB0cnVuY2F0ZVRvV2lkdGggfSBmcm9tICcuL2Zvcm1hdC5qcydcbmltcG9ydCB7IGZpbmRHaXRSb290LCBnZXREZWZhdWx0QnJhbmNoLCBnZXRJc0NsZWFuLCBnaXRFeGUgfSBmcm9tICcuL2dpdC5qcydcbmltcG9ydCB7IHNhZmVQYXJzZUpTT04gfSBmcm9tICcuL2pzb24uanMnXG5pbXBvcnQgeyBsb2dFcnJvciB9IGZyb20gJy4vbG9nLmpzJ1xuaW1wb3J0IHsgY3JlYXRlU3lzdGVtTWVzc2FnZSwgY3JlYXRlVXNlck1lc3NhZ2UgfSBmcm9tICcuL21lc3NhZ2VzLmpzJ1xuaW1wb3J0IHsgZ2V0TWFpbkxvb3BNb2RlbCB9IGZyb20gJy4vbW9kZWwvbW9kZWwuanMnXG5pbXBvcnQgeyBpc1RyYW5zY3JpcHRNZXNzYWdlIH0gZnJvbSAnLi9zZXNzaW9uU3RvcmFnZS5qcydcbmltcG9ydCB7IGdldFNldHRpbmdzX0RFUFJFQ0FURUQgfSBmcm9tICcuL3NldHRpbmdzL3NldHRpbmdzLmpzJ1xuaW1wb3J0IHsganNvblN0cmluZ2lmeSB9IGZyb20gJy4vc2xvd09wZXJhdGlvbnMuanMnXG5pbXBvcnQgeyBhc1N5c3RlbVByb21wdCB9IGZyb20gJy4vc3lzdGVtUHJvbXB0VHlwZS5qcydcbmltcG9ydCB7XG4gIGZldGNoU2Vzc2lvbixcbiAgdHlwZSBHaXRSZXBvc2l0b3J5T3V0Y29tZSxcbiAgdHlwZSBHaXRTb3VyY2UsXG4gIGdldEJyYW5jaEZyb21TZXNzaW9uLFxuICBnZXRPQXV0aEhlYWRlcnMsXG4gIHR5cGUgU2Vzc2lvblJlc291cmNlLFxufSBmcm9tICcuL3RlbGVwb3J0L2FwaS5qcydcbmltcG9ydCB7IGZldGNoRW52aXJvbm1lbnRzIH0gZnJvbSAnLi90ZWxlcG9ydC9lbnZpcm9ubWVudHMuanMnXG5pbXBvcnQgeyBjcmVhdGVBbmRVcGxvYWRHaXRCdW5kbGUgfSBmcm9tICcuL3RlbGVwb3J0L2dpdEJ1bmRsZS5qcydcblxuZXhwb3J0IHR5cGUgVGVsZXBvcnRSZXN1bHQgPSB7XG4gIG1lc3NhZ2VzOiBNZXNzYWdlW11cbiAgYnJhbmNoTmFtZTogc3RyaW5nXG59XG5cbmV4cG9ydCB0eXBlIFRlbGVwb3J0UHJvZ3Jlc3NTdGVwID1cbiAgfCAndmFsaWRhdGluZydcbiAgfCAnZmV0Y2hpbmdfbG9ncydcbiAgfCAnZmV0Y2hpbmdfYnJhbmNoJ1xuICB8ICdjaGVja2luZ19vdXQnXG4gIHwgJ2RvbmUnXG5cbmV4cG9ydCB0eXBlIFRlbGVwb3J0UHJvZ3Jlc3NDYWxsYmFjayA9IChzdGVwOiBUZWxlcG9ydFByb2dyZXNzU3RlcCkgPT4gdm9pZFxuXG4vKipcbiAqIENyZWF0ZXMgYSBzeXN0ZW0gbWVzc2FnZSB0byBpbmZvcm0gYWJvdXQgdGVsZXBvcnQgc2Vzc2lvbiByZXN1bWVcbiAqIEByZXR1cm5zIFN5c3RlbU1lc3NhZ2UgaW5kaWNhdGluZyBzZXNzaW9uIHdhcyByZXN1bWVkIGZyb20gYW5vdGhlciBtYWNoaW5lXG4gKi9cbmZ1bmN0aW9uIGNyZWF0ZVRlbGVwb3J0UmVzdW1lU3lzdGVtTWVzc2FnZShcbiAgYnJhbmNoRXJyb3I6IEVycm9yIHwgbnVsbCxcbik6IFN5c3RlbU1lc3NhZ2Uge1xuICBpZiAoYnJhbmNoRXJyb3IgPT09IG51bGwpIHtcbiAgICByZXR1cm4gY3JlYXRlU3lzdGVtTWVzc2FnZSgnU2Vzc2lvbiByZXN1bWVkJywgJ3N1Z2dlc3Rpb24nKVxuICB9XG4gIGNvbnN0IGZvcm1hdHRlZEVycm9yID1cbiAgICBicmFuY2hFcnJvciBpbnN0YW5jZW9mIFRlbGVwb3J0T3BlcmF0aW9uRXJyb3JcbiAgICAgID8gYnJhbmNoRXJyb3IuZm9ybWF0dGVkTWVzc2FnZVxuICAgICAgOiBicmFuY2hFcnJvci5tZXNzYWdlXG4gIHJldHVybiBjcmVhdGVTeXN0ZW1NZXNzYWdlKFxuICAgIGBTZXNzaW9uIHJlc3VtZWQgd2l0aG91dCBicmFuY2g6ICR7Zm9ybWF0dGVkRXJyb3J9YCxcbiAgICAnd2FybmluZycsXG4gIClcbn1cblxuLyoqXG4gKiBDcmVhdGVzIGEgdXNlciBtZXNzYWdlIHRvIGluZm9ybSB0aGUgbW9kZWwgYWJvdXQgdGVsZXBvcnQgc2Vzc2lvbiByZXN1bWVcbiAqIEByZXR1cm5zIFVzZXIgbWVzc2FnZSBpbmRpY2F0aW5nIHNlc3Npb24gd2FzIHJlc3VtZWQgZnJvbSBhbm90aGVyIG1hY2hpbmVcbiAqL1xuZnVuY3Rpb24gY3JlYXRlVGVsZXBvcnRSZXN1bWVVc2VyTWVzc2FnZSgpIHtcbiAgcmV0dXJuIGNyZWF0ZVVzZXJNZXNzYWdlKHtcbiAgICBjb250ZW50OiBgVGhpcyBzZXNzaW9uIGlzIGJlaW5nIGNvbnRpbnVlZCBmcm9tIGFub3RoZXIgbWFjaGluZS4gQXBwbGljYXRpb24gc3RhdGUgbWF5IGhhdmUgY2hhbmdlZC4gVGhlIHVwZGF0ZWQgd29ya2luZyBkaXJlY3RvcnkgaXMgJHtnZXRPcmlnaW5hbEN3ZCgpfWAsXG4gICAgaXNNZXRhOiB0cnVlLFxuICB9KVxufVxuXG50eXBlIFRlbGVwb3J0VG9SZW1vdGVSZXNwb25zZSA9IHtcbiAgaWQ6IHN0cmluZ1xuICB0aXRsZTogc3RyaW5nXG59XG5cbmNvbnN0IFNFU1NJT05fVElUTEVfQU5EX0JSQU5DSF9QUk9NUFQgPSBgWW91IGFyZSBjb21pbmcgdXAgd2l0aCBhIHN1Y2NpbmN0IHRpdGxlIGFuZCBnaXQgYnJhbmNoIG5hbWUgZm9yIGEgY29kaW5nIHNlc3Npb24gYmFzZWQgb24gdGhlIHByb3ZpZGVkIGRlc2NyaXB0aW9uLiBUaGUgdGl0bGUgc2hvdWxkIGJlIGNsZWFyLCBjb25jaXNlLCBhbmQgYWNjdXJhdGVseSByZWZsZWN0IHRoZSBjb250ZW50IG9mIHRoZSBjb2RpbmcgdGFzay5cbllvdSBzaG91bGQga2VlcCBpdCBzaG9ydCBhbmQgc2ltcGxlLCBpZGVhbGx5IG5vIG1vcmUgdGhhbiA2IHdvcmRzLiBBdm9pZCB1c2luZyBqYXJnb24gb3Igb3Zlcmx5IHRlY2huaWNhbCB0ZXJtcyB1bmxlc3MgYWJzb2x1dGVseSBuZWNlc3NhcnkuIFRoZSB0aXRsZSBzaG91bGQgYmUgZWFzeSB0byB1bmRlcnN0YW5kIGZvciBhbnlvbmUgcmVhZGluZyBpdC5cblVzZSBzZW50ZW5jZSBjYXNlIGZvciB0aGUgdGl0bGUgKGNhcGl0YWxpemUgb25seSB0aGUgZmlyc3Qgd29yZCBhbmQgcHJvcGVyIG5vdW5zKSwgbm90IFRpdGxlIENhc2UuXG5cblRoZSBicmFuY2ggbmFtZSBzaG91bGQgYmUgY2xlYXIsIGNvbmNpc2UsIGFuZCBhY2N1cmF0ZWx5IHJlZmxlY3QgdGhlIGNvbnRlbnQgb2YgdGhlIGNvZGluZyB0YXNrLlxuWW91IHNob3VsZCBrZWVwIGl0IHNob3J0IGFuZCBzaW1wbGUsIGlkZWFsbHkgbm8gbW9yZSB0aGFuIDQgd29yZHMuIFRoZSBicmFuY2ggc2hvdWxkIGFsd2F5cyBzdGFydCB3aXRoIFwiY2xhdWRlL1wiIGFuZCBzaG91bGQgYmUgYWxsIGxvd2VyIGNhc2UsIHdpdGggd29yZHMgc2VwYXJhdGVkIGJ5IGRhc2hlcy5cblxuUmV0dXJuIGEgSlNPTiBvYmplY3Qgd2l0aCBcInRpdGxlXCIgYW5kIFwiYnJhbmNoXCIgZmllbGRzLlxuXG5FeGFtcGxlIDE6IHtcInRpdGxlXCI6IFwiRml4IGxvZ2luIGJ1dHRvbiBub3Qgd29ya2luZyBvbiBtb2JpbGVcIiwgXCJicmFuY2hcIjogXCJjbGF1ZGUvZml4LW1vYmlsZS1sb2dpbi1idXR0b25cIn1cbkV4YW1wbGUgMjoge1widGl0bGVcIjogXCJVcGRhdGUgUkVBRE1FIHdpdGggaW5zdGFsbGF0aW9uIGluc3RydWN0aW9uc1wiLCBcImJyYW5jaFwiOiBcImNsYXVkZS91cGRhdGUtcmVhZG1lXCJ9XG5FeGFtcGxlIDM6IHtcInRpdGxlXCI6IFwiSW1wcm92ZSBwZXJmb3JtYW5jZSBvZiBkYXRhIHByb2Nlc3Npbmcgc2NyaXB0XCIsIFwiYnJhbmNoXCI6IFwiY2xhdWRlL2ltcHJvdmUtZGF0YS1wcm9jZXNzaW5nXCJ9XG5cbkhlcmUgaXMgdGhlIHNlc3Npb24gZGVzY3JpcHRpb246XG48ZGVzY3JpcHRpb24+e2Rlc2NyaXB0aW9ufTwvZGVzY3JpcHRpb24+XG5QbGVhc2UgZ2VuZXJhdGUgYSB0aXRsZSBhbmQgYnJhbmNoIG5hbWUgZm9yIHRoaXMgc2Vzc2lvbi5gXG5cbnR5cGUgVGl0bGVBbmRCcmFuY2ggPSB7XG4gIHRpdGxlOiBzdHJpbmdcbiAgYnJhbmNoTmFtZTogc3RyaW5nXG59XG5cbi8qKlxuICogR2VuZXJhdGVzIGEgdGl0bGUgYW5kIGJyYW5jaCBuYW1lIGZvciBhIGNvZGluZyBzZXNzaW9uIHVzaW5nIENsYXVkZSBIYWlrdVxuICogQHBhcmFtIGRlc2NyaXB0aW9uIFRoZSBkZXNjcmlwdGlvbi9wcm9tcHQgZm9yIHRoZSBzZXNzaW9uXG4gKiBAcmV0dXJucyBQcm9taXNlPFRpdGxlQW5kQnJhbmNoPiBUaGUgZ2VuZXJhdGVkIHRpdGxlIGFuZCBicmFuY2ggbmFtZVxuICovXG5hc3luYyBmdW5jdGlvbiBnZW5lcmF0ZVRpdGxlQW5kQnJhbmNoKFxuICBkZXNjcmlwdGlvbjogc3RyaW5nLFxuICBzaWduYWw6IEFib3J0U2lnbmFsLFxuKTogUHJvbWlzZTxUaXRsZUFuZEJyYW5jaD4ge1xuICBjb25zdCBmYWxsYmFja1RpdGxlID0gdHJ1bmNhdGVUb1dpZHRoKGRlc2NyaXB0aW9uLCA3NSlcbiAgY29uc3QgZmFsbGJhY2tCcmFuY2ggPSAnY2xhdWRlL3Rhc2snXG5cbiAgdHJ5IHtcbiAgICBjb25zdCB1c2VyUHJvbXB0ID0gU0VTU0lPTl9USVRMRV9BTkRfQlJBTkNIX1BST01QVC5yZXBsYWNlKFxuICAgICAgJ3tkZXNjcmlwdGlvbn0nLFxuICAgICAgZGVzY3JpcHRpb24sXG4gICAgKVxuXG4gICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCBxdWVyeUhhaWt1KHtcbiAgICAgIHN5c3RlbVByb21wdDogYXNTeXN0ZW1Qcm9tcHQoW10pLFxuICAgICAgdXNlclByb21wdCxcbiAgICAgIG91dHB1dEZvcm1hdDoge1xuICAgICAgICB0eXBlOiAnanNvbl9zY2hlbWEnLFxuICAgICAgICBzY2hlbWE6IHtcbiAgICAgICAgICB0eXBlOiAnb2JqZWN0JyxcbiAgICAgICAgICBwcm9wZXJ0aWVzOiB7XG4gICAgICAgICAgICB0aXRsZTogeyB0eXBlOiAnc3RyaW5nJyB9LFxuICAgICAgICAgICAgYnJhbmNoOiB7IHR5cGU6ICdzdHJpbmcnIH0sXG4gICAgICAgICAgfSxcbiAgICAgICAgICByZXF1aXJlZDogWyd0aXRsZScsICdicmFuY2gnXSxcbiAgICAgICAgICBhZGRpdGlvbmFsUHJvcGVydGllczogZmFsc2UsXG4gICAgICAgIH0sXG4gICAgICB9LFxuICAgICAgc2lnbmFsLFxuICAgICAgb3B0aW9uczoge1xuICAgICAgICBxdWVyeVNvdXJjZTogJ3RlbGVwb3J0X2dlbmVyYXRlX3RpdGxlJyxcbiAgICAgICAgYWdlbnRzOiBbXSxcbiAgICAgICAgaXNOb25JbnRlcmFjdGl2ZVNlc3Npb246IGZhbHNlLFxuICAgICAgICBoYXNBcHBlbmRTeXN0ZW1Qcm9tcHQ6IGZhbHNlLFxuICAgICAgICBtY3BUb29sczogW10sXG4gICAgICB9LFxuICAgIH0pXG5cbiAgICAvLyBFeHRyYWN0IHRleHQgZnJvbSB0aGUgcmVzcG9uc2VcbiAgICBjb25zdCBmaXJzdEJsb2NrID0gcmVzcG9uc2UubWVzc2FnZS5jb250ZW50WzBdXG4gICAgaWYgKGZpcnN0QmxvY2s/LnR5cGUgIT09ICd0ZXh0Jykge1xuICAgICAgcmV0dXJuIHsgdGl0bGU6IGZhbGxiYWNrVGl0bGUsIGJyYW5jaE5hbWU6IGZhbGxiYWNrQnJhbmNoIH1cbiAgICB9XG5cbiAgICBjb25zdCBwYXJzZWQgPSBzYWZlUGFyc2VKU09OKGZpcnN0QmxvY2sudGV4dC50cmltKCkpXG4gICAgY29uc3QgcGFyc2VSZXN1bHQgPSB6XG4gICAgICAub2JqZWN0KHsgdGl0bGU6IHouc3RyaW5nKCksIGJyYW5jaDogei5zdHJpbmcoKSB9KVxuICAgICAgLnNhZmVQYXJzZShwYXJzZWQpXG4gICAgaWYgKHBhcnNlUmVzdWx0LnN1Y2Nlc3MpIHtcbiAgICAgIHJldHVybiB7XG4gICAgICAgIHRpdGxlOiBwYXJzZVJlc3VsdC5kYXRhLnRpdGxlIHx8IGZhbGxiYWNrVGl0bGUsXG4gICAgICAgIGJyYW5jaE5hbWU6IHBhcnNlUmVzdWx0LmRhdGEuYnJhbmNoIHx8IGZhbGxiYWNrQnJhbmNoLFxuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiB7IHRpdGxlOiBmYWxsYmFja1RpdGxlLCBicmFuY2hOYW1lOiBmYWxsYmFja0JyYW5jaCB9XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgbG9nRXJyb3IobmV3IEVycm9yKGBFcnJvciBnZW5lcmF0aW5nIHRpdGxlIGFuZCBicmFuY2g6ICR7ZXJyb3J9YCkpXG4gICAgcmV0dXJuIHsgdGl0bGU6IGZhbGxiYWNrVGl0bGUsIGJyYW5jaE5hbWU6IGZhbGxiYWNrQnJhbmNoIH1cbiAgfVxufVxuXG4vKipcbiAqIFZhbGlkYXRlcyB0aGF0IHRoZSBnaXQgd29ya2luZyBkaXJlY3RvcnkgaXMgY2xlYW4gKGlnbm9yaW5nIHVudHJhY2tlZCBmaWxlcylcbiAqIFVudHJhY2tlZCBmaWxlcyBhcmUgaWdub3JlZCBiZWNhdXNlIHRoZXkgd29uJ3QgYmUgbG9zdCBkdXJpbmcgYnJhbmNoIHN3aXRjaGluZ1xuICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gdmFsaWRhdGVHaXRTdGF0ZSgpOiBQcm9taXNlPHZvaWQ+IHtcbiAgY29uc3QgaXNDbGVhbiA9IGF3YWl0IGdldElzQ2xlYW4oeyBpZ25vcmVVbnRyYWNrZWQ6IHRydWUgfSlcbiAgaWYgKCFpc0NsZWFuKSB7XG4gICAgbG9nRXZlbnQoJ3Rlbmd1X3RlbGVwb3J0X2Vycm9yX2dpdF9ub3RfY2xlYW4nLCB7fSlcbiAgICBjb25zdCBlcnJvciA9IG5ldyBUZWxlcG9ydE9wZXJhdGlvbkVycm9yKFxuICAgICAgJ0dpdCB3b3JraW5nIGRpcmVjdG9yeSBpcyBub3QgY2xlYW4uIFBsZWFzZSBjb21taXQgb3Igc3Rhc2ggeW91ciBjaGFuZ2VzIGJlZm9yZSB1c2luZyAtLXRlbGVwb3J0LicsXG4gICAgICBjaGFsay5yZWQoXG4gICAgICAgICdFcnJvcjogR2l0IHdvcmtpbmcgZGlyZWN0b3J5IGlzIG5vdCBjbGVhbi4gUGxlYXNlIGNvbW1pdCBvciBzdGFzaCB5b3VyIGNoYW5nZXMgYmVmb3JlIHVzaW5nIC0tdGVsZXBvcnQuXFxuJyxcbiAgICAgICksXG4gICAgKVxuICAgIHRocm93IGVycm9yXG4gIH1cbn1cblxuLyoqXG4gKiBGZXRjaGVzIGEgc3BlY2lmaWMgYnJhbmNoIGZyb20gcmVtb3RlIG9yaWdpblxuICogQHBhcmFtIGJyYW5jaCBUaGUgYnJhbmNoIHRvIGZldGNoLiBJZiBub3Qgc3BlY2lmaWVkLCBmZXRjaGVzIGFsbCBicmFuY2hlcy5cbiAqL1xuYXN5bmMgZnVuY3Rpb24gZmV0Y2hGcm9tT3JpZ2luKGJyYW5jaD86IHN0cmluZyk6IFByb21pc2U8dm9pZD4ge1xuICBjb25zdCBmZXRjaEFyZ3MgPSBicmFuY2hcbiAgICA/IFsnZmV0Y2gnLCAnb3JpZ2luJywgYCR7YnJhbmNofToke2JyYW5jaH1gXVxuICAgIDogWydmZXRjaCcsICdvcmlnaW4nXVxuXG4gIGNvbnN0IHsgY29kZTogZmV0Y2hDb2RlLCBzdGRlcnI6IGZldGNoU3RkZXJyIH0gPSBhd2FpdCBleGVjRmlsZU5vVGhyb3coXG4gICAgZ2l0RXhlKCksXG4gICAgZmV0Y2hBcmdzLFxuICApXG4gIGlmIChmZXRjaENvZGUgIT09IDApIHtcbiAgICAvLyBJZiBmZXRjaGluZyBhIHNwZWNpZmljIGJyYW5jaCBmYWlscywgaXQgbWlnaHQgbm90IGV4aXN0IGxvY2FsbHkgeWV0XG4gICAgLy8gVHJ5IGZldGNoaW5nIGp1c3QgdGhlIHJlZiB3aXRob3V0IG1hcHBpbmcgdG8gbG9jYWwgYnJhbmNoXG4gICAgaWYgKGJyYW5jaCAmJiBmZXRjaFN0ZGVyci5pbmNsdWRlcygncmVmc3BlYycpKSB7XG4gICAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICAgIGBTcGVjaWZpYyBicmFuY2ggZmV0Y2ggZmFpbGVkLCB0cnlpbmcgdG8gZmV0Y2ggcmVmOiAke2JyYW5jaH1gLFxuICAgICAgKVxuICAgICAgY29uc3QgeyBjb2RlOiByZWZGZXRjaENvZGUsIHN0ZGVycjogcmVmRmV0Y2hTdGRlcnIgfSA9XG4gICAgICAgIGF3YWl0IGV4ZWNGaWxlTm9UaHJvdyhnaXRFeGUoKSwgWydmZXRjaCcsICdvcmlnaW4nLCBicmFuY2hdKVxuICAgICAgaWYgKHJlZkZldGNoQ29kZSAhPT0gMCkge1xuICAgICAgICBsb2dFcnJvcihcbiAgICAgICAgICBuZXcgRXJyb3IoYEZhaWxlZCB0byBmZXRjaCBmcm9tIHJlbW90ZSBvcmlnaW46ICR7cmVmRmV0Y2hTdGRlcnJ9YCksXG4gICAgICAgIClcbiAgICAgIH1cbiAgICB9IGVsc2Uge1xuICAgICAgbG9nRXJyb3IobmV3IEVycm9yKGBGYWlsZWQgdG8gZmV0Y2ggZnJvbSByZW1vdGUgb3JpZ2luOiAke2ZldGNoU3RkZXJyfWApKVxuICAgIH1cbiAgfVxufVxuXG4vKipcbiAqIEVuc3VyZXMgdGhhdCB0aGUgY3VycmVudCBicmFuY2ggaGFzIGFuIHVwc3RyZWFtIHNldFxuICogSWYgbm90LCBzZXRzIGl0IHRvIG9yaWdpbi88YnJhbmNoTmFtZT4gaWYgdGhhdCByZW1vdGUgYnJhbmNoIGV4aXN0c1xuICovXG5hc3luYyBmdW5jdGlvbiBlbnN1cmVVcHN0cmVhbUlzU2V0KGJyYW5jaE5hbWU6IHN0cmluZyk6IFByb21pc2U8dm9pZD4ge1xuICAvLyBDaGVjayBpZiB1cHN0cmVhbSBpcyBhbHJlYWR5IHNldFxuICBjb25zdCB7IGNvZGU6IHVwc3RyZWFtQ2hlY2tDb2RlIH0gPSBhd2FpdCBleGVjRmlsZU5vVGhyb3coZ2l0RXhlKCksIFtcbiAgICAncmV2LXBhcnNlJyxcbiAgICAnLS1hYmJyZXYtcmVmJyxcbiAgICBgJHticmFuY2hOYW1lfUB7dXBzdHJlYW19YCxcbiAgXSlcblxuICBpZiAodXBzdHJlYW1DaGVja0NvZGUgPT09IDApIHtcbiAgICAvLyBVcHN0cmVhbSBpcyBhbHJlYWR5IHNldFxuICAgIGxvZ0ZvckRlYnVnZ2luZyhgQnJhbmNoICcke2JyYW5jaE5hbWV9JyBhbHJlYWR5IGhhcyB1cHN0cmVhbSBzZXRgKVxuICAgIHJldHVyblxuICB9XG5cbiAgLy8gQ2hlY2sgaWYgb3JpZ2luLzxicmFuY2hOYW1lPiBleGlzdHNcbiAgY29uc3QgeyBjb2RlOiByZW1vdGVDaGVja0NvZGUgfSA9IGF3YWl0IGV4ZWNGaWxlTm9UaHJvdyhnaXRFeGUoKSwgW1xuICAgICdyZXYtcGFyc2UnLFxuICAgICctLXZlcmlmeScsXG4gICAgYG9yaWdpbi8ke2JyYW5jaE5hbWV9YCxcbiAgXSlcblxuICBpZiAocmVtb3RlQ2hlY2tDb2RlID09PSAwKSB7XG4gICAgLy8gUmVtb3RlIGJyYW5jaCBleGlzdHMsIHNldCB1cHN0cmVhbVxuICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgIGBTZXR0aW5nIHVwc3RyZWFtIGZvciAnJHticmFuY2hOYW1lfScgdG8gJ29yaWdpbi8ke2JyYW5jaE5hbWV9J2AsXG4gICAgKVxuICAgIGNvbnN0IHsgY29kZTogc2V0VXBzdHJlYW1Db2RlLCBzdGRlcnI6IHNldFVwc3RyZWFtU3RkZXJyIH0gPVxuICAgICAgYXdhaXQgZXhlY0ZpbGVOb1Rocm93KGdpdEV4ZSgpLCBbXG4gICAgICAgICdicmFuY2gnLFxuICAgICAgICAnLS1zZXQtdXBzdHJlYW0tdG8nLFxuICAgICAgICBgb3JpZ2luLyR7YnJhbmNoTmFtZX1gLFxuICAgICAgICBicmFuY2hOYW1lLFxuICAgICAgXSlcblxuICAgIGlmIChzZXRVcHN0cmVhbUNvZGUgIT09IDApIHtcbiAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgYEZhaWxlZCB0byBzZXQgdXBzdHJlYW0gZm9yICcke2JyYW5jaE5hbWV9JzogJHtzZXRVcHN0cmVhbVN0ZGVycn1gLFxuICAgICAgKVxuICAgICAgLy8gRG9uJ3QgdGhyb3csIGp1c3QgbG9nIC0gdGhpcyBpcyBub3QgY3JpdGljYWxcbiAgICB9IGVsc2Uge1xuICAgICAgbG9nRm9yRGVidWdnaW5nKGBTdWNjZXNzZnVsbHkgc2V0IHVwc3RyZWFtIGZvciAnJHticmFuY2hOYW1lfSdgKVxuICAgIH1cbiAgfSBlbHNlIHtcbiAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICBgUmVtb3RlIGJyYW5jaCAnb3JpZ2luLyR7YnJhbmNoTmFtZX0nIGRvZXMgbm90IGV4aXN0LCBza2lwcGluZyB1cHN0cmVhbSBzZXR1cGAsXG4gICAgKVxuICB9XG59XG5cbi8qKlxuICogQ2hlY2tzIG91dCBhIHNwZWNpZmljIGJyYW5jaFxuICovXG5hc3luYyBmdW5jdGlvbiBjaGVja291dEJyYW5jaChicmFuY2hOYW1lOiBzdHJpbmcpOiBQcm9taXNlPHZvaWQ+IHtcbiAgLy8gRmlyc3QgdHJ5IHRvIGNoZWNrb3V0IHRoZSBicmFuY2ggYXMtaXMgKG1pZ2h0IGJlIGxvY2FsKVxuICBsZXQgeyBjb2RlOiBjaGVja291dENvZGUsIHN0ZGVycjogY2hlY2tvdXRTdGRlcnIgfSA9IGF3YWl0IGV4ZWNGaWxlTm9UaHJvdyhcbiAgICBnaXRFeGUoKSxcbiAgICBbJ2NoZWNrb3V0JywgYnJhbmNoTmFtZV0sXG4gIClcblxuICAvLyBJZiB0aGF0IGZhaWxzLCB0cnkgdG8gY2hlY2tvdXQgZnJvbSBvcmlnaW5cbiAgaWYgKGNoZWNrb3V0Q29kZSAhPT0gMCkge1xuICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgIGBMb2NhbCBjaGVja291dCBmYWlsZWQsIHRyeWluZyB0byBjaGVja291dCBmcm9tIG9yaWdpbjogJHtjaGVja291dFN0ZGVycn1gLFxuICAgIClcblxuICAgIC8vIFRyeSB0byBjaGVja291dCB0aGUgcmVtb3RlIGJyYW5jaCBhbmQgY3JlYXRlIGEgbG9jYWwgdHJhY2tpbmcgYnJhbmNoXG4gICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgZXhlY0ZpbGVOb1Rocm93KGdpdEV4ZSgpLCBbXG4gICAgICAnY2hlY2tvdXQnLFxuICAgICAgJy1iJyxcbiAgICAgIGJyYW5jaE5hbWUsXG4gICAgICAnLS10cmFjaycsXG4gICAgICBgb3JpZ2luLyR7YnJhbmNoTmFtZX1gLFxuICAgIF0pXG5cbiAgICBjaGVja291dENvZGUgPSByZXN1bHQuY29kZVxuICAgIGNoZWNrb3V0U3RkZXJyID0gcmVzdWx0LnN0ZGVyclxuXG4gICAgLy8gSWYgdGhhdCBhbHNvIGZhaWxzLCB0cnkgd2l0aG91dCAtYiBpbiBjYXNlIHRoZSBicmFuY2ggZXhpc3RzIGJ1dCBpc24ndCBjaGVja2VkIG91dFxuICAgIGlmIChjaGVja291dENvZGUgIT09IDApIHtcbiAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgYFJlbW90ZSBjaGVja291dCB3aXRoIC1iIGZhaWxlZCwgdHJ5aW5nIHdpdGhvdXQgLWI6ICR7Y2hlY2tvdXRTdGRlcnJ9YCxcbiAgICAgIClcbiAgICAgIGNvbnN0IGZpbmFsUmVzdWx0ID0gYXdhaXQgZXhlY0ZpbGVOb1Rocm93KGdpdEV4ZSgpLCBbXG4gICAgICAgICdjaGVja291dCcsXG4gICAgICAgICctLXRyYWNrJyxcbiAgICAgICAgYG9yaWdpbi8ke2JyYW5jaE5hbWV9YCxcbiAgICAgIF0pXG4gICAgICBjaGVja291dENvZGUgPSBmaW5hbFJlc3VsdC5jb2RlXG4gICAgICBjaGVja291dFN0ZGVyciA9IGZpbmFsUmVzdWx0LnN0ZGVyclxuICAgIH1cbiAgfVxuXG4gIGlmIChjaGVja291dENvZGUgIT09IDApIHtcbiAgICBsb2dFdmVudCgndGVuZ3VfdGVsZXBvcnRfZXJyb3JfYnJhbmNoX2NoZWNrb3V0X2ZhaWxlZCcsIHt9KVxuICAgIHRocm93IG5ldyBUZWxlcG9ydE9wZXJhdGlvbkVycm9yKFxuICAgICAgYEZhaWxlZCB0byBjaGVja291dCBicmFuY2ggJyR7YnJhbmNoTmFtZX0nOiAke2NoZWNrb3V0U3RkZXJyfWAsXG4gICAgICBjaGFsay5yZWQoYEZhaWxlZCB0byBjaGVja291dCBicmFuY2ggJyR7YnJhbmNoTmFtZX0nXFxuYCksXG4gICAgKVxuICB9XG5cbiAgLy8gQWZ0ZXIgc3VjY2Vzc2Z1bCBjaGVja291dCwgZW5zdXJlIHVwc3RyZWFtIGlzIHNldFxuICBhd2FpdCBlbnN1cmVVcHN0cmVhbUlzU2V0KGJyYW5jaE5hbWUpXG59XG5cbi8qKlxuICogR2V0cyB0aGUgY3VycmVudCBicmFuY2ggbmFtZVxuICovXG5hc3luYyBmdW5jdGlvbiBnZXRDdXJyZW50QnJhbmNoKCk6IFByb21pc2U8c3RyaW5nPiB7XG4gIGNvbnN0IHsgc3Rkb3V0OiBjdXJyZW50QnJhbmNoIH0gPSBhd2FpdCBleGVjRmlsZU5vVGhyb3coZ2l0RXhlKCksIFtcbiAgICAnYnJhbmNoJyxcbiAgICAnLS1zaG93LWN1cnJlbnQnLFxuICBdKVxuICByZXR1cm4gY3VycmVudEJyYW5jaC50cmltKClcbn1cblxuLyoqXG4gKiBQcm9jZXNzZXMgbWVzc2FnZXMgZm9yIHRlbGVwb3J0IHJlc3VtZSwgcmVtb3ZpbmcgaW5jb21wbGV0ZSB0b29sX3VzZSBibG9ja3NcbiAqIGFuZCBhZGRpbmcgdGVsZXBvcnQgbm90aWNlIG1lc3NhZ2VzXG4gKiBAcGFyYW0gbWVzc2FnZXMgVGhlIGNvbnZlcnNhdGlvbiBtZXNzYWdlc1xuICogQHBhcmFtIGVycm9yIE9wdGlvbmFsIGVycm9yIGZyb20gYnJhbmNoIGNoZWNrb3V0XG4gKiBAcmV0dXJucyBQcm9jZXNzZWQgbWVzc2FnZXMgcmVhZHkgZm9yIHJlc3VtZVxuICovXG5leHBvcnQgZnVuY3Rpb24gcHJvY2Vzc01lc3NhZ2VzRm9yVGVsZXBvcnRSZXN1bWUoXG4gIG1lc3NhZ2VzOiBNZXNzYWdlW10sXG4gIGVycm9yOiBFcnJvciB8IG51bGwsXG4pOiBNZXNzYWdlW10ge1xuICAvLyBTaGFyZWQgbG9naWMgd2l0aCByZXN1bWUgZm9yIGhhbmRsaW5nIGludGVycnVwZWQgc2Vzc2lvbiB0cmFuc2NyaXB0c1xuICBjb25zdCBkZXNlcmlhbGl6ZWRNZXNzYWdlcyA9IGRlc2VyaWFsaXplTWVzc2FnZXMobWVzc2FnZXMpXG5cbiAgLy8gQWRkIHVzZXIgbWVzc2FnZSBhYm91dCB0ZWxlcG9ydCByZXN1bWUgKHZpc2libGUgdG8gbW9kZWwpXG4gIGNvbnN0IG1lc3NhZ2VzV2l0aFRlbGVwb3J0Tm90aWNlID0gW1xuICAgIC4uLmRlc2VyaWFsaXplZE1lc3NhZ2VzLFxuICAgIGNyZWF0ZVRlbGVwb3J0UmVzdW1lVXNlck1lc3NhZ2UoKSxcbiAgICBjcmVhdGVUZWxlcG9ydFJlc3VtZVN5c3RlbU1lc3NhZ2UoZXJyb3IpLFxuICBdXG5cbiAgcmV0dXJuIG1lc3NhZ2VzV2l0aFRlbGVwb3J0Tm90aWNlXG59XG5cbi8qKlxuICogQ2hlY2tzIG91dCB0aGUgc3BlY2lmaWVkIGJyYW5jaCBmb3IgYSB0ZWxlcG9ydGVkIHNlc3Npb25cbiAqIEBwYXJhbSBicmFuY2ggT3B0aW9uYWwgYnJhbmNoIHRvIGNoZWNrb3V0XG4gKiBAcmV0dXJucyBUaGUgY3VycmVudCBicmFuY2ggbmFtZSBhbmQgYW55IGVycm9yIHRoYXQgb2NjdXJyZWRcbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGNoZWNrT3V0VGVsZXBvcnRlZFNlc3Npb25CcmFuY2goXG4gIGJyYW5jaD86IHN0cmluZyxcbik6IFByb21pc2U8eyBicmFuY2hOYW1lOiBzdHJpbmc7IGJyYW5jaEVycm9yOiBFcnJvciB8IG51bGwgfT4ge1xuICB0cnkge1xuICAgIGNvbnN0IGN1cnJlbnRCcmFuY2ggPSBhd2FpdCBnZXRDdXJyZW50QnJhbmNoKClcbiAgICBsb2dGb3JEZWJ1Z2dpbmcoYEN1cnJlbnQgYnJhbmNoIGJlZm9yZSB0ZWxlcG9ydDogJyR7Y3VycmVudEJyYW5jaH0nYClcblxuICAgIGlmIChicmFuY2gpIHtcbiAgICAgIGxvZ0ZvckRlYnVnZ2luZyhgU3dpdGNoaW5nIHRvIGJyYW5jaCAnJHticmFuY2h9Jy4uLmApXG4gICAgICBhd2FpdCBmZXRjaEZyb21PcmlnaW4oYnJhbmNoKVxuICAgICAgYXdhaXQgY2hlY2tvdXRCcmFuY2goYnJhbmNoKVxuICAgICAgY29uc3QgbmV3QnJhbmNoID0gYXdhaXQgZ2V0Q3VycmVudEJyYW5jaCgpXG4gICAgICBsb2dGb3JEZWJ1Z2dpbmcoYEJyYW5jaCBhZnRlciBjaGVja291dDogJyR7bmV3QnJhbmNofSdgKVxuICAgIH0gZWxzZSB7XG4gICAgICBsb2dGb3JEZWJ1Z2dpbmcoJ05vIGJyYW5jaCBzcGVjaWZpZWQsIHN0YXlpbmcgb24gY3VycmVudCBicmFuY2gnKVxuICAgIH1cblxuICAgIGNvbnN0IGJyYW5jaE5hbWUgPSBhd2FpdCBnZXRDdXJyZW50QnJhbmNoKClcbiAgICByZXR1cm4geyBicmFuY2hOYW1lLCBicmFuY2hFcnJvcjogbnVsbCB9XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgY29uc3QgYnJhbmNoTmFtZSA9IGF3YWl0IGdldEN1cnJlbnRCcmFuY2goKVxuICAgIGNvbnN0IGJyYW5jaEVycm9yID0gdG9FcnJvcihlcnJvcilcbiAgICByZXR1cm4geyBicmFuY2hOYW1lLCBicmFuY2hFcnJvciB9XG4gIH1cbn1cblxuLyoqXG4gKiBSZXN1bHQgb2YgcmVwb3NpdG9yeSB2YWxpZGF0aW9uIGZvciB0ZWxlcG9ydFxuICovXG5leHBvcnQgdHlwZSBSZXBvVmFsaWRhdGlvblJlc3VsdCA9IHtcbiAgc3RhdHVzOiAnbWF0Y2gnIHwgJ21pc21hdGNoJyB8ICdub3RfaW5fcmVwbycgfCAnbm9fcmVwb19yZXF1aXJlZCcgfCAnZXJyb3InXG4gIHNlc3Npb25SZXBvPzogc3RyaW5nXG4gIGN1cnJlbnRSZXBvPzogc3RyaW5nIHwgbnVsbFxuICAvKiogSG9zdCBvZiB0aGUgc2Vzc2lvbiByZXBvIChlLmcuIFwiZ2l0aHViLmNvbVwiIG9yIFwiZ2hlLmNvcnAuY29tXCIpIOKAlCBmb3IgZGlzcGxheSBvbmx5ICovXG4gIHNlc3Npb25Ib3N0Pzogc3RyaW5nXG4gIC8qKiBIb3N0IG9mIHRoZSBjdXJyZW50IHJlcG8gKGUuZy4gXCJnaXRodWIuY29tXCIgb3IgXCJnaGUuY29ycC5jb21cIikg4oCUIGZvciBkaXNwbGF5IG9ubHkgKi9cbiAgY3VycmVudEhvc3Q/OiBzdHJpbmdcbiAgZXJyb3JNZXNzYWdlPzogc3RyaW5nXG59XG5cbi8qKlxuICogVmFsaWRhdGVzIHRoYXQgdGhlIGN1cnJlbnQgcmVwb3NpdG9yeSBtYXRjaGVzIHRoZSBzZXNzaW9uJ3MgcmVwb3NpdG9yeS5cbiAqIFJldHVybnMgYSByZXN1bHQgb2JqZWN0IGluc3RlYWQgb2YgdGhyb3dpbmcsIGFsbG93aW5nIHRoZSBjYWxsZXIgdG8gaGFuZGxlIG1pc21hdGNoZXMuXG4gKlxuICogQHBhcmFtIHNlc3Npb25EYXRhIFRoZSBzZXNzaW9uIHJlc291cmNlIHRvIHZhbGlkYXRlIGFnYWluc3RcbiAqIEByZXR1cm5zIFZhbGlkYXRpb24gcmVzdWx0IHdpdGggc3RhdHVzIGFuZCByZXBvIGluZm9ybWF0aW9uXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiB2YWxpZGF0ZVNlc3Npb25SZXBvc2l0b3J5KFxuICBzZXNzaW9uRGF0YTogU2Vzc2lvblJlc291cmNlLFxuKTogUHJvbWlzZTxSZXBvVmFsaWRhdGlvblJlc3VsdD4ge1xuICBjb25zdCBjdXJyZW50UGFyc2VkID0gYXdhaXQgZGV0ZWN0Q3VycmVudFJlcG9zaXRvcnlXaXRoSG9zdCgpXG4gIGNvbnN0IGN1cnJlbnRSZXBvID0gY3VycmVudFBhcnNlZFxuICAgID8gYCR7Y3VycmVudFBhcnNlZC5vd25lcn0vJHtjdXJyZW50UGFyc2VkLm5hbWV9YFxuICAgIDogbnVsbFxuXG4gIGNvbnN0IGdpdFNvdXJjZSA9IHNlc3Npb25EYXRhLnNlc3Npb25fY29udGV4dC5zb3VyY2VzLmZpbmQoXG4gICAgKHNvdXJjZSk6IHNvdXJjZSBpcyBHaXRTb3VyY2UgPT4gc291cmNlLnR5cGUgPT09ICdnaXRfcmVwb3NpdG9yeScsXG4gIClcblxuICBpZiAoIWdpdFNvdXJjZT8udXJsKSB7XG4gICAgLy8gU2Vzc2lvbiBoYXMgbm8gcmVwbyByZXF1aXJlbWVudFxuICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgIGN1cnJlbnRSZXBvXG4gICAgICAgID8gJ1Nlc3Npb24gaGFzIG5vIGFzc29jaWF0ZWQgcmVwb3NpdG9yeSwgcHJvY2VlZGluZyB3aXRob3V0IHZhbGlkYXRpb24nXG4gICAgICAgIDogJ1Nlc3Npb24gaGFzIG5vIHJlcG8gcmVxdWlyZW1lbnQgYW5kIG5vdCBpbiBnaXQgZGlyZWN0b3J5LCBwcm9jZWVkaW5nJyxcbiAgICApXG4gICAgcmV0dXJuIHsgc3RhdHVzOiAnbm9fcmVwb19yZXF1aXJlZCcgfVxuICB9XG5cbiAgY29uc3Qgc2Vzc2lvblBhcnNlZCA9IHBhcnNlR2l0UmVtb3RlKGdpdFNvdXJjZS51cmwpXG4gIGNvbnN0IHNlc3Npb25SZXBvID0gc2Vzc2lvblBhcnNlZFxuICAgID8gYCR7c2Vzc2lvblBhcnNlZC5vd25lcn0vJHtzZXNzaW9uUGFyc2VkLm5hbWV9YFxuICAgIDogcGFyc2VHaXRIdWJSZXBvc2l0b3J5KGdpdFNvdXJjZS51cmwpXG4gIGlmICghc2Vzc2lvblJlcG8pIHtcbiAgICByZXR1cm4geyBzdGF0dXM6ICdub19yZXBvX3JlcXVpcmVkJyB9XG4gIH1cblxuICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgYFNlc3Npb24gaXMgZm9yIHJlcG9zaXRvcnk6ICR7c2Vzc2lvblJlcG99LCBjdXJyZW50IHJlcG86ICR7Y3VycmVudFJlcG8gPz8gJ25vbmUnfWAsXG4gIClcblxuICBpZiAoIWN1cnJlbnRSZXBvKSB7XG4gICAgLy8gTm90IGluIGEgZ2l0IHJlcG8sIGJ1dCBzZXNzaW9uIHJlcXVpcmVzIG9uZVxuICAgIHJldHVybiB7XG4gICAgICBzdGF0dXM6ICdub3RfaW5fcmVwbycsXG4gICAgICBzZXNzaW9uUmVwbyxcbiAgICAgIHNlc3Npb25Ib3N0OiBzZXNzaW9uUGFyc2VkPy5ob3N0LFxuICAgICAgY3VycmVudFJlcG86IG51bGwsXG4gICAgfVxuICB9XG5cbiAgLy8gQ29tcGFyZSBib3RoIG93bmVyL3JlcG8gYW5kIGhvc3QgdG8gYXZvaWQgY3Jvc3MtaW5zdGFuY2UgbWlzbWF0Y2hlcy5cbiAgLy8gU3RyaXAgcG9ydHMgYmVmb3JlIGNvbXBhcmluZyBob3N0cyDigJQgU1NIIHJlbW90ZXMgb21pdCB0aGUgcG9ydCB3aGlsZVxuICAvLyBIVFRQUyByZW1vdGVzIG1heSBpbmNsdWRlIGEgbm9uLXN0YW5kYXJkIHBvcnQgKGUuZy4gZ2hlLmNvcnAuY29tOjg0NDMpLFxuICAvLyB3aGljaCB3b3VsZCBjYXVzZSBhIGZhbHNlIG1pc21hdGNoLlxuICBjb25zdCBzdHJpcFBvcnQgPSAoaG9zdDogc3RyaW5nKTogc3RyaW5nID0+IGhvc3QucmVwbGFjZSgvOlxcZCskLywgJycpXG4gIGNvbnN0IHJlcG9NYXRjaCA9IGN1cnJlbnRSZXBvLnRvTG93ZXJDYXNlKCkgPT09IHNlc3Npb25SZXBvLnRvTG93ZXJDYXNlKClcbiAgY29uc3QgaG9zdE1hdGNoID1cbiAgICAhY3VycmVudFBhcnNlZCB8fFxuICAgICFzZXNzaW9uUGFyc2VkIHx8XG4gICAgc3RyaXBQb3J0KGN1cnJlbnRQYXJzZWQuaG9zdC50b0xvd2VyQ2FzZSgpKSA9PT1cbiAgICAgIHN0cmlwUG9ydChzZXNzaW9uUGFyc2VkLmhvc3QudG9Mb3dlckNhc2UoKSlcblxuICBpZiAocmVwb01hdGNoICYmIGhvc3RNYXRjaCkge1xuICAgIHJldHVybiB7XG4gICAgICBzdGF0dXM6ICdtYXRjaCcsXG4gICAgICBzZXNzaW9uUmVwbyxcbiAgICAgIGN1cnJlbnRSZXBvLFxuICAgIH1cbiAgfVxuXG4gIC8vIFJlcG8gbWlzbWF0Y2gg4oCUIGtlZXAgc2Vzc2lvblJlcG8vY3VycmVudFJlcG8gYXMgcGxhaW4gXCJvd25lci9yZXBvXCIgc29cbiAgLy8gZG93bnN0cmVhbSBjb25zdW1lcnMgKGUuZy4gZ2V0S25vd25QYXRoc0ZvclJlcG8pIGNhbiB1c2UgdGhlbSBhcyBsb29rdXAga2V5cy5cbiAgLy8gSW5jbHVkZSBob3N0IGluZm9ybWF0aW9uIGluIHNlcGFyYXRlIGZpZWxkcyBmb3IgZGlzcGxheSBwdXJwb3Nlcy5cbiAgcmV0dXJuIHtcbiAgICBzdGF0dXM6ICdtaXNtYXRjaCcsXG4gICAgc2Vzc2lvblJlcG8sXG4gICAgY3VycmVudFJlcG8sXG4gICAgc2Vzc2lvbkhvc3Q6IHNlc3Npb25QYXJzZWQ/Lmhvc3QsXG4gICAgY3VycmVudEhvc3Q6IGN1cnJlbnRQYXJzZWQ/Lmhvc3QsXG4gIH1cbn1cblxuLyoqXG4gKiBIYW5kbGVzIHRlbGVwb3J0aW5nIGZyb20gYSBjb2RlIHNlc3Npb24gSUQuXG4gKiBGZXRjaGVzIHNlc3Npb24gbG9ncyBhbmQgdmFsaWRhdGVzIHJlcG8uXG4gKiBAcGFyYW0gc2Vzc2lvbklkIFRoZSBzZXNzaW9uIElEIHRvIHJlc3VtZVxuICogQHBhcmFtIG9uUHJvZ3Jlc3MgT3B0aW9uYWwgY2FsbGJhY2sgZm9yIHByb2dyZXNzIHVwZGF0ZXNcbiAqIEByZXR1cm5zIFRoZSByYXcgc2Vzc2lvbiBsb2cgYW5kIGJyYW5jaCBuYW1lXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiB0ZWxlcG9ydFJlc3VtZUNvZGVTZXNzaW9uKFxuICBzZXNzaW9uSWQ6IHN0cmluZyxcbiAgb25Qcm9ncmVzcz86IFRlbGVwb3J0UHJvZ3Jlc3NDYWxsYmFjayxcbik6IFByb21pc2U8VGVsZXBvcnRSZW1vdGVSZXNwb25zZT4ge1xuICBpZiAoIWlzUG9saWN5QWxsb3dlZCgnYWxsb3dfcmVtb3RlX3Nlc3Npb25zJykpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICBcIlJlbW90ZSBzZXNzaW9ucyBhcmUgZGlzYWJsZWQgYnkgeW91ciBvcmdhbml6YXRpb24ncyBwb2xpY3kuXCIsXG4gICAgKVxuICB9XG5cbiAgbG9nRm9yRGVidWdnaW5nKGBSZXN1bWluZyBjb2RlIHNlc3Npb24gSUQ6ICR7c2Vzc2lvbklkfWApXG5cbiAgdHJ5IHtcbiAgICBjb25zdCBhY2Nlc3NUb2tlbiA9IGdldENsYXVkZUFJT0F1dGhUb2tlbnMoKT8uYWNjZXNzVG9rZW5cbiAgICBpZiAoIWFjY2Vzc1Rva2VuKSB7XG4gICAgICBsb2dFdmVudCgndGVuZ3VfdGVsZXBvcnRfcmVzdW1lX2Vycm9yJywge1xuICAgICAgICBlcnJvcl90eXBlOlxuICAgICAgICAgICdub19hY2Nlc3NfdG9rZW4nIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICB9KVxuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICAnQ2xhdWRlIENvZGUgd2ViIHNlc3Npb25zIHJlcXVpcmUgYXV0aGVudGljYXRpb24gd2l0aCBhIENsYXVkZS5haSBhY2NvdW50LiBBUEkga2V5IGF1dGhlbnRpY2F0aW9uIGlzIG5vdCBzdWZmaWNpZW50LiBQbGVhc2UgcnVuIC9sb2dpbiB0byBhdXRoZW50aWNhdGUsIG9yIGNoZWNrIHlvdXIgYXV0aGVudGljYXRpb24gc3RhdHVzIHdpdGggL3N0YXR1cy4nLFxuICAgICAgKVxuICAgIH1cblxuICAgIC8vIEdldCBvcmdhbml6YXRpb24gVVVJRFxuICAgIGNvbnN0IG9yZ1VVSUQgPSBhd2FpdCBnZXRPcmdhbml6YXRpb25VVUlEKClcbiAgICBpZiAoIW9yZ1VVSUQpIHtcbiAgICAgIGxvZ0V2ZW50KCd0ZW5ndV90ZWxlcG9ydF9yZXN1bWVfZXJyb3InLCB7XG4gICAgICAgIGVycm9yX3R5cGU6XG4gICAgICAgICAgJ25vX29yZ191dWlkJyBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgfSlcbiAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgJ1VuYWJsZSB0byBnZXQgb3JnYW5pemF0aW9uIFVVSUQgZm9yIGNvbnN0cnVjdGluZyBzZXNzaW9uIFVSTCcsXG4gICAgICApXG4gICAgfVxuXG4gICAgLy8gRmV0Y2ggYW5kIHZhbGlkYXRlIHJlcG9zaXRvcnkgbWF0Y2hlcyBiZWZvcmUgcmVzdW1pbmdcbiAgICBvblByb2dyZXNzPy4oJ3ZhbGlkYXRpbmcnKVxuICAgIGNvbnN0IHNlc3Npb25EYXRhID0gYXdhaXQgZmV0Y2hTZXNzaW9uKHNlc3Npb25JZClcbiAgICBjb25zdCByZXBvVmFsaWRhdGlvbiA9IGF3YWl0IHZhbGlkYXRlU2Vzc2lvblJlcG9zaXRvcnkoc2Vzc2lvbkRhdGEpXG5cbiAgICBzd2l0Y2ggKHJlcG9WYWxpZGF0aW9uLnN0YXR1cykge1xuICAgICAgY2FzZSAnbWF0Y2gnOlxuICAgICAgY2FzZSAnbm9fcmVwb19yZXF1aXJlZCc6XG4gICAgICAgIC8vIFByb2NlZWQgd2l0aCB0ZWxlcG9ydFxuICAgICAgICBicmVha1xuICAgICAgY2FzZSAnbm90X2luX3JlcG8nOiB7XG4gICAgICAgIGxvZ0V2ZW50KCd0ZW5ndV90ZWxlcG9ydF9lcnJvcl9yZXBvX25vdF9pbl9naXRfZGlyX3Nlc3Npb25zX2FwaScsIHtcbiAgICAgICAgICBzZXNzaW9uSWQ6XG4gICAgICAgICAgICBzZXNzaW9uSWQgYXMgQW5hbHl0aWNzTWV0YWRhdGFfSV9WRVJJRklFRF9USElTX0lTX05PVF9DT0RFX09SX0ZJTEVQQVRIUyxcbiAgICAgICAgfSlcbiAgICAgICAgLy8gSW5jbHVkZSBob3N0IGZvciBHSEUgdXNlcnMgc28gdGhleSBrbm93IHdoaWNoIGluc3RhbmNlIHRoZSByZXBvIGlzIG9uXG4gICAgICAgIGNvbnN0IG5vdEluUmVwb0Rpc3BsYXkgPVxuICAgICAgICAgIHJlcG9WYWxpZGF0aW9uLnNlc3Npb25Ib3N0ICYmXG4gICAgICAgICAgcmVwb1ZhbGlkYXRpb24uc2Vzc2lvbkhvc3QudG9Mb3dlckNhc2UoKSAhPT0gJ2dpdGh1Yi5jb20nXG4gICAgICAgICAgICA/IGAke3JlcG9WYWxpZGF0aW9uLnNlc3Npb25Ib3N0fS8ke3JlcG9WYWxpZGF0aW9uLnNlc3Npb25SZXBvfWBcbiAgICAgICAgICAgIDogcmVwb1ZhbGlkYXRpb24uc2Vzc2lvblJlcG9cbiAgICAgICAgdGhyb3cgbmV3IFRlbGVwb3J0T3BlcmF0aW9uRXJyb3IoXG4gICAgICAgICAgYFlvdSBtdXN0IHJ1biBjbGF1ZGUgLS10ZWxlcG9ydCAke3Nlc3Npb25JZH0gZnJvbSBhIGNoZWNrb3V0IG9mICR7bm90SW5SZXBvRGlzcGxheX0uYCxcbiAgICAgICAgICBjaGFsay5yZWQoXG4gICAgICAgICAgICBgWW91IG11c3QgcnVuIGNsYXVkZSAtLXRlbGVwb3J0ICR7c2Vzc2lvbklkfSBmcm9tIGEgY2hlY2tvdXQgb2YgJHtjaGFsay5ib2xkKG5vdEluUmVwb0Rpc3BsYXkpfS5cXG5gLFxuICAgICAgICAgICksXG4gICAgICAgIClcbiAgICAgIH1cbiAgICAgIGNhc2UgJ21pc21hdGNoJzoge1xuICAgICAgICBsb2dFdmVudCgndGVuZ3VfdGVsZXBvcnRfZXJyb3JfcmVwb19taXNtYXRjaF9zZXNzaW9uc19hcGknLCB7XG4gICAgICAgICAgc2Vzc2lvbklkOlxuICAgICAgICAgICAgc2Vzc2lvbklkIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIH0pXG4gICAgICAgIC8vIE9ubHkgaW5jbHVkZSBob3N0IHByZWZpeCB3aGVuIGhvc3RzIGFjdHVhbGx5IGRpZmZlciB0byBkaXNhbWJpZ3VhdGVcbiAgICAgICAgLy8gY3Jvc3MtaW5zdGFuY2UgbWlzbWF0Y2hlczsgZm9yIHNhbWUtaG9zdCBtaXNtYXRjaGVzIHRoZSBob3N0IGlzIG5vaXNlLlxuICAgICAgICBjb25zdCBob3N0c0RpZmZlciA9XG4gICAgICAgICAgcmVwb1ZhbGlkYXRpb24uc2Vzc2lvbkhvc3QgJiZcbiAgICAgICAgICByZXBvVmFsaWRhdGlvbi5jdXJyZW50SG9zdCAmJlxuICAgICAgICAgIHJlcG9WYWxpZGF0aW9uLnNlc3Npb25Ib3N0LnJlcGxhY2UoLzpcXGQrJC8sICcnKS50b0xvd2VyQ2FzZSgpICE9PVxuICAgICAgICAgICAgcmVwb1ZhbGlkYXRpb24uY3VycmVudEhvc3QucmVwbGFjZSgvOlxcZCskLywgJycpLnRvTG93ZXJDYXNlKClcbiAgICAgICAgY29uc3Qgc2Vzc2lvbkRpc3BsYXkgPSBob3N0c0RpZmZlclxuICAgICAgICAgID8gYCR7cmVwb1ZhbGlkYXRpb24uc2Vzc2lvbkhvc3R9LyR7cmVwb1ZhbGlkYXRpb24uc2Vzc2lvblJlcG99YFxuICAgICAgICAgIDogcmVwb1ZhbGlkYXRpb24uc2Vzc2lvblJlcG9cbiAgICAgICAgY29uc3QgY3VycmVudERpc3BsYXkgPSBob3N0c0RpZmZlclxuICAgICAgICAgID8gYCR7cmVwb1ZhbGlkYXRpb24uY3VycmVudEhvc3R9LyR7cmVwb1ZhbGlkYXRpb24uY3VycmVudFJlcG99YFxuICAgICAgICAgIDogcmVwb1ZhbGlkYXRpb24uY3VycmVudFJlcG9cbiAgICAgICAgdGhyb3cgbmV3IFRlbGVwb3J0T3BlcmF0aW9uRXJyb3IoXG4gICAgICAgICAgYFlvdSBtdXN0IHJ1biBjbGF1ZGUgLS10ZWxlcG9ydCAke3Nlc3Npb25JZH0gZnJvbSBhIGNoZWNrb3V0IG9mICR7c2Vzc2lvbkRpc3BsYXl9LlxcblRoaXMgcmVwbyBpcyAke2N1cnJlbnREaXNwbGF5fS5gLFxuICAgICAgICAgIGNoYWxrLnJlZChcbiAgICAgICAgICAgIGBZb3UgbXVzdCBydW4gY2xhdWRlIC0tdGVsZXBvcnQgJHtzZXNzaW9uSWR9IGZyb20gYSBjaGVja291dCBvZiAke2NoYWxrLmJvbGQoc2Vzc2lvbkRpc3BsYXkpfS5cXG5UaGlzIHJlcG8gaXMgJHtjaGFsay5ib2xkKGN1cnJlbnREaXNwbGF5KX0uXFxuYCxcbiAgICAgICAgICApLFxuICAgICAgICApXG4gICAgICB9XG4gICAgICBjYXNlICdlcnJvcic6XG4gICAgICAgIHRocm93IG5ldyBUZWxlcG9ydE9wZXJhdGlvbkVycm9yKFxuICAgICAgICAgIHJlcG9WYWxpZGF0aW9uLmVycm9yTWVzc2FnZSB8fFxuICAgICAgICAgICAgJ0ZhaWxlZCB0byB2YWxpZGF0ZSBzZXNzaW9uIHJlcG9zaXRvcnknLFxuICAgICAgICAgIGNoYWxrLnJlZChcbiAgICAgICAgICAgIGBFcnJvcjogJHtyZXBvVmFsaWRhdGlvbi5lcnJvck1lc3NhZ2UgfHwgJ0ZhaWxlZCB0byB2YWxpZGF0ZSBzZXNzaW9uIHJlcG9zaXRvcnknfVxcbmAsXG4gICAgICAgICAgKSxcbiAgICAgICAgKVxuICAgICAgZGVmYXVsdDoge1xuICAgICAgICBjb25zdCBfZXhoYXVzdGl2ZTogbmV2ZXIgPSByZXBvVmFsaWRhdGlvbi5zdGF0dXNcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBVbmhhbmRsZWQgcmVwbyB2YWxpZGF0aW9uIHN0YXR1czogJHtfZXhoYXVzdGl2ZX1gKVxuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiBhd2FpdCB0ZWxlcG9ydEZyb21TZXNzaW9uc0FQSShcbiAgICAgIHNlc3Npb25JZCxcbiAgICAgIG9yZ1VVSUQsXG4gICAgICBhY2Nlc3NUb2tlbixcbiAgICAgIG9uUHJvZ3Jlc3MsXG4gICAgICBzZXNzaW9uRGF0YSxcbiAgICApXG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgaWYgKGVycm9yIGluc3RhbmNlb2YgVGVsZXBvcnRPcGVyYXRpb25FcnJvcikge1xuICAgICAgdGhyb3cgZXJyb3JcbiAgICB9XG5cbiAgICBjb25zdCBlcnIgPSB0b0Vycm9yKGVycm9yKVxuICAgIGxvZ0Vycm9yKGVycilcbiAgICBsb2dFdmVudCgndGVuZ3VfdGVsZXBvcnRfcmVzdW1lX2Vycm9yJywge1xuICAgICAgZXJyb3JfdHlwZTpcbiAgICAgICAgJ3Jlc3VtZV9zZXNzaW9uX2lkX2NhdGNoJyBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgIH0pXG5cbiAgICB0aHJvdyBuZXcgVGVsZXBvcnRPcGVyYXRpb25FcnJvcihcbiAgICAgIGVyci5tZXNzYWdlLFxuICAgICAgY2hhbGsucmVkKGBFcnJvcjogJHtlcnIubWVzc2FnZX1cXG5gKSxcbiAgICApXG4gIH1cbn1cblxuLyoqXG4gKiBIZWxwZXIgZnVuY3Rpb24gdG8gaGFuZGxlIHRlbGVwb3J0IHByZXJlcXVpc2l0ZXMgKGF1dGhlbnRpY2F0aW9uIGFuZCBnaXQgc3RhdGUpXG4gKiBTaG93cyBUZWxlcG9ydEVycm9yIGRpYWxvZyByZW5kZXJlZCBpbnRvIHRoZSBleGlzdGluZyByb290IGlmIG5lZWRlZFxuICovXG5hc3luYyBmdW5jdGlvbiBoYW5kbGVUZWxlcG9ydFByZXJlcXVpc2l0ZXMoXG4gIHJvb3Q6IFJvb3QsXG4gIGVycm9yc1RvSWdub3JlPzogU2V0PFRlbGVwb3J0TG9jYWxFcnJvclR5cGU+LFxuKTogUHJvbWlzZTx2b2lkPiB7XG4gIGNvbnN0IGVycm9ycyA9IGF3YWl0IGdldFRlbGVwb3J0RXJyb3JzKClcbiAgaWYgKGVycm9ycy5zaXplID4gMCkge1xuICAgIC8vIExvZyB0ZWxlcG9ydCBlcnJvcnMgZGV0ZWN0ZWRcbiAgICBsb2dFdmVudCgndGVuZ3VfdGVsZXBvcnRfZXJyb3JzX2RldGVjdGVkJywge1xuICAgICAgZXJyb3JfdHlwZXM6IEFycmF5LmZyb20oZXJyb3JzKS5qb2luKFxuICAgICAgICAnLCcsXG4gICAgICApIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICBlcnJvcnNfaWdub3JlZDogQXJyYXkuZnJvbShlcnJvcnNUb0lnbm9yZSB8fCBbXSkuam9pbihcbiAgICAgICAgJywnLFxuICAgICAgKSBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgIH0pXG5cbiAgICAvLyBTaG93IFRlbGVwb3J0RXJyb3IgZGlhbG9nIGZvciB1c2VyIGludGVyYWN0aW9uXG4gICAgYXdhaXQgbmV3IFByb21pc2U8dm9pZD4ocmVzb2x2ZSA9PiB7XG4gICAgICByb290LnJlbmRlcihcbiAgICAgICAgPEFwcFN0YXRlUHJvdmlkZXI+XG4gICAgICAgICAgPEtleWJpbmRpbmdTZXR1cD5cbiAgICAgICAgICAgIDxUZWxlcG9ydEVycm9yXG4gICAgICAgICAgICAgIGVycm9yc1RvSWdub3JlPXtlcnJvcnNUb0lnbm9yZX1cbiAgICAgICAgICAgICAgb25Db21wbGV0ZT17KCkgPT4ge1xuICAgICAgICAgICAgICAgIC8vIExvZyB3aGVuIGVycm9ycyBhcmUgcmVzb2x2ZWRcbiAgICAgICAgICAgICAgICBsb2dFdmVudCgndGVuZ3VfdGVsZXBvcnRfZXJyb3JzX3Jlc29sdmVkJywge1xuICAgICAgICAgICAgICAgICAgZXJyb3JfdHlwZXM6IEFycmF5LmZyb20oZXJyb3JzKS5qb2luKFxuICAgICAgICAgICAgICAgICAgICAnLCcsXG4gICAgICAgICAgICAgICAgICApIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgICAgICAgICAgfSlcbiAgICAgICAgICAgICAgICB2b2lkIHJlc29sdmUoKVxuICAgICAgICAgICAgICB9fVxuICAgICAgICAgICAgLz5cbiAgICAgICAgICA8L0tleWJpbmRpbmdTZXR1cD5cbiAgICAgICAgPC9BcHBTdGF0ZVByb3ZpZGVyPixcbiAgICAgIClcbiAgICB9KVxuICB9XG59XG5cbi8qKlxuICogQ3JlYXRlcyBhIHJlbW90ZSBDbGF1ZGUuYWkgc2Vzc2lvbiB3aXRoIGVycm9yIGhhbmRsaW5nIGFuZCBVSSBmZWVkYmFjay5cbiAqIFNob3dzIHByZXJlcXVpc2l0ZSBlcnJvciBkaWFsb2cgaW4gdGhlIGV4aXN0aW5nIHJvb3QgaWYgbmVlZGVkLlxuICogQHBhcmFtIHJvb3QgVGhlIGV4aXN0aW5nIEluayByb290IHRvIHJlbmRlciBkaWFsb2dzIGludG9cbiAqIEBwYXJhbSBkZXNjcmlwdGlvbiBUaGUgZGVzY3JpcHRpb24vcHJvbXB0IGZvciB0aGUgbmV3IHNlc3Npb24gKG51bGwgZm9yIG5vIGluaXRpYWwgcHJvbXB0KVxuICogQHBhcmFtIHNpZ25hbCBBYm9ydFNpZ25hbCBmb3IgY2FuY2VsbGF0aW9uXG4gKiBAcGFyYW0gYnJhbmNoTmFtZSBPcHRpb25hbCBicmFuY2ggbmFtZSBmb3IgdGhlIHJlbW90ZSBzZXNzaW9uIHRvIHVzZVxuICogQHJldHVybnMgUHJvbWlzZTxUZWxlcG9ydFRvUmVtb3RlUmVzcG9uc2UgfCBudWxsPiBUaGUgY3JlYXRlZCBzZXNzaW9uIG9yIG51bGwgaWYgY3JlYXRpb24gZmFpbHNcbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHRlbGVwb3J0VG9SZW1vdGVXaXRoRXJyb3JIYW5kbGluZyhcbiAgcm9vdDogUm9vdCxcbiAgZGVzY3JpcHRpb246IHN0cmluZyB8IG51bGwsXG4gIHNpZ25hbDogQWJvcnRTaWduYWwsXG4gIGJyYW5jaE5hbWU/OiBzdHJpbmcsXG4pOiBQcm9taXNlPFRlbGVwb3J0VG9SZW1vdGVSZXNwb25zZSB8IG51bGw+IHtcbiAgY29uc3QgZXJyb3JzVG9JZ25vcmUgPSBuZXcgU2V0PFRlbGVwb3J0TG9jYWxFcnJvclR5cGU+KFsnbmVlZHNHaXRTdGFzaCddKVxuICBhd2FpdCBoYW5kbGVUZWxlcG9ydFByZXJlcXVpc2l0ZXMocm9vdCwgZXJyb3JzVG9JZ25vcmUpXG4gIHJldHVybiB0ZWxlcG9ydFRvUmVtb3RlKHtcbiAgICBpbml0aWFsTWVzc2FnZTogZGVzY3JpcHRpb24sXG4gICAgc2lnbmFsLFxuICAgIGJyYW5jaE5hbWUsXG4gICAgb25CdW5kbGVGYWlsOiBtc2cgPT4gcHJvY2Vzcy5zdGRlcnIud3JpdGUoYFxcbiR7bXNnfVxcbmApLFxuICB9KVxufVxuXG4vKipcbiAqIEZldGNoZXMgc2Vzc2lvbiBkYXRhIGZyb20gdGhlIHNlc3Npb24gaW5ncmVzcyBBUEkgKC92MS9zZXNzaW9uX2luZ3Jlc3MvKVxuICogVXNlcyBzZXNzaW9uIGxvZ3MgaW5zdGVhZCBvZiBTREsgZXZlbnRzIHRvIGdldCB0aGUgY29ycmVjdCBtZXNzYWdlIHN0cnVjdHVyZVxuICogQHBhcmFtIHNlc3Npb25JZCBUaGUgc2Vzc2lvbiBJRCB0byBmZXRjaFxuICogQHBhcmFtIG9yZ1VVSUQgVGhlIG9yZ2FuaXphdGlvbiBVVUlEXG4gKiBAcGFyYW0gYWNjZXNzVG9rZW4gVGhlIE9BdXRoIGFjY2VzcyB0b2tlblxuICogQHBhcmFtIG9uUHJvZ3Jlc3MgT3B0aW9uYWwgY2FsbGJhY2sgZm9yIHByb2dyZXNzIHVwZGF0ZXNcbiAqIEBwYXJhbSBzZXNzaW9uRGF0YSBPcHRpb25hbCBzZXNzaW9uIGRhdGEgKHVzZWQgdG8gZXh0cmFjdCBicmFuY2ggaW5mbylcbiAqIEByZXR1cm5zIFRlbGVwb3J0UmVtb3RlUmVzcG9uc2Ugd2l0aCBzZXNzaW9uIGxvZ3MgYXMgTWVzc2FnZVtdXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiB0ZWxlcG9ydEZyb21TZXNzaW9uc0FQSShcbiAgc2Vzc2lvbklkOiBzdHJpbmcsXG4gIG9yZ1VVSUQ6IHN0cmluZyxcbiAgYWNjZXNzVG9rZW46IHN0cmluZyxcbiAgb25Qcm9ncmVzcz86IFRlbGVwb3J0UHJvZ3Jlc3NDYWxsYmFjayxcbiAgc2Vzc2lvbkRhdGE/OiBTZXNzaW9uUmVzb3VyY2UsXG4pOiBQcm9taXNlPFRlbGVwb3J0UmVtb3RlUmVzcG9uc2U+IHtcbiAgY29uc3Qgc3RhcnRUaW1lID0gRGF0ZS5ub3coKVxuXG4gIHRyeSB7XG4gICAgLy8gRmV0Y2ggc2Vzc2lvbiBsb2dzIHZpYSBzZXNzaW9uIGluZ3Jlc3NcbiAgICBsb2dGb3JEZWJ1Z2dpbmcoYFt0ZWxlcG9ydF0gU3RhcnRpbmcgZmV0Y2ggZm9yIHNlc3Npb246ICR7c2Vzc2lvbklkfWApXG4gICAgb25Qcm9ncmVzcz8uKCdmZXRjaGluZ19sb2dzJylcblxuICAgIGNvbnN0IGxvZ3NTdGFydFRpbWUgPSBEYXRlLm5vdygpXG4gICAgLy8gVHJ5IENDUiB2MiBmaXJzdCAoR2V0VGVsZXBvcnRFdmVudHMg4oCUIHNlcnZlciBkaXNwYXRjaGVzIFNwYW5uZXIvXG4gICAgLy8gdGhyZWFkc3RvcmUpLiBGYWxsIGJhY2sgdG8gc2Vzc2lvbi1pbmdyZXNzIGlmIGl0IHJldHVybnMgbnVsbFxuICAgIC8vIChlbmRwb2ludCBub3QgeWV0IGRlcGxveWVkLCBvciB0cmFuc2llbnQgZXJyb3IpLiBPbmNlIHNlc3Npb24taW5ncmVzc1xuICAgIC8vIGlzIGdvbmUsIHRoZSBmYWxsYmFjayBiZWNvbWVzIGEgbm8tb3Ag4oCUIGdldFNlc3Npb25Mb2dzVmlhT0F1dGggd2lsbFxuICAgIC8vIHJldHVybiBudWxsIHRvbyBhbmQgd2UgZmFpbCB3aXRoIFwiRmFpbGVkIHRvIGZldGNoIHNlc3Npb24gbG9nc1wiLlxuICAgIGxldCBsb2dzID0gYXdhaXQgZ2V0VGVsZXBvcnRFdmVudHMoc2Vzc2lvbklkLCBhY2Nlc3NUb2tlbiwgb3JnVVVJRClcbiAgICBpZiAobG9ncyA9PT0gbnVsbCkge1xuICAgICAgbG9nRm9yRGVidWdnaW5nKFxuICAgICAgICAnW3RlbGVwb3J0XSB2MiBlbmRwb2ludCByZXR1cm5lZCBudWxsLCB0cnlpbmcgc2Vzc2lvbi1pbmdyZXNzJyxcbiAgICAgIClcbiAgICAgIGxvZ3MgPSBhd2FpdCBnZXRTZXNzaW9uTG9nc1ZpYU9BdXRoKHNlc3Npb25JZCwgYWNjZXNzVG9rZW4sIG9yZ1VVSUQpXG4gICAgfVxuICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgIGBbdGVsZXBvcnRdIFNlc3Npb24gbG9ncyBmZXRjaGVkIGluICR7RGF0ZS5ub3coKSAtIGxvZ3NTdGFydFRpbWV9bXNgLFxuICAgIClcblxuICAgIGlmIChsb2dzID09PSBudWxsKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ0ZhaWxlZCB0byBmZXRjaCBzZXNzaW9uIGxvZ3MnKVxuICAgIH1cblxuICAgIC8vIEZpbHRlciB0byBnZXQgb25seSB0cmFuc2NyaXB0IG1lc3NhZ2VzLCBleGNsdWRpbmcgc2lkZWNoYWluIG1lc3NhZ2VzXG4gICAgY29uc3QgZmlsdGVyU3RhcnRUaW1lID0gRGF0ZS5ub3coKVxuICAgIGNvbnN0IG1lc3NhZ2VzID0gbG9ncy5maWx0ZXIoXG4gICAgICBlbnRyeSA9PiBpc1RyYW5zY3JpcHRNZXNzYWdlKGVudHJ5KSAmJiAhZW50cnkuaXNTaWRlY2hhaW4sXG4gICAgKSBhcyBNZXNzYWdlW11cbiAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICBgW3RlbGVwb3J0XSBGaWx0ZXJlZCAke2xvZ3MubGVuZ3RofSBlbnRyaWVzIHRvICR7bWVzc2FnZXMubGVuZ3RofSBtZXNzYWdlcyBpbiAke0RhdGUubm93KCkgLSBmaWx0ZXJTdGFydFRpbWV9bXNgLFxuICAgIClcblxuICAgIC8vIEV4dHJhY3QgYnJhbmNoIGluZm8gZnJvbSBzZXNzaW9uIGRhdGFcbiAgICBvblByb2dyZXNzPy4oJ2ZldGNoaW5nX2JyYW5jaCcpXG4gICAgY29uc3QgYnJhbmNoID0gc2Vzc2lvbkRhdGEgPyBnZXRCcmFuY2hGcm9tU2Vzc2lvbihzZXNzaW9uRGF0YSkgOiB1bmRlZmluZWRcbiAgICBpZiAoYnJhbmNoKSB7XG4gICAgICBsb2dGb3JEZWJ1Z2dpbmcoYFt0ZWxlcG9ydF0gRm91bmQgYnJhbmNoOiAke2JyYW5jaH1gKVxuICAgIH1cblxuICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgIGBbdGVsZXBvcnRdIFRvdGFsIHRlbGVwb3J0RnJvbVNlc3Npb25zQVBJIHRpbWU6ICR7RGF0ZS5ub3coKSAtIHN0YXJ0VGltZX1tc2AsXG4gICAgKVxuXG4gICAgcmV0dXJuIHtcbiAgICAgIGxvZzogbWVzc2FnZXMsXG4gICAgICBicmFuY2gsXG4gICAgfVxuICB9IGNhdGNoIChlcnJvcikge1xuICAgIGNvbnN0IGVyciA9IHRvRXJyb3IoZXJyb3IpXG5cbiAgICAvLyBIYW5kbGUgNDA0IHNwZWNpZmljYWxseVxuICAgIGlmIChheGlvcy5pc0F4aW9zRXJyb3IoZXJyb3IpICYmIGVycm9yLnJlc3BvbnNlPy5zdGF0dXMgPT09IDQwNCkge1xuICAgICAgbG9nRXZlbnQoJ3Rlbmd1X3RlbGVwb3J0X2Vycm9yX3Nlc3Npb25fbm90X2ZvdW5kXzQwNCcsIHtcbiAgICAgICAgc2Vzc2lvbklkOlxuICAgICAgICAgIHNlc3Npb25JZCBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgfSlcbiAgICAgIHRocm93IG5ldyBUZWxlcG9ydE9wZXJhdGlvbkVycm9yKFxuICAgICAgICBgJHtzZXNzaW9uSWR9IG5vdCBmb3VuZC5gLFxuICAgICAgICBgJHtzZXNzaW9uSWR9IG5vdCBmb3VuZC5cXG4ke2NoYWxrLmRpbSgnUnVuIC9zdGF0dXMgaW4gQ2xhdWRlIENvZGUgdG8gY2hlY2sgeW91ciBhY2NvdW50LicpfWAsXG4gICAgICApXG4gICAgfVxuXG4gICAgbG9nRXJyb3IoZXJyKVxuXG4gICAgdGhyb3cgbmV3IEVycm9yKGBGYWlsZWQgdG8gZmV0Y2ggc2Vzc2lvbiBmcm9tIFNlc3Npb25zIEFQSTogJHtlcnIubWVzc2FnZX1gKVxuICB9XG59XG5cbi8qKlxuICogUmVzcG9uc2UgdHlwZSBmb3IgcG9sbGluZyByZW1vdGUgc2Vzc2lvbiBldmVudHMgKHVzZXMgU0RLIGV2ZW50cyBmb3JtYXQpXG4gKi9cbmV4cG9ydCB0eXBlIFBvbGxSZW1vdGVTZXNzaW9uUmVzcG9uc2UgPSB7XG4gIG5ld0V2ZW50czogU0RLTWVzc2FnZVtdXG4gIGxhc3RFdmVudElkOiBzdHJpbmcgfCBudWxsXG4gIGJyYW5jaD86IHN0cmluZ1xuICBzZXNzaW9uU3RhdHVzPzogJ2lkbGUnIHwgJ3J1bm5pbmcnIHwgJ3JlcXVpcmVzX2FjdGlvbicgfCAnYXJjaGl2ZWQnXG59XG5cbi8qKlxuICogUG9sbHMgcmVtb3RlIHNlc3Npb24gZXZlbnRzLiBQYXNzIHRoZSBwcmV2aW91cyByZXNwb25zZSdzIGBsYXN0RXZlbnRJZGBcbiAqIGFzIGBhZnRlcklkYCB0byBmZXRjaCBvbmx5IHRoZSBkZWx0YS4gU2V0IGBza2lwTWV0YWRhdGFgIHRvIGF2b2lkIHRoZVxuICogcGVyLWNhbGwgR0VUIC92MS9zZXNzaW9ucy97aWR9IHdoZW4gYnJhbmNoL3N0YXR1cyBhcmVuJ3QgbmVlZGVkLlxuICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gcG9sbFJlbW90ZVNlc3Npb25FdmVudHMoXG4gIHNlc3Npb25JZDogc3RyaW5nLFxuICBhZnRlcklkOiBzdHJpbmcgfCBudWxsID0gbnVsbCxcbiAgb3B0cz86IHsgc2tpcE1ldGFkYXRhPzogYm9vbGVhbiB9LFxuKTogUHJvbWlzZTxQb2xsUmVtb3RlU2Vzc2lvblJlc3BvbnNlPiB7XG4gIGNvbnN0IGFjY2Vzc1Rva2VuID0gZ2V0Q2xhdWRlQUlPQXV0aFRva2VucygpPy5hY2Nlc3NUb2tlblxuICBpZiAoIWFjY2Vzc1Rva2VuKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKCdObyBhY2Nlc3MgdG9rZW4gZm9yIHBvbGxpbmcnKVxuICB9XG5cbiAgY29uc3Qgb3JnVVVJRCA9IGF3YWl0IGdldE9yZ2FuaXphdGlvblVVSUQoKVxuICBpZiAoIW9yZ1VVSUQpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoJ05vIG9yZyBVVUlEIGZvciBwb2xsaW5nJylcbiAgfVxuXG4gIGNvbnN0IGhlYWRlcnMgPSB7XG4gICAgLi4uZ2V0T0F1dGhIZWFkZXJzKGFjY2Vzc1Rva2VuKSxcbiAgICAnYW50aHJvcGljLWJldGEnOiAnY2NyLWJ5b2MtMjAyNS0wNy0yOScsXG4gICAgJ3gtb3JnYW5pemF0aW9uLXV1aWQnOiBvcmdVVUlELFxuICB9XG4gIGNvbnN0IGV2ZW50c1VybCA9IGAke2dldE9hdXRoQ29uZmlnKCkuQkFTRV9BUElfVVJMfS92MS9zZXNzaW9ucy8ke3Nlc3Npb25JZH0vZXZlbnRzYFxuXG4gIHR5cGUgRXZlbnRzUmVzcG9uc2UgPSB7XG4gICAgZGF0YTogdW5rbm93bltdXG4gICAgaGFzX21vcmU6IGJvb2xlYW5cbiAgICBmaXJzdF9pZDogc3RyaW5nIHwgbnVsbFxuICAgIGxhc3RfaWQ6IHN0cmluZyB8IG51bGxcbiAgfVxuXG4gIC8vIENhcCBpcyBhIHNhZmV0eSB2YWx2ZSBhZ2FpbnN0IHN0dWNrIGN1cnNvcnM7IHN0ZWFkeS1zdGF0ZSBpcyAw4oCTMSBwYWdlcy5cbiAgY29uc3QgTUFYX0VWRU5UX1BBR0VTID0gNTBcbiAgY29uc3Qgc2RrTWVzc2FnZXM6IFNES01lc3NhZ2VbXSA9IFtdXG4gIGxldCBjdXJzb3IgPSBhZnRlcklkXG4gIGZvciAobGV0IHBhZ2UgPSAwOyBwYWdlIDwgTUFYX0VWRU5UX1BBR0VTOyBwYWdlKyspIHtcbiAgICBjb25zdCBldmVudHNSZXNwb25zZSA9IGF3YWl0IGF4aW9zLmdldChldmVudHNVcmwsIHtcbiAgICAgIGhlYWRlcnMsXG4gICAgICBwYXJhbXM6IGN1cnNvciA/IHsgYWZ0ZXJfaWQ6IGN1cnNvciB9IDogdW5kZWZpbmVkLFxuICAgICAgdGltZW91dDogMzAwMDAsXG4gICAgfSlcblxuICAgIGlmIChldmVudHNSZXNwb25zZS5zdGF0dXMgIT09IDIwMCkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICBgRmFpbGVkIHRvIGZldGNoIHNlc3Npb24gZXZlbnRzOiAke2V2ZW50c1Jlc3BvbnNlLnN0YXR1c1RleHR9YCxcbiAgICAgIClcbiAgICB9XG5cbiAgICBjb25zdCBldmVudHNEYXRhOiBFdmVudHNSZXNwb25zZSA9IGV2ZW50c1Jlc3BvbnNlLmRhdGFcbiAgICBpZiAoIWV2ZW50c0RhdGE/LmRhdGEgfHwgIUFycmF5LmlzQXJyYXkoZXZlbnRzRGF0YS5kYXRhKSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKCdJbnZhbGlkIGV2ZW50cyByZXNwb25zZScpXG4gICAgfVxuXG4gICAgZm9yIChjb25zdCBldmVudCBvZiBldmVudHNEYXRhLmRhdGEpIHtcbiAgICAgIGlmIChldmVudCAmJiB0eXBlb2YgZXZlbnQgPT09ICdvYmplY3QnICYmICd0eXBlJyBpbiBldmVudCkge1xuICAgICAgICBpZiAoXG4gICAgICAgICAgZXZlbnQudHlwZSA9PT0gJ2Vudl9tYW5hZ2VyX2xvZycgfHxcbiAgICAgICAgICBldmVudC50eXBlID09PSAnY29udHJvbF9yZXNwb25zZSdcbiAgICAgICAgKSB7XG4gICAgICAgICAgY29udGludWVcbiAgICAgICAgfVxuICAgICAgICBpZiAoJ3Nlc3Npb25faWQnIGluIGV2ZW50KSB7XG4gICAgICAgICAgc2RrTWVzc2FnZXMucHVzaChldmVudCBhcyBTREtNZXNzYWdlKVxuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuXG4gICAgaWYgKCFldmVudHNEYXRhLmxhc3RfaWQpIGJyZWFrXG4gICAgY3Vyc29yID0gZXZlbnRzRGF0YS5sYXN0X2lkXG4gICAgaWYgKCFldmVudHNEYXRhLmhhc19tb3JlKSBicmVha1xuICB9XG5cbiAgaWYgKG9wdHM/LnNraXBNZXRhZGF0YSkge1xuICAgIHJldHVybiB7IG5ld0V2ZW50czogc2RrTWVzc2FnZXMsIGxhc3RFdmVudElkOiBjdXJzb3IgfVxuICB9XG5cbiAgLy8gRmV0Y2ggc2Vzc2lvbiBtZXRhZGF0YSAoYnJhbmNoLCBzdGF0dXMpXG4gIGxldCBicmFuY2g6IHN0cmluZyB8IHVuZGVmaW5lZFxuICBsZXQgc2Vzc2lvblN0YXR1czogUG9sbFJlbW90ZVNlc3Npb25SZXNwb25zZVsnc2Vzc2lvblN0YXR1cyddXG4gIHRyeSB7XG4gICAgY29uc3Qgc2Vzc2lvbkRhdGEgPSBhd2FpdCBmZXRjaFNlc3Npb24oc2Vzc2lvbklkKVxuICAgIGJyYW5jaCA9IGdldEJyYW5jaEZyb21TZXNzaW9uKHNlc3Npb25EYXRhKVxuICAgIHNlc3Npb25TdGF0dXMgPVxuICAgICAgc2Vzc2lvbkRhdGEuc2Vzc2lvbl9zdGF0dXMgYXMgUG9sbFJlbW90ZVNlc3Npb25SZXNwb25zZVsnc2Vzc2lvblN0YXR1cyddXG4gIH0gY2F0Y2ggKGUpIHtcbiAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICBgdGVsZXBvcnQ6IGZhaWxlZCB0byBmZXRjaCBzZXNzaW9uICR7c2Vzc2lvbklkfSBtZXRhZGF0YTogJHtlfWAsXG4gICAgICB7IGxldmVsOiAnZGVidWcnIH0sXG4gICAgKVxuICB9XG5cbiAgcmV0dXJuIHsgbmV3RXZlbnRzOiBzZGtNZXNzYWdlcywgbGFzdEV2ZW50SWQ6IGN1cnNvciwgYnJhbmNoLCBzZXNzaW9uU3RhdHVzIH1cbn1cblxuLyoqXG4gKiBDcmVhdGVzIGEgcmVtb3RlIENsYXVkZS5haSBzZXNzaW9uIHVzaW5nIHRoZSBTZXNzaW9ucyBBUEkuXG4gKlxuICogVHdvIHNvdXJjZSBtb2RlczpcbiAqIC0gR2l0SHViIChkZWZhdWx0KTogYmFja2VuZCBjbG9uZXMgZnJvbSB0aGUgcmVwbydzIG9yaWdpbiBVUkwuIFJlcXVpcmVzIGFcbiAqICAgR2l0SHViIHJlbW90ZSArIENDUi1zaWRlIEdpdEh1YiBjb25uZWN0aW9uLiA0MyUgb2YgQ0xJIHNlc3Npb25zIGhhdmUgYW5cbiAqICAgb3JpZ2luIHJlbW90ZTsgZmFyIGZld2VyIHBhc3MgdGhlIGZ1bGwgcHJlY29uZGl0aW9uIGNoYWluLlxuICogLSBCdW5kbGUgKENDUl9GT1JDRV9CVU5ETEU9MSk6IENMSSBjcmVhdGVzIGBnaXQgYnVuZGxlIC0tYWxsYCwgdXBsb2FkcyB2aWEgRmlsZXNcbiAqICAgQVBJLCBwYXNzZXMgZmlsZV9pZCBhcyBzZWVkX2J1bmRsZV9maWxlX2lkIG9uIHRoZSBzZXNzaW9uIGNvbnRleHQuIENDUlxuICogICBkb3dubG9hZHMgaXQgYW5kIGNsb25lcyBmcm9tIHRoZSBidW5kbGUuIE5vIEdpdEh1YiBkZXBlbmRlbmN5IOKAlCB3b3JrcyBmb3JcbiAqICAgbG9jYWwtb25seSByZXBvcy4gUmVhY2g6IDU0JSBvZiBDTEkgc2Vzc2lvbnMgKGFueXRoaW5nIHdpdGggLmdpdC8pLlxuICogICBCYWNrZW5kOiBhbnRocm9waWMjMzAzODU2LlxuICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gdGVsZXBvcnRUb1JlbW90ZShvcHRpb25zOiB7XG4gIGluaXRpYWxNZXNzYWdlOiBzdHJpbmcgfCBudWxsXG4gIGJyYW5jaE5hbWU/OiBzdHJpbmdcbiAgdGl0bGU/OiBzdHJpbmdcbiAgLyoqXG4gICAqIFRoZSBkZXNjcmlwdGlvbiBvZiB0aGUgc2Vzc2lvbi4gVGhpcyBpcyB1c2VkIHRvIGdlbmVyYXRlIHRoZSB0aXRsZSBhbmRcbiAgICogc2Vzc2lvbiBicmFuY2ggbmFtZSAodW5sZXNzIHRoZXkgYXJlIGV4cGxpY2l0bHkgcHJvdmlkZWQpLlxuICAgKi9cbiAgZGVzY3JpcHRpb24/OiBzdHJpbmdcbiAgbW9kZWw/OiBzdHJpbmdcbiAgcGVybWlzc2lvbk1vZGU/OiBQZXJtaXNzaW9uTW9kZVxuICB1bHRyYXBsYW4/OiBib29sZWFuXG4gIHNpZ25hbDogQWJvcnRTaWduYWxcbiAgdXNlRGVmYXVsdEVudmlyb25tZW50PzogYm9vbGVhblxuICAvKipcbiAgICogRXhwbGljaXQgZW52aXJvbm1lbnRfaWQgKGUuZy4gdGhlIGNvZGVfcmV2aWV3IHN5bnRoZXRpYyBlbnYpLiBCeXBhc3Nlc1xuICAgKiBmZXRjaEVudmlyb25tZW50czsgdGhlIHVzdWFsIHJlcG8tZGV0ZWN0aW9uIOKGkiBnaXQgc291cmNlIHN0aWxsIHJ1bnMgc29cbiAgICogdGhlIGNvbnRhaW5lciBnZXRzIHRoZSByZXBvIGNoZWNrZWQgb3V0IChvcmNoZXN0cmF0b3IgcmVhZHMgLS1yZXBvLWRpclxuICAgKiBmcm9tIHB3ZCwgaXQgZG9lc24ndCBjbG9uZSkuXG4gICAqL1xuICBlbnZpcm9ubWVudElkPzogc3RyaW5nXG4gIC8qKlxuICAgKiBQZXItc2Vzc2lvbiBlbnYgdmFycyBtZXJnZWQgaW50byBzZXNzaW9uX2NvbnRleHQuZW52aXJvbm1lbnRfdmFyaWFibGVzLlxuICAgKiBXcml0ZS1vbmx5IGF0IHRoZSBBUEkgbGF5ZXIgKHN0cmlwcGVkIGZyb20gR2V0L0xpc3QgcmVzcG9uc2VzKS4gV2hlblxuICAgKiBlbnZpcm9ubWVudElkIGlzIHNldCwgQ0xBVURFX0NPREVfT0FVVEhfVE9LRU4gaXMgYXV0by1pbmplY3RlZCBmcm9tIHRoZVxuICAgKiBjYWxsZXIncyBhY2Nlc3NUb2tlbiBzbyB0aGUgY29udGFpbmVyJ3MgaG9vayBjYW4gaGl0IGluZmVyZW5jZSAodGhlXG4gICAqIHNlcnZlciBvbmx5IHBhc3NlcyB0aHJvdWdoIHdoYXQgdGhlIGNhbGxlciBzZW5kczsgYnVnaHVudGVyLmdvIG1pbnRzXG4gICAqIGl0cyBvd24sIHVzZXIgc2Vzc2lvbnMgZG9uJ3QgZ2V0IG9uZSBhdXRvbWF0aWNhbGx5KS5cbiAgICovXG4gIGVudmlyb25tZW50VmFyaWFibGVzPzogUmVjb3JkPHN0cmluZywgc3RyaW5nPlxuICAvKipcbiAgICogV2hlbiBzZXQgd2l0aCBlbnZpcm9ubWVudElkLCBjcmVhdGVzIGFuZCB1cGxvYWRzIGEgZ2l0IGJ1bmRsZSBvZiB0aGVcbiAgICogbG9jYWwgd29ya2luZyB0cmVlIChjcmVhdGVBbmRVcGxvYWRHaXRCdW5kbGUgaGFuZGxlcyB0aGUgc3Rhc2gtY3JlYXRlXG4gICAqIGZvciB1bmNvbW1pdHRlZCBjaGFuZ2VzKSBhbmQgcGFzc2VzIGl0IGFzIHNlZWRfYnVuZGxlX2ZpbGVfaWQuIEJhY2tlbmRcbiAgICogY2xvbmVzIGZyb20gdGhlIGJ1bmRsZSBpbnN0ZWFkIG9mIEdpdEh1YiDigJQgY29udGFpbmVyIGdldHMgdGhlIGNhbGxlcidzXG4gICAqIGV4YWN0IGxvY2FsIHN0YXRlLiBOZWVkcyAuZ2l0LyBvbmx5LCBub3QgYSBHaXRIdWIgcmVtb3RlLlxuICAgKi9cbiAgdXNlQnVuZGxlPzogYm9vbGVhblxuICAvKipcbiAgICogQ2FsbGVkIHdpdGggYSB1c2VyLWZhY2luZyBtZXNzYWdlIHdoZW4gdGhlIGJ1bmRsZSBwYXRoIGlzIGF0dGVtcHRlZCBidXRcbiAgICogZmFpbHMuIFRoZSB3cmFwcGVyIHN0ZGVyci53cml0ZXMgaXQgKHByZS1SRVBMKS4gUmVtb3RlLWFnZW50IGNhbGxlcnNcbiAgICogY2FwdHVyZSBpdCB0byBpbmNsdWRlIGluIHRoZWlyIHRocm93IChpbi1SRVBMLCBJbmstcmVuZGVyZWQpLlxuICAgKi9cbiAgb25CdW5kbGVGYWlsPzogKG1lc3NhZ2U6IHN0cmluZykgPT4gdm9pZFxuICAvKipcbiAgICogV2hlbiB0cnVlLCBkaXNhYmxlcyB0aGUgZ2l0LWJ1bmRsZSBmYWxsYmFjayBlbnRpcmVseS4gVXNlIGZvciBmbG93cyBsaWtlXG4gICAqIGF1dG9maXggd2hlcmUgQ0NSIG11c3QgcHVzaCB0byBHaXRIdWIg4oCUIGEgYnVuZGxlIGNhbid0IGRvIHRoYXQuXG4gICAqL1xuICBza2lwQnVuZGxlPzogYm9vbGVhblxuICAvKipcbiAgICogV2hlbiBzZXQsIHJldXNlcyB0aGlzIGJyYW5jaCBhcyB0aGUgb3V0Y29tZSBicmFuY2ggaW5zdGVhZCBvZiBnZW5lcmF0aW5nXG4gICAqIGEgbmV3IGNsYXVkZS8gYnJhbmNoLiBTZXRzIGFsbG93X3VucmVzdHJpY3RlZF9naXRfcHVzaCBvbiB0aGUgc291cmNlIGFuZFxuICAgKiByZXVzZV9vdXRjb21lX2JyYW5jaGVzIG9uIHRoZSBzZXNzaW9uIGNvbnRleHQgc28gdGhlIHJlbW90ZSBwdXNoZXMgdG8gdGhlXG4gICAqIGNhbGxlcidzIGJyYW5jaCBkaXJlY3RseS5cbiAgICovXG4gIHJldXNlT3V0Y29tZUJyYW5jaD86IHN0cmluZ1xuICAvKipcbiAgICogR2l0SHViIFBSIHRvIGF0dGFjaCB0byB0aGUgc2Vzc2lvbiBjb250ZXh0LiBCYWNrZW5kIHVzZXMgdGhpcyB0b1xuICAgKiBpZGVudGlmeSB0aGUgUFIgYXNzb2NpYXRlZCB3aXRoIHRoaXMgc2Vzc2lvbi5cbiAgICovXG4gIGdpdGh1YlByPzogeyBvd25lcjogc3RyaW5nOyByZXBvOiBzdHJpbmc7IG51bWJlcjogbnVtYmVyIH1cbn0pOiBQcm9taXNlPFRlbGVwb3J0VG9SZW1vdGVSZXNwb25zZSB8IG51bGw+IHtcbiAgY29uc3QgeyBpbml0aWFsTWVzc2FnZSwgc2lnbmFsIH0gPSBvcHRpb25zXG4gIHRyeSB7XG4gICAgLy8gQ2hlY2sgYXV0aGVudGljYXRpb25cbiAgICBhd2FpdCBjaGVja0FuZFJlZnJlc2hPQXV0aFRva2VuSWZOZWVkZWQoKVxuICAgIGNvbnN0IGFjY2Vzc1Rva2VuID0gZ2V0Q2xhdWRlQUlPQXV0aFRva2VucygpPy5hY2Nlc3NUb2tlblxuICAgIGlmICghYWNjZXNzVG9rZW4pIHtcbiAgICAgIGxvZ0Vycm9yKG5ldyBFcnJvcignTm8gYWNjZXNzIHRva2VuIGZvdW5kIGZvciByZW1vdGUgc2Vzc2lvbiBjcmVhdGlvbicpKVxuICAgICAgcmV0dXJuIG51bGxcbiAgICB9XG5cbiAgICAvLyBHZXQgb3JnYW5pemF0aW9uIFVVSURcbiAgICBjb25zdCBvcmdVVUlEID0gYXdhaXQgZ2V0T3JnYW5pemF0aW9uVVVJRCgpXG4gICAgaWYgKCFvcmdVVUlEKSB7XG4gICAgICBsb2dFcnJvcihcbiAgICAgICAgbmV3IEVycm9yKFxuICAgICAgICAgICdVbmFibGUgdG8gZ2V0IG9yZ2FuaXphdGlvbiBVVUlEIGZvciByZW1vdGUgc2Vzc2lvbiBjcmVhdGlvbicsXG4gICAgICAgICksXG4gICAgICApXG4gICAgICByZXR1cm4gbnVsbFxuICAgIH1cblxuICAgIC8vIEV4cGxpY2l0IGVudmlyb25tZW50SWQgc2hvcnQtY2lyY3VpdHMgSGFpa3UgdGl0bGUtZ2VuICsgZW52IHNlbGVjdGlvbi5cbiAgICAvLyBTdGlsbCBydW5zIHJlcG8gZGV0ZWN0aW9uIHNvIHRoZSBjb250YWluZXIgZ2V0cyBhIHdvcmtpbmcgZGlyZWN0b3J5IOKAlFxuICAgIC8vIHRoZSBjb2RlX3JldmlldyBvcmNoZXN0cmF0b3IgcmVhZHMgLS1yZXBvLWRpciAkKHB3ZCksIGl0IGRvZXNuJ3QgY2xvbmVcbiAgICAvLyAoYnVnaHVudGVyLmdvOjUyMCBzZXRzIGEgZ2l0IHNvdXJjZSB0b287IGVudi1tYW5hZ2VyIGRvZXMgdGhlIGNoZWNrb3V0XG4gICAgLy8gYmVmb3JlIHRoZSBTZXNzaW9uU3RhcnQgaG9vayBmaXJlcykuXG4gICAgaWYgKG9wdGlvbnMuZW52aXJvbm1lbnRJZCkge1xuICAgICAgY29uc3QgdXJsID0gYCR7Z2V0T2F1dGhDb25maWcoKS5CQVNFX0FQSV9VUkx9L3YxL3Nlc3Npb25zYFxuICAgICAgY29uc3QgaGVhZGVycyA9IHtcbiAgICAgICAgLi4uZ2V0T0F1dGhIZWFkZXJzKGFjY2Vzc1Rva2VuKSxcbiAgICAgICAgJ2FudGhyb3BpYy1iZXRhJzogJ2Njci1ieW9jLTIwMjUtMDctMjknLFxuICAgICAgICAneC1vcmdhbml6YXRpb24tdXVpZCc6IG9yZ1VVSUQsXG4gICAgICB9XG4gICAgICBjb25zdCBlbnZWYXJzID0ge1xuICAgICAgICBDTEFVREVfQ09ERV9PQVVUSF9UT0tFTjogYWNjZXNzVG9rZW4sXG4gICAgICAgIC4uLihvcHRpb25zLmVudmlyb25tZW50VmFyaWFibGVzID8/IHt9KSxcbiAgICAgIH1cblxuICAgICAgLy8gQnVuZGxlIG1vZGU6IHVwbG9hZCBsb2NhbCB3b3JraW5nIHRyZWUgKHVuY29tbWl0dGVkIGNoYW5nZXMgdmlhXG4gICAgICAvLyByZWZzL3NlZWQvc3Rhc2gpLCBjb250YWluZXIgY2xvbmVzIGZyb20gdGhlIGJ1bmRsZS4gTm8gR2l0SHViLlxuICAgICAgLy8gT3RoZXJ3aXNlOiBnaXRodWIuY29tIHNvdXJjZSDigJQgY2FsbGVyIGNoZWNrZWQgZWxpZ2liaWxpdHkuXG4gICAgICBsZXQgZ2l0U291cmNlOiBHaXRTb3VyY2UgfCBudWxsID0gbnVsbFxuICAgICAgbGV0IHNlZWRCdW5kbGVGaWxlSWQ6IHN0cmluZyB8IG51bGwgPSBudWxsXG4gICAgICBpZiAob3B0aW9ucy51c2VCdW5kbGUpIHtcbiAgICAgICAgY29uc3QgYnVuZGxlID0gYXdhaXQgY3JlYXRlQW5kVXBsb2FkR2l0QnVuZGxlKFxuICAgICAgICAgIHtcbiAgICAgICAgICAgIG9hdXRoVG9rZW46IGFjY2Vzc1Rva2VuLFxuICAgICAgICAgICAgc2Vzc2lvbklkOiBnZXRTZXNzaW9uSWQoKSxcbiAgICAgICAgICAgIGJhc2VVcmw6IGdldE9hdXRoQ29uZmlnKCkuQkFTRV9BUElfVVJMLFxuICAgICAgICAgIH0sXG4gICAgICAgICAgeyBzaWduYWwgfSxcbiAgICAgICAgKVxuICAgICAgICBpZiAoIWJ1bmRsZS5zdWNjZXNzKSB7XG4gICAgICAgICAgbG9nRXJyb3IobmV3IEVycm9yKGBCdW5kbGUgdXBsb2FkIGZhaWxlZDogJHtidW5kbGUuZXJyb3J9YCkpXG4gICAgICAgICAgcmV0dXJuIG51bGxcbiAgICAgICAgfVxuICAgICAgICBzZWVkQnVuZGxlRmlsZUlkID0gYnVuZGxlLmZpbGVJZFxuICAgICAgICBsb2dFdmVudCgndGVuZ3VfdGVsZXBvcnRfYnVuZGxlX21vZGUnLCB7XG4gICAgICAgICAgc2l6ZV9ieXRlczogYnVuZGxlLmJ1bmRsZVNpemVCeXRlcyxcbiAgICAgICAgICBzY29wZTpcbiAgICAgICAgICAgIGJ1bmRsZS5zY29wZSBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICAgIGhhc193aXA6IGJ1bmRsZS5oYXNXaXAsXG4gICAgICAgICAgcmVhc29uOlxuICAgICAgICAgICAgJ2V4cGxpY2l0X2Vudl9idW5kbGUnIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICAgIH0pXG4gICAgICB9IGVsc2Uge1xuICAgICAgICBjb25zdCByZXBvSW5mbyA9IGF3YWl0IGRldGVjdEN1cnJlbnRSZXBvc2l0b3J5V2l0aEhvc3QoKVxuICAgICAgICBpZiAocmVwb0luZm8pIHtcbiAgICAgICAgICBnaXRTb3VyY2UgPSB7XG4gICAgICAgICAgICB0eXBlOiAnZ2l0X3JlcG9zaXRvcnknLFxuICAgICAgICAgICAgdXJsOiBgaHR0cHM6Ly8ke3JlcG9JbmZvLmhvc3R9LyR7cmVwb0luZm8ub3duZXJ9LyR7cmVwb0luZm8ubmFtZX1gLFxuICAgICAgICAgICAgcmV2aXNpb246IG9wdGlvbnMuYnJhbmNoTmFtZSxcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgY29uc3QgcmVxdWVzdEJvZHkgPSB7XG4gICAgICAgIHRpdGxlOiBvcHRpb25zLnRpdGxlIHx8IG9wdGlvbnMuZGVzY3JpcHRpb24gfHwgJ1JlbW90ZSB0YXNrJyxcbiAgICAgICAgZXZlbnRzOiBbXSxcbiAgICAgICAgc2Vzc2lvbl9jb250ZXh0OiB7XG4gICAgICAgICAgc291cmNlczogZ2l0U291cmNlID8gW2dpdFNvdXJjZV0gOiBbXSxcbiAgICAgICAgICAuLi4oc2VlZEJ1bmRsZUZpbGVJZCAmJiB7IHNlZWRfYnVuZGxlX2ZpbGVfaWQ6IHNlZWRCdW5kbGVGaWxlSWQgfSksXG4gICAgICAgICAgb3V0Y29tZXM6IFtdLFxuICAgICAgICAgIGVudmlyb25tZW50X3ZhcmlhYmxlczogZW52VmFycyxcbiAgICAgICAgfSxcbiAgICAgICAgZW52aXJvbm1lbnRfaWQ6IG9wdGlvbnMuZW52aXJvbm1lbnRJZCxcbiAgICAgIH1cbiAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgYFt0ZWxlcG9ydFRvUmVtb3RlXSBleHBsaWNpdCBlbnYgJHtvcHRpb25zLmVudmlyb25tZW50SWR9LCAke09iamVjdC5rZXlzKGVudlZhcnMpLmxlbmd0aH0gZW52IHZhcnMsICR7c2VlZEJ1bmRsZUZpbGVJZCA/IGBidW5kbGU9JHtzZWVkQnVuZGxlRmlsZUlkfWAgOiBgc291cmNlPSR7Z2l0U291cmNlPy51cmwgPz8gJ25vbmUnfUAke29wdGlvbnMuYnJhbmNoTmFtZSA/PyAnZGVmYXVsdCd9YH1gLFxuICAgICAgKVxuICAgICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCBheGlvcy5wb3N0KHVybCwgcmVxdWVzdEJvZHksIHsgaGVhZGVycywgc2lnbmFsIH0pXG4gICAgICBpZiAocmVzcG9uc2Uuc3RhdHVzICE9PSAyMDAgJiYgcmVzcG9uc2Uuc3RhdHVzICE9PSAyMDEpIHtcbiAgICAgICAgbG9nRXJyb3IoXG4gICAgICAgICAgbmV3IEVycm9yKFxuICAgICAgICAgICAgYENyZWF0ZVNlc3Npb24gJHtyZXNwb25zZS5zdGF0dXN9OiAke2pzb25TdHJpbmdpZnkocmVzcG9uc2UuZGF0YSl9YCxcbiAgICAgICAgICApLFxuICAgICAgICApXG4gICAgICAgIHJldHVybiBudWxsXG4gICAgICB9XG4gICAgICBjb25zdCBzZXNzaW9uRGF0YSA9IHJlc3BvbnNlLmRhdGEgYXMgU2Vzc2lvblJlc291cmNlXG4gICAgICBpZiAoIXNlc3Npb25EYXRhIHx8IHR5cGVvZiBzZXNzaW9uRGF0YS5pZCAhPT0gJ3N0cmluZycpIHtcbiAgICAgICAgbG9nRXJyb3IoXG4gICAgICAgICAgbmV3IEVycm9yKFxuICAgICAgICAgICAgYE5vIHNlc3Npb24gaWQgaW4gcmVzcG9uc2U6ICR7anNvblN0cmluZ2lmeShyZXNwb25zZS5kYXRhKX1gLFxuICAgICAgICAgICksXG4gICAgICAgIClcbiAgICAgICAgcmV0dXJuIG51bGxcbiAgICAgIH1cbiAgICAgIHJldHVybiB7XG4gICAgICAgIGlkOiBzZXNzaW9uRGF0YS5pZCxcbiAgICAgICAgdGl0bGU6IHNlc3Npb25EYXRhLnRpdGxlIHx8IHJlcXVlc3RCb2R5LnRpdGxlLFxuICAgICAgfVxuICAgIH1cblxuICAgIGxldCBnaXRTb3VyY2U6IEdpdFNvdXJjZSB8IG51bGwgPSBudWxsXG4gICAgbGV0IGdpdE91dGNvbWU6IEdpdFJlcG9zaXRvcnlPdXRjb21lIHwgbnVsbCA9IG51bGxcbiAgICBsZXQgc2VlZEJ1bmRsZUZpbGVJZDogc3RyaW5nIHwgbnVsbCA9IG51bGxcblxuICAgIC8vIFNvdXJjZSBzZWxlY3Rpb24gbGFkZGVyOiBHaXRIdWIgY2xvbmUgKGlmIENDUiBjYW4gYWN0dWFsbHkgcHVsbCBpdCkg4oaSXG4gICAgLy8gYnVuZGxlIGZhbGxiYWNrIChpZiAuZ2l0IGV4aXN0cykg4oaSIGVtcHR5IHNhbmRib3guXG4gICAgLy9cbiAgICAvLyBUaGUgcHJlZmxpZ2h0IGlzIHRoZSBzYW1lIGNvZGUgcGF0aCB0aGUgY29udGFpbmVyJ3MgZ2l0LXByb3h5IGNsb25lXG4gICAgLy8gd2lsbCBoaXQgKGdldF9naXRodWJfY2xpZW50X3dpdGhfdXNlcl9hdXRoIOKGkiBub19zeW5jX3VzZXJfdG9rZW5fZm91bmQpLlxuICAgIC8vIDUwJSBvZiB1c2VycyB3aG8gcmVhY2ggdGhlIFwiaW5zdGFsbCBHaXRIdWIgQXBwXCIgc3RlcCBuZXZlciBmaW5pc2ggaXQ7XG4gICAgLy8gd2l0aG91dCB0aGUgcHJlZmxpZ2h0LCBldmVyeSBvbmUgb2YgdGhlbSBnZXRzIGEgY29udGFpbmVyIHRoYXQgNDAxc1xuICAgIC8vIG9uIGNsb25lLiBXaXRoIGl0LCB0aGV5IHNpbGVudGx5IGZhbGwgYmFjayB0byBidW5kbGUuXG4gICAgLy9cbiAgICAvLyBDQ1JfRk9SQ0VfQlVORExFPTEgc2tpcHMgdGhlIHByZWZsaWdodCBlbnRpcmVseSDigJQgdXNlZnVsIGZvciB0ZXN0aW5nXG4gICAgLy8gb3Igd2hlbiB5b3Uga25vdyB5b3VyIEdpdEh1YiBhdXRoIGlzIGJ1c3RlZC4gUmVhZCBoZXJlIChub3QgaW4gdGhlXG4gICAgLy8gY2FsbGVyKSBzbyBpdCB3b3JrcyBmb3IgcmVtb3RlLWFnZW50IHRvbywgbm90IGp1c3QgLS1yZW1vdGUuXG5cbiAgICBjb25zdCByZXBvSW5mbyA9IGF3YWl0IGRldGVjdEN1cnJlbnRSZXBvc2l0b3J5V2l0aEhvc3QoKVxuXG4gICAgLy8gR2VuZXJhdGUgdGl0bGUgYW5kIGJyYW5jaCBuYW1lIGZvciB0aGUgc2Vzc2lvbi4gU2tpcCB0aGUgSGFpa3UgY2FsbFxuICAgIC8vIHdoZW4gYm90aCB0aXRsZSBhbmQgb3V0Y29tZSBicmFuY2ggYXJlIGV4cGxpY2l0bHkgcHJvdmlkZWQuXG4gICAgbGV0IHNlc3Npb25UaXRsZTogc3RyaW5nXG4gICAgbGV0IHNlc3Npb25CcmFuY2g6IHN0cmluZ1xuICAgIGlmIChvcHRpb25zLnRpdGxlICYmIG9wdGlvbnMucmV1c2VPdXRjb21lQnJhbmNoKSB7XG4gICAgICBzZXNzaW9uVGl0bGUgPSBvcHRpb25zLnRpdGxlXG4gICAgICBzZXNzaW9uQnJhbmNoID0gb3B0aW9ucy5yZXVzZU91dGNvbWVCcmFuY2hcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3QgZ2VuZXJhdGVkID0gYXdhaXQgZ2VuZXJhdGVUaXRsZUFuZEJyYW5jaChcbiAgICAgICAgb3B0aW9ucy5kZXNjcmlwdGlvbiB8fCBpbml0aWFsTWVzc2FnZSB8fCAnQmFja2dyb3VuZCB0YXNrJyxcbiAgICAgICAgc2lnbmFsLFxuICAgICAgKVxuICAgICAgc2Vzc2lvblRpdGxlID0gb3B0aW9ucy50aXRsZSB8fCBnZW5lcmF0ZWQudGl0bGVcbiAgICAgIHNlc3Npb25CcmFuY2ggPSBvcHRpb25zLnJldXNlT3V0Y29tZUJyYW5jaCB8fCBnZW5lcmF0ZWQuYnJhbmNoTmFtZVxuICAgIH1cblxuICAgIC8vIFByZWZsaWdodDogZG9lcyBDQ1IgaGF2ZSBhIHRva2VuIHRoYXQgY2FuIGNsb25lIHRoaXMgcmVwbz9cbiAgICAvLyBPbmx5IGNoZWNrZWQgZm9yIGdpdGh1Yi5jb20g4oCUIEdIRVMgbmVlZHMgZ2hlX2NvbmZpZ3VyYXRpb25faWQgd2hpY2hcbiAgICAvLyB3ZSBkb24ndCBoYXZlLCBhbmQgR0hFUyB1c2VycyBhcmUgcG93ZXIgdXNlcnMgd2hvIHByb2JhYmx5IGZpbmlzaGVkXG4gICAgLy8gc2V0dXAuIEZvciB0aGVtIChhbmQgZm9yIG5vbi1HaXRIdWIgaG9zdHMgdGhhdCBwYXJzZUdpdFJlbW90ZVxuICAgIC8vIHNvbWVob3cgYWNjZXB0ZWQpLCBmYWxsIHRocm91Z2ggb3B0aW1pc3RpY2FsbHk7IGlmIHRoZSBiYWNrZW5kXG4gICAgLy8gcmVqZWN0cyB0aGUgaG9zdCwgYnVuZGxlIG5leHQgdGltZS5cbiAgICBsZXQgZ2hWaWFibGUgPSBmYWxzZVxuICAgIGxldCBzb3VyY2VSZWFzb246XG4gICAgICB8ICdnaXRodWJfcHJlZmxpZ2h0X29rJ1xuICAgICAgfCAnZ2hlc19vcHRpbWlzdGljJ1xuICAgICAgfCAnZ2l0aHViX3ByZWZsaWdodF9mYWlsZWQnXG4gICAgICB8ICdub19naXRodWJfcmVtb3RlJ1xuICAgICAgfCAnZm9yY2VkX2J1bmRsZSdcbiAgICAgIHwgJ25vX2dpdF9hdF9hbGwnID0gJ25vX2dpdF9hdF9hbGwnXG5cbiAgICAvLyBnaXRSb290IGdhdGVzIGJvdGggYnVuZGxlIGNyZWF0aW9uIGFuZCB0aGUgZ2F0ZSBjaGVjayBpdHNlbGYg4oCUIG5vXG4gICAgLy8gcG9pbnQgYXdhaXRpbmcgR3Jvd3RoQm9vayB3aGVuIHRoZXJlJ3Mgbm90aGluZyB0byBidW5kbGUuXG4gICAgY29uc3QgZ2l0Um9vdCA9IGZpbmRHaXRSb290KGdldEN3ZCgpKVxuICAgIGNvbnN0IGZvcmNlQnVuZGxlID1cbiAgICAgICFvcHRpb25zLnNraXBCdW5kbGUgJiYgaXNFbnZUcnV0aHkocHJvY2Vzcy5lbnYuQ0NSX0ZPUkNFX0JVTkRMRSlcbiAgICBjb25zdCBidW5kbGVTZWVkR2F0ZU9uID1cbiAgICAgICFvcHRpb25zLnNraXBCdW5kbGUgJiZcbiAgICAgIGdpdFJvb3QgIT09IG51bGwgJiZcbiAgICAgIChpc0VudlRydXRoeShwcm9jZXNzLmVudi5DQ1JfRU5BQkxFX0JVTkRMRSkgfHxcbiAgICAgICAgKGF3YWl0IGNoZWNrR2F0ZV9DQUNIRURfT1JfQkxPQ0tJTkcoJ3Rlbmd1X2Njcl9idW5kbGVfc2VlZF9lbmFibGVkJykpKVxuXG4gICAgaWYgKHJlcG9JbmZvICYmICFmb3JjZUJ1bmRsZSkge1xuICAgICAgaWYgKHJlcG9JbmZvLmhvc3QgPT09ICdnaXRodWIuY29tJykge1xuICAgICAgICBnaFZpYWJsZSA9IGF3YWl0IGNoZWNrR2l0aHViQXBwSW5zdGFsbGVkKFxuICAgICAgICAgIHJlcG9JbmZvLm93bmVyLFxuICAgICAgICAgIHJlcG9JbmZvLm5hbWUsXG4gICAgICAgICAgc2lnbmFsLFxuICAgICAgICApXG4gICAgICAgIHNvdXJjZVJlYXNvbiA9IGdoVmlhYmxlXG4gICAgICAgICAgPyAnZ2l0aHViX3ByZWZsaWdodF9vaydcbiAgICAgICAgICA6ICdnaXRodWJfcHJlZmxpZ2h0X2ZhaWxlZCdcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGdoVmlhYmxlID0gdHJ1ZVxuICAgICAgICBzb3VyY2VSZWFzb24gPSAnZ2hlc19vcHRpbWlzdGljJ1xuICAgICAgfVxuICAgIH0gZWxzZSBpZiAoZm9yY2VCdW5kbGUpIHtcbiAgICAgIHNvdXJjZVJlYXNvbiA9ICdmb3JjZWRfYnVuZGxlJ1xuICAgIH0gZWxzZSBpZiAoZ2l0Um9vdCkge1xuICAgICAgc291cmNlUmVhc29uID0gJ25vX2dpdGh1Yl9yZW1vdGUnXG4gICAgfVxuXG4gICAgLy8gUHJlZmxpZ2h0IGZhaWxlZCBidXQgYnVuZGxlIGlzIG9mZiDigJQgZmFsbCB0aHJvdWdoIG9wdGltaXN0aWNhbGx5IGxpa2VcbiAgICAvLyBwcmUtcHJlZmxpZ2h0IGJlaGF2aW9yLiBCYWNrZW5kIHJlcG9ydHMgdGhlIHJlYWwgYXV0aCBlcnJvci5cbiAgICBpZiAoIWdoVmlhYmxlICYmICFidW5kbGVTZWVkR2F0ZU9uICYmIHJlcG9JbmZvKSB7XG4gICAgICBnaFZpYWJsZSA9IHRydWVcbiAgICB9XG5cbiAgICBpZiAoZ2hWaWFibGUgJiYgcmVwb0luZm8pIHtcbiAgICAgIGNvbnN0IHsgaG9zdCwgb3duZXIsIG5hbWUgfSA9IHJlcG9JbmZvXG4gICAgICAvLyBSZXNvbHZlIHRoZSBiYXNlIGJyYW5jaDogcHJlZmVyIGV4cGxpY2l0IGJyYW5jaE5hbWUsIGZhbGwgYmFjayB0byBkZWZhdWx0IGJyYW5jaFxuICAgICAgY29uc3QgcmV2aXNpb24gPVxuICAgICAgICBvcHRpb25zLmJyYW5jaE5hbWUgPz8gKGF3YWl0IGdldERlZmF1bHRCcmFuY2goKSkgPz8gdW5kZWZpbmVkXG4gICAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICAgIGBbdGVsZXBvcnRUb1JlbW90ZV0gR2l0IHNvdXJjZTogJHtob3N0fS8ke293bmVyfS8ke25hbWV9LCByZXZpc2lvbjogJHtyZXZpc2lvbiA/PyAnbm9uZSd9YCxcbiAgICAgIClcbiAgICAgIGdpdFNvdXJjZSA9IHtcbiAgICAgICAgdHlwZTogJ2dpdF9yZXBvc2l0b3J5JyxcbiAgICAgICAgdXJsOiBgaHR0cHM6Ly8ke2hvc3R9LyR7b3duZXJ9LyR7bmFtZX1gLFxuICAgICAgICAvLyBUaGUgcmV2aXNpb24gc3BlY2lmaWVzIHdoaWNoIHJlZiB0byBjaGVja291dCBhcyB0aGUgYmFzZSBicmFuY2hcbiAgICAgICAgcmV2aXNpb24sXG4gICAgICAgIC4uLihvcHRpb25zLnJldXNlT3V0Y29tZUJyYW5jaCAmJiB7XG4gICAgICAgICAgYWxsb3dfdW5yZXN0cmljdGVkX2dpdF9wdXNoOiB0cnVlLFxuICAgICAgICB9KSxcbiAgICAgIH1cbiAgICAgIC8vIHR5cGU6ICdnaXRodWInIGlzIHVzZWQgZm9yIGFsbCBHaXRIdWItY29tcGF0aWJsZSBob3N0cyAoZ2l0aHViLmNvbSBhbmQgR0hFKS5cbiAgICAgIC8vIFRoZSBDTEkgY2FuJ3QgZGlzdGluZ3Vpc2ggR0hFIGZyb20gbm9uLUdpdEh1YiBob3N0cyAoR2l0TGFiLCBCaXRidWNrZXQpXG4gICAgICAvLyBjbGllbnQtc2lkZSDigJQgdGhlIGJhY2tlbmQgdmFsaWRhdGVzIHRoZSBVUkwgYWdhaW5zdCBjb25maWd1cmVkIEdIRSBpbnN0YW5jZXNcbiAgICAgIC8vIGFuZCBpZ25vcmVzIGdpdF9pbmZvIGZvciB1bnJlY29nbml6ZWQgaG9zdHMuXG4gICAgICBnaXRPdXRjb21lID0ge1xuICAgICAgICB0eXBlOiAnZ2l0X3JlcG9zaXRvcnknLFxuICAgICAgICBnaXRfaW5mbzoge1xuICAgICAgICAgIHR5cGU6ICdnaXRodWInLFxuICAgICAgICAgIHJlcG86IGAke293bmVyfS8ke25hbWV9YCxcbiAgICAgICAgICBicmFuY2hlczogW3Nlc3Npb25CcmFuY2hdLFxuICAgICAgICB9LFxuICAgICAgfVxuICAgIH1cblxuICAgIC8vIEJ1bmRsZSBmYWxsYmFjay4gT25seSB0cnkgYnVuZGxlIGlmIEdpdEh1YiB3YXNuJ3QgdmlhYmxlLCB0aGUgZ2F0ZSBpc1xuICAgIC8vIG9uLCBhbmQgdGhlcmUncyBhIC5naXQvIHRvIGJ1bmRsZSBmcm9tLiBSZWFjaGluZyBoZXJlIHdpdGhcbiAgICAvLyBnaFZpYWJsZT1mYWxzZSBhbmQgcmVwb0luZm8gbm9uLW51bGwgbWVhbnMgdGhlIHByZWZsaWdodCBmYWlsZWQg4oCUXG4gICAgLy8gLmdpdCBkZWZpbml0ZWx5IGV4aXN0cyAoZGV0ZWN0Q3VycmVudFJlcG9zaXRvcnlXaXRoSG9zdCByZWFkIHRoZVxuICAgIC8vIHJlbW90ZSBmcm9tIGl0KS5cbiAgICBpZiAoIWdpdFNvdXJjZSAmJiBidW5kbGVTZWVkR2F0ZU9uKSB7XG4gICAgICBsb2dGb3JEZWJ1Z2dpbmcoYFt0ZWxlcG9ydFRvUmVtb3RlXSBCdW5kbGluZyAocmVhc29uOiAke3NvdXJjZVJlYXNvbn0pYClcbiAgICAgIGNvbnN0IGJ1bmRsZSA9IGF3YWl0IGNyZWF0ZUFuZFVwbG9hZEdpdEJ1bmRsZShcbiAgICAgICAge1xuICAgICAgICAgIG9hdXRoVG9rZW46IGFjY2Vzc1Rva2VuLFxuICAgICAgICAgIHNlc3Npb25JZDogZ2V0U2Vzc2lvbklkKCksXG4gICAgICAgICAgYmFzZVVybDogZ2V0T2F1dGhDb25maWcoKS5CQVNFX0FQSV9VUkwsXG4gICAgICAgIH0sXG4gICAgICAgIHsgc2lnbmFsIH0sXG4gICAgICApXG4gICAgICBpZiAoIWJ1bmRsZS5zdWNjZXNzKSB7XG4gICAgICAgIGxvZ0Vycm9yKG5ldyBFcnJvcihgQnVuZGxlIHVwbG9hZCBmYWlsZWQ6ICR7YnVuZGxlLmVycm9yfWApKVxuICAgICAgICAvLyBPbmx5IHN0ZWVyIHVzZXJzIHRvIEdpdEh1YiBzZXR1cCB3aGVuIHRoZXJlJ3MgYSByZW1vdGUgdG8gY2xvbmUgZnJvbS5cbiAgICAgICAgY29uc3Qgc2V0dXAgPSByZXBvSW5mb1xuICAgICAgICAgID8gJy4gUGxlYXNlIHNldHVwIEdpdEh1YiBvbiBodHRwczovL2NsYXVkZS5haS9jb2RlJ1xuICAgICAgICAgIDogJydcbiAgICAgICAgbGV0IG1zZzogc3RyaW5nXG4gICAgICAgIHN3aXRjaCAoYnVuZGxlLmZhaWxSZWFzb24pIHtcbiAgICAgICAgICBjYXNlICdlbXB0eV9yZXBvJzpcbiAgICAgICAgICAgIG1zZyA9XG4gICAgICAgICAgICAgICdSZXBvc2l0b3J5IGhhcyBubyBjb21taXRzIOKAlCBydW4gYGdpdCBhZGQgLiAmJiBnaXQgY29tbWl0IC1tIFwiaW5pdGlhbFwiYCB0aGVuIHJldHJ5J1xuICAgICAgICAgICAgYnJlYWtcbiAgICAgICAgICBjYXNlICd0b29fbGFyZ2UnOlxuICAgICAgICAgICAgbXNnID0gYFJlcG8gaXMgdG9vIGxhcmdlIHRvIHRlbGVwb3J0JHtzZXR1cH1gXG4gICAgICAgICAgICBicmVha1xuICAgICAgICAgIGNhc2UgJ2dpdF9lcnJvcic6XG4gICAgICAgICAgICBtc2cgPSBgRmFpbGVkIHRvIGNyZWF0ZSBnaXQgYnVuZGxlICgke2J1bmRsZS5lcnJvcn0pJHtzZXR1cH1gXG4gICAgICAgICAgICBicmVha1xuICAgICAgICAgIGNhc2UgdW5kZWZpbmVkOlxuICAgICAgICAgICAgbXNnID0gYEJ1bmRsZSB1cGxvYWQgZmFpbGVkOiAke2J1bmRsZS5lcnJvcn0ke3NldHVwfWBcbiAgICAgICAgICAgIGJyZWFrXG4gICAgICAgICAgZGVmYXVsdDoge1xuICAgICAgICAgICAgY29uc3QgX2V4aGF1c3RpdmU6IG5ldmVyID0gYnVuZGxlLmZhaWxSZWFzb25cbiAgICAgICAgICAgIHZvaWQgX2V4aGF1c3RpdmVcbiAgICAgICAgICAgIG1zZyA9IGBCdW5kbGUgdXBsb2FkIGZhaWxlZDogJHtidW5kbGUuZXJyb3J9YFxuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICBvcHRpb25zLm9uQnVuZGxlRmFpbD8uKG1zZylcbiAgICAgICAgcmV0dXJuIG51bGxcbiAgICAgIH1cbiAgICAgIHNlZWRCdW5kbGVGaWxlSWQgPSBidW5kbGUuZmlsZUlkXG4gICAgICBsb2dFdmVudCgndGVuZ3VfdGVsZXBvcnRfYnVuZGxlX21vZGUnLCB7XG4gICAgICAgIHNpemVfYnl0ZXM6IGJ1bmRsZS5idW5kbGVTaXplQnl0ZXMsXG4gICAgICAgIHNjb3BlOlxuICAgICAgICAgIGJ1bmRsZS5zY29wZSBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgICBoYXNfd2lwOiBidW5kbGUuaGFzV2lwLFxuICAgICAgICByZWFzb246XG4gICAgICAgICAgc291cmNlUmVhc29uIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgICB9KVxuICAgIH1cblxuICAgIGxvZ0V2ZW50KCd0ZW5ndV90ZWxlcG9ydF9zb3VyY2VfZGVjaXNpb24nLCB7XG4gICAgICByZWFzb246XG4gICAgICAgIHNvdXJjZVJlYXNvbiBhcyBBbmFseXRpY3NNZXRhZGF0YV9JX1ZFUklGSUVEX1RISVNfSVNfTk9UX0NPREVfT1JfRklMRVBBVEhTLFxuICAgICAgcGF0aDogKGdpdFNvdXJjZVxuICAgICAgICA/ICdnaXRodWInXG4gICAgICAgIDogc2VlZEJ1bmRsZUZpbGVJZFxuICAgICAgICAgID8gJ2J1bmRsZSdcbiAgICAgICAgICA6ICdlbXB0eScpIGFzIEFuYWx5dGljc01ldGFkYXRhX0lfVkVSSUZJRURfVEhJU19JU19OT1RfQ09ERV9PUl9GSUxFUEFUSFMsXG4gICAgfSlcblxuICAgIGlmICghZ2l0U291cmNlICYmICFzZWVkQnVuZGxlRmlsZUlkKSB7XG4gICAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICAgICdbdGVsZXBvcnRUb1JlbW90ZV0gTm8gcmVwb3NpdG9yeSBkZXRlY3RlZCDigJQgc2Vzc2lvbiB3aWxsIGhhdmUgYW4gZW1wdHkgc2FuZGJveCcsXG4gICAgICApXG4gICAgfVxuXG4gICAgLy8gRmV0Y2ggYXZhaWxhYmxlIGVudmlyb25tZW50c1xuICAgIGxldCBlbnZpcm9ubWVudHMgPSBhd2FpdCBmZXRjaEVudmlyb25tZW50cygpXG4gICAgaWYgKCFlbnZpcm9ubWVudHMgfHwgZW52aXJvbm1lbnRzLmxlbmd0aCA9PT0gMCkge1xuICAgICAgbG9nRXJyb3IobmV3IEVycm9yKCdObyBlbnZpcm9ubWVudHMgYXZhaWxhYmxlIGZvciBzZXNzaW9uIGNyZWF0aW9uJykpXG4gICAgICByZXR1cm4gbnVsbFxuICAgIH1cblxuICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgIGBBdmFpbGFibGUgZW52aXJvbm1lbnRzOiAke2Vudmlyb25tZW50cy5tYXAoZSA9PiBgJHtlLmVudmlyb25tZW50X2lkfSAoJHtlLm5hbWV9LCAke2Uua2luZH0pYCkuam9pbignLCAnKX1gLFxuICAgIClcblxuICAgIC8vIFNlbGVjdCBlbnZpcm9ubWVudCBiYXNlZCBvbiBzZXR0aW5ncywgdGhlbiBhbnRocm9waWNfY2xvdWQgcHJlZmVyZW5jZSwgdGhlbiBmaXJzdCBhdmFpbGFibGUuXG4gICAgLy8gUHJlZmVyIGFudGhyb3BpY19jbG91ZCBlbnZpcm9ubWVudHMgb3ZlciBieW9jOiBhbnRocm9waWNfY2xvdWQgZW52aXJvbm1lbnRzIChlLmcuIFwiRGVmYXVsdFwiKVxuICAgIC8vIGFyZSB0aGUgc3RhbmRhcmQgY29tcHV0ZSBlbnZpcm9ubWVudHMgd2l0aCBmdWxsIHJlcG8gYWNjZXNzLCB3aGVyZWFzIGJ5b2MgZW52aXJvbm1lbnRzXG4gICAgLy8gKGUuZy4gXCJtb25vcmVwb1wiKSBhcmUgdXNlci1vd25lZCBjb21wdXRlIHRoYXQgbWF5IG5vdCBzdXBwb3J0IHRoZSBjdXJyZW50IHJlcG9zaXRvcnkuXG4gICAgY29uc3Qgc2V0dGluZ3MgPSBnZXRTZXR0aW5nc19ERVBSRUNBVEVEKClcbiAgICBjb25zdCBkZWZhdWx0RW52aXJvbm1lbnRJZCA9IG9wdGlvbnMudXNlRGVmYXVsdEVudmlyb25tZW50XG4gICAgICA/IHVuZGVmaW5lZFxuICAgICAgOiBzZXR0aW5ncz8ucmVtb3RlPy5kZWZhdWx0RW52aXJvbm1lbnRJZFxuICAgIGxldCBjbG91ZEVudiA9IGVudmlyb25tZW50cy5maW5kKGVudiA9PiBlbnYua2luZCA9PT0gJ2FudGhyb3BpY19jbG91ZCcpXG4gICAgLy8gV2hlbiB0aGUgY2FsbGVyIG9wdHMgb3V0IG9mIHRoZWlyIGNvbmZpZ3VyZWQgZGVmYXVsdCwgZG8gbm90IGZhbGxcbiAgICAvLyB0aHJvdWdoIHRvIGEgQllPQyBlbnYgdGhhdCBtYXkgbm90IHN1cHBvcnQgdGhlIGN1cnJlbnQgcmVwbyBvciB0aGVcbiAgICAvLyByZXF1ZXN0ZWQgcGVybWlzc2lvbiBtb2RlLiBSZXRyeSBvbmNlIGZvciBldmVudHVhbCBjb25zaXN0ZW5jeSxcbiAgICAvLyB0aGVuIGZhaWwgbG91ZGx5LlxuICAgIGlmIChvcHRpb25zLnVzZURlZmF1bHRFbnZpcm9ubWVudCAmJiAhY2xvdWRFbnYpIHtcbiAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgYE5vIGFudGhyb3BpY19jbG91ZCBpbiBlbnYgbGlzdCAoJHtlbnZpcm9ubWVudHMubGVuZ3RofSBlbnZzKTsgcmV0cnlpbmcgZmV0Y2hFbnZpcm9ubWVudHNgLFxuICAgICAgKVxuICAgICAgY29uc3QgcmV0cmllZCA9IGF3YWl0IGZldGNoRW52aXJvbm1lbnRzKClcbiAgICAgIGNsb3VkRW52ID0gcmV0cmllZD8uZmluZChlbnYgPT4gZW52LmtpbmQgPT09ICdhbnRocm9waWNfY2xvdWQnKVxuICAgICAgaWYgKCFjbG91ZEVudikge1xuICAgICAgICBsb2dFcnJvcihcbiAgICAgICAgICBuZXcgRXJyb3IoXG4gICAgICAgICAgICBgTm8gYW50aHJvcGljX2Nsb3VkIGVudmlyb25tZW50IGF2YWlsYWJsZSBhZnRlciByZXRyeSAoZ290OiAkeyhyZXRyaWVkID8/IGVudmlyb25tZW50cykubWFwKGUgPT4gYCR7ZS5uYW1lfSAoJHtlLmtpbmR9KWApLmpvaW4oJywgJyl9KS4gU2lsZW50IGJ5b2MgZmFsbHRocm91Z2ggd291bGQgbGF1bmNoIGludG8gYSBkZWFkIGVudiDigJQgZmFpbCBmYXN0IGluc3RlYWQuYCxcbiAgICAgICAgICApLFxuICAgICAgICApXG4gICAgICAgIHJldHVybiBudWxsXG4gICAgICB9XG4gICAgICBpZiAocmV0cmllZCkgZW52aXJvbm1lbnRzID0gcmV0cmllZFxuICAgIH1cbiAgICBjb25zdCBzZWxlY3RlZEVudmlyb25tZW50ID1cbiAgICAgIChkZWZhdWx0RW52aXJvbm1lbnRJZCAmJlxuICAgICAgICBlbnZpcm9ubWVudHMuZmluZChcbiAgICAgICAgICBlbnYgPT4gZW52LmVudmlyb25tZW50X2lkID09PSBkZWZhdWx0RW52aXJvbm1lbnRJZCxcbiAgICAgICAgKSkgfHxcbiAgICAgIGNsb3VkRW52IHx8XG4gICAgICBlbnZpcm9ubWVudHMuZmluZChlbnYgPT4gZW52LmtpbmQgIT09ICdicmlkZ2UnKSB8fFxuICAgICAgZW52aXJvbm1lbnRzWzBdXG5cbiAgICBpZiAoIXNlbGVjdGVkRW52aXJvbm1lbnQpIHtcbiAgICAgIGxvZ0Vycm9yKG5ldyBFcnJvcignTm8gZW52aXJvbm1lbnRzIGF2YWlsYWJsZSBmb3Igc2Vzc2lvbiBjcmVhdGlvbicpKVxuICAgICAgcmV0dXJuIG51bGxcbiAgICB9XG5cbiAgICBpZiAoZGVmYXVsdEVudmlyb25tZW50SWQpIHtcbiAgICAgIGNvbnN0IG1hdGNoZWREZWZhdWx0ID1cbiAgICAgICAgc2VsZWN0ZWRFbnZpcm9ubWVudC5lbnZpcm9ubWVudF9pZCA9PT0gZGVmYXVsdEVudmlyb25tZW50SWRcbiAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgbWF0Y2hlZERlZmF1bHRcbiAgICAgICAgICA/IGBVc2luZyBjb25maWd1cmVkIGRlZmF1bHQgZW52aXJvbm1lbnQ6ICR7ZGVmYXVsdEVudmlyb25tZW50SWR9YFxuICAgICAgICAgIDogYENvbmZpZ3VyZWQgZGVmYXVsdCBlbnZpcm9ubWVudCAke2RlZmF1bHRFbnZpcm9ubWVudElkfSBub3QgZm91bmQsIHVzaW5nIGZpcnN0IGF2YWlsYWJsZWAsXG4gICAgICApXG4gICAgfVxuXG4gICAgY29uc3QgZW52aXJvbm1lbnRJZCA9IHNlbGVjdGVkRW52aXJvbm1lbnQuZW52aXJvbm1lbnRfaWRcbiAgICBsb2dGb3JEZWJ1Z2dpbmcoXG4gICAgICBgU2VsZWN0ZWQgZW52aXJvbm1lbnQ6ICR7ZW52aXJvbm1lbnRJZH0gKCR7c2VsZWN0ZWRFbnZpcm9ubWVudC5uYW1lfSwgJHtzZWxlY3RlZEVudmlyb25tZW50LmtpbmR9KWAsXG4gICAgKVxuXG4gICAgLy8gUHJlcGFyZSBBUEkgcmVxdWVzdCBmb3IgU2Vzc2lvbnMgQVBJXG4gICAgY29uc3QgdXJsID0gYCR7Z2V0T2F1dGhDb25maWcoKS5CQVNFX0FQSV9VUkx9L3YxL3Nlc3Npb25zYFxuXG4gICAgY29uc3QgaGVhZGVycyA9IHtcbiAgICAgIC4uLmdldE9BdXRoSGVhZGVycyhhY2Nlc3NUb2tlbiksXG4gICAgICAnYW50aHJvcGljLWJldGEnOiAnY2NyLWJ5b2MtMjAyNS0wNy0yOScsXG4gICAgICAneC1vcmdhbml6YXRpb24tdXVpZCc6IG9yZ1VVSUQsXG4gICAgfVxuXG4gICAgY29uc3Qgc2Vzc2lvbkNvbnRleHQgPSB7XG4gICAgICBzb3VyY2VzOiBnaXRTb3VyY2UgPyBbZ2l0U291cmNlXSA6IFtdLFxuICAgICAgLi4uKHNlZWRCdW5kbGVGaWxlSWQgJiYgeyBzZWVkX2J1bmRsZV9maWxlX2lkOiBzZWVkQnVuZGxlRmlsZUlkIH0pLFxuICAgICAgb3V0Y29tZXM6IGdpdE91dGNvbWUgPyBbZ2l0T3V0Y29tZV0gOiBbXSxcbiAgICAgIG1vZGVsOiBvcHRpb25zLm1vZGVsID8/IGdldE1haW5Mb29wTW9kZWwoKSxcbiAgICAgIC4uLihvcHRpb25zLnJldXNlT3V0Y29tZUJyYW5jaCAmJiB7IHJldXNlX291dGNvbWVfYnJhbmNoZXM6IHRydWUgfSksXG4gICAgICAuLi4ob3B0aW9ucy5naXRodWJQciAmJiB7IGdpdGh1Yl9wcjogb3B0aW9ucy5naXRodWJQciB9KSxcbiAgICB9XG5cbiAgICAvLyBDcmVhdGVDQ1JTZXNzaW9uUGF5bG9hZCBoYXMgbm8gcGVybWlzc2lvbl9tb2RlIGZpZWxkIOKAlCBhIHRvcC1sZXZlbFxuICAgIC8vIGJvZHkgZW50cnkgaXMgc2lsZW50bHkgZHJvcHBlZCBieSB0aGUgcHJvdG8gcGFyc2VyIHNlcnZlci1zaWRlLlxuICAgIC8vIEluc3RlYWQgcHJlcGVuZCBhIHNldF9wZXJtaXNzaW9uX21vZGUgY29udHJvbF9yZXF1ZXN0IGV2ZW50LiBJbml0aWFsXG4gICAgLy8gZXZlbnRzIGFyZSB3cml0dGVuIHRvIHRocmVhZHN0b3JlIGJlZm9yZSB0aGUgY29udGFpbmVyIGNvbm5lY3RzLCBzb1xuICAgIC8vIHRoZSBDTEkgYXBwbGllcyB0aGUgbW9kZSBiZWZvcmUgdGhlIGZpcnN0IHVzZXIgdHVybiDigJQgbm8gcmVhZGluZXNzIHJhY2UuXG4gICAgY29uc3QgZXZlbnRzOiBBcnJheTx7IHR5cGU6ICdldmVudCc7IGRhdGE6IFJlY29yZDxzdHJpbmcsIHVua25vd24+IH0+ID0gW11cbiAgICBpZiAob3B0aW9ucy5wZXJtaXNzaW9uTW9kZSkge1xuICAgICAgZXZlbnRzLnB1c2goe1xuICAgICAgICB0eXBlOiAnZXZlbnQnLFxuICAgICAgICBkYXRhOiB7XG4gICAgICAgICAgdHlwZTogJ2NvbnRyb2xfcmVxdWVzdCcsXG4gICAgICAgICAgcmVxdWVzdF9pZDogYHNldC1tb2RlLSR7cmFuZG9tVVVJRCgpfWAsXG4gICAgICAgICAgcmVxdWVzdDoge1xuICAgICAgICAgICAgc3VidHlwZTogJ3NldF9wZXJtaXNzaW9uX21vZGUnLFxuICAgICAgICAgICAgbW9kZTogb3B0aW9ucy5wZXJtaXNzaW9uTW9kZSxcbiAgICAgICAgICAgIHVsdHJhcGxhbjogb3B0aW9ucy51bHRyYXBsYW4sXG4gICAgICAgICAgfSxcbiAgICAgICAgfSxcbiAgICAgIH0pXG4gICAgfVxuICAgIGlmIChpbml0aWFsTWVzc2FnZSkge1xuICAgICAgZXZlbnRzLnB1c2goe1xuICAgICAgICB0eXBlOiAnZXZlbnQnLFxuICAgICAgICBkYXRhOiB7XG4gICAgICAgICAgdXVpZDogcmFuZG9tVVVJRCgpLFxuICAgICAgICAgIHNlc3Npb25faWQ6ICcnLFxuICAgICAgICAgIHR5cGU6ICd1c2VyJyxcbiAgICAgICAgICBwYXJlbnRfdG9vbF91c2VfaWQ6IG51bGwsXG4gICAgICAgICAgbWVzc2FnZToge1xuICAgICAgICAgICAgcm9sZTogJ3VzZXInLFxuICAgICAgICAgICAgY29udGVudDogaW5pdGlhbE1lc3NhZ2UsXG4gICAgICAgICAgfSxcbiAgICAgICAgfSxcbiAgICAgIH0pXG4gICAgfVxuXG4gICAgY29uc3QgcmVxdWVzdEJvZHkgPSB7XG4gICAgICB0aXRsZTogb3B0aW9ucy51bHRyYXBsYW4gPyBgdWx0cmFwbGFuOiAke3Nlc3Npb25UaXRsZX1gIDogc2Vzc2lvblRpdGxlLFxuICAgICAgZXZlbnRzLFxuICAgICAgc2Vzc2lvbl9jb250ZXh0OiBzZXNzaW9uQ29udGV4dCxcbiAgICAgIGVudmlyb25tZW50X2lkOiBlbnZpcm9ubWVudElkLFxuICAgIH1cblxuICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgIGBDcmVhdGluZyBzZXNzaW9uIHdpdGggcGF5bG9hZDogJHtqc29uU3RyaW5naWZ5KHJlcXVlc3RCb2R5LCBudWxsLCAyKX1gLFxuICAgIClcblxuICAgIC8vIE1ha2UgQVBJIGNhbGxcbiAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IGF4aW9zLnBvc3QodXJsLCByZXF1ZXN0Qm9keSwgeyBoZWFkZXJzLCBzaWduYWwgfSlcbiAgICBjb25zdCBpc1N1Y2Nlc3MgPSByZXNwb25zZS5zdGF0dXMgPT09IDIwMCB8fCByZXNwb25zZS5zdGF0dXMgPT09IDIwMVxuXG4gICAgaWYgKCFpc1N1Y2Nlc3MpIHtcbiAgICAgIGxvZ0Vycm9yKFxuICAgICAgICBuZXcgRXJyb3IoXG4gICAgICAgICAgYEFQSSByZXF1ZXN0IGZhaWxlZCB3aXRoIHN0YXR1cyAke3Jlc3BvbnNlLnN0YXR1c306ICR7cmVzcG9uc2Uuc3RhdHVzVGV4dH1cXG5cXG5SZXNwb25zZSBkYXRhOiAke2pzb25TdHJpbmdpZnkocmVzcG9uc2UuZGF0YSwgbnVsbCwgMil9YCxcbiAgICAgICAgKSxcbiAgICAgIClcbiAgICAgIHJldHVybiBudWxsXG4gICAgfVxuXG4gICAgLy8gUGFyc2UgcmVzcG9uc2UgYXMgU2Vzc2lvblJlc291cmNlXG4gICAgY29uc3Qgc2Vzc2lvbkRhdGEgPSByZXNwb25zZS5kYXRhIGFzIFNlc3Npb25SZXNvdXJjZVxuICAgIGlmICghc2Vzc2lvbkRhdGEgfHwgdHlwZW9mIHNlc3Npb25EYXRhLmlkICE9PSAnc3RyaW5nJykge1xuICAgICAgbG9nRXJyb3IoXG4gICAgICAgIG5ldyBFcnJvcihcbiAgICAgICAgICBgQ2Fubm90IGRldGVybWluZSBzZXNzaW9uIElEIGZyb20gQVBJIHJlc3BvbnNlOiAke2pzb25TdHJpbmdpZnkocmVzcG9uc2UuZGF0YSl9YCxcbiAgICAgICAgKSxcbiAgICAgIClcbiAgICAgIHJldHVybiBudWxsXG4gICAgfVxuXG4gICAgbG9nRm9yRGVidWdnaW5nKGBTdWNjZXNzZnVsbHkgY3JlYXRlZCByZW1vdGUgc2Vzc2lvbjogJHtzZXNzaW9uRGF0YS5pZH1gKVxuICAgIHJldHVybiB7XG4gICAgICBpZDogc2Vzc2lvbkRhdGEuaWQsXG4gICAgICB0aXRsZTogc2Vzc2lvbkRhdGEudGl0bGUgfHwgcmVxdWVzdEJvZHkudGl0bGUsXG4gICAgfVxuICB9IGNhdGNoIChlcnJvcikge1xuICAgIGNvbnN0IGVyciA9IHRvRXJyb3IoZXJyb3IpXG4gICAgbG9nRXJyb3IoZXJyKVxuICAgIHJldHVybiBudWxsXG4gIH1cbn1cblxuLyoqXG4gKiBCZXN0LWVmZm9ydCBzZXNzaW9uIGFyY2hpdmUuIFBPU1QgL3YxL3Nlc3Npb25zL3tpZH0vYXJjaGl2ZSBoYXMgbm9cbiAqIHJ1bm5pbmctc3RhdHVzIGNoZWNrICh1bmxpa2UgREVMRVRFIHdoaWNoIDQwOXMgb24gUlVOTklORyksIHNvIGl0IHdvcmtzXG4gKiBtaWQtaW1wbGVtZW50YXRpb24uIEFyY2hpdmVkIHNlc3Npb25zIHJlamVjdCBuZXcgZXZlbnRzIChzZW5kX2V2ZW50cy5nbyksXG4gKiBzbyB0aGUgcmVtb3RlIHN0b3BzIG9uIGl0cyBuZXh0IHdyaXRlLiA0MDkgKGFscmVhZHkgYXJjaGl2ZWQpIHRyZWF0ZWQgYXNcbiAqIHN1Y2Nlc3MuIEZpcmUtYW5kLWZvcmdldDsgZmFpbHVyZSBsZWFrcyBhIHZpc2libGUgc2Vzc2lvbiB1bnRpbCB0aGVcbiAqIHJlYXBlciBjb2xsZWN0cyBpdC5cbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGFyY2hpdmVSZW1vdGVTZXNzaW9uKHNlc3Npb25JZDogc3RyaW5nKTogUHJvbWlzZTx2b2lkPiB7XG4gIGNvbnN0IGFjY2Vzc1Rva2VuID0gZ2V0Q2xhdWRlQUlPQXV0aFRva2VucygpPy5hY2Nlc3NUb2tlblxuICBpZiAoIWFjY2Vzc1Rva2VuKSByZXR1cm5cbiAgY29uc3Qgb3JnVVVJRCA9IGF3YWl0IGdldE9yZ2FuaXphdGlvblVVSUQoKVxuICBpZiAoIW9yZ1VVSUQpIHJldHVyblxuICBjb25zdCBoZWFkZXJzID0ge1xuICAgIC4uLmdldE9BdXRoSGVhZGVycyhhY2Nlc3NUb2tlbiksXG4gICAgJ2FudGhyb3BpYy1iZXRhJzogJ2Njci1ieW9jLTIwMjUtMDctMjknLFxuICAgICd4LW9yZ2FuaXphdGlvbi11dWlkJzogb3JnVVVJRCxcbiAgfVxuICBjb25zdCB1cmwgPSBgJHtnZXRPYXV0aENvbmZpZygpLkJBU0VfQVBJX1VSTH0vdjEvc2Vzc2lvbnMvJHtzZXNzaW9uSWR9L2FyY2hpdmVgXG4gIHRyeSB7XG4gICAgY29uc3QgcmVzcCA9IGF3YWl0IGF4aW9zLnBvc3QoXG4gICAgICB1cmwsXG4gICAgICB7fSxcbiAgICAgIHsgaGVhZGVycywgdGltZW91dDogMTAwMDAsIHZhbGlkYXRlU3RhdHVzOiBzID0+IHMgPCA1MDAgfSxcbiAgICApXG4gICAgaWYgKHJlc3Auc3RhdHVzID09PSAyMDAgfHwgcmVzcC5zdGF0dXMgPT09IDQwOSkge1xuICAgICAgbG9nRm9yRGVidWdnaW5nKGBbYXJjaGl2ZVJlbW90ZVNlc3Npb25dIGFyY2hpdmVkICR7c2Vzc2lvbklkfWApXG4gICAgfSBlbHNlIHtcbiAgICAgIGxvZ0ZvckRlYnVnZ2luZyhcbiAgICAgICAgYFthcmNoaXZlUmVtb3RlU2Vzc2lvbl0gJHtzZXNzaW9uSWR9IGZhaWxlZCAke3Jlc3Auc3RhdHVzfTogJHtqc29uU3RyaW5naWZ5KHJlc3AuZGF0YSl9YCxcbiAgICAgIClcbiAgICB9XG4gIH0gY2F0Y2ggKGVycikge1xuICAgIGxvZ0Vycm9yKGVycilcbiAgfVxufVxuIl0sIm1hcHBpbmdzIjoiQUFBQSxPQUFPQSxLQUFLLE1BQU0sT0FBTztBQUN6QixPQUFPQyxLQUFLLE1BQU0sT0FBTztBQUN6QixTQUFTQyxVQUFVLFFBQVEsUUFBUTtBQUNuQyxPQUFPQyxLQUFLLE1BQU0sT0FBTztBQUN6QixTQUFTQyxjQUFjLEVBQUVDLFlBQVksUUFBUSx3QkFBd0I7QUFDckUsU0FBU0MsNEJBQTRCLFFBQVEsc0NBQXNDO0FBQ25GLFNBQ0UsS0FBS0MsMERBQTBELEVBQy9EQyxRQUFRLFFBQ0gsaUNB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