@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,90 @@
1
+ /**
2
+ * GPT-5 Responses API state management
3
+ * Manages previous_response_id for conversation continuity and reasoning context reuse
4
+ */
5
+
6
+ interface ConversationState {
7
+ previousResponseId?: string
8
+ lastUpdate: number
9
+ }
10
+
11
+ class ResponseStateManager {
12
+ private conversationStates = new Map<string, ConversationState>()
13
+
14
+ // Cache cleanup after 1 hour of inactivity
15
+ private readonly CLEANUP_INTERVAL = 60 * 60 * 1000
16
+
17
+ constructor() {
18
+ // Periodic cleanup of stale conversations
19
+ setInterval(() => {
20
+ this.cleanup()
21
+ }, this.CLEANUP_INTERVAL)
22
+ }
23
+
24
+ /**
25
+ * Set the previous response ID for a conversation
26
+ */
27
+ setPreviousResponseId(conversationId: string, responseId: string): void {
28
+ this.conversationStates.set(conversationId, {
29
+ previousResponseId: responseId,
30
+ lastUpdate: Date.now()
31
+ })
32
+ }
33
+
34
+ /**
35
+ * Get the previous response ID for a conversation
36
+ */
37
+ getPreviousResponseId(conversationId: string): string | undefined {
38
+ const state = this.conversationStates.get(conversationId)
39
+ if (state) {
40
+ // Update last access time
41
+ state.lastUpdate = Date.now()
42
+ return state.previousResponseId
43
+ }
44
+ return undefined
45
+ }
46
+
47
+ /**
48
+ * Clear state for a conversation
49
+ */
50
+ clearConversation(conversationId: string): void {
51
+ this.conversationStates.delete(conversationId)
52
+ }
53
+
54
+ /**
55
+ * Clear all conversation states
56
+ */
57
+ clearAll(): void {
58
+ this.conversationStates.clear()
59
+ }
60
+
61
+ /**
62
+ * Clean up stale conversations
63
+ */
64
+ private cleanup(): void {
65
+ const now = Date.now()
66
+ for (const [conversationId, state] of this.conversationStates.entries()) {
67
+ if (now - state.lastUpdate > this.CLEANUP_INTERVAL) {
68
+ this.conversationStates.delete(conversationId)
69
+ }
70
+ }
71
+ }
72
+
73
+ /**
74
+ * Get current state size (for debugging/monitoring)
75
+ */
76
+ getStateSize(): number {
77
+ return this.conversationStates.size
78
+ }
79
+ }
80
+
81
+ // Singleton instance
82
+ export const responseStateManager = new ResponseStateManager()
83
+
84
+ /**
85
+ * Helper to generate conversation ID from context
86
+ */
87
+ export function getConversationId(agentId?: string, messageId?: string): string {
88
+ // Use agentId as primary identifier, fallback to messageId or timestamp
89
+ return agentId || messageId || `conv_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`
90
+ }
@@ -0,0 +1,3 @@
1
+ export function initSentry(): void {}
2
+
3
+ export async function captureException(error: unknown): Promise<void> {}
@@ -0,0 +1,172 @@
1
+ import React from 'react'
2
+ import { memoize } from 'lodash-es'
3
+ import chalk from 'chalk'
4
+ import {
5
+ StatsigClient,
6
+ StatsigOptions,
7
+ StatsigEvent,
8
+ LogLevel,
9
+ } from '@statsig/js-client'
10
+ import './browserMocks.js' // Initialize browser mocks
11
+ import { FileSystemStorageProvider } from './statsigStorage'
12
+ import { STATSIG_CLIENT_KEY } from '../constants/keys'
13
+ import { env } from '../utils/env'
14
+ import { getUser } from '../utils/user'
15
+ import { logError } from '../utils/log'
16
+ import { SESSION_ID } from '../utils/log'
17
+ import { getBetas } from '../utils/betas'
18
+ import { getIsGit } from '../utils/git'
19
+ import { getModelManager } from '../utils/model'
20
+ import { MACRO } from '../constants/macros'
21
+ const gateValues: Record<string, boolean> = {}
22
+ let client: StatsigClient | null = null
23
+
24
+ export const initializeStatsig = memoize(
25
+ async (): Promise<StatsigClient | null> => {
26
+ if (env.isCI || process.env.NODE_ENV === 'test') {
27
+ return null
28
+ }
29
+
30
+ const user = await getUser()
31
+ const options: StatsigOptions = {
32
+ networkConfig: {
33
+ api: 'https://statsig.anthropic.com/v1/',
34
+ },
35
+ environment: {
36
+ tier:
37
+ env.isCI ||
38
+ ['test', 'development'].includes(process.env.NODE_ENV ?? '')
39
+ ? 'dev'
40
+ : 'production',
41
+ },
42
+ logLevel: LogLevel.None,
43
+ storageProvider: new FileSystemStorageProvider(),
44
+ }
45
+
46
+ client = new StatsigClient(STATSIG_CLIENT_KEY, user, options)
47
+ client.on('error', errorEvent => {
48
+ logError(`Statsig error: ${errorEvent}`)
49
+ })
50
+ await client.initializeAsync()
51
+ process.on('exit', () => {
52
+ client?.flush()
53
+ })
54
+ return client
55
+ },
56
+ )
57
+
58
+ export function logEvent(
59
+ eventName: string,
60
+ metadata: { [key: string]: string | undefined },
61
+ ): void {
62
+ // console.log('logEvent', eventName, metadata)
63
+ if (env.isCI || process.env.NODE_ENV === 'test') {
64
+ return
65
+ }
66
+ Promise.all([
67
+ initializeStatsig(),
68
+ getIsGit(),
69
+ getBetas(),
70
+ metadata.model ? Promise.resolve(metadata.model) : Promise.resolve(getModelManager().getModelName('main') || 'unknown'),
71
+ ]).then(([statsigClient, isGit, betas, model]) => {
72
+ if (!statsigClient) return
73
+
74
+ const eventMetadata: Record<string, string> = {
75
+ ...metadata,
76
+ model,
77
+ sessionId: SESSION_ID,
78
+ userType: process.env.USER_TYPE || '',
79
+ ...(process.env.SWE_BENCH_RUN_ID
80
+ ? { sweBenchId: process.env.SWE_BENCH_RUN_ID }
81
+ : {}),
82
+ ...(betas.length > 0 ? { betas: betas.join(',') } : {}),
83
+ env: JSON.stringify({
84
+ isGit,
85
+ platform: env.platform,
86
+ nodeVersion: env.nodeVersion,
87
+ terminal: env.terminal,
88
+ version: MACRO.VERSION,
89
+ }),
90
+ }
91
+
92
+ // Debug logging when debug mode is enabled
93
+ if (process.argv.includes('--debug') || process.argv.includes('-d')) {
94
+ console.log(
95
+ chalk.dim(
96
+ `[DEBUG-ONLY] Statsig event: ${eventName} ${JSON.stringify(metadata, null, 0)}`,
97
+ ),
98
+ )
99
+ }
100
+
101
+ const event: StatsigEvent = {
102
+ eventName,
103
+ metadata: eventMetadata,
104
+ }
105
+ // statsigClient.logEvent(event)
106
+ })
107
+ }
108
+
109
+ export const checkGate = memoize(async (gateName: string): Promise<boolean> => {
110
+ // Default to disabled gates when Statsig is not active
111
+ return false
112
+ // if (env.isCI || process.env.NODE_ENV === 'test') {
113
+ // return false
114
+ // }
115
+ // const statsigClient = await initializeStatsig()
116
+ // if (!statsigClient) return false
117
+
118
+ // const value = statsigClient.checkGate(gateName)
119
+ // gateValues[gateName] = value
120
+ // return value
121
+ })
122
+
123
+ export const useStatsigGate = (gateName: string, defaultValue = false) => {
124
+ return false
125
+ // const [gateValue, setGateValue] = React.useState(defaultValue)
126
+ // React.useEffect(() => {
127
+ // checkGate(gateName).then(setGateValue)
128
+ // }, [gateName])
129
+ // return gateValue
130
+ }
131
+
132
+ export function getGateValues(): Record<string, boolean> {
133
+ return { ...gateValues }
134
+ }
135
+
136
+ export const getExperimentValue = memoize(
137
+ async <T>(experimentName: string, defaultValue: T): Promise<T> => {
138
+ return defaultValue
139
+ // if (env.isCI || process.env.NODE_ENV === 'test') {
140
+ // return defaultValue
141
+ // }
142
+ // const statsigClient = await initializeStatsig()
143
+ // if (!statsigClient) return defaultValue
144
+
145
+ // const experiment = statsigClient.getExperiment(experimentName)
146
+ // if (Object.keys(experiment.value).length === 0) {
147
+ // logError(`getExperimentValue got empty value for ${experimentName}`)
148
+ // return defaultValue
149
+ // }
150
+ // return experiment.value as T
151
+ },
152
+ )
153
+
154
+ // NB Not memoized like other methods, to allow for dynamic config changes
155
+ export const getDynamicConfig = async <T>(
156
+ configName: string,
157
+ defaultValue: T,
158
+ ): Promise<T> => {
159
+ return defaultValue
160
+ // if (env.isCI || process.env.NODE_ENV === 'test') {
161
+ // return defaultValue
162
+ // }
163
+ // const statsigClient = await initializeStatsig()
164
+ // if (!statsigClient) return defaultValue
165
+
166
+ // const config = statsigClient.getDynamicConfig(configName)
167
+ // if (Object.keys(config.value).length === 0) {
168
+ // logError(`getDynamicConfig got empty value for ${configName}`)
169
+ // return defaultValue
170
+ // }
171
+ // return config.value as T
172
+ }
@@ -0,0 +1,86 @@
1
+ import { StorageProvider } from '@statsig/client-core'
2
+ import * as fs from 'fs'
3
+ import * as path from 'path'
4
+ import { homedir } from 'os'
5
+ import { logError } from '../utils/log'
6
+ import { existsSync, unlinkSync } from 'fs'
7
+ import { CONFIG_BASE_DIR } from '../constants/product'
8
+
9
+ // Support both KODE_CONFIG_DIR and CLAUDE_CONFIG_DIR environment variables
10
+ const CONFIG_DIR = process.env.KODE_CONFIG_DIR ?? process.env.CLAUDE_CONFIG_DIR ?? path.join(homedir(), CONFIG_BASE_DIR)
11
+ const STATSIG_DIR = path.join(CONFIG_DIR, 'statsig')
12
+
13
+ // Ensure the directory exists
14
+ try {
15
+ fs.mkdirSync(STATSIG_DIR, { recursive: true })
16
+ } catch (error) {
17
+ logError(`Failed to create statsig storage directory: ${error}`)
18
+ }
19
+
20
+ export class FileSystemStorageProvider implements StorageProvider {
21
+ private cache: Map<string, string> = new Map()
22
+ private ready = false
23
+
24
+ constructor() {
25
+ // Load all existing files into cache on startup
26
+ try {
27
+ if (!fs.existsSync(STATSIG_DIR)) {
28
+ fs.mkdirSync(STATSIG_DIR, { recursive: true })
29
+ }
30
+ const files = fs.readdirSync(STATSIG_DIR)
31
+ for (const file of files) {
32
+ const key = decodeURIComponent(file)
33
+ const value = fs.readFileSync(path.join(STATSIG_DIR, file), 'utf8')
34
+ this.cache.set(key, value)
35
+ }
36
+ this.ready = true
37
+ } catch (error) {
38
+ logError(`Failed to initialize statsig storage: ${error}`)
39
+ this.ready = true // Still mark as ready to avoid blocking
40
+ }
41
+ }
42
+
43
+ isReady(): boolean {
44
+ return this.ready
45
+ }
46
+
47
+ isReadyResolver(): Promise<void> | null {
48
+ return this.ready ? Promise.resolve() : null
49
+ }
50
+
51
+ getProviderName(): string {
52
+ return 'FileSystemStorageProvider'
53
+ }
54
+
55
+ getItem(key: string): string | null {
56
+ return this.cache.get(key) ?? null
57
+ }
58
+
59
+ setItem(key: string, value: string): void {
60
+ this.cache.set(key, value)
61
+ try {
62
+ const encodedKey = encodeURIComponent(key)
63
+ fs.writeFileSync(path.join(STATSIG_DIR, encodedKey), value, 'utf8')
64
+ } catch (error) {
65
+ logError(`Failed to write statsig storage item: ${error}`)
66
+ }
67
+ }
68
+
69
+ removeItem(key: string): void {
70
+ this.cache.delete(key)
71
+ const encodedKey = encodeURIComponent(key)
72
+ const file = path.join(STATSIG_DIR, encodedKey)
73
+ if (!existsSync(file)) {
74
+ return
75
+ }
76
+ try {
77
+ unlinkSync(file)
78
+ } catch (error) {
79
+ logError(`Failed to remove statsig storage item: ${error}`)
80
+ }
81
+ }
82
+
83
+ getAllKeys(): readonly string[] {
84
+ return Array.from(this.cache.keys())
85
+ }
86
+ }