@shareai-lab/kode 1.0.70 → 1.0.73

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 (278) hide show
  1. package/README.md +342 -75
  2. package/README.zh-CN.md +292 -0
  3. package/cli.js +62 -0
  4. package/package.json +49 -25
  5. package/scripts/postinstall.js +56 -0
  6. package/src/ProjectOnboarding.tsx +198 -0
  7. package/src/Tool.ts +82 -0
  8. package/src/commands/agents.tsx +3401 -0
  9. package/src/commands/approvedTools.ts +53 -0
  10. package/src/commands/bug.tsx +20 -0
  11. package/src/commands/clear.ts +43 -0
  12. package/src/commands/compact.ts +120 -0
  13. package/src/commands/config.tsx +19 -0
  14. package/src/commands/cost.ts +18 -0
  15. package/src/commands/ctx_viz.ts +209 -0
  16. package/src/commands/doctor.ts +24 -0
  17. package/src/commands/help.tsx +19 -0
  18. package/src/commands/init.ts +37 -0
  19. package/src/commands/listen.ts +42 -0
  20. package/src/commands/login.tsx +51 -0
  21. package/src/commands/logout.tsx +40 -0
  22. package/src/commands/mcp.ts +41 -0
  23. package/src/commands/model.tsx +40 -0
  24. package/src/commands/modelstatus.tsx +20 -0
  25. package/src/commands/onboarding.tsx +34 -0
  26. package/src/commands/pr_comments.ts +59 -0
  27. package/src/commands/refreshCommands.ts +54 -0
  28. package/src/commands/release-notes.ts +34 -0
  29. package/src/commands/resume.tsx +31 -0
  30. package/src/commands/review.ts +49 -0
  31. package/src/commands/terminalSetup.ts +221 -0
  32. package/src/commands.ts +139 -0
  33. package/src/components/ApproveApiKey.tsx +93 -0
  34. package/src/components/AsciiLogo.tsx +13 -0
  35. package/src/components/AutoUpdater.tsx +148 -0
  36. package/src/components/Bug.tsx +367 -0
  37. package/src/components/Config.tsx +293 -0
  38. package/src/components/ConsoleOAuthFlow.tsx +327 -0
  39. package/src/components/Cost.tsx +23 -0
  40. package/src/components/CostThresholdDialog.tsx +46 -0
  41. package/src/components/CustomSelect/option-map.ts +42 -0
  42. package/src/components/CustomSelect/select-option.tsx +78 -0
  43. package/src/components/CustomSelect/select.tsx +152 -0
  44. package/src/components/CustomSelect/theme.ts +45 -0
  45. package/src/components/CustomSelect/use-select-state.ts +414 -0
  46. package/src/components/CustomSelect/use-select.ts +35 -0
  47. package/src/components/FallbackToolUseRejectedMessage.tsx +15 -0
  48. package/src/components/FileEditToolUpdatedMessage.tsx +66 -0
  49. package/src/components/Help.tsx +215 -0
  50. package/src/components/HighlightedCode.tsx +33 -0
  51. package/src/components/InvalidConfigDialog.tsx +113 -0
  52. package/src/components/Link.tsx +32 -0
  53. package/src/components/LogSelector.tsx +86 -0
  54. package/src/components/Logo.tsx +145 -0
  55. package/src/components/MCPServerApprovalDialog.tsx +100 -0
  56. package/src/components/MCPServerDialogCopy.tsx +25 -0
  57. package/src/components/MCPServerMultiselectDialog.tsx +109 -0
  58. package/src/components/Message.tsx +221 -0
  59. package/src/components/MessageResponse.tsx +15 -0
  60. package/src/components/MessageSelector.tsx +211 -0
  61. package/src/components/ModeIndicator.tsx +88 -0
  62. package/src/components/ModelConfig.tsx +301 -0
  63. package/src/components/ModelListManager.tsx +227 -0
  64. package/src/components/ModelSelector.tsx +3386 -0
  65. package/src/components/ModelStatusDisplay.tsx +230 -0
  66. package/src/components/Onboarding.tsx +274 -0
  67. package/src/components/PressEnterToContinue.tsx +11 -0
  68. package/src/components/PromptInput.tsx +740 -0
  69. package/src/components/SentryErrorBoundary.ts +33 -0
  70. package/src/components/Spinner.tsx +129 -0
  71. package/src/components/StickerRequestForm.tsx +16 -0
  72. package/src/components/StructuredDiff.tsx +191 -0
  73. package/src/components/TextInput.tsx +259 -0
  74. package/src/components/TodoItem.tsx +11 -0
  75. package/src/components/TokenWarning.tsx +31 -0
  76. package/src/components/ToolUseLoader.tsx +40 -0
  77. package/src/components/TrustDialog.tsx +106 -0
  78. package/src/components/binary-feedback/BinaryFeedback.tsx +63 -0
  79. package/src/components/binary-feedback/BinaryFeedbackOption.tsx +111 -0
  80. package/src/components/binary-feedback/BinaryFeedbackView.tsx +172 -0
  81. package/src/components/binary-feedback/utils.ts +220 -0
  82. package/src/components/messages/AssistantBashOutputMessage.tsx +22 -0
  83. package/src/components/messages/AssistantLocalCommandOutputMessage.tsx +49 -0
  84. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +19 -0
  85. package/src/components/messages/AssistantTextMessage.tsx +144 -0
  86. package/src/components/messages/AssistantThinkingMessage.tsx +40 -0
  87. package/src/components/messages/AssistantToolUseMessage.tsx +133 -0
  88. package/src/components/messages/TaskProgressMessage.tsx +32 -0
  89. package/src/components/messages/TaskToolMessage.tsx +58 -0
  90. package/src/components/messages/UserBashInputMessage.tsx +28 -0
  91. package/src/components/messages/UserCommandMessage.tsx +30 -0
  92. package/src/components/messages/UserKodingInputMessage.tsx +28 -0
  93. package/src/components/messages/UserPromptMessage.tsx +35 -0
  94. package/src/components/messages/UserTextMessage.tsx +39 -0
  95. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +12 -0
  96. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +36 -0
  97. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +31 -0
  98. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +57 -0
  99. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +35 -0
  100. package/src/components/messages/UserToolResultMessage/utils.tsx +56 -0
  101. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +121 -0
  102. package/src/components/permissions/FallbackPermissionRequest.tsx +153 -0
  103. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +182 -0
  104. package/src/components/permissions/FileEditPermissionRequest/FileEditToolDiff.tsx +77 -0
  105. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +164 -0
  106. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +83 -0
  107. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +240 -0
  108. package/src/components/permissions/PermissionRequest.tsx +101 -0
  109. package/src/components/permissions/PermissionRequestTitle.tsx +69 -0
  110. package/src/components/permissions/hooks.ts +44 -0
  111. package/src/components/permissions/toolUseOptions.ts +59 -0
  112. package/src/components/permissions/utils.ts +23 -0
  113. package/src/constants/betas.ts +5 -0
  114. package/src/constants/claude-asterisk-ascii-art.tsx +238 -0
  115. package/src/constants/figures.ts +4 -0
  116. package/src/constants/keys.ts +3 -0
  117. package/src/constants/macros.ts +8 -0
  118. package/src/constants/modelCapabilities.ts +179 -0
  119. package/src/constants/models.ts +1025 -0
  120. package/src/constants/oauth.ts +18 -0
  121. package/src/constants/product.ts +17 -0
  122. package/src/constants/prompts.ts +177 -0
  123. package/src/constants/releaseNotes.ts +7 -0
  124. package/src/context/PermissionContext.tsx +149 -0
  125. package/src/context.ts +278 -0
  126. package/src/cost-tracker.ts +84 -0
  127. package/src/entrypoints/cli.tsx +1518 -0
  128. package/src/entrypoints/mcp.ts +176 -0
  129. package/src/history.ts +25 -0
  130. package/src/hooks/useApiKeyVerification.ts +59 -0
  131. package/src/hooks/useArrowKeyHistory.ts +55 -0
  132. package/src/hooks/useCanUseTool.ts +138 -0
  133. package/src/hooks/useCancelRequest.ts +39 -0
  134. package/src/hooks/useDoublePress.ts +42 -0
  135. package/src/hooks/useExitOnCtrlCD.ts +31 -0
  136. package/src/hooks/useInterval.ts +25 -0
  137. package/src/hooks/useLogMessages.ts +16 -0
  138. package/src/hooks/useLogStartupTime.ts +12 -0
  139. package/src/hooks/useNotifyAfterTimeout.ts +65 -0
  140. package/src/hooks/usePermissionRequestLogging.ts +44 -0
  141. package/src/hooks/useTerminalSize.ts +49 -0
  142. package/src/hooks/useTextInput.ts +318 -0
  143. package/src/hooks/useUnifiedCompletion.ts +1404 -0
  144. package/src/messages.ts +38 -0
  145. package/src/permissions.ts +268 -0
  146. package/src/query.ts +707 -0
  147. package/src/screens/ConfigureNpmPrefix.tsx +197 -0
  148. package/src/screens/Doctor.tsx +219 -0
  149. package/src/screens/LogList.tsx +68 -0
  150. package/src/screens/REPL.tsx +798 -0
  151. package/src/screens/ResumeConversation.tsx +68 -0
  152. package/src/services/adapters/base.ts +38 -0
  153. package/src/services/adapters/chatCompletions.ts +90 -0
  154. package/src/services/adapters/responsesAPI.ts +170 -0
  155. package/src/services/browserMocks.ts +66 -0
  156. package/src/services/claude.ts +2083 -0
  157. package/src/services/customCommands.ts +704 -0
  158. package/src/services/fileFreshness.ts +377 -0
  159. package/src/services/gpt5ConnectionTest.ts +340 -0
  160. package/src/services/mcpClient.ts +564 -0
  161. package/src/services/mcpServerApproval.tsx +50 -0
  162. package/src/services/mentionProcessor.ts +273 -0
  163. package/src/services/modelAdapterFactory.ts +69 -0
  164. package/src/services/notifier.ts +40 -0
  165. package/src/services/oauth.ts +357 -0
  166. package/src/services/openai.ts +1305 -0
  167. package/src/services/responseStateManager.ts +90 -0
  168. package/src/services/sentry.ts +3 -0
  169. package/src/services/statsig.ts +171 -0
  170. package/src/services/statsigStorage.ts +86 -0
  171. package/src/services/systemReminder.ts +507 -0
  172. package/src/services/vcr.ts +161 -0
  173. package/src/test/testAdapters.ts +96 -0
  174. package/src/tools/ArchitectTool/ArchitectTool.tsx +122 -0
  175. package/src/tools/ArchitectTool/prompt.ts +15 -0
  176. package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +569 -0
  177. package/src/tools/BashTool/BashTool.tsx +243 -0
  178. package/src/tools/BashTool/BashToolResultMessage.tsx +38 -0
  179. package/src/tools/BashTool/OutputLine.tsx +49 -0
  180. package/src/tools/BashTool/prompt.ts +174 -0
  181. package/src/tools/BashTool/utils.ts +56 -0
  182. package/src/tools/FileEditTool/FileEditTool.tsx +315 -0
  183. package/src/tools/FileEditTool/prompt.ts +51 -0
  184. package/src/tools/FileEditTool/utils.ts +58 -0
  185. package/src/tools/FileReadTool/FileReadTool.tsx +404 -0
  186. package/src/tools/FileReadTool/prompt.ts +7 -0
  187. package/src/tools/FileWriteTool/FileWriteTool.tsx +297 -0
  188. package/src/tools/FileWriteTool/prompt.ts +10 -0
  189. package/src/tools/GlobTool/GlobTool.tsx +119 -0
  190. package/src/tools/GlobTool/prompt.ts +8 -0
  191. package/src/tools/GrepTool/GrepTool.tsx +147 -0
  192. package/src/tools/GrepTool/prompt.ts +11 -0
  193. package/src/tools/MCPTool/MCPTool.tsx +107 -0
  194. package/src/tools/MCPTool/prompt.ts +3 -0
  195. package/src/tools/MemoryReadTool/MemoryReadTool.tsx +127 -0
  196. package/src/tools/MemoryReadTool/prompt.ts +3 -0
  197. package/src/tools/MemoryWriteTool/MemoryWriteTool.tsx +89 -0
  198. package/src/tools/MemoryWriteTool/prompt.ts +3 -0
  199. package/src/tools/MultiEditTool/MultiEditTool.tsx +366 -0
  200. package/src/tools/MultiEditTool/prompt.ts +45 -0
  201. package/src/tools/NotebookEditTool/NotebookEditTool.tsx +298 -0
  202. package/src/tools/NotebookEditTool/prompt.ts +3 -0
  203. package/src/tools/NotebookReadTool/NotebookReadTool.tsx +258 -0
  204. package/src/tools/NotebookReadTool/prompt.ts +3 -0
  205. package/src/tools/StickerRequestTool/StickerRequestTool.tsx +93 -0
  206. package/src/tools/StickerRequestTool/prompt.ts +19 -0
  207. package/src/tools/TaskTool/TaskTool.tsx +466 -0
  208. package/src/tools/TaskTool/constants.ts +1 -0
  209. package/src/tools/TaskTool/prompt.ts +92 -0
  210. package/src/tools/ThinkTool/ThinkTool.tsx +54 -0
  211. package/src/tools/ThinkTool/prompt.ts +12 -0
  212. package/src/tools/TodoWriteTool/TodoWriteTool.tsx +290 -0
  213. package/src/tools/TodoWriteTool/prompt.ts +63 -0
  214. package/src/tools/lsTool/lsTool.tsx +272 -0
  215. package/src/tools/lsTool/prompt.ts +2 -0
  216. package/src/tools.ts +63 -0
  217. package/src/types/PermissionMode.ts +120 -0
  218. package/src/types/RequestContext.ts +72 -0
  219. package/src/types/conversation.ts +51 -0
  220. package/src/types/logs.ts +58 -0
  221. package/src/types/modelCapabilities.ts +64 -0
  222. package/src/types/notebook.ts +87 -0
  223. package/src/utils/Cursor.ts +436 -0
  224. package/src/utils/PersistentShell.ts +373 -0
  225. package/src/utils/advancedFuzzyMatcher.ts +290 -0
  226. package/src/utils/agentLoader.ts +284 -0
  227. package/src/utils/agentStorage.ts +97 -0
  228. package/src/utils/array.ts +3 -0
  229. package/src/utils/ask.tsx +99 -0
  230. package/src/utils/auth.ts +13 -0
  231. package/src/utils/autoCompactCore.ts +223 -0
  232. package/src/utils/autoUpdater.ts +318 -0
  233. package/src/utils/betas.ts +20 -0
  234. package/src/utils/browser.ts +14 -0
  235. package/src/utils/cleanup.ts +72 -0
  236. package/src/utils/commands.ts +261 -0
  237. package/src/utils/commonUnixCommands.ts +161 -0
  238. package/src/utils/config.ts +942 -0
  239. package/src/utils/conversationRecovery.ts +55 -0
  240. package/src/utils/debugLogger.ts +1123 -0
  241. package/src/utils/diff.ts +42 -0
  242. package/src/utils/env.ts +57 -0
  243. package/src/utils/errors.ts +21 -0
  244. package/src/utils/exampleCommands.ts +109 -0
  245. package/src/utils/execFileNoThrow.ts +51 -0
  246. package/src/utils/expertChatStorage.ts +136 -0
  247. package/src/utils/file.ts +402 -0
  248. package/src/utils/fileRecoveryCore.ts +71 -0
  249. package/src/utils/format.tsx +44 -0
  250. package/src/utils/fuzzyMatcher.ts +328 -0
  251. package/src/utils/generators.ts +62 -0
  252. package/src/utils/git.ts +92 -0
  253. package/src/utils/globalLogger.ts +77 -0
  254. package/src/utils/http.ts +10 -0
  255. package/src/utils/imagePaste.ts +38 -0
  256. package/src/utils/json.ts +13 -0
  257. package/src/utils/log.ts +382 -0
  258. package/src/utils/markdown.ts +213 -0
  259. package/src/utils/messageContextManager.ts +289 -0
  260. package/src/utils/messages.tsx +939 -0
  261. package/src/utils/model.ts +836 -0
  262. package/src/utils/permissions/filesystem.ts +118 -0
  263. package/src/utils/responseState.ts +23 -0
  264. package/src/utils/ripgrep.ts +167 -0
  265. package/src/utils/secureFile.ts +559 -0
  266. package/src/utils/sessionState.ts +49 -0
  267. package/src/utils/state.ts +25 -0
  268. package/src/utils/style.ts +29 -0
  269. package/src/utils/terminal.ts +50 -0
  270. package/src/utils/theme.ts +133 -0
  271. package/src/utils/thinking.ts +144 -0
  272. package/src/utils/todoStorage.ts +431 -0
  273. package/src/utils/tokens.ts +43 -0
  274. package/src/utils/toolExecutionController.ts +163 -0
  275. package/src/utils/unaryLogging.ts +26 -0
  276. package/src/utils/user.ts +37 -0
  277. package/src/utils/validate.ts +165 -0
  278. package/cli.mjs +0 -1803
