claude-code/components/mcp/MCPListPanel.tsx

504 lines
57 KiB
TypeScript
Raw Normal View History

import { c as _c } from "react/compiler-runtime";
import figures from 'figures';
import React, { useCallback, useState } from 'react';
import type { CommandResultDisplay } from '../../commands.js';
import { Box, color, Link, Text, useTheme } from '../../ink.js';
import { useKeybindings } from '../../keybindings/useKeybinding.js';
import type { ConfigScope } from '../../services/mcp/types.js';
import { describeMcpConfigFilePath } from '../../services/mcp/utils.js';
import { isDebugMode } from '../../utils/debug.js';
import { plural } from '../../utils/stringUtils.js';
import { ConfigurableShortcutHint } from '../ConfigurableShortcutHint.js';
import { Byline } from '../design-system/Byline.js';
import { Dialog } from '../design-system/Dialog.js';
import { KeyboardShortcutHint } from '../design-system/KeyboardShortcutHint.js';
import { McpParsingWarnings } from './McpParsingWarnings.js';
import type { AgentMcpServerInfo, ServerInfo } from './types.js';
type Props = {
servers: ServerInfo[];
agentServers?: AgentMcpServerInfo[];
onSelectServer: (server: ServerInfo) => void;
onSelectAgentServer?: (agentServer: AgentMcpServerInfo) => void;
onComplete: (result?: string, options?: {
display?: CommandResultDisplay;
}) => void;
defaultTab?: string;
};
type SelectableItem = {
type: 'server';
server: ServerInfo;
} | {
type: 'agent-server';
agentServer: AgentMcpServerInfo;
};
// Define scope order for display (constant, outside component)
// 'dynamic' (built-in) is rendered separately at the end
const SCOPE_ORDER: ConfigScope[] = ['project', 'local', 'user', 'enterprise'];
// Get scope heading parts (label is bold, path is grey)
function getScopeHeading(scope: ConfigScope): {
label: string;
path?: string;
} {
switch (scope) {
case 'project':
return {
label: 'Project MCPs',
path: describeMcpConfigFilePath(scope)
};
case 'user':
return {
label: 'User MCPs',
path: describeMcpConfigFilePath(scope)
};
case 'local':
return {
label: 'Local MCPs',
path: describeMcpConfigFilePath(scope)
};
case 'enterprise':
return {
label: 'Enterprise MCPs'
};
case 'dynamic':
return {
label: 'Built-in MCPs',
path: 'always available'
};
default:
return {
label: scope
};
}
}
// Group servers by scope
function groupServersByScope(serverList: ServerInfo[]): Map<ConfigScope, ServerInfo[]> {
const groups = new Map<ConfigScope, ServerInfo[]>();
for (const server of serverList) {
const scope = server.scope;
if (!groups.has(scope)) {
groups.set(scope, []);
}
groups.get(scope)!.push(server);
}
// Sort servers within each group alphabetically
for (const [, groupServers] of groups) {
groupServers.sort((a, b) => a.name.localeCompare(b.name));
}
return groups;
}
export function MCPListPanel(t0) {
const $ = _c(78);
const {
servers,
agentServers: t1,
onSelectServer,
onSelectAgentServer,
onComplete
} = t0;
let t2;
if ($[0] !== t1) {
t2 = t1 === undefined ? [] : t1;
$[0] = t1;
$[1] = t2;
} else {
t2 = $[1];
}
const agentServers = t2;
const [theme] = useTheme();
const [selectedIndex, setSelectedIndex] = useState(0);
let t3;
if ($[2] !== servers) {
const regularServers = servers.filter(_temp);
t3 = groupServersByScope(regularServers);
$[2] = servers;
$[3] = t3;
} else {
t3 = $[3];
}
const serversByScope = t3;
let t4;
if ($[4] !== servers) {
t4 = servers.filter(_temp2).sort(_temp3);
$[4] = servers;
$[5] = t4;
} else {
t4 = $[5];
}
const claudeAiServers = t4;
let t5;
if ($[6] !== serversByScope) {
t5 = (serversByScope.get("dynamic") ?? []).sort(_temp4);
$[6] = serversByScope;
$[7] = t5;
} else {
t5 = $[7];
}
const dynamicServers = t5;
let t6;
if ($[8] === Symbol.for("react.memo_cache_sentinel")) {
t6 = getScopeHeading("dynamic");
$[8] = t6;
} else {
t6 = $[8];
}
const dynamicHeading = t6;
let items;
if ($[9] !== agentServers || $[10] !== claudeAiServers || $[11] !== dynamicServers || $[12] !== serversByScope) {
items = [];
for (const scope of SCOPE_ORDER) {
const scopeServers = serversByScope.get(scope) ?? [];
for (const server of scopeServers) {
items.push({
type: "server",
server
});
}
}
for (const server_0 of claudeAiServers) {
items.push({
type: "server",
server: server_0
});
}
for (const agentServer of agentServers) {
items.push({
type: "agent-server",
agentServer
});
}
for (const server_1 of dynamicServers) {
items.push({
type: "server",
server: server_1
});
}
$[9] = agentServers;
$[10] = claudeAiServers;
$[11] = dynamicServers;
$[12] = serversByScope;
$[13] = items;
} else {
items = $[13];
}
const selectableItems = items;
let t7;
if ($[14] !== onComplete) {
t7 = () => {
onComplete("MCP dialog dismissed", {
display: "system"
});
};
$[14] = onComplete;
$[15] = t7;
} else {
t7 = $[15];
}
const handleCancel = t7;
let t8;
if ($[16] !== onSelectAgentServer || $[17] !== onSelectServer || $[18] !== selectableItems || $[19] !== selectedIndex) {
t8 = () => {
const item = selectableItems[selectedIndex];
if (!item) {
return;
}
if (item.type === "server") {
onSelectServer(item.server);
} else {
if (item.type === "agent-server" && onSelectAgentServer) {
onSelectAgentServer(item.agentServer);
}
}
};
$[16] = onSelectAgentServer;
$[17] = onSelectServer;
$[18] = selectableItems;
$[19] = selectedIndex;
$[20] = t8;
} else {
t8 = $[20];
}
const handleSelect = t8;
let t10;
let t9;
if ($[21] !== selectableItems) {
t9 = () => setSelectedIndex(prev => prev === 0 ? selectableItems.length - 1 : prev - 1);
t10 = () => setSelectedIndex(prev_0 => prev_0 === selectableItems.length - 1 ? 0 : prev_0 + 1);
$[21] = selectableItems;
$[22] = t10;
$[23] = t9;
} else {
t10 = $[22];
t9 = $[23];
}
let t11;
if ($[24] !== handleCancel || $[25] !== handleSelect || $[26] !== t10 || $[27] !== t9) {
t11 = {
"confirm:previous": t9,
"confirm:next": t10,
"confirm:yes": handleSelect,
"confirm:no": handleCancel
};
$[24] = handleCancel;
$[25] = handleSelect;
$[26] = t10;
$[27] = t9;
$[28] = t11;
} else {
t11 = $[28];
}
let t12;
if ($[29] === Symbol.for("react.memo_cache_sentinel")) {
t12 = {
context: "Confirmation"
};
$[29] = t12;
} else {
t12 = $[29];
}
useKeybindings(t11, t12);
let t13;
if ($[30] !== selectableItems) {
t13 = server_2 => selectableItems.findIndex(item_0 => item_0.type === "server" && item_0.server === server_2);
$[30] = selectableItems;
$[31] = t13;
} else {
t13 = $[31];
}
const getServerIndex = t13;
let t14;
if ($[32] !== selectableItems) {
t14 = agentServer_0 => selectableItems.findIndex(item_1 => item_1.type === "agent-server" && item_1.agentServer === agentServer_0);
$[32] = selectableItems;
$[33] = t14;
} else {
t14 = $[33];
}
const getAgentServerIndex = t14;
let t15;
if ($[34] === Symbol.for("react.memo_cache_sentinel")) {
t15 = isDebugMode();
$[34] = t15;
} else {
t15 = $[34];
}
const debugMode = t15;
let t16;
if ($[35] !== servers) {
t16 = servers.some(_temp5);
$[35] = servers;
$[36] = t16;
} else {
t16 = $[36];
}
const hasFailedClients = t16;
if (servers.length === 0 && agentServers.length === 0) {
return null;
}
let t17;
if ($[37] !== getServerIndex || $[38] !== selectedIndex || $[39] !== theme) {
t17 = server_3 => {
const index = getServerIndex(server_3);
const isSelected = selectedIndex === index;
let statusIcon;
let statusText;
if (server_3.client.type === "disabled") {
statusIcon = color("inactive", theme)(figures.radioOff);
statusText = "disabled";
} else {
if (server_3.client.type === "connected") {
statusIcon = color("success", theme)(figures.tick);
statusText = "connected";
} else {
if (server_3.client.type === "pending") {
statusIcon = color("inactive", theme)(figures.radioOff);
const {
reconnectAttempt,
maxReconnectAttempts
} = server_3.client;
if (reconnectAttempt && maxReconnectAttempts) {
statusText = `reconnecting (${reconnectAttempt}/${maxReconnectAttempts})…`;
} else {
statusText = "connecting\u2026";
}
} else {
if (server_3.client.type === "needs-auth") {
statusIcon = color("warning", theme)(figures.triangleUpOutline);
statusText = "needs authentication";
} else {
statusIcon = color("error", theme)(figures.cross);
statusText = "failed";
}
}
}
}
return <Box key={`${server_3.name}-${index}`}><Text color={isSelected ? "suggestion" : undefined}>{isSelected ? `${figures.pointer} ` : " "}</Text><Text color={isSelected ? "suggestion" : undefined}>{server_3.name}</Text><Text dimColor={!isSelected}> · {statusIcon} </Text><Text dimColor={!isSelected}>{statusText}</Text></Box>;
};
$[37] = getServerIndex;
$[38] = selectedIndex;
$[39] = theme;
$[40] = t17;
} else {
t17 = $[40];
}
const renderServerItem = t17;
let t18;
if ($[41] !== getAgentServerIndex || $[42] !== selectedIndex || $[43] !== theme) {
t18 = agentServer_1 => {
const index_0 = getAgentServerIndex(agentServer_1);
const isSelected_0 = selectedIndex === index_0;
const statusIcon_0 = agentServer_1.needsAuth ? color("warning", theme)(figures.triangleUpOutline) : color("inactive", theme)(figures.radioOff);
const statusText_0 = agentServer_1.needsAuth ? "may need auth" : "agent-only";
return <Box key={`agent-${agentServer_1.name}-${index_0}`}><Text color={isSelected_0 ? "suggestion" : undefined}>{isSelected_0 ? `${figures.pointer} ` : " "}</Text><Text color={isSelected_0 ? "suggestion" : undefined}>{agentServer_1.name}</Text><Text dimColor={!isSelected_0}> · {statusIcon_0} </Text><Text dimColor={!isSelected_0}>{statusText_0}</Text></Box>;
};
$[41] = getAgentServerIndex;
$[42] = selectedIndex;
$[43] = theme;
$[44] = t18;
} else {
t18 = $[44];
}
const renderAgentServerItem = t18;
const totalServers = servers.length + agentServers.length;
let t19;
if ($[45] === Symbol.for("react.memo_cache_sentinel")) {
t19 = <McpParsingWarnings />;
$[45] = t19;
} else {
t19 = $[45];
}
let t20;
if ($[46] !== totalServers) {
t20 = plural(totalServers, "server");
$[46] = totalServers;
$[47] = t20;
} else {
t20 = $[47];
}
const t21 = `${totalServers} ${t20}`;
let t22;
if ($[48] !== renderServerItem || $[49] !== serversByScope) {
t22 = SCOPE_ORDER.map(scope_0 => {
const scopeServers_0 = serversByScope.get(scope_0);
if (!scopeServers_0 || scopeServers_0.length === 0) {
return null;
}
const heading = getScopeHeading(scope_0);
return <Box key={scope_0} flexDirection="column" marginBottom={1}><Box paddingLeft={2}><Text bold={true}>{heading.label}</Text>{heading.path && <Text dimColor={true}> ({heading.path})</Text>}</Box>{scopeServers_0.map(server_4 => renderServerItem(server_4))}</Box>;
});
$[48] = renderServerItem;
$[49] = serversByScope;
$[50] = t22;
} else {
t22 = $[50];
}
let t23;
if ($[51] !== claudeAiServers || $[52] !== renderServerItem) {
t23 = claudeAiServers.length > 0 && <Box flexDirection="column" marginBottom={1}><Box paddingLeft={2}><Text bold={true}>claude.ai</Text></Box>{claudeAiServers.map(server_5 => renderServerItem(server_5))}</Box>;
$[51] = claudeAiServers;
$[52] = renderServerItem;
$[53] = t23;
} else {
t23 = $[53];
}
let t24;
if ($[54] !== agentServers || $[55] !== renderAgentServerItem) {
t24 = agentServers.length > 0 && <Box flexDirection="column" marginBottom={1}><Box paddingLeft={2}><Text bold={true}>Agent MCPs</Text></Box>{[...new Set(agentServers.flatMap(_temp6))].map(agentName => <Box key={agentName} flexDirection="column" marginTop={1}><Box paddingLeft={2}><Text dimColor={true}>@{agentName}</Text></Box>{agentServers.filter(s_3 => s_3.sourceAgents.includes(agentName)).map(agentServer_2 => renderAgentServerItem(agentServer_2))}</Box>)}</Box>;
$[54] = agentServers;
$[55] = renderAgentServerItem;
$[56] = t24;
} else {
t24 = $[56];
}
let t25;
if ($[57] !== dynamicServers || $[58] !== renderServerItem) {
t25 = dynamicServers.length > 0 && <Box flexDirection="column" marginBottom={1}><Box paddingLeft={2}><Text bold={true}>{dynamicHeading.label}</Text>{dynamicHeading.path && <Text dimColor={true}> ({dynamicHeading.path})</Text>}</Box>{dynamicServers.map(server_6 => renderServerItem(server_6))}</Box>;
$[57] = dynamicServers;
$[58] = renderServerItem;
$[59] = t25;
} else {
t25 = $[59];
}
let t26;
if ($[60] !== hasFailedClients) {
t26 = hasFailedClients && <Text dimColor={true}>{debugMode ? "\u203B Error logs shown inline with --debug" : "\u203B Run claude --debug to see error logs"}</Text>;
$[60] = hasFailedClients;
$[61] = t26;
} else {
t26 = $[61];
}
let t27;
if ($[62] === Symbol.for("react.memo_cache_sentinel")) {
t27 = <Text dimColor={true}><Link url="https://code.claude.com/docs/en/mcp">https://code.claude.com/docs/en/mcp</Link>{" "}for help</Text>;
$[62] = t27;
} else {
t27 = $[62];
}
let t28;
if ($[63] !== t26) {
t28 = <Box flexDirection="column">{t26}{t27}</Box>;
$[63] = t26;
$[64] = t28;
} else {
t28 = $[64];
}
let t29;
if ($[65] !== t22 || $[66] !== t23 || $[67] !== t24 || $[68] !== t25 || $[69] !== t28) {
t29 = <Box flexDirection="column">{t22}{t23}{t24}{t25}{t28}</Box>;
$[65] = t22;
$[66] = t23;
$[67] = t24;
$[68] = t25;
$[69] = t28;
$[70] = t29;
} else {
t29 = $[70];
}
let t30;
if ($[71] !== handleCancel || $[72] !== t21 || $[73] !== t29) {
t30 = <Dialog title="Manage MCP servers" subtitle={t21} onCancel={handleCancel} hideInputGuide={true}>{t29}</Dialog>;
$[71] = handleCancel;
$[72] = t21;
$[73] = t29;
$[74] = t30;
} else {
t30 = $[74];
}
let t31;
if ($[75] === Symbol.for("react.memo_cache_sentinel")) {
t31 = <Box paddingX={1}><Text dimColor={true} italic={true}><Byline><KeyboardShortcutHint shortcut={"\u2191\u2193"} action="navigate" /><KeyboardShortcutHint shortcut="Enter" action="confirm" /><ConfigurableShortcutHint action="confirm:no" context="Confirmation" fallback="Esc" description="cancel" /></Byline></Text></Box>;
$[75] = t31;
} else {
t31 = $[75];
}
let t32;
if ($[76] !== t30) {
t32 = <Box flexDirection="column">{t19}{t30}{t31}</Box>;
$[76] = t30;
$[77] = t32;
} else {
t32 = $[77];
}
return t32;
}
function _temp6(s_2) {
return s_2.sourceAgents;
}
function _temp5(s_1) {
return s_1.client.type === "failed";
}
function _temp4(a_0, b_0) {
return a_0.name.localeCompare(b_0.name);
}
function _temp3(a, b) {
return a.name.localeCompare(b.name);
}
function _temp2(s_0) {
return s_0.client.config.type === "claudeai-proxy";
}
function _temp(s) {
return s.client.config.type !== "claudeai-proxy";
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmaWd1cmVzIiwiUmVhY3QiLCJ1c2VDYWxsYmFjayIsInVzZVN0YXRlIiwiQ29tbWFuZFJlc3VsdERpc3BsYXkiLCJCb3giLCJjb2xvciIsIkxpbmsiLCJUZXh0IiwidXNlVGhlbWUiLCJ1c2VLZXliaW5kaW5ncyIsIkNvbmZpZ1Njb3BlIiwiZGVzY3JpYmVNY3BDb25maWdGaWxlUGF0aCIsImlzRGVidWdNb2RlIiwicGx1cmFsIiwiQ29uZmlndXJhYmxlU2hvcnRjdXRIaW50IiwiQnlsaW5lIiwiRGlhbG9nIiwiS2V5Ym9hcmRTaG9ydGN1dEhpbnQiLCJNY3BQYXJzaW5nV2FybmluZ3MiLCJBZ2VudE1jcFNlcnZlckluZm8iLCJTZXJ2ZXJJbmZvIiwiUHJvcHMiLCJzZXJ2ZXJzIiwiYWdlbnRTZXJ2ZXJzIiwib25TZWxlY3RTZXJ2ZXIiLCJzZXJ2ZXIiLCJvblNlbGVjdEFnZW50U2VydmVyIiwiYWdlbnRTZXJ2ZXIiLCJvbkNvbXBsZXRlIiwicmVzdWx0Iiwib3B0aW9ucyIsImRpc3BsYXkiLCJkZWZhdWx0VGFiIiwiU2VsZWN0YWJsZUl0ZW0iLCJ0eXBlIiwiU0NPUEVfT1JERVIiLCJnZXRTY29wZUhlYWRpbmciLCJzY29wZSIsImxhYmVsIiwicGF0aCIsImdyb3VwU2VydmVyc0J5U2NvcGUiLCJzZXJ2ZXJMaXN0IiwiTWFwIiwiZ3JvdXBzIiwiaGFzIiwic2V0IiwiZ2V0IiwicHVzaCIsImdyb3VwU2VydmVycyIsInNvcnQiLCJhIiwiYiIsIm5hbWUiLCJsb2NhbGVDb21wYXJlIiwiTUNQTGlzdFBhbmVsIiwidDAiLCIkIiwiX2MiLCJ0MSIsInQyIiwidW5kZWZpbmVkIiwidGhlbWUiLCJzZWxlY3RlZEluZGV4Iiwic2V0U2VsZWN0ZWRJbmRleCIsInQzIiwicmVndWxhclNlcnZlcnMiLCJmaWx0ZXIiLCJfdGVtcCIsInNlcnZlcnNCeVNjb3BlIiwidDQiLCJfdGVtcDIiLCJfdGVtcDMiLCJjbGF1ZGVBaVNlcnZlcnMiLCJ0NSIsIl90ZW1wNCIsImR5bmFtaWNTZXJ2ZXJzIiwidDYiLCJTeW1ib2wiLCJmb3IiLCJkeW5hbWljSGVhZGluZyIsIml0ZW1zIiwic2NvcGVTZXJ2ZXJzIiwic2VydmVyXzAiLCJzZXJ2ZXJfMSIsInNlbGVjdGFibGVJdGVtcyIsInQ3IiwiaGFuZGxlQ2FuY2VsIiwidDgiLCJpdGVtIiwiaGFuZGxlU2VsZWN0IiwidDEwIiwidDkiLCJwcmV2IiwibGVuZ3RoIiwicHJldl8wIiwidDExIiwidDEyIiwiY29udGV4dCIsInQxMyIsInNlcnZlcl8yIiwiZmluZEluZGV4IiwiaXRlbV8wIiwiZ2V0U2VydmVySW5kZXgiLCJ0MTQiLCJhZ2VudFNlcnZlcl8wIiwiaXRlbV8xIiwiZ2V0QWdlbnRTZXJ2ZXJJbmRleCIsInQxNSIsImRlYnVnTW9kZSIsInQxNiIsInNvbWUiLCJfdGVtcDUiLCJoYXNGYWlsZWRDbGllbnRzIiwidDE3Iiwic2VydmVyXzMiLCJpbmRleCIsImlzU2VsZWN0ZWQiLCJzdGF0dXNJY29uIiwic3RhdHVzVGV4dCIsImNsaWVudCIsInJhZGlvT2ZmIiwidGljayIsInJlY29ubmVjdEF0dGVtcHQiLCJtYXhSZWNvbm5lY3RBdHRlbXB0cyIsInRyaWFuZ2xlVXBPdXRsaW5lIiwiY3Jvc3MiLCJwb2ludGVyIiwicmVuZGVyU2VydmVySXRlbSIsInQxOCIsImFnZW50U2VydmVyXzEiLCJpbmRleF8wIiwiaXNTZWxlY3RlZF8wIiwic3RhdHVzSWNvbl8wIiwibmVlZHNBdXRoIiwic3RhdHVzVGV4dF8wIiwicmVuZGVyQWdlbnRTZXJ2ZXJJdGVtIiwidG90YWxTZXJ2ZXJzIiwidDE5IiwidDIwIiwidDIxIiwidDIyIiwibWFwIiwic2NvcGVfMCIsInNjb3BlU2VydmVyc18wIiwiaGVhZGluZyIsInNlcnZlcl80IiwidDIzIiwic2VydmVyXzUiLCJ0MjQiLCJTZXQiLCJmbGF0TWFwIiwiX3RlbXA2IiwiYWdlbnROYW1lIiwic18zIiwicyIsInNvdXJjZUFnZW50cyIsImluY2x1ZGVzIiwiYWdlbnRTZXJ2ZXJfMiIsInQyNSIsInNlcnZlcl82IiwidDI2IiwidDI3IiwidDI4IiwidDI5IiwidDMwIiwidDMxIiwidDMyIiwic18yIiwic18xIiwiYV8wIiwiYl8wIiwic18wIiwiY29uZmlnIl0sInNvdXJjZXMiOlsiTUNQTGlzdFBhbmVsLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgZmlndXJlcyBmcm9tICdmaWd1cmVzJ1xuaW1wb3J0IFJlYWN0LCB7IHVzZUNhbGxiYWNrLCB1c2VTdGF0ZSB9IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHR5cGUgeyBDb21tYW5kUmVzdWx0RGlzcGxheSB9IGZyb20gJy4uLy4uL2NvbW1hbmRzLmpzJ1xuaW1wb3J0IHsgQm94LCBjb2xvciwgTGluaywgVGV4dCwgdXNlVGhlbWUgfSBmcm9tICcuLi8uLi9pbmsuanMnXG5pbXBvcnQgeyB1c2VLZXliaW5kaW5ncyB9IGZyb20gJy4uLy4uL2tleWJpbmRpbmdzL3VzZUtleWJpbmRpbmcuanMnXG5pbXBvcnQgdHlwZSB7IENvbmZpZ1Njb3BlIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvbWNwL3R5cGVzLmpzJ1xuaW1wb3J0IHsgZGVzY3JpYmVNY3BDb25maWdGaWxlUGF0aCB9IGZyb20gJy4uLy4uL3NlcnZpY2VzL21jcC91dGlscy5qcydcbmltcG9ydCB7IGlzRGVidWdNb2RlIH0gZnJvbSAnLi4vLi4vdXRpbHMvZGVidWcuanMnXG5pbXBvcnQgeyBwbHVyYWwgfSBmcm9tICcuLi8uLi91dGlscy9zdHJpbmdVdGlscy5qcydcbmltcG9ydCB7IENvbmZpZ3VyYWJsZVNob3J0Y3V0SGludCB9IGZyb20gJy4uL0NvbmZpZ3VyYWJsZVNob3J0Y3V0SGludC5qcydcbmltcG9ydCB7IEJ5bGluZSB9IGZyb20gJy4uL2Rlc2lnbi1zeXN0ZW0vQnlsaW5lLmpzJ1xuaW1wb3J0IHsgRGlhbG9nIH0gZnJvbSAnLi4vZGVzaWduLXN5c3RlbS9EaWFsb2cuanMnXG5pbXBvcnQgeyBLZXlib2FyZFNob3J0Y3V0SGludCB9IGZyb20gJy4uL2Rlc2lnbi1zeXN0ZW0vS2V5Ym9hcmRTaG9ydGN1dEhpbnQuanMnXG5pbXBvcnQgeyBNY3BQYXJzaW5nV2FybmluZ3MgfSBmcm9tICcuL01jcFBhcnNpbmdXYXJuaW5ncy5qcydcbmltcG9ydCB0eXBlIHsgQWdlbnRNY3BTZXJ2ZXJJbmZvLCBTZXJ2ZXJJbmZvIH0gZnJvbSAnLi90eXBlcy5qcydcblxudHlwZSBQcm9wcyA9IHtcbiAgc2VydmVyczogU2VydmVySW5mb1tdXG4gIGFnZW50U2VydmVycz86IEFnZW50TWNwU2VydmVySW5mb1tdXG4gIG9uU2VsZWN0U2VydmVyOiAoc2VydmVyOiBTZXJ2ZXJJbmZvKSA9PiB2b2lkXG4gIG9uU2VsZWN0QWdlbnRTZXJ2ZXI/OiAoYWdlbnRTZXJ2ZXI6IEF