@shareai-lab/kode 1.1.12 → 1.1.14

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 (593) hide show
  1. package/cli.js +83 -57
  2. package/dist/ProjectOnboarding.js +99 -0
  3. package/dist/ProjectOnboarding.js.map +7 -0
  4. package/dist/Tool.js +1 -0
  5. package/dist/Tool.js.map +7 -0
  6. package/dist/commands/agents.js +2087 -0
  7. package/dist/commands/agents.js.map +7 -0
  8. package/dist/commands/approvedTools.js +36 -0
  9. package/dist/commands/approvedTools.js.map +7 -0
  10. package/dist/commands/bug.js +21 -0
  11. package/dist/commands/bug.js.map +7 -0
  12. package/dist/commands/clear.js +37 -0
  13. package/dist/commands/clear.js.map +7 -0
  14. package/dist/commands/compact.js +104 -0
  15. package/dist/commands/compact.js.map +7 -0
  16. package/dist/commands/config.js +20 -0
  17. package/dist/commands/config.js.map +7 -0
  18. package/dist/commands/cost.js +19 -0
  19. package/dist/commands/cost.js.map +7 -0
  20. package/dist/commands/ctx_viz.js +152 -0
  21. package/dist/commands/ctx_viz.js.map +7 -0
  22. package/dist/commands/doctor.js +25 -0
  23. package/dist/commands/doctor.js.map +7 -0
  24. package/dist/commands/help.js +20 -0
  25. package/dist/commands/help.js.map +7 -0
  26. package/dist/commands/init.js +38 -0
  27. package/dist/commands/init.js.map +7 -0
  28. package/dist/commands/listen.js +37 -0
  29. package/dist/commands/listen.js.map +7 -0
  30. package/dist/commands/login.js +37 -0
  31. package/dist/commands/login.js.map +7 -0
  32. package/dist/commands/logout.js +33 -0
  33. package/dist/commands/logout.js.map +7 -0
  34. package/dist/commands/mcp.js +34 -0
  35. package/dist/commands/mcp.js.map +7 -0
  36. package/dist/commands/model.js +41 -0
  37. package/dist/commands/model.js.map +7 -0
  38. package/dist/commands/modelstatus.js +21 -0
  39. package/dist/commands/modelstatus.js.map +7 -0
  40. package/dist/commands/onboarding.js +36 -0
  41. package/dist/commands/onboarding.js.map +7 -0
  42. package/dist/commands/pr_comments.js +61 -0
  43. package/dist/commands/pr_comments.js.map +7 -0
  44. package/dist/commands/refreshCommands.js +37 -0
  45. package/dist/commands/refreshCommands.js.map +7 -0
  46. package/dist/commands/release-notes.js +30 -0
  47. package/dist/commands/release-notes.js.map +7 -0
  48. package/dist/commands/resume.js +35 -0
  49. package/dist/commands/resume.js.map +7 -0
  50. package/dist/commands/review.js +51 -0
  51. package/dist/commands/review.js.map +7 -0
  52. package/dist/commands/terminalSetup.js +163 -0
  53. package/dist/commands/terminalSetup.js.map +7 -0
  54. package/dist/commands.js +84 -0
  55. package/dist/commands.js.map +7 -0
  56. package/dist/components/ApproveApiKey.js +74 -0
  57. package/dist/components/ApproveApiKey.js.map +7 -0
  58. package/dist/components/AsciiLogo.js +12 -0
  59. package/dist/components/AsciiLogo.js.map +7 -0
  60. package/dist/components/AutoUpdater.js +74 -0
  61. package/dist/components/AutoUpdater.js.map +7 -0
  62. package/dist/components/Bug.js +147 -0
  63. package/dist/components/Bug.js.map +7 -0
  64. package/dist/components/Config.js +166 -0
  65. package/dist/components/Config.js.map +7 -0
  66. package/dist/components/ConsoleOAuthFlow.js +188 -0
  67. package/dist/components/ConsoleOAuthFlow.js.map +7 -0
  68. package/dist/components/Cost.js +13 -0
  69. package/dist/components/Cost.js.map +7 -0
  70. package/dist/components/CostThresholdDialog.js +38 -0
  71. package/dist/components/CostThresholdDialog.js.map +7 -0
  72. package/dist/components/CustomSelect/option-map.js +32 -0
  73. package/dist/components/CustomSelect/option-map.js.map +7 -0
  74. package/dist/components/CustomSelect/select-option.js +34 -0
  75. package/dist/components/CustomSelect/select-option.js.map +7 -0
  76. package/dist/components/CustomSelect/select.js +64 -0
  77. package/dist/components/CustomSelect/select.js.map +7 -0
  78. package/dist/components/CustomSelect/theme.js +1 -0
  79. package/dist/components/CustomSelect/theme.js.map +7 -0
  80. package/dist/components/CustomSelect/use-select-state.js +220 -0
  81. package/dist/components/CustomSelect/use-select-state.js.map +7 -0
  82. package/dist/components/CustomSelect/use-select.js +21 -0
  83. package/dist/components/CustomSelect/use-select.js.map +7 -0
  84. package/dist/components/FallbackToolUseRejectedMessage.js +11 -0
  85. package/dist/components/FallbackToolUseRejectedMessage.js.map +7 -0
  86. package/dist/components/FileEditToolUpdatedMessage.js +31 -0
  87. package/dist/components/FileEditToolUpdatedMessage.js.map +7 -0
  88. package/dist/components/Help.js +41 -0
  89. package/dist/components/Help.js.map +7 -0
  90. package/dist/components/HighlightedCode.js +30 -0
  91. package/dist/components/HighlightedCode.js.map +7 -0
  92. package/dist/components/InvalidConfigDialog.js +83 -0
  93. package/dist/components/InvalidConfigDialog.js.map +7 -0
  94. package/dist/components/Link.js +18 -0
  95. package/dist/components/Link.js.map +7 -0
  96. package/dist/components/LogSelector.js +50 -0
  97. package/dist/components/LogSelector.js.map +7 -0
  98. package/dist/components/Logo.js +89 -0
  99. package/dist/components/Logo.js.map +7 -0
  100. package/dist/components/MCPServerApprovalDialog.js +79 -0
  101. package/dist/components/MCPServerApprovalDialog.js.map +7 -0
  102. package/dist/components/MCPServerDialogCopy.js +11 -0
  103. package/dist/components/MCPServerDialogCopy.js.map +7 -0
  104. package/dist/components/MCPServerMultiselectDialog.js +80 -0
  105. package/dist/components/MCPServerMultiselectDialog.js.map +7 -0
  106. package/dist/components/Message.js +146 -0
  107. package/dist/components/Message.js.map +7 -0
  108. package/dist/components/MessageResponse.js +9 -0
  109. package/dist/components/MessageResponse.js.map +7 -0
  110. package/dist/components/MessageSelector.js +133 -0
  111. package/dist/components/MessageSelector.js.map +7 -0
  112. package/dist/components/ModeIndicator.js +38 -0
  113. package/dist/components/ModeIndicator.js.map +7 -0
  114. package/dist/components/ModelConfig.js +208 -0
  115. package/dist/components/ModelConfig.js.map +7 -0
  116. package/dist/components/ModelListManager.js +140 -0
  117. package/dist/components/ModelListManager.js.map +7 -0
  118. package/dist/components/ModelSelector.js +1985 -0
  119. package/dist/components/ModelSelector.js.map +7 -0
  120. package/dist/components/ModelStatusDisplay.js +87 -0
  121. package/dist/components/ModelStatusDisplay.js.map +7 -0
  122. package/dist/components/Onboarding.js +153 -0
  123. package/dist/components/Onboarding.js.map +7 -0
  124. package/dist/components/PressEnterToContinue.js +10 -0
  125. package/dist/components/PressEnterToContinue.js.map +7 -0
  126. package/dist/components/PromptInput.js +501 -0
  127. package/dist/components/PromptInput.js.map +7 -0
  128. package/dist/components/SentryErrorBoundary.js +27 -0
  129. package/dist/components/SentryErrorBoundary.js.map +7 -0
  130. package/dist/components/Spinner.js +101 -0
  131. package/dist/components/Spinner.js.map +7 -0
  132. package/dist/components/StickerRequestForm.js +7 -0
  133. package/dist/components/StickerRequestForm.js.map +7 -0
  134. package/dist/components/StructuredDiff.js +148 -0
  135. package/dist/components/StructuredDiff.js.map +7 -0
  136. package/dist/components/TextInput.js +100 -0
  137. package/dist/components/TextInput.js.map +7 -0
  138. package/dist/components/TodoItem.js +35 -0
  139. package/dist/components/TodoItem.js.map +7 -0
  140. package/dist/components/TokenWarning.js +19 -0
  141. package/dist/components/TokenWarning.js.map +7 -0
  142. package/dist/components/ToolUseLoader.js +24 -0
  143. package/dist/components/ToolUseLoader.js.map +7 -0
  144. package/dist/components/TrustDialog.js +76 -0
  145. package/dist/components/TrustDialog.js.map +7 -0
  146. package/dist/components/binary-feedback/BinaryFeedback.js +50 -0
  147. package/dist/components/binary-feedback/BinaryFeedback.js.map +7 -0
  148. package/dist/components/binary-feedback/BinaryFeedbackOption.js +94 -0
  149. package/dist/components/binary-feedback/BinaryFeedbackOption.js.map +7 -0
  150. package/dist/components/binary-feedback/BinaryFeedbackView.js +139 -0
  151. package/dist/components/binary-feedback/BinaryFeedbackView.js.map +7 -0
  152. package/dist/components/binary-feedback/utils.js +161 -0
  153. package/dist/components/binary-feedback/utils.js.map +7 -0
  154. package/dist/components/messages/AssistantBashOutputMessage.js +23 -0
  155. package/dist/components/messages/AssistantBashOutputMessage.js.map +7 -0
  156. package/dist/components/messages/AssistantLocalCommandOutputMessage.js +36 -0
  157. package/dist/components/messages/AssistantLocalCommandOutputMessage.js.map +7 -0
  158. package/dist/components/messages/AssistantRedactedThinkingMessage.js +12 -0
  159. package/dist/components/messages/AssistantRedactedThinkingMessage.js.map +7 -0
  160. package/dist/components/messages/AssistantTextMessage.js +78 -0
  161. package/dist/components/messages/AssistantTextMessage.js.map +7 -0
  162. package/dist/components/messages/AssistantThinkingMessage.js +27 -0
  163. package/dist/components/messages/AssistantThinkingMessage.js.map +7 -0
  164. package/dist/components/messages/AssistantToolUseMessage.js +91 -0
  165. package/dist/components/messages/AssistantToolUseMessage.js.map +7 -0
  166. package/dist/components/messages/TaskProgressMessage.js +11 -0
  167. package/dist/components/messages/TaskProgressMessage.js.map +7 -0
  168. package/dist/components/messages/TaskToolMessage.js +39 -0
  169. package/dist/components/messages/TaskToolMessage.js.map +7 -0
  170. package/dist/components/messages/UserBashInputMessage.js +18 -0
  171. package/dist/components/messages/UserBashInputMessage.js.map +7 -0
  172. package/dist/components/messages/UserCommandMessage.js +20 -0
  173. package/dist/components/messages/UserCommandMessage.js.map +7 -0
  174. package/dist/components/messages/UserKodingInputMessage.js +18 -0
  175. package/dist/components/messages/UserKodingInputMessage.js.map +7 -0
  176. package/dist/components/messages/UserPromptMessage.js +20 -0
  177. package/dist/components/messages/UserPromptMessage.js.map +7 -0
  178. package/dist/components/messages/UserTextMessage.js +25 -0
  179. package/dist/components/messages/UserTextMessage.js.map +7 -0
  180. package/dist/components/messages/UserToolResultMessage/UserToolCanceledMessage.js +10 -0
  181. package/dist/components/messages/UserToolResultMessage/UserToolCanceledMessage.js.map +7 -0
  182. package/dist/components/messages/UserToolResultMessage/UserToolErrorMessage.js +15 -0
  183. package/dist/components/messages/UserToolResultMessage/UserToolErrorMessage.js.map +7 -0
  184. package/dist/components/messages/UserToolResultMessage/UserToolRejectMessage.js +25 -0
  185. package/dist/components/messages/UserToolResultMessage/UserToolRejectMessage.js.map +7 -0
  186. package/dist/components/messages/UserToolResultMessage/UserToolResultMessage.js +47 -0
  187. package/dist/components/messages/UserToolResultMessage/UserToolResultMessage.js.map +7 -0
  188. package/dist/components/messages/UserToolResultMessage/UserToolSuccessMessage.js +23 -0
  189. package/dist/components/messages/UserToolResultMessage/UserToolSuccessMessage.js.map +7 -0
  190. package/dist/components/messages/UserToolResultMessage/utils.js +42 -0
  191. package/dist/components/messages/UserToolResultMessage/utils.js.map +7 -0
  192. package/dist/components/permissions/BashPermissionRequest/BashPermissionRequest.js +112 -0
  193. package/dist/components/permissions/BashPermissionRequest/BashPermissionRequest.js.map +7 -0
  194. package/dist/components/permissions/FallbackPermissionRequest.js +131 -0
  195. package/dist/components/permissions/FallbackPermissionRequest.js.map +7 -0
  196. package/dist/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.js +159 -0
  197. package/dist/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.js.map +7 -0
  198. package/dist/components/permissions/FileEditPermissionRequest/FileEditToolDiff.js +58 -0
  199. package/dist/components/permissions/FileEditPermissionRequest/FileEditToolDiff.js.map +7 -0
  200. package/dist/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.js +153 -0
  201. package/dist/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.js.map +7 -0
  202. package/dist/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.js +70 -0
  203. package/dist/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.js.map +7 -0
  204. package/dist/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.js +212 -0
  205. package/dist/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.js.map +7 -0
  206. package/dist/components/permissions/PermissionRequest.js +70 -0
  207. package/dist/components/permissions/PermissionRequest.js.map +7 -0
  208. package/dist/components/permissions/PermissionRequestTitle.js +52 -0
  209. package/dist/components/permissions/PermissionRequestTitle.js.map +7 -0
  210. package/dist/components/permissions/hooks.js +28 -0
  211. package/dist/components/permissions/hooks.js.map +7 -0
  212. package/dist/components/permissions/toolUseOptions.js +46 -0
  213. package/dist/components/permissions/toolUseOptions.js.map +7 -0
  214. package/dist/components/permissions/utils.js +21 -0
  215. package/dist/components/permissions/utils.js.map +7 -0
  216. package/dist/constants/betas.js +11 -0
  217. package/dist/constants/betas.js.map +7 -0
  218. package/dist/constants/claude-asterisk-ascii-art.js +242 -0
  219. package/dist/constants/claude-asterisk-ascii-art.js.map +7 -0
  220. package/dist/constants/figures.js +6 -0
  221. package/dist/constants/figures.js.map +7 -0
  222. package/dist/constants/keys.js +7 -0
  223. package/dist/constants/keys.js.map +7 -0
  224. package/dist/constants/macros.js +13 -0
  225. package/dist/constants/macros.js.map +7 -0
  226. package/dist/constants/modelCapabilities.js +154 -0
  227. package/dist/constants/modelCapabilities.js.map +7 -0
  228. package/dist/constants/models.js +1029 -0
  229. package/dist/constants/models.js.map +7 -0
  230. package/dist/constants/oauth.js +18 -0
  231. package/dist/constants/oauth.js.map +7 -0
  232. package/dist/constants/product.js +26 -0
  233. package/dist/constants/product.js.map +7 -0
  234. package/dist/constants/prompts.js +168 -0
  235. package/dist/constants/prompts.js.map +7 -0
  236. package/dist/constants/releaseNotes.js +9 -0
  237. package/dist/constants/releaseNotes.js.map +7 -0
  238. package/dist/context/PermissionContext.js +111 -0
  239. package/dist/context/PermissionContext.js.map +7 -0
  240. package/dist/context.js +259 -0
  241. package/dist/context.js.map +7 -0
  242. package/dist/cost-tracker.js +76 -0
  243. package/dist/cost-tracker.js.map +7 -0
  244. package/dist/entrypoints/cli.js +1101 -0
  245. package/dist/entrypoints/cli.js.map +7 -0
  246. package/dist/entrypoints/mcp.js +150 -0
  247. package/dist/entrypoints/mcp.js.map +7 -0
  248. package/dist/history.js +25 -0
  249. package/dist/history.js.map +7 -0
  250. package/dist/hooks/useApiKeyVerification.js +12 -0
  251. package/dist/hooks/useApiKeyVerification.js.map +7 -0
  252. package/dist/hooks/useArrowKeyHistory.js +50 -0
  253. package/dist/hooks/useArrowKeyHistory.js.map +7 -0
  254. package/dist/hooks/useCanUseTool.js +112 -0
  255. package/dist/hooks/useCanUseTool.js.map +7 -0
  256. package/dist/hooks/useCancelRequest.js +30 -0
  257. package/dist/hooks/useCancelRequest.js.map +7 -0
  258. package/dist/hooks/useDoublePress.js +31 -0
  259. package/dist/hooks/useDoublePress.js.map +7 -0
  260. package/dist/hooks/useExitOnCtrlCD.js +26 -0
  261. package/dist/hooks/useExitOnCtrlCD.js.map +7 -0
  262. package/dist/hooks/useInterval.js +18 -0
  263. package/dist/hooks/useInterval.js.map +7 -0
  264. package/dist/hooks/useLogMessages.js +14 -0
  265. package/dist/hooks/useLogMessages.js.map +7 -0
  266. package/dist/hooks/useLogStartupTime.js +15 -0
  267. package/dist/hooks/useLogStartupTime.js.map +7 -0
  268. package/dist/hooks/useNotifyAfterTimeout.js +42 -0
  269. package/dist/hooks/useNotifyAfterTimeout.js.map +7 -0
  270. package/dist/hooks/usePermissionRequestLogging.js +28 -0
  271. package/dist/hooks/usePermissionRequestLogging.js.map +7 -0
  272. package/dist/hooks/useTerminalSize.js +38 -0
  273. package/dist/hooks/useTerminalSize.js.map +7 -0
  274. package/dist/hooks/useTextInput.js +250 -0
  275. package/dist/hooks/useTextInput.js.map +7 -0
  276. package/dist/hooks/useUnifiedCompletion.js +929 -0
  277. package/dist/hooks/useUnifiedCompletion.js.map +7 -0
  278. package/dist/index.js +26 -0
  279. package/dist/index.js.map +7 -0
  280. package/dist/messages.js +33 -0
  281. package/dist/messages.js.map +7 -0
  282. package/dist/package.json +1 -0
  283. package/dist/permissions.js +194 -0
  284. package/dist/permissions.js.map +7 -0
  285. package/dist/query.js +492 -0
  286. package/dist/query.js.map +7 -0
  287. package/dist/screens/ConfigureNpmPrefix.js +128 -0
  288. package/dist/screens/ConfigureNpmPrefix.js.map +7 -0
  289. package/dist/screens/Doctor.js +143 -0
  290. package/dist/screens/Doctor.js.map +7 -0
  291. package/dist/screens/LogList.js +55 -0
  292. package/dist/screens/LogList.js.map +7 -0
  293. package/dist/screens/REPL.js +596 -0
  294. package/dist/screens/REPL.js.map +7 -0
  295. package/dist/screens/ResumeConversation.js +56 -0
  296. package/dist/screens/ResumeConversation.js.map +7 -0
  297. package/dist/services/adapters/base.js +29 -0
  298. package/dist/services/adapters/base.js.map +7 -0
  299. package/dist/services/adapters/chatCompletions.js +69 -0
  300. package/dist/services/adapters/chatCompletions.js.map +7 -0
  301. package/dist/services/adapters/responsesAPI.js +126 -0
  302. package/dist/services/adapters/responsesAPI.js.map +7 -0
  303. package/dist/services/browserMocks.js +48 -0
  304. package/dist/services/browserMocks.js.map +7 -0
  305. package/dist/services/claude.js +1605 -0
  306. package/dist/services/claude.js.map +7 -0
  307. package/dist/services/customCommands.js +359 -0
  308. package/dist/services/customCommands.js.map +7 -0
  309. package/dist/services/fileFreshness.js +280 -0
  310. package/dist/services/fileFreshness.js.map +7 -0
  311. package/dist/services/gpt5ConnectionTest.js +248 -0
  312. package/dist/services/gpt5ConnectionTest.js.map +7 -0
  313. package/dist/services/mcpClient.js +435 -0
  314. package/dist/services/mcpClient.js.map +7 -0
  315. package/dist/services/mcpServerApproval.js +55 -0
  316. package/dist/services/mcpServerApproval.js.map +7 -0
  317. package/dist/services/mentionProcessor.js +200 -0
  318. package/dist/services/mentionProcessor.js.map +7 -0
  319. package/dist/services/modelAdapterFactory.js +47 -0
  320. package/dist/services/modelAdapterFactory.js.map +7 -0
  321. package/dist/services/notifier.js +35 -0
  322. package/dist/services/notifier.js.map +7 -0
  323. package/dist/services/oauth.js +259 -0
  324. package/dist/services/oauth.js.map +7 -0
  325. package/dist/services/openai.js +998 -0
  326. package/dist/services/openai.js.map +7 -0
  327. package/dist/services/responseStateManager.js +68 -0
  328. package/dist/services/responseStateManager.js.map +7 -0
  329. package/dist/services/sentry.js +9 -0
  330. package/dist/services/sentry.js.map +7 -0
  331. package/dist/services/statsig.js +112 -0
  332. package/dist/services/statsig.js.map +7 -0
  333. package/dist/services/statsigStorage.js +75 -0
  334. package/dist/services/statsigStorage.js.map +7 -0
  335. package/dist/services/systemReminder.js +353 -0
  336. package/dist/services/systemReminder.js.map +7 -0
  337. package/dist/services/vcr.js +133 -0
  338. package/dist/services/vcr.js.map +7 -0
  339. package/dist/test/testAdapters.js +88 -0
  340. package/dist/test/testAdapters.js.map +1 -0
  341. package/dist/tools/ArchitectTool/ArchitectTool.js +119 -0
  342. package/dist/tools/ArchitectTool/ArchitectTool.js.map +7 -0
  343. package/dist/tools/ArchitectTool/prompt.js +18 -0
  344. package/dist/tools/ArchitectTool/prompt.js.map +7 -0
  345. package/dist/tools/AskExpertModelTool/AskExpertModelTool.js +423 -0
  346. package/dist/tools/AskExpertModelTool/AskExpertModelTool.js.map +7 -0
  347. package/dist/tools/BashTool/BashTool.js +188 -0
  348. package/dist/tools/BashTool/BashTool.js.map +7 -0
  349. package/dist/tools/BashTool/BashToolResultMessage.js +21 -0
  350. package/dist/tools/BashTool/BashToolResultMessage.js.map +7 -0
  351. package/dist/tools/BashTool/OutputLine.js +30 -0
  352. package/dist/tools/BashTool/OutputLine.js.map +7 -0
  353. package/dist/tools/BashTool/prompt.js +179 -0
  354. package/dist/tools/BashTool/prompt.js.map +7 -0
  355. package/dist/tools/BashTool/utils.js +51 -0
  356. package/dist/tools/BashTool/utils.js.map +7 -0
  357. package/dist/tools/FileEditTool/FileEditTool.js +228 -0
  358. package/dist/tools/FileEditTool/FileEditTool.js.map +7 -0
  359. package/dist/tools/FileEditTool/prompt.js +54 -0
  360. package/dist/tools/FileEditTool/prompt.js.map +7 -0
  361. package/dist/tools/FileEditTool/utils.js +42 -0
  362. package/dist/tools/FileEditTool/utils.js.map +7 -0
  363. package/dist/tools/FileReadTool/FileReadTool.js +272 -0
  364. package/dist/tools/FileReadTool/FileReadTool.js.map +7 -0
  365. package/dist/tools/FileReadTool/prompt.js +10 -0
  366. package/dist/tools/FileReadTool/prompt.js.map +7 -0
  367. package/dist/tools/FileWriteTool/FileWriteTool.js +204 -0
  368. package/dist/tools/FileWriteTool/FileWriteTool.js.map +7 -0
  369. package/dist/tools/FileWriteTool/prompt.js +14 -0
  370. package/dist/tools/FileWriteTool/prompt.js.map +7 -0
  371. package/dist/tools/GlobTool/GlobTool.js +88 -0
  372. package/dist/tools/GlobTool/GlobTool.js.map +7 -0
  373. package/dist/tools/GlobTool/prompt.js +12 -0
  374. package/dist/tools/GlobTool/prompt.js.map +7 -0
  375. package/dist/tools/GrepTool/GrepTool.js +107 -0
  376. package/dist/tools/GrepTool/GrepTool.js.map +7 -0
  377. package/dist/tools/GrepTool/prompt.js +15 -0
  378. package/dist/tools/GrepTool/prompt.js.map +7 -0
  379. package/dist/tools/MCPTool/MCPTool.js +90 -0
  380. package/dist/tools/MCPTool/MCPTool.js.map +7 -0
  381. package/dist/tools/MCPTool/prompt.js +7 -0
  382. package/dist/tools/MCPTool/prompt.js.map +7 -0
  383. package/dist/tools/MemoryReadTool/MemoryReadTool.js +103 -0
  384. package/dist/tools/MemoryReadTool/MemoryReadTool.js.map +7 -0
  385. package/dist/tools/MemoryReadTool/prompt.js +7 -0
  386. package/dist/tools/MemoryReadTool/prompt.js.map +7 -0
  387. package/dist/tools/MemoryWriteTool/MemoryWriteTool.js +77 -0
  388. package/dist/tools/MemoryWriteTool/MemoryWriteTool.js.map +7 -0
  389. package/dist/tools/MemoryWriteTool/prompt.js +7 -0
  390. package/dist/tools/MemoryWriteTool/prompt.js.map +7 -0
  391. package/dist/tools/MultiEditTool/MultiEditTool.js +293 -0
  392. package/dist/tools/MultiEditTool/MultiEditTool.js.map +7 -0
  393. package/dist/tools/MultiEditTool/prompt.js +48 -0
  394. package/dist/tools/MultiEditTool/prompt.js.map +7 -0
  395. package/dist/tools/NotebookEditTool/NotebookEditTool.js +238 -0
  396. package/dist/tools/NotebookEditTool/NotebookEditTool.js.map +7 -0
  397. package/dist/tools/NotebookEditTool/prompt.js +7 -0
  398. package/dist/tools/NotebookEditTool/prompt.js.map +7 -0
  399. package/dist/tools/NotebookReadTool/NotebookReadTool.js +212 -0
  400. package/dist/tools/NotebookReadTool/NotebookReadTool.js.map +7 -0
  401. package/dist/tools/NotebookReadTool/prompt.js +7 -0
  402. package/dist/tools/NotebookReadTool/prompt.js.map +7 -0
  403. package/dist/tools/StickerRequestTool/StickerRequestTool.js +86 -0
  404. package/dist/tools/StickerRequestTool/StickerRequestTool.js.map +7 -0
  405. package/dist/tools/StickerRequestTool/prompt.js +23 -0
  406. package/dist/tools/StickerRequestTool/prompt.js.map +7 -0
  407. package/dist/tools/TaskTool/TaskTool.js +308 -0
  408. package/dist/tools/TaskTool/TaskTool.js.map +7 -0
  409. package/dist/tools/TaskTool/constants.js +5 -0
  410. package/dist/tools/TaskTool/constants.js.map +7 -0
  411. package/dist/tools/TaskTool/prompt.js +82 -0
  412. package/dist/tools/TaskTool/prompt.js.map +7 -0
  413. package/dist/tools/ThinkTool/ThinkTool.js +48 -0
  414. package/dist/tools/ThinkTool/ThinkTool.js.map +7 -0
  415. package/dist/tools/ThinkTool/prompt.js +16 -0
  416. package/dist/tools/ThinkTool/prompt.js.map +7 -0
  417. package/dist/tools/TodoWriteTool/TodoWriteTool.js +216 -0
  418. package/dist/tools/TodoWriteTool/TodoWriteTool.js.map +7 -0
  419. package/dist/tools/TodoWriteTool/prompt.js +66 -0
  420. package/dist/tools/TodoWriteTool/prompt.js.map +7 -0
  421. package/dist/tools/URLFetcherTool/URLFetcherTool.js +137 -0
  422. package/dist/tools/URLFetcherTool/URLFetcherTool.js.map +7 -0
  423. package/dist/tools/URLFetcherTool/cache.js +45 -0
  424. package/dist/tools/URLFetcherTool/cache.js.map +7 -0
  425. package/dist/tools/URLFetcherTool/htmlToMarkdown.js +42 -0
  426. package/dist/tools/URLFetcherTool/htmlToMarkdown.js.map +7 -0
  427. package/dist/tools/URLFetcherTool/prompt.js +22 -0
  428. package/dist/tools/URLFetcherTool/prompt.js.map +7 -0
  429. package/dist/tools/WebSearchTool/WebSearchTool.js +86 -0
  430. package/dist/tools/WebSearchTool/WebSearchTool.js.map +7 -0
  431. package/dist/tools/WebSearchTool/prompt.js +17 -0
  432. package/dist/tools/WebSearchTool/prompt.js.map +7 -0
  433. package/dist/tools/WebSearchTool/searchProviders.js +48 -0
  434. package/dist/tools/WebSearchTool/searchProviders.js.map +7 -0
  435. package/dist/tools/lsTool/lsTool.js +201 -0
  436. package/dist/tools/lsTool/lsTool.js.map +7 -0
  437. package/dist/tools/lsTool/prompt.js +5 -0
  438. package/dist/tools/lsTool/prompt.js.map +7 -0
  439. package/dist/tools.js +64 -0
  440. package/dist/tools.js.map +7 -0
  441. package/dist/types/PermissionMode.js +82 -0
  442. package/dist/types/PermissionMode.js.map +7 -0
  443. package/dist/types/RequestContext.js +47 -0
  444. package/dist/types/RequestContext.js.map +7 -0
  445. package/dist/types/common.d.js +1 -0
  446. package/dist/types/common.d.js.map +7 -0
  447. package/dist/types/conversation.js +1 -0
  448. package/dist/types/conversation.js.map +7 -0
  449. package/dist/types/logs.js +1 -0
  450. package/dist/types/logs.js.map +7 -0
  451. package/dist/types/modelCapabilities.js +1 -0
  452. package/dist/types/modelCapabilities.js.map +7 -0
  453. package/dist/types/notebook.js +1 -0
  454. package/dist/types/notebook.js.map +7 -0
  455. package/dist/utils/Cursor.js +313 -0
  456. package/dist/utils/Cursor.js.map +7 -0
  457. package/dist/utils/PersistentShell.js +382 -0
  458. package/dist/utils/PersistentShell.js.map +7 -0
  459. package/dist/utils/advancedFuzzyMatcher.js +206 -0
  460. package/dist/utils/advancedFuzzyMatcher.js.map +7 -0
  461. package/dist/utils/agentLoader.js +199 -0
  462. package/dist/utils/agentLoader.js.map +7 -0
  463. package/dist/utils/agentStorage.js +59 -0
  464. package/dist/utils/agentStorage.js.map +7 -0
  465. package/dist/utils/array.js +7 -0
  466. package/dist/utils/array.js.map +7 -0
  467. package/dist/utils/ask.js +77 -0
  468. package/dist/utils/ask.js.map +7 -0
  469. package/dist/utils/auth.js +11 -0
  470. package/dist/utils/auth.js.map +7 -0
  471. package/dist/utils/autoCompactCore.js +149 -0
  472. package/dist/utils/autoCompactCore.js.map +7 -0
  473. package/dist/utils/autoUpdater.js +362 -0
  474. package/dist/utils/autoUpdater.js.map +7 -0
  475. package/dist/utils/betas.js +21 -0
  476. package/dist/utils/betas.js.map +7 -0
  477. package/dist/utils/browser.js +15 -0
  478. package/dist/utils/browser.js.map +7 -0
  479. package/dist/utils/cleanup.js +54 -0
  480. package/dist/utils/cleanup.js.map +7 -0
  481. package/dist/utils/commands.js +207 -0
  482. package/dist/utils/commands.js.map +7 -0
  483. package/dist/utils/commonUnixCommands.js +687 -0
  484. package/dist/utils/commonUnixCommands.js.map +7 -0
  485. package/dist/utils/config.js +655 -0
  486. package/dist/utils/config.js.map +7 -0
  487. package/dist/utils/conversationRecovery.js +35 -0
  488. package/dist/utils/conversationRecovery.js.map +7 -0
  489. package/dist/utils/debugLogger.js +891 -0
  490. package/dist/utils/debugLogger.js.map +7 -0
  491. package/dist/utils/diff.js +32 -0
  492. package/dist/utils/diff.js.map +7 -0
  493. package/dist/utils/env.js +44 -0
  494. package/dist/utils/env.js.map +7 -0
  495. package/dist/utils/errors.js +23 -0
  496. package/dist/utils/errors.js.map +7 -0
  497. package/dist/utils/exampleCommands.js +80 -0
  498. package/dist/utils/exampleCommands.js.map +7 -0
  499. package/dist/utils/execFileNoThrow.js +44 -0
  500. package/dist/utils/execFileNoThrow.js.map +7 -0
  501. package/dist/utils/expertChatStorage.js +78 -0
  502. package/dist/utils/expertChatStorage.js.map +7 -0
  503. package/dist/utils/file.js +282 -0
  504. package/dist/utils/file.js.map +7 -0
  505. package/dist/utils/fileRecoveryCore.js +41 -0
  506. package/dist/utils/fileRecoveryCore.js.map +7 -0
  507. package/dist/utils/format.js +41 -0
  508. package/dist/utils/format.js.map +7 -0
  509. package/dist/utils/fuzzyMatcher.js +252 -0
  510. package/dist/utils/fuzzyMatcher.js.map +7 -0
  511. package/dist/utils/generators.js +46 -0
  512. package/dist/utils/generators.js.map +7 -0
  513. package/dist/utils/git.js +83 -0
  514. package/dist/utils/git.js.map +7 -0
  515. package/dist/utils/globalLogger.js +54 -0
  516. package/dist/utils/globalLogger.js.map +7 -0
  517. package/dist/utils/http.js +7 -0
  518. package/dist/utils/http.js.map +7 -0
  519. package/dist/utils/imagePaste.js +29 -0
  520. package/dist/utils/imagePaste.js.map +7 -0
  521. package/dist/utils/json.js +16 -0
  522. package/dist/utils/json.js.map +7 -0
  523. package/dist/utils/log.js +298 -0
  524. package/dist/utils/log.js.map +7 -0
  525. package/dist/utils/markdown.js +187 -0
  526. package/dist/utils/markdown.js.map +7 -0
  527. package/dist/utils/messageContextManager.js +195 -0
  528. package/dist/utils/messageContextManager.js.map +7 -0
  529. package/dist/utils/messages.js +633 -0
  530. package/dist/utils/messages.js.map +7 -0
  531. package/dist/utils/model.js +687 -0
  532. package/dist/utils/model.js.map +7 -0
  533. package/dist/utils/permissions/filesystem.js +80 -0
  534. package/dist/utils/permissions/filesystem.js.map +7 -0
  535. package/dist/utils/responseState.js +20 -0
  536. package/dist/utils/responseState.js.map +7 -0
  537. package/dist/utils/ripgrep.js +131 -0
  538. package/dist/utils/ripgrep.js.map +7 -0
  539. package/dist/utils/secureFile.js +483 -0
  540. package/dist/utils/secureFile.js.map +7 -0
  541. package/dist/utils/sessionState.js +31 -0
  542. package/dist/utils/sessionState.js.map +7 -0
  543. package/dist/utils/state.js +24 -0
  544. package/dist/utils/state.js.map +7 -0
  545. package/dist/utils/style.js +31 -0
  546. package/dist/utils/style.js.map +7 -0
  547. package/dist/utils/terminal.js +43 -0
  548. package/dist/utils/terminal.js.map +7 -0
  549. package/dist/utils/theme.js +102 -0
  550. package/dist/utils/theme.js.map +7 -0
  551. package/dist/utils/thinking.js +103 -0
  552. package/dist/utils/thinking.js.map +7 -0
  553. package/dist/utils/todoStorage.js +291 -0
  554. package/dist/utils/todoStorage.js.map +7 -0
  555. package/dist/utils/tokens.js +30 -0
  556. package/dist/utils/tokens.js.map +7 -0
  557. package/dist/utils/toolExecutionController.js +109 -0
  558. package/dist/utils/toolExecutionController.js.map +7 -0
  559. package/dist/utils/unaryLogging.js +14 -0
  560. package/dist/utils/unaryLogging.js.map +7 -0
  561. package/dist/utils/user.js +40 -0
  562. package/dist/utils/user.js.map +7 -0
  563. package/dist/utils/validate.js +132 -0
  564. package/dist/utils/validate.js.map +7 -0
  565. package/dist/yoga.wasm +0 -0
  566. package/package.json +28 -7
  567. package/src/Tool.ts +4 -3
  568. package/src/commands/agents.tsx +10 -4
  569. package/src/components/messages/AssistantToolUseMessage.tsx +5 -6
  570. package/src/constants/macros.ts +5 -2
  571. package/src/entrypoints/cli.tsx +38 -19
  572. package/src/entrypoints/mcp.ts +1 -2
  573. package/src/hooks/useDoublePress.ts +0 -1
  574. package/src/hooks/useTextInput.ts +4 -5
  575. package/src/hooks/useUnifiedCompletion.ts +2 -2
  576. package/src/index.ts +34 -0
  577. package/src/query.ts +13 -8
  578. package/src/screens/Doctor.tsx +1 -1
  579. package/src/screens/REPL.tsx +13 -9
  580. package/src/services/openai.ts +25 -4
  581. package/src/tools/ArchitectTool/ArchitectTool.tsx +18 -5
  582. package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +21 -14
  583. package/src/tools/FileEditTool/FileEditTool.tsx +6 -2
  584. package/src/tools/FileWriteTool/FileWriteTool.tsx +7 -3
  585. package/src/tools/MultiEditTool/MultiEditTool.tsx +26 -4
  586. package/src/tools/NotebookReadTool/NotebookReadTool.tsx +1 -1
  587. package/src/tools/StickerRequestTool/StickerRequestTool.tsx +28 -14
  588. package/src/tools/TaskTool/TaskTool.tsx +8 -36
  589. package/src/types/common.d.ts +2 -0
  590. package/src/utils/generators.ts +1 -1
  591. package/src/utils/messageContextManager.ts +5 -0
  592. package/src/utils/messages.tsx +8 -2
  593. package/src/utils/thinking.ts +1 -1
