@within-7/minto 0.3.10 → 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 +2 -2
- package/dist/commands/agents/AgentsCommand.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 +19 -9
- 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/mcp-interactive.js +14 -8
- 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 +45 -2
- 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/plugin/utils.js +33 -1
- package/dist/commands/plugin/utils.js.map +2 -2
- package/dist/commands/plugin.js +10 -1
- package/dist/commands/plugin.js.map +2 -2
- package/dist/commands/refreshCommands.js +2 -0
- package/dist/commands/refreshCommands.js.map +2 -2
- package/dist/commands/review.js +51 -0
- package/dist/commands/review.js.map +7 -0
- package/dist/commands/terminalSetup.js +6 -0
- package/dist/commands/terminalSetup.js.map +2 -2
- package/dist/commands/undo.js +8 -0
- 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/HighlightedCode.js +1 -0
- package/dist/components/HighlightedCode.js.map +2 -2
- package/dist/components/ModelSelector/ModelSelector.js +250 -143
- package/dist/components/ModelSelector/ModelSelector.js.map +2 -2
- package/dist/components/PromptInput.js +21 -6
- 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/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/SubagentBlock.js +43 -6
- package/dist/components/SubagentBlock.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 +1 -1
- 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 +2 -2
- package/dist/core/backupHook.js.map +2 -2
- package/dist/core/config/defaults.js +4 -1
- package/dist/core/config/defaults.js.map +2 -2
- package/dist/core/config/schema.js +7 -1
- package/dist/core/config/schema.js.map +2 -2
- package/dist/core/costTracker.js +18 -0
- 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/entrypoints/cli.js +65 -84
- 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 +8 -9
- package/dist/i18n/locales/en.js.map +2 -2
- package/dist/i18n/locales/zh-CN.js +8 -9
- 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/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/skillMarketplace.js +4 -1
- 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 +350 -3
- 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 +1 -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 +3 -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.map +1 -1
- package/dist/tools/NotebookEditTool/NotebookEditTool.js.map +1 -1
- 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 +75 -5
- 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 +1 -1
- package/dist/types/plugin.js.map +2 -2
- package/dist/utils/agentLoader.js +25 -3
- package/dist/utils/agentLoader.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/config.js +26 -128
- 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/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/memoizeWithTTL.js +25 -0
- package/dist/utils/memoizeWithTTL.js.map +7 -0
- package/dist/utils/model.js +34 -9
- package/dist/utils/model.js.map +2 -2
- package/dist/utils/pluginInstaller.js +34 -5
- package/dist/utils/pluginInstaller.js.map +2 -2
- package/dist/utils/pluginLoader.js +201 -32
- package/dist/utils/pluginLoader.js.map +2 -2
- package/dist/utils/safeFetch.js +45 -0
- package/dist/utils/safeFetch.js.map +7 -0
- package/dist/utils/skillLoader.js +59 -6
- 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/style.js +6 -3
- package/dist/utils/style.js.map +2 -2
- package/dist/utils/teamConfig.js +9 -3
- package/dist/utils/teamConfig.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 +2 -1
package/dist/types/plugin.js.map
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../src/types/plugin.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Plugin System Type Definitions\n */\n\nimport { z } from 'zod'\n\n/**\n * Author information (can be string or object)\n */\nexport const AuthorSchema = z.union([\n z.string(),\n z.object({\n name: z.string(),\n email: z.string().email().optional(),\n url: z.string().url().optional(),\n }),\n])\n\n/**\n * MCP Server Configuration Schema\n * Compatible with Claude Code CLI's .mcp.json format\n */\nexport const MCPServerConfigSchema = z.object({\n // Command-based (stdio) servers\n command: z.string().optional(),\n args: z.array(z.string()).optional().default([]),\n env: z.record(z.string()).optional().default({}),\n timeout: z.number().optional(),\n\n // HTTP/SSE servers\n type: z.enum(['stdio', 'http', 'sse']).optional().default('stdio'),\n url: z.string().optional(),\n headers: z.record(z.string()).optional(),\n})\n\nexport type MCPServerConfig = z.infer<typeof MCPServerConfigSchema>\n\n/**\n * Plugin Manifest Schema (plugin.json)\n * This is the source of truth for plugin metadata\n */\nexport const PluginManifestSchema = z.object({\n // Required fields\n name: z\n .string()\n .min(1)\n .regex(\n /^[a-z0-9-]+$/,\n 'Plugin name must be lowercase alphanumeric with hyphens',\n ),\n version: z.string().min(1), // Supports semver and commit hashes (for CC sync)\n description: z.string().min(1),\n\n // Optional metadata\n displayName: z.string().optional(),\n author: AuthorSchema.optional(),\n homepage: z.string().url().optional(),\n repository: z.string().url().optional(),\n license: z.string().optional(),\n\n // Plugin components (relative paths from plugin root)\n agents: z.array(z.string()).optional().default([]),\n commands: z.array(z.string()).optional().default([]),\n skills: z.array(z.string()).optional().default([]),\n hooks: z.array(z.string()).optional().default([]),\n\n // MCP servers support both:\n // - Array of file paths: [\"mcp-servers/server1.json\"]\n // - Object with inline configs: { \"server1\": { command: \"...\" } }\n mcpServers: z\n .union([z.array(z.string()), z.record(MCPServerConfigSchema)])\n .optional()\n .default([]),\n\n // Dependencies\n dependencies: z.record(z.string()).optional().default({}),\n\n // Minto/Claude Code version requirements\n engines: z\n .object({\n minto: z.string().optional(),\n 'claude-code': z.string().optional(),\n node: z.string().optional(),\n })\n .optional(),\n\n // Plugin-specific configuration schema\n configSchema: z.record(z.any()).optional(),\n})\n\nexport type PluginManifest = z.infer<typeof PluginManifestSchema>\n\n/**\n * Loaded Plugin (runtime representation)\n */\nexport interface LoadedPlugin {\n name: string\n manifest: PluginManifest\n location: string // Absolute path to plugin directory\n source: PluginSource\n agents: LoadedAgent[]\n commands: LoadedCommand[]\n skills: LoadedSkill[]\n hooks: LoadedHook[]\n mcpServers: LoadedMCPServer[]\n enabled: boolean\n config?: Record<string, any>\n}\n\n/**\n * Plugin source types\n */\nexport type PluginSource =\n | { type: 'local'; path: string }\n | { type: 'git'; repo: string; ref?: string }\n | { type: 'npm'; package: string; version?: string }\n | { type: 'marketplace'; marketplace: string; name: string }\n | { type: 'claude-code'; marketplace: string; name: string }\n\n/**\n * Agent component (from plugin)\n */\nexport interface LoadedAgent {\n name: string\n filePath: string\n config: {\n name: string\n description: string\n tools?: string | string[]\n model?: string\n content: string\n }\n pluginName: string\n}\n\n/**\n * Command component (from plugin)\n */\nexport interface LoadedCommand {\n name: string\n filePath: string\n config: {\n name: string\n description?: string\n aliases?: string[]\n enabled?: boolean\n hidden?: boolean\n progressMessage?: string\n argNames?: string[]\n 'allowed-tools'?: string[]\n content: string\n }\n pluginName: string\n}\n\n/**\n * Skill configuration (Claude Code specification compliant)\n */\nexport interface SkillConfig {\n // Identity\n name: string\n description: string\n argumentHint?: string // Hint for expected arguments\n\n // Invocation control (Claude Code spec)\n disableModelInvocation?: boolean // If true, model cannot auto-invoke (user-only)\n userInvocable?: boolean // If true, user can invoke via /command (default: true)\n\n // Execution configuration\n allowedTools?: string[] // Tools the skill can use\n model?: 'sonnet' | 'opus' | 'haiku' | 'inherit' // Model to use\n\n // Fork context (Claude Code spec)\n context?: 'fork' // Execute in isolated context\n agent?: string // Agent type to use for fork execution\n\n // Content\n content: string\n}\n\n/**\n * Skill component (from plugin or standalone)\n */\nexport interface LoadedSkill {\n name: string\n filePath: string\n config: SkillConfig\n pluginName: string // 'standalone' for standalone skills\n source: 'plugin' | 'user' | 'project' // Where the skill was loaded from\n}\n\n/**\n * Hook component (from plugin)\n */\nexport interface LoadedHook {\n name: string\n filePath: string\n config: {\n event: HookEvent\n matcher?: string\n type: 'command' | 'message' | 'notification'\n command?: string\n message?: string\n blocking?: boolean\n timeout?: number\n }\n pluginName: string\n event: HookEvent // Also store event at root level for easier access\n matcher?: string // Also store matcher at root level\n}\n\n/**\n * Hook lifecycle events\n */\nexport type HookEvent =\n | 'PreToolUse'\n | 'PostToolUse'\n | 'UserPromptSubmit'\n | 'SessionStart'\n | 'SessionEnd'\n | 'Stop'\n | 'SubagentStop'\n | 'Notification'\n | 'PreCompact'\n\n/**\n * MCP Server component (from plugin)\n */\nexport interface LoadedMCPServer {\n name: string\n filePath: string\n config: {\n command: string\n args?: string[]\n env?: Record<string, string>\n timeout?: number\n }\n pluginName: string\n}\n\n/**\n * Plugin installation options\n */\nexport interface PluginInstallOptions {\n source: PluginSource\n global?: boolean // Install globally vs project-level\n enable?: boolean // Auto-enable after install\n config?: Record<string, any> // Plugin-specific config\n}\n\n/**\n * Plugin marketplace configuration\n */\nexport interface PluginMarketplace {\n name: string\n source: PluginSource\n plugins: PluginManifest[]\n}\n\n/**\n * Plugin error types\n */\nexport class PluginError extends Error {\n constructor(\n message: string,\n public code: PluginErrorCode,\n public pluginName?: string,\n public details?: any,\n ) {\n super(message)\n this.name = 'PluginError'\n }\n}\n\nexport enum PluginErrorCode {\n MANIFEST_INVALID = 'MANIFEST_INVALID',\n MANIFEST_NOT_FOUND = 'MANIFEST_NOT_FOUND',\n VERSION_MISMATCH = 'VERSION_MISMATCH',\n DEPENDENCY_MISSING = 'DEPENDENCY_MISSING',\n COMPONENT_NOT_FOUND = 'COMPONENT_NOT_FOUND',\n COMPONENT_INVALID = 'COMPONENT_INVALID',\n ALREADY_INSTALLED = 'ALREADY_INSTALLED',\n NOT_INSTALLED = 'NOT_INSTALLED',\n PERMISSION_DENIED = 'PERMISSION_DENIED',\n}\n\n// ============================================================================\n// Session Plugin Types (Claude Code CLI Compatible)\n// ============================================================================\n\n/**\n * A loaded plugin in the current session\n * Compatible with Claude Code CLI session plugins\n */\nexport interface SessionPlugin {\n /** Plugin name from manifest */\n name: string\n /** Root directory of the plugin */\n rootDir: string\n /** Path to the manifest file */\n manifestPath: string\n /** Raw manifest data */\n manifest: unknown\n /** Directories containing custom commands */\n commandsDirs: string[]\n /** Directories containing skills */\n skillsDirs: string[]\n /** Directories containing agents */\n agentsDirs: string[]\n /** Paths to hooks configuration files */\n hooksFiles: string[]\n /** Directories containing output styles */\n outputStylesDirs: string[]\n /** Paths to MCP server configuration files */\n mcpConfigFiles: string[]\n /** Paths to LSP server configuration files (Claude Code CLI specific) */\n lspConfigFiles: string[]\n}\n\n/**\n * Plugin scope - where the plugin is installed/activated\n */\nexport type PluginScope = 'user' | 'project' | 'local'\n\n/**\n * Installed skill plugin record\n */\nexport interface InstalledSkillPlugin {\n plugin: string\n marketplace: string\n scope: PluginScope\n kind?: 'skill-pack' | 'plugin-pack'\n isEnabled?: boolean\n projectPath?: string\n installedAt: string\n pluginRoot?: string\n skills: string[]\n commands: string[]\n sourceMarketplacePath: string\n}\n"],
|
|
5
|
-
"mappings": "AAIA,SAAS,SAAS;AAKX,MAAM,eAAe,EAAE,MAAM;AAAA,EAClC,EAAE,OAAO;AAAA,EACT,EAAE,OAAO;AAAA,IACP,MAAM,EAAE,OAAO;AAAA,IACf,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,SAAS;AAAA,IACnC,KAAK,EAAE,OAAO,EAAE,IAAI,EAAE,SAAS;AAAA,EACjC,CAAC;AACH,CAAC;AAMM,MAAM,wBAAwB,EAAE,OAAO;AAAA;AAAA,EAE5C,SAAS,EAAE,OAAO,EAAE,SAAS;AAAA,EAC7B,MAAM,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;AAAA,EAC/C,KAAK,EAAE,OAAO,EAAE,OAAO,CAAC,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;AAAA,EAC/C,SAAS,EAAE,OAAO,EAAE,SAAS;AAAA;AAAA,EAG7B,MAAM,EAAE,KAAK,CAAC,SAAS,QAAQ,KAAK,CAAC,EAAE,SAAS,EAAE,QAAQ,OAAO;AAAA,EACjE,KAAK,EAAE,OAAO,EAAE,SAAS;AAAA,EACzB,SAAS,EAAE,OAAO,EAAE,OAAO,CAAC,EAAE,SAAS;AACzC,CAAC;AAQM,MAAM,uBAAuB,EAAE,OAAO;AAAA;AAAA,EAE3C,MAAM,EACH,OAAO,EACP,IAAI,CAAC,EACL;AAAA,IACC;AAAA,IACA;AAAA,EACF;AAAA,EACF,SAAS,EAAE,OAAO,EAAE,IAAI,CAAC;AAAA;AAAA,EACzB,aAAa,EAAE,OAAO,EAAE,IAAI,CAAC;AAAA;AAAA,EAG7B,aAAa,EAAE,OAAO,EAAE,SAAS;AAAA,EACjC,QAAQ,aAAa,SAAS;AAAA,EAC9B,UAAU,EAAE,OAAO,EAAE,IAAI,EAAE,SAAS;AAAA,EACpC,YAAY,EAAE,OAAO,EAAE,IAAI,EAAE,SAAS;AAAA,EACtC,SAAS,EAAE,OAAO,EAAE,SAAS;AAAA;AAAA,EAG7B,QAAQ,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;AAAA,EACjD,UAAU,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;AAAA,EACnD,QAAQ,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;AAAA,EACjD,OAAO,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;AAAA;AAAA;AAAA;AAAA,EAKhD,YAAY,EACT,MAAM,CAAC,EAAE,MAAM,EAAE,OAAO,CAAC,GAAG,EAAE,OAAO,qBAAqB,CAAC,CAAC,EAC5D,SAAS,EACT,QAAQ,CAAC,CAAC;AAAA;AAAA,EAGb,cAAc,EAAE,OAAO,EAAE,OAAO,CAAC,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;AAAA;AAAA,EAGxD,SAAS,EACN,OAAO;AAAA,IACN,OAAO,EAAE,OAAO,EAAE,SAAS;AAAA,IAC3B,eAAe,EAAE,OAAO,EAAE,SAAS;AAAA,IACnC,MAAM,EAAE,OAAO,EAAE,SAAS;AAAA,EAC5B,CAAC,EACA,SAAS;AAAA;AAAA,EAGZ,cAAc,EAAE,OAAO,EAAE,IAAI,CAAC,EAAE,SAAS;AAC3C,CAAC;
|
|
4
|
+
"sourcesContent": ["/**\n * Plugin System Type Definitions\n */\n\nimport { z } from 'zod'\n\n/**\n * Author information (can be string or object)\n */\nexport const AuthorSchema = z.union([\n z.string(),\n z.object({\n name: z.string(),\n email: z.string().email().optional(),\n url: z.string().url().optional(),\n }),\n])\n\n/**\n * MCP Server Configuration Schema\n * Compatible with Claude Code CLI's .mcp.json format\n */\nexport const MCPServerConfigSchema = z.object({\n // Command-based (stdio) servers\n command: z.string().optional(),\n args: z.array(z.string()).optional().default([]),\n env: z.record(z.string()).optional().default({}),\n timeout: z.number().optional(),\n\n // Transport type: stdio (default), http (Streamable HTTP), sse (legacy SSE)\n type: z.enum(['stdio', 'http', 'sse']).optional().default('stdio'),\n url: z.string().optional(),\n headers: z.record(z.string()).optional(),\n})\n\nexport type MCPServerConfig = z.infer<typeof MCPServerConfigSchema>\n\n/**\n * Plugin Manifest Schema (plugin.json)\n * This is the source of truth for plugin metadata\n */\nexport const PluginManifestSchema = z.object({\n // Required fields\n name: z\n .string()\n .min(1)\n .regex(\n /^[a-z0-9-]+$/,\n 'Plugin name must be lowercase alphanumeric with hyphens',\n ),\n version: z.string().min(1), // Supports semver and commit hashes (for CC sync)\n description: z.string().min(1),\n\n // Optional metadata\n displayName: z.string().optional(),\n author: AuthorSchema.optional(),\n homepage: z.string().url().optional(),\n repository: z.string().url().optional(),\n license: z.string().optional(),\n\n // Plugin components (relative paths from plugin root)\n agents: z.array(z.string()).optional().default([]),\n commands: z.array(z.string()).optional().default([]),\n skills: z.array(z.string()).optional().default([]),\n hooks: z.array(z.string()).optional().default([]),\n\n // MCP servers support both:\n // - Array of file paths: [\"mcp-servers/server1.json\"]\n // - Object with inline configs: { \"server1\": { command: \"...\" } }\n mcpServers: z\n .union([z.array(z.string()), z.record(MCPServerConfigSchema)])\n .optional()\n .default([]),\n\n // Dependencies\n dependencies: z.record(z.string()).optional().default({}),\n\n // Minto/Claude Code version requirements\n engines: z\n .object({\n minto: z.string().optional(),\n 'claude-code': z.string().optional(),\n node: z.string().optional(),\n })\n .optional(),\n\n // Plugin-specific configuration schema\n configSchema: z.record(z.any()).optional(),\n})\n\nexport type PluginManifest = z.infer<typeof PluginManifestSchema>\n\n/**\n * Loaded Plugin (runtime representation)\n */\nexport interface LoadedPlugin {\n name: string\n manifest: PluginManifest\n location: string // Absolute path to plugin directory\n source: PluginSource\n agents: LoadedAgent[]\n commands: LoadedCommand[]\n skills: LoadedSkill[]\n hooks: LoadedHook[]\n mcpServers: LoadedMCPServer[]\n lspServers: LoadedLSPServer[]\n enabled: boolean\n config?: Record<string, any>\n /** Plugin agent settings (CC format: { defaultAgent, maxTurns, ... }) */\n settings?: Record<string, any> | null\n}\n\n/**\n * Plugin source types\n */\nexport type PluginSource =\n | { type: 'local'; path: string }\n | { type: 'git'; repo: string; ref?: string }\n | { type: 'npm'; package: string; version?: string }\n | { type: 'marketplace'; marketplace: string; name: string }\n | { type: 'claude-code'; marketplace: string; name: string }\n\n/**\n * Agent component (from plugin)\n */\nexport interface LoadedAgent {\n name: string\n filePath: string\n config: {\n name: string\n description: string\n tools?: string | string[]\n model?: string\n content: string\n }\n pluginName: string\n}\n\n/**\n * Command component (from plugin)\n */\nexport interface LoadedCommand {\n name: string\n filePath: string\n config: {\n name: string\n description?: string\n aliases?: string[]\n enabled?: boolean\n hidden?: boolean\n progressMessage?: string\n argNames?: string[]\n 'allowed-tools'?: string[]\n content: string\n }\n pluginName: string\n}\n\n/**\n * Skill hook definition (inline in SKILL.md frontmatter)\n */\nexport interface SkillHookDefinition {\n event: HookEvent\n matcher?: string\n type?: 'command' | 'prompt' | 'agent'\n command?: string\n prompt?: string\n timeout?: number\n}\n\n/**\n * Skill configuration (Claude Code specification compliant)\n */\nexport interface SkillConfig {\n // Identity\n name: string\n description: string\n argumentHint?: string // Hint for expected arguments\n\n // Invocation control (Claude Code spec)\n disableModelInvocation?: boolean // If true, model cannot auto-invoke (user-only)\n userInvocable?: boolean // If true, user can invoke via /command (default: true)\n\n // Execution configuration\n allowedTools?: string[] // Tools the skill can use\n model?: 'sonnet' | 'opus' | 'haiku' | 'inherit' // Model to use\n\n // Fork context (Claude Code spec)\n context?: 'fork' // Execute in isolated context\n agent?: string // Agent type to use for fork execution\n\n // Lifecycle hooks (Claude Code spec)\n hooks?: SkillHookDefinition[]\n\n // Metadata (Claude Code spec standard fields)\n license?: string\n compatibility?: string | Record<string, string>\n metadata?: Record<string, unknown>\n\n // Content\n content: string\n}\n\n/**\n * Skill component (from plugin or standalone)\n */\nexport interface LoadedSkill {\n name: string\n filePath: string\n config: SkillConfig\n pluginName: string // 'standalone' for standalone skills\n source: 'plugin' | 'user' | 'project' // Where the skill was loaded from\n}\n\n/**\n * Hook component (from plugin)\n */\nexport interface LoadedHook {\n name: string\n filePath: string\n config: {\n event: HookEvent\n matcher?: string\n type: 'command' | 'prompt' | 'agent'\n command?: string\n prompt?: string\n blocking?: boolean\n timeout?: number\n async?: boolean\n once?: boolean\n /** Tool input field patterns for matching (CC spec) */\n toolInput?: Record<string, string>\n }\n pluginName: string\n event: HookEvent // Also store event at root level for easier access\n matcher?: string // Also store matcher at root level\n}\n\n/**\n * Hook lifecycle events\n */\nexport type HookEvent =\n | 'PreToolUse'\n | 'PostToolUse'\n | 'PostToolUseFailure'\n | 'PermissionRequest'\n | 'UserPromptSubmit'\n | 'SessionStart'\n | 'SessionEnd'\n | 'Stop'\n | 'SubagentStart'\n | 'SubagentStop'\n | 'Notification'\n | 'PreCompact'\n | 'PostCompact'\n | 'ConfigChange'\n | 'TaskCompleted'\n | 'TeammateIdle'\n\n/**\n * MCP Server component (from plugin)\n */\nexport interface LoadedMCPServer {\n name: string\n filePath: string\n config: {\n command: string\n args?: string[]\n env?: Record<string, string>\n timeout?: number\n }\n pluginName: string\n}\n\n/**\n * LSP Server component (from plugin)\n * Claude Code CLI compatible: plugins can define language servers\n * via .lsp.json or inline lspServers in plugin.json manifest.\n */\nexport interface LoadedLSPServer {\n name: string\n filePath: string\n config: {\n command: string\n args?: string[]\n env?: Record<string, string>\n /** File patterns this LSP server handles (e.g. [\"*.py\", \"*.pyi\"]) */\n filePatterns?: string[]\n /** Language identifiers (e.g. [\"python\", \"typescript\"]) */\n languages?: string[]\n /** Initialization options for the LSP server */\n initializationOptions?: Record<string, unknown>\n }\n pluginName: string\n}\n\n/**\n * Plugin installation options\n */\nexport interface PluginInstallOptions {\n source: PluginSource\n global?: boolean // Install globally vs project-level\n enable?: boolean // Auto-enable after install\n config?: Record<string, any> // Plugin-specific config\n}\n\n/**\n * Plugin marketplace configuration\n */\nexport interface PluginMarketplace {\n name: string\n source: PluginSource\n plugins: PluginManifest[]\n}\n\n/**\n * Plugin error types\n */\nexport class PluginError extends Error {\n constructor(\n message: string,\n public code: PluginErrorCode,\n public pluginName?: string,\n public details?: any,\n ) {\n super(message)\n this.name = 'PluginError'\n }\n}\n\nexport enum PluginErrorCode {\n MANIFEST_INVALID = 'MANIFEST_INVALID',\n MANIFEST_NOT_FOUND = 'MANIFEST_NOT_FOUND',\n VERSION_MISMATCH = 'VERSION_MISMATCH',\n DEPENDENCY_MISSING = 'DEPENDENCY_MISSING',\n COMPONENT_NOT_FOUND = 'COMPONENT_NOT_FOUND',\n COMPONENT_INVALID = 'COMPONENT_INVALID',\n ALREADY_INSTALLED = 'ALREADY_INSTALLED',\n NOT_INSTALLED = 'NOT_INSTALLED',\n PERMISSION_DENIED = 'PERMISSION_DENIED',\n}\n\n// ============================================================================\n// Session Plugin Types (Claude Code CLI Compatible)\n// ============================================================================\n\n/**\n * A loaded plugin in the current session\n * Compatible with Claude Code CLI session plugins\n */\nexport interface SessionPlugin {\n /** Plugin name from manifest */\n name: string\n /** Root directory of the plugin */\n rootDir: string\n /** Path to the manifest file */\n manifestPath: string\n /** Raw manifest data */\n manifest: unknown\n /** Directories containing custom commands */\n commandsDirs: string[]\n /** Directories containing skills */\n skillsDirs: string[]\n /** Directories containing agents */\n agentsDirs: string[]\n /** Paths to hooks configuration files */\n hooksFiles: string[]\n /** Directories containing output styles */\n outputStylesDirs: string[]\n /** Paths to MCP server configuration files */\n mcpConfigFiles: string[]\n /** Paths to LSP server configuration files (Claude Code CLI specific) */\n lspConfigFiles: string[]\n}\n\n/**\n * Plugin scope - where the plugin is installed/activated\n */\nexport type PluginScope = 'user' | 'project' | 'local'\n\n/**\n * Installed skill plugin record\n */\nexport interface InstalledSkillPlugin {\n plugin: string\n marketplace: string\n scope: PluginScope\n kind?: 'skill-pack' | 'plugin-pack'\n isEnabled?: boolean\n projectPath?: string\n installedAt: string\n pluginRoot?: string\n skills: string[]\n commands: string[]\n sourceMarketplacePath: string\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["PluginErrorCode"]
|
|
7
7
|
}
|
|
@@ -265,7 +265,12 @@ async function scanAgentDirectory(dirPath, location) {
|
|
|
265
265
|
...frontmatter.maxThinkingTokens && {
|
|
266
266
|
maxThinkingTokens: Number(frontmatter.maxThinkingTokens)
|
|
267
267
|
},
|
|
268
|
-
...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 }
|
|
269
274
|
};
|
|
270
275
|
agents.push(agent);
|
|
271
276
|
} catch (error) {
|
|
@@ -339,6 +344,13 @@ async function loadPluginAgents() {
|
|
|
339
344
|
},
|
|
340
345
|
...frontmatter.hooks && {
|
|
341
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
|
|
342
354
|
}
|
|
343
355
|
};
|
|
344
356
|
agents.push(agent);
|
|
@@ -381,7 +393,8 @@ async function loadAllAgents() {
|
|
|
381
393
|
agentMap.set(agent.agentType, agent);
|
|
382
394
|
}
|
|
383
395
|
for (const agent of pluginAgents) {
|
|
384
|
-
|
|
396
|
+
const namespacedType = agent.pluginName ? `${agent.pluginName}:${agent.agentType}` : agent.agentType;
|
|
397
|
+
agentMap.set(namespacedType, { ...agent, agentType: namespacedType });
|
|
385
398
|
}
|
|
386
399
|
for (const agent of userClaudeAgents) {
|
|
387
400
|
agentMap.set(agent.agentType, agent);
|
|
@@ -430,7 +443,16 @@ function clearAgentCache() {
|
|
|
430
443
|
const getAgentByType = memoize(
|
|
431
444
|
async (agentType) => {
|
|
432
445
|
const agents = await getActiveAgents();
|
|
433
|
-
|
|
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;
|
|
434
456
|
}
|
|
435
457
|
);
|
|
436
458
|
const getAvailableAgentTypes = memoize(async () => {
|
|
@@ -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 // 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 }\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 }\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 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 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\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 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": "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
|
}
|
|
@@ -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
|
}
|
|
@@ -1,3 +1,6 @@
|
|
|
1
|
+
import { existsSync, mkdirSync, appendFileSync } from "fs";
|
|
2
|
+
import { join } from "path";
|
|
3
|
+
import { homedir } from "os";
|
|
1
4
|
import { countTokens } from "./tokens.js";
|
|
2
5
|
import { getMessagesSetter } from "../messages.js";
|
|
3
6
|
import { getContext } from "../context.js";
|
|
@@ -10,6 +13,10 @@ import { addLineNumbers } from "./file.js";
|
|
|
10
13
|
import { getModelManager } from "./model.js";
|
|
11
14
|
import { getCompressionPrompt } from "../constants/compressionPrompts.js";
|
|
12
15
|
import { getCompressionMode } from "./compressionMode.js";
|
|
16
|
+
import { getHookManager } from "./hookManager.js";
|
|
17
|
+
import { emitReminderEvent } from "../services/systemReminder.js";
|
|
18
|
+
import { CONFIG_BASE_DIR } from "../constants/product.js";
|
|
19
|
+
import { getGlobalConfig } from "./config.js";
|
|
13
20
|
const AUTO_COMPACT_THRESHOLD_RATIO = 0.92;
|
|
14
21
|
async function getCompressionModelContextLimit() {
|
|
15
22
|
try {
|
|
@@ -44,6 +51,11 @@ async function checkAutoCompact(messages, toolUseContext) {
|
|
|
44
51
|
return { messages, wasCompacted: false };
|
|
45
52
|
}
|
|
46
53
|
try {
|
|
54
|
+
const hookManager = getHookManager();
|
|
55
|
+
if (hookManager) {
|
|
56
|
+
hookManager.executePreCompact("auto").catch(() => {
|
|
57
|
+
});
|
|
58
|
+
}
|
|
47
59
|
const compactedMessages = await executeAutoCompact(messages, toolUseContext);
|
|
48
60
|
return {
|
|
49
61
|
messages: compactedMessages,
|
|
@@ -57,6 +69,52 @@ async function checkAutoCompact(messages, toolUseContext) {
|
|
|
57
69
|
return { messages, wasCompacted: false };
|
|
58
70
|
}
|
|
59
71
|
}
|
|
72
|
+
const REMEMBER_PATTERNS = [
|
|
73
|
+
/\bremember\s+(this|that)\b/i,
|
|
74
|
+
/\balways\s+(use|do|prefer|avoid|skip)\b/i,
|
|
75
|
+
/\bnever\s+(use|do|commit|push|deploy)\b/i,
|
|
76
|
+
/\bfrom\s+now\s+on\b/i,
|
|
77
|
+
/\bin\s+the\s+future\b/i,
|
|
78
|
+
/\bgoing\s+forward\b/i,
|
|
79
|
+
/\bkeep\s+in\s+mind\b/i,
|
|
80
|
+
/\bdon'?t\s+forget\b/i
|
|
81
|
+
];
|
|
82
|
+
function extractAutoMemory(messages) {
|
|
83
|
+
try {
|
|
84
|
+
const config = getGlobalConfig();
|
|
85
|
+
if (!config.autoMemory) return;
|
|
86
|
+
const memories = [];
|
|
87
|
+
for (const msg of messages) {
|
|
88
|
+
if (msg.type !== "user") continue;
|
|
89
|
+
const content = msg.message.content;
|
|
90
|
+
const text = typeof content === "string" ? content : Array.isArray(content) ? content.filter((b) => b.type === "text").map((b) => b.text).join("\n") : "";
|
|
91
|
+
if (!text) continue;
|
|
92
|
+
const matchesPattern = REMEMBER_PATTERNS.some((p) => p.test(text));
|
|
93
|
+
if (matchesPattern) {
|
|
94
|
+
const lines = text.split("\n");
|
|
95
|
+
for (const line of lines) {
|
|
96
|
+
const trimmed = line.trim();
|
|
97
|
+
if (trimmed && REMEMBER_PATTERNS.some((p) => p.test(trimmed))) {
|
|
98
|
+
memories.push(trimmed);
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
if (memories.length === 0) return;
|
|
104
|
+
const autoMemoryDir = join(homedir(), CONFIG_BASE_DIR);
|
|
105
|
+
const autoMemoryPath = join(autoMemoryDir, "auto-memory.md");
|
|
106
|
+
if (!existsSync(autoMemoryDir)) {
|
|
107
|
+
mkdirSync(autoMemoryDir, { recursive: true });
|
|
108
|
+
}
|
|
109
|
+
const timestamp = (/* @__PURE__ */ new Date()).toISOString().split("T")[0];
|
|
110
|
+
const entry = `
|
|
111
|
+
## ${timestamp}
|
|
112
|
+
${memories.map((m) => `- ${m}`).join("\n")}
|
|
113
|
+
`;
|
|
114
|
+
appendFileSync(autoMemoryPath, entry, "utf-8");
|
|
115
|
+
} catch {
|
|
116
|
+
}
|
|
117
|
+
}
|
|
60
118
|
async function executeAutoCompact(messages, toolUseContext) {
|
|
61
119
|
const compressionMode = getCompressionMode();
|
|
62
120
|
const compressionPrompt = getCompressionPrompt(compressionMode);
|
|
@@ -112,8 +170,10 @@ ${contentWithLines}
|
|
|
112
170
|
*Automatically recovered (${file.tokens} tokens)${file.truncated ? " [truncated]" : ""}*`
|
|
113
171
|
);
|
|
114
172
|
compactedMessages.push(recoveryMessage);
|
|
173
|
+
emitReminderEvent("compact:file_reference", { filePath: file.path });
|
|
115
174
|
}
|
|
116
175
|
}
|
|
176
|
+
extractAutoMemory(messages);
|
|
117
177
|
getMessagesSetter()([]);
|
|
118
178
|
getContext.cache.clear?.();
|
|
119
179
|
getCodeStyle.cache.clear?.();
|