@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.
- package/cli.js +83 -57
- package/dist/ProjectOnboarding.js +99 -0
- package/dist/ProjectOnboarding.js.map +7 -0
- package/dist/Tool.js +1 -0
- package/dist/Tool.js.map +7 -0
- package/dist/commands/agents.js +2087 -0
- package/dist/commands/agents.js.map +7 -0
- package/dist/commands/approvedTools.js +36 -0
- package/dist/commands/approvedTools.js.map +7 -0
- package/dist/commands/bug.js +21 -0
- package/dist/commands/bug.js.map +7 -0
- package/dist/commands/clear.js +37 -0
- package/dist/commands/clear.js.map +7 -0
- package/dist/commands/compact.js +104 -0
- package/dist/commands/compact.js.map +7 -0
- package/dist/commands/config.js +20 -0
- package/dist/commands/config.js.map +7 -0
- package/dist/commands/cost.js +19 -0
- package/dist/commands/cost.js.map +7 -0
- package/dist/commands/ctx_viz.js +152 -0
- package/dist/commands/ctx_viz.js.map +7 -0
- package/dist/commands/doctor.js +25 -0
- package/dist/commands/doctor.js.map +7 -0
- package/dist/commands/help.js +20 -0
- package/dist/commands/help.js.map +7 -0
- package/dist/commands/init.js +38 -0
- package/dist/commands/init.js.map +7 -0
- package/dist/commands/listen.js +37 -0
- package/dist/commands/listen.js.map +7 -0
- package/dist/commands/login.js +37 -0
- package/dist/commands/login.js.map +7 -0
- package/dist/commands/logout.js +33 -0
- package/dist/commands/logout.js.map +7 -0
- package/dist/commands/mcp.js +34 -0
- package/dist/commands/mcp.js.map +7 -0
- package/dist/commands/model.js +41 -0
- package/dist/commands/model.js.map +7 -0
- package/dist/commands/modelstatus.js +21 -0
- package/dist/commands/modelstatus.js.map +7 -0
- package/dist/commands/onboarding.js +36 -0
- package/dist/commands/onboarding.js.map +7 -0
- package/dist/commands/pr_comments.js +61 -0
- package/dist/commands/pr_comments.js.map +7 -0
- package/dist/commands/refreshCommands.js +37 -0
- package/dist/commands/refreshCommands.js.map +7 -0
- package/dist/commands/release-notes.js +30 -0
- package/dist/commands/release-notes.js.map +7 -0
- package/dist/commands/resume.js +35 -0
- package/dist/commands/resume.js.map +7 -0
- package/dist/commands/review.js +51 -0
- package/dist/commands/review.js.map +7 -0
- package/dist/commands/terminalSetup.js +163 -0
- package/dist/commands/terminalSetup.js.map +7 -0
- package/dist/commands.js +84 -0
- package/dist/commands.js.map +7 -0
- package/dist/components/ApproveApiKey.js +74 -0
- package/dist/components/ApproveApiKey.js.map +7 -0
- package/dist/components/AsciiLogo.js +12 -0
- package/dist/components/AsciiLogo.js.map +7 -0
- package/dist/components/AutoUpdater.js +74 -0
- package/dist/components/AutoUpdater.js.map +7 -0
- package/dist/components/Bug.js +147 -0
- package/dist/components/Bug.js.map +7 -0
- package/dist/components/Config.js +166 -0
- package/dist/components/Config.js.map +7 -0
- package/dist/components/ConsoleOAuthFlow.js +188 -0
- package/dist/components/ConsoleOAuthFlow.js.map +7 -0
- package/dist/components/Cost.js +13 -0
- package/dist/components/Cost.js.map +7 -0
- package/dist/components/CostThresholdDialog.js +38 -0
- package/dist/components/CostThresholdDialog.js.map +7 -0
- package/dist/components/CustomSelect/option-map.js +32 -0
- package/dist/components/CustomSelect/option-map.js.map +7 -0
- package/dist/components/CustomSelect/select-option.js +34 -0
- package/dist/components/CustomSelect/select-option.js.map +7 -0
- package/dist/components/CustomSelect/select.js +64 -0
- package/dist/components/CustomSelect/select.js.map +7 -0
- package/dist/components/CustomSelect/theme.js +1 -0
- package/dist/components/CustomSelect/theme.js.map +7 -0
- package/dist/components/CustomSelect/use-select-state.js +220 -0
- package/dist/components/CustomSelect/use-select-state.js.map +7 -0
- package/dist/components/CustomSelect/use-select.js +21 -0
- package/dist/components/CustomSelect/use-select.js.map +7 -0
- package/dist/components/FallbackToolUseRejectedMessage.js +11 -0
- package/dist/components/FallbackToolUseRejectedMessage.js.map +7 -0
- package/dist/components/FileEditToolUpdatedMessage.js +31 -0
- package/dist/components/FileEditToolUpdatedMessage.js.map +7 -0
- package/dist/components/Help.js +41 -0
- package/dist/components/Help.js.map +7 -0
- package/dist/components/HighlightedCode.js +30 -0
- package/dist/components/HighlightedCode.js.map +7 -0
- package/dist/components/InvalidConfigDialog.js +83 -0
- package/dist/components/InvalidConfigDialog.js.map +7 -0
- package/dist/components/Link.js +18 -0
- package/dist/components/Link.js.map +7 -0
- package/dist/components/LogSelector.js +50 -0
- package/dist/components/LogSelector.js.map +7 -0
- package/dist/components/Logo.js +89 -0
- package/dist/components/Logo.js.map +7 -0
- package/dist/components/MCPServerApprovalDialog.js +79 -0
- package/dist/components/MCPServerApprovalDialog.js.map +7 -0
- package/dist/components/MCPServerDialogCopy.js +11 -0
- package/dist/components/MCPServerDialogCopy.js.map +7 -0
- package/dist/components/MCPServerMultiselectDialog.js +80 -0
- package/dist/components/MCPServerMultiselectDialog.js.map +7 -0
- package/dist/components/Message.js +146 -0
- package/dist/components/Message.js.map +7 -0
- package/dist/components/MessageResponse.js +9 -0
- package/dist/components/MessageResponse.js.map +7 -0
- package/dist/components/MessageSelector.js +133 -0
- package/dist/components/MessageSelector.js.map +7 -0
- package/dist/components/ModeIndicator.js +38 -0
- package/dist/components/ModeIndicator.js.map +7 -0
- package/dist/components/ModelConfig.js +208 -0
- package/dist/components/ModelConfig.js.map +7 -0
- package/dist/components/ModelListManager.js +140 -0
- package/dist/components/ModelListManager.js.map +7 -0
- package/dist/components/ModelSelector.js +1985 -0
- package/dist/components/ModelSelector.js.map +7 -0
- package/dist/components/ModelStatusDisplay.js +87 -0
- package/dist/components/ModelStatusDisplay.js.map +7 -0
- package/dist/components/Onboarding.js +153 -0
- package/dist/components/Onboarding.js.map +7 -0
- package/dist/components/PressEnterToContinue.js +10 -0
- package/dist/components/PressEnterToContinue.js.map +7 -0
- package/dist/components/PromptInput.js +501 -0
- package/dist/components/PromptInput.js.map +7 -0
- package/dist/components/SentryErrorBoundary.js +27 -0
- package/dist/components/SentryErrorBoundary.js.map +7 -0
- package/dist/components/Spinner.js +101 -0
- package/dist/components/Spinner.js.map +7 -0
- package/dist/components/StickerRequestForm.js +7 -0
- package/dist/components/StickerRequestForm.js.map +7 -0
- package/dist/components/StructuredDiff.js +148 -0
- package/dist/components/StructuredDiff.js.map +7 -0
- package/dist/components/TextInput.js +100 -0
- package/dist/components/TextInput.js.map +7 -0
- package/dist/components/TodoItem.js +35 -0
- package/dist/components/TodoItem.js.map +7 -0
- package/dist/components/TokenWarning.js +19 -0
- package/dist/components/TokenWarning.js.map +7 -0
- package/dist/components/ToolUseLoader.js +24 -0
- package/dist/components/ToolUseLoader.js.map +7 -0
- package/dist/components/TrustDialog.js +76 -0
- package/dist/components/TrustDialog.js.map +7 -0
- package/dist/components/binary-feedback/BinaryFeedback.js +50 -0
- package/dist/components/binary-feedback/BinaryFeedback.js.map +7 -0
- package/dist/components/binary-feedback/BinaryFeedbackOption.js +94 -0
- package/dist/components/binary-feedback/BinaryFeedbackOption.js.map +7 -0
- package/dist/components/binary-feedback/BinaryFeedbackView.js +139 -0
- package/dist/components/binary-feedback/BinaryFeedbackView.js.map +7 -0
- package/dist/components/binary-feedback/utils.js +161 -0
- package/dist/components/binary-feedback/utils.js.map +7 -0
- package/dist/components/messages/AssistantBashOutputMessage.js +23 -0
- package/dist/components/messages/AssistantBashOutputMessage.js.map +7 -0
- package/dist/components/messages/AssistantLocalCommandOutputMessage.js +36 -0
- package/dist/components/messages/AssistantLocalCommandOutputMessage.js.map +7 -0
- package/dist/components/messages/AssistantRedactedThinkingMessage.js +12 -0
- package/dist/components/messages/AssistantRedactedThinkingMessage.js.map +7 -0
- package/dist/components/messages/AssistantTextMessage.js +78 -0
- package/dist/components/messages/AssistantTextMessage.js.map +7 -0
- package/dist/components/messages/AssistantThinkingMessage.js +27 -0
- package/dist/components/messages/AssistantThinkingMessage.js.map +7 -0
- package/dist/components/messages/AssistantToolUseMessage.js +91 -0
- package/dist/components/messages/AssistantToolUseMessage.js.map +7 -0
- package/dist/components/messages/TaskProgressMessage.js +11 -0
- package/dist/components/messages/TaskProgressMessage.js.map +7 -0
- package/dist/components/messages/TaskToolMessage.js +39 -0
- package/dist/components/messages/TaskToolMessage.js.map +7 -0
- package/dist/components/messages/UserBashInputMessage.js +18 -0
- package/dist/components/messages/UserBashInputMessage.js.map +7 -0
- package/dist/components/messages/UserCommandMessage.js +20 -0
- package/dist/components/messages/UserCommandMessage.js.map +7 -0
- package/dist/components/messages/UserKodingInputMessage.js +18 -0
- package/dist/components/messages/UserKodingInputMessage.js.map +7 -0
- package/dist/components/messages/UserPromptMessage.js +20 -0
- package/dist/components/messages/UserPromptMessage.js.map +7 -0
- package/dist/components/messages/UserTextMessage.js +25 -0
- package/dist/components/messages/UserTextMessage.js.map +7 -0
- package/dist/components/messages/UserToolResultMessage/UserToolCanceledMessage.js +10 -0
- package/dist/components/messages/UserToolResultMessage/UserToolCanceledMessage.js.map +7 -0
- package/dist/components/messages/UserToolResultMessage/UserToolErrorMessage.js +15 -0
- package/dist/components/messages/UserToolResultMessage/UserToolErrorMessage.js.map +7 -0
- package/dist/components/messages/UserToolResultMessage/UserToolRejectMessage.js +25 -0
- package/dist/components/messages/UserToolResultMessage/UserToolRejectMessage.js.map +7 -0
- package/dist/components/messages/UserToolResultMessage/UserToolResultMessage.js +47 -0
- package/dist/components/messages/UserToolResultMessage/UserToolResultMessage.js.map +7 -0
- package/dist/components/messages/UserToolResultMessage/UserToolSuccessMessage.js +23 -0
- package/dist/components/messages/UserToolResultMessage/UserToolSuccessMessage.js.map +7 -0
- package/dist/components/messages/UserToolResultMessage/utils.js +42 -0
- package/dist/components/messages/UserToolResultMessage/utils.js.map +7 -0
- package/dist/components/permissions/BashPermissionRequest/BashPermissionRequest.js +112 -0
- package/dist/components/permissions/BashPermissionRequest/BashPermissionRequest.js.map +7 -0
- package/dist/components/permissions/FallbackPermissionRequest.js +131 -0
- package/dist/components/permissions/FallbackPermissionRequest.js.map +7 -0
- package/dist/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.js +159 -0
- package/dist/components/permissions/FileEditPermissionRequest/FileEditPermissionRequest.js.map +7 -0
- package/dist/components/permissions/FileEditPermissionRequest/FileEditToolDiff.js +58 -0
- package/dist/components/permissions/FileEditPermissionRequest/FileEditToolDiff.js.map +7 -0
- package/dist/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.js +153 -0
- package/dist/components/permissions/FileWritePermissionRequest/FileWritePermissionRequest.js.map +7 -0
- package/dist/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.js +70 -0
- package/dist/components/permissions/FileWritePermissionRequest/FileWriteToolDiff.js.map +7 -0
- package/dist/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.js +212 -0
- package/dist/components/permissions/FilesystemPermissionRequest/FilesystemPermissionRequest.js.map +7 -0
- package/dist/components/permissions/PermissionRequest.js +70 -0
- package/dist/components/permissions/PermissionRequest.js.map +7 -0
- package/dist/components/permissions/PermissionRequestTitle.js +52 -0
- package/dist/components/permissions/PermissionRequestTitle.js.map +7 -0
- package/dist/components/permissions/hooks.js +28 -0
- package/dist/components/permissions/hooks.js.map +7 -0
- package/dist/components/permissions/toolUseOptions.js +46 -0
- package/dist/components/permissions/toolUseOptions.js.map +7 -0
- package/dist/components/permissions/utils.js +21 -0
- package/dist/components/permissions/utils.js.map +7 -0
- package/dist/constants/betas.js +11 -0
- package/dist/constants/betas.js.map +7 -0
- package/dist/constants/claude-asterisk-ascii-art.js +242 -0
- package/dist/constants/claude-asterisk-ascii-art.js.map +7 -0
- package/dist/constants/figures.js +6 -0
- package/dist/constants/figures.js.map +7 -0
- package/dist/constants/keys.js +7 -0
- package/dist/constants/keys.js.map +7 -0
- package/dist/constants/macros.js +13 -0
- package/dist/constants/macros.js.map +7 -0
- package/dist/constants/modelCapabilities.js +154 -0
- package/dist/constants/modelCapabilities.js.map +7 -0
- package/dist/constants/models.js +1029 -0
- package/dist/constants/models.js.map +7 -0
- package/dist/constants/oauth.js +18 -0
- package/dist/constants/oauth.js.map +7 -0
- package/dist/constants/product.js +26 -0
- package/dist/constants/product.js.map +7 -0
- package/dist/constants/prompts.js +168 -0
- package/dist/constants/prompts.js.map +7 -0
- package/dist/constants/releaseNotes.js +9 -0
- package/dist/constants/releaseNotes.js.map +7 -0
- package/dist/context/PermissionContext.js +111 -0
- package/dist/context/PermissionContext.js.map +7 -0
- package/dist/context.js +259 -0
- package/dist/context.js.map +7 -0
- package/dist/cost-tracker.js +76 -0
- package/dist/cost-tracker.js.map +7 -0
- package/dist/entrypoints/cli.js +1101 -0
- package/dist/entrypoints/cli.js.map +7 -0
- package/dist/entrypoints/mcp.js +150 -0
- package/dist/entrypoints/mcp.js.map +7 -0
- package/dist/history.js +25 -0
- package/dist/history.js.map +7 -0
- package/dist/hooks/useApiKeyVerification.js +12 -0
- package/dist/hooks/useApiKeyVerification.js.map +7 -0
- package/dist/hooks/useArrowKeyHistory.js +50 -0
- package/dist/hooks/useArrowKeyHistory.js.map +7 -0
- package/dist/hooks/useCanUseTool.js +112 -0
- package/dist/hooks/useCanUseTool.js.map +7 -0
- package/dist/hooks/useCancelRequest.js +30 -0
- package/dist/hooks/useCancelRequest.js.map +7 -0
- package/dist/hooks/useDoublePress.js +31 -0
- package/dist/hooks/useDoublePress.js.map +7 -0
- package/dist/hooks/useExitOnCtrlCD.js +26 -0
- package/dist/hooks/useExitOnCtrlCD.js.map +7 -0
- package/dist/hooks/useInterval.js +18 -0
- package/dist/hooks/useInterval.js.map +7 -0
- package/dist/hooks/useLogMessages.js +14 -0
- package/dist/hooks/useLogMessages.js.map +7 -0
- package/dist/hooks/useLogStartupTime.js +15 -0
- package/dist/hooks/useLogStartupTime.js.map +7 -0
- package/dist/hooks/useNotifyAfterTimeout.js +42 -0
- package/dist/hooks/useNotifyAfterTimeout.js.map +7 -0
- package/dist/hooks/usePermissionRequestLogging.js +28 -0
- package/dist/hooks/usePermissionRequestLogging.js.map +7 -0
- package/dist/hooks/useTerminalSize.js +38 -0
- package/dist/hooks/useTerminalSize.js.map +7 -0
- package/dist/hooks/useTextInput.js +250 -0
- package/dist/hooks/useTextInput.js.map +7 -0
- package/dist/hooks/useUnifiedCompletion.js +929 -0
- package/dist/hooks/useUnifiedCompletion.js.map +7 -0
- package/dist/index.js +26 -0
- package/dist/index.js.map +7 -0
- package/dist/messages.js +33 -0
- package/dist/messages.js.map +7 -0
- package/dist/package.json +1 -0
- package/dist/permissions.js +194 -0
- package/dist/permissions.js.map +7 -0
- package/dist/query.js +492 -0
- package/dist/query.js.map +7 -0
- package/dist/screens/ConfigureNpmPrefix.js +128 -0
- package/dist/screens/ConfigureNpmPrefix.js.map +7 -0
- package/dist/screens/Doctor.js +143 -0
- package/dist/screens/Doctor.js.map +7 -0
- package/dist/screens/LogList.js +55 -0
- package/dist/screens/LogList.js.map +7 -0
- package/dist/screens/REPL.js +596 -0
- package/dist/screens/REPL.js.map +7 -0
- package/dist/screens/ResumeConversation.js +56 -0
- package/dist/screens/ResumeConversation.js.map +7 -0
- package/dist/services/adapters/base.js +29 -0
- package/dist/services/adapters/base.js.map +7 -0
- package/dist/services/adapters/chatCompletions.js +69 -0
- package/dist/services/adapters/chatCompletions.js.map +7 -0
- package/dist/services/adapters/responsesAPI.js +126 -0
- package/dist/services/adapters/responsesAPI.js.map +7 -0
- package/dist/services/browserMocks.js +48 -0
- package/dist/services/browserMocks.js.map +7 -0
- package/dist/services/claude.js +1605 -0
- package/dist/services/claude.js.map +7 -0
- package/dist/services/customCommands.js +359 -0
- package/dist/services/customCommands.js.map +7 -0
- package/dist/services/fileFreshness.js +280 -0
- package/dist/services/fileFreshness.js.map +7 -0
- package/dist/services/gpt5ConnectionTest.js +248 -0
- package/dist/services/gpt5ConnectionTest.js.map +7 -0
- package/dist/services/mcpClient.js +435 -0
- package/dist/services/mcpClient.js.map +7 -0
- package/dist/services/mcpServerApproval.js +55 -0
- package/dist/services/mcpServerApproval.js.map +7 -0
- package/dist/services/mentionProcessor.js +200 -0
- package/dist/services/mentionProcessor.js.map +7 -0
- package/dist/services/modelAdapterFactory.js +47 -0
- package/dist/services/modelAdapterFactory.js.map +7 -0
- package/dist/services/notifier.js +35 -0
- package/dist/services/notifier.js.map +7 -0
- package/dist/services/oauth.js +259 -0
- package/dist/services/oauth.js.map +7 -0
- package/dist/services/openai.js +998 -0
- package/dist/services/openai.js.map +7 -0
- package/dist/services/responseStateManager.js +68 -0
- package/dist/services/responseStateManager.js.map +7 -0
- package/dist/services/sentry.js +9 -0
- package/dist/services/sentry.js.map +7 -0
- package/dist/services/statsig.js +112 -0
- package/dist/services/statsig.js.map +7 -0
- package/dist/services/statsigStorage.js +75 -0
- package/dist/services/statsigStorage.js.map +7 -0
- package/dist/services/systemReminder.js +353 -0
- package/dist/services/systemReminder.js.map +7 -0
- package/dist/services/vcr.js +133 -0
- package/dist/services/vcr.js.map +7 -0
- package/dist/test/testAdapters.js +88 -0
- package/dist/test/testAdapters.js.map +1 -0
- package/dist/tools/ArchitectTool/ArchitectTool.js +119 -0
- package/dist/tools/ArchitectTool/ArchitectTool.js.map +7 -0
- package/dist/tools/ArchitectTool/prompt.js +18 -0
- package/dist/tools/ArchitectTool/prompt.js.map +7 -0
- package/dist/tools/AskExpertModelTool/AskExpertModelTool.js +423 -0
- package/dist/tools/AskExpertModelTool/AskExpertModelTool.js.map +7 -0
- package/dist/tools/BashTool/BashTool.js +188 -0
- package/dist/tools/BashTool/BashTool.js.map +7 -0
- package/dist/tools/BashTool/BashToolResultMessage.js +21 -0
- package/dist/tools/BashTool/BashToolResultMessage.js.map +7 -0
- package/dist/tools/BashTool/OutputLine.js +30 -0
- package/dist/tools/BashTool/OutputLine.js.map +7 -0
- package/dist/tools/BashTool/prompt.js +179 -0
- package/dist/tools/BashTool/prompt.js.map +7 -0
- package/dist/tools/BashTool/utils.js +51 -0
- package/dist/tools/BashTool/utils.js.map +7 -0
- package/dist/tools/FileEditTool/FileEditTool.js +228 -0
- package/dist/tools/FileEditTool/FileEditTool.js.map +7 -0
- package/dist/tools/FileEditTool/prompt.js +54 -0
- package/dist/tools/FileEditTool/prompt.js.map +7 -0
- package/dist/tools/FileEditTool/utils.js +42 -0
- package/dist/tools/FileEditTool/utils.js.map +7 -0
- package/dist/tools/FileReadTool/FileReadTool.js +272 -0
- package/dist/tools/FileReadTool/FileReadTool.js.map +7 -0
- package/dist/tools/FileReadTool/prompt.js +10 -0
- package/dist/tools/FileReadTool/prompt.js.map +7 -0
- package/dist/tools/FileWriteTool/FileWriteTool.js +204 -0
- package/dist/tools/FileWriteTool/FileWriteTool.js.map +7 -0
- package/dist/tools/FileWriteTool/prompt.js +14 -0
- package/dist/tools/FileWriteTool/prompt.js.map +7 -0
- package/dist/tools/GlobTool/GlobTool.js +88 -0
- package/dist/tools/GlobTool/GlobTool.js.map +7 -0
- package/dist/tools/GlobTool/prompt.js +12 -0
- package/dist/tools/GlobTool/prompt.js.map +7 -0
- package/dist/tools/GrepTool/GrepTool.js +107 -0
- package/dist/tools/GrepTool/GrepTool.js.map +7 -0
- package/dist/tools/GrepTool/prompt.js +15 -0
- package/dist/tools/GrepTool/prompt.js.map +7 -0
- package/dist/tools/MCPTool/MCPTool.js +90 -0
- package/dist/tools/MCPTool/MCPTool.js.map +7 -0
- package/dist/tools/MCPTool/prompt.js +7 -0
- package/dist/tools/MCPTool/prompt.js.map +7 -0
- package/dist/tools/MemoryReadTool/MemoryReadTool.js +103 -0
- package/dist/tools/MemoryReadTool/MemoryReadTool.js.map +7 -0
- package/dist/tools/MemoryReadTool/prompt.js +7 -0
- package/dist/tools/MemoryReadTool/prompt.js.map +7 -0
- package/dist/tools/MemoryWriteTool/MemoryWriteTool.js +77 -0
- package/dist/tools/MemoryWriteTool/MemoryWriteTool.js.map +7 -0
- package/dist/tools/MemoryWriteTool/prompt.js +7 -0
- package/dist/tools/MemoryWriteTool/prompt.js.map +7 -0
- package/dist/tools/MultiEditTool/MultiEditTool.js +293 -0
- package/dist/tools/MultiEditTool/MultiEditTool.js.map +7 -0
- package/dist/tools/MultiEditTool/prompt.js +48 -0
- package/dist/tools/MultiEditTool/prompt.js.map +7 -0
- package/dist/tools/NotebookEditTool/NotebookEditTool.js +238 -0
- package/dist/tools/NotebookEditTool/NotebookEditTool.js.map +7 -0
- package/dist/tools/NotebookEditTool/prompt.js +7 -0
- package/dist/tools/NotebookEditTool/prompt.js.map +7 -0
- package/dist/tools/NotebookReadTool/NotebookReadTool.js +212 -0
- package/dist/tools/NotebookReadTool/NotebookReadTool.js.map +7 -0
- package/dist/tools/NotebookReadTool/prompt.js +7 -0
- package/dist/tools/NotebookReadTool/prompt.js.map +7 -0
- package/dist/tools/StickerRequestTool/StickerRequestTool.js +86 -0
- package/dist/tools/StickerRequestTool/StickerRequestTool.js.map +7 -0
- package/dist/tools/StickerRequestTool/prompt.js +23 -0
- package/dist/tools/StickerRequestTool/prompt.js.map +7 -0
- package/dist/tools/TaskTool/TaskTool.js +308 -0
- package/dist/tools/TaskTool/TaskTool.js.map +7 -0
- package/dist/tools/TaskTool/constants.js +5 -0
- package/dist/tools/TaskTool/constants.js.map +7 -0
- package/dist/tools/TaskTool/prompt.js +82 -0
- package/dist/tools/TaskTool/prompt.js.map +7 -0
- package/dist/tools/ThinkTool/ThinkTool.js +48 -0
- package/dist/tools/ThinkTool/ThinkTool.js.map +7 -0
- package/dist/tools/ThinkTool/prompt.js +16 -0
- package/dist/tools/ThinkTool/prompt.js.map +7 -0
- package/dist/tools/TodoWriteTool/TodoWriteTool.js +216 -0
- package/dist/tools/TodoWriteTool/TodoWriteTool.js.map +7 -0
- package/dist/tools/TodoWriteTool/prompt.js +66 -0
- package/dist/tools/TodoWriteTool/prompt.js.map +7 -0
- package/dist/tools/URLFetcherTool/URLFetcherTool.js +137 -0
- package/dist/tools/URLFetcherTool/URLFetcherTool.js.map +7 -0
- package/dist/tools/URLFetcherTool/cache.js +45 -0
- package/dist/tools/URLFetcherTool/cache.js.map +7 -0
- package/dist/tools/URLFetcherTool/htmlToMarkdown.js +42 -0
- package/dist/tools/URLFetcherTool/htmlToMarkdown.js.map +7 -0
- package/dist/tools/URLFetcherTool/prompt.js +22 -0
- package/dist/tools/URLFetcherTool/prompt.js.map +7 -0
- package/dist/tools/WebSearchTool/WebSearchTool.js +86 -0
- package/dist/tools/WebSearchTool/WebSearchTool.js.map +7 -0
- package/dist/tools/WebSearchTool/prompt.js +17 -0
- package/dist/tools/WebSearchTool/prompt.js.map +7 -0
- package/dist/tools/WebSearchTool/searchProviders.js +48 -0
- package/dist/tools/WebSearchTool/searchProviders.js.map +7 -0
- package/dist/tools/lsTool/lsTool.js +201 -0
- package/dist/tools/lsTool/lsTool.js.map +7 -0
- package/dist/tools/lsTool/prompt.js +5 -0
- package/dist/tools/lsTool/prompt.js.map +7 -0
- package/dist/tools.js +64 -0
- package/dist/tools.js.map +7 -0
- package/dist/types/PermissionMode.js +82 -0
- package/dist/types/PermissionMode.js.map +7 -0
- package/dist/types/RequestContext.js +47 -0
- package/dist/types/RequestContext.js.map +7 -0
- package/dist/types/common.d.js +1 -0
- package/dist/types/common.d.js.map +7 -0
- package/dist/types/conversation.js +1 -0
- package/dist/types/conversation.js.map +7 -0
- package/dist/types/logs.js +1 -0
- package/dist/types/logs.js.map +7 -0
- package/dist/types/modelCapabilities.js +1 -0
- package/dist/types/modelCapabilities.js.map +7 -0
- package/dist/types/notebook.js +1 -0
- package/dist/types/notebook.js.map +7 -0
- package/dist/utils/Cursor.js +313 -0
- package/dist/utils/Cursor.js.map +7 -0
- package/dist/utils/PersistentShell.js +382 -0
- package/dist/utils/PersistentShell.js.map +7 -0
- package/dist/utils/advancedFuzzyMatcher.js +206 -0
- package/dist/utils/advancedFuzzyMatcher.js.map +7 -0
- package/dist/utils/agentLoader.js +199 -0
- package/dist/utils/agentLoader.js.map +7 -0
- package/dist/utils/agentStorage.js +59 -0
- package/dist/utils/agentStorage.js.map +7 -0
- package/dist/utils/array.js +7 -0
- package/dist/utils/array.js.map +7 -0
- package/dist/utils/ask.js +77 -0
- package/dist/utils/ask.js.map +7 -0
- package/dist/utils/auth.js +11 -0
- package/dist/utils/auth.js.map +7 -0
- package/dist/utils/autoCompactCore.js +149 -0
- package/dist/utils/autoCompactCore.js.map +7 -0
- package/dist/utils/autoUpdater.js +362 -0
- package/dist/utils/autoUpdater.js.map +7 -0
- package/dist/utils/betas.js +21 -0
- package/dist/utils/betas.js.map +7 -0
- package/dist/utils/browser.js +15 -0
- package/dist/utils/browser.js.map +7 -0
- package/dist/utils/cleanup.js +54 -0
- package/dist/utils/cleanup.js.map +7 -0
- package/dist/utils/commands.js +207 -0
- package/dist/utils/commands.js.map +7 -0
- package/dist/utils/commonUnixCommands.js +687 -0
- package/dist/utils/commonUnixCommands.js.map +7 -0
- package/dist/utils/config.js +655 -0
- package/dist/utils/config.js.map +7 -0
- package/dist/utils/conversationRecovery.js +35 -0
- package/dist/utils/conversationRecovery.js.map +7 -0
- package/dist/utils/debugLogger.js +891 -0
- package/dist/utils/debugLogger.js.map +7 -0
- package/dist/utils/diff.js +32 -0
- package/dist/utils/diff.js.map +7 -0
- package/dist/utils/env.js +44 -0
- package/dist/utils/env.js.map +7 -0
- package/dist/utils/errors.js +23 -0
- package/dist/utils/errors.js.map +7 -0
- package/dist/utils/exampleCommands.js +80 -0
- package/dist/utils/exampleCommands.js.map +7 -0
- package/dist/utils/execFileNoThrow.js +44 -0
- package/dist/utils/execFileNoThrow.js.map +7 -0
- package/dist/utils/expertChatStorage.js +78 -0
- package/dist/utils/expertChatStorage.js.map +7 -0
- package/dist/utils/file.js +282 -0
- package/dist/utils/file.js.map +7 -0
- package/dist/utils/fileRecoveryCore.js +41 -0
- package/dist/utils/fileRecoveryCore.js.map +7 -0
- package/dist/utils/format.js +41 -0
- package/dist/utils/format.js.map +7 -0
- package/dist/utils/fuzzyMatcher.js +252 -0
- package/dist/utils/fuzzyMatcher.js.map +7 -0
- package/dist/utils/generators.js +46 -0
- package/dist/utils/generators.js.map +7 -0
- package/dist/utils/git.js +83 -0
- package/dist/utils/git.js.map +7 -0
- package/dist/utils/globalLogger.js +54 -0
- package/dist/utils/globalLogger.js.map +7 -0
- package/dist/utils/http.js +7 -0
- package/dist/utils/http.js.map +7 -0
- package/dist/utils/imagePaste.js +29 -0
- package/dist/utils/imagePaste.js.map +7 -0
- package/dist/utils/json.js +16 -0
- package/dist/utils/json.js.map +7 -0
- package/dist/utils/log.js +298 -0
- package/dist/utils/log.js.map +7 -0
- package/dist/utils/markdown.js +187 -0
- package/dist/utils/markdown.js.map +7 -0
- package/dist/utils/messageContextManager.js +195 -0
- package/dist/utils/messageContextManager.js.map +7 -0
- package/dist/utils/messages.js +633 -0
- package/dist/utils/messages.js.map +7 -0
- package/dist/utils/model.js +687 -0
- package/dist/utils/model.js.map +7 -0
- package/dist/utils/permissions/filesystem.js +80 -0
- package/dist/utils/permissions/filesystem.js.map +7 -0
- package/dist/utils/responseState.js +20 -0
- package/dist/utils/responseState.js.map +7 -0
- package/dist/utils/ripgrep.js +131 -0
- package/dist/utils/ripgrep.js.map +7 -0
- package/dist/utils/secureFile.js +483 -0
- package/dist/utils/secureFile.js.map +7 -0
- package/dist/utils/sessionState.js +31 -0
- package/dist/utils/sessionState.js.map +7 -0
- package/dist/utils/state.js +24 -0
- package/dist/utils/state.js.map +7 -0
- package/dist/utils/style.js +31 -0
- package/dist/utils/style.js.map +7 -0
- package/dist/utils/terminal.js +43 -0
- package/dist/utils/terminal.js.map +7 -0
- package/dist/utils/theme.js +102 -0
- package/dist/utils/theme.js.map +7 -0
- package/dist/utils/thinking.js +103 -0
- package/dist/utils/thinking.js.map +7 -0
- package/dist/utils/todoStorage.js +291 -0
- package/dist/utils/todoStorage.js.map +7 -0
- package/dist/utils/tokens.js +30 -0
- package/dist/utils/tokens.js.map +7 -0
- package/dist/utils/toolExecutionController.js +109 -0
- package/dist/utils/toolExecutionController.js.map +7 -0
- package/dist/utils/unaryLogging.js +14 -0
- package/dist/utils/unaryLogging.js.map +7 -0
- package/dist/utils/user.js +40 -0
- package/dist/utils/user.js.map +7 -0
- package/dist/utils/validate.js +132 -0
- package/dist/utils/validate.js.map +7 -0
- package/dist/yoga.wasm +0 -0
- package/package.json +28 -7
- package/src/Tool.ts +4 -3
- package/src/commands/agents.tsx +10 -4
- package/src/components/messages/AssistantToolUseMessage.tsx +5 -6
- package/src/constants/macros.ts +5 -2
- package/src/entrypoints/cli.tsx +38 -19
- package/src/entrypoints/mcp.ts +1 -2
- package/src/hooks/useDoublePress.ts +0 -1
- package/src/hooks/useTextInput.ts +4 -5
- package/src/hooks/useUnifiedCompletion.ts +2 -2
- package/src/index.ts +34 -0
- package/src/query.ts +13 -8
- package/src/screens/Doctor.tsx +1 -1
- package/src/screens/REPL.tsx +13 -9
- package/src/services/openai.ts +25 -4
- package/src/tools/ArchitectTool/ArchitectTool.tsx +18 -5
- package/src/tools/AskExpertModelTool/AskExpertModelTool.tsx +21 -14
- package/src/tools/FileEditTool/FileEditTool.tsx +6 -2
- package/src/tools/FileWriteTool/FileWriteTool.tsx +7 -3
- package/src/tools/MultiEditTool/MultiEditTool.tsx +26 -4
- package/src/tools/NotebookReadTool/NotebookReadTool.tsx +1 -1
- package/src/tools/StickerRequestTool/StickerRequestTool.tsx +28 -14
- package/src/tools/TaskTool/TaskTool.tsx +8 -36
- package/src/types/common.d.ts +2 -0
- package/src/utils/generators.ts +1 -1
- package/src/utils/messageContextManager.ts +5 -0
- package/src/utils/messages.tsx +8 -2
- 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": "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",
|
|
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
|
+
}
|