@shareai-lab/kode 1.0.9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (286) hide show
  1. package/LICENSE +201 -0
  2. package/README.md +426 -0
  3. package/README.zh-CN.md +326 -0
  4. package/cli.js +79 -0
  5. package/package.json +119 -0
  6. package/scripts/postinstall.js +18 -0
  7. package/src/ProjectOnboarding.tsx +198 -0
  8. package/src/Tool.ts +82 -0
  9. package/src/commands/agents.tsx +3410 -0
  10. package/src/commands/approvedTools.ts +53 -0
  11. package/src/commands/bug.tsx +20 -0
  12. package/src/commands/clear.ts +43 -0
  13. package/src/commands/compact.ts +120 -0
  14. package/src/commands/config.tsx +19 -0
  15. package/src/commands/cost.ts +18 -0
  16. package/src/commands/ctx_viz.ts +209 -0
  17. package/src/commands/doctor.ts +24 -0
  18. package/src/commands/help.tsx +19 -0
  19. package/src/commands/init.ts +37 -0
  20. package/src/commands/listen.ts +42 -0
  21. package/src/commands/login.tsx +51 -0
  22. package/src/commands/logout.tsx +40 -0
  23. package/src/commands/mcp.ts +41 -0
  24. package/src/commands/model.tsx +40 -0
  25. package/src/commands/modelstatus.tsx +20 -0
  26. package/src/commands/onboarding.tsx +34 -0
  27. package/src/commands/pr_comments.ts +59 -0
  28. package/src/commands/refreshCommands.ts +54 -0
  29. package/src/commands/release-notes.ts +34 -0
  30. package/src/commands/resume.tsx +31 -0
  31. package/src/commands/review.ts +49 -0
  32. package/src/commands/terminalSetup.ts +221 -0
  33. package/src/commands.ts +139 -0
  34. package/src/components/ApproveApiKey.tsx +93 -0
  35. package/src/components/AsciiLogo.tsx +13 -0
  36. package/src/components/AutoUpdater.tsx +148 -0
  37. package/src/components/Bug.tsx +367 -0
  38. package/src/components/Config.tsx +293 -0
  39. package/src/components/ConsoleOAuthFlow.tsx +327 -0
  40. package/src/components/Cost.tsx +23 -0
  41. package/src/components/CostThresholdDialog.tsx +46 -0
  42. package/src/components/CustomSelect/option-map.ts +42 -0
  43. package/src/components/CustomSelect/select-option.tsx +78 -0
  44. package/src/components/CustomSelect/select.tsx +152 -0
  45. package/src/components/CustomSelect/theme.ts +45 -0
  46. package/src/components/CustomSelect/use-select-state.ts +414 -0
  47. package/src/components/CustomSelect/use-select.ts +35 -0
  48. package/src/components/FallbackToolUseRejectedMessage.tsx +15 -0
  49. package/src/components/FileEditToolUpdatedMessage.tsx +66 -0
  50. package/src/components/Help.tsx +215 -0
  51. package/src/components/HighlightedCode.tsx +33 -0
  52. package/src/components/InvalidConfigDialog.tsx +113 -0
  53. package/src/components/Link.tsx +32 -0
  54. package/src/components/LogSelector.tsx +86 -0
  55. package/src/components/Logo.tsx +170 -0
  56. package/src/components/MCPServerApprovalDialog.tsx +100 -0
  57. package/src/components/MCPServerDialogCopy.tsx +25 -0
  58. package/src/components/MCPServerMultiselectDialog.tsx +109 -0
  59. package/src/components/Message.tsx +221 -0
  60. package/src/components/MessageResponse.tsx +15 -0
  61. package/src/components/MessageSelector.tsx +211 -0
  62. package/src/components/ModeIndicator.tsx +88 -0
  63. package/src/components/ModelConfig.tsx +301 -0
  64. package/src/components/ModelListManager.tsx +227 -0
  65. package/src/components/ModelSelector.tsx +3387 -0
  66. package/src/components/ModelStatusDisplay.tsx +230 -0
  67. package/src/components/Onboarding.tsx +274 -0
  68. package/src/components/PressEnterToContinue.tsx +11 -0
  69. package/src/components/PromptInput.tsx +760 -0
  70. package/src/components/SentryErrorBoundary.ts +39 -0
  71. package/src/components/Spinner.tsx +129 -0
  72. package/src/components/StickerRequestForm.tsx +16 -0
  73. package/src/components/StructuredDiff.tsx +191 -0
  74. package/src/components/TextInput.tsx +259 -0
  75. package/src/components/TodoItem.tsx +47 -0
  76. package/src/components/TokenWarning.tsx +31 -0
  77. package/src/components/ToolUseLoader.tsx +40 -0
  78. package/src/components/TrustDialog.tsx +106 -0
  79. package/src/components/binary-feedback/BinaryFeedback.tsx +63 -0
  80. package/src/components/binary-feedback/BinaryFeedbackOption.tsx +111 -0
  81. package/src/components/binary-feedback/BinaryFeedbackView.tsx +172 -0
  82. package/src/components/binary-feedback/utils.ts +220 -0
  83. package/src/components/messages/AssistantBashOutputMessage.tsx +22 -0
  84. package/src/components/messages/AssistantLocalCommandOutputMessage.tsx +49 -0
  85. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +19 -0
  86. package/src/components/messages/AssistantTextMessage.tsx +144 -0
  87. package/src/components/messages/AssistantThinkingMessage.tsx +40 -0
  88. package/src/components/messages/AssistantToolUseMessage.tsx +133 -0
  89. package/src/components/messages/TaskProgressMessage.tsx +32 -0
  90. package/src/components/messages/TaskToolMessage.tsx +58 -0
  91. package/src/components/messages/UserBashInputMessage.tsx +28 -0
  92. package/src/components/messages/UserCommandMessage.tsx +30 -0
  93. package/src/components/messages/UserKodingInputMessage.tsx +28 -0
  94. package/src/components/messages/UserPromptMessage.tsx +35 -0
  95. package/src/components/messages/UserTextMessage.tsx +39 -0
  96. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +12 -0
  97. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +36 -0
  98. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +31 -0
  99. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +57 -0
  100. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +35 -0
  101. package/src/components/messages/UserToolResultMessage/utils.tsx +56 -0
  102. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +121 -0
  103. package/src/components/permissions/FallbackPermissionRequest.tsx +153 -0
  104. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +182 -0
  105. package/src/components/permissions/FileEditPermissionRequest/FileEditToolDiff.tsx +77 -0
  106. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +164 -0
  107. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +83 -0
  108. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +240 -0
  109. package/src/components/permissions/PermissionRequest.tsx +101 -0
  110. package/src/components/permissions/PermissionRequestTitle.tsx +69 -0
  111. package/src/components/permissions/hooks.ts +44 -0
  112. package/src/components/permissions/toolUseOptions.ts +59 -0
  113. package/src/components/permissions/utils.ts +23 -0
  114. package/src/constants/betas.ts +5 -0
  115. package/src/constants/claude-asterisk-ascii-art.tsx +238 -0
  116. package/src/constants/figures.ts +4 -0
  117. package/src/constants/keys.ts +3 -0
  118. package/src/constants/macros.ts +8 -0
  119. package/src/constants/modelCapabilities.ts +179 -0
  120. package/src/constants/models.ts +1025 -0
  121. package/src/constants/oauth.ts +18 -0
  122. package/src/constants/product.ts +17 -0
  123. package/src/constants/prompts.ts +168 -0
  124. package/src/constants/releaseNotes.ts +7 -0
  125. package/src/context/PermissionContext.tsx +149 -0
  126. package/src/context.ts +278 -0
  127. package/src/cost-tracker.ts +84 -0
  128. package/src/entrypoints/cli.tsx +1542 -0
  129. package/src/entrypoints/mcp.ts +176 -0
  130. package/src/history.ts +25 -0
  131. package/src/hooks/useApiKeyVerification.ts +59 -0
  132. package/src/hooks/useArrowKeyHistory.ts +55 -0
  133. package/src/hooks/useCanUseTool.ts +138 -0
  134. package/src/hooks/useCancelRequest.ts +39 -0
  135. package/src/hooks/useDoublePress.ts +42 -0
  136. package/src/hooks/useExitOnCtrlCD.ts +31 -0
  137. package/src/hooks/useInterval.ts +25 -0
  138. package/src/hooks/useLogMessages.ts +16 -0
  139. package/src/hooks/useLogStartupTime.ts +12 -0
  140. package/src/hooks/useNotifyAfterTimeout.ts +65 -0
  141. package/src/hooks/usePermissionRequestLogging.ts +44 -0
  142. package/src/hooks/useTerminalSize.ts +49 -0
  143. package/src/hooks/useTextInput.ts +318 -0
  144. package/src/hooks/useUnifiedCompletion.ts +1405 -0
  145. package/src/messages.ts +38 -0
  146. package/src/permissions.ts +268 -0
  147. package/src/query.ts +715 -0
  148. package/src/screens/ConfigureNpmPrefix.tsx +197 -0
  149. package/src/screens/Doctor.tsx +219 -0
  150. package/src/screens/LogList.tsx +68 -0
  151. package/src/screens/REPL.tsx +809 -0
  152. package/src/screens/ResumeConversation.tsx +68 -0
  153. package/src/services/adapters/base.ts +38 -0
  154. package/src/services/adapters/chatCompletions.ts +90 -0
  155. package/src/services/adapters/responsesAPI.ts +170 -0
  156. package/src/services/browserMocks.ts +66 -0
  157. package/src/services/claude.ts +2197 -0
  158. package/src/services/customCommands.ts +704 -0
  159. package/src/services/fileFreshness.ts +377 -0
  160. package/src/services/gpt5ConnectionTest.ts +340 -0
  161. package/src/services/mcpClient.ts +564 -0
  162. package/src/services/mcpServerApproval.tsx +50 -0
  163. package/src/services/mentionProcessor.ts +273 -0
  164. package/src/services/modelAdapterFactory.ts +69 -0
  165. package/src/services/notifier.ts +40 -0
  166. package/src/services/oauth.ts +357 -0
  167. package/src/services/openai.ts +1338 -0
  168. package/src/services/responseStateManager.ts +90 -0
  169. package/src/services/sentry.ts +3 -0
  170. package/src/services/statsig.ts +172 -0
  171. package/src/services/statsigStorage.ts +86 -0
  172. package/src/services/systemReminder.ts +507 -0
  173. package/src/services/vcr.ts +161 -0
  174. package/src/test/testAdapters.ts +96 -0
  175. package/src/tools/ArchitectTool/ArchitectTool.tsx +122 -0
  176. package/src/tools/ArchitectTool/prompt.ts +15 -0
  177. package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +569 -0
  178. package/src/tools/BashTool/BashTool.tsx +243 -0
  179. package/src/tools/BashTool/BashToolResultMessage.tsx +38 -0
  180. package/src/tools/BashTool/OutputLine.tsx +49 -0
  181. package/src/tools/BashTool/prompt.ts +174 -0
  182. package/src/tools/BashTool/utils.ts +56 -0
  183. package/src/tools/FileEditTool/FileEditTool.tsx +315 -0
  184. package/src/tools/FileEditTool/prompt.ts +51 -0
  185. package/src/tools/FileEditTool/utils.ts +58 -0
  186. package/src/tools/FileReadTool/FileReadTool.tsx +404 -0
  187. package/src/tools/FileReadTool/prompt.ts +7 -0
  188. package/src/tools/FileWriteTool/FileWriteTool.tsx +297 -0
  189. package/src/tools/FileWriteTool/prompt.ts +10 -0
  190. package/src/tools/GlobTool/GlobTool.tsx +119 -0
  191. package/src/tools/GlobTool/prompt.ts +8 -0
  192. package/src/tools/GrepTool/GrepTool.tsx +147 -0
  193. package/src/tools/GrepTool/prompt.ts +11 -0
  194. package/src/tools/MCPTool/MCPTool.tsx +107 -0
  195. package/src/tools/MCPTool/prompt.ts +3 -0
  196. package/src/tools/MemoryReadTool/MemoryReadTool.tsx +127 -0
  197. package/src/tools/MemoryReadTool/prompt.ts +3 -0
  198. package/src/tools/MemoryWriteTool/MemoryWriteTool.tsx +89 -0
  199. package/src/tools/MemoryWriteTool/prompt.ts +3 -0
  200. package/src/tools/MultiEditTool/MultiEditTool.tsx +366 -0
  201. package/src/tools/MultiEditTool/prompt.ts +45 -0
  202. package/src/tools/NotebookEditTool/NotebookEditTool.tsx +298 -0
  203. package/src/tools/NotebookEditTool/prompt.ts +3 -0
  204. package/src/tools/NotebookReadTool/NotebookReadTool.tsx +258 -0
  205. package/src/tools/NotebookReadTool/prompt.ts +3 -0
  206. package/src/tools/StickerRequestTool/StickerRequestTool.tsx +93 -0
  207. package/src/tools/StickerRequestTool/prompt.ts +19 -0
  208. package/src/tools/TaskTool/TaskTool.tsx +466 -0
  209. package/src/tools/TaskTool/constants.ts +1 -0
  210. package/src/tools/TaskTool/prompt.ts +92 -0
  211. package/src/tools/ThinkTool/ThinkTool.tsx +54 -0
  212. package/src/tools/ThinkTool/prompt.ts +12 -0
  213. package/src/tools/TodoWriteTool/TodoWriteTool.tsx +313 -0
  214. package/src/tools/TodoWriteTool/prompt.ts +63 -0
  215. package/src/tools/URLFetcherTool/URLFetcherTool.tsx +178 -0
  216. package/src/tools/URLFetcherTool/cache.ts +55 -0
  217. package/src/tools/URLFetcherTool/htmlToMarkdown.ts +55 -0
  218. package/src/tools/URLFetcherTool/prompt.ts +17 -0
  219. package/src/tools/WebSearchTool/WebSearchTool.tsx +103 -0
  220. package/src/tools/WebSearchTool/prompt.ts +13 -0
  221. package/src/tools/WebSearchTool/searchProviders.ts +66 -0
  222. package/src/tools/lsTool/lsTool.tsx +272 -0
  223. package/src/tools/lsTool/prompt.ts +2 -0
  224. package/src/tools.ts +67 -0
  225. package/src/types/PermissionMode.ts +120 -0
  226. package/src/types/RequestContext.ts +72 -0
  227. package/src/types/conversation.ts +51 -0
  228. package/src/types/logs.ts +58 -0
  229. package/src/types/modelCapabilities.ts +64 -0
  230. package/src/types/notebook.ts +87 -0
  231. package/src/utils/Cursor.ts +436 -0
  232. package/src/utils/PersistentShell.ts +552 -0
  233. package/src/utils/advancedFuzzyMatcher.ts +290 -0
  234. package/src/utils/agentLoader.ts +278 -0
  235. package/src/utils/agentStorage.ts +97 -0
  236. package/src/utils/array.ts +3 -0
  237. package/src/utils/ask.tsx +99 -0
  238. package/src/utils/auth.ts +13 -0
  239. package/src/utils/autoCompactCore.ts +223 -0
  240. package/src/utils/autoUpdater.ts +458 -0
  241. package/src/utils/betas.ts +20 -0
  242. package/src/utils/browser.ts +14 -0
  243. package/src/utils/cleanup.ts +72 -0
  244. package/src/utils/commands.ts +261 -0
  245. package/src/utils/commonUnixCommands.ts +161 -0
  246. package/src/utils/config.ts +945 -0
  247. package/src/utils/conversationRecovery.ts +55 -0
  248. package/src/utils/debugLogger.ts +1235 -0
  249. package/src/utils/diff.ts +42 -0
  250. package/src/utils/env.ts +57 -0
  251. package/src/utils/errors.ts +21 -0
  252. package/src/utils/exampleCommands.ts +109 -0
  253. package/src/utils/execFileNoThrow.ts +51 -0
  254. package/src/utils/expertChatStorage.ts +136 -0
  255. package/src/utils/file.ts +405 -0
  256. package/src/utils/fileRecoveryCore.ts +71 -0
  257. package/src/utils/format.tsx +44 -0
  258. package/src/utils/fuzzyMatcher.ts +328 -0
  259. package/src/utils/generators.ts +62 -0
  260. package/src/utils/git.ts +92 -0
  261. package/src/utils/globalLogger.ts +77 -0
  262. package/src/utils/http.ts +10 -0
  263. package/src/utils/imagePaste.ts +38 -0
  264. package/src/utils/json.ts +13 -0
  265. package/src/utils/log.ts +382 -0
  266. package/src/utils/markdown.ts +213 -0
  267. package/src/utils/messageContextManager.ts +289 -0
  268. package/src/utils/messages.tsx +939 -0
  269. package/src/utils/model.ts +914 -0
  270. package/src/utils/permissions/filesystem.ts +127 -0
  271. package/src/utils/responseState.ts +23 -0
  272. package/src/utils/ripgrep.ts +167 -0
  273. package/src/utils/secureFile.ts +564 -0
  274. package/src/utils/sessionState.ts +49 -0
  275. package/src/utils/state.ts +25 -0
  276. package/src/utils/style.ts +29 -0
  277. package/src/utils/terminal.ts +50 -0
  278. package/src/utils/theme.ts +127 -0
  279. package/src/utils/thinking.ts +144 -0
  280. package/src/utils/todoStorage.ts +431 -0
  281. package/src/utils/tokens.ts +43 -0
  282. package/src/utils/toolExecutionController.ts +163 -0
  283. package/src/utils/unaryLogging.ts +26 -0
  284. package/src/utils/user.ts +37 -0
  285. package/src/utils/validate.ts +165 -0
  286. package/yoga.wasm +0 -0
