@shareai-lab/kode 1.1.14 → 1.1.16-dev.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (289) hide show
  1. package/cli.js +77 -82
  2. package/dist/entrypoints/cli.js +59 -38
  3. package/dist/entrypoints/cli.js.map +3 -3
  4. package/dist/index.js +5 -26
  5. package/dist/package.json +4 -1
  6. package/package.json +11 -104
  7. package/dist/test/testAdapters.js +0 -88
  8. package/dist/test/testAdapters.js.map +0 -1
  9. package/src/ProjectOnboarding.tsx +0 -198
  10. package/src/Tool.ts +0 -83
  11. package/src/commands/agents.tsx +0 -3416
  12. package/src/commands/approvedTools.ts +0 -53
  13. package/src/commands/bug.tsx +0 -20
  14. package/src/commands/clear.ts +0 -43
  15. package/src/commands/compact.ts +0 -120
  16. package/src/commands/config.tsx +0 -19
  17. package/src/commands/cost.ts +0 -18
  18. package/src/commands/ctx_viz.ts +0 -209
  19. package/src/commands/doctor.ts +0 -24
  20. package/src/commands/help.tsx +0 -19
  21. package/src/commands/init.ts +0 -37
  22. package/src/commands/listen.ts +0 -42
  23. package/src/commands/login.tsx +0 -51
  24. package/src/commands/logout.tsx +0 -40
  25. package/src/commands/mcp.ts +0 -41
  26. package/src/commands/model.tsx +0 -40
  27. package/src/commands/modelstatus.tsx +0 -20
  28. package/src/commands/onboarding.tsx +0 -34
  29. package/src/commands/pr_comments.ts +0 -59
  30. package/src/commands/refreshCommands.ts +0 -54
  31. package/src/commands/release-notes.ts +0 -34
  32. package/src/commands/resume.tsx +0 -31
  33. package/src/commands/review.ts +0 -49
  34. package/src/commands/terminalSetup.ts +0 -221
  35. package/src/commands.ts +0 -139
  36. package/src/components/ApproveApiKey.tsx +0 -93
  37. package/src/components/AsciiLogo.tsx +0 -13
  38. package/src/components/AutoUpdater.tsx +0 -148
  39. package/src/components/Bug.tsx +0 -367
  40. package/src/components/Config.tsx +0 -293
  41. package/src/components/ConsoleOAuthFlow.tsx +0 -327
  42. package/src/components/Cost.tsx +0 -23
  43. package/src/components/CostThresholdDialog.tsx +0 -46
  44. package/src/components/CustomSelect/option-map.ts +0 -42
  45. package/src/components/CustomSelect/select-option.tsx +0 -78
  46. package/src/components/CustomSelect/select.tsx +0 -152
  47. package/src/components/CustomSelect/theme.ts +0 -45
  48. package/src/components/CustomSelect/use-select-state.ts +0 -414
  49. package/src/components/CustomSelect/use-select.ts +0 -35
  50. package/src/components/FallbackToolUseRejectedMessage.tsx +0 -15
  51. package/src/components/FileEditToolUpdatedMessage.tsx +0 -66
  52. package/src/components/Help.tsx +0 -215
  53. package/src/components/HighlightedCode.tsx +0 -33
  54. package/src/components/InvalidConfigDialog.tsx +0 -113
  55. package/src/components/Link.tsx +0 -32
  56. package/src/components/LogSelector.tsx +0 -86
  57. package/src/components/Logo.tsx +0 -170
  58. package/src/components/MCPServerApprovalDialog.tsx +0 -100
  59. package/src/components/MCPServerDialogCopy.tsx +0 -25
  60. package/src/components/MCPServerMultiselectDialog.tsx +0 -109
  61. package/src/components/Message.tsx +0 -221
  62. package/src/components/MessageResponse.tsx +0 -15
  63. package/src/components/MessageSelector.tsx +0 -211
  64. package/src/components/ModeIndicator.tsx +0 -88
  65. package/src/components/ModelConfig.tsx +0 -301
  66. package/src/components/ModelListManager.tsx +0 -227
  67. package/src/components/ModelSelector.tsx +0 -3387
  68. package/src/components/ModelStatusDisplay.tsx +0 -230
  69. package/src/components/Onboarding.tsx +0 -274
  70. package/src/components/PressEnterToContinue.tsx +0 -11
  71. package/src/components/PromptInput.tsx +0 -760
  72. package/src/components/SentryErrorBoundary.ts +0 -39
  73. package/src/components/Spinner.tsx +0 -129
  74. package/src/components/StickerRequestForm.tsx +0 -16
  75. package/src/components/StructuredDiff.tsx +0 -191
  76. package/src/components/TextInput.tsx +0 -259
  77. package/src/components/TodoItem.tsx +0 -47
  78. package/src/components/TokenWarning.tsx +0 -31
  79. package/src/components/ToolUseLoader.tsx +0 -40
  80. package/src/components/TrustDialog.tsx +0 -106
  81. package/src/components/binary-feedback/BinaryFeedback.tsx +0 -63
  82. package/src/components/binary-feedback/BinaryFeedbackOption.tsx +0 -111
  83. package/src/components/binary-feedback/BinaryFeedbackView.tsx +0 -172
  84. package/src/components/binary-feedback/utils.ts +0 -220
  85. package/src/components/messages/AssistantBashOutputMessage.tsx +0 -22
  86. package/src/components/messages/AssistantLocalCommandOutputMessage.tsx +0 -49
  87. package/src/components/messages/AssistantRedactedThinkingMessage.tsx +0 -19
  88. package/src/components/messages/AssistantTextMessage.tsx +0 -144
  89. package/src/components/messages/AssistantThinkingMessage.tsx +0 -40
  90. package/src/components/messages/AssistantToolUseMessage.tsx +0 -132
  91. package/src/components/messages/TaskProgressMessage.tsx +0 -32
  92. package/src/components/messages/TaskToolMessage.tsx +0 -58
  93. package/src/components/messages/UserBashInputMessage.tsx +0 -28
  94. package/src/components/messages/UserCommandMessage.tsx +0 -30
  95. package/src/components/messages/UserKodingInputMessage.tsx +0 -28
  96. package/src/components/messages/UserPromptMessage.tsx +0 -35
  97. package/src/components/messages/UserTextMessage.tsx +0 -39
  98. package/src/components/messages/UserToolResultMessage/UserToolCanceledMessage.tsx +0 -12
  99. package/src/components/messages/UserToolResultMessage/UserToolErrorMessage.tsx +0 -36
  100. package/src/components/messages/UserToolResultMessage/UserToolRejectMessage.tsx +0 -31
  101. package/src/components/messages/UserToolResultMessage/UserToolResultMessage.tsx +0 -57
  102. package/src/components/messages/UserToolResultMessage/UserToolSuccessMessage.tsx +0 -35
  103. package/src/components/messages/UserToolResultMessage/utils.tsx +0 -56
  104. package/src/components/permissions/BashPermissionRequest/BashPermissionRequest.tsx +0 -121
  105. package/src/components/permissions/FallbackPermissionRequest.tsx +0 -153
  106. package/src/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.tsx +0 -182
  107. package/src/components/permissions/FileEditPermissionRequest/FileEditToolDiff.tsx +0 -77
  108. package/src/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.tsx +0 -164
  109. package/src/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.tsx +0 -83
  110. package/src/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.tsx +0 -240
  111. package/src/components/permissions/PermissionRequest.tsx +0 -101
  112. package/src/components/permissions/PermissionRequestTitle.tsx +0 -69
  113. package/src/components/permissions/hooks.ts +0 -44
  114. package/src/components/permissions/toolUseOptions.ts +0 -59
  115. package/src/components/permissions/utils.ts +0 -23
  116. package/src/constants/betas.ts +0 -5
  117. package/src/constants/claude-asterisk-ascii-art.tsx +0 -238
  118. package/src/constants/figures.ts +0 -4
  119. package/src/constants/keys.ts +0 -3
  120. package/src/constants/macros.ts +0 -11
  121. package/src/constants/modelCapabilities.ts +0 -179
  122. package/src/constants/models.ts +0 -1025
  123. package/src/constants/oauth.ts +0 -18
  124. package/src/constants/product.ts +0 -17
  125. package/src/constants/prompts.ts +0 -168
  126. package/src/constants/releaseNotes.ts +0 -7
  127. package/src/context/PermissionContext.tsx +0 -149
  128. package/src/context.ts +0 -278
  129. package/src/cost-tracker.ts +0 -84
  130. package/src/entrypoints/cli.tsx +0 -1561
  131. package/src/entrypoints/mcp.ts +0 -175
  132. package/src/history.ts +0 -25
  133. package/src/hooks/useApiKeyVerification.ts +0 -59
  134. package/src/hooks/useArrowKeyHistory.ts +0 -55
  135. package/src/hooks/useCanUseTool.ts +0 -138
  136. package/src/hooks/useCancelRequest.ts +0 -39
  137. package/src/hooks/useDoublePress.ts +0 -41
  138. package/src/hooks/useExitOnCtrlCD.ts +0 -31
  139. package/src/hooks/useInterval.ts +0 -25
  140. package/src/hooks/useLogMessages.ts +0 -16
  141. package/src/hooks/useLogStartupTime.ts +0 -12
  142. package/src/hooks/useNotifyAfterTimeout.ts +0 -65
  143. package/src/hooks/usePermissionRequestLogging.ts +0 -44
  144. package/src/hooks/useTerminalSize.ts +0 -49
  145. package/src/hooks/useTextInput.ts +0 -317
  146. package/src/hooks/useUnifiedCompletion.ts +0 -1405
  147. package/src/index.ts +0 -34
  148. package/src/messages.ts +0 -38
  149. package/src/permissions.ts +0 -268
  150. package/src/query.ts +0 -720
  151. package/src/screens/ConfigureNpmPrefix.tsx +0 -197
  152. package/src/screens/Doctor.tsx +0 -219
  153. package/src/screens/LogList.tsx +0 -68
  154. package/src/screens/REPL.tsx +0 -813
  155. package/src/screens/ResumeConversation.tsx +0 -68
  156. package/src/services/adapters/base.ts +0 -38
  157. package/src/services/adapters/chatCompletions.ts +0 -90
  158. package/src/services/adapters/responsesAPI.ts +0 -170
  159. package/src/services/browserMocks.ts +0 -66
  160. package/src/services/claude.ts +0 -2197
  161. package/src/services/customCommands.ts +0 -704
  162. package/src/services/fileFreshness.ts +0 -377
  163. package/src/services/gpt5ConnectionTest.ts +0 -340
  164. package/src/services/mcpClient.ts +0 -564
  165. package/src/services/mcpServerApproval.tsx +0 -50
  166. package/src/services/mentionProcessor.ts +0 -273
  167. package/src/services/modelAdapterFactory.ts +0 -69
  168. package/src/services/notifier.ts +0 -40
  169. package/src/services/oauth.ts +0 -357
  170. package/src/services/openai.ts +0 -1359
  171. package/src/services/responseStateManager.ts +0 -90
  172. package/src/services/sentry.ts +0 -3
  173. package/src/services/statsig.ts +0 -172
  174. package/src/services/statsigStorage.ts +0 -86
  175. package/src/services/systemReminder.ts +0 -507
  176. package/src/services/vcr.ts +0 -161
  177. package/src/test/testAdapters.ts +0 -96
  178. package/src/tools/ArchitectTool/ArchitectTool.tsx +0 -135
  179. package/src/tools/ArchitectTool/prompt.ts +0 -15
  180. package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +0 -576
  181. package/src/tools/BashTool/BashTool.tsx +0 -243
  182. package/src/tools/BashTool/BashToolResultMessage.tsx +0 -38
  183. package/src/tools/BashTool/OutputLine.tsx +0 -49
  184. package/src/tools/BashTool/prompt.ts +0 -174
  185. package/src/tools/BashTool/utils.ts +0 -56
  186. package/src/tools/FileEditTool/FileEditTool.tsx +0 -319
  187. package/src/tools/FileEditTool/prompt.ts +0 -51
  188. package/src/tools/FileEditTool/utils.ts +0 -58
  189. package/src/tools/FileReadTool/FileReadTool.tsx +0 -404
  190. package/src/tools/FileReadTool/prompt.ts +0 -7
  191. package/src/tools/FileWriteTool/FileWriteTool.tsx +0 -301
  192. package/src/tools/FileWriteTool/prompt.ts +0 -10
  193. package/src/tools/GlobTool/GlobTool.tsx +0 -119
  194. package/src/tools/GlobTool/prompt.ts +0 -8
  195. package/src/tools/GrepTool/GrepTool.tsx +0 -147
  196. package/src/tools/GrepTool/prompt.ts +0 -11
  197. package/src/tools/MCPTool/MCPTool.tsx +0 -107
  198. package/src/tools/MCPTool/prompt.ts +0 -3
  199. package/src/tools/MemoryReadTool/MemoryReadTool.tsx +0 -127
  200. package/src/tools/MemoryReadTool/prompt.ts +0 -3
  201. package/src/tools/MemoryWriteTool/MemoryWriteTool.tsx +0 -89
  202. package/src/tools/MemoryWriteTool/prompt.ts +0 -3
  203. package/src/tools/MultiEditTool/MultiEditTool.tsx +0 -388
  204. package/src/tools/MultiEditTool/prompt.ts +0 -45
  205. package/src/tools/NotebookEditTool/NotebookEditTool.tsx +0 -298
  206. package/src/tools/NotebookEditTool/prompt.ts +0 -3
  207. package/src/tools/NotebookReadTool/NotebookReadTool.tsx +0 -258
  208. package/src/tools/NotebookReadTool/prompt.ts +0 -3
  209. package/src/tools/StickerRequestTool/StickerRequestTool.tsx +0 -107
  210. package/src/tools/StickerRequestTool/prompt.ts +0 -19
  211. package/src/tools/TaskTool/TaskTool.tsx +0 -438
  212. package/src/tools/TaskTool/constants.ts +0 -1
  213. package/src/tools/TaskTool/prompt.ts +0 -92
  214. package/src/tools/ThinkTool/ThinkTool.tsx +0 -54
  215. package/src/tools/ThinkTool/prompt.ts +0 -12
  216. package/src/tools/TodoWriteTool/TodoWriteTool.tsx +0 -313
  217. package/src/tools/TodoWriteTool/prompt.ts +0 -63
  218. package/src/tools/URLFetcherTool/URLFetcherTool.tsx +0 -178
  219. package/src/tools/URLFetcherTool/cache.ts +0 -55
  220. package/src/tools/URLFetcherTool/htmlToMarkdown.ts +0 -55
  221. package/src/tools/URLFetcherTool/prompt.ts +0 -17
  222. package/src/tools/WebSearchTool/WebSearchTool.tsx +0 -103
  223. package/src/tools/WebSearchTool/prompt.ts +0 -13
  224. package/src/tools/WebSearchTool/searchProviders.ts +0 -66
  225. package/src/tools/lsTool/lsTool.tsx +0 -272
  226. package/src/tools/lsTool/prompt.ts +0 -2
  227. package/src/tools.ts +0 -67
  228. package/src/types/PermissionMode.ts +0 -120
  229. package/src/types/RequestContext.ts +0 -72
  230. package/src/types/common.d.ts +0 -2
  231. package/src/types/conversation.ts +0 -51
  232. package/src/types/logs.ts +0 -58
  233. package/src/types/modelCapabilities.ts +0 -64
  234. package/src/types/notebook.ts +0 -87
  235. package/src/utils/Cursor.ts +0 -436
  236. package/src/utils/PersistentShell.ts +0 -552
  237. package/src/utils/advancedFuzzyMatcher.ts +0 -290
  238. package/src/utils/agentLoader.ts +0 -278
  239. package/src/utils/agentStorage.ts +0 -97
  240. package/src/utils/array.ts +0 -3
  241. package/src/utils/ask.tsx +0 -99
  242. package/src/utils/auth.ts +0 -13
  243. package/src/utils/autoCompactCore.ts +0 -223
  244. package/src/utils/autoUpdater.ts +0 -458
  245. package/src/utils/betas.ts +0 -20
  246. package/src/utils/browser.ts +0 -14
  247. package/src/utils/cleanup.ts +0 -72
  248. package/src/utils/commands.ts +0 -261
  249. package/src/utils/commonUnixCommands.ts +0 -161
  250. package/src/utils/config.ts +0 -945
  251. package/src/utils/conversationRecovery.ts +0 -55
  252. package/src/utils/debugLogger.ts +0 -1235
  253. package/src/utils/diff.ts +0 -42
  254. package/src/utils/env.ts +0 -57
  255. package/src/utils/errors.ts +0 -21
  256. package/src/utils/exampleCommands.ts +0 -109
  257. package/src/utils/execFileNoThrow.ts +0 -51
  258. package/src/utils/expertChatStorage.ts +0 -136
  259. package/src/utils/file.ts +0 -405
  260. package/src/utils/fileRecoveryCore.ts +0 -71
  261. package/src/utils/format.tsx +0 -44
  262. package/src/utils/fuzzyMatcher.ts +0 -328
  263. package/src/utils/generators.ts +0 -62
  264. package/src/utils/git.ts +0 -92
  265. package/src/utils/globalLogger.ts +0 -77
  266. package/src/utils/http.ts +0 -10
  267. package/src/utils/imagePaste.ts +0 -38
  268. package/src/utils/json.ts +0 -13
  269. package/src/utils/log.ts +0 -382
  270. package/src/utils/markdown.ts +0 -213
  271. package/src/utils/messageContextManager.ts +0 -294
  272. package/src/utils/messages.tsx +0 -945
  273. package/src/utils/model.ts +0 -914
  274. package/src/utils/permissions/filesystem.ts +0 -127
  275. package/src/utils/responseState.ts +0 -23
  276. package/src/utils/ripgrep.ts +0 -167
  277. package/src/utils/secureFile.ts +0 -564
  278. package/src/utils/sessionState.ts +0 -49
  279. package/src/utils/state.ts +0 -25
  280. package/src/utils/style.ts +0 -29
  281. package/src/utils/terminal.ts +0 -50
  282. package/src/utils/theme.ts +0 -127
  283. package/src/utils/thinking.ts +0 -144
  284. package/src/utils/todoStorage.ts +0 -431
  285. package/src/utils/tokens.ts +0 -43
  286. package/src/utils/toolExecutionController.ts +0 -163
  287. package/src/utils/unaryLogging.ts +0 -26
  288. package/src/utils/user.ts +0 -37
  289. package/src/utils/validate.ts +0 -165
