@shareai-lab/kode 1.0.70 → 1.0.73

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 (278) hide show
  1. package/README.md +342 -75
  2. package/README.zh-CN.md +292 -0
  3. package/cli.js +62 -0
  4. package/package.json +49 -25
  5. package/scripts/postinstall.js +56 -0
  6. package/src/ProjectOnboarding.tsx +198 -0
  7. package/src/Tool.ts +82 -0
  8. package/src/commands/agents.tsx +3401 -0
  9. package/src/commands/approvedTools.ts +53 -0
  10. package/src/commands/bug.tsx +20 -0
  11. package/src/commands/clear.ts +43 -0
  12. package/src/commands/compact.ts +120 -0
  13. package/src/commands/config.tsx +19 -0
  14. package/src/commands/cost.ts +18 -0
  15. package/src/commands/ctx_viz.ts +209 -0
  16. package/src/commands/doctor.ts +24 -0
  17. package/src/commands/help.tsx +19 -0
  18. package/src/commands/init.ts +37 -0
  19. package/src/commands/listen.ts +42 -0
  20. package/src/commands/login.tsx +51 -0
  21. package/src/commands/logout.tsx +40 -0
  22. package/src/commands/mcp.ts +41 -0
  23. package/src/commands/model.tsx +40 -0
  24. package/src/commands/modelstatus.tsx +20 -0
  25. package/src/commands/onboarding.tsx +34 -0
  26. package/src/commands/pr_comments.ts +59 -0
  27. package/src/commands/refreshCommands.ts +54 -0
  28. package/src/commands/release-notes.ts +34 -0
  29. package/src/commands/resume.tsx +31 -0
  30. package/src/commands/review.ts +49 -0
  31. package/src/commands/terminalSetup.ts +221 -0
  32. package/src/commands.ts +139 -0
  33. package/src/components/ApproveApiKey.tsx +93 -0
  34. package/src/components/AsciiLogo.tsx +13 -0
  35. package/src/components/AutoUpdater.tsx +148 -0
  36. package/src/components/Bug.tsx +367 -0
  37. package/src/components/Config.tsx +293 -0
  38. package/src/components/ConsoleOAuthFlow.tsx +327 -0
  39. package/src/components/Cost.tsx +23 -0
  40. package/src/components/CostThresholdDialog.tsx +46 -0
  41. package/src/components/CustomSelect/option-map.ts +42 -0
  42. package/src/components/CustomSelect/select-option.tsx +78 -0
  43. package/src/components/CustomSelect/select.tsx +152 -0
  44. package/src/components/CustomSelect/theme.ts +45 -0
  45. package/src/components/CustomSelect/use-select-state.ts +414 -0
  46. package/src/components/CustomSelect/use-select.ts +35 -0
  47. package/src/components/FallbackToolUseRejectedMessage.tsx +15 -0
  48. package/src/components/FileEditToolUpdatedMessage.tsx +66 -0
  49. package/src/components/Help.tsx +215 -0
  50. package/src/components/HighlightedCode.tsx +33 -0
  51. package/src/components/InvalidConfigDialog.tsx +113 -0
  52. package/src/components/Link.tsx +32 -0
  53. package/src/components/LogSelector.tsx +86 -0
  54. package/src/components/Logo.tsx +145 -0
  55. package/src/components/MCPServerApprovalDialog.tsx +100 -0
  56. package/src/components/MCPServerDialogCopy.tsx +25 -0
  57. package/src/components/MCPServerMultiselectDialog.tsx +109 -0
  58. package/src/components/Message.tsx +221 -0
  59. package/src/components/MessageResponse.tsx +15 -0
  60. package/src/components/MessageSelector.tsx +211 -0
  61. package/src/components/ModeIndicator.tsx +88 -0
  62. package/src/components/ModelConfig.tsx +301 -0
  63. package/src/components/ModelListManager.tsx +227 -0
  64. package/src/components/ModelSelector.tsx +3386 -0
  65. package/src/components/ModelStatusDisplay.tsx +230 -0
  66. package/src/components/Onboarding.tsx +274 -0
  67. package/src/components/PressEnterToContinue.tsx +11 -0
  68. package/src/components/PromptInput.tsx +740 -0
  69. package/src/components/SentryErrorBoundary.ts +33 -0
  70. package/src/components/Spinner.tsx +129 -0
  71. package/src/components/StickerRequestForm.tsx +16 -0
  72. package/src/components/StructuredDiff.tsx +191 -0
  73. package/src/components/TextInput.tsx +259 -0
  74. package/src/components/TodoItem.tsx +11 -0
  75. package/src/components/TokenWarning.tsx +31 -0
  76. package/src/components/ToolUseLoader.tsx +40 -0
  77. package/src/components/TrustDialog.tsx +106 -0
  78. package/src/components/binary-feedback/BinaryFeedback.tsx +63 -0
  79. package/src/components/binary-feedback/BinaryFeedbackOption.tsx +111 -0
  80. package/src/components/binary-feedback/BinaryFeedbackView.tsx +172 -0
  81. package/src/components/binary-feedback/utils.ts +220 -0
  82. package/src/components/messages/AssistantBashOutputMessage.tsx +22 -0
  83. package/src/components/messages/AssistantLocalCommandOutputMessage.tsx +49 -0
  84. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +19 -0
  85. package/src/components/messages/AssistantTextMessage.tsx +144 -0
  86. package/src/components/messages/AssistantThinkingMessage.tsx +40 -0
  87. package/src/components/messages/AssistantToolUseMessage.tsx +133 -0
  88. package/src/components/messages/TaskProgressMessage.tsx +32 -0
  89. package/src/components/messages/TaskToolMessage.tsx +58 -0
  90. package/src/components/messages/UserBashInputMessage.tsx +28 -0
  91. package/src/components/messages/UserCommandMessage.tsx +30 -0
  92. package/src/components/messages/UserKodingInputMessage.tsx +28 -0
  93. package/src/components/messages/UserPromptMessage.tsx +35 -0
  94. package/src/components/messages/UserTextMessage.tsx +39 -0
  95. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +12 -0
  96. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +36 -0
  97. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +31 -0
  98. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +57 -0
  99. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +35 -0
  100. package/src/components/messages/UserToolResultMessage/utils.tsx +56 -0
  101. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +121 -0
  102. package/src/components/permissions/FallbackPermissionRequest.tsx +153 -0
  103. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +182 -0
  104. package/src/components/permissions/FileEditPermissionRequest/FileEditToolDiff.tsx +77 -0
  105. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +164 -0
  106. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +83 -0
  107. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +240 -0
  108. package/src/components/permissions/PermissionRequest.tsx +101 -0
  109. package/src/components/permissions/PermissionRequestTitle.tsx +69 -0
  110. package/src/components/permissions/hooks.ts +44 -0
  111. package/src/components/permissions/toolUseOptions.ts +59 -0
  112. package/src/components/permissions/utils.ts +23 -0
  113. package/src/constants/betas.ts +5 -0
  114. package/src/constants/claude-asterisk-ascii-art.tsx +238 -0
  115. package/src/constants/figures.ts +4 -0
  116. package/src/constants/keys.ts +3 -0
  117. package/src/constants/macros.ts +8 -0
  118. package/src/constants/modelCapabilities.ts +179 -0
  119. package/src/constants/models.ts +1025 -0
  120. package/src/constants/oauth.ts +18 -0
  121. package/src/constants/product.ts +17 -0
  122. package/src/constants/prompts.ts +177 -0
  123. package/src/constants/releaseNotes.ts +7 -0
  124. package/src/context/PermissionContext.tsx +149 -0
  125. package/src/context.ts +278 -0
  126. package/src/cost-tracker.ts +84 -0
  127. package/src/entrypoints/cli.tsx +1518 -0
  128. package/src/entrypoints/mcp.ts +176 -0
  129. package/src/history.ts +25 -0
  130. package/src/hooks/useApiKeyVerification.ts +59 -0
  131. package/src/hooks/useArrowKeyHistory.ts +55 -0
  132. package/src/hooks/useCanUseTool.ts +138 -0
  133. package/src/hooks/useCancelRequest.ts +39 -0
  134. package/src/hooks/useDoublePress.ts +42 -0
  135. package/src/hooks/useExitOnCtrlCD.ts +31 -0
  136. package/src/hooks/useInterval.ts +25 -0
  137. package/src/hooks/useLogMessages.ts +16 -0
  138. package/src/hooks/useLogStartupTime.ts +12 -0
  139. package/src/hooks/useNotifyAfterTimeout.ts +65 -0
  140. package/src/hooks/usePermissionRequestLogging.ts +44 -0
  141. package/src/hooks/useTerminalSize.ts +49 -0
  142. package/src/hooks/useTextInput.ts +318 -0
  143. package/src/hooks/useUnifiedCompletion.ts +1404 -0
  144. package/src/messages.ts +38 -0
  145. package/src/permissions.ts +268 -0
  146. package/src/query.ts +707 -0
  147. package/src/screens/ConfigureNpmPrefix.tsx +197 -0
  148. package/src/screens/Doctor.tsx +219 -0
  149. package/src/screens/LogList.tsx +68 -0
  150. package/src/screens/REPL.tsx +798 -0
  151. package/src/screens/ResumeConversation.tsx +68 -0
  152. package/src/services/adapters/base.ts +38 -0
  153. package/src/services/adapters/chatCompletions.ts +90 -0
  154. package/src/services/adapters/responsesAPI.ts +170 -0
  155. package/src/services/browserMocks.ts +66 -0
  156. package/src/services/claude.ts +2083 -0
  157. package/src/services/customCommands.ts +704 -0
  158. package/src/services/fileFreshness.ts +377 -0
  159. package/src/services/gpt5ConnectionTest.ts +340 -0
  160. package/src/services/mcpClient.ts +564 -0
  161. package/src/services/mcpServerApproval.tsx +50 -0
  162. package/src/services/mentionProcessor.ts +273 -0
  163. package/src/services/modelAdapterFactory.ts +69 -0
  164. package/src/services/notifier.ts +40 -0
  165. package/src/services/oauth.ts +357 -0
  166. package/src/services/openai.ts +1305 -0
  167. package/src/services/responseStateManager.ts +90 -0
  168. package/src/services/sentry.ts +3 -0
  169. package/src/services/statsig.ts +171 -0
  170. package/src/services/statsigStorage.ts +86 -0
  171. package/src/services/systemReminder.ts +507 -0
  172. package/src/services/vcr.ts +161 -0
  173. package/src/test/testAdapters.ts +96 -0
  174. package/src/tools/ArchitectTool/ArchitectTool.tsx +122 -0
  175. package/src/tools/ArchitectTool/prompt.ts +15 -0
  176. package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +569 -0
  177. package/src/tools/BashTool/BashTool.tsx +243 -0
  178. package/src/tools/BashTool/BashToolResultMessage.tsx +38 -0
  179. package/src/tools/BashTool/OutputLine.tsx +49 -0
  180. package/src/tools/BashTool/prompt.ts +174 -0
  181. package/src/tools/BashTool/utils.ts +56 -0
  182. package/src/tools/FileEditTool/FileEditTool.tsx +315 -0
  183. package/src/tools/FileEditTool/prompt.ts +51 -0
  184. package/src/tools/FileEditTool/utils.ts +58 -0
  185. package/src/tools/FileReadTool/FileReadTool.tsx +404 -0
  186. package/src/tools/FileReadTool/prompt.ts +7 -0
  187. package/src/tools/FileWriteTool/FileWriteTool.tsx +297 -0
  188. package/src/tools/FileWriteTool/prompt.ts +10 -0
  189. package/src/tools/GlobTool/GlobTool.tsx +119 -0
  190. package/src/tools/GlobTool/prompt.ts +8 -0
  191. package/src/tools/GrepTool/GrepTool.tsx +147 -0
  192. package/src/tools/GrepTool/prompt.ts +11 -0
  193. package/src/tools/MCPTool/MCPTool.tsx +107 -0
  194. package/src/tools/MCPTool/prompt.ts +3 -0
  195. package/src/tools/MemoryReadTool/MemoryReadTool.tsx +127 -0
  196. package/src/tools/MemoryReadTool/prompt.ts +3 -0
  197. package/src/tools/MemoryWriteTool/MemoryWriteTool.tsx +89 -0
  198. package/src/tools/MemoryWriteTool/prompt.ts +3 -0
  199. package/src/tools/MultiEditTool/MultiEditTool.tsx +366 -0
  200. package/src/tools/MultiEditTool/prompt.ts +45 -0
  201. package/src/tools/NotebookEditTool/NotebookEditTool.tsx +298 -0
  202. package/src/tools/NotebookEditTool/prompt.ts +3 -0
  203. package/src/tools/NotebookReadTool/NotebookReadTool.tsx +258 -0
  204. package/src/tools/NotebookReadTool/prompt.ts +3 -0
  205. package/src/tools/StickerRequestTool/StickerRequestTool.tsx +93 -0
  206. package/src/tools/StickerRequestTool/prompt.ts +19 -0
  207. package/src/tools/TaskTool/TaskTool.tsx +466 -0
  208. package/src/tools/TaskTool/constants.ts +1 -0
  209. package/src/tools/TaskTool/prompt.ts +92 -0
  210. package/src/tools/ThinkTool/ThinkTool.tsx +54 -0
  211. package/src/tools/ThinkTool/prompt.ts +12 -0
  212. package/src/tools/TodoWriteTool/TodoWriteTool.tsx +290 -0
  213. package/src/tools/TodoWriteTool/prompt.ts +63 -0
  214. package/src/tools/lsTool/lsTool.tsx +272 -0
  215. package/src/tools/lsTool/prompt.ts +2 -0
  216. package/src/tools.ts +63 -0
  217. package/src/types/PermissionMode.ts +120 -0
  218. package/src/types/RequestContext.ts +72 -0
  219. package/src/types/conversation.ts +51 -0
  220. package/src/types/logs.ts +58 -0
  221. package/src/types/modelCapabilities.ts +64 -0
  222. package/src/types/notebook.ts +87 -0
  223. package/src/utils/Cursor.ts +436 -0
  224. package/src/utils/PersistentShell.ts +373 -0
  225. package/src/utils/advancedFuzzyMatcher.ts +290 -0
  226. package/src/utils/agentLoader.ts +284 -0
  227. package/src/utils/agentStorage.ts +97 -0
  228. package/src/utils/array.ts +3 -0
  229. package/src/utils/ask.tsx +99 -0
  230. package/src/utils/auth.ts +13 -0
  231. package/src/utils/autoCompactCore.ts +223 -0
  232. package/src/utils/autoUpdater.ts +318 -0
  233. package/src/utils/betas.ts +20 -0
  234. package/src/utils/browser.ts +14 -0
  235. package/src/utils/cleanup.ts +72 -0
  236. package/src/utils/commands.ts +261 -0
  237. package/src/utils/commonUnixCommands.ts +161 -0
  238. package/src/utils/config.ts +942 -0
  239. package/src/utils/conversationRecovery.ts +55 -0
  240. package/src/utils/debugLogger.ts +1123 -0
  241. package/src/utils/diff.ts +42 -0
  242. package/src/utils/env.ts +57 -0
  243. package/src/utils/errors.ts +21 -0
  244. package/src/utils/exampleCommands.ts +109 -0
  245. package/src/utils/execFileNoThrow.ts +51 -0
  246. package/src/utils/expertChatStorage.ts +136 -0
  247. package/src/utils/file.ts +402 -0
  248. package/src/utils/fileRecoveryCore.ts +71 -0
  249. package/src/utils/format.tsx +44 -0
  250. package/src/utils/fuzzyMatcher.ts +328 -0
  251. package/src/utils/generators.ts +62 -0
  252. package/src/utils/git.ts +92 -0
  253. package/src/utils/globalLogger.ts +77 -0
  254. package/src/utils/http.ts +10 -0
  255. package/src/utils/imagePaste.ts +38 -0
  256. package/src/utils/json.ts +13 -0
  257. package/src/utils/log.ts +382 -0
  258. package/src/utils/markdown.ts +213 -0
  259. package/src/utils/messageContextManager.ts +289 -0
  260. package/src/utils/messages.tsx +939 -0
  261. package/src/utils/model.ts +836 -0
  262. package/src/utils/permissions/filesystem.ts +118 -0
  263. package/src/utils/responseState.ts +23 -0
  264. package/src/utils/ripgrep.ts +167 -0
  265. package/src/utils/secureFile.ts +559 -0
  266. package/src/utils/sessionState.ts +49 -0
  267. package/src/utils/state.ts +25 -0
  268. package/src/utils/style.ts +29 -0
  269. package/src/utils/terminal.ts +50 -0
  270. package/src/utils/theme.ts +133 -0
  271. package/src/utils/thinking.ts +144 -0
  272. package/src/utils/todoStorage.ts +431 -0
  273. package/src/utils/tokens.ts +43 -0
  274. package/src/utils/toolExecutionController.ts +163 -0
  275. package/src/utils/unaryLogging.ts +26 -0
  276. package/src/utils/user.ts +37 -0
  277. package/src/utils/validate.ts +165 -0
  278. package/cli.mjs +0 -1803
