@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,58 @@
1
+ import React, { useEffect, useState, useMemo } from 'react'
2
+ import { Text } from 'ink'
3
+ import { getAgentByType } from '../../utils/agentLoader'
4
+ import { getTheme } from '../../utils/theme'
5
+
6
+ interface Props {
7
+ agentType: string
8
+ children: React.ReactNode
9
+ bold?: boolean
10
+ }
11
+
12
+ // Simple cache to prevent re-fetching agent configs
13
+ const agentConfigCache = new Map<string, any>()
14
+
15
+ export function TaskToolMessage({ agentType, children, bold = true }: Props) {
16
+ const theme = getTheme()
17
+ const [agentConfig, setAgentConfig] = useState<any>(() => {
18
+ // Return cached config immediately if available
19
+ return agentConfigCache.get(agentType) || null
20
+ })
21
+
22
+ useEffect(() => {
23
+ // Skip if already cached
24
+ if (agentConfigCache.has(agentType)) {
25
+ setAgentConfig(agentConfigCache.get(agentType))
26
+ return
27
+ }
28
+
29
+ // Load and cache agent configuration
30
+ let mounted = true
31
+ getAgentByType(agentType).then(config => {
32
+ if (mounted) {
33
+ agentConfigCache.set(agentType, config)
34
+ setAgentConfig(config)
35
+ }
36
+ }).catch(() => {
37
+ // Silently handle errors to prevent console noise
38
+ if (mounted) {
39
+ agentConfigCache.set(agentType, null)
40
+ }
41
+ })
42
+
43
+ return () => {
44
+ mounted = false
45
+ }
46
+ }, [agentType])
47
+
48
+ // Memoize color calculation to prevent unnecessary re-renders
49
+ const color = useMemo(() => {
50
+ return agentConfig?.color || theme.text
51
+ }, [agentConfig?.color, theme.text])
52
+
53
+ return (
54
+ <Text color={color} bold={bold}>
55
+ {children}
56
+ </Text>
57
+ )
58
+ }
@@ -0,0 +1,28 @@
1
+ import { Box, Text } from 'ink'
2
+ import * as React from 'react'
3
+ import { extractTag } from '../../utils/messages'
4
+ import { getTheme } from '../../utils/theme'
5
+ import { TextBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'
6
+
7
+ type Props = {
8
+ addMargin: boolean
9
+ param: TextBlockParam
10
+ }
11
+
12
+ export function UserBashInputMessage({
13
+ param: { text },
14
+ addMargin,
15
+ }: Props): React.ReactNode {
16
+ const input = extractTag(text, 'bash-input')
17
+ if (!input) {
18
+ return null
19
+ }
20
+ return (
21
+ <Box flexDirection="column" marginTop={addMargin ? 1 : 0} width="100%">
22
+ <Box>
23
+ <Text color={getTheme().bashBorder}>!</Text>
24
+ <Text color={getTheme().secondaryText}> {input}</Text>
25
+ </Box>
26
+ </Box>
27
+ )
28
+ }
@@ -0,0 +1,30 @@
1
+ import { Box, Text } from 'ink'
2
+ import * as React from 'react'
3
+ import { getTheme } from '../../utils/theme'
4
+ import { extractTag } from '../../utils/messages'
5
+ import { TextBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'
6
+
7
+ type Props = {
8
+ addMargin: boolean
9
+ param: TextBlockParam
10
+ }
11
+
12
+ export function UserCommandMessage({
13
+ addMargin,
14
+ param: { text },
15
+ }: Props): React.ReactNode {
16
+ const commandMessage = extractTag(text, 'command-message')
17
+ const args = extractTag(text, 'command-args')
18
+ if (!commandMessage) {
19
+ return null
20
+ }
21
+
22
+ const theme = getTheme()
23
+ return (
24
+ <Box flexDirection="column" marginTop={addMargin ? 1 : 0} width="100%">
25
+ <Text color={theme.secondaryText}>
26
+ &gt; /{commandMessage} {args}
27
+ </Text>
28
+ </Box>
29
+ )
30
+ }
@@ -0,0 +1,28 @@
1
+ import { Box, Text } from 'ink'
2
+ import * as React from 'react'
3
+ import { extractTag } from '../../utils/messages'
4
+ import { getTheme } from '../../utils/theme'
5
+ import { TextBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'
6
+
7
+ type Props = {
8
+ addMargin: boolean
9
+ param: TextBlockParam
10
+ }
11
+
12
+ export function UserKodingInputMessage({
13
+ param: { text },
14
+ addMargin,
15
+ }: Props): React.ReactNode {
16
+ const input = extractTag(text, 'koding-input')
17
+ if (!input) {
18
+ return null
19
+ }
20
+ return (
21
+ <Box flexDirection="column" marginTop={addMargin ? 1 : 0} width="100%">
22
+ <Box>
23
+ <Text color={getTheme().noting}>#</Text>
24
+ <Text color={getTheme().secondaryText}> {input}</Text>
25
+ </Box>
26
+ </Box>
27
+ )
28
+ }
@@ -0,0 +1,35 @@
1
+ import React from 'react'
2
+ import { TextBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'
3
+ import { Box, Text } from 'ink'
4
+ import { getTheme } from '../../utils/theme'
5
+ import { logError } from '../../utils/log'
6
+ import { useTerminalSize } from '../../hooks/useTerminalSize'
7
+
8
+ type Props = {
9
+ addMargin: boolean
10
+ param: TextBlockParam
11
+ }
12
+
13
+ export function UserPromptMessage({
14
+ addMargin,
15
+ param: { text },
16
+ }: Props): React.ReactNode {
17
+ const { columns } = useTerminalSize()
18
+ if (!text) {
19
+ logError('No content found in user prompt message')
20
+ return null
21
+ }
22
+
23
+ return (
24
+ <Box flexDirection="row" marginTop={addMargin ? 1 : 0} width="100%">
25
+ <Box minWidth={2} width={2}>
26
+ <Text color={getTheme().secondaryText}>&gt;</Text>
27
+ </Box>
28
+ <Box flexDirection="column" width={columns - 4}>
29
+ <Text color={getTheme().secondaryText} wrap="wrap">
30
+ {text}
31
+ </Text>
32
+ </Box>
33
+ </Box>
34
+ )
35
+ }
@@ -0,0 +1,39 @@
1
+ import { TextBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'
2
+ import { UserBashInputMessage } from './UserBashInputMessage'
3
+ import { UserKodingInputMessage } from './UserKodingInputMessage'
4
+ import { UserCommandMessage } from './UserCommandMessage'
5
+ import { UserPromptMessage } from './UserPromptMessage'
6
+ import * as React from 'react'
7
+ import { NO_CONTENT_MESSAGE } from '../../services/claude'
8
+
9
+ type Props = {
10
+ addMargin: boolean
11
+ param: TextBlockParam
12
+ }
13
+
14
+ export function UserTextMessage({ addMargin, param }: Props): React.ReactNode {
15
+ if (param.text.trim() === NO_CONTENT_MESSAGE) {
16
+ return null
17
+ }
18
+
19
+ // Koding inputs!
20
+ if (param.text.includes('<koding-input>')) {
21
+ return <UserKodingInputMessage addMargin={addMargin} param={param} />
22
+ }
23
+
24
+ // Bash inputs!
25
+ if (param.text.includes('<bash-input>')) {
26
+ return <UserBashInputMessage addMargin={addMargin} param={param} />
27
+ }
28
+
29
+ // Slash commands/
30
+ if (
31
+ param.text.includes('<command-name>') ||
32
+ param.text.includes('<command-message>')
33
+ ) {
34
+ return <UserCommandMessage addMargin={addMargin} param={param} />
35
+ }
36
+
37
+ // User prompts>
38
+ return <UserPromptMessage addMargin={addMargin} param={param} />
39
+ }
@@ -0,0 +1,12 @@
1
+ import { Text } from 'ink'
2
+ import * as React from 'react'
3
+ import { getTheme } from '../../../utils/theme'
4
+
5
+ export function UserToolCanceledMessage(): React.ReactNode {
6
+ return (
7
+ <Text>
8
+ &nbsp;&nbsp;⎿ &nbsp;
9
+ <Text color={getTheme().error}>Interrupted by user</Text>
10
+ </Text>
11
+ )
12
+ }
@@ -0,0 +1,36 @@
1
+ import { ToolResultBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'
2
+ import { Box, Text } from 'ink'
3
+ import * as React from 'react'
4
+ import { getTheme } from '../../../utils/theme'
5
+
6
+ const MAX_RENDERED_LINES = 10
7
+
8
+ type Props = {
9
+ param: ToolResultBlockParam
10
+ verbose: boolean
11
+ }
12
+
13
+ export function UserToolErrorMessage({
14
+ param,
15
+ verbose,
16
+ }: Props): React.ReactNode {
17
+ const error =
18
+ typeof param.content === 'string' ? param.content.trim() : 'Error'
19
+ return (
20
+ <Box flexDirection="row" width="100%">
21
+ <Text>&nbsp;&nbsp;⎿ &nbsp;</Text>
22
+ <Box flexDirection="column">
23
+ <Text color={getTheme().error}>
24
+ {verbose
25
+ ? error
26
+ : error.split('\n').slice(0, MAX_RENDERED_LINES).join('\n') || ''}
27
+ </Text>
28
+ {!verbose && error.split('\n').length > MAX_RENDERED_LINES && (
29
+ <Text color={getTheme().secondaryText}>
30
+ ... (+{error.split('\n').length - MAX_RENDERED_LINES} lines)
31
+ </Text>
32
+ )}
33
+ </Box>
34
+ </Box>
35
+ )
36
+ }
@@ -0,0 +1,31 @@
1
+ import * as React from 'react'
2
+ import { Tool } from '../../../Tool'
3
+ import { Message } from '../../../query'
4
+ import { FallbackToolUseRejectedMessage } from '../../FallbackToolUseRejectedMessage'
5
+ import { useGetToolFromMessages } from './utils'
6
+ import { useTerminalSize } from '../../../hooks/useTerminalSize'
7
+
8
+ type Props = {
9
+ toolUseID: string
10
+ messages: Message[]
11
+ tools: Tool[]
12
+ verbose: boolean
13
+ }
14
+
15
+ export function UserToolRejectMessage({
16
+ toolUseID,
17
+ tools,
18
+ messages,
19
+ verbose,
20
+ }: Props): React.ReactNode {
21
+ const { columns } = useTerminalSize()
22
+ const { tool, toolUse } = useGetToolFromMessages(toolUseID, tools, messages)
23
+ const input = tool.inputSchema.safeParse(toolUse.input)
24
+ if (input.success) {
25
+ return tool.renderToolUseRejectedMessage(input.data, {
26
+ columns,
27
+ verbose,
28
+ })
29
+ }
30
+ return <FallbackToolUseRejectedMessage />
31
+ }
@@ -0,0 +1,57 @@
1
+ import { ToolResultBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'
2
+ import * as React from 'react'
3
+ import { Tool } from '../../../Tool'
4
+ import { Message, UserMessage } from '../../../query'
5
+ import { CANCEL_MESSAGE, REJECT_MESSAGE } from '../../../utils/messages'
6
+ import { UserToolCanceledMessage } from './UserToolCanceledMessage'
7
+ import { UserToolErrorMessage } from './UserToolErrorMessage'
8
+ import { UserToolRejectMessage } from './UserToolRejectMessage'
9
+ import { UserToolSuccessMessage } from './UserToolSuccessMessage'
10
+
11
+ type Props = {
12
+ param: ToolResultBlockParam
13
+ message: UserMessage
14
+ messages: Message[]
15
+ tools: Tool[]
16
+ verbose: boolean
17
+ width: number | string
18
+ }
19
+
20
+ export function UserToolResultMessage({
21
+ param,
22
+ message,
23
+ messages,
24
+ tools,
25
+ verbose,
26
+ width,
27
+ }: Props): React.ReactNode {
28
+ if (param.content === CANCEL_MESSAGE) {
29
+ return <UserToolCanceledMessage />
30
+ }
31
+
32
+ if (param.content === REJECT_MESSAGE) {
33
+ return (
34
+ <UserToolRejectMessage
35
+ toolUseID={param.tool_use_id}
36
+ tools={tools}
37
+ messages={messages}
38
+ verbose={verbose}
39
+ />
40
+ )
41
+ }
42
+
43
+ if (param.is_error) {
44
+ return <UserToolErrorMessage param={param} verbose={verbose} />
45
+ }
46
+
47
+ return (
48
+ <UserToolSuccessMessage
49
+ param={param}
50
+ message={message}
51
+ messages={messages}
52
+ tools={tools}
53
+ verbose={verbose}
54
+ width={width}
55
+ />
56
+ )
57
+ }
@@ -0,0 +1,35 @@
1
+ import { ToolResultBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'
2
+ import { Box } from 'ink'
3
+ import * as React from 'react'
4
+ import { Tool } from '../../../Tool'
5
+ import { Message, UserMessage } from '../../../query'
6
+ import { useGetToolFromMessages } from './utils'
7
+
8
+ type Props = {
9
+ param: ToolResultBlockParam
10
+ message: UserMessage
11
+ messages: Message[]
12
+ verbose: boolean
13
+ tools: Tool[]
14
+ width: number | string
15
+ }
16
+
17
+ export function UserToolSuccessMessage({
18
+ param,
19
+ message,
20
+ messages,
21
+ tools,
22
+ verbose,
23
+ width,
24
+ }: Props): React.ReactNode {
25
+ const { tool } = useGetToolFromMessages(param.tool_use_id, tools, messages)
26
+
27
+ return (
28
+ // TODO: Distinguish UserMessage from UserToolResultMessage
29
+ <Box flexDirection="column" width={width}>
30
+ {tool.renderToolResultMessage?.(message.toolUseResult!.data as never, {
31
+ verbose,
32
+ })}
33
+ </Box>
34
+ )
35
+ }
@@ -0,0 +1,56 @@
1
+ import { ToolUseBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'
2
+ import { Message } from '../../../query'
3
+ import { useMemo } from 'react'
4
+ import { Tool } from '../../../Tool'
5
+ import { GlobTool } from '../../../tools/GlobTool/GlobTool'
6
+ import { GrepTool } from '../../../tools/GrepTool/GrepTool'
7
+ import { logEvent } from '../../../services/statsig'
8
+
9
+ function getToolUseFromMessages(
10
+ toolUseID: string,
11
+ messages: Message[],
12
+ ): ToolUseBlockParam | null {
13
+ let toolUse: ToolUseBlockParam | null = null
14
+ for (const message of messages) {
15
+ if (
16
+ message.type !== 'assistant' ||
17
+ !Array.isArray(message.message.content)
18
+ ) {
19
+ continue
20
+ }
21
+ for (const content of message.message.content) {
22
+ if (content.type === 'tool_use' && content.id === toolUseID) {
23
+ toolUse = content
24
+ }
25
+ }
26
+ }
27
+ return toolUse
28
+ }
29
+
30
+ export function useGetToolFromMessages(
31
+ toolUseID: string,
32
+ tools: Tool[],
33
+ messages: Message[],
34
+ ) {
35
+ return useMemo(() => {
36
+ const toolUse = getToolUseFromMessages(toolUseID, messages)
37
+ if (!toolUse) {
38
+ throw new ReferenceError(
39
+ `Tool use not found for tool_use_id ${toolUseID}`,
40
+ )
41
+ }
42
+ // Hack: we don't expose GlobTool and GrepTool in getTools anymore,
43
+ // but we still want to be able to load old transcripts.
44
+ // TODO: Remove this when logging hits zero
45
+ const tool = [...tools, GlobTool, GrepTool].find(
46
+ _ => _.name === toolUse.name,
47
+ )
48
+ if (tool === GlobTool || tool === GrepTool) {
49
+ logEvent('tengu_legacy_tool_lookup', {})
50
+ }
51
+ if (!tool) {
52
+ throw new ReferenceError(`Tool not found for ${toolUse.name}`)
53
+ }
54
+ return { tool, toolUse }
55
+ }, [toolUseID, messages, tools])
56
+ }
@@ -0,0 +1,121 @@
1
+ import { Box, Text } from 'ink'
2
+ import React, { useMemo } from 'react'
3
+ import { UnaryEvent } from '../../../hooks/usePermissionRequestLogging'
4
+ import { savePermission } from '../../../permissions'
5
+ import { BashTool } from '../../../tools/BashTool/BashTool'
6
+ import { getTheme } from '../../../utils/theme'
7
+ import { usePermissionRequestLogging } from '../hooks'
8
+ import {
9
+ type ToolUseConfirm,
10
+ toolUseConfirmGetPrefix,
11
+ } from '../PermissionRequest.js'
12
+ import { PermissionRequestTitle } from '../PermissionRequestTitle'
13
+ import { logUnaryPermissionEvent } from '../utils'
14
+ import { Select } from '../../CustomSelect/select'
15
+ import { toolUseOptions } from '../toolUseOptions'
16
+
17
+ type Props = {
18
+ toolUseConfirm: ToolUseConfirm
19
+ onDone(): void
20
+ }
21
+
22
+ export function BashPermissionRequest({
23
+ toolUseConfirm,
24
+ onDone,
25
+ }: Props): React.ReactNode {
26
+ const theme = getTheme()
27
+
28
+ // ok to use parse since we've already validated args earliers
29
+ const { command } = BashTool.inputSchema.parse(toolUseConfirm.input)
30
+
31
+ const unaryEvent = useMemo<UnaryEvent>(
32
+ () => ({ completion_type: 'tool_use_single', language_name: 'none' }),
33
+ [],
34
+ )
35
+
36
+ usePermissionRequestLogging(toolUseConfirm, unaryEvent)
37
+
38
+ return (
39
+ <Box
40
+ flexDirection="column"
41
+ borderStyle="round"
42
+ borderColor={theme.permission}
43
+ marginTop={1}
44
+ paddingLeft={1}
45
+ paddingRight={1}
46
+ paddingBottom={1}
47
+ >
48
+ <PermissionRequestTitle
49
+ title="Bash command"
50
+ riskScore={toolUseConfirm.riskScore}
51
+ />
52
+ <Box flexDirection="column" paddingX={2} paddingY={1}>
53
+ <Text>{BashTool.renderToolUseMessage({ command })}</Text>
54
+ <Text color={theme.secondaryText}>{toolUseConfirm.description}</Text>
55
+ </Box>
56
+
57
+ <Box flexDirection="column">
58
+ <Text>Do you want to proceed?</Text>
59
+ <Select
60
+ options={toolUseOptions({ toolUseConfirm, command })}
61
+ onChange={newValue => {
62
+ switch (newValue) {
63
+ case 'yes':
64
+ logUnaryPermissionEvent(
65
+ 'tool_use_single',
66
+ toolUseConfirm,
67
+ 'accept',
68
+ )
69
+ toolUseConfirm.onAllow('temporary')
70
+ onDone()
71
+ break
72
+ case 'yes-dont-ask-again-prefix': {
73
+ const prefix = toolUseConfirmGetPrefix(toolUseConfirm)
74
+ if (prefix !== null) {
75
+ logUnaryPermissionEvent(
76
+ 'tool_use_single',
77
+ toolUseConfirm,
78
+ 'accept',
79
+ )
80
+ savePermission(
81
+ toolUseConfirm.tool,
82
+ toolUseConfirm.input,
83
+ prefix,
84
+ ).then(() => {
85
+ toolUseConfirm.onAllow('permanent')
86
+ onDone()
87
+ })
88
+ }
89
+ break
90
+ }
91
+ case 'yes-dont-ask-again-full':
92
+ logUnaryPermissionEvent(
93
+ 'tool_use_single',
94
+ toolUseConfirm,
95
+ 'accept',
96
+ )
97
+ savePermission(
98
+ toolUseConfirm.tool,
99
+ toolUseConfirm.input,
100
+ null, // Save without prefix
101
+ ).then(() => {
102
+ toolUseConfirm.onAllow('permanent')
103
+ onDone()
104
+ })
105
+ break
106
+ case 'no':
107
+ logUnaryPermissionEvent(
108
+ 'tool_use_single',
109
+ toolUseConfirm,
110
+ 'reject',
111
+ )
112
+ toolUseConfirm.onReject()
113
+ onDone()
114
+ break
115
+ }
116
+ }}
117
+ />
118
+ </Box>
119
+ </Box>
120
+ )
121
+ }