@@ -1,377 +0,0 @@
1
- import { statSync, existsSync, watchFile, unwatchFile } from 'fs'
2
- import {
3
- emitReminderEvent,
4
- systemReminderService,
5
- } from '../services/systemReminder'
6
- import { getAgentFilePath } from '../utils/agentStorage'
7
- import { logEvent } from '../services/statsig'
8
-
9
- interface FileTimestamp {
10
- path: string
11
- lastRead: number
12
- lastModified: number
13
- size: number
14
- lastAgentEdit?: number // Track when Agent last edited this file
15
- }
16
-
17
- interface FileFreshnessState {
18
- readTimestamps: Map<string, FileTimestamp>
19
- editConflicts: Set<string>
20
- sessionFiles: Set<string>
21
- watchedTodoFiles: Map<string, string> // agentId -> filePath
22
- }
23
-
24
- class FileFreshnessService {
25
- private state: FileFreshnessState = {
26
- readTimestamps: new Map(),
27
- editConflicts: new Set(),
28
- sessionFiles: new Set(),
29
- watchedTodoFiles: new Map(),
30
- }
31
-
32
- constructor() {
33
- this.setupEventListeners()
34
- }
35
-
36
- /**
37
- * Setup event listeners for session management
38
- */
39
- private setupEventListeners(): void {
40
- // Listen for session startup events through the SystemReminderService
41
- systemReminderService.addEventListener(
42
- 'session:startup',
43
- (context: any) => {
44
- // Reset session state on startup
45
- this.resetSession()
46
-
47
- // Log session startup
48
- logEvent('file_freshness_session_startup', {
49
- agentId: context.agentId || 'default',
50
- timestamp: context.timestamp,
51
- })
52
- },
53
- )
54
- }
55
-
56
- /**
57
- * Record file read operation with timestamp tracking
58
- */
59
- public recordFileRead(filePath: string): void {
60
- try {
61
- if (!existsSync(filePath)) {
62
- return
63
- }
64
-
65
- const stats = statSync(filePath)
66
- const timestamp: FileTimestamp = {
67
- path: filePath,
68
- lastRead: Date.now(),
69
- lastModified: stats.mtimeMs,
70
- size: stats.size,
71
- }
72
-
73
- this.state.readTimestamps.set(filePath, timestamp)
74
- this.state.sessionFiles.add(filePath)
75
-
76
- // Emit file read event for system reminders
77
- emitReminderEvent('file:read', {
78
- filePath,
79
- timestamp: timestamp.lastRead,
80
- size: timestamp.size,
81
- modified: timestamp.lastModified,
82
- })
83
- } catch (error) {
84
- console.error(`Error recording file read for ${filePath}:`, error)
85
- }
86
- }
87
-
88
- /**
89
- * Check if file has been modified since last read
90
- */
91
- public checkFileFreshness(filePath: string): {
92
- isFresh: boolean
93
- lastRead?: number
94
- currentModified?: number
95
- conflict: boolean
96
- } {
97
- const recorded = this.state.readTimestamps.get(filePath)
98
-
99
- if (!recorded) {
100
- return { isFresh: true, conflict: false }
101
- }
102
-
103
- try {
104
- if (!existsSync(filePath)) {
105
- return { isFresh: false, conflict: true }
106
- }
107
-
108
- const currentStats = statSync(filePath)
109
- const isFresh = currentStats.mtimeMs <= recorded.lastModified
110
- const conflict = !isFresh
111
-
112
- if (conflict) {
113
- this.state.editConflicts.add(filePath)
114
-
115
- // Emit file conflict event
116
- emitReminderEvent('file:conflict', {
117
- filePath,
118
- lastRead: recorded.lastRead,
119
- lastModified: recorded.lastModified,
120
- currentModified: currentStats.mtimeMs,
121
- sizeDiff: currentStats.size - recorded.size,
122
- })
123
- }
124
-
125
- return {
126
- isFresh,
127
- lastRead: recorded.lastRead,
128
- currentModified: currentStats.mtimeMs,
129
- conflict,
130
- }
131
- } catch (error) {
132
- console.error(`Error checking freshness for ${filePath}:`, error)
133
- return { isFresh: false, conflict: true }
134
- }
135
- }
136
-
137
- /**
138
- * Record file edit operation by Agent
139
- */
140
- public recordFileEdit(filePath: string, content?: string): void {
141
- try {
142
- const now = Date.now()
143
-
144
- // Update recorded timestamp after edit
145
- if (existsSync(filePath)) {
146
- const stats = statSync(filePath)
147
- const existing = this.state.readTimestamps.get(filePath)
148
-
149
- if (existing) {
150
- existing.lastModified = stats.mtimeMs
151
- existing.size = stats.size
152
- existing.lastAgentEdit = now // Mark this as Agent-initiated edit
153
- this.state.readTimestamps.set(filePath, existing)
154
- } else {
155
- // Create new record for Agent-edited file
156
- const timestamp: FileTimestamp = {
157
- path: filePath,
158
- lastRead: now,
159
- lastModified: stats.mtimeMs,
160
- size: stats.size,
161
- lastAgentEdit: now,
162
- }
163
- this.state.readTimestamps.set(filePath, timestamp)
164
- }
165
- }
166
-
167
- // Remove from conflicts since we just edited it
168
- this.state.editConflicts.delete(filePath)
169
-
170
- // Emit file edit event
171
- emitReminderEvent('file:edited', {
172
- filePath,
173
- timestamp: now,
174
- contentLength: content?.length || 0,
175
- source: 'agent',
176
- })
177
- } catch (error) {
178
- console.error(`Error recording file edit for ${filePath}:`, error)
179
- }
180
- }
181
-
182
- public generateFileModificationReminder(filePath: string): string | null {
183
- const recorded = this.state.readTimestamps.get(filePath)
184
-
185
- if (!recorded) {
186
- return null
187
- }
188
-
189
- try {
190
- if (!existsSync(filePath)) {
191
- return `Note: ${filePath} was deleted since last read.`
192
- }
193
-
194
- const currentStats = statSync(filePath)
195
- const isModified = currentStats.mtimeMs > recorded.lastModified
196
-
197
- if (!isModified) {
198
- return null
199
- }
200
-
201
- // Check if this was an Agent-initiated change
202
- // Use small time tolerance to handle filesystem timestamp precision issues
203
- const TIME_TOLERANCE_MS = 100
204
- if (
205
- recorded.lastAgentEdit &&
206
- recorded.lastAgentEdit >= recorded.lastModified - TIME_TOLERANCE_MS
207
- ) {
208
- // Agent modified this file recently, no reminder needed
209
- // (context already contains before/after content)
210
- return null
211
- }
212
-
213
- // External modification detected - generate reminder
214
- return `Note: ${filePath} was modified externally since last read. The file may have changed outside of this session.`
215
- } catch (error) {
216
- console.error(`Error checking modification for ${filePath}:`, error)
217
- return null
218
- }
219
- }
220
-
221
- public getConflictedFiles(): string[] {
222
- return Array.from(this.state.editConflicts)
223
- }
224
-
225
- public getSessionFiles(): string[] {
226
- return Array.from(this.state.sessionFiles)
227
- }
228
-
229
- public resetSession(): void {
230
- // Clean up existing todo file watchers
231
- this.state.watchedTodoFiles.forEach(filePath => {
232
- try {
233
- unwatchFile(filePath)
234
- } catch (error) {
235
- console.error(`Error unwatching file ${filePath}:`, error)
236
- }
237
- })
238
-
239
- this.state = {
240
- readTimestamps: new Map(),
241
- editConflicts: new Set(),
242
- sessionFiles: new Set(),
243
- watchedTodoFiles: new Map(),
244
- }
245
- }
246
-
247
- /**
248
- * Start watching todo file for an agent
249
- */
250
- public startWatchingTodoFile(agentId: string): void {
251
- try {
252
- const filePath = getAgentFilePath(agentId)
253
-
254
- // Don't watch if already watching
255
- if (this.state.watchedTodoFiles.has(agentId)) {
256
- return
257
- }
258
-
259
- this.state.watchedTodoFiles.set(agentId, filePath)
260
-
261
- // Record initial state if file exists
262
- if (existsSync(filePath)) {
263
- this.recordFileRead(filePath)
264
- }
265
-
266
- // Start watching for changes
267
- watchFile(filePath, { interval: 1000 }, (curr, prev) => {
268
- // Check if this was an external modification
269
- const reminder = this.generateFileModificationReminder(filePath)
270
- if (reminder) {
271
- // File was modified externally, emit todo change reminder
272
- emitReminderEvent('todo:file_changed', {
273
- agentId,
274
- filePath,
275
- reminder,
276
- timestamp: Date.now(),
277
- currentStats: { mtime: curr.mtime, size: curr.size },
278
- previousStats: { mtime: prev.mtime, size: prev.size },
279
- })
280
- }
281
- })
282
- } catch (error) {
283
- console.error(
284
- `Error starting todo file watch for agent ${agentId}:`,
285
- error,
286
- )
287
- }
288
- }
289
-
290
- /**
291
- * Stop watching todo file for an agent
292
- */
293
- public stopWatchingTodoFile(agentId: string): void {
294
- try {
295
- const filePath = this.state.watchedTodoFiles.get(agentId)
296
- if (filePath) {
297
- unwatchFile(filePath)
298
- this.state.watchedTodoFiles.delete(agentId)
299
- }
300
- } catch (error) {
301
- console.error(
302
- `Error stopping todo file watch for agent ${agentId}:`,
303
- error,
304
- )
305
- }
306
- }
307
-
308
- public getFileInfo(filePath: string): FileTimestamp | null {
309
- return this.state.readTimestamps.get(filePath) || null
310
- }
311
-
312
- public isFileTracked(filePath: string): boolean {
313
- return this.state.readTimestamps.has(filePath)
314
- }
315
-
316
- /**
317
- * Retrieves files prioritized for recovery during conversation compression
318
- *
319
- * Selects recently accessed files based on:
320
- * - File access recency (most recent first)
321
- * - File type relevance (excludes dependencies, build artifacts)
322
- * - Development workflow importance
323
- *
324
- * Used to maintain coding context when conversation history is compressed
325
- */
326
- public getImportantFiles(maxFiles: number = 5): Array<{
327
- path: string
328
- timestamp: number
329
- size: number
330
- }> {
331
- return Array.from(this.state.readTimestamps.entries())
332
- .map(([path, info]) => ({
333
- path,
334
- timestamp: info.lastRead,
335
- size: info.size,
336
- }))
337
- .filter(file => this.isValidForRecovery(file.path))
338
- .sort((a, b) => b.timestamp - a.timestamp) // Newest first
339
- .slice(0, maxFiles)
340
- }
341
-
342
- /**
343
- * Determines which files are suitable for automatic recovery
344
- *
345
- * Excludes files that are typically not relevant for development context:
346
- * - Build artifacts and generated files
347
- * - Dependencies and cached files
348
- * - Temporary files and system directories
349
- */
350
- private isValidForRecovery(filePath: string): boolean {
351
- return (
352
- !filePath.includes('node_modules') &&
353
- !filePath.includes('.git') &&
354
- !filePath.startsWith('/tmp') &&
355
- !filePath.includes('.cache') &&
356
- !filePath.includes('dist/') &&
357
- !filePath.includes('build/')
358
- )
359
- }
360
- }
361
-
362
- export const fileFreshnessService = new FileFreshnessService()
363
-
364
- export const recordFileRead = (filePath: string) =>
365
- fileFreshnessService.recordFileRead(filePath)
366
- export const recordFileEdit = (filePath: string, content?: string) =>
367
- fileFreshnessService.recordFileEdit(filePath, content)
368
- export const checkFileFreshness = (filePath: string) =>
369
- fileFreshnessService.checkFileFreshness(filePath)
370
- export const generateFileModificationReminder = (filePath: string) =>
371
- fileFreshnessService.generateFileModificationReminder(filePath)
372
- export const resetFileFreshnessSession = () =>
373
- fileFreshnessService.resetSession()
374
- export const startWatchingTodoFile = (agentId: string) =>
375
- fileFreshnessService.startWatchingTodoFile(agentId)
376
- export const stopWatchingTodoFile = (agentId: string) =>
377
- fileFreshnessService.stopWatchingTodoFile(agentId)
@@ -1,340 +0,0 @@
1
- /**
2
- * 🔥 GPT-5 Connection Test Service
3
- *
4
- * Specialized connection testing for GPT-5 models that supports both
5
- * Responses API and Chat Completions API with proper fallback handling.
6
- */
7
-
8
- import { getModelFeatures } from './openai'
9
-
10
- export interface ConnectionTestResult {
11
- success: boolean
12
- message: string
13
- endpoint?: string
14
- details?: string
15
- apiUsed?: 'responses' | 'chat_completions'
16
- responseTime?: number
17
- }
18
-
19
- export interface GPT5TestConfig {
20
- model: string
21
- apiKey: string
22
- baseURL?: string
23
- maxTokens?: number
24
- provider?: string
25
- }
26
-
27
- /**
28
- * Test GPT-5 model connection with intelligent API selection
29
- */
30
- export async function testGPT5Connection(config: GPT5TestConfig): Promise<ConnectionTestResult> {
31
- const startTime = Date.now()
32
-
33
- // Validate configuration
34
- if (!config.model || !config.apiKey) {
35
- return {
36
- success: false,
37
- message: 'Invalid configuration',
38
- details: 'Model name and API key are required',
39
- }
40
- }
41
-
42
- const isGPT5 = config.model.toLowerCase().includes('gpt-5')
43
- const modelFeatures = getModelFeatures(config.model)
44
- const baseURL = config.baseURL || 'https://api.openai.com/v1'
45
- const isOfficialOpenAI = !config.baseURL || config.baseURL.includes('api.openai.com')
46
-
47
- console.log(`🔧 Testing GPT-5 connection for model: ${config.model}`)
48
- console.log(`🔧 Base URL: ${baseURL}`)
49
- console.log(`🔧 Official OpenAI: ${isOfficialOpenAI}`)
50
- console.log(`🔧 Supports Responses API: ${modelFeatures.supportsResponsesAPI}`)
51
-
52
- // Try Responses API first for official GPT-5 models
53
- if (isGPT5 && modelFeatures.supportsResponsesAPI && isOfficialOpenAI) {
54
- console.log(`🚀 Attempting Responses API for ${config.model}`)
55
- const responsesResult = await testResponsesAPI(config, baseURL, startTime)
56
-
57
- if (responsesResult.success) {
58
- console.log(`✅ Responses API test successful for ${config.model}`)
59
- return responsesResult
60
- } else {
61
- console.log(`⚠️ Responses API failed, falling back to Chat Completions: ${responsesResult.details}`)
62
- }
63
- }
64
-
65
- // Fallback to Chat Completions API
66
- console.log(`🔄 Using Chat Completions API for ${config.model}`)
67
- return await testChatCompletionsAPI(config, baseURL, startTime)
68
- }
69
-
70
- /**
71
- * Test using GPT-5 Responses API
72
- */
73
- async function testResponsesAPI(
74
- config: GPT5TestConfig,
75
- baseURL: string,
76
- startTime: number
77
- ): Promise<ConnectionTestResult> {
78
- const testURL = `${baseURL.replace(/\/+$/, '')}/responses`
79
-
80
- const testPayload = {
81
- model: config.model,
82
- input: [
83
- {
84
- role: 'user',
85
- content: 'Please respond with exactly "YES" (in capital letters) to confirm this connection is working.',
86
- },
87
- ],
88
- max_completion_tokens: Math.max(config.maxTokens || 8192, 8192),
89
- temperature: 1, // GPT-5 requirement
90
- reasoning: {
91
- effort: 'low', // Fast response for connection test
92
- },
93
- }
94
-
95
- const headers = {
96
- 'Content-Type': 'application/json',
97
- 'Authorization': `Bearer ${config.apiKey}`,
98
- }
99
-
100
- console.log(`🔧 Responses API URL: ${testURL}`)
101
- console.log(`🔧 Responses API payload:`, JSON.stringify(testPayload, null, 2))
102
-
103
- try {
104
- const response = await fetch(testURL, {
105
- method: 'POST',
106
- headers,
107
- body: JSON.stringify(testPayload),
108
- })
109
-
110
- const responseTime = Date.now() - startTime
111
-
112
- if (response.ok) {
113
- const data = await response.json()
114
- console.log(`✅ Responses API successful response:`, data)
115
-
116
- // Extract content from Responses API format
117
- let responseContent = ''
118
- if (data.output_text) {
119
- responseContent = data.output_text
120
- } else if (data.output && Array.isArray(data.output)) {
121
- // Extract from structured output format
122
- const messageOutput = data.output.find(item => item.type === 'message')
123
- if (messageOutput && messageOutput.content) {
124
- const textContent = messageOutput.content.find(c => c.type === 'output_text')
125
- responseContent = textContent?.text || ''
126
- }
127
- }
128
-
129
- const containsYes = responseContent.toLowerCase().includes('yes')
130
-
131
- if (containsYes) {
132
- return {
133
- success: true,
134
- message: '✅ GPT-5 Responses API connection successful',
135
- endpoint: '/responses',
136
- details: `Model responded correctly: "${responseContent.trim()}"`,
137
- apiUsed: 'responses',
138
- responseTime,
139
- }
140
- } else {
141
- return {
142
- success: false,
143
- message: '⚠️ Responses API connected but unexpected response',
144
- endpoint: '/responses',
145
- details: `Expected "YES" but got: "${responseContent.trim() || '(empty response)'}"`,
146
- apiUsed: 'responses',
147
- responseTime,
148
- }
149
- }
150
- } else {
151
- const errorData = await response.json().catch(() => null)
152
- const errorMessage = errorData?.error?.message || errorData?.message || response.statusText
153
-
154
- console.log(`❌ Responses API error (${response.status}):`, errorData)
155
-
156
- return {
157
- success: false,
158
- message: `❌ Responses API failed (${response.status})`,
159
- endpoint: '/responses',
160
- details: `Error: ${errorMessage}`,
161
- apiUsed: 'responses',
162
- responseTime: Date.now() - startTime,
163
- }
164
- }
165
- } catch (error) {
166
- console.log(`❌ Responses API connection error:`, error)
167
-
168
- return {
169
- success: false,
170
- message: '❌ Responses API connection failed',
171
- endpoint: '/responses',
172
- details: error instanceof Error ? error.message : String(error),
173
- apiUsed: 'responses',
174
- responseTime: Date.now() - startTime,
175
- }
176
- }
177
- }
178
-
179
- /**
180
- * Test using Chat Completions API with GPT-5 compatibility
181
- */
182
- async function testChatCompletionsAPI(
183
- config: GPT5TestConfig,
184
- baseURL: string,
185
- startTime: number
186
- ): Promise<ConnectionTestResult> {
187
- const testURL = `${baseURL.replace(/\/+$/, '')}/chat/completions`
188
-
189
- const isGPT5 = config.model.toLowerCase().includes('gpt-5')
190
-
191
- // Create test payload with GPT-5 compatibility
192
- const testPayload: any = {
193
- model: config.model,
194
- messages: [
195
- {
196
- role: 'user',
197
- content: 'Please respond with exactly "YES" (in capital letters) to confirm this connection is working.',
198
- },
199
- ],
200
- temperature: isGPT5 ? 1 : 0, // GPT-5 requires temperature=1
201
- stream: false,
202
- }
203
-
204
- // 🔧 Apply GPT-5 parameter transformations
205
- if (isGPT5) {
206
- testPayload.max_completion_tokens = Math.max(config.maxTokens || 8192, 8192)
207
- delete testPayload.max_tokens // 🔥 CRITICAL: Remove max_tokens for GPT-5
208
- console.log(`🔧 GPT-5 mode: Using max_completion_tokens = ${testPayload.max_completion_tokens}`)
209
- } else {
210
- testPayload.max_tokens = Math.max(config.maxTokens || 8192, 8192)
211
- }
212
-
213
- const headers = {
214
- 'Content-Type': 'application/json',
215
- }
216
-
217
- // Add provider-specific headers
218
- if (config.provider === 'azure') {
219
- headers['api-key'] = config.apiKey
220
- } else {
221
- headers['Authorization'] = `Bearer ${config.apiKey}`
222
- }
223
-
224
- console.log(`🔧 Chat Completions URL: ${testURL}`)
225
- console.log(`🔧 Chat Completions payload:`, JSON.stringify(testPayload, null, 2))
226
-
227
- try {
228
- const response = await fetch(testURL, {
229
- method: 'POST',
230
- headers,
231
- body: JSON.stringify(testPayload),
232
- })
233
-
234
- const responseTime = Date.now() - startTime
235
-
236
- if (response.ok) {
237
- const data = await response.json()
238
- console.log(`✅ Chat Completions successful response:`, data)
239
-
240
- const responseContent = data.choices?.[0]?.message?.content || ''
241
- const containsYes = responseContent.toLowerCase().includes('yes')
242
-
243
- if (containsYes) {
244
- return {
245
- success: true,
246
- message: `✅ ${isGPT5 ? 'GPT-5' : 'Model'} Chat Completions connection successful`,
247
- endpoint: '/chat/completions',
248
- details: `Model responded correctly: "${responseContent.trim()}"`,
249
- apiUsed: 'chat_completions',
250
- responseTime,
251
- }
252
- } else {
253
- return {
254
- success: false,
255
- message: '⚠️ Chat Completions connected but unexpected response',
256
- endpoint: '/chat/completions',
257
- details: `Expected "YES" but got: "${responseContent.trim() || '(empty response)'}"`,
258
- apiUsed: 'chat_completions',
259
- responseTime,
260
- }
261
- }
262
- } else {
263
- const errorData = await response.json().catch(() => null)
264
- const errorMessage = errorData?.error?.message || errorData?.message || response.statusText
265
-
266
- console.log(`❌ Chat Completions error (${response.status}):`, errorData)
267
-
268
- // 🔧 Provide specific guidance for GPT-5 errors
269
- let details = `Error: ${errorMessage}`
270
- if (response.status === 400 && errorMessage.includes('max_tokens') && isGPT5) {
271
- details += '\n\n🔧 GPT-5 Fix Applied: This error suggests a parameter compatibility issue. Please check if the provider supports GPT-5 with max_completion_tokens.'
272
- }
273
-
274
- return {
275
- success: false,
276
- message: `❌ Chat Completions failed (${response.status})`,
277
- endpoint: '/chat/completions',
278
- details: details,
279
- apiUsed: 'chat_completions',
280
- responseTime: Date.now() - startTime,
281
- }
282
- }
283
- } catch (error) {
284
- console.log(`❌ Chat Completions connection error:`, error)
285
-
286
- return {
287
- success: false,
288
- message: '❌ Chat Completions connection failed',
289
- endpoint: '/chat/completions',
290
- details: error instanceof Error ? error.message : String(error),
291
- apiUsed: 'chat_completions',
292
- responseTime: Date.now() - startTime,
293
- }
294
- }
295
- }
296
-
297
- /**
298
- * Quick validation for GPT-5 configuration
299
- */
300
- export function validateGPT5Config(config: GPT5TestConfig): { valid: boolean; errors: string[] } {
301
- console.log(`🔧 validateGPT5Config called with:`, {
302
- model: config.model,
303
- hasApiKey: !!config.apiKey,
304
- baseURL: config.baseURL,
305
- provider: config.provider,
306
- })
307
-
308
- const errors: string[] = []
309
-
310
- if (!config.model) {
311
- errors.push('Model name is required')
312
- }
313
-
314
- if (!config.apiKey) {
315
- errors.push('API key is required')
316
- }
317
-
318
- if (config.apiKey && config.apiKey.length < 10) {
319
- errors.push('API key appears to be invalid (too short)')
320
- }
321
-
322
- const isGPT5 = config.model?.toLowerCase().includes('gpt-5')
323
- if (isGPT5) {
324
- console.log(`🔧 GPT-5 validation: model=${config.model}, maxTokens=${config.maxTokens}`)
325
-
326
- if (config.maxTokens && config.maxTokens < 1000) {
327
- errors.push('GPT-5 models typically require at least 1000 max tokens')
328
- }
329
-
330
- // 完全移除第三方provider限制,允许所有代理中转站使用GPT-5
331
- console.log(`🔧 No third-party restrictions applied for GPT-5`)
332
- }
333
-
334
- console.log(`🔧 Validation result:`, { valid: errors.length === 0, errors })
335
-
336
- return {
337
- valid: errors.length === 0,
338
- errors,
339
- }
340
- }