Filehigh importancesource

PlanApprovalMessage.tsx

components/messages/PlanApprovalMessage.tsx

222
Lines
25339
Bytes
4
Exports
8
Imports
10
Keywords

What this is

This page documents one file from the repository and includes its full source so you can read it without leaving the docs site.

Beginner explanation

This file is one piece of the larger system. Its name, directory, imports, and exports show where it fits. Start by reading the exports and related files first.

How it is used

Start from the exports list and related files. Those are the easiest clues for where this file fits into the system.

Expert explanation

Architecturally, this file intersects with planner-verifier-agents, ui-flow. It contains 222 lines, 8 detected imports, and 4 detected exports.

Important relationships

Detected exports

  • PlanApprovalRequestDisplay
  • PlanApprovalResponseDisplay
  • tryRenderPlanApprovalMessage
  • formatTeammateMessageContent

Keywords

requestplancontenttextresponseelsesendernamemessageapprovalflexdirection

Detected imports

  • react/compiler-runtime
  • react
  • ../../components/Markdown.js
  • ../../ink.js
  • ../../utils/slowOperations.js
  • ../../utils/teammateMailbox.js
  • ./ShutdownMessage.js
  • ./TaskAssignmentMessage.js

Source notes

This page embeds the full file contents. Small or leaf files are still indexed honestly instead of being over-explained.

Open parent directory

Full source

import { c as _c } from "react/compiler-runtime";
import * as React from 'react';
import { Markdown } from '../../components/Markdown.js';
import { Box, Text } from '../../ink.js';
import { jsonParse } from '../../utils/slowOperations.js';
import { type IdleNotificationMessage, isIdleNotification, isPlanApprovalRequest, isPlanApprovalResponse, type PlanApprovalRequestMessage, type PlanApprovalResponseMessage } from '../../utils/teammateMailbox.js';
import { getShutdownMessageSummary } from './ShutdownMessage.js';
import { getTaskAssignmentSummary } from './TaskAssignmentMessage.js';
type PlanApprovalRequestProps = {
  request: PlanApprovalRequestMessage;
};

/**
 * Renders a plan approval request with a planMode-colored border,
 * showing the plan content and instructions for approving/rejecting.
 */
export function PlanApprovalRequestDisplay(t0) {
  const $ = _c(10);
  const {
    request
  } = t0;
  let t1;
  if ($[0] !== request.from) {
    t1 = <Box marginBottom={1}><Text color="planMode" bold={true}>Plan Approval Request from {request.from}</Text></Box>;
    $[0] = request.from;
    $[1] = t1;
  } else {
    t1 = $[1];
  }
  let t2;
  if ($[2] !== request.planContent) {
    t2 = <Box borderStyle="dashed" borderColor="subtle" borderLeft={false} borderRight={false} flexDirection="column" paddingX={1} marginBottom={1}><Markdown>{request.planContent}</Markdown></Box>;
    $[2] = request.planContent;
    $[3] = t2;
  } else {
    t2 = $[3];
  }
  let t3;
  if ($[4] !== request.planFilePath) {
    t3 = <Text dimColor={true}>Plan file: {request.planFilePath}</Text>;
    $[4] = request.planFilePath;
    $[5] = t3;
  } else {
    t3 = $[5];
  }
  let t4;
  if ($[6] !== t1 || $[7] !== t2 || $[8] !== t3) {
    t4 = <Box flexDirection="column" marginY={1}><Box borderStyle="round" borderColor="planMode" flexDirection="column" paddingX={1}>{t1}{t2}{t3}</Box></Box>;
    $[6] = t1;
    $[7] = t2;
    $[8] = t3;
    $[9] = t4;
  } else {
    t4 = $[9];
  }
  return t4;
}
type PlanApprovalResponseProps = {
  response: PlanApprovalResponseMessage;
  senderName: string;
};

/**
 * Renders a plan approval response with a success (green) or error (red) border.
 */
