@shareai-lab/kode 1.0.69 → 1.0.71

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 (253) hide show
  1. package/README.md +205 -72
  2. package/README.zh-CN.md +246 -0
  3. package/cli.js +62 -0
  4. package/package.json +45 -25
  5. package/scripts/postinstall.js +56 -0
  6. package/src/ProjectOnboarding.tsx +180 -0
  7. package/src/Tool.ts +53 -0
  8. package/src/commands/approvedTools.ts +53 -0
  9. package/src/commands/bug.tsx +20 -0
  10. package/src/commands/clear.ts +43 -0
  11. package/src/commands/compact.ts +120 -0
  12. package/src/commands/config.tsx +19 -0
  13. package/src/commands/cost.ts +18 -0
  14. package/src/commands/ctx_viz.ts +209 -0
  15. package/src/commands/doctor.ts +24 -0
  16. package/src/commands/help.tsx +19 -0
  17. package/src/commands/init.ts +37 -0
  18. package/src/commands/listen.ts +42 -0
  19. package/src/commands/login.tsx +51 -0
  20. package/src/commands/logout.tsx +40 -0
  21. package/src/commands/mcp.ts +41 -0
  22. package/src/commands/model.tsx +40 -0
  23. package/src/commands/modelstatus.tsx +20 -0
  24. package/src/commands/onboarding.tsx +34 -0
  25. package/src/commands/pr_comments.ts +59 -0
  26. package/src/commands/refreshCommands.ts +54 -0
  27. package/src/commands/release-notes.ts +34 -0
  28. package/src/commands/resume.tsx +30 -0
  29. package/src/commands/review.ts +49 -0
  30. package/src/commands/terminalSetup.ts +221 -0
  31. package/src/commands.ts +136 -0
  32. package/src/components/ApproveApiKey.tsx +93 -0
  33. package/src/components/AsciiLogo.tsx +13 -0
  34. package/src/components/AutoUpdater.tsx +148 -0
  35. package/src/components/Bug.tsx +367 -0
  36. package/src/components/Config.tsx +289 -0
  37. package/src/components/ConsoleOAuthFlow.tsx +326 -0
  38. package/src/components/Cost.tsx +23 -0
  39. package/src/components/CostThresholdDialog.tsx +46 -0
  40. package/src/components/CustomSelect/option-map.ts +42 -0
  41. package/src/components/CustomSelect/select-option.tsx +52 -0
  42. package/src/components/CustomSelect/select.tsx +143 -0
  43. package/src/components/CustomSelect/use-select-state.ts +414 -0
  44. package/src/components/CustomSelect/use-select.ts +35 -0
  45. package/src/components/FallbackToolUseRejectedMessage.tsx +15 -0
  46. package/src/components/FileEditToolUpdatedMessage.tsx +66 -0
  47. package/src/components/Help.tsx +215 -0
  48. package/src/components/HighlightedCode.tsx +33 -0
  49. package/src/components/InvalidConfigDialog.tsx +113 -0
  50. package/src/components/Link.tsx +32 -0
  51. package/src/components/LogSelector.tsx +86 -0
  52. package/src/components/Logo.tsx +145 -0
  53. package/src/components/MCPServerApprovalDialog.tsx +100 -0
  54. package/src/components/MCPServerDialogCopy.tsx +25 -0
  55. package/src/components/MCPServerMultiselectDialog.tsx +109 -0
  56. package/src/components/Message.tsx +219 -0
  57. package/src/components/MessageResponse.tsx +15 -0
  58. package/src/components/MessageSelector.tsx +211 -0
  59. package/src/components/ModeIndicator.tsx +88 -0
  60. package/src/components/ModelConfig.tsx +301 -0
  61. package/src/components/ModelListManager.tsx +223 -0
  62. package/src/components/ModelSelector.tsx +3208 -0
  63. package/src/components/ModelStatusDisplay.tsx +228 -0
  64. package/src/components/Onboarding.tsx +274 -0
  65. package/src/components/PressEnterToContinue.tsx +11 -0
  66. package/src/components/PromptInput.tsx +710 -0
  67. package/src/components/SentryErrorBoundary.ts +33 -0
  68. package/src/components/Spinner.tsx +129 -0
  69. package/src/components/StructuredDiff.tsx +184 -0
  70. package/src/components/TextInput.tsx +246 -0
  71. package/src/components/TokenWarning.tsx +31 -0
  72. package/src/components/ToolUseLoader.tsx +40 -0
  73. package/src/components/TrustDialog.tsx +106 -0
  74. package/src/components/binary-feedback/BinaryFeedback.tsx +63 -0
  75. package/src/components/binary-feedback/BinaryFeedbackOption.tsx +111 -0
  76. package/src/components/binary-feedback/BinaryFeedbackView.tsx +172 -0
  77. package/src/components/binary-feedback/utils.ts +220 -0
  78. package/src/components/messages/AssistantBashOutputMessage.tsx +22 -0
  79. package/src/components/messages/AssistantLocalCommandOutputMessage.tsx +45 -0
  80. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +19 -0
  81. package/src/components/messages/AssistantTextMessage.tsx +144 -0
  82. package/src/components/messages/AssistantThinkingMessage.tsx +40 -0
  83. package/src/components/messages/AssistantToolUseMessage.tsx +123 -0
  84. package/src/components/messages/UserBashInputMessage.tsx +28 -0
  85. package/src/components/messages/UserCommandMessage.tsx +30 -0
  86. package/src/components/messages/UserKodingInputMessage.tsx +28 -0
  87. package/src/components/messages/UserPromptMessage.tsx +35 -0
  88. package/src/components/messages/UserTextMessage.tsx +39 -0
  89. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +12 -0
  90. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +36 -0
  91. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +31 -0
  92. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +57 -0
  93. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +35 -0
  94. package/src/components/messages/UserToolResultMessage/utils.tsx +56 -0
  95. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +121 -0
  96. package/src/components/permissions/FallbackPermissionRequest.tsx +155 -0
  97. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +182 -0
  98. package/src/components/permissions/FileEditPermissionRequest/FileEditToolDiff.tsx +75 -0
  99. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +164 -0
  100. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +81 -0
  101. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +242 -0
  102. package/src/components/permissions/PermissionRequest.tsx +103 -0
  103. package/src/components/permissions/PermissionRequestTitle.tsx +69 -0
  104. package/src/components/permissions/hooks.ts +44 -0
  105. package/src/components/permissions/toolUseOptions.ts +59 -0
  106. package/src/components/permissions/utils.ts +23 -0
  107. package/src/constants/betas.ts +5 -0
  108. package/src/constants/claude-asterisk-ascii-art.tsx +238 -0
  109. package/src/constants/figures.ts +4 -0
  110. package/src/constants/keys.ts +3 -0
  111. package/src/constants/macros.ts +6 -0
  112. package/src/constants/models.ts +935 -0
  113. package/src/constants/oauth.ts +18 -0
  114. package/src/constants/product.ts +17 -0
  115. package/src/constants/prompts.ts +177 -0
  116. package/src/constants/releaseNotes.ts +7 -0
  117. package/src/context/PermissionContext.tsx +149 -0
  118. package/src/context.ts +278 -0
  119. package/src/cost-tracker.ts +84 -0
  120. package/src/entrypoints/cli.tsx +1498 -0
  121. package/src/entrypoints/mcp.ts +176 -0
  122. package/src/history.ts +25 -0
  123. package/src/hooks/useApiKeyVerification.ts +59 -0
  124. package/src/hooks/useArrowKeyHistory.ts +55 -0
  125. package/src/hooks/useCanUseTool.ts +138 -0
  126. package/src/hooks/useCancelRequest.ts +39 -0
  127. package/src/hooks/useDoublePress.ts +42 -0
  128. package/src/hooks/useExitOnCtrlCD.ts +31 -0
  129. package/src/hooks/useInterval.ts +25 -0
  130. package/src/hooks/useLogMessages.ts +16 -0
  131. package/src/hooks/useLogStartupTime.ts +12 -0
  132. package/src/hooks/useNotifyAfterTimeout.ts +65 -0
  133. package/src/hooks/usePermissionRequestLogging.ts +44 -0
  134. package/src/hooks/useSlashCommandTypeahead.ts +137 -0
  135. package/src/hooks/useTerminalSize.ts +49 -0
  136. package/src/hooks/useTextInput.ts +315 -0
  137. package/src/messages.ts +37 -0
  138. package/src/permissions.ts +268 -0
  139. package/src/query.ts +704 -0
  140. package/src/screens/ConfigureNpmPrefix.tsx +197 -0
  141. package/src/screens/Doctor.tsx +219 -0
  142. package/src/screens/LogList.tsx +68 -0
  143. package/src/screens/REPL.tsx +792 -0
  144. package/src/screens/ResumeConversation.tsx +68 -0
  145. package/src/services/browserMocks.ts +66 -0
  146. package/src/services/claude.ts +1947 -0
  147. package/src/services/customCommands.ts +683 -0
  148. package/src/services/fileFreshness.ts +377 -0
  149. package/src/services/mcpClient.ts +564 -0
  150. package/src/services/mcpServerApproval.tsx +50 -0
  151. package/src/services/notifier.ts +40 -0
  152. package/src/services/oauth.ts +357 -0
  153. package/src/services/openai.ts +796 -0
  154. package/src/services/sentry.ts +3 -0
  155. package/src/services/statsig.ts +171 -0
  156. package/src/services/statsigStorage.ts +86 -0
  157. package/src/services/systemReminder.ts +406 -0
  158. package/src/services/vcr.ts +161 -0
  159. package/src/tools/ArchitectTool/ArchitectTool.tsx +122 -0
  160. package/src/tools/ArchitectTool/prompt.ts +15 -0
  161. package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +505 -0
  162. package/src/tools/BashTool/BashTool.tsx +270 -0
  163. package/src/tools/BashTool/BashToolResultMessage.tsx +38 -0
  164. package/src/tools/BashTool/OutputLine.tsx +48 -0
  165. package/src/tools/BashTool/prompt.ts +174 -0
  166. package/src/tools/BashTool/utils.ts +56 -0
  167. package/src/tools/FileEditTool/FileEditTool.tsx +316 -0
  168. package/src/tools/FileEditTool/prompt.ts +51 -0
  169. package/src/tools/FileEditTool/utils.ts +58 -0
  170. package/src/tools/FileReadTool/FileReadTool.tsx +371 -0
  171. package/src/tools/FileReadTool/prompt.ts +7 -0
  172. package/src/tools/FileWriteTool/FileWriteTool.tsx +297 -0
  173. package/src/tools/FileWriteTool/prompt.ts +10 -0
  174. package/src/tools/GlobTool/GlobTool.tsx +119 -0
  175. package/src/tools/GlobTool/prompt.ts +8 -0
  176. package/src/tools/GrepTool/GrepTool.tsx +147 -0
  177. package/src/tools/GrepTool/prompt.ts +11 -0
  178. package/src/tools/MCPTool/MCPTool.tsx +106 -0
  179. package/src/tools/MCPTool/prompt.ts +3 -0
  180. package/src/tools/MemoryReadTool/MemoryReadTool.tsx +127 -0
  181. package/src/tools/MemoryReadTool/prompt.ts +3 -0
  182. package/src/tools/MemoryWriteTool/MemoryWriteTool.tsx +89 -0
  183. package/src/tools/MemoryWriteTool/prompt.ts +3 -0
  184. package/src/tools/MultiEditTool/MultiEditTool.tsx +366 -0
  185. package/src/tools/MultiEditTool/prompt.ts +45 -0
  186. package/src/tools/NotebookEditTool/NotebookEditTool.tsx +298 -0
  187. package/src/tools/NotebookEditTool/prompt.ts +3 -0
  188. package/src/tools/NotebookReadTool/NotebookReadTool.tsx +266 -0
  189. package/src/tools/NotebookReadTool/prompt.ts +3 -0
  190. package/src/tools/StickerRequestTool/StickerRequestTool.tsx +93 -0
  191. package/src/tools/StickerRequestTool/prompt.ts +19 -0
  192. package/src/tools/TaskTool/TaskTool.tsx +382 -0
  193. package/src/tools/TaskTool/constants.ts +1 -0
  194. package/src/tools/TaskTool/prompt.ts +56 -0
  195. package/src/tools/ThinkTool/ThinkTool.tsx +56 -0
  196. package/src/tools/ThinkTool/prompt.ts +12 -0
  197. package/src/tools/TodoWriteTool/TodoWriteTool.tsx +289 -0
  198. package/src/tools/TodoWriteTool/prompt.ts +63 -0
  199. package/src/tools/lsTool/lsTool.tsx +269 -0
  200. package/src/tools/lsTool/prompt.ts +2 -0
  201. package/src/tools.ts +63 -0
  202. package/src/types/PermissionMode.ts +120 -0
  203. package/src/types/RequestContext.ts +72 -0
  204. package/src/utils/Cursor.ts +436 -0
  205. package/src/utils/PersistentShell.ts +373 -0
  206. package/src/utils/agentStorage.ts +97 -0
  207. package/src/utils/array.ts +3 -0
  208. package/src/utils/ask.tsx +98 -0
  209. package/src/utils/auth.ts +13 -0
  210. package/src/utils/autoCompactCore.ts +223 -0
  211. package/src/utils/autoUpdater.ts +318 -0
  212. package/src/utils/betas.ts +20 -0
  213. package/src/utils/browser.ts +14 -0
  214. package/src/utils/cleanup.ts +72 -0
  215. package/src/utils/commands.ts +261 -0
  216. package/src/utils/config.ts +771 -0
  217. package/src/utils/conversationRecovery.ts +54 -0
  218. package/src/utils/debugLogger.ts +1123 -0
  219. package/src/utils/diff.ts +42 -0
  220. package/src/utils/env.ts +57 -0
  221. package/src/utils/errors.ts +21 -0
  222. package/src/utils/exampleCommands.ts +108 -0
  223. package/src/utils/execFileNoThrow.ts +51 -0
  224. package/src/utils/expertChatStorage.ts +136 -0
  225. package/src/utils/file.ts +402 -0
  226. package/src/utils/fileRecoveryCore.ts +71 -0
  227. package/src/utils/format.tsx +44 -0
  228. package/src/utils/generators.ts +62 -0
  229. package/src/utils/git.ts +92 -0
  230. package/src/utils/globalLogger.ts +77 -0
  231. package/src/utils/http.ts +10 -0
  232. package/src/utils/imagePaste.ts +38 -0
  233. package/src/utils/json.ts +13 -0
  234. package/src/utils/log.ts +382 -0
  235. package/src/utils/markdown.ts +213 -0
  236. package/src/utils/messageContextManager.ts +289 -0
  237. package/src/utils/messages.tsx +938 -0
  238. package/src/utils/model.ts +836 -0
  239. package/src/utils/permissions/filesystem.ts +118 -0
  240. package/src/utils/ripgrep.ts +167 -0
  241. package/src/utils/sessionState.ts +49 -0
  242. package/src/utils/state.ts +25 -0
  243. package/src/utils/style.ts +29 -0
  244. package/src/utils/terminal.ts +49 -0
  245. package/src/utils/theme.ts +122 -0
  246. package/src/utils/thinking.ts +144 -0
  247. package/src/utils/todoStorage.ts +431 -0
  248. package/src/utils/tokens.ts +43 -0
  249. package/src/utils/toolExecutionController.ts +163 -0
  250. package/src/utils/unaryLogging.ts +26 -0
  251. package/src/utils/user.ts +37 -0
  252. package/src/utils/validate.ts +165 -0
  253. package/cli.mjs +0 -1803
