/ utils / statusNoticeDefinitions.tsx
statusNoticeDefinitions.tsx
  1  // biome-ignore-all assist/source/organizeImports: ANT-ONLY import markers must not be reordered
  2  import { Box, Text } from '../ink.js';
  3  import * as React from 'react';
  4  import { getLargeMemoryFiles, MAX_MEMORY_CHARACTER_COUNT, type MemoryFileInfo } from './claudemd.js';
  5  import figures from 'figures';
  6  import { getCwd } from './cwd.js';
  7  import { relative } from 'path';
  8  import { formatNumber } from './format.js';
  9  import type { getGlobalConfig } from './config.js';
 10  import { getAnthropicApiKeyWithSource, getApiKeyFromConfigOrMacOSKeychain, getAuthTokenSource, isClaudeAISubscriber } from './auth.js';
 11  import type { AgentDefinitionsResult } from '../tools/AgentTool/loadAgentsDir.js';
 12  import { getAgentDescriptionsTotalTokens, AGENT_DESCRIPTIONS_THRESHOLD } from './statusNoticeHelpers.js';
 13  import { isSupportedJetBrainsTerminal, toIDEDisplayName, getTerminalIdeType } from './ide.js';
 14  import { isJetBrainsPluginInstalledCachedSync } from './jetbrains.js';
 15  
 16  // Types
 17  export type StatusNoticeType = 'warning' | 'info';
 18  export type StatusNoticeContext = {
 19    config: ReturnType<typeof getGlobalConfig>;
 20    agentDefinitions?: AgentDefinitionsResult;
 21    memoryFiles: MemoryFileInfo[];
 22  };
 23  export type StatusNoticeDefinition = {
 24    id: string;
 25    type: StatusNoticeType;
 26    isActive: (context: StatusNoticeContext) => boolean;
 27    render: (context: StatusNoticeContext) => React.ReactNode;
 28  };
 29  
 30  // Individual notice definitions
 31  const largeMemoryFilesNotice: StatusNoticeDefinition = {
 32    id: 'large-memory-files',
 33    type: 'warning',
 34    isActive: ctx => getLargeMemoryFiles(ctx.memoryFiles).length > 0,
 35    render: ctx => {
 36      const largeMemoryFiles = getLargeMemoryFiles(ctx.memoryFiles);
 37      return <>
 38          {largeMemoryFiles.map(file => {
 39          const displayPath = file.path.startsWith(getCwd()) ? relative(getCwd(), file.path) : file.path;
 40          return <Box key={file.path} flexDirection="row">
 41                <Text color="warning">{figures.warning}</Text>
 42                <Text color="warning">
 43                  Large <Text bold>{displayPath}</Text> will impact performance (
 44                  {formatNumber(file.content.length)} chars &gt;{' '}
 45                  {formatNumber(MAX_MEMORY_CHARACTER_COUNT)})
 46                  <Text dimColor> · /memory to edit</Text>
 47                </Text>
 48              </Box>;
 49        })}
 50        </>;
 51    }
 52  };
 53  const claudeAiSubscriberExternalTokenNotice: StatusNoticeDefinition = {
 54    id: 'claude-ai-external-token',
 55    type: 'warning',
 56    isActive: () => {
 57      const authTokenInfo = getAuthTokenSource();
 58      return isClaudeAISubscriber() && (authTokenInfo.source === 'ANTHROPIC_AUTH_TOKEN' || authTokenInfo.source === 'apiKeyHelper');
 59    },
 60    render: () => {
 61      const authTokenInfo = getAuthTokenSource();
 62      return <Box flexDirection="row" marginTop={1}>
 63          <Text color="warning">{figures.warning}</Text>
 64          <Text color="warning">
 65            Auth conflict: Using {authTokenInfo.source} instead of Claude account
 66            subscription token. Either unset {authTokenInfo.source}, or run
 67            `claude /logout`.
 68          </Text>
 69        </Box>;
 70    }
 71  };
 72  const apiKeyConflictNotice: StatusNoticeDefinition = {
 73    id: 'api-key-conflict',
 74    type: 'warning',
 75    isActive: () => {
 76      const {
 77        source: apiKeySource
 78      } = getAnthropicApiKeyWithSource({
 79        skipRetrievingKeyFromApiKeyHelper: true
 80      });
 81      return !!getApiKeyFromConfigOrMacOSKeychain() && (apiKeySource === 'ANTHROPIC_API_KEY' || apiKeySource === 'apiKeyHelper');
 82    },
 83    render: () => {
 84      const {
 85        source: apiKeySource
 86      } = getAnthropicApiKeyWithSource({
 87        skipRetrievingKeyFromApiKeyHelper: true
 88      });
 89      return <Box flexDirection="row" marginTop={1}>
 90          <Text color="warning">{figures.warning}</Text>
 91          <Text color="warning">
 92            Auth conflict: Using {apiKeySource} instead of Anthropic Console key.
 93            Either unset {apiKeySource}, or run `claude /logout`.
 94          </Text>
 95        </Box>;
 96    }
 97  };
 98  const bothAuthMethodsNotice: StatusNoticeDefinition = {
 99    id: 'both-auth-methods',
100    type: 'warning',
101    isActive: () => {
102      const {
103        source: apiKeySource
104      } = getAnthropicApiKeyWithSource({
105        skipRetrievingKeyFromApiKeyHelper: true
106      });
107      const authTokenInfo = getAuthTokenSource();
108      return apiKeySource !== 'none' && authTokenInfo.source !== 'none' && !(apiKeySource === 'apiKeyHelper' && authTokenInfo.source === 'apiKeyHelper');
109    },
110    render: () => {
111      const {
112        source: apiKeySource
113      } = getAnthropicApiKeyWithSource({
114        skipRetrievingKeyFromApiKeyHelper: true
115      });
116      const authTokenInfo = getAuthTokenSource();
117      return <Box flexDirection="column" marginTop={1}>
118          <Box flexDirection="row">
119            <Text color="warning">{figures.warning}</Text>
120            <Text color="warning">
121              Auth conflict: Both a token ({authTokenInfo.source}) and an API key
122              ({apiKeySource}) are set. This may lead to unexpected behavior.
123            </Text>
124          </Box>
125          <Box flexDirection="column" marginLeft={3}>
126            <Text color="warning">
127              · Trying to use{' '}
128              {authTokenInfo.source === 'claude.ai' ? 'claude.ai' : authTokenInfo.source}
129              ?{' '}
130              {apiKeySource === 'ANTHROPIC_API_KEY' ? 'Unset the ANTHROPIC_API_KEY environment variable, or claude /logout then say "No" to the API key approval before login.' : apiKeySource === 'apiKeyHelper' ? 'Unset the apiKeyHelper setting.' : 'claude /logout'}
131            </Text>
132            <Text color="warning">
133              · Trying to use {apiKeySource}?{' '}
134              {authTokenInfo.source === 'claude.ai' ? 'claude /logout to sign out of claude.ai.' : `Unset the ${authTokenInfo.source} environment variable.`}
135            </Text>
136          </Box>
137        </Box>;
138    }
139  };
140  const largeAgentDescriptionsNotice: StatusNoticeDefinition = {
141    id: 'large-agent-descriptions',
142    type: 'warning',
143    isActive: context => {
144      const totalTokens = getAgentDescriptionsTotalTokens(context.agentDefinitions);
145      return totalTokens > AGENT_DESCRIPTIONS_THRESHOLD;
146    },
147    render: context => {
148      const totalTokens = getAgentDescriptionsTotalTokens(context.agentDefinitions);
149      return <Box flexDirection="row">
150          <Text color="warning">{figures.warning}</Text>
151          <Text color="warning">
152            Large cumulative agent descriptions will impact performance (~
153            {formatNumber(totalTokens)} tokens &gt;{' '}
154            {formatNumber(AGENT_DESCRIPTIONS_THRESHOLD)})
155            <Text dimColor> · /agents to manage</Text>
156          </Text>
157        </Box>;
158    }
159  };
160  const jetbrainsPluginNotice: StatusNoticeDefinition = {
161    id: 'jetbrains-plugin-install',
162    type: 'info',
163    isActive: context => {
164      // Only show if running in JetBrains built-in terminal
165      if (!isSupportedJetBrainsTerminal()) {
166        return false;
167      }
168      // Don't show if auto-install is disabled
169      const shouldAutoInstall = context.config.autoInstallIdeExtension ?? true;
170      if (!shouldAutoInstall) {
171        return false;
172      }
173      // Check if plugin is already installed (cached to avoid repeated filesystem checks)
174      const ideType = getTerminalIdeType();
175      return ideType !== null && !isJetBrainsPluginInstalledCachedSync(ideType);
176    },
177    render: () => {
178      const ideType = getTerminalIdeType();
179      const ideName = toIDEDisplayName(ideType);
180      return <Box flexDirection="row" gap={1} marginLeft={1}>
181          <Text color="ide">{figures.arrowUp}</Text>
182          <Text>
183            Install the <Text color="ide">{ideName}</Text> plugin from the
184            JetBrains Marketplace:{' '}
185            <Text bold>https://docs.claude.com/s/claude-code-jetbrains</Text>
186          </Text>
187        </Box>;
188    }
189  };
190  
191  // All notice definitions
192  export const statusNoticeDefinitions: StatusNoticeDefinition[] = [largeMemoryFilesNotice, largeAgentDescriptionsNotice, claudeAiSubscriberExternalTokenNotice, apiKeyConflictNotice, bothAuthMethodsNotice, jetbrainsPluginNotice];
193  
194  // Helper functions for external use
195  export function getActiveNotices(context: StatusNoticeContext): StatusNoticeDefinition[] {
196    return statusNoticeDefinitions.filter(notice => notice.isActive(context));
197  }
198  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJCb3giLCJUZXh0IiwiUmVhY3QiLCJnZXRMYXJnZU1lbW9yeUZpbGVzIiwiTUFYX01FTU9SWV9DSEFSQUNURVJfQ09VTlQiLCJNZW1vcnlGaWxlSW5mbyIsImZpZ3VyZXMiLCJnZXRDd2QiLCJyZWxhdGl2ZSIsImZvcm1hdE51bWJlciIsImdldEdsb2JhbENvbmZpZyIsImdldEFudGhyb3BpY0FwaUtleVdpdGhTb3VyY2UiLCJnZXRBcGlLZXlGcm9tQ29uZmlnT3JNYWNPU0tleWNoYWluIiwiZ2V0QXV0aFRva2VuU291cmNlIiwiaXNDbGF1ZGVBSVN1YnNjcmliZXIiLCJBZ2VudERlZmluaXRpb25zUmVzdWx0IiwiZ2V0QWdlbnREZXNjcmlwdGlvbnNUb3RhbFRva2VucyIsIkFHRU5UX0RFU0NSSVBUSU9OU19USFJFU0hPTEQiLCJpc1N1cHBvcnRlZEpldEJyYWluc1Rlcm1pbmFsIiwidG9JREVEaXNwbGF5TmFtZSIsImdldFRlcm1pbmFsSWRlVHlwZSIsImlzSmV0QnJhaW5zUGx1Z2luSW5zdGFsbGVkQ2FjaGVkU3luYyIsIlN0YXR1c05vdGljZVR5cGUiLCJTdGF0dXNOb3RpY2VDb250ZXh0IiwiY29uZmlnIiwiUmV0dXJuVHlwZSIsImFnZW50RGVmaW5pdGlvbnMiLCJtZW1vcnlGaWxlcyIsIlN0YXR1c05vdGljZURlZmluaXRpb24iLCJpZCIsInR5cGUiLCJpc0FjdGl2ZSIsImNvbnRleHQiLCJyZW5kZXIiLCJSZWFjdE5vZGUiLCJsYXJnZU1lbW9yeUZpbGVzTm90aWNlIiwiY3R4IiwibGVuZ3RoIiwibGFyZ2VNZW1vcnlGaWxlcyIsIm1hcCIsImZpbGUiLCJkaXNwbGF5UGF0aCIsInBhdGgiLCJzdGFydHNXaXRoIiwid2FybmluZyIsImNvbnRlbnQiLCJjbGF1ZGVBaVN1YnNjcmliZXJFeHRlcm5hbFRva2VuTm90aWNlIiwiYXV0aFRva2VuSW5mbyIsInNvdXJjZSIsImFwaUtleUNvbmZsaWN0Tm90aWNlIiwiYXBpS2V5U291cmNlIiwic2tpcFJldHJpZXZpbmdLZXlGcm9tQXBpS2V5SGVscGVyIiwiYm90aEF1dGhNZXRob2RzTm90aWNlIiwibGFyZ2VBZ2VudERlc2NyaXB0aW9uc05vdGljZSIsInRvdGFsVG9rZW5zIiwiamV0YnJhaW5zUGx1Z2luTm90aWNlIiwic2hvdWxkQXV0b0luc3RhbGwiLCJhdXRvSW5zdGFsbElkZUV4dGVuc2lvbiIsImlkZVR5cGUiLCJpZGVOYW1lIiwiYXJyb3dVcCIsInN0YXR1c05vdGljZURlZmluaXRpb25zIiwiZ2V0QWN0aXZlTm90aWNlcyIsImZpbHRlciIsIm5vdGljZSJdLCJzb3VyY2VzIjpbInN0YXR1c05vdGljZURlZmluaXRpb25zLnRzeCJdLCJzb3VyY2VzQ29udGVudCI6WyIvLyBiaW9tZS1pZ25vcmUtYWxsIGFzc2lzdC9zb3VyY2Uvb3JnYW5pemVJbXBvcnRzOiBBTlQtT05MWSBpbXBvcnQgbWFya2VycyBtdXN0IG5vdCBiZSByZW9yZGVyZWRcbmltcG9ydCB7IEJveCwgVGV4dCB9IGZyb20gJy4uL2luay5qcydcbmltcG9ydCAqIGFzIFJlYWN0IGZyb20gJ3JlYWN0J1xuaW1wb3J0IHtcbiAgZ2V0TGFyZ2VNZW1vcnlGaWxlcyxcbiAgTUFYX01FTU9SWV9DSEFSQUNURVJfQ09VTlQsXG4gIHR5cGUgTWVtb3J5RmlsZUluZm8sXG59IGZyb20gJy4vY2xhdWRlbWQuanMnXG5pbXBvcnQgZmlndXJlcyBmcm9tICdmaWd1cmVzJ1xuaW1wb3J0IHsgZ2V0Q3dkIH0gZnJvbSAnLi9jd2QuanMnXG5pbXBvcnQgeyByZWxhdGl2ZSB9IGZyb20gJ3BhdGgnXG5pbXBvcnQgeyBmb3JtYXROdW1iZXIgfSBmcm9tICcuL2Zvcm1hdC5qcydcbmltcG9ydCB0eXBlIHsgZ2V0R2xvYmFsQ29uZmlnIH0gZnJvbSAnLi9jb25maWcuanMnXG5pbXBvcnQge1xuICBnZXRBbnRocm9waWNBcGlLZXlXaXRoU291cmNlLFxuICBnZXRBcGlLZXlGcm9tQ29uZmlnT3JNYWNPU0tleWNoYWluLFxuICBnZXRBdXRoVG9rZW5Tb3VyY2UsXG4gIGlzQ2xhdWRlQUlTdWJzY3JpYmVyLFxufSBmcm9tICcuL2F1dGguanMnXG5pbXBvcnQgdHlwZSB7IEFnZW50RGVmaW5pdGlvbnNSZXN1bHQgfSBmcm9tICcuLi90b29scy9BZ2VudFRvb2wvbG9hZEFnZW50c0Rpci5qcydcbmltcG9ydCB7XG4gIGdldEFnZW50RGVzY3JpcHRpb25zVG90YWxUb2tlbnMsXG4gIEFHRU5UX0RFU0NSSVBUSU9OU19USFJFU0hPTEQsXG59IGZyb20gJy4vc3RhdHVzTm90aWNlSGVscGVycy5qcydcbmltcG9ydCB7XG4gIGlzU3VwcG9ydGVkSmV0QnJhaW5zVGVybWluYWwsXG4gIHRvSURFRGlzcGxheU5hbWUsXG4gIGdldFRlcm1pbmFsSWRlVHlwZSxcbn0gZnJvbSAnLi9pZGUuanMnXG5pbXBvcnQgeyBpc0pldEJyYWluc1BsdWdpbkluc3RhbGxlZENhY2hlZFN5bmMgfSBmcm9tICcuL2pldGJyYWlucy5qcydcblxuLy8gVHlwZXNcbmV4cG9ydCB0eXBlIFN0YXR1c05vdGljZVR5cGUgPSAnd2FybmluZycgfCAnaW5mbydcblxuZXhwb3J0IHR5cGUgU3RhdHVzTm90aWNlQ29udGV4dCA9IHtcbiAgY29uZmlnOiBSZXR1cm5UeXBlPHR5cGVvZiBnZXRHbG9iYWxDb25maWc+XG4gIGFnZW50RGVmaW5pdGlvbnM/OiBBZ2VudERlZmluaXRpb25zUmVzdWx0XG4gIG1lbW9yeUZpbGVzOiBNZW1vcnlGaWxlSW5mb1tdXG59XG5cbmV4cG9ydCB0eXBlIFN0YXR1c05vdGljZURlZmluaXRpb24gPSB7XG4gIGlkOiBzdHJpbmdcbiAgdHlwZTogU3RhdHVzTm90aWNlVHlwZVxuICBpc0FjdGl2ZTogKGNvbnRleHQ6IFN0YXR1c05vdGljZUNvbnRleHQpID0+IGJvb2xlYW5cbiAgcmVuZGVyOiAoY29udGV4dDogU3RhdHVzTm90aWNlQ29udGV4dCkgPT4gUmVhY3QuUmVhY3ROb2RlXG59XG5cbi8vIEluZGl2aWR1YWwgbm90aWNlIGRlZmluaXRpb25zXG5jb25zdCBsYXJnZU1lbW9yeUZpbGVzTm90aWNlOiBTdGF0dXNOb3RpY2VEZWZpbml0aW9uID0ge1xuICBpZDogJ2xhcmdlLW1lbW9yeS1maWxlcycsXG4gIHR5cGU6ICd3YXJuaW5nJyxcbiAgaXNBY3RpdmU6IGN0eCA9PiBnZXRMYXJnZU1lbW9yeUZpbGVzKGN0eC5tZW1vcnlGaWxlcykubGVuZ3RoID4gMCxcbiAgcmVuZGVyOiBjdHggPT4ge1xuICAgIGNvbnN0IGxhcmdlTWVtb3J5RmlsZXMgPSBnZXRMYXJnZU1lbW9yeUZpbGVzKGN0eC5tZW1vcnlGaWxlcylcbiAgICByZXR1cm4gKFxuICAgICAgPD5cbiAgICAgICAge2xhcmdlTWVtb3J5RmlsZXMubWFwKGZpbGUgPT4ge1xuICAgICAgICAgIGNvbnN0IGRpc3BsYXlQYXRoID0gZmlsZS5wYXRoLnN0YXJ0c1dpdGgoZ2V0Q3dkKCkpXG4gICAgICAgICAgICA/IHJlbGF0aXZlKGdldEN3ZCgpLCBmaWxlLnBhdGgpXG4gICAgICAgICAgICA6IGZpbGUucGF0aFxuXG4gICAgICAgICAgcmV0dXJuIChcbiAgICAgICAgICAgIDxCb3gga2V5PXtmaWxlLnBhdGh9IGZsZXhEaXJlY3Rpb249XCJyb3dcIj5cbiAgICAgICAgICAgICAgPFRleHQgY29sb3I9XCJ3YXJuaW5nXCI+e2ZpZ3VyZXMud2FybmluZ308L1RleHQ+XG4gICAgICAgICAgICAgIDxUZXh0IGNvbG9yPVwid2FybmluZ1wiPlxuICAgICAgICAgICAgICAgIExhcmdlIDxUZXh0IGJvbGQ+e2Rpc3BsYXlQYXRofTwvVGV4dD4gd2lsbCBpbXBhY3QgcGVyZm9ybWFuY2UgKFxuICAgICAgICAgICAgICAgIHtmb3JtYXROdW1iZXIoZmlsZS5jb250ZW50Lmxlbmd0aCl9IGNoYXJzICZndDt7JyAnfVxuICAgICAgICAgICAgICAgIHtmb3JtYXROdW1iZXIoTUFYX01FTU9SWV9DSEFSQUNURVJfQ09VTlQpfSlcbiAgICAgICAgICAgICAgICA8VGV4dCBkaW1Db2xvcj4gwrcgL21lbW9yeSB0byBlZGl0PC9UZXh0PlxuICAgICAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgICA8L0JveD5cbiAgICAgICAgICApXG4gICAgICAgIH0pfVxuICAgICAgPC8+XG4gICAgKVxuICB9LFxufVxuXG5jb25zdCBjbGF1ZGVBaVN1YnNjcmliZXJFeHRlcm5hbFRva2VuTm90aWNlOiBTdGF0dXNOb3RpY2VEZWZpbml0aW9uID0ge1xuICBpZDogJ2NsYXVkZS1haS1leHRlcm5hbC10b2tlbicsXG4gIHR5cGU6ICd3YXJuaW5nJyxcbiAgaXNBY3RpdmU6ICgpID0+IHtcbiAgICBjb25zdCBhdXRoVG9rZW5JbmZvID0gZ2V0QXV0aFRva2VuU291cmNlKClcbiAgICByZXR1cm4gKFxuICAgICAgaXNDbGF1ZGVBSVN1YnNjcmliZXIoKSAmJlxuICAgICAgKGF1dGhUb2tlbkluZm8uc291cmNlID09PSAnQU5USFJPUElDX0FVVEhfVE9LRU4nIHx8XG4gICAgICAgIGF1dGhUb2tlbkluZm8uc291cmNlID09PSAnYXBpS2V5SGVscGVyJylcbiAgICApXG4gIH0sXG4gIHJlbmRlcjogKCkgPT4ge1xuICAgIGNvbnN0IGF1dGhUb2tlbkluZm8gPSBnZXRBdXRoVG9rZW5Tb3VyY2UoKVxuICAgIHJldHVybiAoXG4gICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJyb3dcIiBtYXJnaW5Ub3A9ezF9PlxuICAgICAgICA8VGV4dCBjb2xvcj1cIndhcm5pbmdcIj57ZmlndXJlcy53YXJuaW5nfTwvVGV4dD5cbiAgICAgICAgPFRleHQgY29sb3I9XCJ3YXJuaW5nXCI+XG4gICAgICAgICAgQXV0aCBjb25mbGljdDogVXNpbmcge2F1dGhUb2tlbkluZm8uc291cmNlfSBpbnN0ZWFkIG9mIENsYXVkZSBhY2NvdW50XG4gICAgICAgICAgc3Vic2NyaXB0aW9uIHRva2VuLiBFaXRoZXIgdW5zZXQge2F1dGhUb2tlbkluZm8uc291cmNlfSwgb3IgcnVuXG4gICAgICAgICAgYGNsYXVkZSAvbG9nb3V0YC5cbiAgICAgICAgPC9UZXh0PlxuICAgICAgPC9Cb3g+XG4gICAgKVxuICB9LFxufVxuXG5jb25zdCBhcGlLZXlDb25mbGljdE5vdGljZTogU3RhdHVzTm90aWNlRGVmaW5pdGlvbiA9IHtcbiAgaWQ6ICdhcGkta2V5LWNvbmZsaWN0JyxcbiAgdHlwZTogJ3dhcm5pbmcnLFxuICBpc0FjdGl2ZTogKCkgPT4ge1xuICAgIGNvbnN0IHsgc291cmNlOiBhcGlLZXlTb3VyY2UgfSA9IGdldEFudGhyb3BpY0FwaUtleVdpdGhTb3VyY2Uoe1xuICAgICAgc2tpcFJldHJpZXZpbmdLZXlGcm9tQXBpS2V5SGVscGVyOiB0cnVlLFxuICAgIH0pXG4gICAgcmV0dXJuIChcbiAgICAgICEhZ2V0QXBpS2V5RnJvbUNvbmZpZ09yTWFjT1NLZXljaGFpbigpICYmXG4gICAgICAoYXBpS2V5U291cmNlID09PSAnQU5USFJPUElDX0FQSV9LRVknIHx8IGFwaUtleVNvdXJjZSA9PT0gJ2FwaUtleUhlbHBlcicpXG4gICAgKVxuICB9LFxuICByZW5kZXI6ICgpID0+IHtcbiAgICBjb25zdCB7IHNvdXJjZTogYXBpS2V5U291cmNlIH0gPSBnZXRBbnRocm9waWNBcGlLZXlXaXRoU291cmNlKHtcbiAgICAgIHNraXBSZXRyaWV2aW5nS2V5RnJvbUFwaUtleUhlbHBlcjogdHJ1ZSxcbiAgICB9KVxuICAgIHJldHVybiAoXG4gICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJyb3dcIiBtYXJnaW5Ub3A9ezF9PlxuICAgICAgICA8VGV4dCBjb2xvcj1cIndhcm5pbmdcIj57ZmlndXJlcy53YXJuaW5nfTwvVGV4dD5cbiAgICAgICAgPFRleHQgY29sb3I9XCJ3YXJuaW5nXCI+XG4gICAgICAgICAgQXV0aCBjb25mbGljdDogVXNpbmcge2FwaUtleVNvdXJjZX0gaW5zdGVhZCBvZiBBbnRocm9waWMgQ29uc29sZSBrZXkuXG4gICAgICAgICAgRWl0aGVyIHVuc2V0IHthcGlLZXlTb3VyY2V9LCBvciBydW4gYGNsYXVkZSAvbG9nb3V0YC5cbiAgICAgICAgPC9UZXh0PlxuICAgICAgPC9Cb3g+XG4gICAgKVxuICB9LFxufVxuXG5jb25zdCBib3RoQXV0aE1ldGhvZHNOb3RpY2U6IFN0YXR1c05vdGljZURlZmluaXRpb24gPSB7XG4gIGlkOiAnYm90aC1hdXRoLW1ldGhvZHMnLFxuICB0eXBlOiAnd2FybmluZycsXG4gIGlzQWN0aXZlOiAoKSA9PiB7XG4gICAgY29uc3QgeyBzb3VyY2U6IGFwaUtleVNvdXJjZSB9ID0gZ2V0QW50aHJvcGljQXBpS2V5V2l0aFNvdXJjZSh7XG4gICAgICBza2lwUmV0cmlldmluZ0tleUZyb21BcGlLZXlIZWxwZXI6IHRydWUsXG4gICAgfSlcbiAgICBjb25zdCBhdXRoVG9rZW5JbmZvID0gZ2V0QXV0aFRva2VuU291cmNlKClcbiAgICByZXR1cm4gKFxuICAgICAgYXBpS2V5U291cmNlICE9PSAnbm9uZScgJiZcbiAgICAgIGF1dGhUb2tlbkluZm8uc291cmNlICE9PSAnbm9uZScgJiZcbiAgICAgICEoXG4gICAgICAgIGFwaUtleVNvdXJjZSA9PT0gJ2FwaUtleUhlbHBlcicgJiZcbiAgICAgICAgYXV0aFRva2VuSW5mby5zb3VyY2UgPT09ICdhcGlLZXlIZWxwZXInXG4gICAgICApXG4gICAgKVxuICB9LFxuICByZW5kZXI6ICgpID0+IHtcbiAgICBjb25zdCB7IHNvdXJjZTogYXBpS2V5U291cmNlIH0gPSBnZXRBbnRocm9waWNBcGlLZXlXaXRoU291cmNlKHtcbiAgICAgIHNraXBSZXRyaWV2aW5nS2V5RnJvbUFwaUtleUhlbHBlcjogdHJ1ZSxcbiAgICB9KVxuICAgIGNvbnN0IGF1dGhUb2tlbkluZm8gPSBnZXRBdXRoVG9rZW5Tb3VyY2UoKVxuICAgIHJldHVybiAoXG4gICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJjb2x1bW5cIiBtYXJnaW5Ub3A9ezF9PlxuICAgICAgICA8Qm94IGZsZXhEaXJlY3Rpb249XCJyb3dcIj5cbiAgICAgICAgICA8VGV4dCBjb2xvcj1cIndhcm5pbmdcIj57ZmlndXJlcy53YXJuaW5nfTwvVGV4dD5cbiAgICAgICAgICA8VGV4dCBjb2xvcj1cIndhcm5pbmdcIj5cbiAgICAgICAgICAgIEF1dGggY29uZmxpY3Q6IEJvdGggYSB0b2tlbiAoe2F1dGhUb2tlbkluZm8uc291cmNlfSkgYW5kIGFuIEFQSSBrZXlcbiAgICAgICAgICAgICh7YXBpS2V5U291cmNlfSkgYXJlIHNldC4gVGhpcyBtYXkgbGVhZCB0byB1bmV4cGVjdGVkIGJlaGF2aW9yLlxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICAgIDxCb3ggZmxleERpcmVjdGlvbj1cImNvbHVtblwiIG1hcmdpbkxlZnQ9ezN9PlxuICAgICAgICAgIDxUZXh0IGNvbG9yPVwid2FybmluZ1wiPlxuICAgICAgICAgICAgwrcgVHJ5aW5nIHRvIHVzZXsnICd9XG4gICAgICAgICAgICB7YXV0aFRva2VuSW5mby5zb3VyY2UgPT09ICdjbGF1ZGUuYWknXG4gICAgICAgICAgICAgID8gJ2NsYXVkZS5haSdcbiAgICAgICAgICAgICAgOiBhdXRoVG9rZW5JbmZvLnNvdXJjZX1cbiAgICAgICAgICAgID97JyAnfVxuICAgICAgICAgICAge2FwaUtleVNvdXJjZSA9PT0gJ0FOVEhST1BJQ19BUElfS0VZJ1xuICAgICAgICAgICAgICA/ICdVbnNldCB0aGUgQU5USFJPUElDX0FQSV9LRVkgZW52aXJvbm1lbnQgdmFyaWFibGUsIG9yIGNsYXVkZSAvbG9nb3V0IHRoZW4gc2F5IFwiTm9cIiB0byB0aGUgQVBJIGtleSBhcHByb3ZhbCBiZWZvcmUgbG9naW4uJ1xuICAgICAgICAgICAgICA6IGFwaUtleVNvdXJjZSA9PT0gJ2FwaUtleUhlbHBlcidcbiAgICAgICAgICAgICAgICA/ICdVbnNldCB0aGUgYXBpS2V5SGVscGVyIHNldHRpbmcuJ1xuICAgICAgICAgICAgICAgIDogJ2NsYXVkZSAvbG9nb3V0J31cbiAgICAgICAgICA8L1RleHQ+XG4gICAgICAgICAgPFRleHQgY29sb3I9XCJ3YXJuaW5nXCI+XG4gICAgICAgICAgICDCtyBUcnlpbmcgdG8gdXNlIHthcGlLZXlTb3VyY2V9P3snICd9XG4gICAgICAgICAgICB7YXV0aFRva2VuSW5mby5zb3VyY2UgPT09ICdjbGF1ZGUuYWknXG4gICAgICAgICAgICAgID8gJ2NsYXVkZSAvbG9nb3V0IHRvIHNpZ24gb3V0IG9mIGNsYXVkZS5haS4nXG4gICAgICAgICAgICAgIDogYFVuc2V0IHRoZSAke2F1dGhUb2tlbkluZm8uc291cmNlfSBlbnZpcm9ubWVudCB2YXJpYWJsZS5gfVxuICAgICAgICAgIDwvVGV4dD5cbiAgICAgICAgPC9Cb3g+XG4gICAgICA8L0JveD5cbiAgICApXG4gIH0sXG59XG5cbmNvbnN0IGxhcmdlQWdlbnREZXNjcmlwdGlvbnNOb3RpY2U6IFN0YXR1c05vdGljZURlZmluaXRpb24gPSB7XG4gIGlkOiAnbGFyZ2UtYWdlbnQtZGVzY3JpcHRpb25zJyxcbiAgdHlwZTogJ3dhcm5pbmcnLFxuICBpc0FjdGl2ZTogY29udGV4dCA9PiB7XG4gICAgY29uc3QgdG90YWxUb2tlbnMgPSBnZXRBZ2VudERlc2NyaXB0aW9uc1RvdGFsVG9rZW5zKFxuICAgICAgY29udGV4dC5hZ2VudERlZmluaXRpb25zLFxuICAgIClcbiAgICByZXR1cm4gdG90YWxUb2tlbnMgPiBBR0VOVF9ERVNDUklQVElPTlNfVEhSRVNIT0xEXG4gIH0sXG4gIHJlbmRlcjogY29udGV4dCA9PiB7XG4gICAgY29uc3QgdG90YWxUb2tlbnMgPSBnZXRBZ2VudERlc2NyaXB0aW9uc1RvdGFsVG9rZW5zKFxuICAgICAgY29udGV4dC5hZ2VudERlZmluaXRpb25zLFxuICAgIClcbiAgICByZXR1cm4gKFxuICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwicm93XCI+XG4gICAgICAgIDxUZXh0IGNvbG9yPVwid2FybmluZ1wiPntmaWd1cmVzLndhcm5pbmd9PC9UZXh0PlxuICAgICAgICA8VGV4dCBjb2xvcj1cIndhcm5pbmdcIj5cbiAgICAgICAgICBMYXJnZSBjdW11bGF0aXZlIGFnZW50IGRlc2NyaXB0aW9ucyB3aWxsIGltcGFjdCBwZXJmb3JtYW5jZSAoflxuICAgICAgICAgIHtmb3JtYXROdW1iZXIodG90YWxUb2tlbnMpfSB0b2tlbnMgJmd0O3snICd9XG4gICAgICAgICAge2Zvcm1hdE51bWJlcihBR0VOVF9ERVNDUklQVElPTlNfVEhSRVNIT0xEKX0pXG4gICAgICAgICAgPFRleHQgZGltQ29sb3I+IMK3IC9hZ2VudHMgdG8gbWFuYWdlPC9UZXh0PlxuICAgICAgICA8L1RleHQ+XG4gICAgICA8L0JveD5cbiAgICApXG4gIH0sXG59XG5cbmNvbnN0IGpldGJyYWluc1BsdWdpbk5vdGljZTogU3RhdHVzTm90aWNlRGVmaW5pdGlvbiA9IHtcbiAgaWQ6ICdqZXRicmFpbnMtcGx1Z2luLWluc3RhbGwnLFxuICB0eXBlOiAnaW5mbycsXG4gIGlzQWN0aXZlOiBjb250ZXh0ID0+IHtcbiAgICAvLyBPbmx5IHNob3cgaWYgcnVubmluZyBpbiBKZXRCcmFpbnMgYnVpbHQtaW4gdGVybWluYWxcbiAgICBpZiAoIWlzU3VwcG9ydGVkSmV0QnJhaW5zVGVybWluYWwoKSkge1xuICAgICAgcmV0dXJuIGZhbHNlXG4gICAgfVxuICAgIC8vIERvbid0IHNob3cgaWYgYXV0by1pbnN0YWxsIGlzIGRpc2FibGVkXG4gICAgY29uc3Qgc2hvdWxkQXV0b0luc3RhbGwgPSBjb250ZXh0LmNvbmZpZy5hdXRvSW5zdGFsbElkZUV4dGVuc2lvbiA/PyB0cnVlXG4gICAgaWYgKCFzaG91bGRBdXRvSW5zdGFsbCkge1xuICAgICAgcmV0dXJuIGZhbHNlXG4gICAgfVxuICAgIC8vIENoZWNrIGlmIHBsdWdpbiBpcyBhbHJlYWR5IGluc3RhbGxlZCAoY2FjaGVkIHRvIGF2b2lkIHJlcGVhdGVkIGZpbGVzeXN0ZW0gY2hlY2tzKVxuICAgIGNvbnN0IGlkZVR5cGUgPSBnZXRUZXJtaW5hbElkZVR5cGUoKVxuICAgIHJldHVybiBpZGVUeXBlICE9PSBudWxsICYmICFpc0pldEJyYWluc1BsdWdpbkluc3RhbGxlZENhY2hlZFN5bmMoaWRlVHlwZSlcbiAgfSxcbiAgcmVuZGVyOiAoKSA9PiB7XG4gICAgY29uc3QgaWRlVHlwZSA9IGdldFRlcm1pbmFsSWRlVHlwZSgpXG4gICAgY29uc3QgaWRlTmFtZSA9IHRvSURFRGlzcGxheU5hbWUoaWRlVHlwZSlcbiAgICByZXR1cm4gKFxuICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwicm93XCIgZ2FwPXsxfSBtYXJnaW5MZWZ0PXsxfT5cbiAgICAgICAgPFRleHQgY29sb3I9XCJpZGVcIj57ZmlndXJlcy5hcnJvd1VwfTwvVGV4dD5cbiAgICAgICAgPFRleHQ+XG4gICAgICAgICAgSW5zdGFsbCB0aGUgPFRleHQgY29sb3I9XCJpZGVcIj57aWRlTmFtZX08L1RleHQ+IHBsdWdpbiBmcm9tIHRoZVxuICAgICAgICAgIEpldEJyYWlucyBNYXJrZXRwbGFjZTp7JyAnfVxuICAgICAgICAgIDxUZXh0IGJvbGQ+aHR0cHM6Ly9kb2NzLmNsYXVkZS5jb20vcy9jbGF1ZGUtY29kZS1qZXRicmFpbnM8L1RleHQ+XG4gICAgICAgIDwvVGV4dD5cbiAgICAgIDwvQm94PlxuICAgIClcbiAgfSxcbn1cblxuXG4vLyBBbGwgbm90aWNlIGRlZmluaXRpb25zXG5leHBvcnQgY29uc3Qgc3RhdHVzTm90aWNlRGVmaW5pdGlvbnM6IFN0YXR1c05vdGljZURlZmluaXRpb25bXSA9IFtcbiAgbGFyZ2VNZW1vcnlGaWxlc05vdGljZSxcbiAgbGFyZ2VBZ2VudERlc2NyaXB0aW9uc05vdGljZSxcbiAgY2xhdWRlQWlTdWJzY3JpYmVyRXh0ZXJuYWxUb2tlbk5vdGljZSxcbiAgYXBpS2V5Q29uZmxpY3ROb3RpY2UsXG4gIGJvdGhBdXRoTWV0aG9kc05vdGljZSxcbiAgamV0YnJhaW5zUGx1Z2luTm90aWNlLFxuXVxuXG4vLyBIZWxwZXIgZnVuY3Rpb25zIGZvciBleHRlcm5hbCB1c2VcbmV4cG9ydCBmdW5jdGlvbiBnZXRBY3RpdmVOb3RpY2VzKFxuICBjb250ZXh0OiBTdGF0dXNOb3RpY2VDb250ZXh0LFxuKTogU3RhdHVzTm90aWNlRGVmaW5pdGlvbltdIHtcbiAgcmV0dXJuIHN0YXR1c05vdGljZURlZmluaXRpb25zLmZpbHRlcihub3RpY2UgPT4gbm90aWNlLmlzQWN0aXZlKGNvbnRleHQpKVxufVxuIl0sIm1hcHBpbmdzIjoiQUFBQTtBQUNBLFNBQVNBLEdBQUcsRUFBRUMsSUFBSSxRQUFRLFdBQVc7QUFDckMsT0FBTyxLQUFLQyxLQUFLLE1BQU0sT0FBTztBQUM5QixTQUNFQyxtQkFBbUIsRUFDbkJDLDBCQUEwQixFQUMxQixLQUFLQyxjQUFjLFFBQ2QsZUFBZTtBQUN0QixPQUFPQyxPQUFPLE1BQU0sU0FBUztBQUM3QixTQUFTQyxNQUFNLFFBQVEsVUFBVTtBQUNqQyxTQUFTQyxRQUFRLFFBQVEsTUFBTTtBQUMvQixTQUFTQyxZQUFZLFFBQVEsYUFBYTtBQUMxQyxjQUFjQyxlQUFlLFFBQVEsYUFBYTtBQUNsRCxTQUNFQyw0QkFBNEIsRUFDNUJDLGtDQUFrQyxFQUNsQ0Msa0JBQWtCLEVBQ2xCQyxvQkFBb0IsUUFDZixXQUFXO0FBQ2xCLGNBQWNDLHNCQUFzQixRQUFRLHFDQUFxQztBQUNqRixTQUNFQywrQkFBK0IsRUFDL0JDLDRCQUE0QixRQUN2QiwwQkFBMEI7QUFDakMsU0FDRUMsNEJBQTRCLEVBQzVCQyxnQkFBZ0IsRUFDaEJDLGtCQUFrQixRQUNiLFVBQVU7QUFDakIsU0FBU0Msb0NBQW9DLFFBQVEsZ0JBQWdCOztBQUVyRTtBQUNBLE9BQU8sS0FBS0MsZ0JBQWdCLEdBQUcsU0FBUyxHQUFHLE1BQU07QUFFakQsT0FBTyxLQUFLQyxtQkFBbUIsR0FBRztFQUNoQ0MsTUFBTSxFQUFFQyxVQUFVLENBQUMsT0FBT2YsZUFBZSxDQUFDO0VBQzFDZ0IsZ0JBQWdCLENBQUMsRUFBRVgsc0JBQXNCO0VBQ3pDWSxXQUFXLEVBQUV0QixjQUFjLEVBQUU7QUFDL0IsQ0FBQztBQUVELE9BQU8sS0FBS3VCLHNCQUFzQixHQUFHO0VBQ25DQyxFQUFFLEVBQUUsTUFBTTtFQUNWQyxJQUFJLEVBQUVSLGdCQUFnQjtFQUN0QlMsUUFBUSxFQUFFLENBQUND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