export function PlanApprovalResponseDisplay(t0) {
  const $ = _c(13);
  const {
    response,
    senderName
  } = t0;
  if (response.approved) {
    let t1;
    if ($[0] !== senderName) {
      t1 = <Box><Text color="success" bold={true}>✓ Plan Approved by {senderName}</Text></Box>;
      $[0] = senderName;
      $[1] = t1;
    } else {
      t1 = $[1];
    }
    let t2;
    if ($[2] === Symbol.for("react.memo_cache_sentinel")) {
      t2 = <Box marginTop={1}><Text>You can now proceed with implementation. Your plan mode restrictions have been lifted.</Text></Box>;
      $[2] = t2;
    } else {
      t2 = $[2];
    }
    let t3;
    if ($[3] !== t1) {
      t3 = <Box flexDirection="column" marginY={1}><Box borderStyle="round" borderColor="success" flexDirection="column" paddingX={1} paddingY={1}>{t1}{t2}</Box></Box>;
      $[3] = t1;
      $[4] = t3;
    } else {
      t3 = $[4];
    }
    return t3;
  }
  let t1;
  if ($[5] !== senderName) {
    t1 = <Box><Text color="error" bold={true}>✗ Plan Rejected by {senderName}</Text></Box>;
    $[5] = senderName;
    $[6] = t1;
  } else {
    t1 = $[6];
  }
  let t2;
  if ($[7] !== response.feedback) {
    t2 = response.feedback && <Box marginTop={1} borderStyle="dashed" borderColor="subtle" borderLeft={false} borderRight={false} paddingX={1}><Text>Feedback: {response.feedback}</Text></Box>;
    $[7] = response.feedback;
    $[8] = t2;
  } else {
    t2 = $[8];
  }
  let t3;
  if ($[9] === Symbol.for("react.memo_cache_sentinel")) {
    t3 = <Box marginTop={1}><Text dimColor={true}>Please revise your plan based on the feedback and call ExitPlanMode again.</Text></Box>;
    $[9] = t3;
  } else {
    t3 = $[9];
  }
  let t4;
  if ($[10] !== t1 || $[11] !== t2) {
    t4 = <Box flexDirection="column" marginY={1}><Box borderStyle="round" borderColor="error" flexDirection="column" paddingX={1} paddingY={1}>{t1}{t2}{t3}</Box></Box>;
    $[10] = t1;
    $[11] = t2;
    $[12] = t4;
  } else {
    t4 = $[12];
  }
  return t4;
}

/**
 * Try to parse and render a plan approval message from raw content.
 * Returns the rendered component if it's a plan approval message, null otherwise.
 */
export function tryRenderPlanApprovalMessage(content: string, senderName: string): React.ReactNode | null {
  const request = isPlanApprovalRequest(content);
  if (request) {
    return <PlanApprovalRequestDisplay request={request} />;
  }
  const response = isPlanApprovalResponse(content);
  if (response) {
    return <PlanApprovalResponseDisplay response={response} senderName={senderName} />;
  }
  return null;
}

/**
 * Get a brief summary text for a plan approval message.
 * Used in places like the inbox queue where we want a short description.
 * Returns null if the content is not a plan approval message.
 */
function getPlanApprovalSummary(content: string): string | null {
  const request = isPlanApprovalRequest(content);
  if (request) {
    return `[Plan Approval Request from ${request.from}]`;
  }
  const response = isPlanApprovalResponse(content);
  if (response) {
    if (response.approved) {
      return '[Plan Approved] You can now proceed with implementation';
    } else {
      return `[Plan Rejected] ${response.feedback || 'Please revise your plan'}`;
    }
  }
  return null;
}

/**
 * Get a brief summary text for an idle notification.
 */
function getIdleNotificationSummary(msg: IdleNotificationMessage): string {
  const parts: string[] = ['Agent idle'];
  if (msg.completedTaskId) {
    const status = msg.completedStatus || 'completed';
    parts.push(`Task ${msg.completedTaskId} ${status}`);
  }
  if (msg.summary) {
    parts.push(`Last DM: ${msg.summary}`);
  }
  return parts.join(' · ');
}

/**
 * Format teammate message content for display.
 * If it's a structured message (plan approval, shutdown, or idle), returns a formatted summary.
 * Otherwise returns the original content.
 */
