@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,298 +0,0 @@
1
- import { existsSync, readFileSync } from 'fs'
2
- import { Box, Text } from 'ink'
3
- import { extname, isAbsolute, relative, resolve } from 'path'
4
- import * as React from 'react'
5
- import { z } from 'zod'
6
- import { FallbackToolUseRejectedMessage } from '../../components/FallbackToolUseRejectedMessage'
7
- import { HighlightedCode } from '../../components/HighlightedCode'
8
- import type { Tool } from '../../Tool'
9
- import { NotebookCellType, NotebookContent } from '../../types/notebook'
10
- import {
11
- detectFileEncoding,
12
- detectLineEndings,
13
- writeTextContent,
14
- } from '../../utils/file.js'
15
- import { safeParseJSON } from '../../utils/json'
16
- import { getCwd } from '../../utils/state'
17
- import { DESCRIPTION, PROMPT } from './prompt'
18
- import { hasWritePermission } from '../../utils/permissions/filesystem'
19
- import { emitReminderEvent } from '../../services/systemReminder'
20
- import { recordFileEdit } from '../../services/fileFreshness'
21
-
22
- const inputSchema = z.strictObject({
23
- notebook_path: z
24
- .string()
25
- .describe(
26
- 'The absolute path to the Jupyter notebook file to edit (must be absolute, not relative)',
27
- ),
28
- cell_number: z.number().describe('The index of the cell to edit (0-based)'),
29
- new_source: z.string().describe('The new source for the cell'),
30
- cell_type: z
31
- .enum(['code', 'markdown'])
32
- .optional()
33
- .describe(
34
- 'The type of the cell (code or markdown). If not specified, it defaults to the current cell type. If using edit_mode=insert, this is required.',
35
- ),
36
- edit_mode: z
37
- .string()
38
- .optional()
39
- .describe(
40
- 'The type of edit to make (replace, insert, delete). Defaults to replace.',
41
- ),
42
- })
43
-
44
- export const NotebookEditTool = {
45
- name: 'NotebookEditCell',
46
- async description() {
47
- return DESCRIPTION
48
- },
49
- async prompt() {
50
- return PROMPT
51
- },
52
- inputSchema,
53
- userFacingName() {
54
- return 'Edit Notebook'
55
- },
56
- async isEnabled() {
57
- return true
58
- },
59
- isReadOnly() {
60
- return false
61
- },
62
- isConcurrencySafe() {
63
- return false // NotebookEditTool modifies state/files, not safe for concurrent execution
64
- },
65
- needsPermissions({ notebook_path }) {
66
- return !hasWritePermission(notebook_path)
67
- },
68
- renderResultForAssistant({ cell_number, edit_mode, new_source, error }) {
69
- if (error) {
70
- return error
71
- }
72
- switch (edit_mode) {
73
- case 'replace':
74
- return `Updated cell ${cell_number} with ${new_source}`
75
- case 'insert':
76
- return `Inserted cell ${cell_number} with ${new_source}`
77
- case 'delete':
78
- return `Deleted cell ${cell_number}`
79
- }
80
- },
81
- renderToolUseMessage(input, { verbose }) {
82
- return `notebook_path: ${verbose ? input.notebook_path : relative(getCwd(), input.notebook_path)}, cell: ${input.cell_number}, content: ${input.new_source.slice(0, 30)}…, cell_type: ${input.cell_type}, edit_mode: ${input.edit_mode ?? 'replace'}`
83
- },
84
- renderToolUseRejectedMessage() {
85
- return <FallbackToolUseRejectedMessage />
86
- },
87
- renderToolResultMessage({ cell_number, new_source, language, error }) {
88
- if (error) {
89
- return (
90
- <Box flexDirection="column">
91
- <Text color="red">{error}</Text>
92
- </Box>
93
- )
94
- }
95
-
96
- return (
97
- <Box flexDirection="column">
98
- <Text>Updated cell {cell_number}:</Text>
99
- <Box marginLeft={2}>
100
- <HighlightedCode code={new_source} language={language} />
101
- </Box>
102
- </Box>
103
- )
104
- },
105
- async validateInput({
106
- notebook_path,
107
- cell_number,
108
- cell_type,
109
- edit_mode = 'replace',
110
- }) {
111
- const fullPath = isAbsolute(notebook_path)
112
- ? notebook_path
113
- : resolve(getCwd(), notebook_path)
114
-
115
- if (!existsSync(fullPath)) {
116
- return {
117
- result: false,
118
- message: 'Notebook file does not exist.',
119
- }
120
- }
121
-
122
- if (extname(fullPath) !== '.ipynb') {
123
- return {
124
- result: false,
125
- message:
126
- 'File must be a Jupyter notebook (.ipynb file). For editing other file types, use the FileEdit tool.',
127
- }
128
- }
129
-
130
- if (cell_number < 0) {
131
- return {
132
- result: false,
133
- message: 'Cell number must be non-negative.',
134
- }
135
- }
136
-
137
- if (
138
- edit_mode !== 'replace' &&
139
- edit_mode !== 'insert' &&
140
- edit_mode !== 'delete'
141
- ) {
142
- return {
143
- result: false,
144
- message: 'Edit mode must be replace, insert, or delete.',
145
- }
146
- }
147
-
148
- if (edit_mode === 'insert' && !cell_type) {
149
- return {
150
- result: false,
151
- message: 'Cell type is required when using edit_mode=insert.',
152
- }
153
- }
154
-
155
- const enc = detectFileEncoding(fullPath)
156
- const content = readFileSync(fullPath, enc)
157
- const notebook = safeParseJSON(content) as NotebookContent | null
158
- if (!notebook) {
159
- return {
160
- result: false,
161
- message: 'Notebook is not valid JSON.',
162
- }
163
- }
164
-
165
- if (edit_mode === 'insert' && cell_number > notebook.cells.length) {
166
- return {
167
- result: false,
168
- message: `Cell number is out of bounds. For insert mode, the maximum value is ${notebook.cells.length} (to append at the end).`,
169
- }
170
- } else if (
171
- (edit_mode === 'replace' || edit_mode === 'delete') &&
172
- (cell_number >= notebook.cells.length || !notebook.cells[cell_number])
173
- ) {
174
- return {
175
- result: false,
176
- message: `Cell number is out of bounds. Notebook has ${notebook.cells.length} cells.`,
177
- }
178
- }
179
-
180
- return { result: true }
181
- },
182
- async *call({
183
- notebook_path,
184
- cell_number,
185
- new_source,
186
- cell_type,
187
- edit_mode,
188
- }) {
189
- const fullPath = isAbsolute(notebook_path)
190
- ? notebook_path
191
- : resolve(getCwd(), notebook_path)
192
-
193
- try {
194
- const enc = detectFileEncoding(fullPath)
195
- const content = readFileSync(fullPath, enc)
196
- const notebook = JSON.parse(content) as NotebookContent
197
- const language = notebook.metadata.language_info?.name ?? 'python'
198
-
199
- if (edit_mode === 'delete') {
200
- // Delete the specified cell
201
- notebook.cells.splice(cell_number, 1)
202
- } else if (edit_mode === 'insert') {
203
- // Insert the new cell
204
- const new_cell = {
205
- cell_type: cell_type!, // validateInput ensures cell_type is not undefined
206
- source: new_source,
207
- metadata: {},
208
- }
209
- notebook.cells.splice(
210
- cell_number,
211
- 0,
212
- cell_type == 'markdown' ? new_cell : { ...new_cell, outputs: [] },
213
- )
214
- } else {
215
- // Find the specified cell
216
- const targetCell = notebook.cells[cell_number]! // validateInput ensures cell_number is in bounds
217
- targetCell.source = new_source
218
- // Reset execution count and clear outputs since cell was modified
219
- targetCell.execution_count = undefined
220
- targetCell.outputs = []
221
- if (cell_type && cell_type !== targetCell.cell_type) {
222
- targetCell.cell_type = cell_type
223
- }
224
- }
225
- // Write back to file
226
- const endings = detectLineEndings(fullPath)
227
- const updatedNotebook = JSON.stringify(notebook, null, 1)
228
- writeTextContent(fullPath, updatedNotebook, enc, endings!)
229
-
230
- // Record Agent edit operation for file freshness tracking
231
- recordFileEdit(fullPath, updatedNotebook)
232
-
233
- // Emit file edited event for system reminders
234
- emitReminderEvent('file:edited', {
235
- filePath: fullPath,
236
- cellNumber: cell_number,
237
- newSource: new_source,
238
- cellType: cell_type,
239
- editMode: edit_mode || 'replace',
240
- timestamp: Date.now(),
241
- operation: 'notebook_edit',
242
- })
243
- const data = {
244
- cell_number,
245
- new_source,
246
- cell_type: cell_type ?? 'code',
247
- language,
248
- edit_mode: edit_mode ?? 'replace',
249
- error: '',
250
- }
251
- yield {
252
- type: 'result',
253
- data,
254
- resultForAssistant: this.renderResultForAssistant(data),
255
- }
256
- } catch (error) {
257
- if (error instanceof Error) {
258
- const data = {
259
- cell_number,
260
- new_source,
261
- cell_type: cell_type ?? 'code',
262
- language: 'python',
263
- edit_mode: 'replace',
264
- error: error.message,
265
- }
266
- yield {
267
- type: 'result',
268
- data,
269
- resultForAssistant: this.renderResultForAssistant(data),
270
- }
271
- return
272
- }
273
- const data = {
274
- cell_number,
275
- new_source,
276
- cell_type: cell_type ?? 'code',
277
- language: 'python',
278
- edit_mode: 'replace',
279
- error: 'Unknown error occurred while editing notebook',
280
- }
281
- yield {
282
- type: 'result',
283
- data,
284
- resultForAssistant: this.renderResultForAssistant(data),
285
- }
286
- }
287
- },
288
- } satisfies Tool<
289
- typeof inputSchema,
290
- {
291
- cell_number: number
292
- new_source: string
293
- cell_type: NotebookCellType
294
- language: string
295
- edit_mode: string
296
- error?: string
297
- }
298
- >
@@ -1,3 +0,0 @@
1
- export const DESCRIPTION =
2
- 'Replace the contents of a specific cell in a Jupyter notebook.'
3
- export const PROMPT = `Completely replaces the contents of a specific cell in a Jupyter notebook (.ipynb file) with new source. Jupyter notebooks are interactive documents that combine code, text, and visualizations, commonly used for data analysis and scientific computing. The notebook_path parameter must be an absolute path, not a relative path. The cell_number is 0-indexed. Use edit_mode=insert to add a new cell at the index specified by cell_number. Use edit_mode=delete to delete the cell at the index specified by cell_number.`
@@ -1,258 +0,0 @@
1
- import type {
2
- ImageBlockParam,
3
- TextBlockParam,
4
- } from '@anthropic-ai/sdk/resources/index.mjs'
5
-
6
- import { existsSync, readFileSync } from 'fs'
7
- import { Text } from 'ink'
8
- import { extname, isAbsolute, relative, resolve } from 'path'
9
- import * as React from 'react'
10
- import { z } from 'zod'
11
- import { FallbackToolUseRejectedMessage } from '../../components/FallbackToolUseRejectedMessage'
12
- import { Tool } from '../../Tool'
13
- import {
14
- NotebookCellSource,
15
- NotebookContent,
16
- NotebookCell,
17
- NotebookOutputImage,
18
- NotebookCellSourceOutput,
19
- NotebookCellOutput,
20
- NotebookCellType,
21
- } from '../../types/notebook'
22
- import { formatOutput } from '../BashTool/utils'
23
- import { getCwd } from '../../utils/state'
24
- import { findSimilarFile } from '../../utils/file'
25
- import { DESCRIPTION, PROMPT } from './prompt'
26
- import { hasReadPermission } from '../../utils/permissions/filesystem'
27
-
28
- const inputSchema = z.strictObject({
29
- notebook_path: z
30
- .string()
31
- .describe(
32
- 'The absolute path to the Jupyter notebook file to read (must be absolute, not relative)',
33
- ),
34
- })
35
-
36
- type In = typeof inputSchema
37
- type Out = NotebookCellSource[]
38
-
39
-
40
- export const NotebookReadTool = {
41
- name: 'ReadNotebook',
42
- async description() {
43
- return DESCRIPTION
44
- },
45
- async prompt() {
46
- return PROMPT
47
- },
48
- isReadOnly() {
49
- return true
50
- },
51
- isConcurrencySafe() {
52
- return true // NotebookReadTool is read-only, safe for concurrent execution
53
- },
54
- inputSchema,
55
- userFacingName() {
56
- return 'Read Notebook'
57
- },
58
- async isEnabled() {
59
- return true
60
- },
61
- needsPermissions({ notebook_path }) {
62
- return !hasReadPermission(notebook_path)
63
- },
64
- async validateInput({ notebook_path }) {
65
- const fullFilePath = isAbsolute(notebook_path)
66
- ? notebook_path
67
- : resolve(getCwd(), notebook_path)
68
-
69
- if (!existsSync(fullFilePath)) {
70
- // Try to find a similar file with a different extension
71
- const similarFilename = findSimilarFile(fullFilePath)
72
- let message = 'File does not exist.'
73
-
74
- // If we found a similar file, suggest it to the assistant
75
- if (similarFilename) {
76
- message += ` Did you mean ${similarFilename}?`
77
- }
78
-
79
- return {
80
- result: false,
81
- message,
82
- }
83
- }
84
-
85
- if (extname(fullFilePath) !== '.ipynb') {
86
- return {
87
- result: false,
88
- message: 'File must be a Jupyter notebook (.ipynb file).',
89
- }
90
- }
91
-
92
- return { result: true }
93
- },
94
- renderToolUseMessage(input, { verbose }) {
95
- return `notebook_path: ${verbose ? input.notebook_path : relative(getCwd(), input.notebook_path)}`
96
- },
97
- renderToolUseRejectedMessage() {
98
- return <FallbackToolUseRejectedMessage />
99
- },
100
-
101
- renderToolResultMessage(content) {
102
- if (!content) {
103
- return <Text>No cells found in notebook</Text>
104
- }
105
- if (content.length < 1 || !content[0]) {
106
- return <Text>No cells found in notebook</Text>
107
- }
108
- return <Text>Read {content.length} cells</Text>
109
- },
110
- async *call({ notebook_path }) {
111
- const fullPath = isAbsolute(notebook_path)
112
- ? notebook_path
113
- : resolve(getCwd(), notebook_path)
114
-
115
- const content = readFileSync(fullPath, 'utf-8')
116
- const notebook = JSON.parse(content) as NotebookContent
117
- const language = notebook.metadata.language_info?.name ?? 'python'
118
- const cells = notebook.cells.map((cell, index) =>
119
- processCell(cell, index, language),
120
- )
121
-
122
- yield {
123
- type: 'result',
124
- resultForAssistant: this.renderResultForAssistant(cells),
125
- data: cells,
126
- }
127
- },
128
- renderResultForAssistant(data: NotebookCellSource[]) {
129
- // Convert the complex structure to a string representation for the assistant
130
- return data.map((cell, index) => {
131
- let content = `Cell ${index + 1} (${cell.cellType}):\n${cell.source}`
132
- if (cell.outputs && cell.outputs.length > 0) {
133
- const outputText = cell.outputs.map(output => output.text).filter(Boolean).join('\n')
134
- if (outputText) {
135
- content += `\nOutput:\n${outputText}`
136
- }
137
- }
138
- return content
139
- }).join('\n\n')
140
- },
141
- } satisfies Tool<In, Out>
142
-
143
- function processOutputText(text: string | string[] | undefined): string {
144
- if (!text) return ''
145
- const rawText = Array.isArray(text) ? text.join('') : text
146
- const { truncatedContent } = formatOutput(rawText)
147
- return truncatedContent
148
- }
149
-
150
- function extractImage(
151
- data: Record<string, unknown>,
152
- ): NotebookOutputImage | undefined {
153
- if (typeof data['image/png'] === 'string') {
154
- return {
155
- image_data: data['image/png'] as string,
156
- media_type: 'image/png',
157
- }
158
- }
159
- if (typeof data['image/jpeg'] === 'string') {
160
- return {
161
- image_data: data['image/jpeg'] as string,
162
- media_type: 'image/jpeg',
163
- }
164
- }
165
- return undefined
166
- }
167
-
168
- function processOutput(output: NotebookCellOutput) {
169
- switch (output.output_type) {
170
- case 'stream':
171
- return {
172
- output_type: output.output_type,
173
- text: processOutputText(output.text),
174
- }
175
- case 'execute_result':
176
- case 'display_data':
177
- return {
178
- output_type: output.output_type,
179
- text: processOutputText(output.data?.['text/plain'] as string | string[] | undefined),
180
- image: output.data && extractImage(output.data),
181
- }
182
- case 'error':
183
- return {
184
- output_type: output.output_type,
185
- text: processOutputText(
186
- `${output.ename}: ${output.evalue}\n${output.traceback.join('\n')}`,
187
- ),
188
- }
189
- }
190
- }
191
-
192
- function processCell(
193
- cell: NotebookCell,
194
- index: number,
195
- language: string,
196
- ): NotebookCellSource {
197
- const cellData: NotebookCellSource = {
198
- cell: index,
199
- cellType: cell.cell_type,
200
- source: Array.isArray(cell.source) ? cell.source.join('') : cell.source,
201
- language,
202
- execution_count: cell.execution_count,
203
- }
204
-
205
- if (cell.outputs?.length) {
206
- cellData.outputs = cell.outputs.map(processOutput)
207
- }
208
-
209
- return cellData
210
- }
211
-
212
- function cellContentToToolResult(cell: NotebookCellSource): TextBlockParam {
213
- const metadata = []
214
- if (cell.cellType !== 'code') {
215
- metadata.push(`<cell_type>${cell.cellType}</cell_type>`)
216
- }
217
- if (cell.language !== 'python' && cell.cellType === 'code') {
218
- metadata.push(`<language>${cell.language}</language>`)
219
- }
220
- const cellContent = `<cell ${cell.cell}>${metadata.join('')}${cell.source}</cell ${cell.cell}>`
221
- return {
222
- text: cellContent,
223
- type: 'text',
224
- }
225
- }
226
-
227
- function cellOutputToToolResult(output: NotebookCellSourceOutput) {
228
- const outputs: (TextBlockParam | ImageBlockParam)[] = []
229
- if (output.text) {
230
- outputs.push({
231
- text: `\n${output.text}`,
232
- type: 'text',
233
- })
234
- }
235
- if (output.image) {
236
- outputs.push({
237
- type: 'image',
238
- source: {
239
- data: output.image.image_data,
240
- media_type: output.image.media_type,
241
- type: 'base64',
242
- },
243
- })
244
- }
245
- return outputs
246
- }
247
-
248
- function getToolResultFromCell(cell: NotebookCellSource) {
249
- const contentResult = cellContentToToolResult(cell)
250
- const outputResults = cell.outputs?.flatMap(cellOutputToToolResult)
251
- return [contentResult, ...(outputResults ?? [])]
252
- }
253
-
254
- export function isNotebookCellType(
255
- value: string | null,
256
- ): value is NotebookCellType {
257
- return value === 'code' || value === 'markdown'
258
- }
@@ -1,3 +0,0 @@
1
- export const DESCRIPTION =
2
- 'Extract and read source code from all code cells in a Jupyter notebook.'
3
- export const PROMPT = `Reads a Jupyter notebook (.ipynb file) and returns all of the cells with their outputs. Jupyter notebooks are interactive documents that combine code, text, and visualizations, commonly used for data analysis and scientific computing. The notebook_path parameter must be an absolute path, not a relative path.`
@@ -1,107 +0,0 @@
1
- import { z } from 'zod'
2
- import React from 'react'
3
- import { Text } from 'ink'
4
- import { Tool, ToolUseContext, ExtendedToolUseContext } from '../../Tool'
5
- import { DESCRIPTION, PROMPT } from './prompt'
6
- import {
7
- StickerRequestForm,
8
- FormData,
9
- } from '../../components/StickerRequestForm'
10
- import { checkGate, logEvent } from '../../services/statsig'
11
- import { getTheme } from '../../utils/theme'
12
-
13
- const stickerRequestSchema = z.object({
14
- trigger: z.string(),
15
- })
16
-
17
- export const StickerRequestTool: Tool = {
18
- name: 'StickerRequest',
19
- userFacingName: () => 'Stickers',
20
- description: async () => DESCRIPTION,
21
- inputSchema: stickerRequestSchema,
22
- isEnabled: async () => {
23
- const enabled = await checkGate('tengu_sticker_easter_egg')
24
- return enabled
25
- },
26
- isReadOnly: () => false,
27
- isConcurrencySafe: () => false, // StickerRequestTool modifies state, not safe for concurrent execution
28
- needsPermissions: () => false,
29
- prompt: async () => PROMPT,
30
-
31
- async *call(_, context: ToolUseContext) {
32
- // Log form entry event
33
- logEvent('sticker_request_form_opened', {})
34
-
35
- // Create a promise to track form completion and status
36
- let resolveForm: (success: boolean) => void
37
- const formComplete = new Promise<boolean>(resolve => {
38
- resolveForm = success => resolve(success)
39
- })
40
-
41
- // Check if setToolJSX is available (cast context if needed)
42
- const extendedContext = context as ExtendedToolUseContext
43
- if (extendedContext.setToolJSX) {
44
- extendedContext.setToolJSX({
45
- jsx: (
46
- <StickerRequestForm
47
- onSubmit={(formData: FormData) => {
48
- // Log successful completion with form data
49
- logEvent('sticker_request_form_completed', {
50
- has_address: Boolean(formData.address1).toString(),
51
- has_optional_address: Boolean(formData.address2).toString(),
52
- })
53
- resolveForm(true)
54
- if (extendedContext.setToolJSX) {
55
- extendedContext.setToolJSX(null) // Clear the JSX
56
- }
57
- }}
58
- onClose={() => {
59
- // Log form cancellation
60
- logEvent('sticker_request_form_cancelled', {})
61
- resolveForm(false)
62
- if (extendedContext.setToolJSX) {
63
- extendedContext.setToolJSX(null) // Clear the JSX
64
- }
65
- }}
66
- />
67
- ),
68
- shouldHidePromptInput: true,
69
- })
70
- } else {
71
- // Fallback if setToolJSX is not available
72
- console.log('Sticker form would be displayed here, but setToolJSX is not available')
73
- resolveForm(false)
74
- }
75
-
76
- // Wait for form completion and get status
77
- const success = await formComplete
78
-
79
- if (!success) {
80
- context.abortController.abort()
81
- throw new Error('Sticker request cancelled')
82
- }
83
-
84
- // Return success message
85
- yield {
86
- type: 'result',
87
- resultForAssistant:
88
- 'Sticker request completed! Please tell the user that they will receive stickers in the mail if they have submitted the form!',
89
- data: { success },
90
- }
91
- },
92
-
93
- renderToolUseMessage(_input) {
94
- return ''
95
- },
96
-
97
- renderToolUseRejectedMessage() {
98
- return (
99
- <Text>
100
- &nbsp;&nbsp;⎿ &nbsp;
101
- <Text color={getTheme().error}>No (Sticker request cancelled)</Text>
102
- </Text>
103
- )
104
- },
105
-
106
- renderResultForAssistant: (content: string) => content,
107
- }
@@ -1,19 +0,0 @@
1
- export const DESCRIPTION =
2
- 'Sends the user swag stickers with love from Anthropic.'
3
- export const PROMPT = `This tool should be used whenever a user expresses interest in receiving Anthropic or Claude stickers, swag, or merchandise. When triggered, it will display a shipping form for the user to enter their mailing address and contact details. Once submitted, Anthropic will process the request and ship stickers to the provided address.
4
-
5
- Common trigger phrases to watch for:
6
- - "Can I get some Anthropic stickers please?"
7
- - "How do I get Anthropic swag?"
8
- - "I'd love some Claude stickers"
9
- - "Where can I get merchandise?"
10
- - Any mention of wanting stickers or swag
11
-
12
- The tool handles the entire request process by showing an interactive form to collect shipping information.
13
-
14
- NOTE: Only use this tool if the user has explicitly asked us to send or give them stickers. If there are other requests that include the word "sticker", but do not explicitly ask us to send them stickers, do not use this tool.
15
- For example:
16
- - "How do I make custom stickers for my project?" - Do not use this tool
17
- - "I need to store sticker metadata in a database - what schema do you recommend?" - Do not use this tool
18
- - "Show me how to implement drag-and-drop sticker placement with React" - Do not use this tool
19
- `