@shareai-lab/kode 1.0.69 → 1.0.71

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 (253) hide show
  1. package/README.md +205 -72
  2. package/README.zh-CN.md +246 -0
  3. package/cli.js +62 -0
  4. package/package.json +45 -25
  5. package/scripts/postinstall.js +56 -0
  6. package/src/ProjectOnboarding.tsx +180 -0
  7. package/src/Tool.ts +53 -0
  8. package/src/commands/approvedTools.ts +53 -0
  9. package/src/commands/bug.tsx +20 -0
  10. package/src/commands/clear.ts +43 -0
  11. package/src/commands/compact.ts +120 -0
  12. package/src/commands/config.tsx +19 -0
  13. package/src/commands/cost.ts +18 -0
  14. package/src/commands/ctx_viz.ts +209 -0
  15. package/src/commands/doctor.ts +24 -0
  16. package/src/commands/help.tsx +19 -0
  17. package/src/commands/init.ts +37 -0
  18. package/src/commands/listen.ts +42 -0
  19. package/src/commands/login.tsx +51 -0
  20. package/src/commands/logout.tsx +40 -0
  21. package/src/commands/mcp.ts +41 -0
  22. package/src/commands/model.tsx +40 -0
  23. package/src/commands/modelstatus.tsx +20 -0
  24. package/src/commands/onboarding.tsx +34 -0
  25. package/src/commands/pr_comments.ts +59 -0
  26. package/src/commands/refreshCommands.ts +54 -0
  27. package/src/commands/release-notes.ts +34 -0
  28. package/src/commands/resume.tsx +30 -0
  29. package/src/commands/review.ts +49 -0
  30. package/src/commands/terminalSetup.ts +221 -0
  31. package/src/commands.ts +136 -0
  32. package/src/components/ApproveApiKey.tsx +93 -0
  33. package/src/components/AsciiLogo.tsx +13 -0
  34. package/src/components/AutoUpdater.tsx +148 -0
  35. package/src/components/Bug.tsx +367 -0
  36. package/src/components/Config.tsx +289 -0
  37. package/src/components/ConsoleOAuthFlow.tsx +326 -0
  38. package/src/components/Cost.tsx +23 -0
  39. package/src/components/CostThresholdDialog.tsx +46 -0
  40. package/src/components/CustomSelect/option-map.ts +42 -0
  41. package/src/components/CustomSelect/select-option.tsx +52 -0
  42. package/src/components/CustomSelect/select.tsx +143 -0
  43. package/src/components/CustomSelect/use-select-state.ts +414 -0
  44. package/src/components/CustomSelect/use-select.ts +35 -0
  45. package/src/components/FallbackToolUseRejectedMessage.tsx +15 -0
  46. package/src/components/FileEditToolUpdatedMessage.tsx +66 -0
  47. package/src/components/Help.tsx +215 -0
  48. package/src/components/HighlightedCode.tsx +33 -0
  49. package/src/components/InvalidConfigDialog.tsx +113 -0
  50. package/src/components/Link.tsx +32 -0
  51. package/src/components/LogSelector.tsx +86 -0
  52. package/src/components/Logo.tsx +145 -0
  53. package/src/components/MCPServerApprovalDialog.tsx +100 -0
  54. package/src/components/MCPServerDialogCopy.tsx +25 -0
  55. package/src/components/MCPServerMultiselectDialog.tsx +109 -0
  56. package/src/components/Message.tsx +219 -0
  57. package/src/components/MessageResponse.tsx +15 -0
  58. package/src/components/MessageSelector.tsx +211 -0
  59. package/src/components/ModeIndicator.tsx +88 -0
  60. package/src/components/ModelConfig.tsx +301 -0
  61. package/src/components/ModelListManager.tsx +223 -0
  62. package/src/components/ModelSelector.tsx +3208 -0
  63. package/src/components/ModelStatusDisplay.tsx +228 -0
  64. package/src/components/Onboarding.tsx +274 -0
  65. package/src/components/PressEnterToContinue.tsx +11 -0
  66. package/src/components/PromptInput.tsx +710 -0
  67. package/src/components/SentryErrorBoundary.ts +33 -0
  68. package/src/components/Spinner.tsx +129 -0
  69. package/src/components/StructuredDiff.tsx +184 -0
  70. package/src/components/TextInput.tsx +246 -0
  71. package/src/components/TokenWarning.tsx +31 -0
  72. package/src/components/ToolUseLoader.tsx +40 -0
  73. package/src/components/TrustDialog.tsx +106 -0
  74. package/src/components/binary-feedback/BinaryFeedback.tsx +63 -0
  75. package/src/components/binary-feedback/BinaryFeedbackOption.tsx +111 -0
  76. package/src/components/binary-feedback/BinaryFeedbackView.tsx +172 -0
  77. package/src/components/binary-feedback/utils.ts +220 -0
  78. package/src/components/messages/AssistantBashOutputMessage.tsx +22 -0
  79. package/src/components/messages/AssistantLocalCommandOutputMessage.tsx +45 -0
  80. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +19 -0
  81. package/src/components/messages/AssistantTextMessage.tsx +144 -0
  82. package/src/components/messages/AssistantThinkingMessage.tsx +40 -0
  83. package/src/components/messages/AssistantToolUseMessage.tsx +123 -0
  84. package/src/components/messages/UserBashInputMessage.tsx +28 -0
  85. package/src/components/messages/UserCommandMessage.tsx +30 -0
  86. package/src/components/messages/UserKodingInputMessage.tsx +28 -0
  87. package/src/components/messages/UserPromptMessage.tsx +35 -0
  88. package/src/components/messages/UserTextMessage.tsx +39 -0
  89. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +12 -0
  90. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +36 -0
  91. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +31 -0
  92. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +57 -0
  93. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +35 -0
  94. package/src/components/messages/UserToolResultMessage/utils.tsx +56 -0
  95. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +121 -0
  96. package/src/components/permissions/FallbackPermissionRequest.tsx +155 -0
  97. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +182 -0
  98. package/src/components/permissions/FileEditPermissionRequest/FileEditToolDiff.tsx +75 -0
  99. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +164 -0
  100. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +81 -0
  101. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +242 -0
  102. package/src/components/permissions/PermissionRequest.tsx +103 -0
  103. package/src/components/permissions/PermissionRequestTitle.tsx +69 -0
  104. package/src/components/permissions/hooks.ts +44 -0
  105. package/src/components/permissions/toolUseOptions.ts +59 -0
  106. package/src/components/permissions/utils.ts +23 -0
  107. package/src/constants/betas.ts +5 -0
  108. package/src/constants/claude-asterisk-ascii-art.tsx +238 -0
  109. package/src/constants/figures.ts +4 -0
  110. package/src/constants/keys.ts +3 -0
  111. package/src/constants/macros.ts +6 -0
  112. package/src/constants/models.ts +935 -0
  113. package/src/constants/oauth.ts +18 -0
  114. package/src/constants/product.ts +17 -0
  115. package/src/constants/prompts.ts +177 -0
  116. package/src/constants/releaseNotes.ts +7 -0
  117. package/src/context/PermissionContext.tsx +149 -0
  118. package/src/context.ts +278 -0
  119. package/src/cost-tracker.ts +84 -0
  120. package/src/entrypoints/cli.tsx +1498 -0
  121. package/src/entrypoints/mcp.ts +176 -0
  122. package/src/history.ts +25 -0
  123. package/src/hooks/useApiKeyVerification.ts +59 -0
  124. package/src/hooks/useArrowKeyHistory.ts +55 -0
  125. package/src/hooks/useCanUseTool.ts +138 -0
  126. package/src/hooks/useCancelRequest.ts +39 -0
  127. package/src/hooks/useDoublePress.ts +42 -0
  128. package/src/hooks/useExitOnCtrlCD.ts +31 -0
  129. package/src/hooks/useInterval.ts +25 -0
  130. package/src/hooks/useLogMessages.ts +16 -0
  131. package/src/hooks/useLogStartupTime.ts +12 -0
  132. package/src/hooks/useNotifyAfterTimeout.ts +65 -0
  133. package/src/hooks/usePermissionRequestLogging.ts +44 -0
  134. package/src/hooks/useSlashCommandTypeahead.ts +137 -0
  135. package/src/hooks/useTerminalSize.ts +49 -0
  136. package/src/hooks/useTextInput.ts +315 -0
  137. package/src/messages.ts +37 -0
  138. package/src/permissions.ts +268 -0
  139. package/src/query.ts +704 -0
  140. package/src/screens/ConfigureNpmPrefix.tsx +197 -0
  141. package/src/screens/Doctor.tsx +219 -0
  142. package/src/screens/LogList.tsx +68 -0
  143. package/src/screens/REPL.tsx +792 -0
  144. package/src/screens/ResumeConversation.tsx +68 -0
  145. package/src/services/browserMocks.ts +66 -0
  146. package/src/services/claude.ts +1947 -0
  147. package/src/services/customCommands.ts +683 -0
  148. package/src/services/fileFreshness.ts +377 -0
  149. package/src/services/mcpClient.ts +564 -0
  150. package/src/services/mcpServerApproval.tsx +50 -0
  151. package/src/services/notifier.ts +40 -0
  152. package/src/services/oauth.ts +357 -0
  153. package/src/services/openai.ts +796 -0
  154. package/src/services/sentry.ts +3 -0
  155. package/src/services/statsig.ts +171 -0
  156. package/src/services/statsigStorage.ts +86 -0
  157. package/src/services/systemReminder.ts +406 -0
  158. package/src/services/vcr.ts +161 -0
  159. package/src/tools/ArchitectTool/ArchitectTool.tsx +122 -0
  160. package/src/tools/ArchitectTool/prompt.ts +15 -0
  161. package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +505 -0
  162. package/src/tools/BashTool/BashTool.tsx +270 -0
  163. package/src/tools/BashTool/BashToolResultMessage.tsx +38 -0
  164. package/src/tools/BashTool/OutputLine.tsx +48 -0
  165. package/src/tools/BashTool/prompt.ts +174 -0
  166. package/src/tools/BashTool/utils.ts +56 -0
  167. package/src/tools/FileEditTool/FileEditTool.tsx +316 -0
  168. package/src/tools/FileEditTool/prompt.ts +51 -0
  169. package/src/tools/FileEditTool/utils.ts +58 -0
  170. package/src/tools/FileReadTool/FileReadTool.tsx +371 -0
  171. package/src/tools/FileReadTool/prompt.ts +7 -0
  172. package/src/tools/FileWriteTool/FileWriteTool.tsx +297 -0
  173. package/src/tools/FileWriteTool/prompt.ts +10 -0
  174. package/src/tools/GlobTool/GlobTool.tsx +119 -0
  175. package/src/tools/GlobTool/prompt.ts +8 -0
  176. package/src/tools/GrepTool/GrepTool.tsx +147 -0
  177. package/src/tools/GrepTool/prompt.ts +11 -0
  178. package/src/tools/MCPTool/MCPTool.tsx +106 -0
  179. package/src/tools/MCPTool/prompt.ts +3 -0
  180. package/src/tools/MemoryReadTool/MemoryReadTool.tsx +127 -0
  181. package/src/tools/MemoryReadTool/prompt.ts +3 -0
  182. package/src/tools/MemoryWriteTool/MemoryWriteTool.tsx +89 -0
  183. package/src/tools/MemoryWriteTool/prompt.ts +3 -0
  184. package/src/tools/MultiEditTool/MultiEditTool.tsx +366 -0
  185. package/src/tools/MultiEditTool/prompt.ts +45 -0
  186. package/src/tools/NotebookEditTool/NotebookEditTool.tsx +298 -0
  187. package/src/tools/NotebookEditTool/prompt.ts +3 -0
  188. package/src/tools/NotebookReadTool/NotebookReadTool.tsx +266 -0
  189. package/src/tools/NotebookReadTool/prompt.ts +3 -0
  190. package/src/tools/StickerRequestTool/StickerRequestTool.tsx +93 -0
  191. package/src/tools/StickerRequestTool/prompt.ts +19 -0
  192. package/src/tools/TaskTool/TaskTool.tsx +382 -0
  193. package/src/tools/TaskTool/constants.ts +1 -0
  194. package/src/tools/TaskTool/prompt.ts +56 -0
  195. package/src/tools/ThinkTool/ThinkTool.tsx +56 -0
  196. package/src/tools/ThinkTool/prompt.ts +12 -0
  197. package/src/tools/TodoWriteTool/TodoWriteTool.tsx +289 -0
  198. package/src/tools/TodoWriteTool/prompt.ts +63 -0
  199. package/src/tools/lsTool/lsTool.tsx +269 -0
  200. package/src/tools/lsTool/prompt.ts +2 -0
  201. package/src/tools.ts +63 -0
  202. package/src/types/PermissionMode.ts +120 -0
  203. package/src/types/RequestContext.ts +72 -0
  204. package/src/utils/Cursor.ts +436 -0
  205. package/src/utils/PersistentShell.ts +373 -0
  206. package/src/utils/agentStorage.ts +97 -0
  207. package/src/utils/array.ts +3 -0
  208. package/src/utils/ask.tsx +98 -0
  209. package/src/utils/auth.ts +13 -0
  210. package/src/utils/autoCompactCore.ts +223 -0
  211. package/src/utils/autoUpdater.ts +318 -0
  212. package/src/utils/betas.ts +20 -0
  213. package/src/utils/browser.ts +14 -0
  214. package/src/utils/cleanup.ts +72 -0
  215. package/src/utils/commands.ts +261 -0
  216. package/src/utils/config.ts +771 -0
  217. package/src/utils/conversationRecovery.ts +54 -0
  218. package/src/utils/debugLogger.ts +1123 -0
  219. package/src/utils/diff.ts +42 -0
  220. package/src/utils/env.ts +57 -0
  221. package/src/utils/errors.ts +21 -0
  222. package/src/utils/exampleCommands.ts +108 -0
  223. package/src/utils/execFileNoThrow.ts +51 -0
  224. package/src/utils/expertChatStorage.ts +136 -0
  225. package/src/utils/file.ts +402 -0
  226. package/src/utils/fileRecoveryCore.ts +71 -0
  227. package/src/utils/format.tsx +44 -0
  228. package/src/utils/generators.ts +62 -0
  229. package/src/utils/git.ts +92 -0
  230. package/src/utils/globalLogger.ts +77 -0
  231. package/src/utils/http.ts +10 -0
  232. package/src/utils/imagePaste.ts +38 -0
  233. package/src/utils/json.ts +13 -0
  234. package/src/utils/log.ts +382 -0
  235. package/src/utils/markdown.ts +213 -0
  236. package/src/utils/messageContextManager.ts +289 -0
  237. package/src/utils/messages.tsx +938 -0
  238. package/src/utils/model.ts +836 -0
  239. package/src/utils/permissions/filesystem.ts +118 -0
  240. package/src/utils/ripgrep.ts +167 -0
  241. package/src/utils/sessionState.ts +49 -0
  242. package/src/utils/state.ts +25 -0
  243. package/src/utils/style.ts +29 -0
  244. package/src/utils/terminal.ts +49 -0
  245. package/src/utils/theme.ts +122 -0
  246. package/src/utils/thinking.ts +144 -0
  247. package/src/utils/todoStorage.ts +431 -0
  248. package/src/utils/tokens.ts +43 -0
  249. package/src/utils/toolExecutionController.ts +163 -0
  250. package/src/utils/unaryLogging.ts +26 -0
  251. package/src/utils/user.ts +37 -0
  252. package/src/utils/validate.ts +165 -0
  253. package/cli.mjs +0 -1803
