@shareai-lab/kode 1.1.14 → 1.1.16-dev.1

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 (289) hide show
  1. package/cli.js +77 -82
  2. package/dist/entrypoints/cli.js +59 -38
  3. package/dist/entrypoints/cli.js.map +3 -3
  4. package/dist/index.js +5 -26
  5. package/dist/package.json +4 -1
  6. package/package.json +11 -104
  7. package/dist/test/testAdapters.js +0 -88
  8. package/dist/test/testAdapters.js.map +0 -1
  9. package/src/ProjectOnboarding.tsx +0 -198
  10. package/src/Tool.ts +0 -83
  11. package/src/commands/agents.tsx +0 -3416
  12. package/src/commands/approvedTools.ts +0 -53
  13. package/src/commands/bug.tsx +0 -20
  14. package/src/commands/clear.ts +0 -43
  15. package/src/commands/compact.ts +0 -120
  16. package/src/commands/config.tsx +0 -19
  17. package/src/commands/cost.ts +0 -18
  18. package/src/commands/ctx_viz.ts +0 -209
  19. package/src/commands/doctor.ts +0 -24
  20. package/src/commands/help.tsx +0 -19
  21. package/src/commands/init.ts +0 -37
  22. package/src/commands/listen.ts +0 -42
  23. package/src/commands/login.tsx +0 -51
  24. package/src/commands/logout.tsx +0 -40
  25. package/src/commands/mcp.ts +0 -41
  26. package/src/commands/model.tsx +0 -40
  27. package/src/commands/modelstatus.tsx +0 -20
  28. package/src/commands/onboarding.tsx +0 -34
  29. package/src/commands/pr_comments.ts +0 -59
  30. package/src/commands/refreshCommands.ts +0 -54
  31. package/src/commands/release-notes.ts +0 -34
  32. package/src/commands/resume.tsx +0 -31
  33. package/src/commands/review.ts +0 -49
  34. package/src/commands/terminalSetup.ts +0 -221
  35. package/src/commands.ts +0 -139
  36. package/src/components/ApproveApiKey.tsx +0 -93
  37. package/src/components/AsciiLogo.tsx +0 -13
  38. package/src/components/AutoUpdater.tsx +0 -148
  39. package/src/components/Bug.tsx +0 -367
  40. package/src/components/Config.tsx +0 -293
  41. package/src/components/ConsoleOAuthFlow.tsx +0 -327
  42. package/src/components/Cost.tsx +0 -23
  43. package/src/components/CostThresholdDialog.tsx +0 -46
  44. package/src/components/CustomSelect/option-map.ts +0 -42
  45. package/src/components/CustomSelect/select-option.tsx +0 -78
  46. package/src/components/CustomSelect/select.tsx +0 -152
  47. package/src/components/CustomSelect/theme.ts +0 -45
  48. package/src/components/CustomSelect/use-select-state.ts +0 -414
  49. package/src/components/CustomSelect/use-select.ts +0 -35
  50. package/src/components/FallbackToolUseRejectedMessage.tsx +0 -15
  51. package/src/components/FileEditToolUpdatedMessage.tsx +0 -66
  52. package/src/components/Help.tsx +0 -215
  53. package/src/components/HighlightedCode.tsx +0 -33
  54. package/src/components/InvalidConfigDialog.tsx +0 -113
  55. package/src/components/Link.tsx +0 -32
  56. package/src/components/LogSelector.tsx +0 -86
  57. package/src/components/Logo.tsx +0 -170
  58. package/src/components/MCPServerApprovalDialog.tsx +0 -100
  59. package/src/components/MCPServerDialogCopy.tsx +0 -25
  60. package/src/components/MCPServerMultiselectDialog.tsx +0 -109
  61. package/src/components/Message.tsx +0 -221
  62. package/src/components/MessageResponse.tsx +0 -15
  63. package/src/components/MessageSelector.tsx +0 -211
  64. package/src/components/ModeIndicator.tsx +0 -88
  65. package/src/components/ModelConfig.tsx +0 -301
  66. package/src/components/ModelListManager.tsx +0 -227
  67. package/src/components/ModelSelector.tsx +0 -3387
  68. package/src/components/ModelStatusDisplay.tsx +0 -230
  69. package/src/components/Onboarding.tsx +0 -274
  70. package/src/components/PressEnterToContinue.tsx +0 -11
  71. package/src/components/PromptInput.tsx +0 -760
  72. package/src/components/SentryErrorBoundary.ts +0 -39
  73. package/src/components/Spinner.tsx +0 -129
  74. package/src/components/StickerRequestForm.tsx +0 -16
  75. package/src/components/StructuredDiff.tsx +0 -191
  76. package/src/components/TextInput.tsx +0 -259
  77. package/src/components/TodoItem.tsx +0 -47
  78. package/src/components/TokenWarning.tsx +0 -31
  79. package/src/components/ToolUseLoader.tsx +0 -40
  80. package/src/components/TrustDialog.tsx +0 -106
  81. package/src/components/binary-feedback/BinaryFeedback.tsx +0 -63
  82. package/src/components/binary-feedback/BinaryFeedbackOption.tsx +0 -111
  83. package/src/components/binary-feedback/BinaryFeedbackView.tsx +0 -172
  84. package/src/components/binary-feedback/utils.ts +0 -220
  85. package/src/components/messages/AssistantBashOutputMessage.tsx +0 -22
  86. package/src/components/messages/AssistantLocalCommandOutputMessage.tsx +0 -49
  87. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +0 -19
  88. package/src/components/messages/AssistantTextMessage.tsx +0 -144
  89. package/src/components/messages/AssistantThinkingMessage.tsx +0 -40
  90. package/src/components/messages/AssistantToolUseMessage.tsx +0 -132
  91. package/src/components/messages/TaskProgressMessage.tsx +0 -32
  92. package/src/components/messages/TaskToolMessage.tsx +0 -58
  93. package/src/components/messages/UserBashInputMessage.tsx +0 -28
  94. package/src/components/messages/UserCommandMessage.tsx +0 -30
  95. package/src/components/messages/UserKodingInputMessage.tsx +0 -28
  96. package/src/components/messages/UserPromptMessage.tsx +0 -35
  97. package/src/components/messages/UserTextMessage.tsx +0 -39
  98. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +0 -12
  99. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +0 -36
  100. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +0 -31
  101. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +0 -57
  102. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +0 -35
  103. package/src/components/messages/UserToolResultMessage/utils.tsx +0 -56
  104. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +0 -121
  105. package/src/components/permissions/FallbackPermissionRequest.tsx +0 -153
  106. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +0 -182
  107. package/src/components/permissions/FileEditPermissionRequest/FileEditToolDiff.tsx +0 -77
  108. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +0 -164
  109. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +0 -83
  110. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +0 -240
  111. package/src/components/permissions/PermissionRequest.tsx +0 -101
  112. package/src/components/permissions/PermissionRequestTitle.tsx +0 -69
  113. package/src/components/permissions/hooks.ts +0 -44
  114. package/src/components/permissions/toolUseOptions.ts +0 -59
  115. package/src/components/permissions/utils.ts +0 -23
  116. package/src/constants/betas.ts +0 -5
  117. package/src/constants/claude-asterisk-ascii-art.tsx +0 -238
  118. package/src/constants/figures.ts +0 -4
  119. package/src/constants/keys.ts +0 -3
  120. package/src/constants/macros.ts +0 -11
  121. package/src/constants/modelCapabilities.ts +0 -179
  122. package/src/constants/models.ts +0 -1025
  123. package/src/constants/oauth.ts +0 -18
  124. package/src/constants/product.ts +0 -17
  125. package/src/constants/prompts.ts +0 -168
  126. package/src/constants/releaseNotes.ts +0 -7
  127. package/src/context/PermissionContext.tsx +0 -149
  128. package/src/context.ts +0 -278
  129. package/src/cost-tracker.ts +0 -84
  130. package/src/entrypoints/cli.tsx +0 -1561
  131. package/src/entrypoints/mcp.ts +0 -175
  132. package/src/history.ts +0 -25
  133. package/src/hooks/useApiKeyVerification.ts +0 -59
  134. package/src/hooks/useArrowKeyHistory.ts +0 -55
  135. package/src/hooks/useCanUseTool.ts +0 -138
  136. package/src/hooks/useCancelRequest.ts +0 -39
  137. package/src/hooks/useDoublePress.ts +0 -41
  138. package/src/hooks/useExitOnCtrlCD.ts +0 -31
  139. package/src/hooks/useInterval.ts +0 -25
  140. package/src/hooks/useLogMessages.ts +0 -16
  141. package/src/hooks/useLogStartupTime.ts +0 -12
  142. package/src/hooks/useNotifyAfterTimeout.ts +0 -65
  143. package/src/hooks/usePermissionRequestLogging.ts +0 -44
  144. package/src/hooks/useTerminalSize.ts +0 -49
  145. package/src/hooks/useTextInput.ts +0 -317
  146. package/src/hooks/useUnifiedCompletion.ts +0 -1405
  147. package/src/index.ts +0 -34
  148. package/src/messages.ts +0 -38
  149. package/src/permissions.ts +0 -268
  150. package/src/query.ts +0 -720
  151. package/src/screens/ConfigureNpmPrefix.tsx +0 -197
  152. package/src/screens/Doctor.tsx +0 -219
  153. package/src/screens/LogList.tsx +0 -68
  154. package/src/screens/REPL.tsx +0 -813
  155. package/src/screens/ResumeConversation.tsx +0 -68
  156. package/src/services/adapters/base.ts +0 -38
  157. package/src/services/adapters/chatCompletions.ts +0 -90
  158. package/src/services/adapters/responsesAPI.ts +0 -170
  159. package/src/services/browserMocks.ts +0 -66
  160. package/src/services/claude.ts +0 -2197
  161. package/src/services/customCommands.ts +0 -704
  162. package/src/services/fileFreshness.ts +0 -377
  163. package/src/services/gpt5ConnectionTest.ts +0 -340
  164. package/src/services/mcpClient.ts +0 -564
  165. package/src/services/mcpServerApproval.tsx +0 -50
  166. package/src/services/mentionProcessor.ts +0 -273
  167. package/src/services/modelAdapterFactory.ts +0 -69
  168. package/src/services/notifier.ts +0 -40
  169. package/src/services/oauth.ts +0 -357
  170. package/src/services/openai.ts +0 -1359
  171. package/src/services/responseStateManager.ts +0 -90
  172. package/src/services/sentry.ts +0 -3
  173. package/src/services/statsig.ts +0 -172
  174. package/src/services/statsigStorage.ts +0 -86
  175. package/src/services/systemReminder.ts +0 -507
  176. package/src/services/vcr.ts +0 -161
  177. package/src/test/testAdapters.ts +0 -96
  178. package/src/tools/ArchitectTool/ArchitectTool.tsx +0 -135
  179. package/src/tools/ArchitectTool/prompt.ts +0 -15
  180. package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +0 -576
  181. package/src/tools/BashTool/BashTool.tsx +0 -243
  182. package/src/tools/BashTool/BashToolResultMessage.tsx +0 -38
  183. package/src/tools/BashTool/OutputLine.tsx +0 -49
  184. package/src/tools/BashTool/prompt.ts +0 -174
  185. package/src/tools/BashTool/utils.ts +0 -56
  186. package/src/tools/FileEditTool/FileEditTool.tsx +0 -319
  187. package/src/tools/FileEditTool/prompt.ts +0 -51
  188. package/src/tools/FileEditTool/utils.ts +0 -58
  189. package/src/tools/FileReadTool/FileReadTool.tsx +0 -404
  190. package/src/tools/FileReadTool/prompt.ts +0 -7
  191. package/src/tools/FileWriteTool/FileWriteTool.tsx +0 -301
  192. package/src/tools/FileWriteTool/prompt.ts +0 -10
  193. package/src/tools/GlobTool/GlobTool.tsx +0 -119
  194. package/src/tools/GlobTool/prompt.ts +0 -8
  195. package/src/tools/GrepTool/GrepTool.tsx +0 -147
  196. package/src/tools/GrepTool/prompt.ts +0 -11
  197. package/src/tools/MCPTool/MCPTool.tsx +0 -107
  198. package/src/tools/MCPTool/prompt.ts +0 -3
  199. package/src/tools/MemoryReadTool/MemoryReadTool.tsx +0 -127
  200. package/src/tools/MemoryReadTool/prompt.ts +0 -3
  201. package/src/tools/MemoryWriteTool/MemoryWriteTool.tsx +0 -89
  202. package/src/tools/MemoryWriteTool/prompt.ts +0 -3
  203. package/src/tools/MultiEditTool/MultiEditTool.tsx +0 -388
  204. package/src/tools/MultiEditTool/prompt.ts +0 -45
  205. package/src/tools/NotebookEditTool/NotebookEditTool.tsx +0 -298
  206. package/src/tools/NotebookEditTool/prompt.ts +0 -3
  207. package/src/tools/NotebookReadTool/NotebookReadTool.tsx +0 -258
  208. package/src/tools/NotebookReadTool/prompt.ts +0 -3
  209. package/src/tools/StickerRequestTool/StickerRequestTool.tsx +0 -107
  210. package/src/tools/StickerRequestTool/prompt.ts +0 -19
  211. package/src/tools/TaskTool/TaskTool.tsx +0 -438
  212. package/src/tools/TaskTool/constants.ts +0 -1
  213. package/src/tools/TaskTool/prompt.ts +0 -92
  214. package/src/tools/ThinkTool/ThinkTool.tsx +0 -54
  215. package/src/tools/ThinkTool/prompt.ts +0 -12
  216. package/src/tools/TodoWriteTool/TodoWriteTool.tsx +0 -313
  217. package/src/tools/TodoWriteTool/prompt.ts +0 -63
  218. package/src/tools/URLFetcherTool/URLFetcherTool.tsx +0 -178
  219. package/src/tools/URLFetcherTool/cache.ts +0 -55
  220. package/src/tools/URLFetcherTool/htmlToMarkdown.ts +0 -55
  221. package/src/tools/URLFetcherTool/prompt.ts +0 -17
  222. package/src/tools/WebSearchTool/WebSearchTool.tsx +0 -103
  223. package/src/tools/WebSearchTool/prompt.ts +0 -13
  224. package/src/tools/WebSearchTool/searchProviders.ts +0 -66
  225. package/src/tools/lsTool/lsTool.tsx +0 -272
  226. package/src/tools/lsTool/prompt.ts +0 -2
  227. package/src/tools.ts +0 -67
  228. package/src/types/PermissionMode.ts +0 -120
  229. package/src/types/RequestContext.ts +0 -72
  230. package/src/types/common.d.ts +0 -2
  231. package/src/types/conversation.ts +0 -51
  232. package/src/types/logs.ts +0 -58
  233. package/src/types/modelCapabilities.ts +0 -64
  234. package/src/types/notebook.ts +0 -87
  235. package/src/utils/Cursor.ts +0 -436
  236. package/src/utils/PersistentShell.ts +0 -552
  237. package/src/utils/advancedFuzzyMatcher.ts +0 -290
  238. package/src/utils/agentLoader.ts +0 -278
  239. package/src/utils/agentStorage.ts +0 -97
  240. package/src/utils/array.ts +0 -3
  241. package/src/utils/ask.tsx +0 -99
  242. package/src/utils/auth.ts +0 -13
  243. package/src/utils/autoCompactCore.ts +0 -223
  244. package/src/utils/autoUpdater.ts +0 -458
  245. package/src/utils/betas.ts +0 -20
  246. package/src/utils/browser.ts +0 -14
  247. package/src/utils/cleanup.ts +0 -72
  248. package/src/utils/commands.ts +0 -261
  249. package/src/utils/commonUnixCommands.ts +0 -161
  250. package/src/utils/config.ts +0 -945
  251. package/src/utils/conversationRecovery.ts +0 -55
  252. package/src/utils/debugLogger.ts +0 -1235
  253. package/src/utils/diff.ts +0 -42
  254. package/src/utils/env.ts +0 -57
  255. package/src/utils/errors.ts +0 -21
  256. package/src/utils/exampleCommands.ts +0 -109
  257. package/src/utils/execFileNoThrow.ts +0 -51
  258. package/src/utils/expertChatStorage.ts +0 -136
  259. package/src/utils/file.ts +0 -405
  260. package/src/utils/fileRecoveryCore.ts +0 -71
  261. package/src/utils/format.tsx +0 -44
  262. package/src/utils/fuzzyMatcher.ts +0 -328
  263. package/src/utils/generators.ts +0 -62
  264. package/src/utils/git.ts +0 -92
  265. package/src/utils/globalLogger.ts +0 -77
  266. package/src/utils/http.ts +0 -10
  267. package/src/utils/imagePaste.ts +0 -38
  268. package/src/utils/json.ts +0 -13
  269. package/src/utils/log.ts +0 -382
  270. package/src/utils/markdown.ts +0 -213
  271. package/src/utils/messageContextManager.ts +0 -294
  272. package/src/utils/messages.tsx +0 -945
  273. package/src/utils/model.ts +0 -914
  274. package/src/utils/permissions/filesystem.ts +0 -127
  275. package/src/utils/responseState.ts +0 -23
  276. package/src/utils/ripgrep.ts +0 -167
  277. package/src/utils/secureFile.ts +0 -564
  278. package/src/utils/sessionState.ts +0 -49
  279. package/src/utils/state.ts +0 -25
  280. package/src/utils/style.ts +0 -29
  281. package/src/utils/terminal.ts +0 -50
  282. package/src/utils/theme.ts +0 -127
  283. package/src/utils/thinking.ts +0 -144
  284. package/src/utils/todoStorage.ts +0 -431
  285. package/src/utils/tokens.ts +0 -43
  286. package/src/utils/toolExecutionController.ts +0 -163
  287. package/src/utils/unaryLogging.ts +0 -26
  288. package/src/utils/user.ts +0 -37
  289. package/src/utils/validate.ts +0 -165