@@ -0,0 +1,139 @@
1
+ import React from 'react'
2
+ import bug from './commands/bug'
3
+ import clear from './commands/clear'
4
+ import compact from './commands/compact'
5
+ import config from './commands/config'
6
+ import cost from './commands/cost'
7
+ import ctx_viz from './commands/ctx_viz'
8
+ import doctor from './commands/doctor'
9
+ import help from './commands/help'
10
+ import init from './commands/init'
11
+ import listen from './commands/listen'
12
+ import login from './commands/login'
13
+ import logout from './commands/logout'
14
+ import mcp from './commands/mcp'
15
+ import * as model from './commands/model'
16
+ import modelstatus from './commands/modelstatus'
17
+ import onboarding from './commands/onboarding'
18
+ import pr_comments from './commands/pr_comments'
19
+ import refreshCommands from './commands/refreshCommands'
20
+ import releaseNotes from './commands/release-notes'
21
+ import review from './commands/review'
22
+ import terminalSetup from './commands/terminalSetup'
23
+ import { Tool, ToolUseContext } from './Tool'
24
+ import resume from './commands/resume'
25
+ import agents from './commands/agents'
26
+ import { getMCPCommands } from './services/mcpClient'
27
+ import { loadCustomCommands } from './services/customCommands'
28
+ import type { MessageParam } from '@anthropic-ai/sdk/resources/index.mjs'
29
+ import { memoize } from 'lodash-es'
30
+ import type { Message } from './query'
31
+ import { isAnthropicAuthEnabled } from './utils/auth'
32
+
33
+ type PromptCommand = {
34
+ type: 'prompt'
35
+ progressMessage: string
36
+ argNames?: string[]
37
+ getPromptForCommand(args: string): Promise<MessageParam[]>
38
+ }
39
+
40
+ type LocalCommand = {
41
+ type: 'local'
42
+ call(
43
+ args: string,
44
+ context: {
45
+ options: {
46
+ commands: Command[]
47
+ tools: Tool[]
48
+ slowAndCapableModel: string
49
+ }
50
+ abortController: AbortController
51
+ setForkConvoWithMessagesOnTheNextRender: (
52
+ forkConvoWithMessages: Message[],
53
+ ) => void
54
+ },
55
+ ): Promise<string>
56
+ }
57
+
58
+ type LocalJSXCommand = {
59
+ type: 'local-jsx'
60
+ call(
61
+ onDone: (result?: string) => void,
62
+ context: ToolUseContext & {
63
+ setForkConvoWithMessagesOnTheNextRender: (
64
+ forkConvoWithMessages: Message[],
65
+ ) => void
66
+ },
67
+ ): Promise<React.ReactNode>
68
+ }
69
+
70
+ export type Command = {
71
+ description: string
72
+ isEnabled: boolean
73
+ isHidden: boolean
74
+ name: string
75
+ aliases?: string[]
76
+ userFacingName(): string
77
+ } & (PromptCommand | LocalCommand | LocalJSXCommand)
78
+
79
+ const INTERNAL_ONLY_COMMANDS = [ctx_viz, resume, listen]
80
+
81
+ // Declared as a function so that we don't run this until getCommands is called,
82
+ // since underlying functions read from config, which can't be read at module initialization time
83
+ const COMMANDS = memoize((): Command[] => [
84
+ agents,
85
+ clear,
86
+ compact,
87
+ config,
88
+ cost,
89
+ doctor,
90
+ help,
91
+ init,
92
+ mcp,
93
+ model,
94
+ modelstatus,
95
+ onboarding,
96
+ pr_comments,
97
+ refreshCommands,
98
+ releaseNotes,
99
+ bug,
100
+ review,
101
+ terminalSetup,
102
+ ...(isAnthropicAuthEnabled() ? [logout, login()] : []),
103
+ ...INTERNAL_ONLY_COMMANDS,
104
+ ])
105
+
106
+ export const getCommands = memoize(async (): Promise<Command[]> => {
107
+ const [mcpCommands, customCommands] = await Promise.all([
108
+ getMCPCommands(),
109
+ loadCustomCommands(),
110
+ ])
111
+
112
+ return [...mcpCommands, ...customCommands, ...COMMANDS()].filter(
113
+ _ => _.isEnabled,
114
+ )
115
+ })
116
+
117
+ export function hasCommand(commandName: string, commands: Command[]): boolean {
118
+ return commands.some(
119
+ _ => _.userFacingName() === commandName || _.aliases?.includes(commandName),
120
+ )
121
+ }
122
+
123
+ export function getCommand(commandName: string, commands: Command[]): Command {
124
+ const command = commands.find(
125
+ _ => _.userFacingName() === commandName || _.aliases?.includes(commandName),
126
+ ) as Command | undefined
127
+ if (!command) {
128
+ throw ReferenceError(
129
+ `Command ${commandName} not found. Available commands: ${commands
130
+ .map(_ => {
131
+ const name = _.userFacingName()
132
+ return _.aliases ? `${name} (aliases: ${_.aliases.join(', ')})` : name
133
+ })
134
+ .join(', ')}`,
135
+ )
136
+ }
137
+
138
+ return command
139
+ }
@@ -0,0 +1,93 @@
1
+ import React from 'react'
2
+ import { Box, Text } from 'ink'
3
+ import { getGlobalConfig, saveGlobalConfig } from '../utils/config'
4
+ import { getTheme } from '../utils/theme'
5
+ import { Select } from './CustomSelect/select'
6
+ import { useExitOnCtrlCD } from '../hooks/useExitOnCtrlCD'
7
+ import chalk from 'chalk'
8
+
9
+ type Props = {
10
+ customApiKeyTruncated: string
11
+ onDone(): void
12
+ }
13
+
14
+ export function ApproveApiKey({
15
+ customApiKeyTruncated,
16
+ onDone,
17
+ }: Props): React.ReactNode {
18
+ const theme = getTheme()
19
+
20
+ function onChange(value: 'yes' | 'no') {
21
+ const config = getGlobalConfig()
22
+ switch (value) {
23
+ case 'yes': {
24
+ saveGlobalConfig({
25
+ ...config,
26
+ customApiKeyResponses: {
27
+ ...config.customApiKeyResponses,
28
+ approved: [
29
+ ...(config.customApiKeyResponses?.approved ?? []),
30
+ customApiKeyTruncated,
31
+ ],
32
+ },
33
+ })
34
+ onDone()
35
+ break
36
+ }
37
+ case 'no': {
38
+ saveGlobalConfig({
39
+ ...config,
40
+ customApiKeyResponses: {
41
+ ...config.customApiKeyResponses,
42
+ rejected: [
43
+ ...(config.customApiKeyResponses?.rejected ?? []),
44
+ customApiKeyTruncated,
45
+ ],
46
+ },
47
+ })
48
+ onDone()
49
+ break
50
+ }
51
+ }
52
+ }
53
+
54
+ const exitState = useExitOnCtrlCD(() => process.exit(0))
55
+
56
+ return (
57
+ <>
58
+ <Box
59
+ flexDirection="column"
60
+ gap={1}
61
+ padding={1}
62
+ borderStyle="round"
63
+ borderColor={theme.warning}
64
+ >
65
+ <Text bold color={theme.warning}>
66
+ Detected a custom API key in your environment
67
+ </Text>
68
+ <Text>
69
+ Your environment sets{' '}
70
+ <Text color={theme.warning}>ANTHROPIC_API_KEY</Text>:{' '}
71
+ <Text bold>sk-ant-...{customApiKeyTruncated}</Text>
72
+ </Text>
73
+ <Text>Do you want to use this API key?</Text>
74
+ <Select
75
+ options={[
76
+ { label: `No (${chalk.bold('recommended')})`, value: 'no' },
77
+ { label: 'Yes', value: 'yes' },
78
+ ]}
79
+ onChange={value => onChange(value as 'yes' | 'no')}
80
+ />
81
+ </Box>
82
+ <Box marginLeft={3}>
83
+ <Text dimColor>
84
+ {exitState.pending ? (
85
+ <>Press {exitState.keyName} again to exit</>
86
+ ) : (
87
+ <>Enter to confirm</>
88
+ )}
89
+ </Text>
90
+ </Box>
91
+ </>
92
+ )
93
+ }
@@ -0,0 +1,13 @@
1
+ import { Box, Text } from 'ink'
2
+ import React from 'react'
3
+ import { getTheme } from '../utils/theme'
4
+ import { ASCII_LOGO } from '../constants/product'
5
+
6
+ export function AsciiLogo(): React.ReactNode {
7
+ const theme = getTheme()
8
+ return (
9
+ <Box flexDirection="column" alignItems="flex-start">
10
+ <Text color={theme.kode}>{ASCII_LOGO}</Text>
11
+ </Box>
12
+ )
13
+ }
@@ -0,0 +1,148 @@
1
+ import { Box, Text } from 'ink'
2
+ import * as React from 'react'
3
+ import { getTheme } from '../utils/theme'
4
+ import { gte } from 'semver'
5
+ import { useEffect, useState } from 'react'
6
+ import { isAutoUpdaterDisabled } from '../utils/config'
7
+ import {
8
+ AutoUpdaterResult,
9
+ getLatestVersion,
10
+ installGlobalPackage,
11
+ } from '../utils/autoUpdater.js'
12
+ import { useInterval } from '../hooks/useInterval'
13
+ import { logEvent } from '../services/statsig'
14
+ import { MACRO } from '../constants/macros'
15
+ import { PRODUCT_COMMAND } from '../constants/product'
16
+ type Props = {
17
+ debug: boolean
18
+ isUpdating: boolean
19
+ onChangeIsUpdating: (isUpdating: boolean) => void
20
+ onAutoUpdaterResult: (autoUpdaterResult: AutoUpdaterResult) => void
21
+ autoUpdaterResult: AutoUpdaterResult | null
22
+ }
23
+
24
+ export function AutoUpdater({
25
+ debug,
26
+ isUpdating,
27
+ onChangeIsUpdating,
28
+ onAutoUpdaterResult,
29
+ autoUpdaterResult,
30
+ }: Props): React.ReactNode {
31
+ const theme = getTheme()
32
+ const [versions, setVersions] = useState<{
33
+ global?: string | null
34
+ latest?: string | null
35
+ }>({})
36
+ const checkForUpdates = React.useCallback(async () => {
37
+ if (process.env.NODE_ENV === 'test' || process.env.NODE_ENV === 'dev') {
38
+ return
39
+ }
40
+
41
+ if (isUpdating) {
42
+ return
43
+ }
44
+
45
+ // Get versions
46
+ const globalVersion = MACRO.VERSION
47
+ const latestVersion = await getLatestVersion()
48
+ const isDisabled = true //await isAutoUpdaterDisabled()
49
+
50
+ setVersions({ global: globalVersion, latest: latestVersion })
51
+
52
+ // Check if update needed and perform update
53
+ if (
54
+ !isDisabled &&
55
+ globalVersion &&
56
+ latestVersion &&
57
+ !gte(globalVersion, latestVersion)
58
+ ) {
59
+ const startTime = Date.now()
60
+ onChangeIsUpdating(true)
61
+ const installStatus = await installGlobalPackage()
62
+ onChangeIsUpdating(false)
63
+
64
+ if (installStatus === 'success') {
65
+ logEvent('tengu_auto_updater_success', {
66
+ fromVersion: globalVersion,
67
+ toVersion: latestVersion,
68
+ durationMs: String(Date.now() - startTime),
69
+ })
70
+ } else {
71
+ logEvent('tengu_auto_updater_fail', {
72
+ fromVersion: globalVersion,
73
+ attemptedVersion: latestVersion,
74
+ status: installStatus,
75
+ durationMs: String(Date.now() - startTime),
76
+ })
77
+ }
78
+
79
+ onAutoUpdaterResult({
80
+ version: latestVersion!,
81
+ status: installStatus,
82
+ })
83
+ }
84
+ // Don't re-render when isUpdating changes
85
+ // TODO: Find a cleaner way to do this
86
+ // eslint-disable-next-line react-hooks/exhaustive-deps
87
+ }, [onAutoUpdaterResult])
88
+
89
+ // Initial check
90
+ useEffect(() => {
91
+ // checkForUpdates()
92
+ }, [checkForUpdates])
93
+
94
+ // Check every 30 minutes
95
+ // useInterval(checkForUpdates, 30 * 60 * 1000)
96
+
97
+ if (debug) {
98
+ return (
99
+ <Box flexDirection="row">
100
+ <Text dimColor>
101
+ globalVersion: {versions.global} &middot; latestVersion:{' '}
102
+ {versions.latest}
103
+ </Text>
104
+ </Box>
105
+ )
106
+ }
107
+
108
+ if (!autoUpdaterResult?.version && (!versions.global || !versions.latest)) {
109
+ return null
110
+ }
111
+
112
+ if (!autoUpdaterResult?.version && !isUpdating) {
113
+ return null
114
+ }
115
+
116
+ return (
117
+ <Box flexDirection="row">
118
+ {debug && (
119
+ <Text dimColor>
120
+ globalVersion: {versions.global} &middot; latestVersion:{' '}
121
+ {versions.latest}
122
+ </Text>
123
+ )}
124
+ {isUpdating && (
125
+ <>
126
+ <Box>
127
+ <Text color={theme.secondaryText} dimColor wrap="end">
128
+ Auto-updating to v{versions.latest}…
129
+ </Text>
130
+ </Box>
131
+ </>
132
+ )}
133
+ {autoUpdaterResult?.status === 'success' && autoUpdaterResult?.version ? (
134
+ <Text color={theme.success}>
135
+ ✓ Update installed &middot; Restart to apply
136
+ </Text>
137
+ ) : null}
138
+ {(autoUpdaterResult?.status === 'install_failed' ||
139
+ autoUpdaterResult?.status === 'no_permissions') && (
140
+ <Text color={theme.error}>
141
+ ✗ Auto-update failed &middot; Try{' '}
142
+ <Text bold>{PRODUCT_COMMAND} doctor</Text> or{' '}
143
+ <Text bold>npm i -g {MACRO.PACKAGE_URL}</Text>
144
+ </Text>
145
+ )}
146
+ </Box>
147
+ )
148
+ }