@@ -0,0 +1,42 @@
1
+ import { type Hunk, structuredPatch } from 'diff'
2
+
3
+ const CONTEXT_LINES = 3
4
+
5
+ // For some reason, & confuses the diff library, so we replace it with a token,
6
+ // then substitute it back in after the diff is computed.
7
+ const AMPERSAND_TOKEN = '<<:AMPERSAND_TOKEN:>>'
8
+
9
+ const DOLLAR_TOKEN = '<<:DOLLAR_TOKEN:>>'
10
+
11
+ export function getPatch({
12
+ filePath,
13
+ fileContents,
14
+ oldStr,
15
+ newStr,
16
+ }: {
17
+ filePath: string
18
+ fileContents: string
19
+ oldStr: string
20
+ newStr: string
21
+ }): Hunk[] {
22
+ return structuredPatch(
23
+ filePath,
24
+ filePath,
25
+ fileContents.replaceAll('&', AMPERSAND_TOKEN).replaceAll('$', DOLLAR_TOKEN),
26
+ fileContents
27
+ .replaceAll('&', AMPERSAND_TOKEN)
28
+ .replaceAll('$', DOLLAR_TOKEN)
29
+ .replace(
30
+ oldStr.replaceAll('&', AMPERSAND_TOKEN).replaceAll('$', DOLLAR_TOKEN),
31
+ newStr.replaceAll('&', AMPERSAND_TOKEN).replaceAll('$', DOLLAR_TOKEN),
32
+ ),
33
+ undefined,
34
+ undefined,
35
+ { context: CONTEXT_LINES },
36
+ ).hunks.map(_ => ({
37
+ ..._,
38
+ lines: _.lines.map(_ =>
39
+ _.replaceAll(AMPERSAND_TOKEN, '&').replaceAll(DOLLAR_TOKEN, '$'),
40
+ ),
41
+ }))
42
+ }
@@ -0,0 +1,57 @@
1
+ import { execFileNoThrow } from './execFileNoThrow'
2
+ import { memoize } from 'lodash-es'
3
+ import { join } from 'path'
4
+ import { homedir } from 'os'
5
+ import { CONFIG_BASE_DIR, CONFIG_FILE } from '../constants/product'
6
+ // Base directory for all Any kode data files (except config.json for backwards compatibility)
7
+ // Support both KODE_CONFIG_DIR and CLAUDE_CONFIG_DIR for compatibility
8
+ export const CLAUDE_BASE_DIR =
9
+ process.env.KODE_CONFIG_DIR ?? process.env.CLAUDE_CONFIG_DIR ?? join(homedir(), CONFIG_BASE_DIR)
10
+
11
+ // Config and data paths
12
+ // Support both KODE_CONFIG_DIR and CLAUDE_CONFIG_DIR environment variables
13
+ export const GLOBAL_CLAUDE_FILE = (process.env.KODE_CONFIG_DIR || process.env.CLAUDE_CONFIG_DIR)
14
+ ? join(CLAUDE_BASE_DIR, 'config.json')
15
+ : join(homedir(), CONFIG_FILE)
16
+ export const MEMORY_DIR = join(CLAUDE_BASE_DIR, 'memory')
17
+
18
+ const getIsDocker = memoize(async (): Promise<boolean> => {
19
+ // Check for .dockerenv file
20
+ const { code } = await execFileNoThrow('test', ['-f', '/.dockerenv'])
21
+ if (code !== 0) {
22
+ return false
23
+ }
24
+ return process.platform === 'linux'
25
+ })
26
+
27
+ const hasInternetAccess = memoize(async (): Promise<boolean> => {
28
+ try {
29
+ const controller = new AbortController()
30
+ const timeout = setTimeout(() => controller.abort(), 1000)
31
+
32
+ await fetch('http://1.1.1.1', {
33
+ method: 'HEAD',
34
+ signal: controller.signal,
35
+ })
36
+
37
+ clearTimeout(timeout)
38
+ return true
39
+ } catch {
40
+ return false
41
+ }
42
+ })
43
+
44
+ // all of these should be immutable
45
+ export const env = {
46
+ getIsDocker,
47
+ hasInternetAccess,
48
+ isCI: Boolean(process.env.CI),
49
+ platform:
50
+ process.platform === 'win32'
51
+ ? 'windows'
52
+ : process.platform === 'darwin'
53
+ ? 'macos'
54
+ : 'linux',
55
+ nodeVersion: process.version,
56
+ terminal: process.env.TERM_PROGRAM,
57
+ }
@@ -0,0 +1,21 @@
1
+ export class MalformedCommandError extends TypeError {}
2
+
3
+ export class DeprecatedCommandError extends Error {}
4
+
5
+ export class AbortError extends Error {}
6
+
7
+ /**
8
+ * Custom error class for configuration file parsing errors
9
+ * Includes the file path and the default configuration that should be used
10
+ */
11
+ export class ConfigParseError extends Error {
12
+ filePath: string
13
+ defaultConfig: unknown
14
+
15
+ constructor(message: string, filePath: string, defaultConfig: unknown) {
16
+ super(message)
17
+ this.name = 'ConfigParseError'
18
+ this.filePath = filePath
19
+ this.defaultConfig = defaultConfig
20
+ }
21
+ }
@@ -0,0 +1,108 @@
1
+ import {
2
+ getGlobalConfig,
3
+ saveGlobalConfig,
4
+ getCurrentProjectConfig,
5
+ saveCurrentProjectConfig,
6
+ } from './config.js'
7
+ import { env } from './env'
8
+ import { getCwd } from './state'
9
+ import { exec } from 'child_process'
10
+ import { logError } from './log'
11
+ import { memoize, sample } from 'lodash-es'
12
+ import { promisify } from 'util'
13
+ import { getIsGit } from './git'
14
+
15
+ const execPromise = promisify(exec)
16
+
17
+ async function getFrequentlyModifiedFiles(): Promise<string[]> {
18
+ if (process.env.NODE_ENV === 'test') return []
19
+ if (env.platform === 'windows') return []
20
+ if (!(await getIsGit())) return []
21
+
22
+ try {
23
+ let filenames = ''
24
+ // Look up files modified by the user's recent commits
25
+ // Be careful to do it async, so it doesn't block the main thread
26
+ const { stdout: userFilenames } = await execPromise(
27
+ 'git log -n 1000 --pretty=format: --name-only --diff-filter=M --author=$(git config user.email) | sort | uniq -c | sort -nr | head -n 20',
28
+ { cwd: getCwd(), encoding: 'utf8' },
29
+ )
30
+
31
+ filenames = 'Files modified by user:\n' + userFilenames
32
+
33
+ // Look at other users' commits if we don't have enough files
34
+ if (userFilenames.split('\n').length < 10) {
35
+ const { stdout: allFilenames } = await execPromise(
36
+ 'git log -n 1000 --pretty=format: --name-only --diff-filter=M | sort | uniq -c | sort -nr | head -n 20',
37
+ { cwd: getCwd(), encoding: 'utf8' },
38
+ )
39
+ filenames += '\n\nFiles modified by other users:\n' + allFilenames
40
+ }
41
+ const response = await queryQuick({
42
+ systemPrompt: [
43
+ "You are an expert at analyzing git history. Given a list of files and their modification counts, return exactly five filenames that are frequently modified and represent core application logic (not auto-generated files, dependencies, or configuration). Make sure filenames are diverse, not all in the same folder, and are a mix of user and other users. Return only the filenames' basenames (without the path) separated by newlines with no explanation.",
44
+ ],
45
+ userPrompt: filenames,
46
+ })
47
+
48
+ const content = response.message.content[0]
49
+ if (!content || content.type !== 'text') return []
50
+ const chosenFilenames = content.text.trim().split('\n')
51
+ if (chosenFilenames.length < 5) {
52
+ // Likely error
53
+ return []
54
+ }
55
+ return chosenFilenames
56
+ } catch (err) {
57
+ logError(err)
58
+ return []
59
+ }
60
+ }
61
+
62
+ export const getExampleCommands = memoize(async (): Promise<string[]> => {
63
+ const globalConfig = getGlobalConfig()
64
+ const projectConfig = getCurrentProjectConfig()
65
+ const now = Date.now()
66
+ const lastGenerated = projectConfig.exampleFilesGeneratedAt ?? 0
67
+ const oneWeek = 7 * 24 * 60 * 60 * 1000
68
+
69
+ // Regenerate examples if they're over a week old
70
+ if (now - lastGenerated > oneWeek) {
71
+ projectConfig.exampleFiles = []
72
+ }
73
+
74
+ // Update global startup count
75
+ const newGlobalConfig = {
76
+ ...globalConfig,
77
+ numStartups: (globalConfig.numStartups ?? 0) + 1,
78
+ }
79
+ saveGlobalConfig(newGlobalConfig)
80
+
81
+ // // If no example files cached, kickstart fetch in background
82
+ // if (!projectConfig.exampleFiles?.length) {
83
+ // getFrequentlyModifiedFiles().then(files => {
84
+ // if (files.length) {
85
+ // saveCurrentProjectConfig({
86
+ // ...getCurrentProjectConfig(),
87
+ // exampleFiles: files,
88
+ // exampleFilesGeneratedAt: Date.now(),
89
+ // })
90
+ // }
91
+ // })
92
+ // }
93
+
94
+ const frequentFile = projectConfig.exampleFiles?.length
95
+ ? sample(projectConfig.exampleFiles)
96
+ : '<filepath>'
97
+
98
+ return [
99
+ 'fix lint errors',
100
+ 'fix typecheck errors',
101
+ `how does ${frequentFile} work?`,
102
+ `refactor ${frequentFile}`,
103
+ 'how do I log an error?',
104
+ `edit ${frequentFile} to...`,
105
+ `write a test for ${frequentFile}`,
106
+ 'create a util logging.py that...',
107
+ ]
108
+ })
@@ -0,0 +1,51 @@
1
+ import { execFile } from 'child_process'
2
+ import { getCwd } from './state'
3
+ import { logError } from './log'
4
+
5
+ const MS_IN_SECOND = 1000
6
+ const SECONDS_IN_MINUTE = 60
7
+
8
+ /**
9
+ * execFile, but always resolves (never throws)
10
+ */
11
+ export function execFileNoThrow(
12
+ file: string,
13
+ args: string[],
14
+ abortSignal?: AbortSignal,
15
+ timeout = 10 * SECONDS_IN_MINUTE * MS_IN_SECOND,
16
+ preserveOutputOnError = true,
17
+ ): Promise<{ stdout: string; stderr: string; code: number }> {
18
+ return new Promise(resolve => {
19
+ try {
20
+ execFile(
21
+ file,
22
+ args,
23
+ {
24
+ maxBuffer: 1_000_000,
25
+ signal: abortSignal,
26
+ timeout,
27
+ cwd: getCwd(),
28
+ },
29
+ (error, stdout, stderr) => {
30
+ if (error) {
31
+ if (preserveOutputOnError) {
32
+ const errorCode = typeof error.code === 'number' ? error.code : 1
33
+ resolve({
34
+ stdout: stdout || '',
35
+ stderr: stderr || '',
36
+ code: errorCode,
37
+ })
38
+ } else {
39
+ resolve({ stdout: '', stderr: '', code: 1 })
40
+ }
41
+ } else {
42
+ resolve({ stdout, stderr, code: 0 })
43
+ }
44
+ },
45
+ )
46
+ } catch (error) {
47
+ logError(error)
48
+ resolve({ stdout: '', stderr: '', code: 1 })
49
+ }
50
+ })
51
+ }
@@ -0,0 +1,136 @@
1
+ import { existsSync, readFileSync, writeFileSync, mkdirSync } from 'fs'
2
+ import { join } from 'path'
3
+ import { homedir } from 'os'
4
+ import { randomUUID } from 'crypto'
5
+
6
+ /**
7
+ * Expert Chat Session Storage - 极简版
8
+ * 存储符合OpenAI格式的messages历史
9
+ */
10
+
11
+ export interface ChatMessage {
12
+ role: 'user' | 'assistant'
13
+ content: string
14
+ }
15
+
16
+ export interface ExpertChatSession {
17
+ sessionId: string
18
+ expertModel: string
19
+ messages: ChatMessage[]
20
+ createdAt: number
21
+ lastUpdated: number
22
+ }
23
+
24
+ /**
25
+ * 获取专家聊天存储目录
26
+ */
27
+ function getExpertChatDirectory(): string {
28
+ const configDir =
29
+ process.env.KODE_CONFIG_DIR ?? process.env.ANYKODE_CONFIG_DIR ?? join(homedir(), '.kode')
30
+ const expertChatDir = join(configDir, 'expert-chats')
31
+
32
+ if (!existsSync(expertChatDir)) {
33
+ mkdirSync(expertChatDir, { recursive: true })
34
+ }
35
+
36
+ return expertChatDir
37
+ }
38
+
39
+ /**
40
+ * 获取会话文件路径 - 使用 sessionId.json 格式
41
+ */
42
+ function getSessionFilePath(sessionId: string): string {
43
+ return join(getExpertChatDirectory(), `${sessionId}.json`)
44
+ }
45
+
46
+ /**
47
+ * 创建新的专家聊天会话
48
+ */
49
+ export function createExpertChatSession(
50
+ expertModel: string,
51
+ ): ExpertChatSession {
52
+ const sessionId = randomUUID().slice(0, 5)
53
+ const session: ExpertChatSession = {
54
+ sessionId,
55
+ expertModel,
56
+ messages: [],
57
+ createdAt: Date.now(),
58
+ lastUpdated: Date.now(),
59
+ }
60
+
61
+ saveExpertChatSession(session)
62
+ return session
63
+ }
64
+
65
+ /**
66
+ * 加载现有专家聊天会话
67
+ */
68
+ export function loadExpertChatSession(
69
+ sessionId: string,
70
+ ): ExpertChatSession | null {
71
+ const filePath = getSessionFilePath(sessionId)
72
+
73
+ if (!existsSync(filePath)) {
74
+ return null
75
+ }
76
+
77
+ try {
78
+ const content = readFileSync(filePath, 'utf-8')
79
+ return JSON.parse(content) as ExpertChatSession
80
+ } catch (error) {
81
+ console.error(`Failed to load expert chat session ${sessionId}:`, error)
82
+ return null
83
+ }
84
+ }
85
+
86
+ /**
87
+ * 保存专家聊天会话
88
+ */
89
+ export function saveExpertChatSession(session: ExpertChatSession): void {
90
+ const filePath = getSessionFilePath(session.sessionId)
91
+
92
+ try {
93
+ session.lastUpdated = Date.now()
94
+ writeFileSync(filePath, JSON.stringify(session, null, 2), 'utf-8')
95
+ } catch (error) {
96
+ console.error(
97
+ `Failed to save expert chat session ${session.sessionId}:`,
98
+ error,
99
+ )
100
+ throw error
101
+ }
102
+ }
103
+
104
+ /**
105
+ * 添加消息到会话
106
+ */
107
+ export function addMessageToSession(
108
+ sessionId: string,
109
+ role: 'user' | 'assistant',
110
+ content: string,
111
+ ): ExpertChatSession | null {
112
+ const session = loadExpertChatSession(sessionId)
113
+ if (!session) {
114
+ return null
115
+ }
116
+
117
+ session.messages.push({ role, content })
118
+ saveExpertChatSession(session)
119
+
120
+ return session
121
+ }
122
+
123
+ /**
124
+ * 获取会话的消息历史 - 返回OpenAI格式
125
+ */
126
+ export function getSessionMessages(sessionId: string): ChatMessage[] {
127
+ const session = loadExpertChatSession(sessionId)
128
+ return session?.messages || []
129
+ }
130
+
131
+ /**
132
+ * 生成新的会话ID
133
+ */
134
+ export function generateSessionId(): string {
135
+ return randomUUID().slice(0, 5)
136
+ }