@shareai-lab/kode 1.0.9

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 (286) hide show
  1. package/LICENSE +201 -0
  2. package/README.md +426 -0
  3. package/README.zh-CN.md +326 -0
  4. package/cli.js +79 -0
  5. package/package.json +119 -0
  6. package/scripts/postinstall.js +18 -0
  7. package/src/ProjectOnboarding.tsx +198 -0
  8. package/src/Tool.ts +82 -0
  9. package/src/commands/agents.tsx +3410 -0
  10. package/src/commands/approvedTools.ts +53 -0
  11. package/src/commands/bug.tsx +20 -0
  12. package/src/commands/clear.ts +43 -0
  13. package/src/commands/compact.ts +120 -0
  14. package/src/commands/config.tsx +19 -0
  15. package/src/commands/cost.ts +18 -0
  16. package/src/commands/ctx_viz.ts +209 -0
  17. package/src/commands/doctor.ts +24 -0
  18. package/src/commands/help.tsx +19 -0
  19. package/src/commands/init.ts +37 -0
  20. package/src/commands/listen.ts +42 -0
  21. package/src/commands/login.tsx +51 -0
  22. package/src/commands/logout.tsx +40 -0
  23. package/src/commands/mcp.ts +41 -0
  24. package/src/commands/model.tsx +40 -0
  25. package/src/commands/modelstatus.tsx +20 -0
  26. package/src/commands/onboarding.tsx +34 -0
  27. package/src/commands/pr_comments.ts +59 -0
  28. package/src/commands/refreshCommands.ts +54 -0
  29. package/src/commands/release-notes.ts +34 -0
  30. package/src/commands/resume.tsx +31 -0
  31. package/src/commands/review.ts +49 -0
  32. package/src/commands/terminalSetup.ts +221 -0
  33. package/src/commands.ts +139 -0
  34. package/src/components/ApproveApiKey.tsx +93 -0
  35. package/src/components/AsciiLogo.tsx +13 -0
  36. package/src/components/AutoUpdater.tsx +148 -0
  37. package/src/components/Bug.tsx +367 -0
  38. package/src/components/Config.tsx +293 -0
  39. package/src/components/ConsoleOAuthFlow.tsx +327 -0
  40. package/src/components/Cost.tsx +23 -0
  41. package/src/components/CostThresholdDialog.tsx +46 -0
  42. package/src/components/CustomSelect/option-map.ts +42 -0
  43. package/src/components/CustomSelect/select-option.tsx +78 -0
  44. package/src/components/CustomSelect/select.tsx +152 -0
  45. package/src/components/CustomSelect/theme.ts +45 -0
  46. package/src/components/CustomSelect/use-select-state.ts +414 -0
  47. package/src/components/CustomSelect/use-select.ts +35 -0
  48. package/src/components/FallbackToolUseRejectedMessage.tsx +15 -0
  49. package/src/components/FileEditToolUpdatedMessage.tsx +66 -0
  50. package/src/components/Help.tsx +215 -0
  51. package/src/components/HighlightedCode.tsx +33 -0
  52. package/src/components/InvalidConfigDialog.tsx +113 -0
  53. package/src/components/Link.tsx +32 -0
  54. package/src/components/LogSelector.tsx +86 -0
  55. package/src/components/Logo.tsx +170 -0
  56. package/src/components/MCPServerApprovalDialog.tsx +100 -0
  57. package/src/components/MCPServerDialogCopy.tsx +25 -0
  58. package/src/components/MCPServerMultiselectDialog.tsx +109 -0
  59. package/src/components/Message.tsx +221 -0
  60. package/src/components/MessageResponse.tsx +15 -0
  61. package/src/components/MessageSelector.tsx +211 -0
  62. package/src/components/ModeIndicator.tsx +88 -0
  63. package/src/components/ModelConfig.tsx +301 -0
  64. package/src/components/ModelListManager.tsx +227 -0
  65. package/src/components/ModelSelector.tsx +3387 -0
  66. package/src/components/ModelStatusDisplay.tsx +230 -0
  67. package/src/components/Onboarding.tsx +274 -0
  68. package/src/components/PressEnterToContinue.tsx +11 -0
  69. package/src/components/PromptInput.tsx +760 -0
  70. package/src/components/SentryErrorBoundary.ts +39 -0
  71. package/src/components/Spinner.tsx +129 -0
  72. package/src/components/StickerRequestForm.tsx +16 -0
  73. package/src/components/StructuredDiff.tsx +191 -0
  74. package/src/components/TextInput.tsx +259 -0
  75. package/src/components/TodoItem.tsx +47 -0
  76. package/src/components/TokenWarning.tsx +31 -0
  77. package/src/components/ToolUseLoader.tsx +40 -0
  78. package/src/components/TrustDialog.tsx +106 -0
  79. package/src/components/binary-feedback/BinaryFeedback.tsx +63 -0
  80. package/src/components/binary-feedback/BinaryFeedbackOption.tsx +111 -0
  81. package/src/components/binary-feedback/BinaryFeedbackView.tsx +172 -0
  82. package/src/components/binary-feedback/utils.ts +220 -0
  83. package/src/components/messages/AssistantBashOutputMessage.tsx +22 -0
  84. package/src/components/messages/AssistantLocalCommandOutputMessage.tsx +49 -0
  85. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +19 -0
  86. package/src/components/messages/AssistantTextMessage.tsx +144 -0
  87. package/src/components/messages/AssistantThinkingMessage.tsx +40 -0
  88. package/src/components/messages/AssistantToolUseMessage.tsx +133 -0
  89. package/src/components/messages/TaskProgressMessage.tsx +32 -0
  90. package/src/components/messages/TaskToolMessage.tsx +58 -0
  91. package/src/components/messages/UserBashInputMessage.tsx +28 -0
  92. package/src/components/messages/UserCommandMessage.tsx +30 -0
  93. package/src/components/messages/UserKodingInputMessage.tsx +28 -0
  94. package/src/components/messages/UserPromptMessage.tsx +35 -0
  95. package/src/components/messages/UserTextMessage.tsx +39 -0
  96. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +12 -0
  97. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +36 -0
  98. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +31 -0
  99. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +57 -0
  100. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +35 -0
  101. package/src/components/messages/UserToolResultMessage/utils.tsx +56 -0
  102. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +121 -0
  103. package/src/components/permissions/FallbackPermissionRequest.tsx +153 -0
  104. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +182 -0
  105. package/src/components/permissions/FileEditPermissionRequest/FileEditToolDiff.tsx +77 -0
  106. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +164 -0
  107. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +83 -0
  108. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +240 -0
  109. package/src/components/permissions/PermissionRequest.tsx +101 -0
  110. package/src/components/permissions/PermissionRequestTitle.tsx +69 -0
  111. package/src/components/permissions/hooks.ts +44 -0
  112. package/src/components/permissions/toolUseOptions.ts +59 -0
  113. package/src/components/permissions/utils.ts +23 -0
  114. package/src/constants/betas.ts +5 -0
  115. package/src/constants/claude-asterisk-ascii-art.tsx +238 -0
  116. package/src/constants/figures.ts +4 -0
  117. package/src/constants/keys.ts +3 -0
  118. package/src/constants/macros.ts +8 -0
  119. package/src/constants/modelCapabilities.ts +179 -0
  120. package/src/constants/models.ts +1025 -0
  121. package/src/constants/oauth.ts +18 -0
  122. package/src/constants/product.ts +17 -0
  123. package/src/constants/prompts.ts +168 -0
  124. package/src/constants/releaseNotes.ts +7 -0
  125. package/src/context/PermissionContext.tsx +149 -0
  126. package/src/context.ts +278 -0
  127. package/src/cost-tracker.ts +84 -0
  128. package/src/entrypoints/cli.tsx +1542 -0
  129. package/src/entrypoints/mcp.ts +176 -0
  130. package/src/history.ts +25 -0
  131. package/src/hooks/useApiKeyVerification.ts +59 -0
  132. package/src/hooks/useArrowKeyHistory.ts +55 -0
  133. package/src/hooks/useCanUseTool.ts +138 -0
  134. package/src/hooks/useCancelRequest.ts +39 -0
  135. package/src/hooks/useDoublePress.ts +42 -0
  136. package/src/hooks/useExitOnCtrlCD.ts +31 -0
  137. package/src/hooks/useInterval.ts +25 -0
  138. package/src/hooks/useLogMessages.ts +16 -0
  139. package/src/hooks/useLogStartupTime.ts +12 -0
  140. package/src/hooks/useNotifyAfterTimeout.ts +65 -0
  141. package/src/hooks/usePermissionRequestLogging.ts +44 -0
  142. package/src/hooks/useTerminalSize.ts +49 -0
  143. package/src/hooks/useTextInput.ts +318 -0
  144. package/src/hooks/useUnifiedCompletion.ts +1405 -0
  145. package/src/messages.ts +38 -0
  146. package/src/permissions.ts +268 -0
  147. package/src/query.ts +715 -0
  148. package/src/screens/ConfigureNpmPrefix.tsx +197 -0
  149. package/src/screens/Doctor.tsx +219 -0
  150. package/src/screens/LogList.tsx +68 -0
  151. package/src/screens/REPL.tsx +809 -0
  152. package/src/screens/ResumeConversation.tsx +68 -0
  153. package/src/services/adapters/base.ts +38 -0
  154. package/src/services/adapters/chatCompletions.ts +90 -0
  155. package/src/services/adapters/responsesAPI.ts +170 -0
  156. package/src/services/browserMocks.ts +66 -0
  157. package/src/services/claude.ts +2197 -0
  158. package/src/services/customCommands.ts +704 -0
  159. package/src/services/fileFreshness.ts +377 -0
  160. package/src/services/gpt5ConnectionTest.ts +340 -0
  161. package/src/services/mcpClient.ts +564 -0
  162. package/src/services/mcpServerApproval.tsx +50 -0
  163. package/src/services/mentionProcessor.ts +273 -0
  164. package/src/services/modelAdapterFactory.ts +69 -0
  165. package/src/services/notifier.ts +40 -0
  166. package/src/services/oauth.ts +357 -0
  167. package/src/services/openai.ts +1338 -0
  168. package/src/services/responseStateManager.ts +90 -0
  169. package/src/services/sentry.ts +3 -0
  170. package/src/services/statsig.ts +172 -0
  171. package/src/services/statsigStorage.ts +86 -0
  172. package/src/services/systemReminder.ts +507 -0
  173. package/src/services/vcr.ts +161 -0
  174. package/src/test/testAdapters.ts +96 -0
  175. package/src/tools/ArchitectTool/ArchitectTool.tsx +122 -0
  176. package/src/tools/ArchitectTool/prompt.ts +15 -0
  177. package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +569 -0
  178. package/src/tools/BashTool/BashTool.tsx +243 -0
  179. package/src/tools/BashTool/BashToolResultMessage.tsx +38 -0
  180. package/src/tools/BashTool/OutputLine.tsx +49 -0
  181. package/src/tools/BashTool/prompt.ts +174 -0
  182. package/src/tools/BashTool/utils.ts +56 -0
  183. package/src/tools/FileEditTool/FileEditTool.tsx +315 -0
  184. package/src/tools/FileEditTool/prompt.ts +51 -0
  185. package/src/tools/FileEditTool/utils.ts +58 -0
  186. package/src/tools/FileReadTool/FileReadTool.tsx +404 -0
  187. package/src/tools/FileReadTool/prompt.ts +7 -0
  188. package/src/tools/FileWriteTool/FileWriteTool.tsx +297 -0
  189. package/src/tools/FileWriteTool/prompt.ts +10 -0
  190. package/src/tools/GlobTool/GlobTool.tsx +119 -0
  191. package/src/tools/GlobTool/prompt.ts +8 -0
  192. package/src/tools/GrepTool/GrepTool.tsx +147 -0
  193. package/src/tools/GrepTool/prompt.ts +11 -0
  194. package/src/tools/MCPTool/MCPTool.tsx +107 -0
  195. package/src/tools/MCPTool/prompt.ts +3 -0
  196. package/src/tools/MemoryReadTool/MemoryReadTool.tsx +127 -0
  197. package/src/tools/MemoryReadTool/prompt.ts +3 -0
  198. package/src/tools/MemoryWriteTool/MemoryWriteTool.tsx +89 -0
  199. package/src/tools/MemoryWriteTool/prompt.ts +3 -0
  200. package/src/tools/MultiEditTool/MultiEditTool.tsx +366 -0
  201. package/src/tools/MultiEditTool/prompt.ts +45 -0
  202. package/src/tools/NotebookEditTool/NotebookEditTool.tsx +298 -0
  203. package/src/tools/NotebookEditTool/prompt.ts +3 -0
  204. package/src/tools/NotebookReadTool/NotebookReadTool.tsx +258 -0
  205. package/src/tools/NotebookReadTool/prompt.ts +3 -0
  206. package/src/tools/StickerRequestTool/StickerRequestTool.tsx +93 -0
  207. package/src/tools/StickerRequestTool/prompt.ts +19 -0
  208. package/src/tools/TaskTool/TaskTool.tsx +466 -0
  209. package/src/tools/TaskTool/constants.ts +1 -0
  210. package/src/tools/TaskTool/prompt.ts +92 -0
  211. package/src/tools/ThinkTool/ThinkTool.tsx +54 -0
  212. package/src/tools/ThinkTool/prompt.ts +12 -0
  213. package/src/tools/TodoWriteTool/TodoWriteTool.tsx +313 -0
  214. package/src/tools/TodoWriteTool/prompt.ts +63 -0
  215. package/src/tools/URLFetcherTool/URLFetcherTool.tsx +178 -0
  216. package/src/tools/URLFetcherTool/cache.ts +55 -0
  217. package/src/tools/URLFetcherTool/htmlToMarkdown.ts +55 -0
  218. package/src/tools/URLFetcherTool/prompt.ts +17 -0
  219. package/src/tools/WebSearchTool/WebSearchTool.tsx +103 -0
  220. package/src/tools/WebSearchTool/prompt.ts +13 -0
  221. package/src/tools/WebSearchTool/searchProviders.ts +66 -0
  222. package/src/tools/lsTool/lsTool.tsx +272 -0
  223. package/src/tools/lsTool/prompt.ts +2 -0
  224. package/src/tools.ts +67 -0
  225. package/src/types/PermissionMode.ts +120 -0
  226. package/src/types/RequestContext.ts +72 -0
  227. package/src/types/conversation.ts +51 -0
  228. package/src/types/logs.ts +58 -0
  229. package/src/types/modelCapabilities.ts +64 -0
  230. package/src/types/notebook.ts +87 -0
  231. package/src/utils/Cursor.ts +436 -0
  232. package/src/utils/PersistentShell.ts +552 -0
  233. package/src/utils/advancedFuzzyMatcher.ts +290 -0
  234. package/src/utils/agentLoader.ts +278 -0
  235. package/src/utils/agentStorage.ts +97 -0
  236. package/src/utils/array.ts +3 -0
  237. package/src/utils/ask.tsx +99 -0
  238. package/src/utils/auth.ts +13 -0
  239. package/src/utils/autoCompactCore.ts +223 -0
  240. package/src/utils/autoUpdater.ts +458 -0
  241. package/src/utils/betas.ts +20 -0
  242. package/src/utils/browser.ts +14 -0
  243. package/src/utils/cleanup.ts +72 -0
  244. package/src/utils/commands.ts +261 -0
  245. package/src/utils/commonUnixCommands.ts +161 -0
  246. package/src/utils/config.ts +945 -0
  247. package/src/utils/conversationRecovery.ts +55 -0
  248. package/src/utils/debugLogger.ts +1235 -0
  249. package/src/utils/diff.ts +42 -0
  250. package/src/utils/env.ts +57 -0
  251. package/src/utils/errors.ts +21 -0
  252. package/src/utils/exampleCommands.ts +109 -0
  253. package/src/utils/execFileNoThrow.ts +51 -0
  254. package/src/utils/expertChatStorage.ts +136 -0
  255. package/src/utils/file.ts +405 -0
  256. package/src/utils/fileRecoveryCore.ts +71 -0
  257. package/src/utils/format.tsx +44 -0
  258. package/src/utils/fuzzyMatcher.ts +328 -0
  259. package/src/utils/generators.ts +62 -0
  260. package/src/utils/git.ts +92 -0
  261. package/src/utils/globalLogger.ts +77 -0
  262. package/src/utils/http.ts +10 -0
  263. package/src/utils/imagePaste.ts +38 -0
  264. package/src/utils/json.ts +13 -0
  265. package/src/utils/log.ts +382 -0
  266. package/src/utils/markdown.ts +213 -0
  267. package/src/utils/messageContextManager.ts +289 -0
  268. package/src/utils/messages.tsx +939 -0
  269. package/src/utils/model.ts +914 -0
  270. package/src/utils/permissions/filesystem.ts +127 -0
  271. package/src/utils/responseState.ts +23 -0
  272. package/src/utils/ripgrep.ts +167 -0
  273. package/src/utils/secureFile.ts +564 -0
  274. package/src/utils/sessionState.ts +49 -0
  275. package/src/utils/state.ts +25 -0
  276. package/src/utils/style.ts +29 -0
  277. package/src/utils/terminal.ts +50 -0
  278. package/src/utils/theme.ts +127 -0
  279. package/src/utils/thinking.ts +144 -0
  280. package/src/utils/todoStorage.ts +431 -0
  281. package/src/utils/tokens.ts +43 -0
  282. package/src/utils/toolExecutionController.ts +163 -0
  283. package/src/utils/unaryLogging.ts +26 -0
  284. package/src/utils/user.ts +37 -0
  285. package/src/utils/validate.ts +165 -0
  286. package/yoga.wasm +0 -0