@@ -0,0 +1,382 @@
1
+ import { existsSync, mkdirSync } from 'fs'
2
+ import { dirname, join } from 'path'
3
+ import { writeFileSync, readFileSync } from 'fs'
4
+ import { captureException } from '../services/sentry'
5
+ import { randomUUID } from 'crypto'
6
+ import envPaths from 'env-paths'
7
+ import { promises as fsPromises } from 'fs'
8
+ import type { LogOption, SerializedMessage } from '../types/logs'
9
+ import { MACRO } from '../constants/macros'
10
+ import { PRODUCT_COMMAND } from '../constants/product'
11
+ const IN_MEMORY_ERROR_LOG: Array<{
12
+ error: string
13
+ timestamp: string
14
+ }> = []
15
+ const MAX_IN_MEMORY_ERRORS = 100 // Limit to prevent memory issues
16
+
17
+ export const SESSION_ID = randomUUID()
18
+
19
+ const paths = envPaths(PRODUCT_COMMAND)
20
+
21
+ function getProjectDir(cwd: string): string {
22
+ return cwd.replace(/[^a-zA-Z0-9]/g, '-')
23
+ }
24
+
25
+ export const CACHE_PATHS = {
26
+ errors: () => join(paths.cache, getProjectDir(process.cwd()), 'errors'),
27
+ messages: () => join(paths.cache, getProjectDir(process.cwd()), 'messages'),
28
+ mcpLogs: (serverName: string) =>
29
+ join(paths.cache, getProjectDir(process.cwd()), `mcp-logs-${serverName}`),
30
+ }
31
+
32
+ export function dateToFilename(date: Date): string {
33
+ return date.toISOString().replace(/[:.]/g, '-')
34
+ }
35
+
36
+ const DATE = dateToFilename(new Date())
37
+
38
+ function getErrorsPath(): string {
39
+ return join(CACHE_PATHS.errors(), DATE + '.txt')
40
+ }
41
+
42
+ export function getMessagesPath(
43
+ messageLogName: string,
44
+ forkNumber: number,
45
+ sidechainNumber: number,
46
+ ): string {
47
+ return join(
48
+ CACHE_PATHS.messages(),
49
+ `${messageLogName}${forkNumber > 0 ? `-${forkNumber}` : ''}${
50
+ sidechainNumber > 0 ? `-sidechain-${sidechainNumber}` : ''
51
+ }.json`,
52
+ )
53
+ }
54
+
55
+ export function logError(error: unknown): void {
56
+ try {
57
+ if (process.env.NODE_ENV === 'test') {
58
+ console.error(error)
59
+ }
60
+
61
+ const errorStr =
62
+ error instanceof Error ? error.stack || error.message : String(error)
63
+
64
+ const errorInfo = {
65
+ error: errorStr,
66
+ timestamp: new Date().toISOString(),
67
+ }
68
+
69
+ if (IN_MEMORY_ERROR_LOG.length >= MAX_IN_MEMORY_ERRORS) {
70
+ IN_MEMORY_ERROR_LOG.shift() // Remove oldest error
71
+ }
72
+ IN_MEMORY_ERROR_LOG.push(errorInfo)
73
+
74
+ appendToLog(getErrorsPath(), {
75
+ error: errorStr,
76
+ })
77
+ } catch {
78
+ // pass
79
+ }
80
+ // Also send to Sentry with session ID, but don't await
81
+ captureException(error)
82
+ }
83
+
84
+ export function getErrorsLog(): object[] {
85
+ return readLog(getErrorsPath())
86
+ }
87
+
88
+ export function getInMemoryErrors(): object[] {
89
+ return [...IN_MEMORY_ERROR_LOG]
90
+ }
91
+
92
+ function readLog(path: string): object[] {
93
+ if (!existsSync(path)) {
94
+ return []
95
+ }
96
+ try {
97
+ return JSON.parse(readFileSync(path, 'utf8'))
98
+ } catch {
99
+ return []
100
+ }
101
+ }
102
+
103
+ function appendToLog(path: string, message: object): void {
104
+ if (process.env.USER_TYPE === 'external') {
105
+ return
106
+ }
107
+
108
+ const dir = dirname(path)
109
+ if (!existsSync(dir)) {
110
+ mkdirSync(dir, { recursive: true })
111
+ }
112
+
113
+ // Create messages file with empty array if it doesn't exist
114
+ if (!existsSync(path)) {
115
+ writeFileSync(path, '[]', 'utf8')
116
+ }
117
+
118
+ const messages = readLog(path)
119
+ const messageWithTimestamp = {
120
+ ...message,
121
+ cwd: process.cwd(),
122
+ userType: process.env.USER_TYPE,
123
+ sessionId: SESSION_ID,
124
+ timestamp: new Date().toISOString(),
125
+ version: MACRO.VERSION,
126
+ }
127
+ messages.push(messageWithTimestamp)
128
+
129
+ writeFileSync(path, JSON.stringify(messages, null, 2), 'utf8')
130
+ }
131
+
132
+ export function overwriteLog(path: string, messages: object[]): void {
133
+ if (process.env.USER_TYPE === 'external') {
134
+ return
135
+ }
136
+
137
+ if (!messages.length) {
138
+ return
139
+ }
140
+
141
+ const dir = dirname(path)
142
+ if (!existsSync(dir)) {
143
+ mkdirSync(dir, { recursive: true })
144
+ }
145
+
146
+ const messagesWithMetadata = messages.map(message => ({
147
+ ...message,
148
+ cwd: process.cwd(),
149
+ userType: process.env.USER_TYPE,
150
+ sessionId: SESSION_ID,
151
+ timestamp: new Date().toISOString(),
152
+ version: MACRO.VERSION,
153
+ }))
154
+
155
+ writeFileSync(path, JSON.stringify(messagesWithMetadata, null, 2), 'utf8')
156
+ }
157
+
158
+ export async function loadLogList(
159
+ path = CACHE_PATHS.messages(),
160
+ ): Promise<LogOption[]> {
161
+ if (!existsSync(path)) {
162
+ logError(`No logs found at ${path}`)
163
+ return []
164
+ }
165
+
166
+ const files = await fsPromises.readdir(path)
167
+ const logData = await Promise.all(
168
+ files.map(async (file, i) => {
169
+ const fullPath = join(path, file)
170
+ const content = await fsPromises.readFile(fullPath, 'utf8')
171
+ const messages = JSON.parse(content) as SerializedMessage[]
172
+ const firstMessage = messages[0]
173
+ const lastMessage = messages[messages.length - 1]
174
+ const firstPrompt =
175
+ firstMessage?.type === 'user' &&
176
+ typeof firstMessage?.message?.content === 'string'
177
+ ? firstMessage?.message?.content
178
+ : 'No prompt'
179
+
180
+ const { date, forkNumber, sidechainNumber } = parseLogFilename(file)
181
+ return {
182
+ date,
183
+ forkNumber,
184
+ fullPath,
185
+ messages,
186
+ value: i, // hack: overwritten after sorting, right below this
187
+ created: parseISOString(firstMessage?.timestamp || date),
188
+ modified: lastMessage?.timestamp
189
+ ? parseISOString(lastMessage.timestamp)
190
+ : parseISOString(date),
191
+ firstPrompt:
192
+ firstPrompt.split('\n')[0]?.slice(0, 50) +
193
+ (firstPrompt.length > 50 ? '…' : '') || 'No prompt',
194
+ messageCount: messages.length,
195
+ sidechainNumber,
196
+ }
197
+ }),
198
+ )
199
+
200
+ return sortLogs(logData.filter(_ => _.messages.length)).map((_, i) => ({
201
+ ..._,
202
+ value: i,
203
+ }))
204
+ }
205
+
206
+ export function parseLogFilename(filename: string): {
207
+ date: string
208
+ forkNumber: number | undefined
209
+ sidechainNumber: number | undefined
210
+ } {
211
+ const base = filename.split('.')[0]!
212
+ // Default timestamp format has 6 segments: 2025-01-27T01-31-35-104Z
213
+ const segments = base.split('-')
214
+ const hasSidechain = base.includes('-sidechain-')
215
+
216
+ let date = base
217
+ let forkNumber: number | undefined = undefined
218
+ let sidechainNumber: number | undefined = undefined
219
+
220
+ if (hasSidechain) {
221
+ const sidechainIndex = segments.indexOf('sidechain')
222
+ sidechainNumber = Number(segments[sidechainIndex + 1])
223
+ // Fork number is before sidechain if exists
224
+ if (sidechainIndex > 6) {
225
+ forkNumber = Number(segments[sidechainIndex - 1])
226
+ date = segments.slice(0, 6).join('-')
227
+ } else {
228
+ date = segments.slice(0, 6).join('-')
229
+ }
230
+ } else if (segments.length > 6) {
231
+ // Has fork number
232
+ const lastSegment = Number(segments[segments.length - 1])
233
+ forkNumber = lastSegment >= 0 ? lastSegment : undefined
234
+ date = segments.slice(0, 6).join('-')
235
+ } else {
236
+ // Basic timestamp only
237
+ date = base
238
+ }
239
+
240
+ return { date, forkNumber, sidechainNumber }
241
+ }
242
+
243
+ export function getNextAvailableLogForkNumber(
244
+ date: string,
245
+ forkNumber: number,
246
+ // Main chain has sidechainNumber 0
247
+ sidechainNumber: number,
248
+ ): number {
249
+ while (existsSync(getMessagesPath(date, forkNumber, sidechainNumber))) {
250
+ forkNumber++
251
+ }
252
+ return forkNumber
253
+ }
254
+
255
+ export function getNextAvailableLogSidechainNumber(
256
+ date: string,
257
+ forkNumber: number,
258
+ ): number {
259
+ let sidechainNumber = 1
260
+ while (existsSync(getMessagesPath(date, forkNumber, sidechainNumber))) {
261
+ sidechainNumber++
262
+ }
263
+ return sidechainNumber
264
+ }
265
+
266
+ export function getForkNumberFromFilename(
267
+ filename: string,
268
+ ): number | undefined {
269
+ const base = filename.split('.')[0]!
270
+ const segments = base.split('-')
271
+ const hasSidechain = base.includes('-sidechain-')
272
+
273
+ if (hasSidechain) {
274
+ const sidechainIndex = segments.indexOf('sidechain')
275
+ if (sidechainIndex > 6) {
276
+ return Number(segments[sidechainIndex - 1])
277
+ }
278
+ return undefined
279
+ }
280
+
281
+ if (segments.length > 6) {
282
+ const lastNumber = Number(segments[segments.length - 1])
283
+ return lastNumber >= 0 ? lastNumber : undefined
284
+ }
285
+ return undefined
286
+ }
287
+
288
+ export function sortLogs(logs: LogOption[]): LogOption[] {
289
+ return logs.sort((a, b) => {
290
+ // Sort by modified date (newest first)
291
+ const modifiedDiff = b.modified.getTime() - a.modified.getTime()
292
+ if (modifiedDiff !== 0) {
293
+ return modifiedDiff
294
+ }
295
+
296
+ // If modified dates are equal, sort by created date
297
+ const createdDiff = b.created.getTime() - a.created.getTime()
298
+ if (createdDiff !== 0) {
299
+ return createdDiff
300
+ }
301
+
302
+ // If both dates are equal, sort by fork number
303
+ return (b.forkNumber ?? 0) - (a.forkNumber ?? 0)
304
+ })
305
+ }
306
+
307
+ export function formatDate(date: Date): string {
308
+ const now = new Date()
309
+ const yesterday = new Date(now)
310
+ yesterday.setDate(yesterday.getDate() - 1)
311
+
312
+ const isToday = date.toDateString() === now.toDateString()
313
+ const isYesterday = date.toDateString() === yesterday.toDateString()
314
+
315
+ const timeStr = date
316
+ .toLocaleTimeString('en-US', {
317
+ hour: 'numeric',
318
+ minute: '2-digit',
319
+ hour12: true,
320
+ })
321
+ .toLowerCase()
322
+
323
+ if (isToday) {
324
+ return `Today at ${timeStr}`
325
+ } else if (isYesterday) {
326
+ return `Yesterday at ${timeStr}`
327
+ } else {
328
+ return (
329
+ date.toLocaleDateString('en-US', {
330
+ month: 'short',
331
+ day: 'numeric',
332
+ }) + ` at ${timeStr}`
333
+ )
334
+ }
335
+ }
336
+
337
+ export function parseISOString(s: string): Date {
338
+ const b = s.split(/\D+/)
339
+ return new Date(
340
+ Date.UTC(
341
+ parseInt(b[0]!, 10),
342
+ parseInt(b[1]!, 10) - 1,
343
+ parseInt(b[2]!, 10),
344
+ parseInt(b[3]!, 10),
345
+ parseInt(b[4]!, 10),
346
+ parseInt(b[5]!, 10),
347
+ parseInt(b[6]!, 10),
348
+ ),
349
+ )
350
+ }
351
+
352
+ export function logMCPError(serverName: string, error: unknown): void {
353
+ try {
354
+ const logDir = CACHE_PATHS.mcpLogs(serverName)
355
+ const errorStr =
356
+ error instanceof Error ? error.stack || error.message : String(error)
357
+ const timestamp = new Date().toISOString()
358
+
359
+ const logFile = join(logDir, DATE + '.txt')
360
+
361
+ if (!existsSync(logDir)) {
362
+ mkdirSync(logDir, { recursive: true })
363
+ }
364
+
365
+ if (!existsSync(logFile)) {
366
+ writeFileSync(logFile, '[]', 'utf8')
367
+ }
368
+
369
+ const errorInfo = {
370
+ error: errorStr,
371
+ timestamp,
372
+ sessionId: SESSION_ID,
373
+ cwd: process.cwd(),
374
+ }
375
+
376
+ const messages = readLog(logFile)
377
+ messages.push(errorInfo)
378
+ writeFileSync(logFile, JSON.stringify(messages, null, 2), 'utf8')
379
+ } catch {
380
+ // Silently fail
381
+ }
382
+ }
@@ -0,0 +1,213 @@
1
+ import { marked, Token } from 'marked'
2
+ import { stripSystemMessages } from './messages'
3
+ import chalk from 'chalk'
4
+ import { EOL } from 'os'
5
+ import { highlight, supportsLanguage } from 'cli-highlight'
6
+ import { logError } from './log'
7
+
8
+ export function applyMarkdown(content: string): string {
9
+ return marked
10
+ .lexer(stripSystemMessages(content))
11
+ .map(_ => format(_))
12
+ .join('')
13
+ .trim()
14
+ }
15
+
16
+ function format(
17
+ token: Token,
18
+ listDepth = 0,
19
+ orderedListNumber: number | null = null,
20
+ parent: Token | null = null,
21
+ ): string {
22
+ switch (token.type) {
23
+ case 'blockquote':
24
+ return chalk.dim.italic((token.tokens ?? []).map(_ => format(_)).join(''))
25
+ case 'code':
26
+ if (token.lang && supportsLanguage(token.lang)) {
27
+ return highlight(token.text, { language: token.lang }) + EOL
28
+ } else {
29
+ logError(
30
+ `Language not supported while highlighting code, falling back to markdown: ${token.lang}`,
31
+ )
32
+ return highlight(token.text, { language: 'markdown' }) + EOL
33
+ }
34
+ case 'codespan':
35
+ // inline code
36
+ return chalk.blue(token.text)
37
+ case 'em':
38
+ return chalk.italic((token.tokens ?? []).map(_ => format(_)).join(''))
39
+ case 'strong':
40
+ return chalk.bold((token.tokens ?? []).map(_ => format(_)).join(''))
41
+ case 'heading':
42
+ switch (token.depth) {
43
+ case 1: // h1
44
+ return (
45
+ chalk.bold.italic.underline(
46
+ (token.tokens ?? []).map(_ => format(_)).join(''),
47
+ ) +
48
+ EOL +
49
+ EOL
50
+ )
51
+ case 2: // h2
52
+ return (
53
+ chalk.bold((token.tokens ?? []).map(_ => format(_)).join('')) +
54
+ EOL +
55
+ EOL
56
+ )
57
+ default: // h3+
58
+ return (
59
+ chalk.bold.dim((token.tokens ?? []).map(_ => format(_)).join('')) +
60
+ EOL +
61
+ EOL
62
+ )
63
+ }
64
+ case 'hr':
65
+ return '---'
66
+ case 'image':
67
+ return `[Image: ${token.title}: ${token.href}]`
68
+ case 'link':
69
+ return chalk.blue(token.href)
70
+ case 'list': {
71
+ return token.items
72
+ .map((_: Token, index: number) =>
73
+ format(
74
+ _,
75
+ listDepth,
76
+ token.ordered ? token.start + index : null,
77
+ token,
78
+ ),
79
+ )
80
+ .join('')
81
+ }
82
+ case 'list_item':
83
+ return (token.tokens ?? [])
84
+ .map(
85
+ _ =>
86
+ `${' '.repeat(listDepth)}${format(_, listDepth + 1, orderedListNumber, token)}`,
87
+ )
88
+ .join('')
89
+ case 'paragraph':
90
+ return (token.tokens ?? []).map(_ => format(_)).join('') + EOL
91
+ case 'space':
92
+ return EOL
93
+ case 'text':
94
+ if (parent?.type === 'list_item') {
95
+ return `${orderedListNumber === null ? '-' : getListNumber(listDepth, orderedListNumber) + '.'} ${token.tokens ? token.tokens.map(_ => format(_, listDepth, orderedListNumber, token)).join('') : token.text}${EOL}`
96
+ } else {
97
+ return token.text
98
+ }
99
+ }
100
+ // TODO: tables
101
+ return ''
102
+ }
103
+
104
+ const DEPTH_1_LIST_NUMBERS = [
105
+ 'a',
106
+ 'b',
107
+ 'c',
108
+ 'd',
109
+ 'e',
110
+ 'f',
111
+ 'g',
112
+ 'h',
113
+ 'i',
114
+ 'j',
115
+ 'k',
116
+ 'l',
117
+ 'm',
118
+ 'n',
119
+ 'o',
120
+ 'p',
121
+ 'q',
122
+ 'r',
123
+ 's',
124
+ 't',
125
+ 'u',
126
+ 'v',
127
+ 'w',
128
+ 'x',
129
+ 'y',
130
+ 'z',
131
+ 'aa',
132
+ 'ab',
133
+ 'ac',
134
+ 'ad',
135
+ 'ae',
136
+ 'af',
137
+ 'ag',
138
+ 'ah',
139
+ 'ai',
140
+ 'aj',
141
+ 'ak',
142
+ 'al',
143
+ 'am',
144
+ 'an',
145
+ 'ao',
146
+ 'ap',
147
+ 'aq',
148
+ 'ar',
149
+ 'as',
150
+ 'at',
151
+ 'au',
152
+ 'av',
153
+ 'aw',
154
+ 'ax',
155
+ 'ay',
156
+ 'az',
157
+ ]
158
+ const DEPTH_2_LIST_NUMBERS = [
159
+ 'i',
160
+ 'ii',
161
+ 'iii',
162
+ 'iv',
163
+ 'v',
164
+ 'vi',
165
+ 'vii',
166
+ 'viii',
167
+ 'ix',
168
+ 'x',
169
+ 'xi',
170
+ 'xii',
171
+ 'xiii',
172
+ 'xiv',
173
+ 'xv',
174
+ 'xvi',
175
+ 'xvii',
176
+ 'xviii',
177
+ 'xix',
178
+ 'xx',
179
+ 'xxi',
180
+ 'xxii',
181
+ 'xxiii',
182
+ 'xxiv',
183
+ 'xxv',
184
+ 'xxvi',
185
+ 'xxvii',
186
+ 'xxviii',
187
+ 'xxix',
188
+ 'xxx',
189
+ 'xxxi',
190
+ 'xxxii',
191
+ 'xxxiii',
192
+ 'xxxiv',
193
+ 'xxxv',
194
+ 'xxxvi',
195
+ 'xxxvii',
196
+ 'xxxviii',
197
+ 'xxxix',
198
+ 'xl',
199
+ ]
200
+
201
+ function getListNumber(listDepth: number, orderedListNumber: number): string {
202
+ switch (listDepth) {
203
+ case 0:
204
+ case 1:
205
+ return orderedListNumber.toString()
206
+ case 2:
207
+ return DEPTH_1_LIST_NUMBERS[orderedListNumber - 1]! // TODO: don't hard code the list
208
+ case 3:
209
+ return DEPTH_2_LIST_NUMBERS[orderedListNumber - 1]! // TODO: don't hard code the list
210
+ default:
211
+ return orderedListNumber.toString()
212
+ }
213
+ }