@@ -0,0 +1,77 @@
1
+ import * as React from 'react'
2
+ import { existsSync, readFileSync } from 'fs'
3
+ import { useMemo } from 'react'
4
+ import { StructuredDiff } from '../../StructuredDiff'
5
+ import { Box, Text } from 'ink'
6
+ import { getTheme } from '../../../utils/theme'
7
+ import { intersperse } from '../../../utils/array'
8
+ import { getCwd } from '../../../utils/state'
9
+ import { relative } from 'path'
10
+ import { getPatch } from '../../../utils/diff'
11
+
12
+ type Props = {
13
+ file_path: string
14
+ new_string: string
15
+ old_string: string
16
+ verbose: boolean
17
+ useBorder?: boolean
18
+ width: number
19
+ }
20
+
21
+ export function FileEditToolDiff({
22
+ file_path,
23
+ new_string,
24
+ old_string,
25
+ verbose,
26
+ useBorder = true,
27
+ width,
28
+ }: Props): React.ReactNode {
29
+ const file = useMemo(
30
+ () => (existsSync(file_path) ? readFileSync(file_path, 'utf8') : ''),
31
+ [file_path],
32
+ )
33
+ const patch = useMemo(
34
+ () =>
35
+ getPatch({
36
+ filePath: file_path,
37
+ fileContents: file,
38
+ oldStr: old_string,
39
+ newStr: new_string,
40
+ }),
41
+ [file_path, file, old_string, new_string],
42
+ )
43
+
44
+ return (
45
+ <Box flexDirection="column">
46
+ <Box
47
+ borderColor={getTheme().secondaryBorder}
48
+ borderStyle={useBorder ? 'round' : undefined}
49
+ flexDirection="column"
50
+ paddingX={1}
51
+ >
52
+ <Box paddingBottom={1}>
53
+ <Text bold>
54
+ {verbose ? file_path : relative(getCwd(), file_path)}
55
+ </Text>
56
+ </Box>
57
+ {intersperse(
58
+ patch.map(_ => (
59
+ <StructuredDiff
60
+ key={_.newStart}
61
+ patch={_}
62
+ dim={false}
63
+ width={width}
64
+ />
65
+ )),
66
+ i => (
67
+ <React.Fragment key={`ellipsis-${i}`}>
68
+ <Text color={getTheme().secondaryText}>
69
+ ...
70
+ </Text>
71
+ </React.Fragment>
72
+ ),
73
+ )}
74
+ </Box>
75
+ </Box>
76
+ )
77
+ }
@@ -0,0 +1,164 @@
1
+ import { Box, Text } from 'ink'
2
+ import React, { useMemo } from 'react'
3
+ import { Select } from '../../CustomSelect/select'
4
+ import { basename, extname } from 'path'
5
+ import { getTheme } from '../../../utils/theme'
6
+ import {
7
+ PermissionRequestTitle,
8
+ textColorForRiskScore,
9
+ } from '../PermissionRequestTitle.js'
10
+ import { logUnaryEvent } from '../../../utils/unaryLogging'
11
+ import { env } from '../../../utils/env'
12
+ import { savePermission } from '../../../permissions'
13
+ import {
14
+ type ToolUseConfirm,
15
+ toolUseConfirmGetPrefix,
16
+ } from '../PermissionRequest.js'
17
+ import { existsSync } from 'fs'
18
+ import chalk from 'chalk'
19
+ import {
20
+ UnaryEvent,
21
+ usePermissionRequestLogging,
22
+ } from '../../../hooks/usePermissionRequestLogging.js'
23
+ import { FileWriteToolDiff } from './FileWriteToolDiff'
24
+ import { useTerminalSize } from '../../../hooks/useTerminalSize'
25
+
26
+ type Props = {
27
+ toolUseConfirm: ToolUseConfirm
28
+ onDone(): void
29
+ verbose: boolean
30
+ }
31
+
32
+ export function FileWritePermissionRequest({
33
+ toolUseConfirm,
34
+ onDone,
35
+ verbose,
36
+ }: Props): React.ReactNode {
37
+ const { file_path, content } = toolUseConfirm.input as {
38
+ file_path: string
39
+ content: string
40
+ }
41
+ const fileExists = useMemo(() => existsSync(file_path), [file_path])
42
+ const unaryEvent = useMemo<UnaryEvent>(
43
+ () => ({
44
+ completion_type: 'write_file_single',
45
+ language_name: extractLanguageName(file_path),
46
+ }),
47
+ [file_path],
48
+ )
49
+ const { columns } = useTerminalSize()
50
+ usePermissionRequestLogging(toolUseConfirm, unaryEvent)
51
+
52
+ return (
53
+ <Box
54
+ flexDirection="column"
55
+ borderStyle="round"
56
+ borderColor={textColorForRiskScore(toolUseConfirm.riskScore)}
57
+ marginTop={1}
58
+ paddingLeft={1}
59
+ paddingRight={1}
60
+ paddingBottom={1}
61
+ >
62
+ <PermissionRequestTitle
63
+ title={`${fileExists ? 'Edit' : 'Create'} file`}
64
+ riskScore={toolUseConfirm.riskScore}
65
+ />
66
+ <Box flexDirection="column">
67
+ <FileWriteToolDiff
68
+ file_path={file_path}
69
+ content={content}
70
+ verbose={verbose}
71
+ width={columns - 12}
72
+ />
73
+ </Box>
74
+ <Box flexDirection="column">
75
+ <Text>
76
+ Do you want to {fileExists ? 'make this edit to' : 'create'}{' '}
77
+ <Text bold>{basename(file_path)}</Text>?
78
+ </Text>
79
+ <Select
80
+ options={[
81
+ {
82
+ label: 'Yes',
83
+ value: 'yes',
84
+ },
85
+ {
86
+ label: "Yes, and don't ask again this session",
87
+ value: 'yes-dont-ask-again',
88
+ },
89
+ {
90
+ label: `No, and provide instructions (${chalk.bold.hex(getTheme().warning)('esc')})`,
91
+ value: 'no',
92
+ },
93
+ ]}
94
+ onChange={newValue => {
95
+ switch (newValue) {
96
+ case 'yes':
97
+ extractLanguageName(file_path).then(language => {
98
+ logUnaryEvent({
99
+ completion_type: 'write_file_single',
100
+ event: 'accept',
101
+ metadata: {
102
+ language_name: language,
103
+ message_id: toolUseConfirm.assistantMessage.message.id,
104
+ platform: env.platform,
105
+ },
106
+ })
107
+ })
108
+ toolUseConfirm.onAllow('temporary')
109
+ onDone()
110
+ break
111
+ case 'yes-dont-ask-again':
112
+ extractLanguageName(file_path).then(language => {
113
+ logUnaryEvent({
114
+ completion_type: 'write_file_single',
115
+ event: 'accept',
116
+ metadata: {
117
+ language_name: language,
118
+ message_id: toolUseConfirm.assistantMessage.message.id,
119
+ platform: env.platform,
120
+ },
121
+ })
122
+ })
123
+ savePermission(
124
+ toolUseConfirm.tool,
125
+ toolUseConfirm.input,
126
+ toolUseConfirmGetPrefix(toolUseConfirm),
127
+ ).then(() => {
128
+ toolUseConfirm.onAllow('permanent')
129
+ onDone()
130
+ })
131
+ break
132
+ case 'no':
133
+ extractLanguageName(file_path).then(language => {
134
+ logUnaryEvent({
135
+ completion_type: 'write_file_single',
136
+ event: 'reject',
137
+ metadata: {
138
+ language_name: language,
139
+ message_id: toolUseConfirm.assistantMessage.message.id,
140
+ platform: env.platform,
141
+ },
142
+ })
143
+ })
144
+ toolUseConfirm.onReject()
145
+ onDone()
146
+ break
147
+ }
148
+ }}
149
+ />
150
+ </Box>
151
+ </Box>
152
+ )
153
+ }
154
+
155
+ async function extractLanguageName(file_path: string): Promise<string> {
156
+ const ext = extname(file_path)
157
+ if (!ext) {
158
+ return 'unknown'
159
+ }
160
+ const Highlight = (await import('highlight.js')) as unknown as {
161
+ default: { getLanguage(ext: string): { name: string | undefined } }
162
+ }
163
+ return Highlight.default.getLanguage(ext.slice(1))?.name ?? 'unknown'
164
+ }
@@ -0,0 +1,83 @@
1
+ import * as React from 'react'
2
+ import { existsSync, readFileSync } from 'fs'
3
+ import { useMemo } from 'react'
4
+ import { StructuredDiff } from '../../StructuredDiff'
5
+ import { Box, Text } from 'ink'
6
+ import { getTheme } from '../../../utils/theme'
7
+ import { intersperse } from '../../../utils/array'
8
+ import { getCwd } from '../../../utils/state'
9
+ import { extname, relative } from 'path'
10
+ import { detectFileEncoding } from '../../../utils/file'
11
+ import { HighlightedCode } from '../../HighlightedCode'
12
+ import { getPatch } from '../../../utils/diff'
13
+
14
+ type Props = {
15
+ file_path: string
16
+ content: string
17
+ verbose: boolean
18
+ width: number
19
+ }
20
+
21
+ export function FileWriteToolDiff({
22
+ file_path,
23
+ content,
24
+ verbose,
25
+ width,
26
+ }: Props): React.ReactNode {
27
+ const fileExists = useMemo(() => existsSync(file_path), [file_path])
28
+ const oldContent = useMemo(() => {
29
+ if (!fileExists) {
30
+ return ''
31
+ }
32
+ const enc = detectFileEncoding(file_path)
33
+ return readFileSync(file_path, enc)
34
+ }, [file_path, fileExists])
35
+ const hunks = useMemo(() => {
36
+ if (!fileExists) {
37
+ return null
38
+ }
39
+ return getPatch({
40
+ filePath: file_path,
41
+ fileContents: oldContent,
42
+ oldStr: oldContent,
43
+ newStr: content,
44
+ })
45
+ }, [fileExists, file_path, oldContent, content])
46
+
47
+ return (
48
+ <Box
49
+ borderColor={getTheme().secondaryBorder}
50
+ borderStyle="round"
51
+ flexDirection="column"
52
+ paddingX={1}
53
+ >
54
+ <Box paddingBottom={1}>
55
+ <Text bold>{verbose ? file_path : relative(getCwd(), file_path)}</Text>
56
+ </Box>
57
+ {hunks ? (
58
+ intersperse(
59
+ hunks.map(_ => (
60
+ <StructuredDiff
61
+ key={_.newStart}
62
+ patch={_}
63
+ dim={false}
64
+ width={width}
65
+ />
66
+ )),
67
+ i => (
68
+ <React.Fragment key={`ellipsis-${i}`}>
69
+ <Text color={getTheme().secondaryText}>
70
+ ...
71
+ </Text>
72
+ </React.Fragment>
73
+ ),
74
+ )
75
+ ) : (
76
+ <HighlightedCode
77
+ code={content || '(No content)'}
78
+ language={extname(file_path).slice(1)}
79
+ />
80
+ )}
81
+ </Box>
82
+ )
83
+ }
@@ -0,0 +1,240 @@
1
+ import { Box, Text } from 'ink'
2
+ import React, { useMemo } from 'react'
3
+ import { Select } from '../../CustomSelect/select'
4
+ import { getTheme } from '../../../utils/theme'
5
+ import {
6
+ PermissionRequestTitle,
7
+ textColorForRiskScore,
8
+ } from '../PermissionRequestTitle.js'
9
+ import { logUnaryEvent } from '../../../utils/unaryLogging'
10
+ import { env } from '../../../utils/env'
11
+ import {
12
+ type PermissionRequestProps,
13
+ type ToolUseConfirm,
14
+ } from '../PermissionRequest.js'
15
+ import chalk from 'chalk'
16
+ import {
17
+ UnaryEvent,
18
+ usePermissionRequestLogging,
19
+ } from '../../../hooks/usePermissionRequestLogging.js'
20
+ import { FileEditTool } from '../../../tools/FileEditTool/FileEditTool'
21
+ import { FileWriteTool } from '../../../tools/FileWriteTool/FileWriteTool'
22
+ import { GrepTool } from '../../../tools/GrepTool/GrepTool'
23
+ import { GlobTool } from '../../../tools/GlobTool/GlobTool'
24
+ import { LSTool } from '../../../tools/lsTool/lsTool'
25
+ import { FileReadTool } from '../../../tools/FileReadTool/FileReadTool'
26
+ import { NotebookEditTool } from '../../../tools/NotebookEditTool/NotebookEditTool'
27
+ import { NotebookReadTool } from '../../../tools/NotebookReadTool/NotebookReadTool'
28
+ import { FallbackPermissionRequest } from '../FallbackPermissionRequest'
29
+ import {
30
+ grantWritePermissionForOriginalDir,
31
+ pathInOriginalCwd,
32
+ toAbsolutePath,
33
+ } from '../../../utils/permissions/filesystem.js'
34
+ import { getCwd } from '../../../utils/state'
35
+
36
+ function pathArgNameForToolUse(toolUseConfirm: ToolUseConfirm): string | null {
37
+ switch (toolUseConfirm.tool) {
38
+ case FileWriteTool:
39
+ case FileEditTool:
40
+ case FileReadTool: {
41
+ return 'file_path'
42
+ }
43
+ case GlobTool:
44
+ case GrepTool:
45
+ case LSTool: {
46
+ return 'path'
47
+ }
48
+ case NotebookEditTool:
49
+ case NotebookReadTool: {
50
+ return 'notebook_path'
51
+ }
52
+ }
53
+ return null
54
+ }
55
+
56
+ function isMultiFile(toolUseConfirm: ToolUseConfirm): boolean {
57
+ switch (toolUseConfirm.tool) {
58
+ case GlobTool:
59
+ case GrepTool:
60
+ case LSTool: {
61
+ return true
62
+ }
63
+ }
64
+ return false
65
+ }
66
+
67
+ function pathFromToolUse(toolUseConfirm: ToolUseConfirm): string | null {
68
+ const pathArgName = pathArgNameForToolUse(toolUseConfirm)
69
+ const input = toolUseConfirm.input
70
+ if (pathArgName && pathArgName in input) {
71
+ if (typeof input[pathArgName] === 'string') {
72
+ return toAbsolutePath(input[pathArgName])
73
+ } else {
74
+ return toAbsolutePath(getCwd())
75
+ }
76
+ }
77
+ return null
78
+ }
79
+
80
+ export function FilesystemPermissionRequest({
81
+ toolUseConfirm,
82
+ onDone,
83
+ verbose,
84
+ }: PermissionRequestProps): React.ReactNode {
85
+ const path = pathFromToolUse(toolUseConfirm)
86
+ if (!path) {
87
+ // Fall back to generic permission request if no path is found
88
+ return (
89
+ <FallbackPermissionRequest
90
+ toolUseConfirm={toolUseConfirm}
91
+ onDone={onDone}
92
+ verbose={verbose}
93
+ />
94
+ )
95
+ }
96
+ return (
97
+ <FilesystemPermissionRequestImpl
98
+ toolUseConfirm={toolUseConfirm}
99
+ path={path}
100
+ onDone={onDone}
101
+ verbose={verbose}
102
+ />
103
+ )
104
+ }
105
+
106
+ function getDontAskAgainOptions(toolUseConfirm: ToolUseConfirm, path: string) {
107
+ if (toolUseConfirm.tool.isReadOnly()) {
108
+ // "Always allow" is not an option for read-only tools,
109
+ // because they always have write permission in the project directory.
110
+ return []
111
+ }
112
+ // Only show don't ask again option for edits in original working directory
113
+ return pathInOriginalCwd(path)
114
+ ? [
115
+ {
116
+ label: "Yes, and don't ask again for file edits this session",
117
+ value: 'yes-dont-ask-again',
118
+ },
119
+ ]
120
+ : []
121
+ }
122
+
123
+ type Props = {
124
+ toolUseConfirm: ToolUseConfirm
125
+ path: string
126
+ onDone(): void
127
+ verbose: boolean
128
+ }
129
+
130
+ function FilesystemPermissionRequestImpl({
131
+ toolUseConfirm,
132
+ path,
133
+ onDone,
134
+ verbose,
135
+ }: Props): React.ReactNode {
136
+ const userFacingName = toolUseConfirm.tool.userFacingName()
137
+
138
+ const userFacingReadOrWrite = toolUseConfirm.tool.isReadOnly()
139
+ ? 'Read'
140
+ : 'Edit'
141
+ const title = `${userFacingReadOrWrite} ${isMultiFile(toolUseConfirm) ? 'files' : 'file'}`
142
+
143
+ const unaryEvent = useMemo<UnaryEvent>(
144
+ () => ({
145
+ completion_type: 'tool_use_single',
146
+ language_name: 'none',
147
+ }),
148
+ [],
149
+ )
150
+
151
+ usePermissionRequestLogging(toolUseConfirm, unaryEvent)
152
+
153
+ return (
154
+ <Box
155
+ flexDirection="column"
156
+ borderStyle="round"
157
+ borderColor={textColorForRiskScore(toolUseConfirm.riskScore)}
158
+ marginTop={1}
159
+ paddingLeft={1}
160
+ paddingRight={1}
161
+ paddingBottom={1}
162
+ >
163
+ <PermissionRequestTitle
164
+ title={title}
165
+ riskScore={toolUseConfirm.riskScore}
166
+ />
167
+ <Box flexDirection="column" paddingX={2} paddingY={1}>
168
+ <Text>
169
+ {userFacingName}(
170
+ {toolUseConfirm.tool.renderToolUseMessage(
171
+ toolUseConfirm.input as never,
172
+ { verbose },
173
+ )}
174
+ )
175
+ </Text>
176
+ </Box>
177
+
178
+ <Box flexDirection="column">
179
+ <Text>Do you want to proceed?</Text>
180
+ <Select
181
+ options={[
182
+ {
183
+ label: 'Yes',
184
+ value: 'yes',
185
+ },
186
+ ...getDontAskAgainOptions(toolUseConfirm, path),
187
+ {
188
+ label: `No, and provide instructions (${chalk.bold.hex(getTheme().warning)('esc')})`,
189
+ value: 'no',
190
+ },
191
+ ]}
192
+ onChange={newValue => {
193
+ switch (newValue) {
194
+ case 'yes':
195
+ logUnaryEvent({
196
+ completion_type: 'tool_use_single',
197
+ event: 'accept',
198
+ metadata: {
199
+ language_name: 'none',
200
+ message_id: toolUseConfirm.assistantMessage.message.id,
201
+ platform: env.platform,
202
+ },
203
+ })
204
+ toolUseConfirm.onAllow('temporary')
205
+ onDone()
206
+ break
207
+ case 'yes-dont-ask-again':
208
+ logUnaryEvent({
209
+ completion_type: 'tool_use_single',
210
+ event: 'accept',
211
+ metadata: {
212
+ language_name: 'none',
213
+ message_id: toolUseConfirm.assistantMessage.message.id,
214
+ platform: env.platform,
215
+ },
216
+ })
217
+ grantWritePermissionForOriginalDir()
218
+ toolUseConfirm.onAllow('permanent')
219
+ onDone()
220
+ break
221
+ case 'no':
222
+ logUnaryEvent({
223
+ completion_type: 'tool_use_single',
224
+ event: 'reject',
225
+ metadata: {
226
+ language_name: 'none',
227
+ message_id: toolUseConfirm.assistantMessage.message.id,
228
+ platform: env.platform,
229
+ },
230
+ })
231
+ toolUseConfirm.onReject()
232
+ onDone()
233
+ break
234
+ }
235
+ }}
236
+ />
237
+ </Box>
238
+ </Box>
239
+ )
240
+ }
@@ -0,0 +1,101 @@
1
+ import { useInput } from 'ink'
2
+ import * as React from 'react'
3
+ import { Tool } from '../../Tool'
4
+ import { AssistantMessage } from '../../query'
5
+ import { FileEditTool } from '../../tools/FileEditTool/FileEditTool'
6
+ import { FileWriteTool } from '../../tools/FileWriteTool/FileWriteTool'
7
+ import { BashTool } from '../../tools/BashTool/BashTool'
8
+ import { FileEditPermissionRequest } from './FileEditPermissionRequest/FileEditPermissionRequest'
9
+ import { BashPermissionRequest } from './BashPermissionRequest/BashPermissionRequest'
10
+ import { FallbackPermissionRequest } from './FallbackPermissionRequest'
11
+ import { useNotifyAfterTimeout } from '../../hooks/useNotifyAfterTimeout'
12
+ import { FileWritePermissionRequest } from './FileWritePermissionRequest/FileWritePermissionRequest'
13
+ import { type CommandSubcommandPrefixResult } from '../../utils/commands'
14
+ import { FilesystemPermissionRequest } from './FilesystemPermissionRequest/FilesystemPermissionRequest'
15
+ import { NotebookEditTool } from '../../tools/NotebookEditTool/NotebookEditTool'
16
+ import { GlobTool } from '../../tools/GlobTool/GlobTool'
17
+ import { GrepTool } from '../../tools/GrepTool/GrepTool'
18
+ import { LSTool } from '../../tools/lsTool/lsTool'
19
+ import { FileReadTool } from '../../tools/FileReadTool/FileReadTool'
20
+ import { NotebookReadTool } from '../../tools/NotebookReadTool/NotebookReadTool'
21
+ import { PRODUCT_NAME } from '../../constants/product'
22
+
23
+ function permissionComponentForTool(tool: Tool) {
24
+ switch (tool) {
25
+ case FileEditTool:
26
+ return FileEditPermissionRequest
27
+ case FileWriteTool:
28
+ return FileWritePermissionRequest
29
+ case BashTool:
30
+ return BashPermissionRequest
31
+ case GlobTool:
32
+ case GrepTool:
33
+ case LSTool:
34
+ case FileReadTool:
35
+ case NotebookReadTool:
36
+ case NotebookEditTool:
37
+ return FilesystemPermissionRequest
38
+ default:
39
+ return FallbackPermissionRequest
40
+ }
41
+ }
42
+
43
+ export type PermissionRequestProps = {
44
+ toolUseConfirm: ToolUseConfirm
45
+ onDone(): void
46
+ verbose: boolean
47
+ }
48
+
49
+ export function toolUseConfirmGetPrefix(
50
+ toolUseConfirm: ToolUseConfirm,
51
+ ): string | null {
52
+ return (
53
+ (toolUseConfirm.commandPrefix &&
54
+ !(toolUseConfirm.commandPrefix as any).commandInjectionDetected &&
55
+ (toolUseConfirm.commandPrefix as any).commandPrefix) ||
56
+ null
57
+ )
58
+ }
59
+
60
+ export type ToolUseConfirm = {
61
+ assistantMessage: AssistantMessage
62
+ tool: Tool
63
+ description: string
64
+ input: { [key: string]: unknown }
65
+ commandPrefix: CommandSubcommandPrefixResult | null
66
+ // TODO: remove riskScore from ToolUseConfirm
67
+ riskScore: number | null
68
+ onAbort(): void
69
+ onAllow(type: 'permanent' | 'temporary'): void
70
+ onReject(): void
71
+ }
72
+
73
+ // TODO: Move this to Tool.renderPermissionRequest
74
+ export function PermissionRequest({
75
+ toolUseConfirm,
76
+ onDone,
77
+ verbose,
78
+ }: PermissionRequestProps): React.ReactNode {
79
+ // Handle Ctrl+C
80
+ useInput((input, key) => {
81
+ if (key.ctrl && input === 'c') {
82
+ onDone()
83
+ toolUseConfirm.onReject()
84
+ }
85
+ })
86
+
87
+ const toolName = toolUseConfirm.tool.userFacingName?.() || 'Tool'
88
+ useNotifyAfterTimeout(
89
+ `${PRODUCT_NAME} needs your permission to use ${toolName}`,
90
+ )
91
+
92
+ const PermissionComponent = permissionComponentForTool(toolUseConfirm.tool)
93
+
94
+ return (
95
+ <PermissionComponent
96
+ toolUseConfirm={toolUseConfirm}
97
+ onDone={onDone}
98
+ verbose={verbose}
99
+ />
100
+ )
101
+ }