@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,40 @@
1
+ import { Box, Text } from 'ink'
2
+ import React from 'react'
3
+ import { useInterval } from '../hooks/useInterval'
4
+ import { getTheme } from '../utils/theme'
5
+ import { BLACK_CIRCLE } from '../constants/figures'
6
+
7
+ type Props = {
8
+ isError: boolean
9
+ isUnresolved: boolean
10
+ shouldAnimate: boolean
11
+ }
12
+
13
+ export function ToolUseLoader({
14
+ isError,
15
+ isUnresolved,
16
+ shouldAnimate,
17
+ }: Props): React.ReactNode {
18
+ const [isVisible, setIsVisible] = React.useState(true)
19
+
20
+ useInterval(() => {
21
+ if (!shouldAnimate) {
22
+ return
23
+ }
24
+ // To avoid flickering when the tool use confirm is visible, we set the loader to be visible
25
+ // when the tool use confirm is visible.
26
+ setIsVisible(_ => !_)
27
+ }, 600)
28
+
29
+ const color = isUnresolved
30
+ ? getTheme().secondaryText
31
+ : isError
32
+ ? getTheme().error
33
+ : getTheme().success
34
+
35
+ return (
36
+ <Box minWidth={2}>
37
+ <Text color={color}>{isVisible ? BLACK_CIRCLE : ' '}</Text>
38
+ </Box>
39
+ )
40
+ }
@@ -0,0 +1,106 @@
1
+ import React from 'react'
2
+ import { Box, Text, useInput } from 'ink'
3
+ import { getTheme } from '../utils/theme'
4
+ import { Select } from './CustomSelect/select'
5
+ import {
6
+ saveCurrentProjectConfig,
7
+ getCurrentProjectConfig,
8
+ } from '../utils/config.js'
9
+ import { PRODUCT_NAME } from '../constants/product'
10
+ import { logEvent } from '../services/statsig'
11
+ import { useExitOnCtrlCD } from '../hooks/useExitOnCtrlCD'
12
+ import { homedir } from 'os'
13
+ import { getCwd } from '../utils/state'
14
+ import Link from './Link'
15
+
16
+ type Props = {
17
+ onDone(): void
18
+ }
19
+
20
+ export function TrustDialog({ onDone }: Props): React.ReactNode {
21
+ const theme = getTheme()
22
+ React.useEffect(() => {
23
+ // Log when dialog is shown
24
+ logEvent('trust_dialog_shown', {})
25
+ }, [])
26
+
27
+ function onChange(value: 'yes' | 'no') {
28
+ const config = getCurrentProjectConfig()
29
+ switch (value) {
30
+ case 'yes': {
31
+ // Log when user accepts
32
+ const isHomeDir = homedir() === getCwd()
33
+ logEvent('trust_dialog_accept', {
34
+ isHomeDir: String(isHomeDir),
35
+ })
36
+
37
+ if (!isHomeDir) {
38
+ saveCurrentProjectConfig({
39
+ ...config,
40
+ hasTrustDialogAccepted: true,
41
+ })
42
+ }
43
+ onDone()
44
+ break
45
+ }
46
+ case 'no': {
47
+ process.exit(1)
48
+ break
49
+ }
50
+ }
51
+ }
52
+
53
+ const exitState = useExitOnCtrlCD(() => process.exit(0))
54
+
55
+ useInput((_input, key) => {
56
+ if (key.escape) {
57
+ process.exit(0)
58
+ return
59
+ }
60
+ })
61
+
62
+ return (
63
+ <>
64
+ <Box
65
+ flexDirection="column"
66
+ gap={1}
67
+ padding={1}
68
+ borderStyle="round"
69
+ borderColor={theme.warning}
70
+ >
71
+ <Text bold color={theme.warning}>
72
+ Do you trust the files in this folder?
73
+ </Text>
74
+ <Text bold>{process.cwd()}</Text>
75
+
76
+ <Box flexDirection="column" gap={1}>
77
+ <Text>
78
+ {PRODUCT_NAME} may read files in this folder. Reading untrusted
79
+ files may lead to {PRODUCT_NAME} to behave in an unexpected ways.
80
+ </Text>
81
+ <Text>
82
+ With your permission {PRODUCT_NAME} may execute files in this
83
+ folder. Executing untrusted code is unsafe.
84
+ </Text>
85
+ </Box>
86
+
87
+ <Select
88
+ options={[
89
+ { label: 'Yes, proceed', value: 'yes' },
90
+ { label: 'No, exit', value: 'no' },
91
+ ]}
92
+ onChange={value => onChange(value as 'yes' | 'no')}
93
+ />
94
+ </Box>
95
+ <Box marginLeft={3}>
96
+ <Text dimColor>
97
+ {exitState.pending ? (
98
+ <>Press {exitState.keyName} again to exit</>
99
+ ) : (
100
+ <>Enter to confirm · Esc to exit</>
101
+ )}
102
+ </Text>
103
+ </Box>
104
+ </>
105
+ )
106
+ }
@@ -0,0 +1,63 @@
1
+ import { default as React, useCallback } from 'react'
2
+ import { useNotifyAfterTimeout } from '../../hooks/useNotifyAfterTimeout'
3
+ import { AssistantMessage, BinaryFeedbackResult } from '../../query'
4
+ import type { Tool } from '../../Tool'
5
+ import type { NormalizedMessage } from '../../utils/messages'
6
+ import { BinaryFeedbackView } from './BinaryFeedbackView'
7
+ import {
8
+ type BinaryFeedbackChoose,
9
+ getBinaryFeedbackResultForChoice,
10
+ logBinaryFeedbackEvent,
11
+ } from './utils.js'
12
+ import { PRODUCT_NAME } from '../../constants/product'
13
+
14
+ type Props = {
15
+ m1: AssistantMessage
16
+ m2: AssistantMessage
17
+ resolve: (result: BinaryFeedbackResult) => void
18
+ debug: boolean
19
+ erroredToolUseIDs: Set<string>
20
+ inProgressToolUseIDs: Set<string>
21
+ normalizedMessages: NormalizedMessage[]
22
+ tools: Tool[]
23
+ unresolvedToolUseIDs: Set<string>
24
+ verbose: boolean
25
+ }
26
+
27
+ export function BinaryFeedback({
28
+ m1,
29
+ m2,
30
+ resolve,
31
+ debug,
32
+ erroredToolUseIDs,
33
+ inProgressToolUseIDs,
34
+ normalizedMessages,
35
+ tools,
36
+ unresolvedToolUseIDs,
37
+ verbose,
38
+ }: Props): React.ReactNode {
39
+ const onChoose = useCallback<BinaryFeedbackChoose>(
40
+ choice => {
41
+ logBinaryFeedbackEvent(m1, m2, choice)
42
+ resolve(getBinaryFeedbackResultForChoice(m1, m2, choice))
43
+ },
44
+ [m1, m2, resolve],
45
+ )
46
+ useNotifyAfterTimeout(
47
+ `${PRODUCT_NAME} needs your input on a response comparison`,
48
+ )
49
+ return (
50
+ <BinaryFeedbackView
51
+ debug={debug}
52
+ erroredToolUseIDs={erroredToolUseIDs}
53
+ inProgressToolUseIDs={inProgressToolUseIDs}
54
+ m1={m1}
55
+ m2={m2}
56
+ normalizedMessages={normalizedMessages}
57
+ tools={tools}
58
+ unresolvedToolUseIDs={unresolvedToolUseIDs}
59
+ verbose={verbose}
60
+ onChoose={onChoose}
61
+ />
62
+ )
63
+ }
@@ -0,0 +1,111 @@
1
+ import { FileEditTool } from '../../tools/FileEditTool/FileEditTool'
2
+ import { FileEditToolDiff } from '../permissions/FileEditPermissionRequest/FileEditToolDiff'
3
+ import { Message } from '../Message'
4
+ import {
5
+ normalizeMessages,
6
+ type NormalizedMessage,
7
+ } from '../../utils/messages.js'
8
+ import type { Tool } from '../../Tool'
9
+ import { useTerminalSize } from '../../hooks/useTerminalSize'
10
+ import { FileWriteTool } from '../../tools/FileWriteTool/FileWriteTool'
11
+ import { FileWriteToolDiff } from '../permissions/FileWritePermissionRequest/FileWriteToolDiff'
12
+ import type { AssistantMessage } from '../../query'
13
+ import * as React from 'react'
14
+ import { Box } from 'ink'
15
+
16
+ type Props = {
17
+ debug: boolean
18
+ erroredToolUseIDs: Set<string>
19
+ inProgressToolUseIDs: Set<string>
20
+ message: AssistantMessage
21
+ normalizedMessages: NormalizedMessage[]
22
+ tools: Tool[]
23
+ unresolvedToolUseIDs: Set<string>
24
+ verbose: boolean
25
+ }
26
+
27
+ export function BinaryFeedbackOption({
28
+ debug,
29
+ erroredToolUseIDs,
30
+ inProgressToolUseIDs,
31
+ message,
32
+ normalizedMessages,
33
+ tools,
34
+ unresolvedToolUseIDs,
35
+ verbose,
36
+ }: Props): React.ReactNode {
37
+ const { columns } = useTerminalSize()
38
+ return normalizeMessages([message])
39
+ .filter(_ => _.type !== 'progress')
40
+ .map((_, index) => (
41
+ <Box flexDirection="column" key={index}>
42
+ <Message
43
+ addMargin={false}
44
+ erroredToolUseIDs={erroredToolUseIDs}
45
+ debug={debug}
46
+ inProgressToolUseIDs={inProgressToolUseIDs}
47
+ message={_}
48
+ messages={normalizedMessages}
49
+ shouldAnimate={false}
50
+ shouldShowDot={true}
51
+ tools={tools}
52
+ unresolvedToolUseIDs={unresolvedToolUseIDs}
53
+ verbose={verbose}
54
+ width={columns / 2 - 6}
55
+ />
56
+ <AdditionalContext message={_} verbose={verbose} />
57
+ </Box>
58
+ ))
59
+ }
60
+
61
+ function AdditionalContext({
62
+ message,
63
+ verbose,
64
+ }: {
65
+ message: NormalizedMessage
66
+ verbose: boolean
67
+ }) {
68
+ const { columns } = useTerminalSize()
69
+ if (message.type !== 'assistant') {
70
+ return null
71
+ }
72
+ const content = message.message.content[0]!
73
+ switch (content.type) {
74
+ case 'tool_use':
75
+ switch (content.name) {
76
+ case FileEditTool.name: {
77
+ const input = FileEditTool.inputSchema.safeParse(content.input)
78
+ if (!input.success) {
79
+ return null
80
+ }
81
+ return (
82
+ <FileEditToolDiff
83
+ file_path={input.data.file_path}
84
+ new_string={input.data.new_string}
85
+ old_string={input.data.old_string}
86
+ verbose={verbose}
87
+ width={columns / 2 - 12}
88
+ />
89
+ )
90
+ }
91
+ case FileWriteTool.name: {
92
+ const input = FileWriteTool.inputSchema.safeParse(content.input)
93
+ if (!input.success) {
94
+ return null
95
+ }
96
+ return (
97
+ <FileWriteToolDiff
98
+ file_path={input.data.file_path}
99
+ content={input.data.content}
100
+ verbose={verbose}
101
+ width={columns / 2 - 12}
102
+ />
103
+ )
104
+ }
105
+ default:
106
+ return null
107
+ }
108
+ default:
109
+ return null
110
+ }
111
+ }
@@ -0,0 +1,172 @@
1
+ import { Option, SelectProps } from '@inkjs/ui'
2
+ import chalk from 'chalk'
3
+ import { Box, Text, useInput } from 'ink'
4
+ import Link from 'ink-link'
5
+ import React, { useState } from 'react'
6
+ import { getTheme } from '../../utils/theme'
7
+ import { Select } from '../CustomSelect/select'
8
+ import type { Tool } from '../../Tool'
9
+ import type { NormalizedMessage } from '../../utils/messages'
10
+ import { BinaryFeedbackOption } from './BinaryFeedbackOption'
11
+ import type { AssistantMessage } from '../../query'
12
+ import type { BinaryFeedbackChoose } from './utils'
13
+ import { useExitOnCtrlCD } from '../../hooks/useExitOnCtrlCD'
14
+ import { BinaryFeedbackChoice } from './utils'
15
+ import { PRODUCT_NAME } from '../../constants/product'
16
+
17
+ const HELP_URL = 'https://go/cli-feedback'
18
+
19
+ type BinaryFeedbackOption = Option & { value: BinaryFeedbackChoice }
20
+
21
+ // Make options a function to avoid early theme access during module initialization
22
+ export function getOptions(): BinaryFeedbackOption[] {
23
+ return [
24
+ {
25
+ // This option combines the follow user intents:
26
+ // - The two options look about equally good to me
27
+ // - I don't feel confident enough to choose
28
+ // - I don't want to choose right now
29
+ label: 'Choose for me',
30
+ value: 'no-preference',
31
+ },
32
+ {
33
+ label: 'Left option looks better',
34
+ value: 'prefer-left',
35
+ },
36
+ {
37
+ label: 'Right option looks better',
38
+ value: 'prefer-right',
39
+ },
40
+ {
41
+ label: `Neither, and tell ${PRODUCT_NAME} what to do differently (${chalk.bold.hex(getTheme().warning)('esc')})`,
42
+ value: 'neither',
43
+ },
44
+ ]
45
+ }
46
+
47
+ type Props = {
48
+ m1: AssistantMessage
49
+ m2: AssistantMessage
50
+ onChoose?: BinaryFeedbackChoose
51
+ debug: boolean
52
+ erroredToolUseIDs: Set<string>
53
+ inProgressToolUseIDs: Set<string>
54
+ normalizedMessages: NormalizedMessage[]
55
+ tools: Tool[]
56
+ unresolvedToolUseIDs: Set<string>
57
+ verbose: boolean
58
+ }
59
+
60
+ export function BinaryFeedbackView({
61
+ m1,
62
+ m2,
63
+ onChoose,
64
+ debug,
65
+ erroredToolUseIDs,
66
+ inProgressToolUseIDs,
67
+ normalizedMessages,
68
+ tools,
69
+ unresolvedToolUseIDs,
70
+ verbose,
71
+ }: Props) {
72
+ const theme = getTheme()
73
+ const [focused, setFocus] = useState('no-preference')
74
+ const [focusValue, setFocusValue] = useState<string | undefined>(undefined)
75
+ const exitState = useExitOnCtrlCD(() => process.exit(1))
76
+
77
+ useInput((_input, key) => {
78
+ if (key.leftArrow) {
79
+ setFocusValue('prefer-left')
80
+ } else if (key.rightArrow) {
81
+ setFocusValue('prefer-right')
82
+ } else if (key.escape) {
83
+ onChoose?.('neither')
84
+ }
85
+ })
86
+
87
+ return (
88
+ <>
89
+ <Box
90
+ flexDirection="column"
91
+ height="100%"
92
+ width="100%"
93
+ borderStyle="round"
94
+ borderColor={theme.permission}
95
+ >
96
+ <Box width="100%" justifyContent="space-between" paddingX={1}>
97
+ <Text bold color={theme.permission}>
98
+ [ANT-ONLY] Help train {PRODUCT_NAME}
99
+ </Text>
100
+ <Text>
101
+ <Link url={HELP_URL}>[?]</Link>
102
+ </Text>
103
+ </Box>
104
+ <Box flexDirection="row" width="100%" flexGrow={1} paddingTop={1}>
105
+ <Box
106
+ flexDirection="column"
107
+ flexGrow={1}
108
+ flexBasis={1}
109
+ gap={1}
110
+ borderStyle={focused === 'prefer-left' ? 'bold' : 'single'}
111
+ borderColor={
112
+ focused === 'prefer-left' ? theme.success : theme.secondaryBorder
113
+ }
114
+ marginRight={1}
115
+ padding={1}
116
+ >
117
+ <BinaryFeedbackOption
118
+ erroredToolUseIDs={erroredToolUseIDs}
119
+ debug={debug}
120
+ inProgressToolUseIDs={inProgressToolUseIDs}
121
+ message={m1}
122
+ normalizedMessages={normalizedMessages}
123
+ tools={tools}
124
+ unresolvedToolUseIDs={unresolvedToolUseIDs}
125
+ verbose={verbose}
126
+ />
127
+ </Box>
128
+ <Box
129
+ flexDirection="column"
130
+ flexGrow={1}
131
+ flexBasis={1}
132
+ gap={1}
133
+ borderStyle={focused === 'prefer-right' ? 'bold' : 'single'}
134
+ borderColor={
135
+ focused === 'prefer-right' ? theme.success : theme.secondaryBorder
136
+ }
137
+ marginLeft={1}
138
+ padding={1}
139
+ >
140
+ <BinaryFeedbackOption
141
+ erroredToolUseIDs={erroredToolUseIDs}
142
+ debug={debug}
143
+ inProgressToolUseIDs={inProgressToolUseIDs}
144
+ message={m2}
145
+ normalizedMessages={normalizedMessages}
146
+ tools={tools}
147
+ unresolvedToolUseIDs={unresolvedToolUseIDs}
148
+ verbose={verbose}
149
+ />
150
+ </Box>
151
+ </Box>
152
+ <Box flexDirection="column" paddingTop={1} paddingX={1}>
153
+ <Text>How do you want to proceed?</Text>
154
+ <Select
155
+ options={getOptions()}
156
+ onFocus={setFocus}
157
+ focusValue={focusValue}
158
+ onChange={onChoose as SelectProps['onChange']}
159
+ />
160
+ </Box>
161
+ </Box>
162
+ {exitState.pending ? (
163
+ <Box marginLeft={3}>
164
+ <Text dimColor>Press {exitState.keyName} again to exit</Text>
165
+ </Box>
166
+ ) : (
167
+ // Render a blank line so that the UI doesn't reflow when the exit message is shown
168
+ <Text> </Text>
169
+ )}
170
+ </>
171
+ )
172
+ }