@within-7/minto 0.3.9 → 0.4.0
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/dist/Tool.js.map +2 -2
- package/dist/commands/agents/AgentsCommand.js +461 -657
- package/dist/commands/agents/AgentsCommand.js.map +2 -2
- package/dist/commands/agents/types.js +1 -0
- package/dist/commands/agents/types.js.map +2 -2
- package/dist/commands/agents/utils/fileOperations.js +96 -36
- package/dist/commands/agents/utils/fileOperations.js.map +3 -3
- package/dist/commands/agents/utils/index.js +3 -1
- package/dist/commands/agents/utils/index.js.map +2 -2
- package/dist/commands/context.js +54 -23
- package/dist/commands/context.js.map +2 -2
- package/dist/commands/ctx_viz.js +1 -1
- package/dist/commands/effort.js +87 -0
- package/dist/commands/effort.js.map +7 -0
- package/dist/commands/export.js +684 -94
- package/dist/commands/export.js.map +2 -2
- package/dist/commands/ide.js +18 -0
- package/dist/commands/ide.js.map +7 -0
- package/dist/commands/language.js +19 -46
- package/dist/commands/language.js.map +2 -2
- package/dist/commands/mcp-interactive.js +425 -217
- package/dist/commands/mcp-interactive.js.map +2 -2
- package/dist/commands/memory.js +168 -0
- package/dist/commands/memory.js.map +7 -0
- package/dist/commands/model.js +457 -65
- package/dist/commands/model.js.map +2 -2
- package/dist/commands/outputStyle.js +64 -0
- package/dist/commands/outputStyle.js.map +7 -0
- package/dist/commands/permissions.js +75 -49
- package/dist/commands/permissions.js.map +2 -2
- package/dist/commands/plugin/utils.js +33 -1
- package/dist/commands/plugin/utils.js.map +2 -2
- package/dist/commands/plugin.js +891 -185
- package/dist/commands/plugin.js.map +3 -3
- package/dist/commands/refreshCommands.js +2 -0
- package/dist/commands/refreshCommands.js.map +2 -2
- package/dist/commands/resume.js +1 -1
- package/dist/commands/resume.js.map +1 -1
- package/dist/commands/review.js +51 -0
- package/dist/commands/review.js.map +7 -0
- package/dist/commands/sandbox.js +168 -70
- package/dist/commands/sandbox.js.map +2 -2
- package/dist/commands/setup.js +593 -107
- package/dist/commands/setup.js.map +2 -2
- package/dist/commands/stats.js +188 -131
- package/dist/commands/stats.js.map +2 -2
- package/dist/commands/status.js +75 -13
- package/dist/commands/status.js.map +2 -2
- package/dist/commands/terminalSetup.js +6 -0
- package/dist/commands/terminalSetup.js.map +2 -2
- package/dist/commands/undo.js +146 -174
- package/dist/commands/undo.js.map +2 -2
- package/dist/commands/vim.js +22 -0
- package/dist/commands/vim.js.map +7 -0
- package/dist/commands.js +12 -0
- package/dist/commands.js.map +2 -2
- package/dist/components/Help.js +165 -32
- package/dist/components/Help.js.map +2 -2
- package/dist/components/HighlightedCode.js +1 -0
- package/dist/components/HighlightedCode.js.map +2 -2
- package/dist/components/InfoPanel/InfoPanel.js +123 -0
- package/dist/components/InfoPanel/InfoPanel.js.map +7 -0
- package/dist/components/InfoPanel/index.js +5 -0
- package/dist/components/InfoPanel/index.js.map +7 -0
- package/dist/components/InfoPanel/types.js +1 -0
- package/dist/components/InfoPanel/types.js.map +7 -0
- package/dist/components/ModelSelector/BrandTextInput.js +43 -0
- package/dist/components/ModelSelector/BrandTextInput.js.map +7 -0
- package/dist/components/ModelSelector/ModelSelector.js +590 -565
- package/dist/components/ModelSelector/ModelSelector.js.map +2 -2
- package/dist/components/ModelSelector/WizardContainer.js +45 -0
- package/dist/components/ModelSelector/WizardContainer.js.map +7 -0
- package/dist/components/ModelSelector/index.js +1 -3
- package/dist/components/ModelSelector/index.js.map +2 -2
- package/dist/components/PromptInput.js +26 -11
- package/dist/components/PromptInput.js.map +2 -2
- package/dist/components/PulseLabel.js +44 -0
- package/dist/components/PulseLabel.js.map +7 -0
- package/dist/components/RequestStatusIndicator.js +1 -1
- package/dist/components/RequestStatusIndicator.js.map +1 -1
- package/dist/components/SimpleSelector/SimpleSelector.js +154 -0
- package/dist/components/SimpleSelector/SimpleSelector.js.map +7 -0
- package/dist/components/SimpleSelector/index.js +5 -0
- package/dist/components/SimpleSelector/index.js.map +7 -0
- package/dist/components/SimpleSelector/types.js +1 -0
- package/dist/components/SimpleSelector/types.js.map +7 -0
- package/dist/components/Spinner.js +12 -42
- package/dist/components/Spinner.js.map +3 -3
- package/dist/components/StartupStatus.js +57 -0
- package/dist/components/StartupStatus.js.map +7 -0
- package/dist/components/StatusOverlayContent.js +21 -0
- package/dist/components/StatusOverlayContent.js.map +7 -0
- package/dist/components/SubagentBlock.js +43 -6
- package/dist/components/SubagentBlock.js.map +2 -2
- package/dist/components/TabbedListView/ScrollableList.js +31 -5
- package/dist/components/TabbedListView/ScrollableList.js.map +2 -2
- package/dist/components/TabbedListView/TabBar.js +13 -8
- package/dist/components/TabbedListView/TabBar.js.map +2 -2
- package/dist/components/TabbedListView/TabbedListView.js +123 -48
- package/dist/components/TabbedListView/TabbedListView.js.map +2 -2
- package/dist/components/TodoPanel.js +1 -1
- package/dist/components/TodoPanel.js.map +1 -1
- package/dist/components/ToolUseLoader.js +5 -0
- package/dist/components/ToolUseLoader.js.map +2 -2
- package/dist/components/TrustDialog.js +0 -2
- package/dist/components/TrustDialog.js.map +2 -2
- package/dist/components/messages/TaskInModuleView.js +1 -1
- package/dist/components/messages/TaskInModuleView.js.map +2 -2
- package/dist/components/messages/TaskToolMessage.js +1 -1
- package/dist/components/messages/TaskToolMessage.js.map +2 -2
- package/dist/components/messages/UserPromptMessage.js +6 -1
- package/dist/components/messages/UserPromptMessage.js.map +2 -2
- package/dist/constants/modelCapabilities.js +103 -18
- package/dist/constants/modelCapabilities.js.map +2 -2
- package/dist/constants/product.js +2 -0
- package/dist/constants/product.js.map +2 -2
- package/dist/constants/prompts/agentPrompt.js +30 -0
- package/dist/constants/prompts/agentPrompt.js.map +7 -0
- package/dist/constants/prompts/codeConventions.js +27 -0
- package/dist/constants/prompts/codeConventions.js.map +7 -0
- package/dist/constants/prompts/doingTasks.js +15 -0
- package/dist/constants/prompts/doingTasks.js.map +7 -0
- package/dist/constants/prompts/envInfo.js +17 -0
- package/dist/constants/prompts/envInfo.js.map +7 -0
- package/dist/constants/prompts/executingWithCare.js +17 -0
- package/dist/constants/prompts/executingWithCare.js.map +7 -0
- package/dist/constants/prompts/identity.js +10 -0
- package/dist/constants/prompts/identity.js.map +7 -0
- package/dist/constants/prompts/index.js +78 -0
- package/dist/constants/prompts/index.js.map +7 -0
- package/dist/constants/prompts/taskManagement.js +60 -0
- package/dist/constants/prompts/taskManagement.js.map +7 -0
- package/dist/constants/prompts/toneAndStyle.js +62 -0
- package/dist/constants/prompts/toneAndStyle.js.map +7 -0
- package/dist/constants/prompts/toolUsagePolicy.js +38 -0
- package/dist/constants/prompts/toolUsagePolicy.js.map +7 -0
- package/dist/constants/prompts.js +5 -176
- package/dist/constants/prompts.js.map +2 -2
- package/dist/constants/providerRegistry.js +235 -0
- package/dist/constants/providerRegistry.js.map +7 -0
- package/dist/constants/providers.js +35 -0
- package/dist/constants/providers.js.map +7 -0
- package/dist/context/PermissionContext.js +0 -1
- package/dist/context/PermissionContext.js.map +2 -2
- package/dist/context.js +87 -31
- package/dist/context.js.map +2 -2
- package/dist/core/backupHook.js +29 -0
- package/dist/core/backupHook.js.map +7 -0
- package/dist/core/config/defaults.js +11 -2
- package/dist/core/config/defaults.js.map +2 -2
- package/dist/core/config/schema.js +21 -3
- package/dist/core/config/schema.js.map +2 -2
- package/dist/core/costTracker.js +18 -16
- package/dist/core/costTracker.js.map +2 -2
- package/dist/core/index.js +0 -1
- package/dist/core/index.js.map +2 -2
- package/dist/core/tokenStatsManager.js +22 -4
- package/dist/core/tokenStatsManager.js.map +2 -2
- package/dist/cost-tracker.js +0 -16
- package/dist/cost-tracker.js.map +2 -2
- package/dist/entrypoints/bootstrap.js +3 -1
- package/dist/entrypoints/bootstrap.js.map +2 -2
- package/dist/entrypoints/cli.js +81 -68
- package/dist/entrypoints/cli.js.map +2 -2
- package/dist/hooks/useAgentTokenStats.js +1 -1
- package/dist/hooks/useAgentTokenStats.js.map +2 -2
- package/dist/hooks/useAgentTranscripts.js +2 -1
- package/dist/hooks/useAgentTranscripts.js.map +2 -2
- package/dist/hooks/useBackgroundShells.js +29 -0
- package/dist/hooks/useBackgroundShells.js.map +7 -0
- package/dist/hooks/useCanUseTool.js +1 -1
- package/dist/hooks/useCanUseTool.js.map +2 -2
- package/dist/hooks/useDeferredLoading.js +64 -0
- package/dist/hooks/useDeferredLoading.js.map +7 -0
- package/dist/hooks/useHookStatus.js +1 -1
- package/dist/hooks/useHookStatus.js.map +2 -2
- package/dist/hooks/useSessionTracking.js +55 -0
- package/dist/hooks/useSessionTracking.js.map +7 -0
- package/dist/hooks/useTerminalSize.js +21 -0
- package/dist/hooks/useTerminalSize.js.map +2 -2
- package/dist/hooks/useTextInput.js +1 -0
- package/dist/hooks/useTextInput.js.map +2 -2
- package/dist/hooks/useUnifiedCompletion.js +3 -2
- package/dist/hooks/useUnifiedCompletion.js.map +2 -2
- package/dist/i18n/locales/en.js +299 -1
- package/dist/i18n/locales/en.js.map +2 -2
- package/dist/i18n/locales/zh-CN.js +300 -2
- package/dist/i18n/locales/zh-CN.js.map +2 -2
- package/dist/i18n/types.js.map +1 -1
- package/dist/messages.js +41 -17
- package/dist/messages.js.map +2 -2
- package/dist/permissions.js +94 -1
- package/dist/permissions.js.map +2 -2
- package/dist/query.js +27 -19
- package/dist/query.js.map +2 -2
- package/dist/screens/REPL.js +83 -74
- package/dist/screens/REPL.js.map +2 -2
- package/dist/services/adapters/responsesAPI.js +6 -0
- package/dist/services/adapters/responsesAPI.js.map +2 -2
- package/dist/services/agentTeams/index.js +35 -0
- package/dist/services/agentTeams/index.js.map +7 -0
- package/dist/services/agentTeams/mailbox.js +114 -0
- package/dist/services/agentTeams/mailbox.js.map +7 -0
- package/dist/services/agentTeams/teamManager.js +149 -0
- package/dist/services/agentTeams/teamManager.js.map +7 -0
- package/dist/services/agentTeams/teamTaskStore.js +114 -0
- package/dist/services/agentTeams/teamTaskStore.js.map +7 -0
- package/dist/services/agentTeams/teammateSpawner.js +80 -0
- package/dist/services/agentTeams/teammateSpawner.js.map +7 -0
- package/dist/services/checkpointManager.js +16 -3
- package/dist/services/checkpointManager.js.map +2 -2
- package/dist/services/claude.js +19 -1728
- package/dist/services/claude.js.map +3 -3
- package/dist/services/customCommands.js +30 -8
- package/dist/services/customCommands.js.map +2 -2
- package/dist/services/gpt5ConnectionTest.js +4 -2
- package/dist/services/gpt5ConnectionTest.js.map +2 -2
- package/dist/services/hookExecutor.js +411 -127
- package/dist/services/hookExecutor.js.map +2 -2
- package/dist/services/llm/anthropicProvider.js +807 -0
- package/dist/services/llm/anthropicProvider.js.map +7 -0
- package/dist/services/llm/dispatch.js +218 -0
- package/dist/services/llm/dispatch.js.map +7 -0
- package/dist/services/llm/index.js +44 -0
- package/dist/services/llm/index.js.map +7 -0
- package/dist/services/llm/mintoContext.js +69 -0
- package/dist/services/llm/mintoContext.js.map +7 -0
- package/dist/services/llm/openaiProvider.js +622 -0
- package/dist/services/llm/openaiProvider.js.map +7 -0
- package/dist/services/llm/types.js +157 -0
- package/dist/services/llm/types.js.map +7 -0
- package/dist/services/mcpClient.js +183 -33
- package/dist/services/mcpClient.js.map +2 -2
- package/dist/services/notifier.js +14 -0
- package/dist/services/notifier.js.map +2 -2
- package/dist/services/oauth.js +4 -2
- package/dist/services/oauth.js.map +2 -2
- package/dist/services/openai.js +66 -56
- package/dist/services/openai.js.map +3 -3
- package/dist/services/outputStyles.js +102 -21
- package/dist/services/outputStyles.js.map +2 -2
- package/dist/services/plugins/lspServers.js +1 -1
- package/dist/services/plugins/lspServers.js.map +2 -2
- package/dist/services/plugins/pluginRuntime.js +2 -1
- package/dist/services/plugins/pluginRuntime.js.map +2 -2
- package/dist/services/plugins/pluginValidation.js +10 -3
- package/dist/services/plugins/pluginValidation.js.map +2 -2
- package/dist/services/plugins/skillMarketplace.js +20 -9
- package/dist/services/plugins/skillMarketplace.js.map +2 -2
- package/dist/services/sentry.js +1 -1
- package/dist/services/sentry.js.map +2 -2
- package/dist/services/sessionMemory.js +16 -3
- package/dist/services/sessionMemory.js.map +2 -2
- package/dist/services/systemReminder.js +367 -9
- package/dist/services/systemReminder.js.map +2 -2
- package/dist/services/taskStore.js +19 -0
- package/dist/services/taskStore.js.map +2 -2
- package/dist/tools/ArchitectTool/ArchitectTool.js.map +1 -1
- package/dist/tools/AskUserQuestionTool/AskUserQuestionTool.js.map +1 -1
- package/dist/tools/BashOutputTool/BashOutputTool.js.map +1 -1
- package/dist/tools/BashTool/BashTool.js +28 -0
- package/dist/tools/BashTool/BashTool.js.map +2 -2
- package/dist/tools/FileEditTool/FileEditTool.js +8 -1
- package/dist/tools/FileEditTool/FileEditTool.js.map +2 -2
- package/dist/tools/FileReadTool/FileReadTool.js +14 -0
- package/dist/tools/FileReadTool/FileReadTool.js.map +2 -2
- package/dist/tools/FileWriteTool/FileWriteTool.js +10 -1
- package/dist/tools/FileWriteTool/FileWriteTool.js.map +2 -2
- package/dist/tools/GlobTool/GlobTool.js.map +1 -1
- package/dist/tools/GrepTool/GrepTool.js.map +1 -1
- package/dist/tools/KillShellTool/KillShellTool.js.map +1 -1
- package/dist/tools/ListMcpResourcesTool/ListMcpResourcesTool.js.map +2 -2
- package/dist/tools/LspTool/LspTool.js +11 -2
- package/dist/tools/LspTool/LspTool.js.map +2 -2
- package/dist/tools/MCPTool/MCPTool.js.map +1 -1
- package/dist/tools/MemoryReadTool/MemoryReadTool.js +2 -1
- package/dist/tools/MemoryReadTool/MemoryReadTool.js.map +2 -2
- package/dist/tools/MemoryWriteTool/MemoryWriteTool.js +2 -1
- package/dist/tools/MemoryWriteTool/MemoryWriteTool.js.map +2 -2
- package/dist/tools/MultiEditTool/MultiEditTool.js +7 -0
- package/dist/tools/MultiEditTool/MultiEditTool.js.map +2 -2
- package/dist/tools/NotebookEditTool/NotebookEditTool.js +2 -0
- package/dist/tools/NotebookEditTool/NotebookEditTool.js.map +2 -2
- package/dist/tools/NotebookReadTool/NotebookReadTool.js.map +1 -1
- package/dist/tools/PlanModeTool/EnterPlanModeTool.js +8 -2
- package/dist/tools/PlanModeTool/EnterPlanModeTool.js.map +2 -2
- package/dist/tools/PlanModeTool/ExitPlanModeTool.js +2 -0
- package/dist/tools/PlanModeTool/ExitPlanModeTool.js.map +2 -2
- package/dist/tools/ReadMcpResourceTool/ReadMcpResourceTool.js.map +1 -1
- package/dist/tools/SlashCommandTool/SlashCommandTool.js +174 -18
- package/dist/tools/SlashCommandTool/SlashCommandTool.js.map +3 -3
- package/dist/tools/TaskCreateTool/TaskCreateTool.js.map +1 -1
- package/dist/tools/TaskGetTool/TaskGetTool.js.map +1 -1
- package/dist/tools/TaskListTool/TaskListTool.js.map +1 -1
- package/dist/tools/TaskOutputTool/TaskOutputTool.js.map +1 -1
- package/dist/tools/TaskStopTool/TaskStopTool.js.map +1 -1
- package/dist/tools/TaskTool/TaskTool.js +84 -11
- package/dist/tools/TaskTool/TaskTool.js.map +2 -2
- package/dist/tools/TaskTool/prompt.js +12 -6
- package/dist/tools/TaskTool/prompt.js.map +2 -2
- package/dist/tools/TaskUpdateTool/TaskUpdateTool.js.map +1 -1
- package/dist/tools/ThinkTool/ThinkTool.js.map +1 -1
- package/dist/tools/TodoWriteTool/TodoWriteTool.js.map +1 -1
- package/dist/tools/URLFetcherTool/URLFetcherTool.js.map +1 -1
- package/dist/tools/WebSearchTool/WebSearchTool.js.map +1 -1
- package/dist/tools/WebSearchTool/searchProviders.js +2 -1
- package/dist/tools/WebSearchTool/searchProviders.js.map +2 -2
- package/dist/tools/lsTool/lsTool.js.map +2 -2
- package/dist/tools/lsTool/prompt.js.map +1 -1
- package/dist/tools.js +14 -3
- package/dist/tools.js.map +2 -2
- package/dist/types/PermissionMode.js +21 -1
- package/dist/types/PermissionMode.js.map +2 -2
- package/dist/types/agentTeams.js +1 -0
- package/dist/types/agentTeams.js.map +7 -0
- package/dist/types/hooks.js +8 -2
- package/dist/types/hooks.js.map +2 -2
- package/dist/types/plugin.js +3 -5
- package/dist/types/plugin.js.map +2 -2
- package/dist/utils/agentHookExecutor.js +1 -4
- package/dist/utils/agentHookExecutor.js.map +2 -2
- package/dist/utils/agentLoader.js +91 -15
- package/dist/utils/agentLoader.js.map +2 -2
- package/dist/utils/agentMemory.js.map +2 -2
- package/dist/utils/animationManager.js +1 -1
- package/dist/utils/animationManager.js.map +2 -2
- package/dist/utils/ask.js +1 -1
- package/dist/utils/async.js +5 -1
- package/dist/utils/async.js.map +2 -2
- package/dist/utils/autoCompactCore.js +60 -0
- package/dist/utils/autoCompactCore.js.map +2 -2
- package/dist/utils/claudeCodeSync.js +439 -0
- package/dist/utils/claudeCodeSync.js.map +7 -0
- package/dist/utils/config.js +27 -151
- package/dist/utils/config.js.map +2 -2
- package/dist/utils/configSchema.js +227 -0
- package/dist/utils/configSchema.js.map +7 -0
- package/dist/utils/debugLogger.js.map +2 -2
- package/dist/utils/env.js +4 -3
- package/dist/utils/env.js.map +2 -2
- package/dist/utils/envConfig.js +34 -0
- package/dist/utils/envConfig.js.map +3 -3
- package/dist/utils/execFileNoThrow.js +2 -1
- package/dist/utils/execFileNoThrow.js.map +2 -2
- package/dist/utils/gpt5.js +146 -0
- package/dist/utils/gpt5.js.map +7 -0
- package/dist/utils/hookManager.js +374 -140
- package/dist/utils/hookManager.js.map +2 -2
- package/dist/utils/markdown.js +47 -0
- package/dist/utils/markdown.js.map +2 -2
- package/dist/utils/marketplaceManager.js +80 -43
- package/dist/utils/marketplaceManager.js.map +2 -2
- package/dist/utils/memoizeWithTTL.js +25 -0
- package/dist/utils/memoizeWithTTL.js.map +7 -0
- package/dist/utils/messages.js +2 -2
- package/dist/utils/messages.js.map +2 -2
- package/dist/utils/model.js +34 -9
- package/dist/utils/model.js.map +2 -2
- package/dist/utils/pluginInstaller.js +68 -29
- package/dist/utils/pluginInstaller.js.map +2 -2
- package/dist/utils/pluginLoader.js +249 -57
- package/dist/utils/pluginLoader.js.map +2 -2
- package/dist/utils/repoFetcher.js +110 -0
- package/dist/utils/repoFetcher.js.map +7 -0
- package/dist/utils/safeFetch.js +45 -0
- package/dist/utils/safeFetch.js.map +7 -0
- package/dist/utils/skillLoader.js +77 -12
- package/dist/utils/skillLoader.js.map +2 -2
- package/dist/utils/streamingState.js +52 -0
- package/dist/utils/streamingState.js.map +7 -0
- package/dist/utils/stringSubstitution.js +4 -5
- package/dist/utils/stringSubstitution.js.map +2 -2
- package/dist/utils/style.js +6 -3
- package/dist/utils/style.js.map +2 -2
- package/dist/utils/teamConfig.js +162 -16
- package/dist/utils/teamConfig.js.map +2 -2
- package/dist/utils/terminal.js +1 -1
- package/dist/utils/terminal.js.map +2 -2
- package/dist/utils/toolRiskClassification.js +0 -6
- package/dist/utils/toolRiskClassification.js.map +2 -2
- package/dist/version.js +2 -2
- package/dist/version.js.map +1 -1
- package/package.json +7 -6
|
@@ -217,12 +217,24 @@ async function scanAgentDirectory(dirPath, location) {
|
|
|
217
217
|
if (!stat.isFile()) continue;
|
|
218
218
|
try {
|
|
219
219
|
const content = readFileSync(filePath, "utf-8");
|
|
220
|
+
if (!content.trim()) continue;
|
|
220
221
|
const { data: frontmatter, content: body } = matter(content);
|
|
221
222
|
if (!frontmatter.name || !frontmatter.description) {
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
)
|
|
225
|
-
|
|
223
|
+
const headingMatch = body.match(/^#\s+(.+)/m);
|
|
224
|
+
const firstParagraph = body.split(/\n\n/).find((p) => p.trim() && !p.trim().startsWith("#"));
|
|
225
|
+
if (headingMatch) {
|
|
226
|
+
if (!frontmatter.name) {
|
|
227
|
+
frontmatter.name = headingMatch[1].trim();
|
|
228
|
+
}
|
|
229
|
+
if (!frontmatter.description) {
|
|
230
|
+
frontmatter.description = firstParagraph?.trim() || frontmatter.name;
|
|
231
|
+
}
|
|
232
|
+
} else {
|
|
233
|
+
console.warn(
|
|
234
|
+
`Skipping ${filePath}: missing required fields (name, description)`
|
|
235
|
+
);
|
|
236
|
+
continue;
|
|
237
|
+
}
|
|
226
238
|
}
|
|
227
239
|
if (frontmatter.model && !frontmatter.model_name && !warnedAgents.has(frontmatter.name) && process.env.MINTO_DEBUG_AGENTS) {
|
|
228
240
|
console.warn(
|
|
@@ -230,7 +242,9 @@ async function scanAgentDirectory(dirPath, location) {
|
|
|
230
242
|
);
|
|
231
243
|
warnedAgents.add(frontmatter.name);
|
|
232
244
|
}
|
|
233
|
-
const disallowedTools = parseDisallowedTools(
|
|
245
|
+
const disallowedTools = parseDisallowedTools(
|
|
246
|
+
frontmatter.disallowedTools
|
|
247
|
+
);
|
|
234
248
|
const skills = parseSkills(frontmatter.skills);
|
|
235
249
|
const memory = parseMemory(frontmatter.memory);
|
|
236
250
|
const permissionMode = parsePermissionMode(frontmatter.permissionMode);
|
|
@@ -240,6 +254,7 @@ async function scanAgentDirectory(dirPath, location) {
|
|
|
240
254
|
tools: parseTools(frontmatter.tools),
|
|
241
255
|
systemPrompt: body.trim(),
|
|
242
256
|
location,
|
|
257
|
+
sourcePath: filePath,
|
|
243
258
|
// Optional fields - only include if defined
|
|
244
259
|
...frontmatter.color && { color: frontmatter.color },
|
|
245
260
|
...frontmatter.model_name && { model_name: frontmatter.model_name },
|
|
@@ -250,7 +265,12 @@ async function scanAgentDirectory(dirPath, location) {
|
|
|
250
265
|
...frontmatter.maxThinkingTokens && {
|
|
251
266
|
maxThinkingTokens: Number(frontmatter.maxThinkingTokens)
|
|
252
267
|
},
|
|
253
|
-
...frontmatter.hooks && { hooks: frontmatter.hooks }
|
|
268
|
+
...frontmatter.hooks && { hooks: frontmatter.hooks },
|
|
269
|
+
...frontmatter.isolation && { isolation: frontmatter.isolation },
|
|
270
|
+
...frontmatter.background && {
|
|
271
|
+
background: frontmatter.background === true
|
|
272
|
+
},
|
|
273
|
+
...frontmatter.mcpServers && { mcpServers: frontmatter.mcpServers }
|
|
254
274
|
};
|
|
255
275
|
agents.push(agent);
|
|
256
276
|
} catch (error) {
|
|
@@ -271,12 +291,24 @@ async function loadPluginAgents() {
|
|
|
271
291
|
for (const pluginAgent of plugin.agents) {
|
|
272
292
|
try {
|
|
273
293
|
const content = readFileSync(pluginAgent.filePath, "utf-8");
|
|
294
|
+
if (!content.trim()) continue;
|
|
274
295
|
const { data: frontmatter, content: body } = matter(content);
|
|
275
296
|
if (!frontmatter.name || !frontmatter.description) {
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
)
|
|
279
|
-
|
|
297
|
+
const headingMatch = body.match(/^#\s+(.+)/m);
|
|
298
|
+
const firstParagraph = body.split(/\n\n/).find((p) => p.trim() && !p.trim().startsWith("#"));
|
|
299
|
+
if (headingMatch) {
|
|
300
|
+
if (!frontmatter.name) {
|
|
301
|
+
frontmatter.name = headingMatch[1].trim();
|
|
302
|
+
}
|
|
303
|
+
if (!frontmatter.description) {
|
|
304
|
+
frontmatter.description = firstParagraph?.trim() || frontmatter.name;
|
|
305
|
+
}
|
|
306
|
+
} else {
|
|
307
|
+
console.warn(
|
|
308
|
+
`Skipping plugin agent ${pluginAgent.filePath}: missing required fields (name, description)`
|
|
309
|
+
);
|
|
310
|
+
continue;
|
|
311
|
+
}
|
|
280
312
|
}
|
|
281
313
|
if (frontmatter.model && !frontmatter.model_name && !warnedAgents.has(frontmatter.name) && process.env.MINTO_DEBUG_AGENTS) {
|
|
282
314
|
console.warn(
|
|
@@ -284,7 +316,9 @@ async function loadPluginAgents() {
|
|
|
284
316
|
);
|
|
285
317
|
warnedAgents.add(frontmatter.name);
|
|
286
318
|
}
|
|
287
|
-
const disallowedTools = parseDisallowedTools(
|
|
319
|
+
const disallowedTools = parseDisallowedTools(
|
|
320
|
+
frontmatter.disallowedTools
|
|
321
|
+
);
|
|
288
322
|
const skills = parseSkills(frontmatter.skills);
|
|
289
323
|
const memory = parseMemory(frontmatter.memory);
|
|
290
324
|
const permissionMode = parsePermissionMode(frontmatter.permissionMode);
|
|
@@ -295,9 +329,12 @@ async function loadPluginAgents() {
|
|
|
295
329
|
systemPrompt: body.trim(),
|
|
296
330
|
location: "plugin",
|
|
297
331
|
pluginName: plugin.manifest.name,
|
|
332
|
+
sourcePath: pluginAgent.filePath,
|
|
298
333
|
// Optional fields
|
|
299
334
|
...frontmatter.color && { color: frontmatter.color },
|
|
300
|
-
...frontmatter.model_name && {
|
|
335
|
+
...frontmatter.model_name && {
|
|
336
|
+
model_name: frontmatter.model_name
|
|
337
|
+
},
|
|
301
338
|
...disallowedTools && { disallowedTools },
|
|
302
339
|
...permissionMode && { permissionMode },
|
|
303
340
|
...skills && { skills },
|
|
@@ -305,7 +342,16 @@ async function loadPluginAgents() {
|
|
|
305
342
|
...frontmatter.maxThinkingTokens && {
|
|
306
343
|
maxThinkingTokens: Number(frontmatter.maxThinkingTokens)
|
|
307
344
|
},
|
|
308
|
-
...frontmatter.hooks && {
|
|
345
|
+
...frontmatter.hooks && {
|
|
346
|
+
hooks: frontmatter.hooks
|
|
347
|
+
},
|
|
348
|
+
...frontmatter.isolation && { isolation: frontmatter.isolation },
|
|
349
|
+
...frontmatter.background && {
|
|
350
|
+
background: frontmatter.background === true
|
|
351
|
+
},
|
|
352
|
+
...frontmatter.mcpServers && {
|
|
353
|
+
mcpServers: frontmatter.mcpServers
|
|
354
|
+
}
|
|
309
355
|
};
|
|
310
356
|
agents.push(agent);
|
|
311
357
|
} catch (error) {
|
|
@@ -325,11 +371,21 @@ async function loadAllAgents() {
|
|
|
325
371
|
try {
|
|
326
372
|
const home = homedir();
|
|
327
373
|
const cwd = getCwd();
|
|
374
|
+
const userClaudeDir = join(home, ".claude", "agents");
|
|
328
375
|
const userMintoDir = join(home, ".minto", "agents");
|
|
376
|
+
const projectClaudeDir = join(cwd, ".claude", "agents");
|
|
329
377
|
const projectMintoDir = join(cwd, ".minto", "agents");
|
|
330
|
-
const [
|
|
378
|
+
const [
|
|
379
|
+
pluginAgents,
|
|
380
|
+
userClaudeAgents,
|
|
381
|
+
userMintoAgents,
|
|
382
|
+
projectClaudeAgents,
|
|
383
|
+
projectMintoAgents
|
|
384
|
+
] = await Promise.all([
|
|
331
385
|
loadPluginAgents(),
|
|
386
|
+
scanAgentDirectory(userClaudeDir, "user"),
|
|
332
387
|
scanAgentDirectory(userMintoDir, "user"),
|
|
388
|
+
scanAgentDirectory(projectClaudeDir, "project"),
|
|
333
389
|
scanAgentDirectory(projectMintoDir, "project")
|
|
334
390
|
]);
|
|
335
391
|
const agentMap = /* @__PURE__ */ new Map();
|
|
@@ -337,11 +393,18 @@ async function loadAllAgents() {
|
|
|
337
393
|
agentMap.set(agent.agentType, agent);
|
|
338
394
|
}
|
|
339
395
|
for (const agent of pluginAgents) {
|
|
396
|
+
const namespacedType = agent.pluginName ? `${agent.pluginName}:${agent.agentType}` : agent.agentType;
|
|
397
|
+
agentMap.set(namespacedType, { ...agent, agentType: namespacedType });
|
|
398
|
+
}
|
|
399
|
+
for (const agent of userClaudeAgents) {
|
|
340
400
|
agentMap.set(agent.agentType, agent);
|
|
341
401
|
}
|
|
342
402
|
for (const agent of userMintoAgents) {
|
|
343
403
|
agentMap.set(agent.agentType, agent);
|
|
344
404
|
}
|
|
405
|
+
for (const agent of projectClaudeAgents) {
|
|
406
|
+
agentMap.set(agent.agentType, agent);
|
|
407
|
+
}
|
|
345
408
|
for (const agent of projectMintoAgents) {
|
|
346
409
|
agentMap.set(agent.agentType, agent);
|
|
347
410
|
}
|
|
@@ -349,7 +412,9 @@ async function loadAllAgents() {
|
|
|
349
412
|
const allAgents = [
|
|
350
413
|
...BUILTIN_AGENTS,
|
|
351
414
|
...pluginAgents,
|
|
415
|
+
...userClaudeAgents,
|
|
352
416
|
...userMintoAgents,
|
|
417
|
+
...projectClaudeAgents,
|
|
353
418
|
...projectMintoAgents
|
|
354
419
|
];
|
|
355
420
|
return { activeAgents, allAgents };
|
|
@@ -378,7 +443,16 @@ function clearAgentCache() {
|
|
|
378
443
|
const getAgentByType = memoize(
|
|
379
444
|
async (agentType) => {
|
|
380
445
|
const agents = await getActiveAgents();
|
|
381
|
-
|
|
446
|
+
const direct = agents.find((agent) => agent.agentType === agentType);
|
|
447
|
+
if (direct) return direct;
|
|
448
|
+
if (!agentType.includes(":")) {
|
|
449
|
+
for (const agent of agents) {
|
|
450
|
+
if (agent.agentType.includes(":") && agent.agentType.split(":").pop() === agentType) {
|
|
451
|
+
return agent;
|
|
452
|
+
}
|
|
453
|
+
}
|
|
454
|
+
}
|
|
455
|
+
return void 0;
|
|
382
456
|
}
|
|
383
457
|
);
|
|
384
458
|
const getAvailableAgentTypes = memoize(async () => {
|
|
@@ -391,7 +465,9 @@ async function startAgentWatcher(onChange) {
|
|
|
391
465
|
const home = homedir();
|
|
392
466
|
const cwd = getCwd();
|
|
393
467
|
const directories = [
|
|
468
|
+
{ path: join(home, ".claude", "agents"), label: "user/.claude" },
|
|
394
469
|
{ path: join(home, ".minto", "agents"), label: "user/.minto" },
|
|
470
|
+
{ path: join(cwd, ".claude", "agents"), label: "project/.claude" },
|
|
395
471
|
{ path: join(cwd, ".minto", "agents"), label: "project/.minto" }
|
|
396
472
|
];
|
|
397
473
|
const watchDirectory = (dirPath, label) => {
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../src/utils/agentLoader.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Agent configuration loader\n * Loads agent configurations from markdown files with YAML frontmatter.\n * Implements Claude Code Subagent specification for agent configuration.\n */\n\nimport {\n existsSync,\n readFileSync,\n readdirSync,\n statSync,\n watch,\n FSWatcher,\n} from 'fs'\nimport { join, resolve } from 'path'\nimport { homedir } from 'os'\nimport matter from 'gray-matter'\nimport { getCwd } from './state'\nimport { memoize } from 'lodash-es'\nimport { loadAllPlugins } from './pluginLoader'\nimport type { PermissionMode } from '@minto-types/PermissionMode'\nimport type { HookDefinition } from '@minto-types/hooks'\n\n// Track warned agents to avoid spam\nconst warnedAgents = new Set<string>()\n\n/**\n * Agent lifecycle hooks configuration\n * Claude Code specification compliant\n */\nexport interface AgentHooks {\n onStart?: HookDefinition[]\n onComplete?: HookDefinition[]\n onError?: HookDefinition[]\n}\n\n/**\n * Agent memory persistence configuration\n * Claude Code specification compliant\n */\nexport interface AgentMemory {\n scope: 'user' | 'project' | 'local'\n key?: string // Optional namespace key for memory storage\n}\n\n/**\n * Agent configuration interface\n * Extended for Claude Code specification compliance\n */\nexport interface AgentConfig {\n // Identity\n agentType: string // Agent identifier (matches subagent_type)\n whenToUse: string // Description of when to use this agent\n systemPrompt: string // System prompt content\n location: 'built-in' | 'user' | 'project' | 'plugin'\n\n // Tool access control\n tools: string[] | '*' // Tool allowlist (or '*' for all)\n disallowedTools?: string[] // Tool denylist (takes precedence over tools)\n\n // Execution configuration\n model_name?: string // Model override (e.g., 'quick', 'main', 'reasoning')\n permissionMode?: PermissionMode // Permission handling mode\n maxThinkingTokens?: number // Extended thinking token limit\n\n // Skill preloading\n skills?: string[] // Skills to preload into agent context\n\n // Lifecycle hooks\n hooks?: AgentHooks // Agent lifecycle hooks\n\n // Memory persistence\n memory?: AgentMemory // Memory configuration for cross-session persistence\n\n // UI/Display\n color?: string // Optional UI color\n pluginName?: string // Source plugin name (if loaded from plugin)\n}\n\n/**\n * Built-in agents for Claude Code specification compliance\n */\nconst BUILTIN_AGENTS: AgentConfig[] = [\n // general-purpose: Default agent for multi-step tasks\n {\n agentType: 'general-purpose',\n whenToUse:\n 'General-purpose agent for researching complex questions, searching for code, and executing multi-step tasks. When you are searching for a keyword or file and are not confident that you will find the right match in the first few tries use this agent.',\n tools: '*',\n systemPrompt: `You are a general-purpose agent. Given the user's task, use the tools available to complete it efficiently and thoroughly.\n\nWhen to use your capabilities:\n- Searching for code, configurations, and patterns across large codebases\n- Analyzing multiple files to understand system architecture\n- Investigating complex questions that require exploring many files\n- Performing multi-step research tasks\n\nGuidelines:\n- For file searches: Use Grep or Glob when you need to search broadly. Use Read when you know the specific file path.\n- For analysis: Start broad and narrow down. Use multiple search strategies if the first doesn't yield results.\n- Be thorough: Check multiple locations, consider different naming conventions, look for related files.\n- Complete tasks directly using your capabilities.`,\n location: 'built-in',\n },\n\n // Explore: Fast read-only codebase exploration (Claude Code spec)\n {\n agentType: 'Explore',\n whenToUse:\n 'Fast agent specialized for exploring codebases. Use this when you need to quickly find files by patterns, search code for keywords, or answer questions about the codebase. Specify thoroughness: \"quick\" for basic searches, \"medium\" for moderate exploration, \"very thorough\" for comprehensive analysis.',\n tools: ['Read', 'Glob', 'Grep', 'LS', 'WebSearch', 'WebFetch'],\n disallowedTools: ['Edit', 'Write', 'Bash', 'NotebookEdit'],\n model_name: 'quick', // Maps to haiku equivalent\n permissionMode: 'plan', // Read-only mode\n systemPrompt: `You are an exploration agent optimized for fast codebase searches and understanding.\n\nYour role:\n- Quickly find files, functions, classes, and patterns\n- Understand code structure and architecture\n- Answer questions about how code works\n- Identify relevant files for a given task\n\nGuidelines:\n- Use Glob for finding files by pattern (e.g., \"**/*.tsx\", \"src/**/*.ts\")\n- Use Grep for searching code content (e.g., function names, imports, patterns)\n- Use Read to examine specific files you've identified\n- Start broad, then narrow down based on results\n- Report findings with specific file paths and line numbers\n- Be thorough but efficient - check multiple locations and naming conventions\n\nYou are READ-ONLY. You cannot modify any files.`,\n location: 'built-in',\n },\n\n // Plan: Research and planning agent (Claude Code spec)\n {\n agentType: 'Plan',\n whenToUse:\n 'Software architect agent for designing implementation plans. Use this when you need to plan the implementation strategy for a task. Returns step-by-step plans, identifies critical files, and considers architectural trade-offs.',\n tools: ['Read', 'Glob', 'Grep', 'LS', 'WebSearch', 'WebFetch', 'Think'],\n disallowedTools: ['Edit', 'Write', 'Bash', 'NotebookEdit'],\n model_name: 'inherit', // Inherit caller's model\n permissionMode: 'plan', // Read-only mode\n systemPrompt: `You are a planning agent specialized in analyzing codebases and creating implementation plans.\n\nYour role:\n- Understand requirements and constraints\n- Analyze existing code patterns and architecture\n- Create detailed, step-by-step implementation plans\n- Identify potential issues and trade-offs\n- Recommend best approaches based on codebase conventions\n\nGuidelines:\n- Read relevant files to understand current patterns\n- Identify all files that will need modification\n- Consider edge cases and error handling\n- Note any dependencies or prerequisites\n- Provide specific, actionable steps\n\nOutput format:\n1. Summary of the task\n2. Files to modify/create (with paths)\n3. Step-by-step implementation plan\n4. Potential challenges or considerations\n5. Testing approach\n\nYou are READ-ONLY. You cannot modify any files. Focus on planning, not execution.`,\n location: 'built-in',\n },\n\n // Bash: Shell command execution agent (Claude Code spec)\n {\n agentType: 'Bash',\n whenToUse:\n 'Command execution specialist for running bash commands. Use this for git operations, command execution, and other terminal tasks.',\n tools: ['Bash', 'Read', 'LS'],\n systemPrompt: `You are a bash execution agent specialized in running shell commands.\n\nYour role:\n- Execute shell commands safely and effectively\n- Handle build, test, and deployment tasks\n- Manage git operations\n- Run scripts and automation\n\nGuidelines:\n- Check command syntax before execution\n- Handle errors gracefully with clear messages\n- Report command output and exit codes\n- Be cautious with destructive commands (rm, git reset, etc.)\n- Use appropriate flags for verbose output when debugging\n- Chain commands with && for dependent operations\n\nAlways explain what a command will do before running it if it modifies the system.`,\n location: 'built-in',\n },\n]\n\n/**\n * Parse tools field from frontmatter\n */\nfunction parseTools(tools: any): string[] | '*' {\n if (!tools) return '*'\n if (tools === '*') return '*'\n if (Array.isArray(tools)) {\n // Ensure all items are strings and filter out non-strings\n const filteredTools = tools.filter(\n (t): t is string => typeof t === 'string',\n )\n return filteredTools.length > 0 ? filteredTools : '*'\n }\n if (typeof tools === 'string') {\n // Handle comma-separated string: \"Read, Grep, Glob\"\n if (tools.includes(',')) {\n return tools.split(',').map((t: string) => t.trim()).filter(Boolean)\n }\n return [tools]\n }\n return '*'\n}\n\n/**\n * Parse disallowedTools field from frontmatter\n */\nfunction parseDisallowedTools(disallowedTools: any): string[] | undefined {\n if (!disallowedTools) return undefined\n if (Array.isArray(disallowedTools)) {\n const filtered = disallowedTools.filter(\n (t): t is string => typeof t === 'string',\n )\n return filtered.length > 0 ? filtered : undefined\n }\n if (typeof disallowedTools === 'string') {\n // Handle comma-separated string\n if (disallowedTools.includes(',')) {\n return disallowedTools.split(',').map((t: string) => t.trim()).filter(Boolean)\n }\n return [disallowedTools]\n }\n return undefined\n}\n\n/**\n * Parse skills field from frontmatter\n */\nfunction parseSkills(skills: any): string[] | undefined {\n if (!skills) return undefined\n if (Array.isArray(skills)) {\n const filtered = skills.filter((s): s is string => typeof s === 'string')\n return filtered.length > 0 ? filtered : undefined\n }\n if (typeof skills === 'string') {\n if (skills.includes(',')) {\n return skills.split(',').map((s: string) => s.trim()).filter(Boolean)\n }\n return [skills]\n }\n return undefined\n}\n\n/**\n * Parse memory field from frontmatter\n */\nfunction parseMemory(memory: any): AgentMemory | undefined {\n if (!memory) return undefined\n if (typeof memory === 'string') {\n // Simple format: \"user\" | \"project\" | \"local\"\n if (['user', 'project', 'local'].includes(memory)) {\n return { scope: memory as AgentMemory['scope'] }\n }\n return undefined\n }\n if (typeof memory === 'object') {\n const scope = memory.scope\n if (['user', 'project', 'local'].includes(scope)) {\n return {\n scope: scope as AgentMemory['scope'],\n ...(memory.key && { key: String(memory.key) }),\n }\n }\n }\n return undefined\n}\n\n/**\n * Parse permissionMode field from frontmatter\n */\nfunction parsePermissionMode(mode: any): PermissionMode | undefined {\n if (!mode) return undefined\n const validModes: PermissionMode[] = [\n 'default',\n 'acceptEdits',\n 'dontAsk',\n 'plan',\n 'bypassPermissions',\n ]\n if (typeof mode === 'string' && validModes.includes(mode as PermissionMode)) {\n return mode as PermissionMode\n }\n return undefined\n}\n\n/**\n * Scan a directory for agent configuration files\n */\nasync function scanAgentDirectory(\n dirPath: string,\n location: 'user' | 'project',\n): Promise<AgentConfig[]> {\n if (!existsSync(dirPath)) {\n return []\n }\n\n const agents: AgentConfig[] = []\n\n try {\n const files = readdirSync(dirPath)\n\n for (const file of files) {\n if (!file.endsWith('.md')) continue\n\n const filePath = join(dirPath, file)\n const stat = statSync(filePath)\n\n if (!stat.isFile()) continue\n\n try {\n const content = readFileSync(filePath, 'utf-8')\n const { data: frontmatter, content: body } = matter(content)\n\n // Validate required fields\n if (!frontmatter.name || !frontmatter.description) {\n console.warn(\n `Skipping ${filePath}: missing required fields (name, description)`,\n )\n continue\n }\n\n // Silently ignore deprecated 'model' field - no warnings by default\n // Only warn if MINTO_DEBUG_AGENTS environment variable is set\n if (\n frontmatter.model &&\n !frontmatter.model_name &&\n !warnedAgents.has(frontmatter.name) &&\n process.env.MINTO_DEBUG_AGENTS\n ) {\n console.warn(\n `\u26A0\uFE0F Agent ${frontmatter.name}: 'model' field is deprecated and ignored. Use 'model_name' instead, or omit to use default 'task' model.`,\n )\n warnedAgents.add(frontmatter.name)\n }\n\n // Parse optional fields\n const disallowedTools = parseDisallowedTools(frontmatter.disallowedTools)\n const skills = parseSkills(frontmatter.skills)\n const memory = parseMemory(frontmatter.memory)\n const permissionMode = parsePermissionMode(frontmatter.permissionMode)\n\n // Build agent config with all Claude Code spec fields\n const agent: AgentConfig = {\n agentType: frontmatter.name,\n whenToUse: frontmatter.description.replace(/\\\\n/g, '\\n'),\n tools: parseTools(frontmatter.tools),\n systemPrompt: body.trim(),\n location,\n // Optional fields - only include if defined\n ...(frontmatter.color && { color: frontmatter.color }),\n ...(frontmatter.model_name && { model_name: frontmatter.model_name }),\n ...(disallowedTools && { disallowedTools }),\n ...(permissionMode && { permissionMode }),\n ...(skills && { skills }),\n ...(memory && { memory }),\n ...(frontmatter.maxThinkingTokens && {\n maxThinkingTokens: Number(frontmatter.maxThinkingTokens),\n }),\n ...(frontmatter.hooks && { hooks: frontmatter.hooks as AgentHooks }),\n }\n\n agents.push(agent)\n } catch (error) {\n console.warn(`Failed to parse agent file ${filePath}:`, error)\n }\n }\n } catch (error) {\n console.warn(`Failed to scan directory ${dirPath}:`, error)\n }\n\n return agents\n}\n\n/**\n * Load agents from installed plugins\n */\nasync function loadPluginAgents(): Promise<AgentConfig[]> {\n const agents: AgentConfig[] = []\n\n try {\n const plugins = loadAllPlugins()\n\n for (const plugin of plugins) {\n // Skip disabled plugins\n if (!plugin.enabled) continue\n\n for (const pluginAgent of plugin.agents) {\n try {\n // Read the agent file to parse frontmatter and content\n const content = readFileSync(pluginAgent.filePath, 'utf-8')\n const { data: frontmatter, content: body } = matter(content)\n\n // Validate required fields\n if (!frontmatter.name || !frontmatter.description) {\n console.warn(\n `Skipping plugin agent ${pluginAgent.filePath}: missing required fields (name, description)`,\n )\n continue\n }\n\n // Warn about deprecated 'model' field if debug mode is enabled\n if (\n frontmatter.model &&\n !frontmatter.model_name &&\n !warnedAgents.has(frontmatter.name) &&\n process.env.MINTO_DEBUG_AGENTS\n ) {\n console.warn(\n `\u26A0\uFE0F Plugin agent ${frontmatter.name} (from ${plugin.manifest.name}): 'model' field is deprecated and ignored. Use 'model_name' instead, or omit to use default 'task' model.`,\n )\n warnedAgents.add(frontmatter.name)\n }\n\n // Parse optional fields\n const disallowedTools = parseDisallowedTools(frontmatter.disallowedTools)\n const skills = parseSkills(frontmatter.skills)\n const memory = parseMemory(frontmatter.memory)\n const permissionMode = parsePermissionMode(frontmatter.permissionMode)\n\n // Build agent config with all Claude Code spec fields\n const agent: AgentConfig = {\n agentType: frontmatter.name,\n whenToUse: frontmatter.description.replace(/\\\\n/g, '\\n'),\n tools: parseTools(frontmatter.tools),\n systemPrompt: body.trim(),\n location: 'plugin',\n pluginName: plugin.manifest.name,\n // Optional fields\n ...(frontmatter.color && { color: frontmatter.color }),\n ...(frontmatter.model_name && { model_name: frontmatter.model_name }),\n ...(disallowedTools && { disallowedTools }),\n ...(permissionMode && { permissionMode }),\n ...(skills && { skills }),\n ...(memory && { memory }),\n ...(frontmatter.maxThinkingTokens && {\n maxThinkingTokens: Number(frontmatter.maxThinkingTokens),\n }),\n ...(frontmatter.hooks && { hooks: frontmatter.hooks as AgentHooks }),\n }\n\n agents.push(agent)\n } catch (error) {\n console.warn(\n `Failed to load plugin agent ${pluginAgent.name} from ${plugin.manifest.name}:`,\n error,\n )\n }\n }\n }\n } catch (error) {\n console.warn('Failed to load plugin agents:', error)\n }\n\n return agents\n}\n\n/**\n * Load all agent configurations\n *\n * Directory Priority (later overrides earlier):\n * 1. Built-in agents (general-purpose, Explore, Plan, Bash)\n * 2. Plugin agents\n * 3. ~/.minto/agents (Minto user directory)\n * 4. ./.minto/agents (Minto project directory - highest priority)\n */\nasync function loadAllAgents(): Promise<{\n activeAgents: AgentConfig[]\n allAgents: AgentConfig[]\n}> {\n try {\n const home = homedir()\n const cwd = getCwd()\n\n // Define directories in priority order (later overrides earlier)\n const userMintoDir = join(home, '.minto', 'agents')\n const projectMintoDir = join(cwd, '.minto', 'agents')\n\n // Load from all sources in parallel\n const [pluginAgents, userMintoAgents, projectMintoAgents] =\n await Promise.all([\n loadPluginAgents(),\n scanAgentDirectory(userMintoDir, 'user'),\n scanAgentDirectory(projectMintoDir, 'project'),\n ])\n\n // Apply priority override: built-in < plugin < user-minto < project-minto\n // Later entries override earlier ones with the same agentType\n const agentMap = new Map<string, AgentConfig>()\n\n // Add in priority order (later entries override earlier ones)\n for (const agent of BUILTIN_AGENTS) {\n agentMap.set(agent.agentType, agent)\n }\n for (const agent of pluginAgents) {\n agentMap.set(agent.agentType, agent)\n }\n for (const agent of userMintoAgents) {\n agentMap.set(agent.agentType, agent)\n }\n for (const agent of projectMintoAgents) {\n agentMap.set(agent.agentType, agent)\n }\n\n const activeAgents = Array.from(agentMap.values())\n const allAgents = [\n ...BUILTIN_AGENTS,\n ...pluginAgents,\n ...userMintoAgents,\n ...projectMintoAgents,\n ]\n\n return { activeAgents, allAgents }\n } catch (error) {\n console.error('Failed to load agents, falling back to built-in:', error)\n return {\n activeAgents: [...BUILTIN_AGENTS],\n allAgents: [...BUILTIN_AGENTS],\n }\n }\n}\n\n// Memoized version for performance\nexport const getActiveAgents = memoize(async (): Promise<AgentConfig[]> => {\n const { activeAgents } = await loadAllAgents()\n return activeAgents\n})\n\n// Get all agents (both active and overridden)\nexport const getAllAgents = memoize(async (): Promise<AgentConfig[]> => {\n const { allAgents } = await loadAllAgents()\n return allAgents\n})\n\n// Clear cache when needed\nexport function clearAgentCache() {\n getActiveAgents.cache?.clear?.()\n getAllAgents.cache?.clear?.()\n getAgentByType.cache?.clear?.()\n getAvailableAgentTypes.cache?.clear?.()\n}\n\n// Get a specific agent by type\nexport const getAgentByType = memoize(\n async (agentType: string): Promise<AgentConfig | undefined> => {\n const agents = await getActiveAgents()\n return agents.find(agent => agent.agentType === agentType)\n },\n)\n\n// Get all available agent types for validation\nexport const getAvailableAgentTypes = memoize(async (): Promise<string[]> => {\n const agents = await getActiveAgents()\n return agents.map(agent => agent.agentType)\n})\n\n// File watcher for hot reload\nlet watchers: FSWatcher[] = []\n\n/**\n * Start watching agent configuration directories for changes\n */\nexport async function startAgentWatcher(onChange?: () => void): Promise<void> {\n await stopAgentWatcher() // Clean up any existing watchers\n\n const home = homedir()\n const cwd = getCwd()\n\n // Watch .minto directories\n const directories = [\n { path: join(home, '.minto', 'agents'), label: 'user/.minto' },\n { path: join(cwd, '.minto', 'agents'), label: 'project/.minto' },\n ]\n\n const watchDirectory = (dirPath: string, label: string) => {\n if (existsSync(dirPath)) {\n const watcher = watch(\n dirPath,\n { recursive: false },\n async (eventType, filename) => {\n if (filename && filename.endsWith('.md')) {\n console.log(\n `\uD83D\uDD04 Agent configuration changed in ${label}: ${filename}`,\n )\n clearAgentCache()\n // Also clear any other related caches\n getAllAgents.cache?.clear?.()\n onChange?.()\n }\n },\n )\n watchers.push(watcher)\n }\n }\n\n // Watch all directories\n for (const { path, label } of directories) {\n watchDirectory(path, label)\n }\n}\n\n/**\n * Stop watching agent configuration directories\n */\nexport async function stopAgentWatcher(): Promise<void> {\n // FSWatcher.close() is synchronous and does not accept a callback on Node 18/20\n try {\n for (const watcher of watchers) {\n try {\n watcher.close()\n } catch (err) {\n console.error('Failed to close file watcher:', err)\n }\n }\n } finally {\n watchers = []\n }\n}\n"],
|
|
5
|
-
"mappings": "AAMA;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OAEK;AACP,SAAS,YAAqB;AAC9B,SAAS,eAAe;AACxB,OAAO,YAAY;AACnB,SAAS,cAAc;AACvB,SAAS,eAAe;AACxB,SAAS,sBAAsB;AAK/B,MAAM,eAAe,oBAAI,IAAY;
|
|
4
|
+
"sourcesContent": ["/**\n * Agent configuration loader\n * Loads agent configurations from markdown files with YAML frontmatter.\n * Implements Claude Code Subagent specification for agent configuration.\n */\n\nimport {\n existsSync,\n readFileSync,\n readdirSync,\n statSync,\n watch,\n FSWatcher,\n} from 'fs'\nimport { join, resolve } from 'path'\nimport { homedir } from 'os'\nimport matter from 'gray-matter'\nimport { getCwd } from './state'\nimport { memoize } from 'lodash-es'\nimport { loadAllPlugins } from './pluginLoader'\nimport type { PermissionMode } from '@minto-types/PermissionMode'\nimport type { HookDefinition } from '@minto-types/hooks'\n\n// Track warned agents to avoid spam\nconst warnedAgents = new Set<string>()\n\n/**\n * Agent lifecycle hooks configuration\n * Claude Code specification compliant\n */\nexport interface AgentHooks {\n onStart?: HookDefinition[]\n onComplete?: HookDefinition[]\n onError?: HookDefinition[]\n}\n\n/**\n * Agent memory persistence configuration\n * Claude Code specification compliant\n */\nexport interface AgentMemory {\n scope: 'user' | 'project' | 'local'\n key?: string // Optional namespace key for memory storage\n}\n\n/**\n * Agent configuration interface\n * Extended for Claude Code specification compliance\n */\nexport interface AgentConfig {\n // Identity\n agentType: string // Agent identifier (matches subagent_type)\n whenToUse: string // Description of when to use this agent\n systemPrompt: string // System prompt content\n location: 'built-in' | 'user' | 'project' | 'plugin'\n\n // Tool access control\n tools: string[] | '*' // Tool allowlist (or '*' for all)\n disallowedTools?: string[] // Tool denylist (takes precedence over tools)\n\n // Execution configuration\n model_name?: string // Model override (e.g., 'quick', 'main', 'reasoning')\n permissionMode?: PermissionMode // Permission handling mode\n maxThinkingTokens?: number // Extended thinking token limit\n\n // Skill preloading\n skills?: string[] // Skills to preload into agent context\n\n // Lifecycle hooks\n hooks?: AgentHooks // Agent lifecycle hooks\n\n // Memory persistence\n memory?: AgentMemory // Memory configuration for cross-session persistence\n\n // Isolation and execution modes (Claude Code spec)\n isolation?: 'worktree' // Git worktree isolation for parallel file edits\n background?: boolean // Run agent in background (fire-and-forget)\n mcpServers?: Record<\n string,\n { command: string; args?: string[]; env?: Record<string, string> }\n > // Per-agent MCP servers\n\n // UI/Display\n color?: string // Optional UI color\n pluginName?: string // Source plugin name (if loaded from plugin)\n\n // File origin\n sourcePath?: string // Absolute path to the .md file (user/project/plugin agents)\n}\n\n/**\n * Built-in agents for Claude Code specification compliance\n */\nconst BUILTIN_AGENTS: AgentConfig[] = [\n // general-purpose: Default agent for multi-step tasks\n {\n agentType: 'general-purpose',\n whenToUse:\n 'General-purpose agent for researching complex questions, searching for code, and executing multi-step tasks. When you are searching for a keyword or file and are not confident that you will find the right match in the first few tries use this agent.',\n tools: '*',\n systemPrompt: `You are a general-purpose agent. Given the user's task, use the tools available to complete it efficiently and thoroughly.\n\nWhen to use your capabilities:\n- Searching for code, configurations, and patterns across large codebases\n- Analyzing multiple files to understand system architecture\n- Investigating complex questions that require exploring many files\n- Performing multi-step research tasks\n\nGuidelines:\n- For file searches: Use Grep or Glob when you need to search broadly. Use Read when you know the specific file path.\n- For analysis: Start broad and narrow down. Use multiple search strategies if the first doesn't yield results.\n- Be thorough: Check multiple locations, consider different naming conventions, look for related files.\n- Complete tasks directly using your capabilities.`,\n location: 'built-in',\n },\n\n // Explore: Fast read-only codebase exploration (Claude Code spec)\n {\n agentType: 'Explore',\n whenToUse:\n 'Fast agent specialized for exploring codebases. Use this when you need to quickly find files by patterns, search code for keywords, or answer questions about the codebase. Specify thoroughness: \"quick\" for basic searches, \"medium\" for moderate exploration, \"very thorough\" for comprehensive analysis.',\n tools: ['Read', 'Glob', 'Grep', 'LS', 'WebSearch', 'WebFetch'],\n disallowedTools: ['Edit', 'Write', 'Bash', 'NotebookEdit'],\n model_name: 'quick', // Maps to haiku equivalent\n permissionMode: 'plan', // Read-only mode\n systemPrompt: `You are an exploration agent optimized for fast codebase searches and understanding.\n\nYour role:\n- Quickly find files, functions, classes, and patterns\n- Understand code structure and architecture\n- Answer questions about how code works\n- Identify relevant files for a given task\n\nGuidelines:\n- Use Glob for finding files by pattern (e.g., \"**/*.tsx\", \"src/**/*.ts\")\n- Use Grep for searching code content (e.g., function names, imports, patterns)\n- Use Read to examine specific files you've identified\n- Start broad, then narrow down based on results\n- Report findings with specific file paths and line numbers\n- Be thorough but efficient - check multiple locations and naming conventions\n\nYou are READ-ONLY. You cannot modify any files.`,\n location: 'built-in',\n },\n\n // Plan: Research and planning agent (Claude Code spec)\n {\n agentType: 'Plan',\n whenToUse:\n 'Software architect agent for designing implementation plans. Use this when you need to plan the implementation strategy for a task. Returns step-by-step plans, identifies critical files, and considers architectural trade-offs.',\n tools: ['Read', 'Glob', 'Grep', 'LS', 'WebSearch', 'WebFetch', 'Think'],\n disallowedTools: ['Edit', 'Write', 'Bash', 'NotebookEdit'],\n model_name: 'inherit', // Inherit caller's model\n permissionMode: 'plan', // Read-only mode\n systemPrompt: `You are a planning agent specialized in analyzing codebases and creating implementation plans.\n\nYour role:\n- Understand requirements and constraints\n- Analyze existing code patterns and architecture\n- Create detailed, step-by-step implementation plans\n- Identify potential issues and trade-offs\n- Recommend best approaches based on codebase conventions\n\nGuidelines:\n- Read relevant files to understand current patterns\n- Identify all files that will need modification\n- Consider edge cases and error handling\n- Note any dependencies or prerequisites\n- Provide specific, actionable steps\n\nOutput format:\n1. Summary of the task\n2. Files to modify/create (with paths)\n3. Step-by-step implementation plan\n4. Potential challenges or considerations\n5. Testing approach\n\nYou are READ-ONLY. You cannot modify any files. Focus on planning, not execution.`,\n location: 'built-in',\n },\n\n // Bash: Shell command execution agent (Claude Code spec)\n {\n agentType: 'Bash',\n whenToUse:\n 'Command execution specialist for running bash commands. Use this for git operations, command execution, and other terminal tasks.',\n tools: ['Bash', 'Read', 'LS'],\n systemPrompt: `You are a bash execution agent specialized in running shell commands.\n\nYour role:\n- Execute shell commands safely and effectively\n- Handle build, test, and deployment tasks\n- Manage git operations\n- Run scripts and automation\n\nGuidelines:\n- Check command syntax before execution\n- Handle errors gracefully with clear messages\n- Report command output and exit codes\n- Be cautious with destructive commands (rm, git reset, etc.)\n- Use appropriate flags for verbose output when debugging\n- Chain commands with && for dependent operations\n\nAlways explain what a command will do before running it if it modifies the system.`,\n location: 'built-in',\n },\n]\n\n/**\n * Parse tools field from frontmatter\n */\nfunction parseTools(tools: any): string[] | '*' {\n if (!tools) return '*'\n if (tools === '*') return '*'\n if (Array.isArray(tools)) {\n // Ensure all items are strings and filter out non-strings\n const filteredTools = tools.filter(\n (t): t is string => typeof t === 'string',\n )\n return filteredTools.length > 0 ? filteredTools : '*'\n }\n if (typeof tools === 'string') {\n // Handle comma-separated string: \"Read, Grep, Glob\"\n if (tools.includes(',')) {\n return tools\n .split(',')\n .map((t: string) => t.trim())\n .filter(Boolean)\n }\n return [tools]\n }\n return '*'\n}\n\n/**\n * Parse disallowedTools field from frontmatter\n */\nfunction parseDisallowedTools(disallowedTools: any): string[] | undefined {\n if (!disallowedTools) return undefined\n if (Array.isArray(disallowedTools)) {\n const filtered = disallowedTools.filter(\n (t): t is string => typeof t === 'string',\n )\n return filtered.length > 0 ? filtered : undefined\n }\n if (typeof disallowedTools === 'string') {\n // Handle comma-separated string\n if (disallowedTools.includes(',')) {\n return disallowedTools\n .split(',')\n .map((t: string) => t.trim())\n .filter(Boolean)\n }\n return [disallowedTools]\n }\n return undefined\n}\n\n/**\n * Parse skills field from frontmatter\n */\nfunction parseSkills(skills: any): string[] | undefined {\n if (!skills) return undefined\n if (Array.isArray(skills)) {\n const filtered = skills.filter((s): s is string => typeof s === 'string')\n return filtered.length > 0 ? filtered : undefined\n }\n if (typeof skills === 'string') {\n if (skills.includes(',')) {\n return skills\n .split(',')\n .map((s: string) => s.trim())\n .filter(Boolean)\n }\n return [skills]\n }\n return undefined\n}\n\n/**\n * Parse memory field from frontmatter\n */\nfunction parseMemory(memory: any): AgentMemory | undefined {\n if (!memory) return undefined\n if (typeof memory === 'string') {\n // Simple format: \"user\" | \"project\" | \"local\"\n if (['user', 'project', 'local'].includes(memory)) {\n return { scope: memory as AgentMemory['scope'] }\n }\n return undefined\n }\n if (typeof memory === 'object') {\n const scope = memory.scope\n if (['user', 'project', 'local'].includes(scope)) {\n return {\n scope: scope as AgentMemory['scope'],\n ...(memory.key && { key: String(memory.key) }),\n }\n }\n }\n return undefined\n}\n\n/**\n * Parse permissionMode field from frontmatter\n */\nfunction parsePermissionMode(mode: any): PermissionMode | undefined {\n if (!mode) return undefined\n const validModes: PermissionMode[] = [\n 'default',\n 'acceptEdits',\n 'dontAsk',\n 'plan',\n 'bypassPermissions',\n ]\n if (typeof mode === 'string' && validModes.includes(mode as PermissionMode)) {\n return mode as PermissionMode\n }\n return undefined\n}\n\n/**\n * Scan a directory for agent configuration files\n */\nasync function scanAgentDirectory(\n dirPath: string,\n location: 'user' | 'project',\n): Promise<AgentConfig[]> {\n if (!existsSync(dirPath)) {\n return []\n }\n\n const agents: AgentConfig[] = []\n\n try {\n const files = readdirSync(dirPath)\n\n for (const file of files) {\n if (!file.endsWith('.md')) continue\n\n const filePath = join(dirPath, file)\n const stat = statSync(filePath)\n\n if (!stat.isFile()) continue\n\n try {\n const content = readFileSync(filePath, 'utf-8')\n if (!content.trim()) continue // Skip empty files silently\n const { data: frontmatter, content: body } = matter(content)\n\n // Validate required fields \u2014 fallback to markdown heading/first paragraph\n // for CC-style agent files without YAML frontmatter\n if (!frontmatter.name || !frontmatter.description) {\n const headingMatch = body.match(/^#\\s+(.+)/m)\n const firstParagraph = body\n .split(/\\n\\n/)\n .find(p => p.trim() && !p.trim().startsWith('#'))\n if (headingMatch) {\n if (!frontmatter.name) {\n frontmatter.name = headingMatch[1].trim()\n }\n if (!frontmatter.description) {\n frontmatter.description =\n firstParagraph?.trim() || frontmatter.name\n }\n } else {\n console.warn(\n `Skipping ${filePath}: missing required fields (name, description)`,\n )\n continue\n }\n }\n\n // Silently ignore deprecated 'model' field - no warnings by default\n // Only warn if MINTO_DEBUG_AGENTS environment variable is set\n if (\n frontmatter.model &&\n !frontmatter.model_name &&\n !warnedAgents.has(frontmatter.name) &&\n process.env.MINTO_DEBUG_AGENTS\n ) {\n console.warn(\n `\u26A0\uFE0F Agent ${frontmatter.name}: 'model' field is deprecated and ignored. Use 'model_name' instead, or omit to use default 'task' model.`,\n )\n warnedAgents.add(frontmatter.name)\n }\n\n // Parse optional fields\n const disallowedTools = parseDisallowedTools(\n frontmatter.disallowedTools,\n )\n const skills = parseSkills(frontmatter.skills)\n const memory = parseMemory(frontmatter.memory)\n const permissionMode = parsePermissionMode(frontmatter.permissionMode)\n\n // Build agent config with all Claude Code spec fields\n const agent: AgentConfig = {\n agentType: frontmatter.name,\n whenToUse: frontmatter.description.replace(/\\\\n/g, '\\n'),\n tools: parseTools(frontmatter.tools),\n systemPrompt: body.trim(),\n location,\n sourcePath: filePath,\n // Optional fields - only include if defined\n ...(frontmatter.color && { color: frontmatter.color }),\n ...(frontmatter.model_name && { model_name: frontmatter.model_name }),\n ...(disallowedTools && { disallowedTools }),\n ...(permissionMode && { permissionMode }),\n ...(skills && { skills }),\n ...(memory && { memory }),\n ...(frontmatter.maxThinkingTokens && {\n maxThinkingTokens: Number(frontmatter.maxThinkingTokens),\n }),\n ...(frontmatter.hooks && { hooks: frontmatter.hooks as AgentHooks }),\n ...(frontmatter.isolation && { isolation: frontmatter.isolation }),\n ...(frontmatter.background && {\n background: frontmatter.background === true,\n }),\n ...(frontmatter.mcpServers && { mcpServers: frontmatter.mcpServers }),\n }\n\n agents.push(agent)\n } catch (error) {\n console.warn(`Failed to parse agent file ${filePath}:`, error)\n }\n }\n } catch (error) {\n console.warn(`Failed to scan directory ${dirPath}:`, error)\n }\n\n return agents\n}\n\n/**\n * Load agents from installed plugins\n */\nasync function loadPluginAgents(): Promise<AgentConfig[]> {\n const agents: AgentConfig[] = []\n\n try {\n const plugins = loadAllPlugins()\n\n for (const plugin of plugins) {\n // Skip disabled plugins\n if (!plugin.enabled) continue\n\n for (const pluginAgent of plugin.agents) {\n try {\n // Read the agent file to parse frontmatter and content\n const content = readFileSync(pluginAgent.filePath, 'utf-8')\n if (!content.trim()) continue // Skip empty files silently\n const { data: frontmatter, content: body } = matter(content)\n\n // Validate required fields \u2014 fallback to markdown heading/first paragraph\n if (!frontmatter.name || !frontmatter.description) {\n const headingMatch = body.match(/^#\\s+(.+)/m)\n const firstParagraph = body\n .split(/\\n\\n/)\n .find(p => p.trim() && !p.trim().startsWith('#'))\n if (headingMatch) {\n if (!frontmatter.name) {\n frontmatter.name = headingMatch[1].trim()\n }\n if (!frontmatter.description) {\n frontmatter.description =\n firstParagraph?.trim() || frontmatter.name\n }\n } else {\n console.warn(\n `Skipping plugin agent ${pluginAgent.filePath}: missing required fields (name, description)`,\n )\n continue\n }\n }\n\n // Warn about deprecated 'model' field if debug mode is enabled\n if (\n frontmatter.model &&\n !frontmatter.model_name &&\n !warnedAgents.has(frontmatter.name) &&\n process.env.MINTO_DEBUG_AGENTS\n ) {\n console.warn(\n `\u26A0\uFE0F Plugin agent ${frontmatter.name} (from ${plugin.manifest.name}): 'model' field is deprecated and ignored. Use 'model_name' instead, or omit to use default 'task' model.`,\n )\n warnedAgents.add(frontmatter.name)\n }\n\n // Parse optional fields\n const disallowedTools = parseDisallowedTools(\n frontmatter.disallowedTools,\n )\n const skills = parseSkills(frontmatter.skills)\n const memory = parseMemory(frontmatter.memory)\n const permissionMode = parsePermissionMode(frontmatter.permissionMode)\n\n // Build agent config with all Claude Code spec fields\n const agent: AgentConfig = {\n agentType: frontmatter.name,\n whenToUse: frontmatter.description.replace(/\\\\n/g, '\\n'),\n tools: parseTools(frontmatter.tools),\n systemPrompt: body.trim(),\n location: 'plugin',\n pluginName: plugin.manifest.name,\n sourcePath: pluginAgent.filePath,\n // Optional fields\n ...(frontmatter.color && { color: frontmatter.color }),\n ...(frontmatter.model_name && {\n model_name: frontmatter.model_name,\n }),\n ...(disallowedTools && { disallowedTools }),\n ...(permissionMode && { permissionMode }),\n ...(skills && { skills }),\n ...(memory && { memory }),\n ...(frontmatter.maxThinkingTokens && {\n maxThinkingTokens: Number(frontmatter.maxThinkingTokens),\n }),\n ...(frontmatter.hooks && {\n hooks: frontmatter.hooks as AgentHooks,\n }),\n ...(frontmatter.isolation && { isolation: frontmatter.isolation }),\n ...(frontmatter.background && {\n background: frontmatter.background === true,\n }),\n ...(frontmatter.mcpServers && {\n mcpServers: frontmatter.mcpServers,\n }),\n }\n\n agents.push(agent)\n } catch (error) {\n console.warn(\n `Failed to load plugin agent ${pluginAgent.name} from ${plugin.manifest.name}:`,\n error,\n )\n }\n }\n }\n } catch (error) {\n console.warn('Failed to load plugin agents:', error)\n }\n\n return agents\n}\n\n/**\n * Load all agent configurations\n *\n * Directory Priority (later overrides earlier):\n * 1. Built-in agents (general-purpose, Explore, Plan, Bash)\n * 2. Plugin agents\n * 3. ~/.minto/agents (Minto user directory)\n * 4. ./.minto/agents (Minto project directory - highest priority)\n */\nasync function loadAllAgents(): Promise<{\n activeAgents: AgentConfig[]\n allAgents: AgentConfig[]\n}> {\n try {\n const home = homedir()\n const cwd = getCwd()\n\n // Define directories in priority order (later overrides earlier)\n // .claude/ dirs are legacy fallbacks, .minto/ dirs take precedence\n const userClaudeDir = join(home, '.claude', 'agents')\n const userMintoDir = join(home, '.minto', 'agents')\n const projectClaudeDir = join(cwd, '.claude', 'agents')\n const projectMintoDir = join(cwd, '.minto', 'agents')\n\n // Load from all sources in parallel\n const [\n pluginAgents,\n userClaudeAgents,\n userMintoAgents,\n projectClaudeAgents,\n projectMintoAgents,\n ] = await Promise.all([\n loadPluginAgents(),\n scanAgentDirectory(userClaudeDir, 'user'),\n scanAgentDirectory(userMintoDir, 'user'),\n scanAgentDirectory(projectClaudeDir, 'project'),\n scanAgentDirectory(projectMintoDir, 'project'),\n ])\n\n // Apply priority override: built-in < plugin < user/.claude < user/.minto < project/.claude < project/.minto\n // Later entries override earlier ones with the same agentType\n // Plugin agents are namespaced as \"pluginName:agentType\"\n const agentMap = new Map<string, AgentConfig>()\n\n // Add in priority order (later entries override earlier ones)\n for (const agent of BUILTIN_AGENTS) {\n agentMap.set(agent.agentType, agent)\n }\n for (const agent of pluginAgents) {\n // Namespace plugin agents as \"pluginName:agentType\"\n const namespacedType = agent.pluginName\n ? `${agent.pluginName}:${agent.agentType}`\n : agent.agentType\n agentMap.set(namespacedType, { ...agent, agentType: namespacedType })\n }\n for (const agent of userClaudeAgents) {\n agentMap.set(agent.agentType, agent)\n }\n for (const agent of userMintoAgents) {\n agentMap.set(agent.agentType, agent)\n }\n for (const agent of projectClaudeAgents) {\n agentMap.set(agent.agentType, agent)\n }\n for (const agent of projectMintoAgents) {\n agentMap.set(agent.agentType, agent)\n }\n\n const activeAgents = Array.from(agentMap.values())\n const allAgents = [\n ...BUILTIN_AGENTS,\n ...pluginAgents,\n ...userClaudeAgents,\n ...userMintoAgents,\n ...projectClaudeAgents,\n ...projectMintoAgents,\n ]\n\n return { activeAgents, allAgents }\n } catch (error) {\n console.error('Failed to load agents, falling back to built-in:', error)\n return {\n activeAgents: [...BUILTIN_AGENTS],\n allAgents: [...BUILTIN_AGENTS],\n }\n }\n}\n\n// Memoized version for performance\nexport const getActiveAgents = memoize(async (): Promise<AgentConfig[]> => {\n const { activeAgents } = await loadAllAgents()\n return activeAgents\n})\n\n// Get all agents (both active and overridden)\nexport const getAllAgents = memoize(async (): Promise<AgentConfig[]> => {\n const { allAgents } = await loadAllAgents()\n return allAgents\n})\n\n// Clear cache when needed\nexport function clearAgentCache() {\n getActiveAgents.cache?.clear?.()\n getAllAgents.cache?.clear?.()\n getAgentByType.cache?.clear?.()\n getAvailableAgentTypes.cache?.clear?.()\n}\n\n// Get a specific agent by type\n// Supports both namespaced (\"plugin:agent\") and bare (\"agent\") lookups.\n// Bare names match non-plugin agents first; if not found, searches plugin agents by component name.\nexport const getAgentByType = memoize(\n async (agentType: string): Promise<AgentConfig | undefined> => {\n const agents = await getActiveAgents()\n\n // Direct match (exact agentType including namespace)\n const direct = agents.find(agent => agent.agentType === agentType)\n if (direct) return direct\n\n // If no colon in name, try to find among plugin agents by bare component name\n if (!agentType.includes(':')) {\n for (const agent of agents) {\n if (\n agent.agentType.includes(':') &&\n agent.agentType.split(':').pop() === agentType\n ) {\n return agent\n }\n }\n }\n\n return undefined\n },\n)\n\n// Get all available agent types for validation\nexport const getAvailableAgentTypes = memoize(async (): Promise<string[]> => {\n const agents = await getActiveAgents()\n return agents.map(agent => agent.agentType)\n})\n\n// File watcher for hot reload\nlet watchers: FSWatcher[] = []\n\n/**\n * Start watching agent configuration directories for changes\n */\nexport async function startAgentWatcher(onChange?: () => void): Promise<void> {\n await stopAgentWatcher() // Clean up any existing watchers\n\n const home = homedir()\n const cwd = getCwd()\n\n // Watch .minto and .claude (legacy) directories\n const directories = [\n { path: join(home, '.claude', 'agents'), label: 'user/.claude' },\n { path: join(home, '.minto', 'agents'), label: 'user/.minto' },\n { path: join(cwd, '.claude', 'agents'), label: 'project/.claude' },\n { path: join(cwd, '.minto', 'agents'), label: 'project/.minto' },\n ]\n\n const watchDirectory = (dirPath: string, label: string) => {\n if (existsSync(dirPath)) {\n const watcher = watch(\n dirPath,\n { recursive: false },\n async (eventType, filename) => {\n if (filename && filename.endsWith('.md')) {\n console.log(\n `\uD83D\uDD04 Agent configuration changed in ${label}: ${filename}`,\n )\n clearAgentCache()\n // Also clear any other related caches\n getAllAgents.cache?.clear?.()\n onChange?.()\n }\n },\n )\n watchers.push(watcher)\n }\n }\n\n // Watch all directories\n for (const { path, label } of directories) {\n watchDirectory(path, label)\n }\n}\n\n/**\n * Stop watching agent configuration directories\n */\nexport async function stopAgentWatcher(): Promise<void> {\n // FSWatcher.close() is synchronous and does not accept a callback on Node 18/20\n try {\n for (const watcher of watchers) {\n try {\n watcher.close()\n } catch (err) {\n console.error('Failed to close file watcher:', err)\n }\n }\n } finally {\n watchers = []\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../src/utils/agentMemory.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Agent Memory Persistence\n *\n * Implements Claude Code specification for agent memory persistence.\n * Supports three scopes:\n * - user: ~/.minto/memory/ (persists across all projects)\n * - project: ./.minto/memory/ (persists within project, tracked in git)\n * - local: ./.minto/.memory/ (persists within project, gitignored)\n */\n\nimport {\n existsSync,\n readFileSync,\n writeFileSync,\n mkdirSync,\n unlinkSync,\n} from 'fs'\nimport { join, dirname } from 'path'\nimport { homedir } from 'os'\nimport { getCwd } from './state'\nimport type { AgentMemory } from './agentLoader'\n\n/**\n * Memory entry structure\n */\ninterface MemoryEntry {\n content: string\n updatedAt: number\n agentType: string\n version: number\n}\n\n/**\n * Get the base path for memory storage based on scope\n */\nfunction getMemoryBasePath(scope: AgentMemory['scope']): string {\n const cwd = getCwd()\n\n switch (scope) {\n case 'user':\n return join(homedir(), '.minto', 'memory')\n case 'project':\n return join(cwd, '.minto', 'memory')\n case 'local':\n return join(cwd, '.minto', '.memory') // Gitignored by convention\n default:\n return join(cwd, '.minto', 'memory')\n }\n}\n\n/**\n * Get the full path for a memory file\n */\nfunction getMemoryFilePath(
|
|
5
|
-
"mappings": "AAUA;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,MAAM,eAAe;AAC9B,SAAS,eAAe;AACxB,SAAS,cAAc;AAgBvB,SAAS,kBAAkB,OAAqC;AAC9D,QAAM,MAAM,OAAO;AAEnB,UAAQ,OAAO;AAAA,IACb,KAAK;AACH,aAAO,KAAK,QAAQ,GAAG,UAAU,QAAQ;AAAA,IAC3C,KAAK;AACH,aAAO,KAAK,KAAK,UAAU,QAAQ;AAAA,IACrC,KAAK;AACH,aAAO,KAAK,KAAK,UAAU,SAAS;AAAA;AAAA,IACtC;AACE,aAAO,KAAK,KAAK,UAAU,QAAQ;AAAA,EACvC;AACF;AAKA,SAAS,
|
|
4
|
+
"sourcesContent": ["/**\n * Agent Memory Persistence\n *\n * Implements Claude Code specification for agent memory persistence.\n * Supports three scopes:\n * - user: ~/.minto/memory/ (persists across all projects)\n * - project: ./.minto/memory/ (persists within project, tracked in git)\n * - local: ./.minto/.memory/ (persists within project, gitignored)\n */\n\nimport {\n existsSync,\n readFileSync,\n writeFileSync,\n mkdirSync,\n unlinkSync,\n} from 'fs'\nimport { join, dirname } from 'path'\nimport { homedir } from 'os'\nimport { getCwd } from './state'\nimport type { AgentMemory } from './agentLoader'\n\n/**\n * Memory entry structure\n */\ninterface MemoryEntry {\n content: string\n updatedAt: number\n agentType: string\n version: number\n}\n\n/**\n * Get the base path for memory storage based on scope\n */\nfunction getMemoryBasePath(scope: AgentMemory['scope']): string {\n const cwd = getCwd()\n\n switch (scope) {\n case 'user':\n return join(homedir(), '.minto', 'memory')\n case 'project':\n return join(cwd, '.minto', 'memory')\n case 'local':\n return join(cwd, '.minto', '.memory') // Gitignored by convention\n default:\n return join(cwd, '.minto', 'memory')\n }\n}\n\n/**\n * Get the full path for a memory file\n */\nfunction getMemoryFilePath(agentType: string, memory: AgentMemory): string {\n const basePath = getMemoryBasePath(memory.scope)\n const key = memory.key || agentType\n // Sanitize key to be a valid filename\n const sanitizedKey = key.replace(/[^a-zA-Z0-9_-]/g, '_')\n return join(basePath, `${sanitizedKey}.json`)\n}\n\n/**\n * Load agent memory from persistent storage\n *\n * @param agentType - The agent type identifier\n * @param memory - Memory configuration\n * @returns Memory content if found, null otherwise\n */\nexport async function loadAgentMemory(\n agentType: string,\n memory: AgentMemory,\n): Promise<string | null> {\n const filePath = getMemoryFilePath(agentType, memory)\n\n if (!existsSync(filePath)) {\n return null\n }\n\n try {\n const data = JSON.parse(readFileSync(filePath, 'utf-8')) as MemoryEntry\n return data.content\n } catch (error) {\n console.warn(`Failed to load agent memory from ${filePath}:`, error)\n return null\n }\n}\n\n/**\n * Save agent memory to persistent storage\n *\n * @param agentType - The agent type identifier\n * @param memory - Memory configuration\n * @param content - Content to save\n */\nexport async function saveAgentMemory(\n agentType: string,\n memory: AgentMemory,\n content: string,\n): Promise<void> {\n const filePath = getMemoryFilePath(agentType, memory)\n const dirPath = dirname(filePath)\n\n try {\n // Ensure directory exists\n if (!existsSync(dirPath)) {\n mkdirSync(dirPath, { recursive: true })\n }\n\n // Load existing entry to preserve version\n let version = 1\n if (existsSync(filePath)) {\n try {\n const existing = JSON.parse(\n readFileSync(filePath, 'utf-8'),\n ) as MemoryEntry\n version = (existing.version || 0) + 1\n } catch {\n // Ignore parse errors, start fresh\n }\n }\n\n const entry: MemoryEntry = {\n content,\n updatedAt: Date.now(),\n agentType,\n version,\n }\n\n writeFileSync(filePath, JSON.stringify(entry, null, 2), 'utf-8')\n } catch (error) {\n console.error(`Failed to save agent memory to ${filePath}:`, error)\n throw error\n }\n}\n\n/**\n * Delete agent memory from persistent storage\n *\n * @param agentType - The agent type identifier\n * @param memory - Memory configuration\n */\nexport async function deleteAgentMemory(\n agentType: string,\n memory: AgentMemory,\n): Promise<void> {\n const filePath = getMemoryFilePath(agentType, memory)\n\n if (existsSync(filePath)) {\n try {\n unlinkSync(filePath)\n } catch (error) {\n console.warn(`Failed to delete agent memory at ${filePath}:`, error)\n }\n }\n}\n\n/**\n * Check if agent memory exists\n *\n * @param agentType - The agent type identifier\n * @param memory - Memory configuration\n * @returns True if memory exists\n */\nexport function hasAgentMemory(\n agentType: string,\n memory: AgentMemory,\n): boolean {\n const filePath = getMemoryFilePath(agentType, memory)\n return existsSync(filePath)\n}\n\n/**\n * Get memory metadata without loading content\n *\n * @param agentType - The agent type identifier\n * @param memory - Memory configuration\n * @returns Metadata if found, null otherwise\n */\nexport async function getAgentMemoryMetadata(\n agentType: string,\n memory: AgentMemory,\n): Promise<{ updatedAt: number; version: number } | null> {\n const filePath = getMemoryFilePath(agentType, memory)\n\n if (!existsSync(filePath)) {\n return null\n }\n\n try {\n const data = JSON.parse(readFileSync(filePath, 'utf-8')) as MemoryEntry\n return {\n updatedAt: data.updatedAt,\n version: data.version,\n }\n } catch {\n return null\n }\n}\n\n/**\n * Format memory content for injection into system prompt\n *\n * @param content - Memory content\n * @param agentType - Agent type for context\n * @returns Formatted memory section for system prompt\n */\nexport function formatMemoryForPrompt(\n content: string,\n agentType: string,\n): string {\n return `\n<agent-memory agent=\"${agentType}\">\n${content}\n</agent-memory>\n`\n}\n\n/**\n * Ensure the .memory directory is gitignored for 'local' scope\n * This should be called once during initialization\n */\nexport function ensureLocalMemoryGitignored(): void {\n const cwd = getCwd()\n const gitignorePath = join(cwd, '.gitignore')\n const memoryPattern = '.minto/.memory'\n\n try {\n if (existsSync(gitignorePath)) {\n const content = readFileSync(gitignorePath, 'utf-8')\n if (!content.includes(memoryPattern)) {\n // Could optionally append to .gitignore, but for now just warn\n console.debug(\n `Consider adding '${memoryPattern}' to .gitignore for local agent memory`,\n )\n }\n }\n } catch {\n // Ignore errors reading .gitignore\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -128,7 +128,7 @@ function useUnifiedAnimation({
|
|
|
128
128
|
const ELAPSED_INTERVAL = 1e3;
|
|
129
129
|
useEffect(() => {
|
|
130
130
|
if (!enabled) {
|
|
131
|
-
return;
|
|
131
|
+
return void 0;
|
|
132
132
|
}
|
|
133
133
|
const uniqueId = `${componentId}-unified-animation`;
|
|
134
134
|
const unsubscribe = animationManager.subscribe(
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../src/utils/animationManager.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Unified Animation Manager\n *\n * Consolidates multiple animation timers into a single requestAnimationFrame loop\n * to prevent render thrashing and screen flickering.\n *\n * Benefits:\n * - Single animation frame callback instead of multiple setInterval timers\n * - Batched state updates reduce React re-renders\n * - Configurable frame rates for different animation types\n * - Automatic cleanup on unmount\n */\n\ntype AnimationCallback = (deltaTime: number) => void\n\ninterface AnimationSubscription {\n id: string\n callback: AnimationCallback\n interval: number // Minimum interval between updates in ms\n lastUpdate: number\n}\n\nclass AnimationManager {\n private static instance: AnimationManager\n private subscriptions: Map<string, AnimationSubscription> = new Map()\n private rafId: ReturnType<typeof setTimeout> | null = null\n private lastFrameTime: number = 0\n private isRunning: boolean = false\n\n // Use 30fps as base rate to reduce terminal render load\n // Terminal rendering is expensive, unlike browser canvas rendering\n private readonly FRAME_INTERVAL = 33 // ~30fps (was 16ms/60fps)\n\n private constructor() {}\n\n static getInstance(): AnimationManager {\n if (!AnimationManager.instance) {\n AnimationManager.instance = new AnimationManager()\n }\n return AnimationManager.instance\n }\n\n /**\n * Subscribe to animation updates\n * @param id Unique identifier for this subscription\n * @param callback Function to call on each animation frame\n * @param interval Minimum interval between callback invocations (ms)\n * @returns Unsubscribe function\n */\n subscribe(\n id: string,\n callback: AnimationCallback,\n interval: number,\n ): () => void {\n this.subscriptions.set(id, {\n id,\n callback,\n interval,\n lastUpdate: 0,\n })\n\n if (!this.isRunning) {\n this.start()\n }\n\n return () => this.unsubscribe(id)\n }\n\n /**\n * Unsubscribe from animation updates\n */\n unsubscribe(id: string): void {\n this.subscriptions.delete(id)\n\n if (this.subscriptions.size === 0) {\n this.stop()\n }\n }\n\n /**\n * Start the animation loop\n */\n private start(): void {\n if (this.isRunning) return\n\n this.isRunning = true\n this.lastFrameTime = Date.now()\n this.tick()\n }\n\n /**\n * Stop the animation loop\n */\n private stop(): void {\n if (!this.isRunning) return\n\n this.isRunning = false\n if (this.rafId !== null) {\n clearTimeout(this.rafId)\n this.rafId = null\n }\n }\n\n /**\n * Main animation tick\n */\n private tick = (): void => {\n if (!this.isRunning) return\n\n const now = Date.now()\n const deltaTime = now - this.lastFrameTime\n this.lastFrameTime = now\n\n // Batch all callbacks that need to run this frame\n const callbacksToRun: AnimationCallback[] = []\n\n for (const sub of this.subscriptions.values()) {\n const timeSinceLastUpdate = now - sub.lastUpdate\n\n if (timeSinceLastUpdate >= sub.interval) {\n sub.lastUpdate = now\n callbacksToRun.push(sub.callback)\n }\n }\n\n // Execute all callbacks in a single batch\n // This allows React to batch the state updates\n if (callbacksToRun.length > 0) {\n for (const callback of callbacksToRun) {\n try {\n callback(deltaTime)\n } catch (error) {\n // Silently ignore callback errors to prevent animation loop from breaking\n }\n }\n }\n\n // Schedule next frame using setTimeout (works in Node.js/Bun)\n this.rafId = setTimeout(this.tick, this.FRAME_INTERVAL)\n }\n\n /**\n * Get the number of active subscriptions\n */\n getSubscriptionCount(): number {\n return this.subscriptions.size\n }\n\n /**\n * Check if animation loop is running\n */\n isAnimating(): boolean {\n return this.isRunning\n }\n}\n\n// Export singleton instance\nexport const animationManager = AnimationManager.getInstance()\n\n/**\n * React hook for unified animation updates\n *\n * Usage:\n * ```typescript\n * const { spinnerFrame, dotVisible, elapsedTime } = useUnifiedAnimation({\n * enabled: hasRunningTasks,\n * startTime: Date.now(),\n * spinnerFrameCount: 12,\n * })\n * ```\n */\nexport interface UnifiedAnimationState {\n spinnerFrame: number\n dotVisible: boolean\n elapsedTime: number\n}\n\nexport interface UseUnifiedAnimationOptions {\n enabled: boolean\n startTime: number\n spinnerFrameCount: number\n componentId: string\n}\n\nimport { useState, useEffect, useRef, useCallback } from 'react'\n\nexport function useUnifiedAnimation({\n enabled,\n startTime,\n spinnerFrameCount,\n componentId,\n}: UseUnifiedAnimationOptions): UnifiedAnimationState {\n // Use a single state object to batch updates\n const [animState, setAnimState] = useState<UnifiedAnimationState>(() => ({\n spinnerFrame: 0,\n dotVisible: true,\n elapsedTime: Math.floor((Date.now() - startTime) / 1000),\n }))\n\n // Store startTime in ref to avoid closure issues\n const startTimeRef = useRef(startTime)\n startTimeRef.current = startTime\n\n // Track timing internally to reduce state updates\n const timingRef = useRef({\n lastSpinnerUpdate: 0,\n lastDotUpdate: 0,\n lastElapsedUpdate: 0,\n })\n\n // Animation intervals - optimized for less flicker\n // Slower spinner reduces terminal re-render frequency\n const SPINNER_INTERVAL = 200 // 200ms (was 120ms) - reduces flicker by ~40%\n const DOT_INTERVAL = 800 // 800ms (was 600ms) - less visual noise\n const ELAPSED_INTERVAL = 1000 // 1s - unchanged\n\n useEffect(() => {\n if (!enabled) {\n return\n }\n\n const uniqueId = `${componentId}-unified-animation`\n\n const unsubscribe = animationManager.subscribe(\n uniqueId,\n () => {\n const now = Date.now()\n const timing = timingRef.current\n\n // Use functional state update to avoid closure issues\n setAnimState(prevState => {\n let needsUpdate = false\n let newSpinnerFrame = prevState.spinnerFrame\n let newDotVisible = prevState.dotVisible\n let newElapsedTime = prevState.elapsedTime\n\n // Check spinner update\n if (now - timing.lastSpinnerUpdate >= SPINNER_INTERVAL) {\n timing.lastSpinnerUpdate = now\n newSpinnerFrame = (newSpinnerFrame + 1) % spinnerFrameCount\n needsUpdate = true\n }\n\n // Check dot update\n if (now - timing.lastDotUpdate >= DOT_INTERVAL) {\n timing.lastDotUpdate = now\n newDotVisible = !newDotVisible\n needsUpdate = true\n }\n\n // Check elapsed time update\n if (now - timing.lastElapsedUpdate >= ELAPSED_INTERVAL) {\n timing.lastElapsedUpdate = now\n newElapsedTime = Math.floor((now - startTimeRef.current) / 1000)\n needsUpdate = true\n }\n\n // Return same object if no update needed (prevents unnecessary re-render)\n if (!needsUpdate) {\n return prevState\n }\n\n return {\n spinnerFrame: newSpinnerFrame,\n dotVisible: newDotVisible,\n elapsedTime: newElapsedTime,\n }\n })\n },\n // Use fastest needed interval for checking\n Math.min(SPINNER_INTERVAL, DOT_INTERVAL, ELAPSED_INTERVAL),\n )\n\n return unsubscribe\n }, [enabled, spinnerFrameCount, componentId])\n\n return animState\n}\n\n/**\n * Throttle utility for event handlers\n */\nexport function throttle<T extends (...args: any[]) => any>(\n func: T,\n limit: number,\n): (...args: Parameters<T>) => void {\n let inThrottle = false\n let lastArgs: Parameters<T> | null = null\n\n return function (this: any, ...args: Parameters<T>) {\n if (!inThrottle) {\n func.apply(this, args)\n inThrottle = true\n setTimeout(() => {\n inThrottle = false\n if (lastArgs) {\n func.apply(this, lastArgs)\n lastArgs = null\n }\n }, limit)\n } else {\n lastArgs = args\n }\n }\n}\n\n/**\n * Debounce utility for event handlers\n */\nexport function debounce<T extends (...args: any[]) => any>(\n func: T,\n wait: number,\n): (...args: Parameters<T>) => void {\n let timeoutId: ReturnType<typeof setTimeout> | null = null\n\n return function (this: any, ...args: Parameters<T>) {\n if (timeoutId !== null) {\n clearTimeout(timeoutId)\n }\n timeoutId = setTimeout(() => {\n func.apply(this, args)\n timeoutId = null\n }, wait)\n }\n}\n"],
|
|
5
|
-
"mappings": "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
|
|
4
|
+
"sourcesContent": ["/**\n * Unified Animation Manager\n *\n * Consolidates multiple animation timers into a single requestAnimationFrame loop\n * to prevent render thrashing and screen flickering.\n *\n * Benefits:\n * - Single animation frame callback instead of multiple setInterval timers\n * - Batched state updates reduce React re-renders\n * - Configurable frame rates for different animation types\n * - Automatic cleanup on unmount\n */\n\ntype AnimationCallback = (deltaTime: number) => void\n\ninterface AnimationSubscription {\n id: string\n callback: AnimationCallback\n interval: number // Minimum interval between updates in ms\n lastUpdate: number\n}\n\nclass AnimationManager {\n private static instance: AnimationManager\n private subscriptions: Map<string, AnimationSubscription> = new Map()\n private rafId: ReturnType<typeof setTimeout> | null = null\n private lastFrameTime: number = 0\n private isRunning: boolean = false\n\n // Use 30fps as base rate to reduce terminal render load\n // Terminal rendering is expensive, unlike browser canvas rendering\n private readonly FRAME_INTERVAL = 33 // ~30fps (was 16ms/60fps)\n\n private constructor() {}\n\n static getInstance(): AnimationManager {\n if (!AnimationManager.instance) {\n AnimationManager.instance = new AnimationManager()\n }\n return AnimationManager.instance\n }\n\n /**\n * Subscribe to animation updates\n * @param id Unique identifier for this subscription\n * @param callback Function to call on each animation frame\n * @param interval Minimum interval between callback invocations (ms)\n * @returns Unsubscribe function\n */\n subscribe(\n id: string,\n callback: AnimationCallback,\n interval: number,\n ): () => void {\n this.subscriptions.set(id, {\n id,\n callback,\n interval,\n lastUpdate: 0,\n })\n\n if (!this.isRunning) {\n this.start()\n }\n\n return () => this.unsubscribe(id)\n }\n\n /**\n * Unsubscribe from animation updates\n */\n unsubscribe(id: string): void {\n this.subscriptions.delete(id)\n\n if (this.subscriptions.size === 0) {\n this.stop()\n }\n }\n\n /**\n * Start the animation loop\n */\n private start(): void {\n if (this.isRunning) return\n\n this.isRunning = true\n this.lastFrameTime = Date.now()\n this.tick()\n }\n\n /**\n * Stop the animation loop\n */\n private stop(): void {\n if (!this.isRunning) return\n\n this.isRunning = false\n if (this.rafId !== null) {\n clearTimeout(this.rafId)\n this.rafId = null\n }\n }\n\n /**\n * Main animation tick\n */\n private tick = (): void => {\n if (!this.isRunning) return\n\n const now = Date.now()\n const deltaTime = now - this.lastFrameTime\n this.lastFrameTime = now\n\n // Batch all callbacks that need to run this frame\n const callbacksToRun: AnimationCallback[] = []\n\n for (const sub of this.subscriptions.values()) {\n const timeSinceLastUpdate = now - sub.lastUpdate\n\n if (timeSinceLastUpdate >= sub.interval) {\n sub.lastUpdate = now\n callbacksToRun.push(sub.callback)\n }\n }\n\n // Execute all callbacks in a single batch\n // This allows React to batch the state updates\n if (callbacksToRun.length > 0) {\n for (const callback of callbacksToRun) {\n try {\n callback(deltaTime)\n } catch (error) {\n // Silently ignore callback errors to prevent animation loop from breaking\n }\n }\n }\n\n // Schedule next frame using setTimeout (works in Node.js/Bun)\n this.rafId = setTimeout(this.tick, this.FRAME_INTERVAL)\n }\n\n /**\n * Get the number of active subscriptions\n */\n getSubscriptionCount(): number {\n return this.subscriptions.size\n }\n\n /**\n * Check if animation loop is running\n */\n isAnimating(): boolean {\n return this.isRunning\n }\n}\n\n// Export singleton instance\nexport const animationManager = AnimationManager.getInstance()\n\n/**\n * React hook for unified animation updates\n *\n * Usage:\n * ```typescript\n * const { spinnerFrame, dotVisible, elapsedTime } = useUnifiedAnimation({\n * enabled: hasRunningTasks,\n * startTime: Date.now(),\n * spinnerFrameCount: 12,\n * })\n * ```\n */\nexport interface UnifiedAnimationState {\n spinnerFrame: number\n dotVisible: boolean\n elapsedTime: number\n}\n\nexport interface UseUnifiedAnimationOptions {\n enabled: boolean\n startTime: number\n spinnerFrameCount: number\n componentId: string\n}\n\nimport { useState, useEffect, useRef, useCallback } from 'react'\n\nexport function useUnifiedAnimation({\n enabled,\n startTime,\n spinnerFrameCount,\n componentId,\n}: UseUnifiedAnimationOptions): UnifiedAnimationState {\n // Use a single state object to batch updates\n const [animState, setAnimState] = useState<UnifiedAnimationState>(() => ({\n spinnerFrame: 0,\n dotVisible: true,\n elapsedTime: Math.floor((Date.now() - startTime) / 1000),\n }))\n\n // Store startTime in ref to avoid closure issues\n const startTimeRef = useRef(startTime)\n startTimeRef.current = startTime\n\n // Track timing internally to reduce state updates\n const timingRef = useRef({\n lastSpinnerUpdate: 0,\n lastDotUpdate: 0,\n lastElapsedUpdate: 0,\n })\n\n // Animation intervals - optimized for less flicker\n // Slower spinner reduces terminal re-render frequency\n const SPINNER_INTERVAL = 200 // 200ms (was 120ms) - reduces flicker by ~40%\n const DOT_INTERVAL = 800 // 800ms (was 600ms) - less visual noise\n const ELAPSED_INTERVAL = 1000 // 1s - unchanged\n\n useEffect(() => {\n if (!enabled) {\n return undefined\n }\n\n const uniqueId = `${componentId}-unified-animation`\n\n const unsubscribe = animationManager.subscribe(\n uniqueId,\n () => {\n const now = Date.now()\n const timing = timingRef.current\n\n // Use functional state update to avoid closure issues\n setAnimState(prevState => {\n let needsUpdate = false\n let newSpinnerFrame = prevState.spinnerFrame\n let newDotVisible = prevState.dotVisible\n let newElapsedTime = prevState.elapsedTime\n\n // Check spinner update\n if (now - timing.lastSpinnerUpdate >= SPINNER_INTERVAL) {\n timing.lastSpinnerUpdate = now\n newSpinnerFrame = (newSpinnerFrame + 1) % spinnerFrameCount\n needsUpdate = true\n }\n\n // Check dot update\n if (now - timing.lastDotUpdate >= DOT_INTERVAL) {\n timing.lastDotUpdate = now\n newDotVisible = !newDotVisible\n needsUpdate = true\n }\n\n // Check elapsed time update\n if (now - timing.lastElapsedUpdate >= ELAPSED_INTERVAL) {\n timing.lastElapsedUpdate = now\n newElapsedTime = Math.floor((now - startTimeRef.current) / 1000)\n needsUpdate = true\n }\n\n // Return same object if no update needed (prevents unnecessary re-render)\n if (!needsUpdate) {\n return prevState\n }\n\n return {\n spinnerFrame: newSpinnerFrame,\n dotVisible: newDotVisible,\n elapsedTime: newElapsedTime,\n }\n })\n },\n // Use fastest needed interval for checking\n Math.min(SPINNER_INTERVAL, DOT_INTERVAL, ELAPSED_INTERVAL),\n )\n\n return unsubscribe\n }, [enabled, spinnerFrameCount, componentId])\n\n return animState\n}\n\n/**\n * Throttle utility for event handlers\n */\nexport function throttle<T extends (...args: any[]) => any>(\n func: T,\n limit: number,\n): (...args: Parameters<T>) => void {\n let inThrottle = false\n let lastArgs: Parameters<T> | null = null\n\n return function (this: any, ...args: Parameters<T>) {\n if (!inThrottle) {\n func.apply(this, args)\n inThrottle = true\n setTimeout(() => {\n inThrottle = false\n if (lastArgs) {\n func.apply(this, lastArgs)\n lastArgs = null\n }\n }, limit)\n } else {\n lastArgs = args\n }\n }\n}\n\n/**\n * Debounce utility for event handlers\n */\nexport function debounce<T extends (...args: any[]) => any>(\n func: T,\n wait: number,\n): (...args: Parameters<T>) => void {\n let timeoutId: ReturnType<typeof setTimeout> | null = null\n\n return function (this: any, ...args: Parameters<T>) {\n if (timeoutId !== null) {\n clearTimeout(timeoutId)\n }\n timeoutId = setTimeout(() => {\n func.apply(this, args)\n timeoutId = null\n }, wait)\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
package/dist/utils/ask.js
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { last } from "lodash-es";
|
|
2
|
-
import { getSystemPrompt } from "../constants/prompts.js";
|
|
2
|
+
import { getSystemPrompt } from "../constants/prompts/index.js";
|
|
3
3
|
import { getContext } from "../context.js";
|
|
4
4
|
import { getTotalCost } from "../cost-tracker.js";
|
|
5
5
|
import { query } from "../query.js";
|
package/dist/utils/async.js
CHANGED
|
@@ -60,7 +60,8 @@ async function retryWithBackoff(fn, options = {}) {
|
|
|
60
60
|
maxRetries = 3,
|
|
61
61
|
signal,
|
|
62
62
|
config = DEFAULT_RETRY_CONFIG,
|
|
63
|
-
onRetry
|
|
63
|
+
onRetry,
|
|
64
|
+
shouldRetry
|
|
64
65
|
} = options;
|
|
65
66
|
const fullConfig = { ...DEFAULT_RETRY_CONFIG, ...config };
|
|
66
67
|
let lastError;
|
|
@@ -72,6 +73,9 @@ async function retryWithBackoff(fn, options = {}) {
|
|
|
72
73
|
return await fn();
|
|
73
74
|
} catch (error) {
|
|
74
75
|
lastError = error instanceof Error ? error : new Error(String(error));
|
|
76
|
+
if (shouldRetry && !shouldRetry(lastError)) {
|
|
77
|
+
throw lastError;
|
|
78
|
+
}
|
|
75
79
|
if (attempt < maxRetries) {
|
|
76
80
|
onRetry?.(attempt, lastError);
|
|
77
81
|
const delay = getRetryDelay(attempt, void 0, fullConfig);
|
package/dist/utils/async.js.map
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../src/utils/async.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Shared async utilities for consistent async patterns across the codebase.\n * Extracted from claude.ts and openai.ts to eliminate code duplication.\n */\n\nexport interface RetryConfig {\n baseDelayMs: number\n maxDelayMs: number\n maxServerDelayMs: number\n jitterFactor: number\n}\n\nexport const DEFAULT_RETRY_CONFIG: RetryConfig = {\n baseDelayMs: 1000,\n maxDelayMs: 32000,\n maxServerDelayMs: 60000,\n jitterFactor: 0.1,\n}\n\nexport interface RetryOptions {\n maxRetries?: number\n signal?: AbortSignal\n config?: Partial<RetryConfig>\n onRetry?: (attempt: number, error: Error) => void\n}\n\n/**\n * Creates a promise that resolves after the specified delay.\n * Can be aborted via an AbortSignal.\n */\nexport function abortableDelay(\n delayMs: number,\n signal?: AbortSignal,\n): Promise<void> {\n return new Promise((resolve, reject) => {\n if (signal?.aborted) {\n reject(new Error('Request was aborted'))\n return\n }\n\n const timeoutId = setTimeout(() => {\n resolve()\n }, delayMs)\n\n if (signal) {\n const abortHandler = () => {\n clearTimeout(timeoutId)\n reject(new Error('Request was aborted'))\n }\n signal.addEventListener('abort', abortHandler, { once: true })\n }\n })\n}\n\n/**\n * Calculate retry delay with exponential backoff and optional jitter.\n * Respects server-provided Retry-After headers when available.\n */\nexport function getRetryDelay(\n attempt: number,\n retryAfterHeader?: string | null,\n config: RetryConfig = DEFAULT_RETRY_CONFIG,\n): number {\n if (retryAfterHeader) {\n const retryAfterMs = parseInt(retryAfterHeader, 10) * 1000\n if (!isNaN(retryAfterMs) && retryAfterMs > 0) {\n return Math.min(retryAfterMs, config.maxServerDelayMs)\n }\n }\n\n const delay = config.baseDelayMs * Math.pow(2, attempt - 1)\n const jitter = Math.random() * config.jitterFactor * delay\n\n return Math.min(delay + jitter, config.maxDelayMs)\n}\n\n/**\n * Wraps a promise with a timeout.\n * Rejects with a timeout error if the promise doesn't resolve in time.\n */\nexport function withTimeout<T>(\n promise: Promise<T>,\n timeoutMs: number,\n message = 'Operation timed out',\n): Promise<T> {\n return new Promise((resolve, reject) => {\n const timeoutId = setTimeout(() => {\n reject(new Error(message))\n }, timeoutMs)\n\n promise\n .then(result => {\n clearTimeout(timeoutId)\n resolve(result)\n })\n .catch(error => {\n clearTimeout(timeoutId)\n reject(error)\n })\n })\n}\n\n/**\n * Creates an AbortController that automatically aborts after the specified timeout.\n * Returns both the controller and a cleanup function.\n */\nexport function createTimeoutController(timeoutMs: number): {\n controller: AbortController\n cleanup: () => void\n} {\n const controller = new AbortController()\n const timeoutId = setTimeout(() => controller.abort(), timeoutMs)\n\n return {\n controller,\n cleanup: () => clearTimeout(timeoutId),\n }\n}\n\n/**\n * Retries an async function with exponential backoff.\n * Supports abort signals and custom retry conditions.\n */\nexport async function retryWithBackoff<T>(\n fn: () => Promise<T>,\n options: RetryOptions = {},\n): Promise<T> {\n const {\n maxRetries = 3,\n signal,\n config = DEFAULT_RETRY_CONFIG,\n onRetry,\n } = options\n\n const fullConfig = { ...DEFAULT_RETRY_CONFIG, ...config }\n let lastError: Error | undefined\n\n for (let attempt = 1; attempt <= maxRetries; attempt++) {\n if (signal?.aborted) {\n throw new Error('Request was aborted')\n }\n\n try {\n return await fn()\n } catch (error) {\n lastError = error instanceof Error ? error : new Error(String(error))\n\n if (attempt < maxRetries) {\n onRetry?.(attempt, lastError)\n const delay = getRetryDelay(attempt, undefined, fullConfig)\n await abortableDelay(delay, signal)\n }\n }\n }\n\n throw lastError ?? new Error('Retry failed with unknown error')\n}\n\n/**\n * Options for memoizeWithLimit\n */\nexport interface MemoizeOptions<T extends (...args: unknown[]) => unknown> {\n /** Maximum number of cached entries (default: 100) */\n max?: number\n /** Time-to-live in milliseconds (default: 60000) */\n ttl?: number\n /** Custom resolver function to generate cache key from arguments */\n resolver?: (...args: Parameters<T>) => string\n}\n\n/**\n * Creates a memoized version of a function with LRU cache and TTL.\n * Unlike lodash memoize which has unlimited cache, this has bounded memory usage.\n *\n * @param fn - The function to memoize\n * @param options - Cache configuration options\n * @returns Memoized function\n */\nexport function memoizeWithLimit<T extends (...args: unknown[]) => unknown>(\n fn: T,\n options: MemoizeOptions<T> = {},\n): T {\n const { max = 100, ttl = 60000, resolver } = options\n\n interface CacheEntry {\n value: ReturnType<T>\n timestamp: number\n }\n\n const cache = new Map<string, CacheEntry>()\n\n // LRU eviction helper\n const evictOldest = () => {\n if (cache.size >= max) {\n const oldest = cache.keys().next().value\n if (oldest) cache.delete(oldest)\n }\n }\n\n // Check if entry is expired\n const isExpired = (entry: CacheEntry): boolean => {\n return Date.now() - entry.timestamp > ttl\n }\n\n return ((...args: Parameters<T>): ReturnType<T> => {\n const key = resolver ? resolver(...args) : JSON.stringify(args)\n\n // Check cache\n const existing = cache.get(key)\n if (existing && !isExpired(existing)) {\n // Move to end for LRU\n cache.delete(key)\n cache.set(key, existing)\n return existing.value\n }\n\n // Remove expired entry if exists\n if (existing) {\n cache.delete(key)\n }\n\n // Compute new value\n const value = fn(...args) as ReturnType<T>\n\n // Evict oldest if at capacity\n evictOldest()\n\n // Store in cache\n cache.set(key, { value, timestamp: Date.now() })\n\n return value\n }) as T\n}\n\n/**\n * Async version of memoizeWithLimit for async functions\n */\nexport function memoizeAsyncWithLimit<\n T extends (...args: unknown[]) => Promise<unknown>,\n>(fn: T, options: MemoizeOptions<T> = {}): T {\n const { max = 100, ttl = 60000, resolver } = options\n\n interface CacheEntry {\n value: Awaited<ReturnType<T>>\n timestamp: number\n }\n\n const cache = new Map<string, CacheEntry>()\n const pending = new Map<string, Promise<Awaited<ReturnType<T>>>>()\n\n const evictOldest = () => {\n if (cache.size >= max) {\n const oldest = cache.keys().next().value\n if (oldest) cache.delete(oldest)\n }\n }\n\n const isExpired = (entry: CacheEntry): boolean => {\n return Date.now() - entry.timestamp > ttl\n }\n\n return (async (...args: Parameters<T>): Promise<Awaited<ReturnType<T>>> => {\n const key = resolver ? resolver(...args) : JSON.stringify(args)\n\n // Check cache\n const existing = cache.get(key)\n if (existing && !isExpired(existing)) {\n cache.delete(key)\n cache.set(key, existing)\n return existing.value\n }\n\n // Check pending\n const pendingPromise = pending.get(key)\n if (pendingPromise) {\n return pendingPromise\n }\n\n // Remove expired\n if (existing) {\n cache.delete(key)\n }\n\n // Compute new value\n const promise = fn(...args) as Promise<Awaited<ReturnType<T>>>\n pending.set(key, promise)\n\n try {\n const value = await promise\n evictOldest()\n cache.set(key, { value, timestamp: Date.now() })\n return value\n } finally {\n pending.delete(key)\n }\n }) as T\n}\n"],
|
|
5
|
-
"mappings": "AAYO,MAAM,uBAAoC;AAAA,EAC/C,aAAa;AAAA,EACb,YAAY;AAAA,EACZ,kBAAkB;AAAA,EAClB,cAAc;AAChB;
|
|
4
|
+
"sourcesContent": ["/**\n * Shared async utilities for consistent async patterns across the codebase.\n * Extracted from claude.ts and openai.ts to eliminate code duplication.\n */\n\nexport interface RetryConfig {\n baseDelayMs: number\n maxDelayMs: number\n maxServerDelayMs: number\n jitterFactor: number\n}\n\nexport const DEFAULT_RETRY_CONFIG: RetryConfig = {\n baseDelayMs: 1000,\n maxDelayMs: 32000,\n maxServerDelayMs: 60000,\n jitterFactor: 0.1,\n}\n\nexport interface RetryOptions {\n maxRetries?: number\n signal?: AbortSignal\n config?: Partial<RetryConfig>\n onRetry?: (attempt: number, error: Error) => void\n /** Optional predicate to determine if a given error is retryable. Defaults to retrying all errors. */\n shouldRetry?: (error: Error) => boolean\n}\n\n/**\n * Creates a promise that resolves after the specified delay.\n * Can be aborted via an AbortSignal.\n */\nexport function abortableDelay(\n delayMs: number,\n signal?: AbortSignal,\n): Promise<void> {\n return new Promise((resolve, reject) => {\n if (signal?.aborted) {\n reject(new Error('Request was aborted'))\n return\n }\n\n const timeoutId = setTimeout(() => {\n resolve()\n }, delayMs)\n\n if (signal) {\n const abortHandler = () => {\n clearTimeout(timeoutId)\n reject(new Error('Request was aborted'))\n }\n signal.addEventListener('abort', abortHandler, { once: true })\n }\n })\n}\n\n/**\n * Calculate retry delay with exponential backoff and optional jitter.\n * Respects server-provided Retry-After headers when available.\n */\nexport function getRetryDelay(\n attempt: number,\n retryAfterHeader?: string | null,\n config: RetryConfig = DEFAULT_RETRY_CONFIG,\n): number {\n if (retryAfterHeader) {\n const retryAfterMs = parseInt(retryAfterHeader, 10) * 1000\n if (!isNaN(retryAfterMs) && retryAfterMs > 0) {\n return Math.min(retryAfterMs, config.maxServerDelayMs)\n }\n }\n\n const delay = config.baseDelayMs * Math.pow(2, attempt - 1)\n const jitter = Math.random() * config.jitterFactor * delay\n\n return Math.min(delay + jitter, config.maxDelayMs)\n}\n\n/**\n * Wraps a promise with a timeout.\n * Rejects with a timeout error if the promise doesn't resolve in time.\n */\nexport function withTimeout<T>(\n promise: Promise<T>,\n timeoutMs: number,\n message = 'Operation timed out',\n): Promise<T> {\n return new Promise((resolve, reject) => {\n const timeoutId = setTimeout(() => {\n reject(new Error(message))\n }, timeoutMs)\n\n promise\n .then(result => {\n clearTimeout(timeoutId)\n resolve(result)\n })\n .catch(error => {\n clearTimeout(timeoutId)\n reject(error)\n })\n })\n}\n\n/**\n * Creates an AbortController that automatically aborts after the specified timeout.\n * Returns both the controller and a cleanup function.\n */\nexport function createTimeoutController(timeoutMs: number): {\n controller: AbortController\n cleanup: () => void\n} {\n const controller = new AbortController()\n const timeoutId = setTimeout(() => controller.abort(), timeoutMs)\n\n return {\n controller,\n cleanup: () => clearTimeout(timeoutId),\n }\n}\n\n/**\n * Retries an async function with exponential backoff.\n * Supports abort signals and custom retry conditions.\n */\nexport async function retryWithBackoff<T>(\n fn: () => Promise<T>,\n options: RetryOptions = {},\n): Promise<T> {\n const {\n maxRetries = 3,\n signal,\n config = DEFAULT_RETRY_CONFIG,\n onRetry,\n shouldRetry,\n } = options\n\n const fullConfig = { ...DEFAULT_RETRY_CONFIG, ...config }\n let lastError: Error | undefined\n\n for (let attempt = 1; attempt <= maxRetries; attempt++) {\n if (signal?.aborted) {\n throw new Error('Request was aborted')\n }\n\n try {\n return await fn()\n } catch (error) {\n lastError = error instanceof Error ? error : new Error(String(error))\n\n // If shouldRetry predicate is provided and says no, throw immediately\n if (shouldRetry && !shouldRetry(lastError)) {\n throw lastError\n }\n\n if (attempt < maxRetries) {\n onRetry?.(attempt, lastError)\n const delay = getRetryDelay(attempt, undefined, fullConfig)\n await abortableDelay(delay, signal)\n }\n }\n }\n\n throw lastError ?? new Error('Retry failed with unknown error')\n}\n\n/**\n * Options for memoizeWithLimit\n */\nexport interface MemoizeOptions<T extends (...args: unknown[]) => unknown> {\n /** Maximum number of cached entries (default: 100) */\n max?: number\n /** Time-to-live in milliseconds (default: 60000) */\n ttl?: number\n /** Custom resolver function to generate cache key from arguments */\n resolver?: (...args: Parameters<T>) => string\n}\n\n/**\n * Creates a memoized version of a function with LRU cache and TTL.\n * Unlike lodash memoize which has unlimited cache, this has bounded memory usage.\n *\n * @param fn - The function to memoize\n * @param options - Cache configuration options\n * @returns Memoized function\n */\nexport function memoizeWithLimit<T extends (...args: unknown[]) => unknown>(\n fn: T,\n options: MemoizeOptions<T> = {},\n): T {\n const { max = 100, ttl = 60000, resolver } = options\n\n interface CacheEntry {\n value: ReturnType<T>\n timestamp: number\n }\n\n const cache = new Map<string, CacheEntry>()\n\n // LRU eviction helper\n const evictOldest = () => {\n if (cache.size >= max) {\n const oldest = cache.keys().next().value\n if (oldest) cache.delete(oldest)\n }\n }\n\n // Check if entry is expired\n const isExpired = (entry: CacheEntry): boolean => {\n return Date.now() - entry.timestamp > ttl\n }\n\n return ((...args: Parameters<T>): ReturnType<T> => {\n const key = resolver ? resolver(...args) : JSON.stringify(args)\n\n // Check cache\n const existing = cache.get(key)\n if (existing && !isExpired(existing)) {\n // Move to end for LRU\n cache.delete(key)\n cache.set(key, existing)\n return existing.value\n }\n\n // Remove expired entry if exists\n if (existing) {\n cache.delete(key)\n }\n\n // Compute new value\n const value = fn(...args) as ReturnType<T>\n\n // Evict oldest if at capacity\n evictOldest()\n\n // Store in cache\n cache.set(key, { value, timestamp: Date.now() })\n\n return value\n }) as T\n}\n\n/**\n * Async version of memoizeWithLimit for async functions\n */\nexport function memoizeAsyncWithLimit<\n T extends (...args: unknown[]) => Promise<unknown>,\n>(fn: T, options: MemoizeOptions<T> = {}): T {\n const { max = 100, ttl = 60000, resolver } = options\n\n interface CacheEntry {\n value: Awaited<ReturnType<T>>\n timestamp: number\n }\n\n const cache = new Map<string, CacheEntry>()\n const pending = new Map<string, Promise<Awaited<ReturnType<T>>>>()\n\n const evictOldest = () => {\n if (cache.size >= max) {\n const oldest = cache.keys().next().value\n if (oldest) cache.delete(oldest)\n }\n }\n\n const isExpired = (entry: CacheEntry): boolean => {\n return Date.now() - entry.timestamp > ttl\n }\n\n return (async (...args: Parameters<T>): Promise<Awaited<ReturnType<T>>> => {\n const key = resolver ? resolver(...args) : JSON.stringify(args)\n\n // Check cache\n const existing = cache.get(key)\n if (existing && !isExpired(existing)) {\n cache.delete(key)\n cache.set(key, existing)\n return existing.value\n }\n\n // Check pending\n const pendingPromise = pending.get(key)\n if (pendingPromise) {\n return pendingPromise\n }\n\n // Remove expired\n if (existing) {\n cache.delete(key)\n }\n\n // Compute new value\n const promise = fn(...args) as Promise<Awaited<ReturnType<T>>>\n pending.set(key, promise)\n\n try {\n const value = await promise\n evictOldest()\n cache.set(key, { value, timestamp: Date.now() })\n return value\n } finally {\n pending.delete(key)\n }\n }) as T\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|