export function formatTeammateMessageContent(content: string): string {
  const planSummary = getPlanApprovalSummary(content);
  if (planSummary) {
    return planSummary;
  }
  const shutdownSummary = getShutdownMessageSummary(content);
  if (shutdownSummary) {
    return shutdownSummary;
  }
  const idleMsg = isIdleNotification(content);
  if (idleMsg) {
    return getIdleNotificationSummary(idleMsg);
  }
  const taskAssignmentSummary = getTaskAssignmentSummary(content);
  if (taskAssignmentSummary) {
    return taskAssignmentSummary;
  }

  // Check for teammate_terminated message
  try {
    const parsed = jsonParse(content) as {
      type?: string;
      message?: string;
    };
    if (parsed?.type === 'teammate_terminated' && parsed.message) {
      return parsed.message;
    }
  } catch {
    // Not JSON
  }
  return content;
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJSZWFjdCIsIk1hcmtkb3duIiwiQm94IiwiVGV4dCIsImpzb25QYXJzZSIsIklkbGVOb3RpZmljYXRpb25NZXNzYWdlIiwiaXNJZGxlTm90aWZpY2F0aW9uIiwiaXNQbGFuQXBwcm92YWxSZXF1ZXN0IiwiaXNQbGFuQXBwcm92YWxSZXNwb25zZSIsIlBsYW5BcHByb3ZhbFJlcXVlc3RNZXNzYWdlIiwiUGxhbkFwcHJvdmFsUmVzcG9uc2VNZXNzYWdlIiwiZ2V0U2h1dGRvd25NZXNzYWdlU3VtbWFyeSIsImdldFRhc2tBc3NpZ25tZW50U3VtbWFyeSIsIlBsYW5BcHByb3ZhbFJlcXVlc3RQcm9wcyIsInJlcXVlc3QiLCJQbGFuQXBwcm92YWxSZXF1ZXN0RGlzcGxheSIsInQwIiwiJCIsIl9jIiwidDEiLCJmcm9tIiwidDIiLCJwbGFuQ29udGVudCIsInQzIiwicGxhbkZpbGVQYXRoIiwidDQiLCJQbGFuQXBwcm92YWxSZXNwb25zZVByb3BzIiwicmVzcG9uc2UiLCJzZW5kZXJOYW1lIiwiUGxhbkFwcHJvdmFsUmVzcG9uc2VEaXNwbGF5IiwiYXBwcm92ZWQiLCJTeW1ib2wiLCJmb3IiLCJmZWVkYmFjayIsInRyeVJlbmRlclBsYW5BcHByb3ZhbE1lc3NhZ2UiLCJjb250ZW50IiwiUmVhY3ROb2RlIiwiZ2V0UGxhbkFwcHJvdmFsU3VtbWFyeSIsImdldElkbGVOb3RpZmljYXRpb25TdW1tYXJ5IiwibXNnIiwicGFydHMiLCJjb21wbGV0ZWRUYXNrSWQiLCJzdGF0dXMiLCJjb21wbGV0ZWRTdGF0dXMiLCJwdXNoIiwic3VtbWFyeSIsImpvaW4iLCJmb3JtYXRUZWFtbWF0ZU1lc3NhZ2VDb250ZW50IiwicGxhblN1bW1hcnkiLCJzaHV0ZG93blN1bW1hcnkiLCJpZGxlTXNnIiwidGFza0Fzc2lnbm1lbnRTdW1tYXJ5IiwicGFyc2VkIiwidHlwZSIsIm1lc3NhZ2UiXSwic291cmNlcyI6WyJQbGFuQXBwcm92YWxNZXNzYWdlLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCB7IE1hcmtkb3duIH0gZnJvbSAnLi4vLi4vY29tcG9uZW50cy9NYXJrZG93bi5qcydcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB7IGpzb25QYXJzZSB9IGZyb20gJy4uLy4uL3V0aWxzL3Nsb3dPcGVyYXRpb25zLmpzJ1xuaW1wb3J0IHtcbiAgdHlwZSBJZGxlTm90aWZpY2F0aW9uTWVzc2FnZSxcbiAgaXNJZGxlTm90aWZpY2F0aW9uLFxuICBpc1BsYW5BcHByb3ZhbFJlcXVlc3QsXG4gIGlzUGxhbkFwcHJvdmFsUmVzcG9uc2UsXG4gIHR5cGUgUGxhbkFwcHJvdmFsUmVxdWVzdE1lc3NhZ2UsXG4gIHR5cGUgUGxhbkFwcHJvdmFsUmVzcG9uc2VNZXNzYWdlLFxufSBmcm9tICcuLi8uLi91dGlscy90ZWFtbWF0ZU1haWxib3guanMnXG5pbXBvcnQgeyBnZXRTaHV0ZG93bk1lc3NhZ2VTdW1tYXJ5IH0gZnJvbSAnLi9TaHV0ZG93bk1lc3NhZ2UuanMnXG5pbXBvcnQgeyBnZXRUYXNrQXNzaWdubWVudFN1bW1hcnkgfSBmcm9tICcuL1Rhc2tBc3NpZ25tZW50TWVzc2FnZS5qcydcblxudHlwZSBQbGFuQXBwcm92YWxSZXF1ZXN0UHJvcHMgPSB7XG4gIHJlcXVlc3Q6IFBsYW5BcHByb3ZhbFJlcXVlc3RNZXNzYWdlXG59XG5cbi8qKlxuICogUmVuZGVycyBhIHBsYW4gYXBwcm92YWwgcmVxdWVzdCB3aXRoIGEgcGxhbk1vZGUtY29sb3JlZCBib3JkZXIsXG4gKiBzaG93aW5nIHRoZSBwbGFuIGNvbnRlbnQgYW5kIGluc3RydWN0aW9ucyBmb3IgYXBwcm92aW5nL3JlamVjdGluZy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIFBsYW5BcHByb3ZhbFJlcXVlc3REaXNwbGF5KHtcbiAgcmVxdWVzdCxcbn06IFBsYW5BcHByb3ZhbFJlcXVlc3RQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIHJldHVybiAoXG4gICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgbWFyZ2luWT17MX0+XG4gICAgICA8Qm94XG4gICAgICAgIGJvcmRlclN0eWxlPVwicm91bmRcIlxuICAgICAgICBib3JkZXJDb2xvcj1cInBsYW5Nb2RlXCJcbiAgICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICAgIHBhZGRpbmdYPXsxfVxuICAgICAgPlxuICAgICAgICA8Qm94IG1hcmdpbkJvdHRvbT17MX0+XG4gICAgICAgICAgPFRleHQgY29sb3I9XCJwbGFuTW9kZVwiIGJvbGQ+XG4gICAgICAgICAgICBQbGFuIEFwcHJvdmFsIFJlcXVlc3QgZnJvbSB7cmVxdWVzdC5mcm9tfVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICAgIDxCb3hcbiAgICAgICAgICBib3JkZXJTdHlsZT1cImRhc2hlZFwiXG4gICAgICAgICAgYm9yZGVyQ29sb3I9XCJzdWJ0bGVcIlxuICAgICAgICAgIGJvcmRlckxlZnQ9e2ZhbHNlfVxuICAgICAgICAgIGJvcmRlclJpZ2h0PXtmYWxzZX1cbiAgICAgICAgICBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCJcbiAgICAgICAgICBwYWRkaW5nWD17MX1cbiAgICAgICAgICBtYXJnaW5Cb3R0b209ezF9XG4gICAgICAgID5cbiAgICAgICAgICA8TWFya2Rvd24+e3JlcXVlc3QucGxhbkNvbnRlbnR9PC9NYXJrZG93bj5cbiAgICAgICAgPC9Cb3g+XG4gICAgICAgIDxUZXh0IGRpbUNvbG9yPlBsYW4gZmlsZToge3JlcXVlc3QucGxhbkZpbGVQYXRofTwvVGV4dD5cbiAgICAgIDwvQm94PlxuICAgIDwvQm94PlxuICApXG59XG5cbnR5cGUgUGxhbkFwcHJvdmFsUmVzcG9uc2VQcm9wcyA9IHtcbiAgcmVzcG9uc2U6IFBsYW5BcHByb3ZhbFJlc3BvbnNlTWVzc2FnZVxuICBzZW5kZXJOYW1lOiBzdHJpbmdcbn1cblxuLyoqXG4gKiBSZW5kZXJzIGEgcGxhbiBhcHByb3ZhbCByZXNwb25zZSB3aXRoIGEgc3VjY2VzcyAoZ3JlZW4pIG9yIGVycm9yIChyZWQpIGJvcmRlci5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIFBsYW5BcHByb3ZhbFJlc3BvbnNlRGlzcGxheSh7XG4gIHJlc3BvbnNlLFxuICBzZW5kZXJOYW1lLFxufTogUGxhbkFwcHJvdmFsUmVzcG9uc2VQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGlmIChyZXNwb25zZS5hcHByb3ZlZCkge1xuICAgIHJldHVybiAoXG4gICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBtYXJnaW5ZPXsxfT5cbiAgICAgICAgPEJveFxuICAgICAgICAgIGJvcmRlclN0eWxlPVwicm91bmRcIlxuICAgICAgICAgIGJvcmRlckNvbG9yPVwic3VjY2Vzc1wiXG4gICAgICAgICAgZmxleERpcmVjdGlvbj1cImNvbHVtblwiXG4gICAgICAgICAgcGFkZGluZ1g9ezF9XG4gICAgICAgICAgcGFkZGluZ1k9ezF9XG4gICAgICAgID5cbiAgICAgICAgICA8Qm94PlxuICAgICAgICAgICAgPFRleHQgY29sb3I9XCJzdWNjZXNzXCIgYm9sZD5cbiAgICAgICAgICAgICAg4pyTIFBsYW4gQXBwcm92ZWQgYnkge3NlbmRlck5hbWV9XG4gICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgICAgPEJveCBtYXJnaW5Ub3A9ezF9PlxuICAgICAgICAgICAgPFRleHQ+XG4gICAgICAgICAgICAgIFlvdSBjYW4gbm93IHByb2NlZWQgd2l0aCBpbXBsZW1lbnRhdGlvbi4gWW91ciBwbGFuIG1vZGVcbiAgICAgICAgICAgICAgcmVzdHJpY3Rpb25zIGhhdmUgYmVlbiBsaWZ0ZWQuXG4gICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgIDwvQm94PlxuICAgICAgPC9Cb3g+XG4gICAgKVxuICB9XG5cbiAgcmV0dXJuIChcbiAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBtYXJnaW5ZPXsxfT5cbiAgICAgIDxCb3hcbiAgICAgICAgYm9yZGVyU3R5bGU9XCJyb3VuZFwiXG4gICAgICAgIGJvcmRlckNvbG9yPVwiZXJyb3JcIlxuICAgICAgICBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCJcbiAgICAgICAgcGFkZGluZ1g9ezF9XG4gICAgICAgIHBhZGRpbmdZPXsxfVxuICAgICAgPlxuICAgICAgICA8Qm94PlxuICAgICAgICAgIDxUZXh0IGNvbG9yPVwiZXJyb3JcIiBib2xkPlxuICAgICAgICAgICAg4pyXIFBsYW4gUmVqZWN0ZWQgYnkge3NlbmRlck5hbWV9XG4gICAgICAgICAgPC9UZXh0PlxuICAgICAgICA8L0JveD5cbiAgICAgICAge3Jlc3BvbnNlLmZlZWRiYWNrICYmIChcbiAgICAgICAgICA8Qm94XG4gICAgICAgICAgICBtYXJnaW5Ub3A9ezF9XG4gICAgICAgICAgICBib3JkZXJTdHlsZT1cImRhc2hlZFwiXG4gICAgICAgICAgICBib3JkZXJDb2xvcj1cInN1YnRsZVwiXG4gICAgICAgICAgICBib3JkZXJMZWZ0PXtmYWxzZX1cbiAgICAgICAgICAgIGJvcmRlclJpZ2h0PXtmYWxzZX1cbiAgICAgICAgICAgIHBhZGRpbmdYPXsxfVxuICAgICAgICAgID5cbiAgICAgICAgICAgIDxUZXh0PkZlZWRiYWNrOiB7cmVzcG9uc2UuZmVlZGJhY2t9PC9UZXh0PlxuICAgICAgICAgIDwvQm94PlxuICAgICAgICApfVxuICAgICAgICA8Qm94IG1hcmdpblRvcD17MX0+XG4gICAgICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgICAgICBQbGVhc2UgcmV2aXNlIHlvdXIgcGxhbiBiYXNlZCBvbiB0aGUgZmVlZGJhY2sgYW5kIGNhbGwgRXhpdFBsYW5Nb2RlXG4gICAgICAgICAgICBhZ2Fpbi5cbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgIDwvQm94PlxuICAgICAgPC9Cb3g+XG4gICAgPC9Cb3g+XG4gIClcbn1cblxuLyoqXG4gKiBUcnkgdG8gcGFyc2UgYW5kIHJlbmRlciBhIHBsYW4gYXBwcm92YWwgbWVzc2FnZSBmcm9tIHJhdyBjb250ZW50LlxuICogUmV0dXJucyB0aGUgcmVuZGVyZWQgY29tcG9uZW50IGlmIGl0J3MgYSBwbGFuIGFwcHJvdmFsIG1lc3NhZ2UsIG51bGwgb3RoZXJ3aXNlLlxuICovXG5leHBvcnQgZnVuY3Rpb24gdHJ5UmVuZGVyUGxhbkFwcHJvdmFsTWVzc2FnZShcbiAgY29udGVudDogc3RyaW5nLFxuICBzZW5kZXJOYW1lOiBzdHJpbmcsXG4pOiBSZWFjdC5SZWFjdE5vZGUgfCBudWxsIHtcbiAgY29uc3QgcmVxdWVzdCA9IGlzUGxhbkFwcHJvdmFsUmVxdWVzdChjb250ZW50KVxuICBpZiAocmVxdWVzdCkge1xuICAgIHJldHVybiA8UGxhbkFwcHJvdmFsUmVxdWVzdERpc3BsYXkgcmVxdWVzdD17cmVxdWVzdH0gLz5cbiAgfVxuXG4gIGNvbnN0IHJlc3BvbnNlID0gaXNQbGFuQXBwcm92YWxSZXNwb25zZShjb250ZW50KVxuICBpZiAocmVzcG9uc2UpIHtcbiAgICByZXR1cm4gKFxuICAgICAgPFBsYW5BcHByb3ZhbFJlc3BvbnNlRGlzcGxheVxuICAgICAgICByZXNwb25zZT17cmVzcG9uc2V9XG4gICAgICAgIHNlbmRlck5hbWU9e3NlbmRlck5hbWV9XG4gICAgICAvPlxuICAgIClcbiAgfVxuXG4gIHJldHVybiBudWxsXG59XG5cbi8qKlxuICogR2V0IGEgYnJpZWYgc3VtbWFyeSB0ZXh0IGZvciBhIHBsYW4gYXBwcm92YWwgbWVzc2FnZS5cbiAqIFVzZWQgaW4gcGxhY2VzIGxpa2UgdGhlIGluYm94IHF1ZXVlIHdoZXJlIHdlIHdhbnQgYSBzaG9ydCBkZXNjcmlwdGlvbi5cbiAqIFJldHVybnMgbnVsbCBpZiB0aGUgY29udGVudCBpcyBub3QgYSBwbGFuIGFwcHJvdmFsIG1lc3NhZ2UuXG4gKi9cbmZ1bmN0aW9uIGdldFBsYW5BcHByb3ZhbFN1bW1hcnkoY29udGVudDogc3RyaW5nKTogc3RyaW5nIHwgbnVsbCB7XG4gIGNvbnN0IHJlcXVlc3QgPSBpc1BsYW5BcHByb3ZhbFJlcXVlc3QoY29udGVudClcbiAgaWYgKHJlcXVlc3QpIHtcbiAgICByZXR1cm4gYFtQbGFuIEFwcHJvdmFsIFJlcXVlc3QgZnJvbSAke3JlcXVlc3QuZnJvbX1dYFxuICB9XG5cbiAgY29uc3QgcmVzcG9uc2UgPSBpc1BsYW5BcHByb3ZhbFJlc3BvbnNlKGNvbnRlbnQpXG4gIGlmIChyZXNwb25zZSkge1xuICAgIGlmIChyZXNwb25zZS5hcHByb3ZlZCkge1xuICAgICAgcmV0dXJuICdbUGxhbiBBcHByb3ZlZF0gWW91IGNhbiBub3cgcHJvY2VlZCB3aXRoIGltcGxlbWVudGF0aW9uJ1xuICAgIH0gZWxzZSB7XG4gICAgICByZXR1cm4gYFtQbGFuIFJlamVjdGVkXSAke3Jlc3BvbnNlLmZlZWRiYWNrIHx8ICdQbGVhc2UgcmV2aXNlIHlvdXIgcGxhbid9YFxuICAgIH1cbiAgfVxuXG4gIHJldHVybiBudWxsXG59XG5cbi8qKlxuICogR2V0IGEgYnJpZWYgc3VtbWFyeSB0ZXh0IGZvciBhbiBpZGxlIG5vdGlmaWNhdGlvbi5cbiAqL1xuZnVuY3Rpb24gZ2V0SWRsZU5vdGlmaWNhdGlvblN1bW1hcnkobXNnOiBJZGxlTm90aWZpY2F0aW9uTWVzc2FnZSk6IHN0cmluZyB7XG4gIGNvbnN0IHBhcnRzOiBzdHJpbmdbXSA9IFsnQWdlbnQgaWRsZSddXG4gIGlmIChtc2cuY29tcGxldGVkVGFza0lkKSB7XG4gICAgY29uc3Qgc3RhdHVzID0gbXNnLmNvbXBsZXRlZFN0YXR1cyB8fCAnY29tcGxldGVkJ1xuICAgIHBhcnRzLnB1c2goYFRhc2sgJHttc2cuY29tcGxldGVkVGFza0lkfSAke3N0YXR1c31gKVxuICB9XG4gIGlmIChtc2cuc3VtbWFyeSkge1xuICAgIHBhcnRzLnB1c2goYExhc3QgRE06ICR7bXNnLnN1bW1hcnl9YClcbiAgfVxuICByZXR1cm4gcGFydHMuam9pbignIMK3ICcpXG59XG5cbi8qKlxuICogRm9ybWF0IHRlYW1tYXRlIG1lc3NhZ2UgY29udGVudCBmb3IgZGlzcGxheS5cbiAqIElmIGl0J3MgYSBzdHJ1Y3R1cmVkIG1lc3NhZ2UgKHBsYW4gYXBwcm92YWwsIHNodXRkb3duLCBvciBpZGxlKSwgcmV0dXJucyBhIGZvcm1hdHRlZCBzdW1tYXJ5LlxuICogT3RoZXJ3aXNlIHJldHVybnMgdGhlIG9yaWdpbmFsIGNvbnRlbnQuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBmb3JtYXRUZWFtbWF0ZU1lc3NhZ2VDb250ZW50KGNvbnRlbnQ6IHN0cmluZyk6IHN0cmluZyB7XG4gIGNvbnN0IHBsYW5TdW1tYXJ5ID0gZ2V0UGxhbkFwcHJvdmFsU3VtbWFyeShjb250ZW50KVxuICBpZiAocGxhblN1bW1hcnkpIHtcbiAgICByZXR1cm4gcGxhblN1bW1hcnlcbiAgfVxuXG4gIGNvbnN0IHNodXRkb3duU3VtbWFyeSA9IGdldFNodXRkb3duTWVzc2FnZVN1bW1hcnkoY29udGVudClcbiAgaWYgKHNodXRkb3duU3VtbWFyeSkge1xuICAgIHJldHVybiBzaHV0ZG93blN1bW1hcnlcbiAgfVxuXG4gIGNvbnN0IGlkbGVNc2cgPSBpc0lkbGVOb3RpZmljYXRpb24oY29udGVudClcbiAgaWYgKGlkbGVNc2cpIHtcbiAgICByZXR1cm4gZ2V0SWRsZU5vdGlmaWNhdGlvblN1bW1hcnkoaWRsZU1zZylcbiAgfVxuXG4gIGNvbnN0IHRhc2tBc3NpZ25tZW50U3VtbWFyeSA9IGdldFRhc2tBc3NpZ25tZW50U3VtbWFyeShjb250ZW50KVxuICBpZiAodGFza0Fzc2lnbm1lbnRTdW1tYXJ5KSB7XG4gICAgcmV0dXJuIHRhc2tBc3NpZ25tZW50U3VtbWFyeVxuICB9XG5cbiAgLy8gQ2hlY2sgZm9yIHRlYW1tYXRlX3Rlcm1pbmF0ZWQgbWVzc2FnZVxuICB0cnkge1xuICAgIGNvbnN0IHBhcnNlZCA9IGpzb25QYXJzZShjb250ZW50KSBhcyB7IHR5cGU/OiBzdHJpbmc7IG1lc3NhZ2U/OiBzdHJpbmcgfVxuICAgIGlmIChwYXJzZWQ/LnR5cGUgPT09ICd0ZWFtbWF0ZV90ZXJtaW5hdGVkJyAmJiBwYXJzZWQubWVzc2FnZSkge1xuICAgICAgcmV0dXJuIHBhcnNlZC5tZXNzYWdlXG4gICAgfVxuICB9IGNhdGNoIHtcbiAgICAvLyBOb3QgSlNPTlxuICB9XG5cbiAgcmV0dXJuIGNvbnRlbnRcbn1cbiJdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sS0FBS0EsS0FBSyxNQUFNLE9BQU87QUFDOUIsU0FBU0MsUUFBUSxRQUFRLDhCQUE4QjtBQUN2RCxTQUFTQyxHQUFHLEVBQUVDLElBQUksUUFBUSxjQUFjO0FBQ3hDLFNBQVNDLFNBQVMsUUFBUSwrQkFBK0I7QUFDekQsU0FDRSxLQUFLQyx1QkFBdUIsRUFDNUJDLGtCQUFrQixFQUNsQkMscUJBQXFCLEVBQ3JCQyxzQkFBc0IsRUFDdEIsS0FBS0MsMEJBQTBCLEVBQy9CLEtBQUtDLDJCQUEyQixRQUMzQixnQ0FBZ0M7QUFDdkMsU0FBU0MseUJBQXlCLFFBQVEsc0JBQXNCO0FBQ2hFLFNBQVNDLHdCQUF3QixRQUFRLDRCQUE0QjtBQUVyRSxLQUFLQyx3QkFBd0IsR0FBRztFQUM5QkMsT0FBTyxFQUFFTCwwQkFBMEI7QUFDckMsQ0FBQzs7QUFFRDtBQUNBO0FBQ0E7QUFDQTtBQUNBLE9BQU8sU0FBQU0sMkJBQUFDLEVBQUE7RUFBQSxNQUFBQyxDQUFBLEdBQUFDLEVBQUE7RUFBb0M7SUFBQUo7RUFBQSxJQUFBRSxFQUVoQjtFQUFBLElBQUFHLEVBQUE7RUFBQSxJQUFBRixDQUFBLFFBQUFILE9BQUEsQ0FBQU0sSUFBQTtJQVNuQkQsRUFBQSxJQUFDLEdBQUcsQ0FBZSxZQUFDLENBQUQsR0FBQyxDQUNsQixDQUFDLElBQUksQ0FBTyxLQUFVLENBQVYsVUFBVSxDQUFDLElBQUksQ0FBSixLQUFHLENBQUMsQ0FBQywyQkFDRSxDQUFBTCxPQUFPLENBQUFNLElBQUksQ0FDekMsRUFGQyxJQUFJLENBR1AsRUFKQyxHQUFHLENBSUU7SUFBQUgsQ0FBQSxNQUFBSCxPQUFBLENBQUFNLElBQUE7SUFBQUgsQ0FBQSxN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dUIsQ0FBQyxFQUFFLE1BQU0sQ0FBQztFQUN4RSxNQUFNbUMsS0FBSyxFQUFFLE1BQU0sRUFBRSxHQUFHLENBQUMsWUFBWSxDQUFDO0VBQ3RDLElBQUlELEdBQUcsQ0FBQ0UsZUFBZSxFQUFFO0lBQ3ZCLE1BQU1DLE1BQU0sR0FBR0gsR0FBRyxDQUFDSSxlQUFlLElBQUksV0FBVztJQUNqREgsS0FBSyxDQUFDSSxJQUFJLENBQUMsUUFBUUwsR0FBRyxDQUFDRSxlQUFlLElBQUlDLE1BQU0sRUFBRSxDQUFDO0VBQ3JEO0VBQ0EsSUFBSUgsR0FBRyxDQUFDTSxPQUFPLEVBQUU7SUFDZkwsS0FBSyxDQUFDSSxJQUFJLENBQUMsWUFBWUwsR0FBRyxDQUFDTSxPQUFPLEVBQUUsQ0FBQztFQUN2QztFQUNBLE9BQU9MLEtBQUssQ0FBQ00sSUFBSSxDQUFDLEtBQUssQ0FBQztBQUMxQjs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsT0FBTyxTQUFTQyw0QkFBNEJBLENBQUNaLE9BQU8sRUFBRSxNQUFNLENBQUMsRUFBRSxNQUFNLENBQUM7RUFDcEUsTUFBTWEsV0FBVyxHQUFHWCxzQkFBc0IsQ0FBQ0YsT0FBTyxDQUFDO0VBQ25ELElBQUlhLFdBQVcsRUFBRTtJQUNmLE9BQU9BLFdBQVc7RUFDcEI7RUFFQSxNQUFNQyxlQUFlLEdBQUd0Qyx5QkFBeUIsQ0FBQ3dCLE9BQU8sQ0FBQztFQUMxRCxJQUFJYyxlQUFlLEVBQUU7SUFDbkIsT0FBT0EsZUFBZTtFQUN4QjtFQUVBLE1BQU1DLE9BQU8sR0FBRzVDLGtCQUFrQixDQUFDNkIsT0FBTyxDQUFDO0VBQzNDLElBQUllLE9BQU8sRUFBRTtJQUNYLE9BQU9aLDBCQUEwQixDQUFDWSxPQUFPLENBQUM7RUFDNUM7RUFFQSxNQUFNQyxxQkFBcUIsR0FBR3ZDLHdCQUF3QixDQUFDdUIsT0FBTyxDQUFDO0VBQy9ELElBQUlnQixxQkFBcUIsRUFBRTtJQUN6QixPQUFPQSxxQkFBcUI7RUFDOUI7O0VBRUE7RUFDQSxJQUFJO0lBQ0YsTUFBTUMsTUFBTSxHQUFHaEQsU0FBUyxDQUFDK0IsT0FBTyxDQUFDLElBQUk7TUFBRWtCLElBQUksQ0FBQyxFQUFFLE1BQU07TUFBRUMsT0FBTyxDQUFDLEVBQUUsTUFBTTtJQUFDLENBQUM7SUFDeEUsSUFBSUYsTUFBTSxFQUFFQyxJQUFJLEtBQUsscUJBQXFCLElBQUlELE1BQU0sQ0FBQ0UsT0FBTyxFQUFFO01BQzVELE9BQU9GLE1BQU0sQ0FBQ0UsT0FBTztJQUN2QjtFQUNGLENBQUMsQ0FBQyxNQUFNO0lBQ047RUFBQTtFQUdGLE9BQU9uQixPQUFPO0FBQ2hCIiwiaWdub3JlTGlzdCI6W119