@@ -1,319 +0,0 @@
1
- import { Hunk } from 'diff'
2
- import { existsSync, mkdirSync, readFileSync, statSync } from 'fs'
3
- import { Box, Text } from 'ink'
4
- import { dirname, isAbsolute, relative, resolve, sep } from 'path'
5
- import * as React from 'react'
6
- import { z } from 'zod'
7
- import { FileEditToolUpdatedMessage } from '../../components/FileEditToolUpdatedMessage'
8
- import { StructuredDiff } from '../../components/StructuredDiff'
9
- import { FallbackToolUseRejectedMessage } from '../../components/FallbackToolUseRejectedMessage'
10
- import { logEvent } from '../../services/statsig'
11
- import { Tool, ValidationResult } from '../../Tool'
12
- import { intersperse } from '../../utils/array'
13
- import {
14
- addLineNumbers,
15
- detectFileEncoding,
16
- detectLineEndings,
17
- findSimilarFile,
18
- writeTextContent,
19
- } from '../../utils/file.js'
20
- import { logError } from '../../utils/log'
21
- import { getCwd } from '../../utils/state'
22
- import { getTheme } from '../../utils/theme'
23
- import { emitReminderEvent } from '../../services/systemReminder'
24
- import { recordFileEdit } from '../../services/fileFreshness'
25
- import { NotebookEditTool } from '../NotebookEditTool/NotebookEditTool'
26
- import { DESCRIPTION } from './prompt'
27
- import { applyEdit } from './utils'
28
- import { hasWritePermission } from '../../utils/permissions/filesystem'
29
- import { PROJECT_FILE } from '../../constants/product'
30
-
31
- const inputSchema = z.strictObject({
32
- file_path: z.string().describe('The absolute path to the file to modify'),
33
- old_string: z.string().describe('The text to replace'),
34
- new_string: z.string().describe('The text to replace it with'),
35
- })
36
-
37
- export type In = typeof inputSchema
38
-
39
- // Number of lines of context to include before/after the change in our result message
40
- const N_LINES_SNIPPET = 4
41
-
42
- export const FileEditTool = {
43
- name: 'Edit',
44
- async description() {
45
- return 'A tool for editing files'
46
- },
47
- async prompt() {
48
- return DESCRIPTION
49
- },
50
- inputSchema,
51
- userFacingName() {
52
- return 'Edit'
53
- },
54
- async isEnabled() {
55
- return true
56
- },
57
- isReadOnly() {
58
- return false
59
- },
60
- isConcurrencySafe() {
61
- return false // FileEdit modifies files, not safe for concurrent execution
62
- },
63
- needsPermissions({ file_path }) {
64
- return !hasWritePermission(file_path)
65
- },
66
- renderToolUseMessage(input, { verbose }) {
67
- return `file_path: ${verbose ? input.file_path : relative(getCwd(), input.file_path)}`
68
- },
69
- renderToolResultMessage({ filePath, structuredPatch }) {
70
- const verbose = false // Set default value for verbose
71
- return (
72
- <FileEditToolUpdatedMessage
73
- filePath={filePath}
74
- structuredPatch={structuredPatch}
75
- verbose={verbose}
76
- />
77
- )
78
- },
79
- renderToolUseRejectedMessage(
80
- { file_path, old_string, new_string }: any = {},
81
- { columns, verbose }: any = {},
82
- ) {
83
- try {
84
- if (!file_path) {
85
- return <FallbackToolUseRejectedMessage />
86
- }
87
- const { patch } = applyEdit(file_path, old_string, new_string)
88
- return (
89
- <Box flexDirection="column">
90
- <Text>
91
- {' '}⎿{' '}
92
- <Text color={getTheme().error}>
93
- User rejected {old_string === '' ? 'write' : 'update'} to{' '}
94
- </Text>
95
- <Text bold>
96
- {verbose ? file_path : relative(getCwd(), file_path)}
97
- </Text>
98
- </Text>
99
- {intersperse(
100
- patch.map(patch => (
101
- <Box flexDirection="column" paddingLeft={5} key={patch.newStart}>
102
- <StructuredDiff patch={patch} dim={true} width={columns - 12} />
103
- </Box>
104
- )),
105
- i => (
106
- <Box paddingLeft={5} key={`ellipsis-${i}`}>
107
- <Text color={getTheme().secondaryText}>...</Text>
108
- </Box>
109
- ),
110
- )}
111
- </Box>
112
- )
113
- } catch (e) {
114
- // Handle the case where while we were showing the diff, the user manually made the change.
115
- // TODO: Find a way to show the diff in this case
116
- logError(e)
117
- return (
118
- <Box flexDirection="column">
119
- <Text>{' '}⎿ (No changes)</Text>
120
- </Box>
121
- )
122
- }
123
- },
124
- async validateInput(
125
- { file_path, old_string, new_string },
126
- { readFileTimestamps },
127
- ) {
128
- if (old_string === new_string) {
129
- return {
130
- result: false,
131
- message:
132
- 'No changes to make: old_string and new_string are exactly the same.',
133
- meta: {
134
- old_string,
135
- },
136
- } as ValidationResult
137
- }
138
-
139
- const fullFilePath = isAbsolute(file_path)
140
- ? file_path
141
- : resolve(getCwd(), file_path)
142
-
143
- if (existsSync(fullFilePath) && old_string === '') {
144
- return {
145
- result: false,
146
- message: 'Cannot create new file - file already exists.',
147
- }
148
- }
149
-
150
- if (!existsSync(fullFilePath) && old_string === '') {
151
- return {
152
- result: true,
153
- }
154
- }
155
-
156
- if (!existsSync(fullFilePath)) {
157
- // Try to find a similar file with a different extension
158
- const similarFilename = findSimilarFile(fullFilePath)
159
- let message = 'File does not exist.'
160
-
161
- // If we found a similar file, suggest it to the assistant
162
- if (similarFilename) {
163
- message += ` Did you mean ${similarFilename}?`
164
- }
165
-
166
- return {
167
- result: false,
168
- message,
169
- }
170
- }
171
-
172
- if (fullFilePath.endsWith('.ipynb')) {
173
- return {
174
- result: false,
175
- message: `File is a Jupyter Notebook. Use the ${NotebookEditTool.name} to edit this file.`,
176
- }
177
- }
178
-
179
- const readTimestamp = readFileTimestamps[fullFilePath]
180
- if (!readTimestamp) {
181
- return {
182
- result: false,
183
- message:
184
- 'File has not been read yet. Read it first before writing to it.',
185
- meta: {
186
- isFilePathAbsolute: String(isAbsolute(file_path)),
187
- },
188
- }
189
- }
190
-
191
- // Check if file exists and get its last modified time
192
- const stats = statSync(fullFilePath)
193
- const lastWriteTime = stats.mtimeMs
194
- if (lastWriteTime > readTimestamp) {
195
- return {
196
- result: false,
197
- message:
198
- 'File has been modified since read, either by the user or by a linter. Read it again before attempting to write it.',
199
- }
200
- }
201
-
202
- const enc = detectFileEncoding(fullFilePath)
203
- const file = readFileSync(fullFilePath, enc)
204
- if (!file.includes(old_string)) {
205
- return {
206
- result: false,
207
- message: `String to replace not found in file.`,
208
- meta: {
209
- isFilePathAbsolute: String(isAbsolute(file_path)),
210
- },
211
- }
212
- }
213
-
214
- const matches = file.split(old_string).length - 1
215
- if (matches > 1) {
216
- return {
217
- result: false,
218
- message: `Found ${matches} matches of the string to replace. For safety, this tool only supports replacing exactly one occurrence at a time. Add more lines of context to your edit and try again.`,
219
- meta: {
220
- isFilePathAbsolute: String(isAbsolute(file_path)),
221
- },
222
- }
223
- }
224
-
225
- return { result: true }
226
- },
227
- async *call({ file_path, old_string, new_string }, { readFileTimestamps }) {
228
- const { patch, updatedFile } = applyEdit(file_path, old_string, new_string)
229
-
230
- const fullFilePath = isAbsolute(file_path)
231
- ? file_path
232
- : resolve(getCwd(), file_path)
233
- const dir = dirname(fullFilePath)
234
- mkdirSync(dir, { recursive: true })
235
- const enc = existsSync(fullFilePath)
236
- ? detectFileEncoding(fullFilePath)
237
- : 'utf8'
238
- const endings = existsSync(fullFilePath)
239
- ? detectLineEndings(fullFilePath)
240
- : 'LF'
241
- const originalFile = existsSync(fullFilePath)
242
- ? readFileSync(fullFilePath, enc)
243
- : ''
244
- writeTextContent(fullFilePath, updatedFile, enc, endings)
245
-
246
- // Record Agent edit operation for file freshness tracking
247
- recordFileEdit(fullFilePath, updatedFile)
248
-
249
- // Update read timestamp, to invalidate stale writes
250
- readFileTimestamps[fullFilePath] = statSync(fullFilePath).mtimeMs
251
-
252
- // Log when editing CLAUDE.md
253
- if (fullFilePath.endsWith(`${sep}${PROJECT_FILE}`)) {
254
- logEvent('tengu_write_claudemd', {})
255
- }
256
-
257
- // Emit file edited event for system reminders
258
- emitReminderEvent('file:edited', {
259
- filePath: fullFilePath,
260
- oldString: old_string,
261
- newString: new_string,
262
- timestamp: Date.now(),
263
- operation:
264
- old_string === '' ? 'create' : new_string === '' ? 'delete' : 'update',
265
- })
266
-
267
- const data = {
268
- filePath: file_path,
269
- oldString: old_string,
270
- newString: new_string,
271
- originalFile,
272
- structuredPatch: patch,
273
- }
274
- yield {
275
- type: 'result',
276
- data,
277
- resultForAssistant: this.renderResultForAssistant(data),
278
- }
279
- },
280
- renderResultForAssistant({ filePath, originalFile, oldString, newString }) {
281
- const { snippet, startLine } = getSnippet(
282
- originalFile || '',
283
- oldString,
284
- newString,
285
- )
286
- return `The file ${filePath} has been updated. Here's the result of running \`cat -n\` on a snippet of the edited file:
287
- ${addLineNumbers({
288
- content: snippet,
289
- startLine,
290
- })}`
291
- },
292
- } satisfies Tool<
293
- typeof inputSchema,
294
- {
295
- filePath: string
296
- oldString: string
297
- newString: string
298
- originalFile: string
299
- structuredPatch: Hunk[]
300
- }
301
- >
302
-
303
- export function getSnippet(
304
- initialText: string,
305
- oldStr: string,
306
- newStr: string,
307
- ): { snippet: string; startLine: number } {
308
- const before = initialText.split(oldStr)[0] ?? ''
309
- const replacementLine = before.split(/\r?\n/).length - 1
310
- const newFileLines = initialText.replace(oldStr, newStr).split(/\r?\n/)
311
- // Calculate the start and end line numbers for the snippet
312
- const startLine = Math.max(0, replacementLine - N_LINES_SNIPPET)
313
- const endLine =
314
- replacementLine + N_LINES_SNIPPET + newStr.split(/\r?\n/).length
315
- // Get snippet
316
- const snippetLines = newFileLines.slice(startLine, endLine + 1)
317
- const snippet = snippetLines.join('\n')
318
- return { snippet, startLine: startLine + 1 }
319
- }
@@ -1,51 +0,0 @@
1
- import { NotebookEditTool } from '../NotebookEditTool/NotebookEditTool'
2
-
3
- export const DESCRIPTION = `This is a tool for editing files. For moving or renaming files, you should generally use the Bash tool with the 'mv' command instead. For larger edits, use the Write tool to overwrite files. For Jupyter notebooks (.ipynb files), use the ${NotebookEditTool.name} instead.
4
-
5
- Before using this tool:
6
-
7
- 1. Use the View tool to understand the file's contents and context
8
-
9
- 2. Verify the directory path is correct (only applicable when creating new files):
10
- - Use the LS tool to verify the parent directory exists and is the correct location
11
-
12
- To make a file edit, provide the following:
13
- 1. file_path: The absolute path to the file to modify (must be absolute, not relative)
14
- 2. old_string: The text to replace (must be unique within the file, and must match the file contents exactly, including all whitespace and indentation)
15
- 3. new_string: The edited text to replace the old_string
16
-
17
- The tool will replace ONE occurrence of old_string with new_string in the specified file.
18
-
19
- CRITICAL REQUIREMENTS FOR USING THIS TOOL:
20
-
21
- 1. UNIQUENESS: The old_string MUST uniquely identify the specific instance you want to change. This means:
22
- - Include AT LEAST 3-5 lines of context BEFORE the change point
23
- - Include AT LEAST 3-5 lines of context AFTER the change point
24
- - Include all whitespace, indentation, and surrounding code exactly as it appears in the file
25
-
26
- 2. SINGLE INSTANCE: This tool can only change ONE instance at a time. If you need to change multiple instances:
27
- - Make separate calls to this tool for each instance
28
- - Each call must uniquely identify its specific instance using extensive context
29
-
30
- 3. VERIFICATION: Before using this tool:
31
- - Check how many instances of the target text exist in the file
32
- - If multiple instances exist, gather enough context to uniquely identify each one
33
- - Plan separate tool calls for each instance
34
-
35
- WARNING: If you do not follow these requirements:
36
- - The tool will fail if old_string matches multiple locations
37
- - The tool will fail if old_string doesn't match exactly (including whitespace)
38
- - You may change the wrong instance if you don't include enough context
39
-
40
- When making edits:
41
- - Ensure the edit results in idiomatic, correct code
42
- - Do not leave the code in a broken state
43
- - Always use absolute file paths (starting with /)
44
-
45
- If you want to create a new file, use:
46
- - A new file path, including dir name if needed
47
- - An empty old_string
48
- - The new file's contents as new_string
49
-
50
- Remember: when making multiple file edits in a row to the same file, you should prefer to send all edits in a single message with multiple calls to this tool, rather than multiple messages with a single call each.
51
- `
@@ -1,58 +0,0 @@
1
- import { isAbsolute, resolve } from 'path'
2
- import { getCwd } from '../../utils/state'
3
- import { readFileSync } from 'fs'
4
- import { detectFileEncoding } from '../../utils/file'
5
- import { type Hunk } from 'diff'
6
- import { getPatch } from '../../utils/diff'
7
-
8
- /**
9
- * Applies an edit to a file and returns the patch and updated file.
10
- * Does not write the file to disk.
11
- */
12
- export function applyEdit(
13
- file_path: string,
14
- old_string: string,
15
- new_string: string,
16
- ): { patch: Hunk[]; updatedFile: string } {
17
- const fullFilePath = isAbsolute(file_path)
18
- ? file_path
19
- : resolve(getCwd(), file_path)
20
-
21
- let originalFile
22
- let updatedFile
23
- if (old_string === '') {
24
- // Create new file
25
- originalFile = ''
26
- updatedFile = new_string
27
- } else {
28
- // Edit existing file
29
- const enc = detectFileEncoding(fullFilePath)
30
- originalFile = readFileSync(fullFilePath, enc)
31
- if (new_string === '') {
32
- if (
33
- !old_string.endsWith('\n') &&
34
- originalFile.includes(old_string + '\n')
35
- ) {
36
- updatedFile = originalFile.replace(old_string + '\n', () => new_string)
37
- } else {
38
- updatedFile = originalFile.replace(old_string, () => new_string)
39
- }
40
- } else {
41
- updatedFile = originalFile.replace(old_string, () => new_string)
42
- }
43
- if (updatedFile === originalFile) {
44
- throw new Error(
45
- 'Original and edited file match exactly. Failed to apply edit.',
46
- )
47
- }
48
- }
49
-
50
- const patch = getPatch({
51
- filePath: file_path,
52
- fileContents: originalFile,
53
- oldStr: originalFile,
54
- newStr: updatedFile,
55
- })
56
-
57
- return { patch, updatedFile }
58
- }