@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,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
+ }
@@ -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
+ }
@@ -0,0 +1,59 @@
1
+ import { type Option } from '@inkjs/ui'
2
+ import chalk from 'chalk'
3
+ import {
4
+ type ToolUseConfirm,
5
+ toolUseConfirmGetPrefix,
6
+ } from './PermissionRequest.js'
7
+ import { isUnsafeCompoundCommand } from '../../utils/commands'
8
+ import { getCwd } from '../../utils/state'
9
+ import { getTheme } from '../../utils/theme'
10
+ import { type OptionSubtree } from '../CustomSelect/select'
11
+
12
+ /**
13
+ * Generates options for the tool use confirmation dialog
14
+ */
15
+ export function toolUseOptions({
16
+ toolUseConfirm,
17
+ command,
18
+ }: {
19
+ toolUseConfirm: ToolUseConfirm
20
+ command: string
21
+ }): (Option | OptionSubtree)[] {
22
+ // Hide "don't ask again" options if the command is an unsafe compound command, or a potential command injection
23
+ const showDontAskAgainOption =
24
+ !isUnsafeCompoundCommand(command) &&
25
+ toolUseConfirm.commandPrefix &&
26
+ !toolUseConfirm.commandPrefix.commandInjectionDetected
27
+ const prefix = toolUseConfirmGetPrefix(toolUseConfirm)
28
+ const showDontAskAgainPrefixOption = showDontAskAgainOption && prefix !== null
29
+
30
+ let dontShowAgainOptions: (Option | OptionSubtree)[] = []
31
+ if (showDontAskAgainPrefixOption) {
32
+ // Prefix option takes precedence over full command option
33
+ dontShowAgainOptions = [
34
+ {
35
+ label: `Yes, and don't ask again for ${chalk.bold(prefix)} commands in ${chalk.bold(getCwd())}`,
36
+ value: 'yes-dont-ask-again-prefix',
37
+ },
38
+ ]
39
+ } else if (showDontAskAgainOption) {
40
+ dontShowAgainOptions = [
41
+ {
42
+ label: `Yes, and don't ask again for ${chalk.bold(command)} commands in ${chalk.bold(getCwd())}`,
43
+ value: 'yes-dont-ask-again-full',
44
+ },
45
+ ]
46
+ }
47
+
48
+ return [
49
+ {
50
+ label: 'Yes',
51
+ value: 'yes',
52
+ },
53
+ ...dontShowAgainOptions,
54
+ {
55
+ label: `No, and provide instructions (${chalk.bold.hex(getTheme().warning)('esc')})`,
56
+ value: 'no',
57
+ },
58
+ ]
59
+ }
@@ -0,0 +1,23 @@
1
+ import { env } from '../../utils/env'
2
+ import { CompletionType, logUnaryEvent } from '../../utils/unaryLogging'
3
+ import { ToolUseConfirm } from './PermissionRequest'
4
+
5
+ export function logUnaryPermissionEvent(
6
+ completion_type: CompletionType,
7
+ {
8
+ assistantMessage: {
9
+ message: { id: message_id },
10
+ },
11
+ }: ToolUseConfirm,
12
+ event: 'accept' | 'reject',
13
+ ): void {
14
+ logUnaryEvent({
15
+ completion_type,
16
+ event,
17
+ metadata: {
18
+ language_name: 'none',
19
+ message_id,
20
+ platform: env.platform,
21
+ },
22
+ })
23
+ }
@@ -0,0 +1,5 @@
1
+ export const GATE_TOKEN_EFFICIENT_TOOLS = 'tengu-token-efficient-tools'
2
+ export const BETA_HEADER_TOKEN_EFFICIENT_TOOLS =
3
+ 'token-efficient-tools-2024-12-11'
4
+ export const GATE_USE_EXTERNAL_UPDATER = 'tengu-use-external-updater'
5
+ export const CLAUDE_CODE_20250219_BETA_HEADER = 'claude-code-20250219'