@@ -0,0 +1,127 @@
1
+ import { isAbsolute, resolve, relative, sep } from 'path'
2
+ import { getCwd, getOriginalCwd } from '../state'
3
+
4
+ // In-memory storage for file permissions that resets each session
5
+ // Sets of allowed directories for read and write operations
6
+ const readFileAllowedDirectories: Set<string> = new Set()
7
+ const writeFileAllowedDirectories: Set<string> = new Set()
8
+
9
+ /**
10
+ * Ensures a path is absolute by resolving it relative to cwd if necessary
11
+ * @param path The path to normalize
12
+ * @returns Absolute path
13
+ */
14
+ export function toAbsolutePath(path: string): string {
15
+ const abs = isAbsolute(path) ? resolve(path) : resolve(getCwd(), path)
16
+ return normalizeForCompare(abs)
17
+ }
18
+
19
+ function normalizeForCompare(p: string): string {
20
+ // Normalize separators and resolve .. and . segments
21
+ const norm = resolve(p)
22
+ // On Windows, comparisons should be case-insensitive
23
+ return process.platform === 'win32' ? norm.toLowerCase() : norm
24
+ }
25
+
26
+ function isSubpath(base: string, target: string): boolean {
27
+ const rel = relative(base, target)
28
+ // If different drive letters on Windows, relative returns the target path
29
+ if (!rel || rel === '') return true
30
+ // Not a subpath if it goes up to parent
31
+ if (rel.startsWith('..')) return false
32
+ // Not a subpath if absolute
33
+ if (isAbsolute(rel)) return false
34
+ return true
35
+ }
36
+
37
+ /**
38
+ * Ensures a path is in the original cwd path
39
+ * @param directory The directory path to normalize
40
+ * @returns Absolute path
41
+ */
42
+ export function pathInOriginalCwd(path: string): boolean {
43
+ const absolutePath = toAbsolutePath(path)
44
+ const base = toAbsolutePath(getOriginalCwd())
45
+ return isSubpath(base, absolutePath)
46
+ }
47
+
48
+ /**
49
+ * Check if read permission exists for the specified directory
50
+ * @param directory The directory to check permission for
51
+ * @returns true if read permission exists, false otherwise
52
+ */
53
+ export function hasReadPermission(directory: string): boolean {
54
+ const absolutePath = toAbsolutePath(directory)
55
+ for (const allowedPath of readFileAllowedDirectories) {
56
+ if (isSubpath(allowedPath, absolutePath)) return true
57
+ }
58
+ return false
59
+ }
60
+
61
+ /**
62
+ * Check if write permission exists for the specified directory
63
+ * @param directory The directory to check permission for
64
+ * @returns true if write permission exists, false otherwise
65
+ */
66
+ export function hasWritePermission(directory: string): boolean {
67
+ const absolutePath = toAbsolutePath(directory)
68
+ for (const allowedPath of writeFileAllowedDirectories) {
69
+ if (isSubpath(allowedPath, absolutePath)) return true
70
+ }
71
+ return false
72
+ }
73
+
74
+ /**
75
+ * Save read permission for a directory
76
+ * @param directory The directory to grant read permission for
77
+ */
78
+ function saveReadPermission(directory: string): void {
79
+ const absolutePath = toAbsolutePath(directory)
80
+ // Remove any existing subpaths contained by this new path
81
+ for (const allowedPath of Array.from(readFileAllowedDirectories)) {
82
+ if (isSubpath(absolutePath, allowedPath)) {
83
+ readFileAllowedDirectories.delete(allowedPath)
84
+ }
85
+ }
86
+ readFileAllowedDirectories.add(absolutePath)
87
+ }
88
+
89
+ export const saveReadPermissionForTest = saveReadPermission
90
+
91
+ /**
92
+ * Grants read permission for the original project directory.
93
+ * This is useful for initializing read access to the project root.
94
+ */
95
+ export function grantReadPermissionForOriginalDir(): void {
96
+ const originalProjectDir = getOriginalCwd()
97
+ saveReadPermission(originalProjectDir)
98
+ }
99
+
100
+ /**
101
+ * Save write permission for a directory
102
+ * @param directory The directory to grant write permission for
103
+ */
104
+ function saveWritePermission(directory: string): void {
105
+ const absolutePath = toAbsolutePath(directory)
106
+ for (const allowedPath of Array.from(writeFileAllowedDirectories)) {
107
+ if (isSubpath(absolutePath, allowedPath)) {
108
+ writeFileAllowedDirectories.delete(allowedPath)
109
+ }
110
+ }
111
+ writeFileAllowedDirectories.add(absolutePath)
112
+ }
113
+
114
+ /**
115
+ * Grants write permission for the original project directory.
116
+ * This is useful for initializing write access to the project root.
117
+ */
118
+ export function grantWritePermissionForOriginalDir(): void {
119
+ const originalProjectDir = getOriginalCwd()
120
+ saveWritePermission(originalProjectDir)
121
+ }
122
+
123
+ // For testing purposes
124
+ export function clearFilePermissions(): void {
125
+ readFileAllowedDirectories.clear()
126
+ writeFileAllowedDirectories.clear()
127
+ }
@@ -0,0 +1,23 @@
1
+ /**
2
+ * Response state management for Responses API
3
+ * Tracks previous_response_id for conversation chaining
4
+ */
5
+
6
+ // Store the last response ID for each conversation
7
+ const responseIdCache = new Map<string, string>()
8
+
9
+ export function getLastResponseId(conversationId: string): string | undefined {
10
+ return responseIdCache.get(conversationId)
11
+ }
12
+
13
+ export function setLastResponseId(conversationId: string, responseId: string): void {
14
+ responseIdCache.set(conversationId, responseId)
15
+ }
16
+
17
+ export function clearResponseId(conversationId: string): void {
18
+ responseIdCache.delete(conversationId)
19
+ }
20
+
21
+ export function clearAllResponseIds(): void {
22
+ responseIdCache.clear()
23
+ }
@@ -0,0 +1,167 @@
1
+ import { findActualExecutable } from 'spawn-rx'
2
+ import { memoize } from 'lodash-es'
3
+ import { fileURLToPath, resolve } from 'node:url'
4
+ import * as path from 'path'
5
+ import { logError } from './log'
6
+ import { execFileNoThrow } from './execFileNoThrow'
7
+ import { execFile } from 'child_process'
8
+ import debug from 'debug'
9
+
10
+ const __filename = fileURLToPath(import.meta.url)
11
+ const __dirname = resolve(
12
+ __filename,
13
+ process.env.NODE_ENV === 'test' ? '../..' : '.',
14
+ )
15
+
16
+ const d = debug('claude:ripgrep')
17
+
18
+ const useBuiltinRipgrep = !!process.env.USE_BUILTIN_RIPGREP
19
+ if (useBuiltinRipgrep) {
20
+ d('Using builtin ripgrep because USE_BUILTIN_RIPGREP is set')
21
+ }
22
+
23
+ const ripgrepPath = memoize(() => {
24
+ const { cmd } = findActualExecutable('rg', [])
25
+ d(`ripgrep initially resolved as: ${cmd}`)
26
+
27
+ if (cmd !== 'rg' && !useBuiltinRipgrep) {
28
+ // NB: If we're able to find ripgrep in $PATH, cmd will be an absolute
29
+ // path rather than just returning 'rg'
30
+ return cmd
31
+ } else {
32
+ // Use the one we ship in-box
33
+ const rgRoot = path.resolve(__dirname, 'vendor', 'ripgrep')
34
+ if (process.platform === 'win32') {
35
+ // NB: Ripgrep doesn't ship an aarch64 binary for Windows, boooooo
36
+ return path.resolve(rgRoot, 'x64-win32', 'rg.exe')
37
+ }
38
+
39
+ const ret = path.resolve(
40
+ rgRoot,
41
+ `${process.arch}-${process.platform}`,
42
+ 'rg',
43
+ )
44
+
45
+ d('internal ripgrep resolved as: %s', ret)
46
+ return ret
47
+ }
48
+ })
49
+
50
+ export async function ripGrep(
51
+ args: string[],
52
+ target: string,
53
+ abortSignal: AbortSignal,
54
+ ): Promise<string[]> {
55
+ await codesignRipgrepIfNecessary()
56
+ const rg = ripgrepPath()
57
+ d('ripgrep called: %s %o', rg, target, args)
58
+
59
+ // NB: When running interactively, ripgrep does not require a path as its last
60
+ // argument, but when run non-interactively, it will hang unless a path or file
61
+ // pattern is provided
62
+ return new Promise(resolve => {
63
+ execFile(
64
+ ripgrepPath(),
65
+ [...args, target],
66
+ {
67
+ maxBuffer: 1_000_000,
68
+ signal: abortSignal,
69
+ timeout: 10_000,
70
+ },
71
+ (error, stdout) => {
72
+ if (error) {
73
+ // Exit code 1 from ripgrep means "no matches found" - this is normal
74
+ if (error.code !== 1) {
75
+ d('ripgrep error: %o', error)
76
+ logError(error)
77
+ }
78
+ resolve([])
79
+ } else {
80
+ d('ripgrep succeeded with %s', stdout)
81
+ resolve(stdout.trim().split('\n').filter(Boolean))
82
+ }
83
+ },
84
+ )
85
+ })
86
+ }
87
+
88
+ // NB: We do something tricky here. We know that ripgrep processes common
89
+ // ignore files for us, so we just ripgrep for any character, which matches
90
+ // all non-empty files
91
+ export async function listAllContentFiles(
92
+ path: string,
93
+ abortSignal: AbortSignal,
94
+ limit: number,
95
+ ): Promise<string[]> {
96
+ try {
97
+ d('listAllContentFiles called: %s', path)
98
+ return (await ripGrep(['-l', '.', path], path, abortSignal)).slice(0, limit)
99
+ } catch (e) {
100
+ d('listAllContentFiles failed: %o', e)
101
+
102
+ logError(e)
103
+ return []
104
+ }
105
+ }
106
+
107
+ let alreadyDoneSignCheck = false
108
+ async function codesignRipgrepIfNecessary() {
109
+ if (process.platform !== 'darwin' || alreadyDoneSignCheck) {
110
+ return
111
+ }
112
+
113
+ alreadyDoneSignCheck = true
114
+
115
+ // First, check to see if ripgrep is already signed
116
+ d('checking if ripgrep is already signed')
117
+ const lines = (
118
+ await execFileNoThrow(
119
+ 'codesign',
120
+ ['-vv', '-d', ripgrepPath()],
121
+ undefined,
122
+ undefined,
123
+ false,
124
+ )
125
+ ).stdout.split('\n')
126
+
127
+ const needsSigned = lines.find(line => line.includes('linker-signed'))
128
+ if (!needsSigned) {
129
+ d('seems to be already signed')
130
+ return
131
+ }
132
+
133
+ try {
134
+ d('signing ripgrep')
135
+ const signResult = await execFileNoThrow('codesign', [
136
+ '--sign',
137
+ '-',
138
+ '--force',
139
+ '--preserve-metadata=entitlements,requirements,flags,runtime',
140
+ ripgrepPath(),
141
+ ])
142
+
143
+ if (signResult.code !== 0) {
144
+ d('failed to sign ripgrep: %o', signResult)
145
+ logError(
146
+ `Failed to sign ripgrep: ${signResult.stdout} ${signResult.stderr}`,
147
+ )
148
+ }
149
+
150
+ d('removing quarantine')
151
+ const quarantineResult = await execFileNoThrow('xattr', [
152
+ '-d',
153
+ 'com.apple.quarantine',
154
+ ripgrepPath(),
155
+ ])
156
+
157
+ if (quarantineResult.code !== 0) {
158
+ d('failed to remove quarantine: %o', quarantineResult)
159
+ logError(
160
+ `Failed to remove quarantine: ${quarantineResult.stdout} ${quarantineResult.stderr}`,
161
+ )
162
+ }
163
+ } catch (e) {
164
+ d('failed during sign: %o', e)
165
+ logError(e)
166
+ }
167
+ }