@@ -0,0 +1,164 @@
1
+ import { Box, Text } from 'ink'
2
+ import React, { useMemo } from 'react'
3
+ import { Select } from '@inkjs/ui'
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,81 @@
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
+ <Text color={getTheme().secondaryText} key={`ellipsis-${i}`}>
69
+ ...
70
+ </Text>
71
+ ),
72
+ )
73
+ ) : (
74
+ <HighlightedCode
75
+ code={content || '(No content)'}
76
+ language={extname(file_path).slice(1)}
77
+ />
78
+ )}
79
+ </Box>
80
+ )
81
+ }
@@ -0,0 +1,242 @@
1
+ import { Box, Text } from 'ink'
2
+ import React, { useMemo } from 'react'
3
+ import { Select } from '@inkjs/ui'
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
+ toolUseConfirm.input as never,
138
+ )
139
+
140
+ const userFacingReadOrWrite = toolUseConfirm.tool.isReadOnly()
141
+ ? 'Read'
142
+ : 'Edit'
143
+ const title = `${userFacingReadOrWrite} ${isMultiFile(toolUseConfirm) ? 'files' : 'file'}`
144
+
145
+ const unaryEvent = useMemo<UnaryEvent>(
146
+ () => ({
147
+ completion_type: 'tool_use_single',
148
+ language_name: 'none',
149
+ }),
150
+ [],
151
+ )
152
+
153
+ usePermissionRequestLogging(toolUseConfirm, unaryEvent)
154
+
155
+ return (
156
+ <Box
157
+ flexDirection="column"
158
+ borderStyle="round"
159
+ borderColor={textColorForRiskScore(toolUseConfirm.riskScore)}
160
+ marginTop={1}
161
+ paddingLeft={1}
162
+ paddingRight={1}
163
+ paddingBottom={1}
164
+ >
165
+ <PermissionRequestTitle
166
+ title={title}
167
+ riskScore={toolUseConfirm.riskScore}
168
+ />
169
+ <Box flexDirection="column" paddingX={2} paddingY={1}>
170
+ <Text>
171
+ {userFacingName}(
172
+ {toolUseConfirm.tool.renderToolUseMessage(
173
+ toolUseConfirm.input as never,
174
+ { verbose },
175
+ )}
176
+ )
177
+ </Text>
178
+ </Box>
179
+
180
+ <Box flexDirection="column">
181
+ <Text>Do you want to proceed?</Text>
182
+ <Select
183
+ options={[
184
+ {
185
+ label: 'Yes',
186
+ value: 'yes',
187
+ },
188
+ ...getDontAskAgainOptions(toolUseConfirm, path),
189
+ {
190
+ label: `No, and provide instructions (${chalk.bold.hex(getTheme().warning)('esc')})`,
191
+ value: 'no',
192
+ },
193
+ ]}
194
+ onChange={newValue => {
195
+ switch (newValue) {
196
+ case 'yes':
197
+ logUnaryEvent({
198
+ completion_type: 'tool_use_single',
199
+ event: 'accept',
200
+ metadata: {
201
+ language_name: 'none',
202
+ message_id: toolUseConfirm.assistantMessage.message.id,
203
+ platform: env.platform,
204
+ },
205
+ })
206
+ toolUseConfirm.onAllow('temporary')
207
+ onDone()
208
+ break
209
+ case 'yes-dont-ask-again':
210
+ logUnaryEvent({
211
+ completion_type: 'tool_use_single',
212
+ event: 'accept',
213
+ metadata: {
214
+ language_name: 'none',
215
+ message_id: toolUseConfirm.assistantMessage.message.id,
216
+ platform: env.platform,
217
+ },
218
+ })
219
+ grantWritePermissionForOriginalDir()
220
+ toolUseConfirm.onAllow('permanent')
221
+ onDone()
222
+ break
223
+ case 'no':
224
+ logUnaryEvent({
225
+ completion_type: 'tool_use_single',
226
+ event: 'reject',
227
+ metadata: {
228
+ language_name: 'none',
229
+ message_id: toolUseConfirm.assistantMessage.message.id,
230
+ platform: env.platform,
231
+ },
232
+ })
233
+ toolUseConfirm.onReject()
234
+ onDone()
235
+ break
236
+ }
237
+ }}
238
+ />
239
+ </Box>
240
+ </Box>
241
+ )
242
+ }
@@ -0,0 +1,103 @@
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.commandInjectionDetected &&
55
+ toolUseConfirm.commandPrefix.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(
88
+ toolUseConfirm.input as never,
89
+ )
90
+ useNotifyAfterTimeout(
91
+ `${PRODUCT_NAME} needs your permission to use ${toolName}`,
92
+ )
93
+
94
+ const PermissionComponent = permissionComponentForTool(toolUseConfirm.tool)
95
+
96
+ return (
97
+ <PermissionComponent
98
+ toolUseConfirm={toolUseConfirm}
99
+ onDone={onDone}
100
+ verbose={verbose}
101
+ />
102
+ )
103
+ }
@@ -0,0 +1,69 @@
1
+ import * as React from 'react'
2
+ import { Box, Text } from 'ink'
3
+ import { getTheme } from '../../utils/theme'
4
+
5
+ export type RiskScoreCategory = 'low' | 'moderate' | 'high'
6
+
7
+ export function categoryForRiskScore(riskScore: number): RiskScoreCategory {
8
+ return riskScore >= 70 ? 'high' : riskScore >= 30 ? 'moderate' : 'low'
9
+ }
10
+
11
+ function colorSchemeForRiskScoreCategory(category: RiskScoreCategory): {
12
+ highlightColor: string
13
+ textColor: string
14
+ } {
15
+ const theme = getTheme()
16
+ switch (category) {
17
+ case 'low':
18
+ return {
19
+ highlightColor: theme.success,
20
+ textColor: theme.permission,
21
+ }
22
+ case 'moderate':
23
+ return {
24
+ highlightColor: theme.warning,
25
+ textColor: theme.warning,
26
+ }
27
+ case 'high':
28
+ return {
29
+ highlightColor: theme.error,
30
+ textColor: theme.error,
31
+ }
32
+ }
33
+ }
34
+
35
+ export function textColorForRiskScore(riskScore: number | null): string {
36
+ if (riskScore === null) {
37
+ return getTheme().permission
38
+ }
39
+ const category = categoryForRiskScore(riskScore)
40
+ return colorSchemeForRiskScoreCategory(category).textColor
41
+ }
42
+
43
+ export function PermissionRiskScore({
44
+ riskScore,
45
+ }: {
46
+ riskScore: number
47
+ }): React.ReactNode {
48
+ const category = categoryForRiskScore(riskScore)
49
+ return <Text color={textColorForRiskScore(riskScore)}>Risk: {category}</Text>
50
+ }
51
+
52
+ type Props = {
53
+ title: string
54
+ riskScore: number | null
55
+ }
56
+
57
+ export function PermissionRequestTitle({
58
+ title,
59
+ riskScore,
60
+ }: Props): React.ReactNode {
61
+ return (
62
+ <Box flexDirection="column">
63
+ <Text bold color={getTheme().permission}>
64
+ {title}
65
+ </Text>
66
+ {riskScore !== null && <PermissionRiskScore riskScore={riskScore} />}
67
+ </Box>
68
+ )
69
+ }
@@ -0,0 +1,44 @@
1
+ import { useEffect } from 'react'
2
+ import { logUnaryEvent, CompletionType } from '../../utils/unaryLogging'
3
+ import { ToolUseConfirm } from '../../components/permissions/PermissionRequest'
4
+ import { env } from '../../utils/env'
5
+ import { logEvent } from '../../services/statsig'
6
+
7
+ type UnaryEventType = {
8
+ completion_type: CompletionType
9
+ language_name: string | Promise<string>
10
+ }
11
+
12
+ /**
13
+ * Logs permission request events using Statsig and unary logging.
14
+ * Handles both the Statsig event and the unary event logging.
15
+ * Can handle either a string or Promise<string> for language_name.
16
+ */
17
+ export function usePermissionRequestLogging(
18
+ toolUseConfirm: ToolUseConfirm,
19
+ unaryEvent: UnaryEventType,
20
+ ): void {
21
+ useEffect(() => {
22
+ // Log Statsig event
23
+ logEvent('tengu_tool_use_show_permission_request', {
24
+ messageID: toolUseConfirm.assistantMessage.message.id,
25
+ toolName: toolUseConfirm.tool.name,
26
+ })
27
+
28
+ // Handle string or Promise language name
29
+ const languagePromise = Promise.resolve(unaryEvent.language_name)
30
+
31
+ // Log unary event once language is resolved
32
+ languagePromise.then(language => {
33
+ logUnaryEvent({
34
+ completion_type: unaryEvent.completion_type,
35
+ event: 'response',
36
+ metadata: {
37
+ language_name: language,
38
+ message_id: toolUseConfirm.assistantMessage.message.id,
39
+ platform: env.platform,
40
+ },
41
+ })
42
+ })
43
+ }, [toolUseConfirm, unaryEvent])
44
+ }