@shareai-lab/kode 1.1.13 → 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.
- package/dist/entrypoints/cli.js +59 -38
- package/dist/entrypoints/cli.js.map +3 -3
- package/dist/index.js +5 -26
- package/dist/package.json +4 -1
- package/package.json +11 -104
- package/dist/test/testAdapters.js +0 -88
- package/dist/test/testAdapters.js.map +0 -1
- package/src/ProjectOnboarding.tsx +0 -198
- package/src/Tool.ts +0 -83
- package/src/commands/agents.tsx +0 -3416
- package/src/commands/approvedTools.ts +0 -53
- package/src/commands/bug.tsx +0 -20
- package/src/commands/clear.ts +0 -43
- package/src/commands/compact.ts +0 -120
- package/src/commands/config.tsx +0 -19
- package/src/commands/cost.ts +0 -18
- package/src/commands/ctx_viz.ts +0 -209
- package/src/commands/doctor.ts +0 -24
- package/src/commands/help.tsx +0 -19
- package/src/commands/init.ts +0 -37
- package/src/commands/listen.ts +0 -42
- package/src/commands/login.tsx +0 -51
- package/src/commands/logout.tsx +0 -40
- package/src/commands/mcp.ts +0 -41
- package/src/commands/model.tsx +0 -40
- package/src/commands/modelstatus.tsx +0 -20
- package/src/commands/onboarding.tsx +0 -34
- package/src/commands/pr_comments.ts +0 -59
- package/src/commands/refreshCommands.ts +0 -54
- package/src/commands/release-notes.ts +0 -34
- package/src/commands/resume.tsx +0 -31
- package/src/commands/review.ts +0 -49
- package/src/commands/terminalSetup.ts +0 -221
- package/src/commands.ts +0 -139
- package/src/components/ApproveApiKey.tsx +0 -93
- package/src/components/AsciiLogo.tsx +0 -13
- package/src/components/AutoUpdater.tsx +0 -148
- package/src/components/Bug.tsx +0 -367
- package/src/components/Config.tsx +0 -293
- package/src/components/ConsoleOAuthFlow.tsx +0 -327
- package/src/components/Cost.tsx +0 -23
- package/src/components/CostThresholdDialog.tsx +0 -46
- package/src/components/CustomSelect/option-map.ts +0 -42
- package/src/components/CustomSelect/select-option.tsx +0 -78
- package/src/components/CustomSelect/select.tsx +0 -152
- package/src/components/CustomSelect/theme.ts +0 -45
- package/src/components/CustomSelect/use-select-state.ts +0 -414
- package/src/components/CustomSelect/use-select.ts +0 -35
- package/src/components/FallbackToolUseRejectedMessage.tsx +0 -15
- package/src/components/FileEditToolUpdatedMessage.tsx +0 -66
- package/src/components/Help.tsx +0 -215
- package/src/components/HighlightedCode.tsx +0 -33
- package/src/components/InvalidConfigDialog.tsx +0 -113
- package/src/components/Link.tsx +0 -32
- package/src/components/LogSelector.tsx +0 -86
- package/src/components/Logo.tsx +0 -170
- package/src/components/MCPServerApprovalDialog.tsx +0 -100
- package/src/components/MCPServerDialogCopy.tsx +0 -25
- package/src/components/MCPServerMultiselectDialog.tsx +0 -109
- package/src/components/Message.tsx +0 -221
- package/src/components/MessageResponse.tsx +0 -15
- package/src/components/MessageSelector.tsx +0 -211
- package/src/components/ModeIndicator.tsx +0 -88
- package/src/components/ModelConfig.tsx +0 -301
- package/src/components/ModelListManager.tsx +0 -227
- package/src/components/ModelSelector.tsx +0 -3387
- package/src/components/ModelStatusDisplay.tsx +0 -230
- package/src/components/Onboarding.tsx +0 -274
- package/src/components/PressEnterToContinue.tsx +0 -11
- package/src/components/PromptInput.tsx +0 -760
- package/src/components/SentryErrorBoundary.ts +0 -39
- package/src/components/Spinner.tsx +0 -129
- package/src/components/StickerRequestForm.tsx +0 -16
- package/src/components/StructuredDiff.tsx +0 -191
- package/src/components/TextInput.tsx +0 -259
- package/src/components/TodoItem.tsx +0 -47
- package/src/components/TokenWarning.tsx +0 -31
- package/src/components/ToolUseLoader.tsx +0 -40
- package/src/components/TrustDialog.tsx +0 -106
- package/src/components/binary-feedback/BinaryFeedback.tsx +0 -63
- package/src/components/binary-feedback/BinaryFeedbackOption.tsx +0 -111
- package/src/components/binary-feedback/BinaryFeedbackView.tsx +0 -172
- package/src/components/binary-feedback/utils.ts +0 -220
- package/src/components/messages/AssistantBashOutputMessage.tsx +0 -22
- package/src/components/messages/AssistantLocalCommandOutputMessage.tsx +0 -49
- package/src/components/messages/AssistantRedactedThinkingMessage.tsx +0 -19
- package/src/components/messages/AssistantTextMessage.tsx +0 -144
- package/src/components/messages/AssistantThinkingMessage.tsx +0 -40
- package/src/components/messages/AssistantToolUseMessage.tsx +0 -132
- package/src/components/messages/TaskProgressMessage.tsx +0 -32
- package/src/components/messages/TaskToolMessage.tsx +0 -58
- package/src/components/messages/UserBashInputMessage.tsx +0 -28
- package/src/components/messages/UserCommandMessage.tsx +0 -30
- package/src/components/messages/UserKodingInputMessage.tsx +0 -28
- package/src/components/messages/UserPromptMessage.tsx +0 -35
- package/src/components/messages/UserTextMessage.tsx +0 -39
- package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +0 -12
- package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +0 -36
- package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +0 -31
- package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +0 -57
- package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +0 -35
- package/src/components/messages/UserToolResultMessage/utils.tsx +0 -56
- package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +0 -121
- package/src/components/permissions/FallbackPermissionRequest.tsx +0 -153
- package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +0 -182
- package/src/components/permissions/FileEditPermissionRequest/FileEditToolDiff.tsx +0 -77
- package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +0 -164
- package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +0 -83
- package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +0 -240
- package/src/components/permissions/PermissionRequest.tsx +0 -101
- package/src/components/permissions/PermissionRequestTitle.tsx +0 -69
- package/src/components/permissions/hooks.ts +0 -44
- package/src/components/permissions/toolUseOptions.ts +0 -59
- package/src/components/permissions/utils.ts +0 -23
- package/src/constants/betas.ts +0 -5
- package/src/constants/claude-asterisk-ascii-art.tsx +0 -238
- package/src/constants/figures.ts +0 -4
- package/src/constants/keys.ts +0 -3
- package/src/constants/macros.ts +0 -11
- package/src/constants/modelCapabilities.ts +0 -179
- package/src/constants/models.ts +0 -1025
- package/src/constants/oauth.ts +0 -18
- package/src/constants/product.ts +0 -17
- package/src/constants/prompts.ts +0 -168
- package/src/constants/releaseNotes.ts +0 -7
- package/src/context/PermissionContext.tsx +0 -149
- package/src/context.ts +0 -278
- package/src/cost-tracker.ts +0 -84
- package/src/entrypoints/cli.tsx +0 -1561
- package/src/entrypoints/mcp.ts +0 -175
- package/src/history.ts +0 -25
- package/src/hooks/useApiKeyVerification.ts +0 -59
- package/src/hooks/useArrowKeyHistory.ts +0 -55
- package/src/hooks/useCanUseTool.ts +0 -138
- package/src/hooks/useCancelRequest.ts +0 -39
- package/src/hooks/useDoublePress.ts +0 -41
- package/src/hooks/useExitOnCtrlCD.ts +0 -31
- package/src/hooks/useInterval.ts +0 -25
- package/src/hooks/useLogMessages.ts +0 -16
- package/src/hooks/useLogStartupTime.ts +0 -12
- package/src/hooks/useNotifyAfterTimeout.ts +0 -65
- package/src/hooks/usePermissionRequestLogging.ts +0 -44
- package/src/hooks/useTerminalSize.ts +0 -49
- package/src/hooks/useTextInput.ts +0 -317
- package/src/hooks/useUnifiedCompletion.ts +0 -1405
- package/src/index.ts +0 -34
- package/src/messages.ts +0 -38
- package/src/permissions.ts +0 -268
- package/src/query.ts +0 -720
- package/src/screens/ConfigureNpmPrefix.tsx +0 -197
- package/src/screens/Doctor.tsx +0 -219
- package/src/screens/LogList.tsx +0 -68
- package/src/screens/REPL.tsx +0 -813
- package/src/screens/ResumeConversation.tsx +0 -68
- package/src/services/adapters/base.ts +0 -38
- package/src/services/adapters/chatCompletions.ts +0 -90
- package/src/services/adapters/responsesAPI.ts +0 -170
- package/src/services/browserMocks.ts +0 -66
- package/src/services/claude.ts +0 -2197
- package/src/services/customCommands.ts +0 -704
- package/src/services/fileFreshness.ts +0 -377
- package/src/services/gpt5ConnectionTest.ts +0 -340
- package/src/services/mcpClient.ts +0 -564
- package/src/services/mcpServerApproval.tsx +0 -50
- package/src/services/mentionProcessor.ts +0 -273
- package/src/services/modelAdapterFactory.ts +0 -69
- package/src/services/notifier.ts +0 -40
- package/src/services/oauth.ts +0 -357
- package/src/services/openai.ts +0 -1359
- package/src/services/responseStateManager.ts +0 -90
- package/src/services/sentry.ts +0 -3
- package/src/services/statsig.ts +0 -172
- package/src/services/statsigStorage.ts +0 -86
- package/src/services/systemReminder.ts +0 -507
- package/src/services/vcr.ts +0 -161
- package/src/test/testAdapters.ts +0 -96
- package/src/tools/ArchitectTool/ArchitectTool.tsx +0 -135
- package/src/tools/ArchitectTool/prompt.ts +0 -15
- package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +0 -576
- package/src/tools/BashTool/BashTool.tsx +0 -243
- package/src/tools/BashTool/BashToolResultMessage.tsx +0 -38
- package/src/tools/BashTool/OutputLine.tsx +0 -49
- package/src/tools/BashTool/prompt.ts +0 -174
- package/src/tools/BashTool/utils.ts +0 -56
- package/src/tools/FileEditTool/FileEditTool.tsx +0 -319
- package/src/tools/FileEditTool/prompt.ts +0 -51
- package/src/tools/FileEditTool/utils.ts +0 -58
- package/src/tools/FileReadTool/FileReadTool.tsx +0 -404
- package/src/tools/FileReadTool/prompt.ts +0 -7
- package/src/tools/FileWriteTool/FileWriteTool.tsx +0 -301
- package/src/tools/FileWriteTool/prompt.ts +0 -10
- package/src/tools/GlobTool/GlobTool.tsx +0 -119
- package/src/tools/GlobTool/prompt.ts +0 -8
- package/src/tools/GrepTool/GrepTool.tsx +0 -147
- package/src/tools/GrepTool/prompt.ts +0 -11
- package/src/tools/MCPTool/MCPTool.tsx +0 -107
- package/src/tools/MCPTool/prompt.ts +0 -3
- package/src/tools/MemoryReadTool/MemoryReadTool.tsx +0 -127
- package/src/tools/MemoryReadTool/prompt.ts +0 -3
- package/src/tools/MemoryWriteTool/MemoryWriteTool.tsx +0 -89
- package/src/tools/MemoryWriteTool/prompt.ts +0 -3
- package/src/tools/MultiEditTool/MultiEditTool.tsx +0 -388
- package/src/tools/MultiEditTool/prompt.ts +0 -45
- package/src/tools/NotebookEditTool/NotebookEditTool.tsx +0 -298
- package/src/tools/NotebookEditTool/prompt.ts +0 -3
- package/src/tools/NotebookReadTool/NotebookReadTool.tsx +0 -258
- package/src/tools/NotebookReadTool/prompt.ts +0 -3
- package/src/tools/StickerRequestTool/StickerRequestTool.tsx +0 -107
- package/src/tools/StickerRequestTool/prompt.ts +0 -19
- package/src/tools/TaskTool/TaskTool.tsx +0 -438
- package/src/tools/TaskTool/constants.ts +0 -1
- package/src/tools/TaskTool/prompt.ts +0 -92
- package/src/tools/ThinkTool/ThinkTool.tsx +0 -54
- package/src/tools/ThinkTool/prompt.ts +0 -12
- package/src/tools/TodoWriteTool/TodoWriteTool.tsx +0 -313
- package/src/tools/TodoWriteTool/prompt.ts +0 -63
- package/src/tools/URLFetcherTool/URLFetcherTool.tsx +0 -178
- package/src/tools/URLFetcherTool/cache.ts +0 -55
- package/src/tools/URLFetcherTool/htmlToMarkdown.ts +0 -55
- package/src/tools/URLFetcherTool/prompt.ts +0 -17
- package/src/tools/WebSearchTool/WebSearchTool.tsx +0 -103
- package/src/tools/WebSearchTool/prompt.ts +0 -13
- package/src/tools/WebSearchTool/searchProviders.ts +0 -66
- package/src/tools/lsTool/lsTool.tsx +0 -272
- package/src/tools/lsTool/prompt.ts +0 -2
- package/src/tools.ts +0 -67
- package/src/types/PermissionMode.ts +0 -120
- package/src/types/RequestContext.ts +0 -72
- package/src/types/common.d.ts +0 -2
- package/src/types/conversation.ts +0 -51
- package/src/types/logs.ts +0 -58
- package/src/types/modelCapabilities.ts +0 -64
- package/src/types/notebook.ts +0 -87
- package/src/utils/Cursor.ts +0 -436
- package/src/utils/PersistentShell.ts +0 -552
- package/src/utils/advancedFuzzyMatcher.ts +0 -290
- package/src/utils/agentLoader.ts +0 -278
- package/src/utils/agentStorage.ts +0 -97
- package/src/utils/array.ts +0 -3
- package/src/utils/ask.tsx +0 -99
- package/src/utils/auth.ts +0 -13
- package/src/utils/autoCompactCore.ts +0 -223
- package/src/utils/autoUpdater.ts +0 -458
- package/src/utils/betas.ts +0 -20
- package/src/utils/browser.ts +0 -14
- package/src/utils/cleanup.ts +0 -72
- package/src/utils/commands.ts +0 -261
- package/src/utils/commonUnixCommands.ts +0 -161
- package/src/utils/config.ts +0 -945
- package/src/utils/conversationRecovery.ts +0 -55
- package/src/utils/debugLogger.ts +0 -1235
- package/src/utils/diff.ts +0 -42
- package/src/utils/env.ts +0 -57
- package/src/utils/errors.ts +0 -21
- package/src/utils/exampleCommands.ts +0 -109
- package/src/utils/execFileNoThrow.ts +0 -51
- package/src/utils/expertChatStorage.ts +0 -136
- package/src/utils/file.ts +0 -405
- package/src/utils/fileRecoveryCore.ts +0 -71
- package/src/utils/format.tsx +0 -44
- package/src/utils/fuzzyMatcher.ts +0 -328
- package/src/utils/generators.ts +0 -62
- package/src/utils/git.ts +0 -92
- package/src/utils/globalLogger.ts +0 -77
- package/src/utils/http.ts +0 -10
- package/src/utils/imagePaste.ts +0 -38
- package/src/utils/json.ts +0 -13
- package/src/utils/log.ts +0 -382
- package/src/utils/markdown.ts +0 -213
- package/src/utils/messageContextManager.ts +0 -294
- package/src/utils/messages.tsx +0 -945
- package/src/utils/model.ts +0 -914
- package/src/utils/permissions/filesystem.ts +0 -127
- package/src/utils/responseState.ts +0 -23
- package/src/utils/ripgrep.ts +0 -167
- package/src/utils/secureFile.ts +0 -564
- package/src/utils/sessionState.ts +0 -49
- package/src/utils/state.ts +0 -25
- package/src/utils/style.ts +0 -29
- package/src/utils/terminal.ts +0 -50
- package/src/utils/theme.ts +0 -127
- package/src/utils/thinking.ts +0 -144
- package/src/utils/todoStorage.ts +0 -431
- package/src/utils/tokens.ts +0 -43
- package/src/utils/toolExecutionController.ts +0 -163
- package/src/utils/unaryLogging.ts +0 -26
- package/src/utils/user.ts +0 -37
- 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
|
-
⎿
|
|
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
|
-
`
|