@@ -0,0 +1,483 @@
1
+ import { existsSync, readFileSync, writeFileSync, mkdirSync, statSync, unlinkSync, renameSync } from "node:fs";
2
+ import { dirname, normalize, resolve, extname, relative, isAbsolute } from "node:path";
3
+ import { homedir } from "node:os";
4
+ class SecureFileService {
5
+ static instance;
6
+ allowedBasePaths;
7
+ maxFileSize;
8
+ allowedExtensions;
9
+ constructor() {
10
+ this.allowedBasePaths = /* @__PURE__ */ new Set([
11
+ process.cwd(),
12
+ homedir(),
13
+ "/tmp",
14
+ "/var/tmp"
15
+ ]);
16
+ this.maxFileSize = 10 * 1024 * 1024;
17
+ this.allowedExtensions = /* @__PURE__ */ new Set([
18
+ ".txt",
19
+ ".md",
20
+ ".json",
21
+ ".js",
22
+ ".ts",
23
+ ".tsx",
24
+ ".jsx",
25
+ ".yaml",
26
+ ".yml",
27
+ ".toml",
28
+ ".ini",
29
+ ".env",
30
+ ".log",
31
+ ".html",
32
+ ".css",
33
+ ".scss",
34
+ ".less",
35
+ ".xml",
36
+ ".csv",
37
+ ".py",
38
+ ".go",
39
+ ".rs",
40
+ ".java",
41
+ ".cpp",
42
+ ".c",
43
+ ".h",
44
+ ".sh",
45
+ ".bash",
46
+ ".zsh",
47
+ ".fish",
48
+ ".ps1",
49
+ ".bat",
50
+ ".dockerfile",
51
+ ".gitignore",
52
+ ".npmignore",
53
+ ".eslintignore"
54
+ ]);
55
+ }
56
+ static getInstance() {
57
+ if (!SecureFileService.instance) {
58
+ SecureFileService.instance = new SecureFileService();
59
+ }
60
+ return SecureFileService.instance;
61
+ }
62
+ /**
63
+ * 验证文件路径是否安全
64
+ * @param filePath 文件路径
65
+ * @returns 验证结果
66
+ */
67
+ validateFilePath(filePath) {
68
+ try {
69
+ const normalizedPath = normalize(filePath);
70
+ if (normalizedPath.length > 4096) {
71
+ return {
72
+ isValid: false,
73
+ normalizedPath,
74
+ error: "Path too long (max 4096 characters)"
75
+ };
76
+ }
77
+ if (normalizedPath.includes("..") || normalizedPath.includes("~")) {
78
+ return {
79
+ isValid: false,
80
+ normalizedPath,
81
+ error: "Path contains traversal characters"
82
+ };
83
+ }
84
+ const suspiciousPatterns = [
85
+ /\.\./,
86
+ // 父目录
87
+ /~/,
88
+ // 用户目录
89
+ /\$\{/,
90
+ // 环境变量
91
+ /`/,
92
+ // 命令执行
93
+ /\|/,
94
+ // 管道符
95
+ /;/,
96
+ // 命令分隔符
97
+ /&/,
98
+ // 后台执行
99
+ />/,
100
+ // 输出重定向
101
+ /</
102
+ // 输入重定向
103
+ ];
104
+ for (const pattern of suspiciousPatterns) {
105
+ if (pattern.test(normalizedPath)) {
106
+ return {
107
+ isValid: false,
108
+ normalizedPath,
109
+ error: `Path contains suspicious pattern: ${pattern}`
110
+ };
111
+ }
112
+ }
113
+ const absolutePath = resolve(normalizedPath);
114
+ const isInAllowedPath = Array.from(this.allowedBasePaths).some((basePath) => {
115
+ const base = resolve(basePath);
116
+ const rel = relative(base, absolutePath);
117
+ if (!rel || rel === "") return true;
118
+ if (rel.startsWith("..")) return false;
119
+ if (isAbsolute(rel)) return false;
120
+ return true;
121
+ });
122
+ if (!isInAllowedPath) {
123
+ return {
124
+ isValid: false,
125
+ normalizedPath,
126
+ error: "Path is outside allowed directories"
127
+ };
128
+ }
129
+ return { isValid: true, normalizedPath: absolutePath };
130
+ } catch (error) {
131
+ return {
132
+ isValid: false,
133
+ normalizedPath: filePath,
134
+ error: `Path validation failed: ${error instanceof Error ? error.message : String(error)}`
135
+ };
136
+ }
137
+ }
138
+ /**
139
+ * 安全地检查文件是否存在
140
+ * @param filePath 文件路径
141
+ * @returns 文件是否存在
142
+ */
143
+ safeExists(filePath) {
144
+ const validation = this.validateFilePath(filePath);
145
+ if (!validation.isValid) {
146
+ return false;
147
+ }
148
+ try {
149
+ return existsSync(validation.normalizedPath);
150
+ } catch (error) {
151
+ return false;
152
+ }
153
+ }
154
+ /**
155
+ * 安全地读取文件
156
+ * @param filePath 文件路径
157
+ * @param options 读取选项
158
+ * @returns 读取结果
159
+ */
160
+ safeReadFile(filePath, options = {}) {
161
+ const validation = this.validateFilePath(filePath);
162
+ if (!validation.isValid) {
163
+ return { success: false, error: validation.error };
164
+ }
165
+ try {
166
+ const normalizedPath = validation.normalizedPath;
167
+ if (options.checkFileExtension !== false) {
168
+ const ext = extname(normalizedPath).toLowerCase();
169
+ const allowedExts = options.allowedExtensions || Array.from(this.allowedExtensions);
170
+ if (allowedExts.length > 0 && !allowedExts.includes(ext)) {
171
+ return {
172
+ success: false,
173
+ error: `File extension '${ext}' is not allowed`
174
+ };
175
+ }
176
+ }
177
+ if (!existsSync(normalizedPath)) {
178
+ return { success: false, error: "File does not exist" };
179
+ }
180
+ const stats = statSync(normalizedPath);
181
+ const maxSize = options.maxFileSize || this.maxFileSize;
182
+ if (stats.size > maxSize) {
183
+ return {
184
+ success: false,
185
+ error: `File too large (${stats.size} bytes, max ${maxSize} bytes)`
186
+ };
187
+ }
188
+ if (!stats.isFile()) {
189
+ return { success: false, error: "Path is not a file" };
190
+ }
191
+ if ((stats.mode & parseInt("400", 8)) === 0) {
192
+ return { success: false, error: "No read permission" };
193
+ }
194
+ const content = readFileSync(normalizedPath, {
195
+ encoding: options.encoding || "utf8"
196
+ });
197
+ return {
198
+ success: true,
199
+ content,
200
+ stats: {
201
+ size: stats.size,
202
+ mtime: stats.mtime,
203
+ atime: stats.atime,
204
+ mode: stats.mode
205
+ }
206
+ };
207
+ } catch (error) {
208
+ return {
209
+ success: false,
210
+ error: `Failed to read file: ${error instanceof Error ? error.message : String(error)}`
211
+ };
212
+ }
213
+ }
214
+ /**
215
+ * 安全地写入文件
216
+ * @param filePath 文件路径
217
+ * @param content 文件内容
218
+ * @param options 写入选项
219
+ * @returns 写入结果
220
+ */
221
+ safeWriteFile(filePath, content, options = {}) {
222
+ const validation = this.validateFilePath(filePath);
223
+ if (!validation.isValid) {
224
+ return { success: false, error: validation.error };
225
+ }
226
+ try {
227
+ const normalizedPath = validation.normalizedPath;
228
+ if (options.checkFileExtension !== false) {
229
+ const ext = extname(normalizedPath).toLowerCase();
230
+ const allowedExts = options.allowedExtensions || Array.from(this.allowedExtensions);
231
+ if (allowedExts.length > 0 && !allowedExts.includes(ext)) {
232
+ return {
233
+ success: false,
234
+ error: `File extension '${ext}' is not allowed`
235
+ };
236
+ }
237
+ }
238
+ const contentSize = typeof content === "string" ? Buffer.byteLength(content, options.encoding || "utf8") : content.length;
239
+ const maxSize = options.maxSize || this.maxFileSize;
240
+ if (contentSize > maxSize) {
241
+ return {
242
+ success: false,
243
+ error: `Content too large (${contentSize} bytes, max ${maxSize} bytes)`
244
+ };
245
+ }
246
+ if (options.createDirectory) {
247
+ const dir = dirname(normalizedPath);
248
+ if (!existsSync(dir)) {
249
+ mkdirSync(dir, { recursive: true, mode: 493 });
250
+ }
251
+ }
252
+ if (options.atomic) {
253
+ const tempPath = `${normalizedPath}.tmp.${Date.now()}`;
254
+ try {
255
+ writeFileSync(tempPath, content, {
256
+ encoding: options.encoding || "utf8",
257
+ mode: options.mode || 420
258
+ });
259
+ renameSync(tempPath, normalizedPath);
260
+ } catch (renameError) {
261
+ try {
262
+ if (existsSync(tempPath)) {
263
+ unlinkSync(tempPath);
264
+ }
265
+ } catch {
266
+ }
267
+ throw renameError;
268
+ }
269
+ } else {
270
+ writeFileSync(normalizedPath, content, {
271
+ encoding: options.encoding || "utf8",
272
+ mode: options.mode || 420
273
+ });
274
+ }
275
+ return { success: true };
276
+ } catch (error) {
277
+ return {
278
+ success: false,
279
+ error: `Failed to write file: ${error instanceof Error ? error.message : String(error)}`
280
+ };
281
+ }
282
+ }
283
+ /**
284
+ * 安全地删除文件
285
+ * @param filePath 文件路径
286
+ * @returns 删除结果
287
+ */
288
+ safeDeleteFile(filePath) {
289
+ const validation = this.validateFilePath(filePath);
290
+ if (!validation.isValid) {
291
+ return { success: false, error: validation.error };
292
+ }
293
+ try {
294
+ const normalizedPath = validation.normalizedPath;
295
+ if (!existsSync(normalizedPath)) {
296
+ return { success: false, error: "File does not exist" };
297
+ }
298
+ const stats = statSync(normalizedPath);
299
+ if (!stats.isFile()) {
300
+ return { success: false, error: "Path is not a file" };
301
+ }
302
+ if ((stats.mode & parseInt("200", 8)) === 0) {
303
+ return { success: false, error: "No write permission" };
304
+ }
305
+ unlinkSync(normalizedPath);
306
+ return { success: true };
307
+ } catch (error) {
308
+ return {
309
+ success: false,
310
+ error: `Failed to delete file: ${error instanceof Error ? error.message : String(error)}`
311
+ };
312
+ }
313
+ }
314
+ /**
315
+ * 安全地创建目录
316
+ * @param dirPath 目录路径
317
+ * @param mode 目录权限
318
+ * @returns 创建结果
319
+ */
320
+ safeCreateDirectory(dirPath, mode = 493) {
321
+ const validation = this.validateFilePath(dirPath);
322
+ if (!validation.isValid) {
323
+ return { success: false, error: validation.error };
324
+ }
325
+ try {
326
+ const normalizedPath = validation.normalizedPath;
327
+ if (existsSync(normalizedPath)) {
328
+ const stats = statSync(normalizedPath);
329
+ if (!stats.isDirectory()) {
330
+ return { success: false, error: "Path already exists and is not a directory" };
331
+ }
332
+ return { success: true };
333
+ }
334
+ mkdirSync(normalizedPath, { recursive: true, mode });
335
+ return { success: true };
336
+ } catch (error) {
337
+ return {
338
+ success: false,
339
+ error: `Failed to create directory: ${error instanceof Error ? error.message : String(error)}`
340
+ };
341
+ }
342
+ }
343
+ /**
344
+ * 安全地获取文件信息
345
+ * @param filePath 文件路径
346
+ * @returns 文件信息
347
+ */
348
+ safeGetFileInfo(filePath) {
349
+ const validation = this.validateFilePath(filePath);
350
+ if (!validation.isValid) {
351
+ return { success: false, error: validation.error };
352
+ }
353
+ try {
354
+ const normalizedPath = validation.normalizedPath;
355
+ if (!existsSync(normalizedPath)) {
356
+ return { success: false, error: "File does not exist" };
357
+ }
358
+ const stats = statSync(normalizedPath);
359
+ return {
360
+ success: true,
361
+ stats: {
362
+ size: stats.size,
363
+ isFile: stats.isFile(),
364
+ isDirectory: stats.isDirectory(),
365
+ mode: stats.mode,
366
+ atime: stats.atime,
367
+ mtime: stats.mtime,
368
+ ctime: stats.ctime
369
+ }
370
+ };
371
+ } catch (error) {
372
+ return {
373
+ success: false,
374
+ error: `Failed to get file info: ${error instanceof Error ? error.message : String(error)}`
375
+ };
376
+ }
377
+ }
378
+ /**
379
+ * 添加允许的基础路径
380
+ * @param basePath 基础路径
381
+ */
382
+ addAllowedBasePath(basePath) {
383
+ try {
384
+ const normalized = normalize(resolve(basePath));
385
+ if (!existsSync(normalized)) {
386
+ return { success: false, error: "Base path does not exist" };
387
+ }
388
+ this.allowedBasePaths.add(normalized);
389
+ return { success: true };
390
+ } catch (error) {
391
+ return {
392
+ success: false,
393
+ error: `Failed to add base path: ${error instanceof Error ? error.message : String(error)}`
394
+ };
395
+ }
396
+ }
397
+ /**
398
+ * 设置最大文件大小
399
+ * @param maxSize 最大文件大小(字节)
400
+ */
401
+ setMaxFileSize(maxSize) {
402
+ this.maxFileSize = maxSize;
403
+ }
404
+ /**
405
+ * 添加允许的文件扩展名
406
+ * @param extensions 文件扩展名数组
407
+ */
408
+ addAllowedExtensions(extensions) {
409
+ extensions.forEach((ext) => {
410
+ if (!ext.startsWith(".")) {
411
+ ext = "." + ext;
412
+ }
413
+ this.allowedExtensions.add(ext.toLowerCase());
414
+ });
415
+ }
416
+ /**
417
+ * 检查文件是否在允许的基础路径中
418
+ * @param filePath 文件路径
419
+ * @returns 是否允许
420
+ */
421
+ isPathAllowed(filePath) {
422
+ const validation = this.validateFilePath(filePath);
423
+ return validation.isValid;
424
+ }
425
+ /**
426
+ * 验证文件名安全性
427
+ * @param filename 文件名
428
+ * @returns 验证结果
429
+ */
430
+ validateFileName(filename) {
431
+ if (filename.length === 0) {
432
+ return { isValid: false, error: "Filename cannot be empty" };
433
+ }
434
+ if (filename.length > 255) {
435
+ return { isValid: false, error: "Filename too long (max 255 characters)" };
436
+ }
437
+ const invalidChars = /[<>:"/\\|?*\x00-\x1F]/;
438
+ if (invalidChars.test(filename)) {
439
+ return { isValid: false, error: "Filename contains invalid characters" };
440
+ }
441
+ const reservedNames = [
442
+ "CON",
443
+ "PRN",
444
+ "AUX",
445
+ "NUL",
446
+ "COM1",
447
+ "COM2",
448
+ "COM3",
449
+ "COM4",
450
+ "COM5",
451
+ "COM6",
452
+ "COM7",
453
+ "COM8",
454
+ "COM9",
455
+ "LPT1",
456
+ "LPT2",
457
+ "LPT3",
458
+ "LPT4",
459
+ "LPT5",
460
+ "LPT6",
461
+ "LPT7",
462
+ "LPT8",
463
+ "LPT9"
464
+ ];
465
+ const baseName = filename.split(".")[0].toUpperCase();
466
+ if (reservedNames.includes(baseName)) {
467
+ return { isValid: false, error: "Filename is reserved" };
468
+ }
469
+ if (filename.startsWith(".") || filename.endsWith(".")) {
470
+ return { isValid: false, error: "Filename cannot start or end with a dot" };
471
+ }
472
+ if (filename.startsWith(" ") || filename.endsWith(" ")) {
473
+ return { isValid: false, error: "Filename cannot start or end with spaces" };
474
+ }
475
+ return { isValid: true };
476
+ }
477
+ }
478
+ const secureFileService = SecureFileService.getInstance();
479
+ export {
480
+ SecureFileService,
481
+ secureFileService
482
+ };
483
+ //# sourceMappingURL=secureFile.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/utils/secureFile.ts"],
4
+ "sourcesContent": ["import { existsSync, readFileSync, writeFileSync, mkdirSync, statSync, unlinkSync, renameSync } from 'node:fs'\nimport { join, dirname, normalize, resolve, extname, relative, isAbsolute } from 'node:path'\nimport { homedir } from 'node:os'\n\n/**\n * \u5B89\u5168\u6587\u4EF6\u7CFB\u7EDF\u64CD\u4F5C\u670D\u52A1\n * \u89E3\u51B3\u6587\u4EF6\u7CFB\u7EDF\u64CD\u4F5C\u4E2D\u7F3A\u5C11\u9002\u5F53\u9A8C\u8BC1\u548C\u9519\u8BEF\u5904\u7406\u7684\u95EE\u9898\n */\nexport class SecureFileService {\n private static instance: SecureFileService\n private allowedBasePaths: Set<string>\n private maxFileSize: number\n private allowedExtensions: Set<string>\n\n private constructor() {\n // \u5141\u8BB8\u7684\u57FA\u7840\u8DEF\u5F84\n this.allowedBasePaths = new Set([\n process.cwd(),\n homedir(),\n '/tmp',\n '/var/tmp'\n ])\n \n // \u9ED8\u8BA4\u6700\u5927\u6587\u4EF6\u5927\u5C0F (10MB)\n this.maxFileSize = 10 * 1024 * 1024\n \n // \u5141\u8BB8\u7684\u6587\u4EF6\u6269\u5C55\u540D\n this.allowedExtensions = new Set([\n '.txt', '.md', '.json', '.js', '.ts', '.tsx', '.jsx',\n '.yaml', '.yml', '.toml', '.ini', '.env', '.log',\n '.html', '.css', '.scss', '.less', '.xml', '.csv',\n '.py', '.go', '.rs', '.java', '.cpp', '.c', '.h',\n '.sh', '.bash', '.zsh', '.fish', '.ps1', '.bat',\n '.dockerfile', '.gitignore', '.npmignore', '.eslintignore'\n ])\n }\n\n public static getInstance(): SecureFileService {\n if (!SecureFileService.instance) {\n SecureFileService.instance = new SecureFileService()\n }\n return SecureFileService.instance\n }\n\n /**\n * \u9A8C\u8BC1\u6587\u4EF6\u8DEF\u5F84\u662F\u5426\u5B89\u5168\n * @param filePath \u6587\u4EF6\u8DEF\u5F84\n * @returns \u9A8C\u8BC1\u7ED3\u679C\n */\n public validateFilePath(filePath: string): { isValid: boolean; normalizedPath: string; error?: string } {\n try {\n // \u89C4\u8303\u5316\u8DEF\u5F84\n const normalizedPath = normalize(filePath)\n \n // \u68C0\u67E5\u8DEF\u5F84\u957F\u5EA6\n if (normalizedPath.length > 4096) {\n return {\n isValid: false,\n normalizedPath,\n error: 'Path too long (max 4096 characters)'\n }\n }\n\n // \u68C0\u67E5\u662F\u5426\u5305\u542B\u8DEF\u5F84\u904D\u5386\u5B57\u7B26\n if (normalizedPath.includes('..') || normalizedPath.includes('~')) {\n return {\n isValid: false,\n normalizedPath,\n error: 'Path contains traversal characters'\n }\n }\n\n // \u68C0\u67E5\u662F\u5426\u5305\u542B\u53EF\u7591\u7684\u5B57\u7B26\u5E8F\u5217\n const suspiciousPatterns = [\n /\\.\\./, // \u7236\u76EE\u5F55\n /~/, // \u7528\u6237\u76EE\u5F55\n /\\$\\{/, // \u73AF\u5883\u53D8\u91CF\n /`/, // \u547D\u4EE4\u6267\u884C\n /\\|/, // \u7BA1\u9053\u7B26\n /;/, // \u547D\u4EE4\u5206\u9694\u7B26\n /&/, // \u540E\u53F0\u6267\u884C\n />/, // \u8F93\u51FA\u91CD\u5B9A\u5411\n /</, // \u8F93\u5165\u91CD\u5B9A\u5411\n ]\n\n for (const pattern of suspiciousPatterns) {\n if (pattern.test(normalizedPath)) {\n return {\n isValid: false,\n normalizedPath,\n error: `Path contains suspicious pattern: ${pattern}`\n }\n }\n }\n\n // \u89E3\u6790\u4E3A\u7EDD\u5BF9\u8DEF\u5F84\n const absolutePath = resolve(normalizedPath)\n \n // \u68C0\u67E5\u662F\u5426\u5728\u5141\u8BB8\u7684\u57FA\u7840\u8DEF\u5F84\u4E2D\n const isInAllowedPath = Array.from(this.allowedBasePaths).some(basePath => {\n const base = resolve(basePath)\n const rel = relative(base, absolutePath)\n if (!rel || rel === '') return true\n if (rel.startsWith('..')) return false\n if (isAbsolute(rel)) return false\n return true\n })\n\n if (!isInAllowedPath) {\n return {\n isValid: false,\n normalizedPath,\n error: 'Path is outside allowed directories'\n }\n }\n\n return { isValid: true, normalizedPath: absolutePath }\n } catch (error) {\n return {\n isValid: false,\n normalizedPath: filePath,\n error: `Path validation failed: ${error instanceof Error ? error.message : String(error)}`\n }\n }\n }\n\n /**\n * \u5B89\u5168\u5730\u68C0\u67E5\u6587\u4EF6\u662F\u5426\u5B58\u5728\n * @param filePath \u6587\u4EF6\u8DEF\u5F84\n * @returns \u6587\u4EF6\u662F\u5426\u5B58\u5728\n */\n public safeExists(filePath: string): boolean {\n const validation = this.validateFilePath(filePath)\n if (!validation.isValid) {\n return false\n }\n\n try {\n return existsSync(validation.normalizedPath)\n } catch (error) {\n return false\n }\n }\n\n /**\n * \u5B89\u5168\u5730\u8BFB\u53D6\u6587\u4EF6\n * @param filePath \u6587\u4EF6\u8DEF\u5F84\n * @param options \u8BFB\u53D6\u9009\u9879\n * @returns \u8BFB\u53D6\u7ED3\u679C\n */\n public safeReadFile(\n filePath: string, \n options: { \n encoding?: BufferEncoding; \n maxFileSize?: number;\n allowedExtensions?: string[];\n checkFileExtension?: boolean;\n } = {}\n ): { success: boolean; content?: string | Buffer; error?: string; stats?: any } {\n const validation = this.validateFilePath(filePath)\n if (!validation.isValid) {\n return { success: false, error: validation.error }\n }\n\n try {\n const normalizedPath = validation.normalizedPath\n \n // \u68C0\u67E5\u6587\u4EF6\u6269\u5C55\u540D\uFF08\u5982\u679C\u542F\u7528\uFF09\n if (options.checkFileExtension !== false) {\n const ext = extname(normalizedPath).toLowerCase()\n const allowedExts = options.allowedExtensions || \n Array.from(this.allowedExtensions)\n \n if (allowedExts.length > 0 && !allowedExts.includes(ext)) {\n return { \n success: false, \n error: `File extension '${ext}' is not allowed` \n }\n }\n }\n\n // \u68C0\u67E5\u6587\u4EF6\u662F\u5426\u5B58\u5728\n if (!existsSync(normalizedPath)) {\n return { success: false, error: 'File does not exist' }\n }\n\n // \u83B7\u53D6\u6587\u4EF6\u4FE1\u606F\n const stats = statSync(normalizedPath)\n const maxSize = options.maxFileSize || this.maxFileSize\n \n // \u68C0\u67E5\u6587\u4EF6\u5927\u5C0F\n if (stats.size > maxSize) {\n return { \n success: false, \n error: `File too large (${stats.size} bytes, max ${maxSize} bytes)` \n }\n }\n\n // \u68C0\u67E5\u6587\u4EF6\u7C7B\u578B\n if (!stats.isFile()) {\n return { success: false, error: 'Path is not a file' }\n }\n\n // \u68C0\u67E5\u6587\u4EF6\u6743\u9650\n if ((stats.mode & parseInt('400', 8)) === 0) { // \u68C0\u67E5\u8BFB\u6743\u9650\n return { success: false, error: 'No read permission' }\n }\n\n // \u8BFB\u53D6\u6587\u4EF6\u5185\u5BB9\n const content = readFileSync(normalizedPath, {\n encoding: options.encoding || 'utf8'\n })\n\n return { \n success: true, \n content,\n stats: {\n size: stats.size,\n mtime: stats.mtime,\n atime: stats.atime,\n mode: stats.mode\n }\n }\n } catch (error) {\n return { \n success: false, \n error: `Failed to read file: ${error instanceof Error ? error.message : String(error)}` \n }\n }\n }\n\n /**\n * \u5B89\u5168\u5730\u5199\u5165\u6587\u4EF6\n * @param filePath \u6587\u4EF6\u8DEF\u5F84\n * @param content \u6587\u4EF6\u5185\u5BB9\n * @param options \u5199\u5165\u9009\u9879\n * @returns \u5199\u5165\u7ED3\u679C\n */\n public safeWriteFile(\n filePath: string, \n content: string | Buffer, \n options: { \n encoding?: BufferEncoding; \n createDirectory?: boolean;\n atomic?: boolean;\n mode?: number;\n allowedExtensions?: string[];\n checkFileExtension?: boolean;\n maxSize?: number;\n } = {}\n ): { success: boolean; error?: string } {\n const validation = this.validateFilePath(filePath)\n if (!validation.isValid) {\n return { success: false, error: validation.error }\n }\n\n try {\n const normalizedPath = validation.normalizedPath\n \n // \u68C0\u67E5\u6587\u4EF6\u6269\u5C55\u540D\uFF08\u5982\u679C\u542F\u7528\uFF09\n if (options.checkFileExtension !== false) {\n const ext = extname(normalizedPath).toLowerCase()\n const allowedExts = options.allowedExtensions || \n Array.from(this.allowedExtensions)\n \n if (allowedExts.length > 0 && !allowedExts.includes(ext)) {\n return { \n success: false, \n error: `File extension '${ext}' is not allowed` \n }\n }\n }\n\n // \u68C0\u67E5\u5185\u5BB9\u5927\u5C0F\n const contentSize = typeof content === 'string' ? \n Buffer.byteLength(content, options.encoding as BufferEncoding || 'utf8') : \n content.length\n \n const maxSize = options.maxSize || this.maxFileSize\n if (contentSize > maxSize) {\n return { \n success: false, \n error: `Content too large (${contentSize} bytes, max ${maxSize} bytes)` \n }\n }\n\n // \u521B\u5EFA\u76EE\u5F55\uFF08\u5982\u679C\u9700\u8981\uFF09\n if (options.createDirectory) {\n const dir = dirname(normalizedPath)\n if (!existsSync(dir)) {\n mkdirSync(dir, { recursive: true, mode: 0o755 })\n }\n }\n\n // \u539F\u5B50\u5199\u5165\uFF08\u5982\u679C\u542F\u7528\uFF09\n if (options.atomic) {\n const tempPath = `${normalizedPath}.tmp.${Date.now()}`\n \n try {\n // \u5199\u5165\u4E34\u65F6\u6587\u4EF6\n writeFileSync(tempPath, content, {\n encoding: options.encoding as BufferEncoding || 'utf8',\n mode: options.mode || 0o644\n })\n \n // \u91CD\u547D\u540D\u4E3A\u76EE\u6807\u6587\u4EF6\n renameSync(tempPath, normalizedPath)\n } catch (renameError) {\n // \u6E05\u7406\u4E34\u65F6\u6587\u4EF6\n try {\n if (existsSync(tempPath)) {\n unlinkSync(tempPath)\n }\n } catch {\n // \u5FFD\u7565\u6E05\u7406\u9519\u8BEF\n }\n throw renameError\n }\n } else {\n // \u76F4\u63A5\u5199\u5165\n writeFileSync(normalizedPath, content, {\n encoding: options.encoding as BufferEncoding || 'utf8',\n mode: options.mode || 0o644\n })\n }\n\n return { success: true }\n } catch (error) {\n return { \n success: false, \n error: `Failed to write file: ${error instanceof Error ? error.message : String(error)}` \n }\n }\n }\n\n /**\n * \u5B89\u5168\u5730\u5220\u9664\u6587\u4EF6\n * @param filePath \u6587\u4EF6\u8DEF\u5F84\n * @returns \u5220\u9664\u7ED3\u679C\n */\n public safeDeleteFile(filePath: string): { success: boolean; error?: string } {\n const validation = this.validateFilePath(filePath)\n if (!validation.isValid) {\n return { success: false, error: validation.error }\n }\n\n try {\n const normalizedPath = validation.normalizedPath\n \n // \u68C0\u67E5\u6587\u4EF6\u662F\u5426\u5B58\u5728\n if (!existsSync(normalizedPath)) {\n return { success: false, error: 'File does not exist' }\n }\n\n // \u68C0\u67E5\u6587\u4EF6\u7C7B\u578B\n const stats = statSync(normalizedPath)\n if (!stats.isFile()) {\n return { success: false, error: 'Path is not a file' }\n }\n\n // \u68C0\u67E5\u5199\u6743\u9650\n if ((stats.mode & parseInt('200', 8)) === 0) {\n return { success: false, error: 'No write permission' }\n }\n\n // \u5B89\u5168\u5220\u9664\n unlinkSync(normalizedPath)\n return { success: true }\n } catch (error) {\n return { \n success: false, \n error: `Failed to delete file: ${error instanceof Error ? error.message : String(error)}` \n }\n }\n }\n\n /**\n * \u5B89\u5168\u5730\u521B\u5EFA\u76EE\u5F55\n * @param dirPath \u76EE\u5F55\u8DEF\u5F84\n * @param mode \u76EE\u5F55\u6743\u9650\n * @returns \u521B\u5EFA\u7ED3\u679C\n */\n public safeCreateDirectory(dirPath: string, mode: number = 0o755): { success: boolean; error?: string } {\n const validation = this.validateFilePath(dirPath)\n if (!validation.isValid) {\n return { success: false, error: validation.error }\n }\n\n try {\n const normalizedPath = validation.normalizedPath\n \n if (existsSync(normalizedPath)) {\n const stats = statSync(normalizedPath)\n if (!stats.isDirectory()) {\n return { success: false, error: 'Path already exists and is not a directory' }\n }\n return { success: true }\n }\n\n mkdirSync(normalizedPath, { recursive: true, mode })\n return { success: true }\n } catch (error) {\n return { \n success: false, \n error: `Failed to create directory: ${error instanceof Error ? error.message : String(error)}` \n }\n }\n }\n\n /**\n * \u5B89\u5168\u5730\u83B7\u53D6\u6587\u4EF6\u4FE1\u606F\n * @param filePath \u6587\u4EF6\u8DEF\u5F84\n * @returns \u6587\u4EF6\u4FE1\u606F\n */\n public safeGetFileInfo(filePath: string): { \n success: boolean; \n stats?: { \n size: number; \n isFile: boolean; \n isDirectory: boolean; \n mode: number; \n atime: Date; \n mtime: Date; \n ctime: Date; \n }; \n error?: string \n } {\n const validation = this.validateFilePath(filePath)\n if (!validation.isValid) {\n return { success: false, error: validation.error }\n }\n\n try {\n const normalizedPath = validation.normalizedPath\n \n if (!existsSync(normalizedPath)) {\n return { success: false, error: 'File does not exist' }\n }\n\n const stats = statSync(normalizedPath)\n \n return {\n success: true,\n stats: {\n size: stats.size,\n isFile: stats.isFile(),\n isDirectory: stats.isDirectory(),\n mode: stats.mode,\n atime: stats.atime,\n mtime: stats.mtime,\n ctime: stats.ctime\n }\n }\n } catch (error) {\n return { \n success: false, \n error: `Failed to get file info: ${error instanceof Error ? error.message : String(error)}` \n }\n }\n }\n\n /**\n * \u6DFB\u52A0\u5141\u8BB8\u7684\u57FA\u7840\u8DEF\u5F84\n * @param basePath \u57FA\u7840\u8DEF\u5F84\n */\n public addAllowedBasePath(basePath: string): { success: boolean; error?: string } {\n try {\n const normalized = normalize(resolve(basePath))\n \n // \u9A8C\u8BC1\u8DEF\u5F84\u662F\u5426\u5B58\u5728\n if (!existsSync(normalized)) {\n return { success: false, error: 'Base path does not exist' }\n }\n\n this.allowedBasePaths.add(normalized)\n return { success: true }\n } catch (error) {\n return { \n success: false, \n error: `Failed to add base path: ${error instanceof Error ? error.message : String(error)}` \n }\n }\n }\n\n /**\n * \u8BBE\u7F6E\u6700\u5927\u6587\u4EF6\u5927\u5C0F\n * @param maxSize \u6700\u5927\u6587\u4EF6\u5927\u5C0F\uFF08\u5B57\u8282\uFF09\n */\n public setMaxFileSize(maxSize: number): void {\n this.maxFileSize = maxSize\n }\n\n /**\n * \u6DFB\u52A0\u5141\u8BB8\u7684\u6587\u4EF6\u6269\u5C55\u540D\n * @param extensions \u6587\u4EF6\u6269\u5C55\u540D\u6570\u7EC4\n */\n public addAllowedExtensions(extensions: string[]): void {\n extensions.forEach(ext => {\n if (!ext.startsWith('.')) {\n ext = '.' + ext\n }\n this.allowedExtensions.add(ext.toLowerCase())\n })\n }\n\n /**\n * \u68C0\u67E5\u6587\u4EF6\u662F\u5426\u5728\u5141\u8BB8\u7684\u57FA\u7840\u8DEF\u5F84\u4E2D\n * @param filePath \u6587\u4EF6\u8DEF\u5F84\n * @returns \u662F\u5426\u5141\u8BB8\n */\n public isPathAllowed(filePath: string): boolean {\n const validation = this.validateFilePath(filePath)\n return validation.isValid\n }\n\n /**\n * \u9A8C\u8BC1\u6587\u4EF6\u540D\u5B89\u5168\u6027\n * @param filename \u6587\u4EF6\u540D\n * @returns \u9A8C\u8BC1\u7ED3\u679C\n */\n public validateFileName(filename: string): { isValid: boolean; error?: string } {\n // \u68C0\u67E5\u6587\u4EF6\u540D\u957F\u5EA6\n if (filename.length === 0) {\n return { isValid: false, error: 'Filename cannot be empty' }\n }\n\n if (filename.length > 255) {\n return { isValid: false, error: 'Filename too long (max 255 characters)' }\n }\n\n // \u68C0\u67E5\u6587\u4EF6\u540D\u5B57\u7B26\n const invalidChars = /[<>:\"/\\\\|?*\\x00-\\x1F]/\n if (invalidChars.test(filename)) {\n return { isValid: false, error: 'Filename contains invalid characters' }\n }\n\n // \u68C0\u67E5\u4FDD\u7559\u6587\u4EF6\u540D\n const reservedNames = [\n 'CON', 'PRN', 'AUX', 'NUL',\n 'COM1', 'COM2', 'COM3', 'COM4', 'COM5', 'COM6', 'COM7', 'COM8', 'COM9',\n 'LPT1', 'LPT2', 'LPT3', 'LPT4', 'LPT5', 'LPT6', 'LPT7', 'LPT8', 'LPT9'\n ]\n\n const baseName = filename.split('.')[0].toUpperCase()\n if (reservedNames.includes(baseName)) {\n return { isValid: false, error: 'Filename is reserved' }\n }\n\n // \u68C0\u67E5\u662F\u5426\u4EE5\u70B9\u5F00\u5934\u6216\u7ED3\u5C3E\n if (filename.startsWith('.') || filename.endsWith('.')) {\n return { isValid: false, error: 'Filename cannot start or end with a dot' }\n }\n\n // \u68C0\u67E5\u662F\u5426\u4EE5\u7A7A\u683C\u5F00\u5934\u6216\u7ED3\u5C3E\n if (filename.startsWith(' ') || filename.endsWith(' ')) {\n return { isValid: false, error: 'Filename cannot start or end with spaces' }\n }\n\n return { isValid: true }\n }\n}\n\n// \u5BFC\u51FA\u5355\u4F8B\u5B9E\u4F8B\nexport const secureFileService = SecureFileService.getInstance()\n"],
5
+ "mappings": "AAAA,SAAS,YAAY,cAAc,eAAe,WAAW,UAAU,YAAY,kBAAkB;AACrG,SAAe,SAAS,WAAW,SAAS,SAAS,UAAU,kBAAkB;AACjF,SAAS,eAAe;AAMjB,MAAM,kBAAkB;AAAA,EAC7B,OAAe;AAAA,EACP;AAAA,EACA;AAAA,EACA;AAAA,EAEA,cAAc;AAEpB,SAAK,mBAAmB,oBAAI,IAAI;AAAA,MAC9B,QAAQ,IAAI;AAAA,MACZ,QAAQ;AAAA,MACR;AAAA,MACA;AAAA,IACF,CAAC;AAGD,SAAK,cAAc,KAAK,OAAO;AAG/B,SAAK,oBAAoB,oBAAI,IAAI;AAAA,MAC/B;AAAA,MAAQ;AAAA,MAAO;AAAA,MAAS;AAAA,MAAO;AAAA,MAAO;AAAA,MAAQ;AAAA,MAC9C;AAAA,MAAS;AAAA,MAAQ;AAAA,MAAS;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAC1C;AAAA,MAAS;AAAA,MAAQ;AAAA,MAAS;AAAA,MAAS;AAAA,MAAQ;AAAA,MAC3C;AAAA,MAAO;AAAA,MAAO;AAAA,MAAO;AAAA,MAAS;AAAA,MAAQ;AAAA,MAAM;AAAA,MAC5C;AAAA,MAAO;AAAA,MAAS;AAAA,MAAQ;AAAA,MAAS;AAAA,MAAQ;AAAA,MACzC;AAAA,MAAe;AAAA,MAAc;AAAA,MAAc;AAAA,IAC7C,CAAC;AAAA,EACH;AAAA,EAEA,OAAc,cAAiC;AAC7C,QAAI,CAAC,kBAAkB,UAAU;AAC/B,wBAAkB,WAAW,IAAI,kBAAkB;AAAA,IACrD;AACA,WAAO,kBAAkB;AAAA,EAC3B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOO,iBAAiB,UAAgF;AACtG,QAAI;AAEF,YAAM,iBAAiB,UAAU,QAAQ;AAGzC,UAAI,eAAe,SAAS,MAAM;AAChC,eAAO;AAAA,UACL,SAAS;AAAA,UACT;AAAA,UACA,OAAO;AAAA,QACT;AAAA,MACF;AAGA,UAAI,eAAe,SAAS,IAAI,KAAK,eAAe,SAAS,GAAG,GAAG;AACjE,eAAO;AAAA,UACL,SAAS;AAAA,UACT;AAAA,UACA,OAAO;AAAA,QACT;AAAA,MACF;AAGA,YAAM,qBAAqB;AAAA,QACzB;AAAA;AAAA,QACA;AAAA;AAAA,QACA;AAAA;AAAA,QACA;AAAA;AAAA,QACA;AAAA;AAAA,QACA;AAAA;AAAA,QACA;AAAA;AAAA,QACA;AAAA;AAAA,QACA;AAAA;AAAA,MACF;AAEA,iBAAW,WAAW,oBAAoB;AACxC,YAAI,QAAQ,KAAK,cAAc,GAAG;AAChC,iBAAO;AAAA,YACL,SAAS;AAAA,YACT;AAAA,YACA,OAAO,qCAAqC,OAAO;AAAA,UACrD;AAAA,QACF;AAAA,MACF;AAGA,YAAM,eAAe,QAAQ,cAAc;AAG3C,YAAM,kBAAkB,MAAM,KAAK,KAAK,gBAAgB,EAAE,KAAK,cAAY;AACzE,cAAM,OAAO,QAAQ,QAAQ;AAC7B,cAAM,MAAM,SAAS,MAAM,YAAY;AACvC,YAAI,CAAC,OAAO,QAAQ,GAAI,QAAO;AAC/B,YAAI,IAAI,WAAW,IAAI,EAAG,QAAO;AACjC,YAAI,WAAW,GAAG,EAAG,QAAO;AAC5B,eAAO;AAAA,MACT,CAAC;AAED,UAAI,CAAC,iBAAiB;AACpB,eAAO;AAAA,UACL,SAAS;AAAA,UACT;AAAA,UACA,OAAO;AAAA,QACT;AAAA,MACF;AAEA,aAAO,EAAE,SAAS,MAAM,gBAAgB,aAAa;AAAA,IACvD,SAAS,OAAO;AACd,aAAO;AAAA,QACL,SAAS;AAAA,QACT,gBAAgB;AAAA,QAChB,OAAO,2BAA2B,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK,CAAC;AAAA,MAC1F;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOO,WAAW,UAA2B;AAC3C,UAAM,aAAa,KAAK,iBAAiB,QAAQ;AACjD,QAAI,CAAC,WAAW,SAAS;AACvB,aAAO;AAAA,IACT;AAEA,QAAI;AACF,aAAO,WAAW,WAAW,cAAc;AAAA,IAC7C,SAAS,OAAO;AACd,aAAO;AAAA,IACT;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQO,aACL,UACA,UAKI,CAAC,GACyE;AAC9E,UAAM,aAAa,KAAK,iBAAiB,QAAQ;AACjD,QAAI,CAAC,WAAW,SAAS;AACvB,aAAO,EAAE,SAAS,OAAO,OAAO,WAAW,MAAM;AAAA,IACnD;AAEA,QAAI;AACF,YAAM,iBAAiB,WAAW;AAGlC,UAAI,QAAQ,uBAAuB,OAAO;AACxC,cAAM,MAAM,QAAQ,cAAc,EAAE,YAAY;AAChD,cAAM,cAAc,QAAQ,qBACT,MAAM,KAAK,KAAK,iBAAiB;AAEpD,YAAI,YAAY,SAAS,KAAK,CAAC,YAAY,SAAS,GAAG,GAAG;AACxD,iBAAO;AAAA,YACL,SAAS;AAAA,YACT,OAAO,mBAAmB,GAAG;AAAA,UAC/B;AAAA,QACF;AAAA,MACF;AAGA,UAAI,CAAC,WAAW,cAAc,GAAG;AAC/B,eAAO,EAAE,SAAS,OAAO,OAAO,sBAAsB;AAAA,MACxD;AAGA,YAAM,QAAQ,SAAS,cAAc;AACrC,YAAM,UAAU,QAAQ,eAAe,KAAK;AAG5C,UAAI,MAAM,OAAO,SAAS;AACxB,eAAO;AAAA,UACL,SAAS;AAAA,UACT,OAAO,mBAAmB,MAAM,IAAI,eAAe,OAAO;AAAA,QAC5D;AAAA,MACF;AAGA,UAAI,CAAC,MAAM,OAAO,GAAG;AACnB,eAAO,EAAE,SAAS,OAAO,OAAO,qBAAqB;AAAA,MACvD;AAGA,WAAK,MAAM,OAAO,SAAS,OAAO,CAAC,OAAO,GAAG;AAC3C,eAAO,EAAE,SAAS,OAAO,OAAO,qBAAqB;AAAA,MACvD;AAGA,YAAM,UAAU,aAAa,gBAAgB;AAAA,QAC3C,UAAU,QAAQ,YAAY;AAAA,MAChC,CAAC;AAED,aAAO;AAAA,QACL,SAAS;AAAA,QACT;AAAA,QACA,OAAO;AAAA,UACL,MAAM,MAAM;AAAA,UACZ,OAAO,MAAM;AAAA,UACb,OAAO,MAAM;AAAA,UACb,MAAM,MAAM;AAAA,QACd;AAAA,MACF;AAAA,IACF,SAAS,OAAO;AACd,aAAO;AAAA,QACL,SAAS;AAAA,QACT,OAAO,wBAAwB,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK,CAAC;AAAA,MACvF;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASO,cACL,UACA,SACA,UAQI,CAAC,GACiC;AACtC,UAAM,aAAa,KAAK,iBAAiB,QAAQ;AACjD,QAAI,CAAC,WAAW,SAAS;AACvB,aAAO,EAAE,SAAS,OAAO,OAAO,WAAW,MAAM;AAAA,IACnD;AAEA,QAAI;AACF,YAAM,iBAAiB,WAAW;AAGlC,UAAI,QAAQ,uBAAuB,OAAO;AACxC,cAAM,MAAM,QAAQ,cAAc,EAAE,YAAY;AAChD,cAAM,cAAc,QAAQ,qBACT,MAAM,KAAK,KAAK,iBAAiB;AAEpD,YAAI,YAAY,SAAS,KAAK,CAAC,YAAY,SAAS,GAAG,GAAG;AACxD,iBAAO;AAAA,YACL,SAAS;AAAA,YACT,OAAO,mBAAmB,GAAG;AAAA,UAC/B;AAAA,QACF;AAAA,MACF;AAGA,YAAM,cAAc,OAAO,YAAY,WACrC,OAAO,WAAW,SAAS,QAAQ,YAA8B,MAAM,IACvE,QAAQ;AAEV,YAAM,UAAU,QAAQ,WAAW,KAAK;AACxC,UAAI,cAAc,SAAS;AACzB,eAAO;AAAA,UACL,SAAS;AAAA,UACT,OAAO,sBAAsB,WAAW,eAAe,OAAO;AAAA,QAChE;AAAA,MACF;AAGA,UAAI,QAAQ,iBAAiB;AAC3B,cAAM,MAAM,QAAQ,cAAc;AAClC,YAAI,CAAC,WAAW,GAAG,GAAG;AACpB,oBAAU,KAAK,EAAE,WAAW,MAAM,MAAM,IAAM,CAAC;AAAA,QACjD;AAAA,MACF;AAGA,UAAI,QAAQ,QAAQ;AAClB,cAAM,WAAW,GAAG,cAAc,QAAQ,KAAK,IAAI,CAAC;AAEpD,YAAI;AAEF,wBAAc,UAAU,SAAS;AAAA,YAC/B,UAAU,QAAQ,YAA8B;AAAA,YAChD,MAAM,QAAQ,QAAQ;AAAA,UACxB,CAAC;AAGD,qBAAW,UAAU,cAAc;AAAA,QACrC,SAAS,aAAa;AAEpB,cAAI;AACF,gBAAI,WAAW,QAAQ,GAAG;AACxB,yBAAW,QAAQ;AAAA,YACrB;AAAA,UACF,QAAQ;AAAA,UAER;AACA,gBAAM;AAAA,QACR;AAAA,MACF,OAAO;AAEL,sBAAc,gBAAgB,SAAS;AAAA,UACrC,UAAU,QAAQ,YAA8B;AAAA,UAChD,MAAM,QAAQ,QAAQ;AAAA,QACxB,CAAC;AAAA,MACH;AAEA,aAAO,EAAE,SAAS,KAAK;AAAA,IACzB,SAAS,OAAO;AACd,aAAO;AAAA,QACL,SAAS;AAAA,QACT,OAAO,yBAAyB,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK,CAAC;AAAA,MACxF;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOO,eAAe,UAAwD;AAC5E,UAAM,aAAa,KAAK,iBAAiB,QAAQ;AACjD,QAAI,CAAC,WAAW,SAAS;AACvB,aAAO,EAAE,SAAS,OAAO,OAAO,WAAW,MAAM;AAAA,IACnD;AAEA,QAAI;AACF,YAAM,iBAAiB,WAAW;AAGlC,UAAI,CAAC,WAAW,cAAc,GAAG;AAC/B,eAAO,EAAE,SAAS,OAAO,OAAO,sBAAsB;AAAA,MACxD;AAGA,YAAM,QAAQ,SAAS,cAAc;AACrC,UAAI,CAAC,MAAM,OAAO,GAAG;AACnB,eAAO,EAAE,SAAS,OAAO,OAAO,qBAAqB;AAAA,MACvD;AAGA,WAAK,MAAM,OAAO,SAAS,OAAO,CAAC,OAAO,GAAG;AAC3C,eAAO,EAAE,SAAS,OAAO,OAAO,sBAAsB;AAAA,MACxD;AAGA,iBAAW,cAAc;AACzB,aAAO,EAAE,SAAS,KAAK;AAAA,IACzB,SAAS,OAAO;AACd,aAAO;AAAA,QACL,SAAS;AAAA,QACT,OAAO,0BAA0B,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK,CAAC;AAAA,MACzF;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQO,oBAAoB,SAAiB,OAAe,KAA6C;AACtG,UAAM,aAAa,KAAK,iBAAiB,OAAO;AAChD,QAAI,CAAC,WAAW,SAAS;AACvB,aAAO,EAAE,SAAS,OAAO,OAAO,WAAW,MAAM;AAAA,IACnD;AAEA,QAAI;AACF,YAAM,iBAAiB,WAAW;AAElC,UAAI,WAAW,cAAc,GAAG;AAC9B,cAAM,QAAQ,SAAS,cAAc;AACrC,YAAI,CAAC,MAAM,YAAY,GAAG;AACxB,iBAAO,EAAE,SAAS,OAAO,OAAO,6CAA6C;AAAA,QAC/E;AACA,eAAO,EAAE,SAAS,KAAK;AAAA,MACzB;AAEA,gBAAU,gBAAgB,EAAE,WAAW,MAAM,KAAK,CAAC;AACnD,aAAO,EAAE,SAAS,KAAK;AAAA,IACzB,SAAS,OAAO;AACd,aAAO;AAAA,QACL,SAAS;AAAA,QACT,OAAO,+BAA+B,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK,CAAC;AAAA,MAC9F;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOO,gBAAgB,UAYrB;AACA,UAAM,aAAa,KAAK,iBAAiB,QAAQ;AACjD,QAAI,CAAC,WAAW,SAAS;AACvB,aAAO,EAAE,SAAS,OAAO,OAAO,WAAW,MAAM;AAAA,IACnD;AAEA,QAAI;AACF,YAAM,iBAAiB,WAAW;AAElC,UAAI,CAAC,WAAW,cAAc,GAAG;AAC/B,eAAO,EAAE,SAAS,OAAO,OAAO,sBAAsB;AAAA,MACxD;AAEA,YAAM,QAAQ,SAAS,cAAc;AAErC,aAAO;AAAA,QACL,SAAS;AAAA,QACT,OAAO;AAAA,UACL,MAAM,MAAM;AAAA,UACZ,QAAQ,MAAM,OAAO;AAAA,UACrB,aAAa,MAAM,YAAY;AAAA,UAC/B,MAAM,MAAM;AAAA,UACZ,OAAO,MAAM;AAAA,UACb,OAAO,MAAM;AAAA,UACb,OAAO,MAAM;AAAA,QACf;AAAA,MACF;AAAA,IACF,SAAS,OAAO;AACd,aAAO;AAAA,QACL,SAAS;AAAA,QACT,OAAO,4BAA4B,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK,CAAC;AAAA,MAC3F;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA,EAMO,mBAAmB,UAAwD;AAChF,QAAI;AACF,YAAM,aAAa,UAAU,QAAQ,QAAQ,CAAC;AAG9C,UAAI,CAAC,WAAW,UAAU,GAAG;AAC3B,eAAO,EAAE,SAAS,OAAO,OAAO,2BAA2B;AAAA,MAC7D;AAEA,WAAK,iBAAiB,IAAI,UAAU;AACpC,aAAO,EAAE,SAAS,KAAK;AAAA,IACzB,SAAS,OAAO;AACd,aAAO;AAAA,QACL,SAAS;AAAA,QACT,OAAO,4BAA4B,iBAAiB,QAAQ,MAAM,UAAU,OAAO,KAAK,CAAC;AAAA,MAC3F;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA,EAMO,eAAe,SAAuB;AAC3C,SAAK,cAAc;AAAA,EACrB;AAAA;AAAA;AAAA;AAAA;AAAA,EAMO,qBAAqB,YAA4B;AACtD,eAAW,QAAQ,SAAO;AACxB,UAAI,CAAC,IAAI,WAAW,GAAG,GAAG;AACxB,cAAM,MAAM;AAAA,MACd;AACA,WAAK,kBAAkB,IAAI,IAAI,YAAY,CAAC;AAAA,IAC9C,CAAC;AAAA,EACH;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOO,cAAc,UAA2B;AAC9C,UAAM,aAAa,KAAK,iBAAiB,QAAQ;AACjD,WAAO,WAAW;AAAA,EACpB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOO,iBAAiB,UAAwD;AAE9E,QAAI,SAAS,WAAW,GAAG;AACzB,aAAO,EAAE,SAAS,OAAO,OAAO,2BAA2B;AAAA,IAC7D;AAEA,QAAI,SAAS,SAAS,KAAK;AACzB,aAAO,EAAE,SAAS,OAAO,OAAO,yCAAyC;AAAA,IAC3E;AAGA,UAAM,eAAe;AACrB,QAAI,aAAa,KAAK,QAAQ,GAAG;AAC/B,aAAO,EAAE,SAAS,OAAO,OAAO,uCAAuC;AAAA,IACzE;AAGA,UAAM,gBAAgB;AAAA,MACpB;AAAA,MAAO;AAAA,MAAO;AAAA,MAAO;AAAA,MACrB;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAChE;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAAQ;AAAA,MAAQ;AAAA,IAClE;AAEA,UAAM,WAAW,SAAS,MAAM,GAAG,EAAE,CAAC,EAAE,YAAY;AACpD,QAAI,cAAc,SAAS,QAAQ,GAAG;AACpC,aAAO,EAAE,SAAS,OAAO,OAAO,uBAAuB;AAAA,IACzD;AAGA,QAAI,SAAS,WAAW,GAAG,KAAK,SAAS,SAAS,GAAG,GAAG;AACtD,aAAO,EAAE,SAAS,OAAO,OAAO,0CAA0C;AAAA,IAC5E;AAGA,QAAI,SAAS,WAAW,GAAG,KAAK,SAAS,SAAS,GAAG,GAAG;AACtD,aAAO,EAAE,SAAS,OAAO,OAAO,2CAA2C;AAAA,IAC7E;AAEA,WAAO,EAAE,SAAS,KAAK;AAAA,EACzB;AACF;AAGO,MAAM,oBAAoB,kBAAkB,YAAY;",
6
+ "names": []
7
+ }
@@ -0,0 +1,31 @@
1
+ import { logEvent } from "../services/statsig.js";
2
+ const isDebug = process.argv.includes("--debug") || process.argv.includes("-d") || process.env.DEBUG === "true";
3
+ const sessionState = {
4
+ modelErrors: {},
5
+ currentError: null
6
+ };
7
+ function setSessionState(keyOrState, value) {
8
+ if (typeof keyOrState === "string") {
9
+ logEvent("session_state_set", {
10
+ key: keyOrState,
11
+ value: JSON.stringify(value)
12
+ });
13
+ sessionState[keyOrState] = value;
14
+ } else {
15
+ logEvent("session_state_set", {
16
+ key: "partial",
17
+ value: JSON.stringify(keyOrState)
18
+ });
19
+ Object.assign(sessionState, keyOrState);
20
+ }
21
+ }
22
+ function getSessionState(key) {
23
+ return key === void 0 ? sessionState : sessionState[key];
24
+ }
25
+ var sessionState_default = sessionState;
26
+ export {
27
+ sessionState_default as default,
28
+ getSessionState,
29
+ setSessionState
30
+ };
31
+ //# sourceMappingURL=sessionState.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/utils/sessionState.ts"],
4
+ "sourcesContent": ["import { logEvent } from '../services/statsig'\ntype SessionState = {\n modelErrors: Record<string, unknown>\n currentError: string | null\n}\n\nconst isDebug =\n process.argv.includes('--debug') ||\n process.argv.includes('-d') ||\n process.env.DEBUG === 'true'\n\nconst sessionState: SessionState = {\n modelErrors: {},\n currentError: null,\n} as const\n\nfunction setSessionState<K extends keyof SessionState>(\n key: K,\n value: SessionState[K],\n): void\nfunction setSessionState(partialState: Partial<SessionState>): void\nfunction setSessionState(\n keyOrState: keyof SessionState | Partial<SessionState>,\n value?: any,\n): void {\n if (typeof keyOrState === 'string') {\n logEvent('session_state_set', {\n key: keyOrState,\n value: JSON.stringify(value),\n })\n sessionState[keyOrState] = value\n } else {\n logEvent('session_state_set', {\n key: 'partial',\n value: JSON.stringify(keyOrState),\n })\n Object.assign(sessionState, keyOrState)\n }\n}\n\nfunction getSessionState(): SessionState\nfunction getSessionState<K extends keyof SessionState>(key: K): SessionState[K]\nfunction getSessionState<K extends keyof SessionState>(key?: K) {\n return key === undefined ? sessionState : sessionState[key]\n}\n\nexport type { SessionState }\nexport { setSessionState, getSessionState }\nexport default sessionState\n"],
5
+ "mappings": "AAAA,SAAS,gBAAgB;AAMzB,MAAM,UACJ,QAAQ,KAAK,SAAS,SAAS,KAC/B,QAAQ,KAAK,SAAS,IAAI,KAC1B,QAAQ,IAAI,UAAU;AAExB,MAAM,eAA6B;AAAA,EACjC,aAAa,CAAC;AAAA,EACd,cAAc;AAChB;AAOA,SAAS,gBACP,YACA,OACM;AACN,MAAI,OAAO,eAAe,UAAU;AAClC,aAAS,qBAAqB;AAAA,MAC5B,KAAK;AAAA,MACL,OAAO,KAAK,UAAU,KAAK;AAAA,IAC7B,CAAC;AACD,iBAAa,UAAU,IAAI;AAAA,EAC7B,OAAO;AACL,aAAS,qBAAqB;AAAA,MAC5B,KAAK;AAAA,MACL,OAAO,KAAK,UAAU,UAAU;AAAA,IAClC,CAAC;AACD,WAAO,OAAO,cAAc,UAAU;AAAA,EACxC;AACF;AAIA,SAAS,gBAA8C,KAAS;AAC9D,SAAO,QAAQ,SAAY,eAAe,aAAa,GAAG;AAC5D;AAIA,IAAO,uBAAQ;",
6
+ "names": []
7
+ }
@@ -0,0 +1,24 @@
1
+ import { cwd } from "process";
2
+ import { PersistentShell } from "./PersistentShell.js";
3
+ const STATE = {
4
+ originalCwd: cwd()
5
+ };
6
+ async function setCwd(cwd2) {
7
+ await PersistentShell.getInstance().setCwd(cwd2);
8
+ }
9
+ function setOriginalCwd(cwd2) {
10
+ STATE.originalCwd = cwd2;
11
+ }
12
+ function getOriginalCwd() {
13
+ return STATE.originalCwd;
14
+ }
15
+ function getCwd() {
16
+ return PersistentShell.getInstance().pwd();
17
+ }
18
+ export {
19
+ getCwd,
20
+ getOriginalCwd,
21
+ setCwd,
22
+ setOriginalCwd
23
+ };
24
+ //# sourceMappingURL=state.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/utils/state.ts"],
4
+ "sourcesContent": ["import { cwd } from 'process'\nimport { PersistentShell } from './PersistentShell'\n\n// DO NOT ADD MORE STATE HERE OR BORIS WILL CURSE YOU\nconst STATE: {\n originalCwd: string\n} = {\n originalCwd: cwd(),\n}\n\nexport async function setCwd(cwd: string): Promise<void> {\n await PersistentShell.getInstance().setCwd(cwd)\n}\n\nexport function setOriginalCwd(cwd: string): void {\n STATE.originalCwd = cwd\n}\n\nexport function getOriginalCwd(): string {\n return STATE.originalCwd\n}\n\nexport function getCwd(): string {\n return PersistentShell.getInstance().pwd()\n}\n"],
5
+ "mappings": "AAAA,SAAS,WAAW;AACpB,SAAS,uBAAuB;AAGhC,MAAM,QAEF;AAAA,EACF,aAAa,IAAI;AACnB;AAEA,eAAsB,OAAOA,MAA4B;AACvD,QAAM,gBAAgB,YAAY,EAAE,OAAOA,IAAG;AAChD;AAEO,SAAS,eAAeA,MAAmB;AAChD,QAAM,cAAcA;AACtB;AAEO,SAAS,iBAAyB;AACvC,SAAO,MAAM;AACf;AAEO,SAAS,SAAiB;AAC/B,SAAO,gBAAgB,YAAY,EAAE,IAAI;AAC3C;",
6
+ "names": ["cwd"]
7
+ }
@@ -0,0 +1,31 @@
1
+ import { existsSync, readFileSync } from "fs";
2
+ import { join, parse, dirname } from "path";
3
+ import { memoize } from "lodash-es";
4
+ import { getCwd } from "./state.js";
5
+ import { PROJECT_FILE } from "../constants/product.js";
6
+ const STYLE_PROMPT = "The codebase follows strict style guidelines shown below. All code changes must strictly adhere to these guidelines to maintain consistency and quality.";
7
+ const getCodeStyle = memoize(() => {
8
+ const styles = [];
9
+ let currentDir = getCwd();
10
+ while (currentDir !== parse(currentDir).root) {
11
+ const stylePath = join(currentDir, PROJECT_FILE);
12
+ if (existsSync(stylePath)) {
13
+ styles.push(
14
+ `Contents of ${stylePath}:
15
+
16
+ ${readFileSync(stylePath, "utf-8")}`
17
+ );
18
+ }
19
+ currentDir = dirname(currentDir);
20
+ }
21
+ if (styles.length === 0) {
22
+ return "";
23
+ }
24
+ return `${STYLE_PROMPT}
25
+
26
+ ${styles.reverse().join("\n\n")}`;
27
+ });
28
+ export {
29
+ getCodeStyle
30
+ };
31
+ //# sourceMappingURL=style.js.map
@@ -0,0 +1,7 @@
1
+ {
2
+ "version": 3,
3
+ "sources": ["../../src/utils/style.ts"],
4
+ "sourcesContent": ["import { existsSync, readFileSync } from 'fs'\nimport { join, parse, dirname } from 'path'\nimport { memoize } from 'lodash-es'\nimport { getCwd } from './state'\nimport { PROJECT_FILE } from '../constants/product'\n\nconst STYLE_PROMPT =\n 'The codebase follows strict style guidelines shown below. All code changes must strictly adhere to these guidelines to maintain consistency and quality.'\n\nexport const getCodeStyle = memoize((): string => {\n const styles: string[] = []\n let currentDir = getCwd()\n\n while (currentDir !== parse(currentDir).root) {\n const stylePath = join(currentDir, PROJECT_FILE)\n if (existsSync(stylePath)) {\n styles.push(\n `Contents of ${stylePath}:\\n\\n${readFileSync(stylePath, 'utf-8')}`,\n )\n }\n currentDir = dirname(currentDir)\n }\n\n if (styles.length === 0) {\n return ''\n }\n\n return `${STYLE_PROMPT}\\n\\n${styles.reverse().join('\\n\\n')}`\n})\n"],
5
+ "mappings": "AAAA,SAAS,YAAY,oBAAoB;AACzC,SAAS,MAAM,OAAO,eAAe;AACrC,SAAS,eAAe;AACxB,SAAS,cAAc;AACvB,SAAS,oBAAoB;AAE7B,MAAM,eACJ;AAEK,MAAM,eAAe,QAAQ,MAAc;AAChD,QAAM,SAAmB,CAAC;AAC1B,MAAI,aAAa,OAAO;AAExB,SAAO,eAAe,MAAM,UAAU,EAAE,MAAM;AAC5C,UAAM,YAAY,KAAK,YAAY,YAAY;AAC/C,QAAI,WAAW,SAAS,GAAG;AACzB,aAAO;AAAA,QACL,eAAe,SAAS;AAAA;AAAA,EAAQ,aAAa,WAAW,OAAO,CAAC;AAAA,MAClE;AAAA,IACF;AACA,iBAAa,QAAQ,UAAU;AAAA,EACjC;AAEA,MAAI,OAAO,WAAW,GAAG;AACvB,WAAO;AAAA,EACT;AAEA,SAAO,GAAG,YAAY;AAAA;AAAA,EAAO,OAAO,QAAQ,EAAE,KAAK,MAAM,CAAC;AAC5D,CAAC;",
6
+ "names": []
7
+ }