@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,813 +0,0 @@
1
- import { ToolUseBlockParam } from '@anthropic-ai/sdk/resources/index.mjs'
2
- import { Box, Newline, Static, Text } from 'ink'
3
- import ProjectOnboarding, {
4
- markProjectOnboardingComplete,
5
- } from '../ProjectOnboarding.js'
6
- import { CostThresholdDialog } from '../components/CostThresholdDialog'
7
- import * as React from 'react'
8
- import { useEffect, useMemo, useRef, useState, useCallback } from 'react'
9
- import { Command } from '../commands'
10
- import { Logo } from '../components/Logo'
11
- import { Message } from '../components/Message'
12
- import { MessageResponse } from '../components/MessageResponse'
13
- import { MessageSelector } from '../components/MessageSelector'
14
- import {
15
- PermissionRequest,
16
- type ToolUseConfirm,
17
- } from '../components/permissions/PermissionRequest.js'
18
- import PromptInput from '../components/PromptInput'
19
- import { Spinner } from '../components/Spinner'
20
- import { getSystemPrompt } from '../constants/prompts'
21
- import { getContext } from '../context'
22
- import { getTotalCost, useCostSummary } from '../cost-tracker'
23
- import { useLogStartupTime } from '../hooks/useLogStartupTime'
24
- import { addToHistory } from '../history'
25
- import { useApiKeyVerification } from '../hooks/useApiKeyVerification'
26
- import { useCancelRequest } from '../hooks/useCancelRequest'
27
- import useCanUseTool from '../hooks/useCanUseTool'
28
- import { useLogMessages } from '../hooks/useLogMessages'
29
- import { PermissionProvider } from '../context/PermissionContext'
30
- import { ModeIndicator } from '../components/ModeIndicator'
31
- import {
32
- setMessagesGetter,
33
- setMessagesSetter,
34
- setModelConfigChangeHandler,
35
- } from '../messages'
36
- import {
37
- type AssistantMessage,
38
- type BinaryFeedbackResult,
39
- type Message as MessageType,
40
- type ProgressMessage,
41
- query,
42
- } from '../query.js'
43
- import type { WrappedClient } from '../services/mcpClient'
44
- import type { Tool } from '../Tool'
45
- import { AutoUpdaterResult } from '../utils/autoUpdater'
46
- import { getGlobalConfig, saveGlobalConfig } from '../utils/config'
47
- import { MACRO } from '../constants/macros'
48
- import { logEvent } from '../services/statsig'
49
- import { getNextAvailableLogForkNumber } from '../utils/log'
50
- import {
51
- getErroredToolUseMessages,
52
- getInProgressToolUseIDs,
53
- getLastAssistantMessageId,
54
- getToolUseID,
55
- getUnresolvedToolUseIDs,
56
- INTERRUPT_MESSAGE,
57
- isNotEmptyMessage,
58
- type NormalizedMessage,
59
- normalizeMessages,
60
- normalizeMessagesForAPI,
61
- processUserInput,
62
- reorderMessages,
63
- extractTag,
64
- createAssistantMessage,
65
- } from '../utils/messages.js'
66
- import { getModelManager, ModelManager } from '../utils/model'
67
- import { clearTerminal, updateTerminalTitle } from '../utils/terminal'
68
- import { BinaryFeedback } from '../components/binary-feedback/BinaryFeedback'
69
- import { getMaxThinkingTokens } from '../utils/thinking'
70
- import { getOriginalCwd } from '../utils/state'
71
- import { handleHashCommand } from '../commands/terminalSetup'
72
- import { debug as debugLogger } from '../utils/debugLogger'
73
-
74
- type Props = {
75
- commands: Command[]
76
- safeMode?: boolean
77
- debug?: boolean
78
- initialForkNumber?: number | undefined
79
- initialPrompt: string | undefined
80
- // A unique name for the message log file, used to identify the fork
81
- messageLogName: string
82
- shouldShowPromptInput: boolean
83
- tools: Tool[]
84
- verbose: boolean | undefined
85
- // Initial messages to populate the REPL with
86
- initialMessages?: MessageType[]
87
- // MCP clients
88
- mcpClients?: WrappedClient[]
89
- // Flag to indicate if current model is default
90
- isDefaultModel?: boolean
91
- // Update banner info passed from CLI before first render
92
- initialUpdateVersion?: string | null
93
- initialUpdateCommands?: string[] | null
94
- }
95
-
96
- export type BinaryFeedbackContext = {
97
- m1: AssistantMessage
98
- m2: AssistantMessage
99
- resolve: (result: BinaryFeedbackResult) => void
100
- }
101
-
102
- export function REPL({
103
- commands,
104
- safeMode,
105
- debug = false,
106
- initialForkNumber = 0,
107
- initialPrompt,
108
- messageLogName,
109
- shouldShowPromptInput,
110
- tools,
111
- verbose: verboseFromCLI,
112
- initialMessages,
113
- mcpClients = [],
114
- isDefaultModel = true,
115
- initialUpdateVersion,
116
- initialUpdateCommands,
117
- }: Props): React.ReactNode {
118
- // TODO: probably shouldn't re-read config from file synchronously on every keystroke
119
- const verbose = verboseFromCLI ?? getGlobalConfig().verbose
120
-
121
- // Used to force the logo to re-render and conversation log to use a new file
122
- const [forkNumber, setForkNumber] = useState(
123
- getNextAvailableLogForkNumber(messageLogName, initialForkNumber, 0),
124
- )
125
-
126
- const [
127
- forkConvoWithMessagesOnTheNextRender,
128
- setForkConvoWithMessagesOnTheNextRender,
129
- ] = useState<MessageType[] | null>(null)
130
-
131
- // 🔧 Simplified AbortController management - inspired by reference system
132
- const [abortController, setAbortController] = useState<AbortController | null>(null)
133
- const [isLoading, setIsLoading] = useState(false)
134
- const [autoUpdaterResult, setAutoUpdaterResult] =
135
- useState<AutoUpdaterResult | null>(null)
136
- const [toolJSX, setToolJSX] = useState<{
137
- jsx: React.ReactNode | null
138
- shouldHidePromptInput: boolean
139
- } | null>(null)
140
- const [toolUseConfirm, setToolUseConfirm] = useState<ToolUseConfirm | null>(
141
- null,
142
- )
143
- const [messages, setMessages] = useState<MessageType[]>(initialMessages ?? [])
144
- const [inputValue, setInputValue] = useState('')
145
- const [inputMode, setInputMode] = useState<'bash' | 'prompt' | 'koding'>(
146
- 'prompt',
147
- )
148
- const [submitCount, setSubmitCount] = useState(0)
149
- const [isMessageSelectorVisible, setIsMessageSelectorVisible] =
150
- useState(false)
151
- const [showCostDialog, setShowCostDialog] = useState(false)
152
- const [haveShownCostDialog, setHaveShownCostDialog] = useState(
153
- getGlobalConfig().hasAcknowledgedCostThreshold,
154
- )
155
-
156
- const [binaryFeedbackContext, setBinaryFeedbackContext] =
157
- useState<BinaryFeedbackContext | null>(null)
158
- // New version banner: passed in from CLI to guarantee top placement
159
- const updateAvailableVersion = initialUpdateVersion ?? null
160
- const updateCommands = initialUpdateCommands ?? null
161
- // No separate Static for banner; it renders inside Logo
162
-
163
- const getBinaryFeedbackResponse = useCallback(
164
- (
165
- m1: AssistantMessage,
166
- m2: AssistantMessage,
167
- ): Promise<BinaryFeedbackResult> => {
168
- return new Promise<BinaryFeedbackResult>(resolvePromise => {
169
- setBinaryFeedbackContext({
170
- m1,
171
- m2,
172
- resolve: resolvePromise,
173
- })
174
- })
175
- },
176
- [],
177
- )
178
-
179
- const readFileTimestamps = useRef<{
180
- [filename: string]: number
181
- }>({})
182
-
183
- const { status: apiKeyStatus, reverify } = useApiKeyVerification()
184
- function onCancel() {
185
- if (!isLoading) {
186
- return
187
- }
188
- setIsLoading(false)
189
- if (toolUseConfirm) {
190
- toolUseConfirm.onAbort()
191
- } else if (abortController && !abortController.signal.aborted) {
192
- abortController.abort()
193
- }
194
- }
195
-
196
- useCancelRequest(
197
- setToolJSX,
198
- setToolUseConfirm,
199
- setBinaryFeedbackContext,
200
- onCancel,
201
- isLoading,
202
- isMessageSelectorVisible,
203
- abortController?.signal,
204
- )
205
-
206
- useEffect(() => {
207
- if (forkConvoWithMessagesOnTheNextRender) {
208
- setForkNumber(_ => _ + 1)
209
- setForkConvoWithMessagesOnTheNextRender(null)
210
- setMessages(forkConvoWithMessagesOnTheNextRender)
211
- }
212
- }, [forkConvoWithMessagesOnTheNextRender])
213
-
214
- useEffect(() => {
215
- const totalCost = getTotalCost()
216
- if (totalCost >= 5 /* $5 */ && !showCostDialog && !haveShownCostDialog) {
217
- logEvent('tengu_cost_threshold_reached', {})
218
- setShowCostDialog(true)
219
- }
220
- }, [messages, showCostDialog, haveShownCostDialog])
221
-
222
- // Update banner is provided by CLI at startup; no async check here.
223
-
224
- const canUseTool = useCanUseTool(setToolUseConfirm)
225
-
226
- async function onInit() {
227
- reverify()
228
-
229
- if (!initialPrompt) {
230
- return
231
- }
232
-
233
- setIsLoading(true)
234
-
235
- const newAbortController = new AbortController()
236
- setAbortController(newAbortController)
237
-
238
- // 🔧 Force fresh config read to ensure model switching works
239
- const model = new ModelManager(getGlobalConfig()).getModelName('main')
240
- const newMessages = await processUserInput(
241
- initialPrompt,
242
- 'prompt',
243
- setToolJSX,
244
- {
245
- abortController: newAbortController,
246
- options: {
247
- commands,
248
- forkNumber,
249
- messageLogName,
250
- tools,
251
- verbose,
252
- maxThinkingTokens: 0,
253
- },
254
- messageId: getLastAssistantMessageId(messages),
255
- setForkConvoWithMessagesOnTheNextRender,
256
- readFileTimestamps: readFileTimestamps.current,
257
- },
258
- null,
259
- )
260
-
261
- if (newMessages.length) {
262
- for (const message of newMessages) {
263
- if (message.type === 'user') {
264
- addToHistory(initialPrompt)
265
- // TODO: setHistoryIndex
266
- }
267
- }
268
- setMessages(_ => [..._, ...newMessages])
269
-
270
- // The last message is an assistant message if the user input was a bash command,
271
- // or if the user input was an invalid slash command.
272
- const lastMessage = newMessages[newMessages.length - 1]!
273
- if (lastMessage.type === 'assistant') {
274
- setAbortController(null)
275
- setIsLoading(false)
276
- return
277
- }
278
-
279
- const [systemPrompt, context, model, maxThinkingTokens] =
280
- await Promise.all([
281
- getSystemPrompt(),
282
- getContext(),
283
- new ModelManager(getGlobalConfig()).getModelName('main'),
284
- getMaxThinkingTokens([...messages, ...newMessages]),
285
- ])
286
-
287
- for await (const message of query(
288
- [...messages, ...newMessages],
289
- systemPrompt,
290
- context,
291
- canUseTool,
292
- {
293
- options: {
294
- commands,
295
- forkNumber,
296
- messageLogName,
297
- tools,
298
- verbose,
299
- safeMode,
300
- maxThinkingTokens,
301
- },
302
- messageId: getLastAssistantMessageId([...messages, ...newMessages]),
303
- readFileTimestamps: readFileTimestamps.current,
304
- abortController: newAbortController,
305
- setToolJSX,
306
- },
307
- getBinaryFeedbackResponse,
308
- )) {
309
- setMessages(oldMessages => [...oldMessages, message])
310
- }
311
- } else {
312
- addToHistory(initialPrompt)
313
- // TODO: setHistoryIndex
314
- }
315
-
316
- setHaveShownCostDialog(
317
- getGlobalConfig().hasAcknowledgedCostThreshold || false,
318
- )
319
-
320
- // 🔧 Fix: Clean up state after onInit completion
321
- setIsLoading(false)
322
- setAbortController(null)
323
- }
324
-
325
- async function onQuery(newMessages: MessageType[], passedAbortController?: AbortController) {
326
- // Use passed AbortController or create new one
327
- const controllerToUse = passedAbortController || new AbortController()
328
- if (!passedAbortController) {
329
- setAbortController(controllerToUse)
330
- }
331
-
332
- // Check if this is a Koding request based on last message's options
333
- const isKodingRequest =
334
- newMessages.length > 0 &&
335
- newMessages[0].type === 'user' &&
336
- 'options' in newMessages[0] &&
337
- newMessages[0].options?.isKodingRequest === true
338
-
339
- setMessages(oldMessages => [...oldMessages, ...newMessages])
340
-
341
- // Mark onboarding as complete when any user message is sent to Claude
342
- markProjectOnboardingComplete()
343
-
344
- // The last message is an assistant message if the user input was a bash command,
345
- // or if the user input was an invalid slash command.
346
- const lastMessage = newMessages[newMessages.length - 1]!
347
-
348
- // Update terminal title based on user message
349
- if (
350
- lastMessage.type === 'user' &&
351
- typeof lastMessage.message.content === 'string'
352
- ) {
353
- // updateTerminalTitle(lastMessage.message.content)
354
- }
355
- if (lastMessage.type === 'assistant') {
356
- setAbortController(null)
357
- setIsLoading(false)
358
- return
359
- }
360
-
361
- const [systemPrompt, context, model, maxThinkingTokens] =
362
- await Promise.all([
363
- getSystemPrompt(),
364
- getContext(),
365
- new ModelManager(getGlobalConfig()).getModelName('main'),
366
- getMaxThinkingTokens([...messages, lastMessage]),
367
- ])
368
-
369
- let lastAssistantMessage: MessageType | null = null
370
-
371
- // query the API
372
- for await (const message of query(
373
- [...messages, lastMessage],
374
- systemPrompt,
375
- context,
376
- canUseTool,
377
- {
378
- options: {
379
- commands,
380
- forkNumber,
381
- messageLogName,
382
- tools,
383
- verbose,
384
- safeMode,
385
- maxThinkingTokens,
386
- // If this came from Koding mode, pass that along
387
- isKodingRequest: isKodingRequest || undefined,
388
- },
389
- messageId: getLastAssistantMessageId([...messages, lastMessage]),
390
- readFileTimestamps: readFileTimestamps.current,
391
- abortController: controllerToUse,
392
- setToolJSX,
393
- },
394
- getBinaryFeedbackResponse,
395
- )) {
396
- setMessages(oldMessages => [...oldMessages, message])
397
-
398
- // Keep track of the last assistant message for Koding mode
399
- if (message.type === 'assistant') {
400
- lastAssistantMessage = message
401
- }
402
- }
403
-
404
- // If this was a Koding request and we got an assistant message back,
405
- // save it to AGENTS.md (and CLAUDE.md if exists)
406
- if (
407
- isKodingRequest &&
408
- lastAssistantMessage &&
409
- lastAssistantMessage.type === 'assistant'
410
- ) {
411
- try {
412
- const content =
413
- typeof lastAssistantMessage.message.content === 'string'
414
- ? lastAssistantMessage.message.content
415
- : lastAssistantMessage.message.content
416
- .filter(block => block.type === 'text')
417
- .map(block => (block.type === 'text' ? block.text : ''))
418
- .join('\n')
419
-
420
- // Add the content to AGENTS.md (and CLAUDE.md if exists)
421
- if (content && content.trim().length > 0) {
422
- handleHashCommand(content)
423
- }
424
- } catch (error) {
425
- console.error('Error saving response to project docs:', error)
426
- }
427
- }
428
-
429
- setIsLoading(false)
430
- }
431
-
432
- // Register cost summary tracker
433
- useCostSummary()
434
-
435
- // Register messages getter and setter
436
- useEffect(() => {
437
- const getMessages = () => messages
438
- setMessagesGetter(getMessages)
439
- setMessagesSetter(setMessages)
440
- }, [messages])
441
-
442
- // Register model config change handler for UI refresh
443
- useEffect(() => {
444
- setModelConfigChangeHandler(() => {
445
- setForkNumber(prev => prev + 1)
446
- })
447
- }, [])
448
-
449
- // Record transcripts locally, for debugging and conversation recovery
450
- useLogMessages(messages, messageLogName, forkNumber)
451
-
452
- // Log startup time
453
- useLogStartupTime()
454
-
455
- // Initial load
456
- useEffect(() => {
457
- onInit()
458
- // TODO: fix this
459
- // eslint-disable-next-line react-hooks/exhaustive-deps
460
- }, [])
461
-
462
- const normalizedMessages = useMemo(
463
- () => normalizeMessages(messages).filter(isNotEmptyMessage),
464
- [messages],
465
- )
466
-
467
- const unresolvedToolUseIDs = useMemo(
468
- () => getUnresolvedToolUseIDs(normalizedMessages),
469
- [normalizedMessages],
470
- )
471
-
472
- const inProgressToolUseIDs = useMemo(
473
- () => getInProgressToolUseIDs(normalizedMessages),
474
- [normalizedMessages],
475
- )
476
-
477
- const erroredToolUseIDs = useMemo(
478
- () =>
479
- new Set(
480
- getErroredToolUseMessages(normalizedMessages).map(
481
- _ => (_.message.content[0]! as ToolUseBlockParam).id,
482
- ),
483
- ),
484
- [normalizedMessages],
485
- )
486
-
487
- const messagesJSX = useMemo(() => {
488
- return [
489
- {
490
- type: 'static',
491
- jsx: (
492
- <Box flexDirection="column" key={`logo${forkNumber}`}>
493
- <Logo
494
- mcpClients={mcpClients}
495
- isDefaultModel={isDefaultModel}
496
- updateBannerVersion={updateAvailableVersion}
497
- updateBannerCommands={updateCommands}
498
- />
499
- <ProjectOnboarding workspaceDir={getOriginalCwd()} />
500
- </Box>
501
- ),
502
- },
503
- ...reorderMessages(normalizedMessages).map(_ => {
504
- const toolUseID = getToolUseID(_)
505
- const message =
506
- _.type === 'progress' ? (
507
- _.content.message.content[0]?.type === 'text' &&
508
- // Hack: TaskTool interrupts use Progress messages, so don't
509
- // need an extra ⎿ because <Message /> already adds one.
510
- // TODO: Find a cleaner way to do this.
511
- _.content.message.content[0].text === INTERRUPT_MESSAGE ? (
512
- <Message
513
- message={_.content}
514
- messages={_.normalizedMessages}
515
- addMargin={false}
516
- tools={_.tools}
517
- verbose={verbose ?? false}
518
- debug={debug}
519
- erroredToolUseIDs={new Set()}
520
- inProgressToolUseIDs={new Set()}
521
- unresolvedToolUseIDs={new Set()}
522
- shouldAnimate={false}
523
- shouldShowDot={false}
524
- />
525
- ) : (
526
- <MessageResponse children={
527
- <Message
528
- message={_.content}
529
- messages={_.normalizedMessages}
530
- addMargin={false}
531
- tools={_.tools}
532
- verbose={verbose ?? false}
533
- debug={debug}
534
- erroredToolUseIDs={new Set()}
535
- inProgressToolUseIDs={new Set()}
536
- unresolvedToolUseIDs={
537
- new Set([
538
- (_.content.message.content[0]! as ToolUseBlockParam).id,
539
- ])
540
- }
541
- shouldAnimate={false}
542
- shouldShowDot={false}
543
- />
544
- } />
545
- )
546
- ) : (
547
- <Message
548
- message={_}
549
- messages={normalizedMessages}
550
- addMargin={true}
551
- tools={tools}
552
- verbose={verbose}
553
- debug={debug}
554
- erroredToolUseIDs={erroredToolUseIDs}
555
- inProgressToolUseIDs={inProgressToolUseIDs}
556
- shouldAnimate={
557
- !toolJSX &&
558
- !toolUseConfirm &&
559
- !isMessageSelectorVisible &&
560
- (!toolUseID || inProgressToolUseIDs.has(toolUseID))
561
- }
562
- shouldShowDot={true}
563
- unresolvedToolUseIDs={unresolvedToolUseIDs}
564
- />
565
- )
566
-
567
- const type = shouldRenderStatically(
568
- _,
569
- normalizedMessages,
570
- unresolvedToolUseIDs,
571
- )
572
- ? 'static'
573
- : 'transient'
574
-
575
- if (debug) {
576
- return {
577
- type,
578
- jsx: (
579
- <Box
580
- borderStyle="single"
581
- borderColor={type === 'static' ? 'green' : 'red'}
582
- key={_.uuid}
583
- width="100%"
584
- >
585
- {message}
586
- </Box>
587
- ),
588
- }
589
- }
590
-
591
- return {
592
- type,
593
- jsx: (
594
- <Box key={_.uuid} width="100%">
595
- {message}
596
- </Box>
597
- ),
598
- }
599
- }),
600
- ]
601
- }, [
602
- forkNumber,
603
- normalizedMessages,
604
- tools,
605
- verbose,
606
- debug,
607
- erroredToolUseIDs,
608
- inProgressToolUseIDs,
609
- toolJSX,
610
- toolUseConfirm,
611
- isMessageSelectorVisible,
612
- unresolvedToolUseIDs,
613
- mcpClients,
614
- isDefaultModel,
615
- ])
616
-
617
- // only show the dialog once not loading
618
- const showingCostDialog = !isLoading && showCostDialog
619
-
620
- return (
621
- <PermissionProvider
622
- isBypassPermissionsModeAvailable={!safeMode}
623
- children={
624
- <React.Fragment>
625
- {/* Update banner now renders inside Logo for stable placement */}
626
- <ModeIndicator />
627
- <React.Fragment key={`static-messages-${forkNumber}`}>
628
- <Static
629
- items={messagesJSX.filter(_ => _.type === 'static')}
630
- children={_ => ((_ as any).jsx)}
631
- />
632
- </React.Fragment>
633
- {messagesJSX.filter(_ => _.type === 'transient').map(_ => _.jsx)}
634
- <Box
635
- borderColor="red"
636
- borderStyle={debug ? 'single' : undefined}
637
- flexDirection="column"
638
- width="100%"
639
- >
640
- {!toolJSX && !toolUseConfirm && !binaryFeedbackContext && isLoading && (
641
- <Spinner />
642
- )}
643
- {toolJSX ? toolJSX.jsx : null}
644
- {!toolJSX && binaryFeedbackContext && !isMessageSelectorVisible && (
645
- <BinaryFeedback
646
- m1={binaryFeedbackContext.m1}
647
- m2={binaryFeedbackContext.m2}
648
- resolve={result => {
649
- binaryFeedbackContext.resolve(result)
650
- setTimeout(() => setBinaryFeedbackContext(null), 0)
651
- }}
652
- verbose={verbose}
653
- normalizedMessages={normalizedMessages}
654
- tools={tools}
655
- debug={debug}
656
- erroredToolUseIDs={erroredToolUseIDs}
657
- inProgressToolUseIDs={inProgressToolUseIDs}
658
- unresolvedToolUseIDs={unresolvedToolUseIDs}
659
- />
660
- )}
661
- {!toolJSX &&
662
- toolUseConfirm &&
663
- !isMessageSelectorVisible &&
664
- !binaryFeedbackContext && (
665
- <PermissionRequest
666
- toolUseConfirm={toolUseConfirm}
667
- onDone={() => setToolUseConfirm(null)}
668
- verbose={verbose}
669
- />
670
- )}
671
- {!toolJSX &&
672
- !toolUseConfirm &&
673
- !isMessageSelectorVisible &&
674
- !binaryFeedbackContext &&
675
- showingCostDialog && (
676
- <CostThresholdDialog
677
- onDone={() => {
678
- setShowCostDialog(false)
679
- setHaveShownCostDialog(true)
680
- const projectConfig = getGlobalConfig()
681
- saveGlobalConfig({
682
- ...projectConfig,
683
- hasAcknowledgedCostThreshold: true,
684
- })
685
- logEvent('tengu_cost_threshold_acknowledged', {})
686
- }}
687
- />
688
- )}
689
-
690
- {!toolUseConfirm &&
691
- !toolJSX?.shouldHidePromptInput &&
692
- shouldShowPromptInput &&
693
- !isMessageSelectorVisible &&
694
- !binaryFeedbackContext &&
695
- !showingCostDialog && (
696
- <>
697
- <PromptInput
698
- commands={commands}
699
- forkNumber={forkNumber}
700
- messageLogName={messageLogName}
701
- tools={tools}
702
- isDisabled={apiKeyStatus === 'invalid'}
703
- isLoading={isLoading}
704
- onQuery={onQuery}
705
- debug={debug}
706
- verbose={verbose}
707
- messages={messages}
708
- setToolJSX={setToolJSX}
709
- onAutoUpdaterResult={setAutoUpdaterResult}
710
- autoUpdaterResult={autoUpdaterResult}
711
- input={inputValue}
712
- onInputChange={setInputValue}
713
- mode={inputMode}
714
- onModeChange={setInputMode}
715
- submitCount={submitCount}
716
- onSubmitCountChange={setSubmitCount}
717
- setIsLoading={setIsLoading}
718
- setAbortController={setAbortController}
719
- onShowMessageSelector={() =>
720
- setIsMessageSelectorVisible(prev => !prev)
721
- }
722
- setForkConvoWithMessagesOnTheNextRender={
723
- setForkConvoWithMessagesOnTheNextRender
724
- }
725
- readFileTimestamps={readFileTimestamps.current}
726
- abortController={abortController}
727
- onModelChange={() => setForkNumber(prev => prev + 1)}
728
- />
729
- </>
730
- )}
731
- </Box>
732
- {isMessageSelectorVisible && (
733
- <MessageSelector
734
- erroredToolUseIDs={erroredToolUseIDs}
735
- unresolvedToolUseIDs={unresolvedToolUseIDs}
736
- messages={normalizeMessagesForAPI(messages)}
737
- onSelect={async message => {
738
- setIsMessageSelectorVisible(false)
739
-
740
- // If the user selected the current prompt, do nothing
741
- if (!messages.includes(message)) {
742
- return
743
- }
744
-
745
- // Cancel tool use calls/requests
746
- onCancel()
747
-
748
- // Hack: make sure the "Interrupted by user" message is
749
- // rendered in response to the cancellation. Otherwise,
750
- // the screen will be cleared but there will remain a
751
- // vestigial "Interrupted by user" message at the top.
752
- setImmediate(async () => {
753
- // Clear messages, and re-render
754
- await clearTerminal()
755
- setMessages([])
756
- setForkConvoWithMessagesOnTheNextRender(
757
- messages.slice(0, messages.indexOf(message)),
758
- )
759
-
760
- // Populate/reset the prompt input
761
- if (typeof message.message.content === 'string') {
762
- setInputValue(message.message.content)
763
- }
764
- })
765
- }}
766
- onEscape={() => setIsMessageSelectorVisible(false)}
767
- tools={tools}
768
- />
769
- )}
770
- {/** Fix occasional rendering artifact */}
771
- <Newline />
772
- </React.Fragment>
773
- }
774
- />
775
- )
776
- }
777
-
778
- function shouldRenderStatically(
779
- message: NormalizedMessage,
780
- messages: NormalizedMessage[],
781
- unresolvedToolUseIDs: Set<string>,
782
- ): boolean {
783
- switch (message.type) {
784
- case 'user':
785
- case 'assistant': {
786
- const toolUseID = getToolUseID(message)
787
- if (!toolUseID) {
788
- return true
789
- }
790
- if (unresolvedToolUseIDs.has(toolUseID)) {
791
- return false
792
- }
793
-
794
- const correspondingProgressMessage = messages.find(
795
- _ => _.type === 'progress' && _.toolUseID === toolUseID,
796
- ) as ProgressMessage | null
797
- if (!correspondingProgressMessage) {
798
- return true
799
- }
800
-
801
- return !intersects(
802
- unresolvedToolUseIDs,
803
- correspondingProgressMessage.siblingToolUseIDs,
804
- )
805
- }
806
- case 'progress':
807
- return !intersects(unresolvedToolUseIDs, message.siblingToolUseIDs)
808
- }
809
- }
810
-
811
- function intersects<A>(a: Set<A>, b: Set<A>): boolean {
812
- return a.size > 0 && b.size > 0 && [...a].some(_ => b.has(_))
813
- }