source dump of claude code
at main 183 lines 27 kB view raw
1import figures from 'figures'; 2import React, { useCallback, useEffect, useRef, useState } from 'react'; 3import type { CommandResultDisplay } from '../../commands.js'; 4import { Box, color, Link, Text, useTheme } from '../../ink.js'; 5import { useKeybinding } from '../../keybindings/useKeybinding.js'; 6import { AuthenticationCancelledError, performMCPOAuthFlow } from '../../services/mcp/auth.js'; 7import { capitalize } from '../../utils/stringUtils.js'; 8import { ConfigurableShortcutHint } from '../ConfigurableShortcutHint.js'; 9import { Select } from '../CustomSelect/index.js'; 10import { Byline } from '../design-system/Byline.js'; 11import { Dialog } from '../design-system/Dialog.js'; 12import { KeyboardShortcutHint } from '../design-system/KeyboardShortcutHint.js'; 13import { Spinner } from '../Spinner.js'; 14import type { AgentMcpServerInfo } from './types.js'; 15type Props = { 16 agentServer: AgentMcpServerInfo; 17 onCancel: () => void; 18 onComplete?: (result?: string, options?: { 19 display?: CommandResultDisplay; 20 }) => void; 21}; 22 23/** 24 * Menu for agent-specific MCP servers. 25 * These servers are defined in agent frontmatter and only connect when the agent runs. 26 * For HTTP/SSE servers, this allows pre-authentication before using the agent. 27 */ 28export function MCPAgentServerMenu({ 29 agentServer, 30 onCancel, 31 onComplete 32}: Props): React.ReactNode { 33 const [theme] = useTheme(); 34 const [isAuthenticating, setIsAuthenticating] = useState(false); 35 const [error, setError] = useState<string | null>(null); 36 const [authorizationUrl, setAuthorizationUrl] = useState<string | null>(null); 37 const authAbortControllerRef = useRef<AbortController | null>(null); 38 39 // Abort OAuth flow on unmount so the callback server is closed even if a 40 // parent component's Esc handler navigates away before ours fires. 41 useEffect(() => () => authAbortControllerRef.current?.abort(), []); 42 43 // Handle ESC to cancel authentication flow 44 const handleEscCancel = useCallback(() => { 45 if (isAuthenticating) { 46 authAbortControllerRef.current?.abort(); 47 authAbortControllerRef.current = null; 48 setIsAuthenticating(false); 49 setAuthorizationUrl(null); 50 } 51 }, [isAuthenticating]); 52 useKeybinding('confirm:no', handleEscCancel, { 53 context: 'Confirmation', 54 isActive: isAuthenticating 55 }); 56 const handleAuthenticate = useCallback(async () => { 57 if (!agentServer.needsAuth || !agentServer.url) { 58 return; 59 } 60 setIsAuthenticating(true); 61 setError(null); 62 const controller = new AbortController(); 63 authAbortControllerRef.current = controller; 64 try { 65 // Create a temporary config for OAuth 66 const tempConfig = { 67 type: agentServer.transport as 'http' | 'sse', 68 url: agentServer.url 69 }; 70 await performMCPOAuthFlow(agentServer.name, tempConfig, setAuthorizationUrl, controller.signal); 71 onComplete?.(`Authentication successful for ${agentServer.name}. The server will connect when the agent runs.`); 72 } catch (err) { 73 // Don't show error if it was a cancellation 74 if (err instanceof Error && !(err instanceof AuthenticationCancelledError)) { 75 setError(err.message); 76 } 77 } finally { 78 setIsAuthenticating(false); 79 authAbortControllerRef.current = null; 80 } 81 }, [agentServer, onComplete]); 82 const capitalizedServerName = capitalize(String(agentServer.name)); 83 if (isAuthenticating) { 84 return <Box flexDirection="column" gap={1} padding={1}> 85 <Text color="claude">Authenticating with {agentServer.name}</Text> 86 <Box> 87 <Spinner /> 88 <Text> A browser window will open for authentication</Text> 89 </Box> 90 {authorizationUrl && <Box flexDirection="column"> 91 <Text dimColor> 92 If your browser doesn&apos;t open automatically, copy this URL 93 manually: 94 </Text> 95 <Link url={authorizationUrl} /> 96 </Box>} 97 <Box marginLeft={3}> 98 <Text dimColor> 99 Return here after authenticating in your browser.{' '} 100 <ConfigurableShortcutHint action="confirm:no" context="Confirmation" fallback="Esc" description="go back" /> 101 </Text> 102 </Box> 103 </Box>; 104 } 105 const menuOptions = []; 106 107 // Only show authenticate option for HTTP/SSE servers 108 if (agentServer.needsAuth) { 109 menuOptions.push({ 110 label: agentServer.isAuthenticated ? 'Re-authenticate' : 'Authenticate', 111 value: 'auth' 112 }); 113 } 114 menuOptions.push({ 115 label: 'Back', 116 value: 'back' 117 }); 118 return <Dialog title={`${capitalizedServerName} MCP Server`} subtitle="agent-only" onCancel={onCancel} inputGuide={exitState => exitState.pending ? <Text>Press {exitState.keyName} again to exit</Text> : <Byline> 119 <KeyboardShortcutHint shortcut="↑↓" action="navigate" /> 120 <KeyboardShortcutHint shortcut="Enter" action="confirm" /> 121 <ConfigurableShortcutHint action="confirm:no" context="Confirmation" fallback="Esc" description="go back" /> 122 </Byline>}> 123 <Box flexDirection="column" gap={0}> 124 <Box> 125 <Text bold>Type: </Text> 126 <Text dimColor>{agentServer.transport}</Text> 127 </Box> 128 129 {agentServer.url && <Box> 130 <Text bold>URL: </Text> 131 <Text dimColor>{agentServer.url}</Text> 132 </Box>} 133 134 {agentServer.command && <Box> 135 <Text bold>Command: </Text> 136 <Text dimColor>{agentServer.command}</Text> 137 </Box>} 138 139 <Box> 140 <Text bold>Used by: </Text> 141 <Text dimColor>{agentServer.sourceAgents.join(', ')}</Text> 142 </Box> 143 144 <Box marginTop={1}> 145 <Text bold>Status: </Text> 146 <Text> 147 {color('inactive', theme)(figures.radioOff)} not connected 148 (agent-only) 149 </Text> 150 </Box> 151 152 {agentServer.needsAuth && <Box> 153 <Text bold>Auth: </Text> 154 {agentServer.isAuthenticated ? <Text>{color('success', theme)(figures.tick)} authenticated</Text> : <Text> 155 {color('warning', theme)(figures.triangleUpOutline)} may need 156 authentication 157 </Text>} 158 </Box>} 159 </Box> 160 161 <Box> 162 <Text dimColor>This server connects only when running the agent.</Text> 163 </Box> 164 165 {error && <Box> 166 <Text color="error">Error: {error}</Text> 167 </Box>} 168 169 <Box> 170 <Select options={menuOptions} onChange={async value => { 171 switch (value) { 172 case 'auth': 173 await handleAuthenticate(); 174 break; 175 case 'back': 176 onCancel(); 177 break; 178 } 179 }} onCancel={onCancel} /> 180 </Box> 181 </Dialog>; 182} 183//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJmaWd1cmVzIiwiUmVhY3QiLCJ1c2VDYWxsYmFjayIsInVzZUVmZmVjdCIsInVzZVJlZiIsInVzZVN0YXRlIiwiQ29tbWFuZFJlc3VsdERpc3BsYXkiLCJCb3giLCJjb2xvciIsIkxpbmsiLCJUZXh0IiwidXNlVGhlbWUiLCJ1c2VLZXliaW5kaW5nIiwiQXV0aGVudGljYXRpb25DYW5jZWxsZWRFcnJvciIsInBlcmZvcm1NQ1BPQXV0aEZsb3ciLCJjYXBpdGFsaXplIiwiQ29uZmlndXJhYmxlU2hvcnRjdXRIaW50IiwiU2VsZWN0IiwiQnlsaW5lIiwiRGlhbG9nIiwiS2V5Ym9hcmRTaG9ydGN1dEhpbnQiLCJTcGlubmVyIiwiQWdlbnRNY3BTZXJ2ZXJJbmZvIiwiUHJvcHMiLCJhZ2VudFNlcnZlciIsIm9uQ2FuY2VsIiwib25Db21wbGV0ZSIsInJlc3VsdCIsIm9wdGlvbnMiLCJkaXNwbGF5IiwiTUNQQWdlbnRTZXJ2ZXJNZW51IiwiUmVhY3ROb2RlIiwidGhlbWUiLCJpc0F1dGhlbnRpY2F0aW5nIiwic2V0SXNBdXRoZW50aWNhdGluZyIsImVycm9yIiwic2V0RXJyb3IiLCJhdXRob3JpemF0aW9uVXJsIiwic2V0QXV0aG9yaXphdGlvblVybCIsImF1dGhBYm9ydENvbnRyb2xsZXJSZWYiLCJBYm9ydENvbnRyb2xsZXIiLCJjdXJyZW50IiwiYWJvcnQiLCJoYW5kbGVFc2NDYW5jZWwiLCJjb250ZXh0IiwiaXNBY3RpdmUiLCJoYW5kbGVBdXRoZW50aWNhdGUiLCJuZWVkc0F1dGgiLCJ1cmwiLCJjb250cm9sbGVyIiwidGVtcENvbmZpZyIsInR5cGUiLCJ0cmFuc3BvcnQiLCJuYW1lIiwic2lnbmFsIiwiZXJyIiwiRXJyb3IiLCJtZXNzYWdlIiwiY2FwaXRhbGl6ZWRTZXJ2ZXJOYW1lIiwiU3RyaW5nIiwibWVudU9wdGlvbnMiLCJwdXNoIiwibGFiZWwiLCJpc0F1dGhlbnRpY2F0ZWQiLCJ2YWx1ZSIsImV4aXRTdGF0ZSIsInBlbmRpbmciLCJrZXlOYW1lIiwiY29tbWFuZCIsInNvdXJjZUFnZW50cyIsImpvaW4iLCJyYWRpb09mZiIsInRpY2siLCJ0cmlhbmdsZVVwT3V0bGluZSJdLCJzb3VyY2VzIjpbIk1DUEFnZW50U2VydmVyTWVudS50c3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGZpZ3VyZXMgZnJvbSAnZmlndXJlcydcbmltcG9ydCBSZWFjdCwgeyB1c2VDYWxsYmFjaywgdXNlRWZmZWN0LCB1c2VSZWYsIHVzZVN0YXRlIH0gZnJvbSAncmVhY3QnXG5pbXBvcnQgdHlwZSB7IENvbW1hbmRSZXN1bHREaXNwbGF5IH0gZnJvbSAnLi4vLi4vY29tbWFuZHMuanMnXG5pbXBvcnQgeyBCb3gsIGNvbG9yLCBMaW5rLCBUZXh0LCB1c2VUaGVtZSB9IGZyb20gJy4uLy4uL2luay5qcydcbmltcG9ydCB7IHVzZUtleWJpbmRpbmcgfSBmcm9tICcuLi8uLi9rZXliaW5kaW5ncy91c2VLZXliaW5kaW5nLmpzJ1xuaW1wb3J0IHtcbiAgQXV0aGVudGljYXRpb25DYW5jZWxsZWRFcnJvcixcbiAgcGVyZm9ybU1DUE9BdXRoRmxvdyxcbn0gZnJvbSAnLi4vLi4vc2VydmljZXMvbWNwL2F1dGguanMnXG5pbXBvcnQgeyBjYXBpdGFsaXplIH0gZnJvbSAnLi4vLi4vdXRpbHMvc3RyaW5nVXRpbHMuanMnXG5pbXBvcnQgeyBDb25maWd1cmFibGVTaG9ydGN1dEhpbnQgfSBmcm9tICcuLi9Db25maWd1cmFibGVTaG9ydGN1dEhpbnQuanMnXG5pbXBvcnQgeyBTZWxlY3QgfSBmcm9tICcuLi9DdXN0b21TZWxlY3QvaW5kZXguanMnXG5pbXBvcnQgeyBCeWxpbmUgfSBmcm9tICcuLi9kZXNpZ24tc3lzdGVtL0J5bGluZS5qcydcbmltcG9ydCB7IERpYWxvZyB9IGZyb20gJy4uL2Rlc2lnbi1zeXN0ZW0vRGlhbG9nLmpzJ1xuaW1wb3J0IHsgS2V5Ym9hcmRTaG9ydGN1dEhpbnQgfSBmcm9tICcuLi9kZXNpZ24tc3lzdGVtL0tleWJvYXJkU2hvcnRjdXRIaW50LmpzJ1xuaW1wb3J0IHsgU3Bpbm5lciB9IGZyb20gJy4uL1NwaW5uZXIuanMnXG5pbXBvcnQgdHlwZSB7IEFnZW50TWNwU2VydmVySW5mbyB9IGZyb20gJy4vdHlwZXMuanMnXG5cbnR5cGUgUHJvcHMgPSB7XG4gIGFnZW50U2VydmVyOiBBZ2VudE1jcFNlcnZlckluZm9cbiAgb25DYW5jZWw6ICgpID0+IHZvaWRcbiAgb25Db21wbGV0ZT86IChcbiAgICByZXN1bHQ/OiBzdHJpbmcsXG4gICAgb3B0aW9ucz86IHsgZGlzcGxheT86IENvbW1hbmRSZXN1bHREaXNwbGF5IH0sXG4gICkgPT4gdm9pZFxufVxuXG4vKipcbiAqIE1lbnUgZm9yIGFnZW50LXNwZWNpZmljIE1DUCBzZXJ2ZXJzLlxuICogVGhlc2Ugc2VydmVycyBhcmUgZGVmaW5lZCBpbiBhZ2VudCBmcm9udG1hdHRlciBhbmQgb25seSBjb25uZWN0IHdoZW4gdGhlIGFnZW50IHJ1bnMuXG4gKiBGb3IgSFRUUC9TU0Ugc2VydmVycywgdGhpcyBhbGxvd3MgcHJlLWF1dGhlbnRpY2F0aW9uIGJlZm9yZSB1c2luZyB0aGUgYWdlbnQuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBNQ1BBZ2VudFNlcnZlck1lbnUoe1xuICBhZ2VudFNlcnZlcixcbiAgb25DYW5jZWwsXG4gIG9uQ29tcGxldGUsXG59OiBQcm9wcyk6IFJlYWN0LlJlYWN0Tm9kZSB7XG4gIGNvbnN0IFt0aGVtZV0gPSB1c2VUaGVtZSgpXG4gIGNvbnN0IFtpc0F1dGhlbnRpY2F0aW5nLCBzZXRJc0F1dGhlbnRpY2F0aW5nXSA9IHVzZVN0YXRlKGZhbHNlKVxuICBjb25zdCBbZXJyb3IsIHNldEVycm9yXSA9IHVzZVN0YXRlPHN0cmluZyB8IG51bGw+KG51bGwpXG4gIGNvbnN0IFthdXRob3JpemF0aW9uVXJsLCBzZXRBdXRob3JpemF0aW9uVXJsXSA9IHVzZVN0YXRlPHN0cmluZyB8IG51bGw+KG51bGwpXG4gIGNvbnN0IGF1dGhBYm9ydENvbnRyb2xsZXJSZWYgPSB1c2VSZWY8QWJvcnRDb250cm9sbGVyIHwgbnVsbD4obnVsbClcblxuICAvLyBBYm9ydCBPQXV0aCBmbG93IG9uIHVubW91bnQgc28gdGhlIGNhbGxiYWNrIHNlcnZlciBpcyBjbG9zZWQgZXZlbiBpZiBhXG4gIC8vIHBhcmVudCBjb21wb25lbnQncyBFc2MgaGFuZGxlciBuYXZpZ2F0ZXMgYXdheSBiZWZvcmUgb3VycyBmaXJlcy5cbiAgdXNlRWZmZWN0KCgpID0+ICgpID0+IGF1dGhBYm9ydENvbnRyb2xsZXJSZWYuY3VycmVudD8uYWJvcnQoKSwgW10pXG5cbiAgLy8gSGFuZGxlIEVTQyB0byBjYW5jZWwgYXV0aGVudGljYXRpb24gZmxvd1xuICBjb25zdCBoYW5kbGVFc2NDYW5jZWwgPSB1c2VDYWxsYmFjaygoKSA9PiB7XG4gICAgaWYgKGlzQXV0aGVudGljYXRpbmcpIHtcbiAgICAgIGF1dGhBYm9ydENvbnRyb2xsZXJSZWYuY3VycmVudD8uYWJvcnQoKVxuICAgICAgYXV0aEFib3J0Q29udHJvbGxlclJlZi5jdXJyZW50ID0gbnVsbFxuICAgICAgc2V0SXNBdXRoZW50aWNhdGluZyhmYWxzZSlcbiAgICAgIHNldEF1dGhvcml6YXRpb25VcmwobnVsbClcbiAgICB9XG4gIH0sIFtpc0F1dGhlbnRpY2F0aW5nXSlcblxuICB1c2VLZXliaW5kaW5nKCdjb25maXJtOm5vJywgaGFuZGxlRXNjQ2FuY2VsLCB7XG4gICAgY29udGV4dDogJ0NvbmZpcm1hdGlvbicsXG4gICAgaXNBY3RpdmU6IGlzQXV0aGVudGljYXRpbmcsXG4gIH0pXG5cbiAgY29uc3QgaGFuZGxlQXV0aGVudGljYXRlID0gdXNlQ2FsbGJhY2soYXN5bmMgKCkgPT4ge1xuICAgIGlmICghYWdlbnRTZXJ2ZXIubmVlZHNBdXRoIHx8ICFhZ2VudFNlcnZlci51cmwpIHtcbiAgICAgIHJldHVyblxuICAgIH1cblxuICAgIHNldElzQXV0aGVudGljYXRpbmcodHJ1ZSlcbiAgICBzZXRFcnJvcihudWxsKVxuXG4gICAgY29uc3QgY29udHJvbGxlciA9IG5ldyBBYm9ydENvbnRyb2xsZXIoKVxuICAgIGF1dGhBYm9ydENvbnRyb2xsZXJSZWYuY3VycmVudCA9IGNvbnRyb2xsZXJcblxuICAgIHRyeSB7XG4gICAgICAvLyBDcmVhdGUgYSB0ZW1wb3JhcnkgY29uZmlnIGZvciBPQXV0aFxuICAgICAgY29uc3QgdGVtcENvbmZpZyA9IHtcbiAgICAgICAgdHlwZTogYWdlbnRTZXJ2ZXIudHJhbnNwb3J0IGFzICdodHRwJyB8ICdzc2UnLFxuICAgICAgICB1cmw6IGFnZW50U2VydmVyLnVybCxcbiAgICAgIH1cblxuICAgICAgYXdhaXQgcGVyZm9ybU1DUE9BdXRoRmxvdyhcbiAgICAgICAgYWdlbnRTZXJ2ZXIubmFtZSxcbiAgICAgICAgdGVtcENvbmZpZyxcbiAgICAgICAgc2V0QXV0aG9yaXphdGlvblVybCxcbiAgICAgICAgY29udHJvbGxlci5zaWduYWwsXG4gICAgICApXG5cbiAgICAgIG9uQ29tcGxldGU/LihcbiAgICAgICAgYEF1dGhlbnRpY2F0aW9uIHN1Y2Nlc3NmdWwgZm9yICR7YWdlbnRTZXJ2ZXIubmFtZX0uIFRoZSBzZXJ2ZXIgd2lsbCBjb25uZWN0IHdoZW4gdGhlIGFnZW50IHJ1bnMuYCxcbiAgICAgIClcbiAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgIC8vIERvbid0IHNob3cgZXJyb3IgaWYgaXQgd2FzIGEgY2FuY2VsbGF0aW9uXG4gICAgICBpZiAoXG4gICAgICAgIGVyciBpbnN0YW5jZW9mIEVycm9yICYmXG4gICAgICAgICEoZXJyIGluc3RhbmNlb2YgQXV0aGVudGljYXRpb25DYW5jZWxsZWRFcnJvcilcbiAgICAgICkge1xuICAgICAgICBzZXRFcnJvcihlcnIubWVzc2FnZSlcbiAgICAgIH1cbiAgICB9IGZpbmFsbHkge1xuICAgICAgc2V0SXNBdXRoZW50aWNhdGluZyhmYWxzZSlcbiAgICAgIGF1dGhBYm9ydENvbnRyb2xsZXJSZWYuY3VycmVudCA9IG51bGxcbiAgICB9XG4gIH0sIFthZ2VudFNlcnZlciwgb25Db21wbGV0ZV0pXG5cbiAgY29uc3QgY2FwaXRhbGl6ZWRTZXJ2ZXJOYW1lID0gY2FwaXRhbGl6ZShTdHJpbmcoYWdlbnRTZXJ2ZXIubmFtZSkpXG5cbiAgaWYgKGlzQXV0aGVudGljYXRpbmcpIHtcbiAgICByZXR1cm4gKFxuICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCIgZ2FwPXsxfSBwYWRkaW5nPXsxfT5cbiAgICAgICAgPFRleHQgY29sb3I9XCJjbGF1ZGVcIj5BdXRoZW50aWNhdGluZyB3aXRoIHthZ2VudFNlcnZlci5uYW1lfeKApjwvVGV4dD5cbiAgICAgICAgPEJveD5cbiAgICAgICAgICA8U3Bpbm5lciAvPlxuICAgICAgICAgIDxUZXh0PiBBIGJyb3dzZXIgd2luZG93IHdpbGwgb3BlbiBmb3IgYXV0aGVudGljYXRpb248L1RleHQ+XG4gICAgICAgIDwvQm94PlxuICAgICAgICB7YXV0aG9yaXphdGlvblVybCAmJiAoXG4gICAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgICAgICA8VGV4dCBkaW1Db2xvcj5cbiAgICAgICAgICAgICAgSWYgeW91ciBicm93c2VyIGRvZXNuJmFwb3M7dCBvcGVuIGF1dG9tYXRpY2FsbHksIGNvcHkgdGhpcyBVUkxcbiAgICAgICAgICAgICAgbWFudWFsbHk6XG4gICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICA8TGluayB1cmw9e2F1dGhvcml6YXRpb25Vcmx9IC8+XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgICl9XG4gICAgICAgIDxCb3ggbWFyZ2luTGVmdD17M30+XG4gICAgICAgICAgPFRleHQgZGltQ29sb3I+XG4gICAgICAgICAgICBSZXR1cm4gaGVyZSBhZnRlciBhdXRoZW50aWNhdGluZyBpbiB5b3VyIGJyb3dzZXIueycgJ31cbiAgICAgICAgICAgIDxDb25maWd1cmFibGVTaG9ydGN1dEhpbnRcbiAgICAgICAgICAgICAgYWN0aW9uPVwiY29uZmlybTpub1wiXG4gICAgICAgICAgICAgIGNvbnRleHQ9XCJDb25maXJtYXRpb25cIlxuICAgICAgICAgICAgICBmYWxsYmFjaz1cIkVzY1wiXG4gICAgICAgICAgICAgIGRlc2NyaXB0aW9uPVwiZ28gYmFja1wiXG4gICAgICAgICAgICAvPlxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICA8L0JveD5cbiAgICApXG4gIH1cblxuICBjb25zdCBtZW51T3B0aW9ucyA9IFtdXG5cbiAgLy8gT25seSBzaG93IGF1dGhlbnRpY2F0ZSBvcHRpb24gZm9yIEhUVFAvU1NFIHNlcnZlcnNcbiAgaWYgKGFnZW50U2VydmVyLm5lZWRzQXV0aCkge1xuICAgIG1lbnVPcHRpb25zLnB1c2goe1xuICAgICAgbGFiZWw6IGFnZW50U2VydmVyLmlzQXV0aGVudGljYXRlZCA/ICdSZS1hdXRoZW50aWNhdGUnIDogJ0F1dGhlbnRpY2F0ZScsXG4gICAgICB2YWx1ZTogJ2F1dGgnLFxuICAgIH0pXG4gIH1cblxuICBtZW51T3B0aW9ucy5wdXNoKHtcbiAgICBsYWJlbDogJ0JhY2snLFxuICAgIHZhbHVlOiAnYmFjaycsXG4gIH0pXG5cbiAgcmV0dXJuIChcbiAgICA8RGlhbG9nXG4gICAgICB0aXRsZT17YCR7Y2FwaXRhbGl6ZWRTZXJ2ZXJOYW1lfSBNQ1AgU2VydmVyYH1cbiAgICAgIHN1YnRpdGxlPVwiYWdlbnQtb25seVwiXG4gICAgICBvbkNhbmNlbD17b25DYW5jZWx9XG4gICAgICBpbnB1dEd1aWRlPXtleGl0U3RhdGUgPT5cbiAgICAgICAgZXhpdFN0YXRlLnBlbmRpbmcgPyAoXG4gICAgICAgICAgPFRleHQ+UHJlc3Mge2V4aXRTdGF0ZS5rZXlOYW1lfSBhZ2FpbiB0byBleGl0PC9UZXh0PlxuICAgICAgICApIDogKFxuICAgICAgICAgIDxCeWxpbmU+XG4gICAgICAgICAgICA8S2V5Ym9hcmRTaG9ydGN1dEhpbnQgc2hvcnRjdXQ9XCLihpHihpNcIiBhY3Rpb249XCJuYXZpZ2F0ZVwiIC8+XG4gICAgICAgICAgICA8S2V5Ym9hcmRTaG9ydGN1dEhpbnQgc2hvcnRjdXQ9XCJFbnRlclwiIGFjdGlvbj1cImNvbmZpcm1cIiAvPlxuICAgICAgICAgICAgPENvbmZpZ3VyYWJsZVNob3J0Y3V0SGludFxuICAgICAgICAgICAgICBhY3Rpb249XCJjb25maXJtOm5vXCJcbiAgICAgICAgICAgICAgY29udGV4dD1cIkNvbmZpcm1hdGlvblwiXG4gICAgICAgICAgICAgIGZhbGxiYWNrPVwiRXNjXCJcbiAgICAgICAgICAgICAgZGVzY3JpcHRpb249XCJnbyBiYWNrXCJcbiAgICAgICAgICAgIC8+XG4gICAgICAgICAgPC9CeWxpbmU+XG4gICAgICAgIClcbiAgICAgIH1cbiAgICA+XG4gICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBnYXA9ezB9PlxuICAgICAgICA8Qm94PlxuICAgICAgICAgIDxUZXh0IGJvbGQ+VHlwZTogPC9UZXh0PlxuICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPnthZ2VudFNlcnZlci50cmFuc3BvcnR9PC9UZXh0PlxuICAgICAgICA8L0JveD5cblxuICAgICAgICB7YWdlbnRTZXJ2ZXIudXJsICYmIChcbiAgICAgICAgICA8Qm94PlxuICAgICAgICAgICAgPFRleHQgYm9sZD5VUkw6IDwvVGV4dD5cbiAgICAgICAgICAgIDxUZXh0IGRpbUNvbG9yPnthZ2VudFNlcnZlci51cmx9PC9UZXh0PlxuICAgICAgICAgIDwvQm94PlxuICAgICAgICApfVxuXG4gICAgICAgIHthZ2VudFNlcnZlci5jb21tYW5kICYmIChcbiAgICAgICAgICA8Qm94PlxuICAgICAgICAgICAgPFRleHQgYm9sZD5Db21tYW5kOiA8L1RleHQ+XG4gICAgICAgICAgICA8VGV4dCBkaW1Db2xvcj57YWdlbnRTZXJ2ZXIuY29tbWFuZH08L1RleHQ+XG4gICAgICAgICAgPC9Cb3g+XG4gICAgICAgICl9XG5cbiAgICAgICAgPEJveD5cbiAgICAgICAgICA8VGV4dCBib2xkPlVzZWQgYnk6IDwvVGV4dD5cbiAgICAgICAgICA8VGV4dCBkaW1Db2xvcj57YWdlbnRTZXJ2ZXIuc291cmNlQWdlbnRzLmpvaW4oJywgJyl9PC9UZXh0PlxuICAgICAgICA8L0JveD5cblxuICAgICAgICA8Qm94IG1hcmdpblRvcD17MX0+XG4gICAgICAgICAgPFRleHQgYm9sZD5TdGF0dXM6IDwvVGV4dD5cbiAgICAgICAgICA8VGV4dD5cbiAgICAgICAgICAgIHtjb2xvcignaW5hY3RpdmUnLCB0aGVtZSkoZmlndXJlcy5yYWRpb09mZil9IG5vdCBjb25uZWN0ZWRcbiAgICAgICAgICAgIChhZ2VudC1vbmx5KVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG5cbiAgICAgICAge2FnZW50U2VydmVyLm5lZWRzQXV0aCAmJiAoXG4gICAgICAgICAgPEJveD5cbiAgICAgICAgICAgIDxUZXh0IGJvbGQ+QXV0aDogPC9UZXh0PlxuICAgICAgICAgICAge2FnZW50U2VydmVyLmlzQXV0aGVudGljYXRlZCA/IChcbiAgICAgICAgICAgICAgPFRleHQ+e2NvbG9yKCdzdWNjZXNzJywgdGhlbWUpKGZpZ3VyZXMudGljayl9IGF1dGhlbnRpY2F0ZWQ8L1RleHQ+XG4gICAgICAgICAgICApIDogKFxuICAgICAgICAgICAgICA8VGV4dD5cbiAgICAgICAgICAgICAgICB7Y29sb3IoJ3dhcm5pbmcnLCB0aGVtZSkoZmlndXJlcy50cmlhbmdsZVVwT3V0bGluZSl9IG1heSBuZWVkXG4gICAgICAgICAgICAgICAgYXV0aGVudGljYXRpb25cbiAgICAgICAgICAgICAgPC9UZXh0PlxuICAgICAgICAgICAgKX1cbiAgICAgICAgICA8L0JveD5cbiAgICAgICAgKX1cbiAgICAgIDwvQm94PlxuXG4gICAgICA8Qm94PlxuICAgICAgICA8VGV4dCBkaW1Db2xvcj5UaGlzIHNlcnZlciBjb25uZWN0cyBvbmx5IHdoZW4gcnVubmluZyB0aGUgYWdlbnQuPC9UZXh0PlxuICAgICAgPC9Cb3g+XG5cbiAgICAgIHtlcnJvciAmJiAoXG4gICAgICAgIDxCb3g+XG4gICAgICAgICAgPFRleHQgY29sb3I9XCJlcnJvclwiPkVycm9yOiB7ZXJyb3J9PC9UZXh0PlxuICAgICAgICA8L0JveD5cbiAgICAgICl9XG5cbiAgICAgIDxCb3g+XG4gICAgICAgIDxTZWxlY3RcbiAgICAgICAgICBvcHRpb25zPXttZW51T3B0aW9uc31cbiAgICAgICAgICBvbkNoYW5nZT17YXN5bmMgdmFsdWUgPT4ge1xuICAgICAgICAgICAgc3dpdGNoICh2YWx1ZSkge1xuICAgICAgICAgICAgICBjYXNlICdhdXRoJzpcbiAgICAgICAgICAgICAgICBhd2FpdCBoYW5kbGVBdXRoZW50aWNhdGUoKVxuICAgICAgICAgICAgICAgIGJyZWFrXG4gICAgICAgICAgICAgIGNhc2UgJ2JhY2snOlxuICAgICAgICAgICAgICAgIG9uQ2FuY2VsKClcbiAgICAgICAgICAgICAgICBicmVha1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH19XG4gICAgICAgICAgb25DYW5jZWw9e29uQ2FuY2VsfVxuICAgICAgICAvPlxuICAgICAgPC9Cb3g+XG4gICAgPC9EaWFsb2c+XG4gIClcbn1cbiJdLCJtYXBwaW5ncyI6IkFBQUEsT0FBT0EsT0FBTyxNQUFNLFNBQVM7QUFDN0IsT0FBT0MsS0FBSyxJQUFJQyxXQUFXLEVBQUVDLFNBQVMsRUFBRUMsTUFBTSxFQUFFQyxRQUFRLFFBQVEsT0FBTztBQUN2RSxjQUFjQyxvQkFBb0IsUUFBUSxtQkFBbUI7QUFDN0QsU0FBU0MsR0FBRyxFQUFFQyxLQUFLLEVBQUVDLElBQUksRUFBRUMsSUFBSSxFQUFFQyxRQUFRLFFBQVEsY0FBYztBQUMvRCxTQUFTQyxhQUFhLFFBQVEsb0NBQW9DO0FBQ2xFLFNBQ0VDLDRCQUE0QixFQUM1QkMsbUJBQW1CLFFBQ2QsNEJBQTRCO0FBQ25DLFNBQVNDLFVBQVUsUUFBUSw0QkFBNEI7QUFDdkQsU0FBU0Msd0JBQXdCLFFBQVEsZ0NBQWdDO0FBQ3pFLFNBQVNDLE1BQU0sUUFBUSwwQkFBMEI7QUFDakQsU0FBU0MsTUFBTSxRQUFRLDRCQUE0QjtBQUNuRCxT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