@octo-cyber/ai 0.5.4 → 0.5.6
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/shared/IClaudeCodeSessionManager.d.ts +151 -0
- package/dist/shared/IClaudeCodeSessionManager.d.ts.map +1 -0
- package/dist/shared/IClaudeCodeSessionManager.js +6 -0
- package/dist/shared/IClaudeCodeSessionManager.js.map +1 -0
- package/dist/src/BranchDecisionEngine.d.ts +73 -0
- package/dist/src/BranchDecisionEngine.d.ts.map +1 -0
- package/dist/src/BranchDecisionEngine.js +195 -0
- package/dist/src/BranchDecisionEngine.js.map +1 -0
- package/dist/src/agent/agent-approval.controller.d.ts +20 -0
- package/dist/src/agent/agent-approval.controller.d.ts.map +1 -0
- package/dist/src/agent/agent-approval.controller.js +101 -0
- package/dist/src/agent/agent-approval.controller.js.map +1 -0
- package/dist/src/agent/agent-approval.entity.d.ts +15 -0
- package/dist/src/agent/agent-approval.entity.d.ts.map +1 -0
- package/dist/src/agent/agent-approval.entity.js +63 -0
- package/dist/src/agent/agent-approval.entity.js.map +1 -0
- package/dist/src/agent/agent-approval.service.d.ts +35 -0
- package/dist/src/agent/agent-approval.service.d.ts.map +1 -0
- package/dist/src/agent/agent-approval.service.js +109 -0
- package/dist/src/agent/agent-approval.service.js.map +1 -0
- package/dist/src/agent/agent-gateway.controller.d.ts +30 -0
- package/dist/src/agent/agent-gateway.controller.d.ts.map +1 -0
- package/dist/src/agent/agent-gateway.controller.js +112 -0
- package/dist/src/agent/agent-gateway.controller.js.map +1 -0
- package/dist/src/agent/agent-gateway.service.d.ts +33 -0
- package/dist/src/agent/agent-gateway.service.d.ts.map +1 -0
- package/dist/src/agent/agent-gateway.service.js +84 -0
- package/dist/src/agent/agent-gateway.service.js.map +1 -0
- package/dist/src/agent/agent-registry.service.d.ts +30 -0
- package/dist/src/agent/agent-registry.service.d.ts.map +1 -0
- package/dist/src/agent/agent-registry.service.js +45 -0
- package/dist/src/agent/agent-registry.service.js.map +1 -0
- package/dist/src/agent/agent-run.controller.d.ts +42 -0
- package/dist/src/agent/agent-run.controller.d.ts.map +1 -0
- package/dist/src/agent/agent-run.controller.js +195 -0
- package/dist/src/agent/agent-run.controller.js.map +1 -0
- package/dist/src/agent/agent-run.entity.d.ts +35 -0
- package/dist/src/agent/agent-run.entity.d.ts.map +1 -0
- package/dist/src/agent/agent-run.entity.js +98 -0
- package/dist/src/agent/agent-run.entity.js.map +1 -0
- package/dist/src/agent/agent-run.service.d.ts +48 -0
- package/dist/src/agent/agent-run.service.d.ts.map +1 -0
- package/dist/src/agent/agent-run.service.js +155 -0
- package/dist/src/agent/agent-run.service.js.map +1 -0
- package/dist/src/agent/agent-session.controller.d.ts +20 -0
- package/dist/src/agent/agent-session.controller.d.ts.map +1 -0
- package/dist/src/agent/agent-session.controller.js +97 -0
- package/dist/src/agent/agent-session.controller.js.map +1 -0
- package/dist/src/agent/agent-session.entity.d.ts +13 -0
- package/dist/src/agent/agent-session.entity.d.ts.map +1 -0
- package/dist/src/agent/agent-session.entity.js +57 -0
- package/dist/src/agent/agent-session.entity.js.map +1 -0
- package/dist/src/agent/agent-session.service.d.ts +38 -0
- package/dist/src/agent/agent-session.service.d.ts.map +1 -0
- package/dist/src/agent/agent-session.service.js +112 -0
- package/dist/src/agent/agent-session.service.js.map +1 -0
- package/dist/src/agent/coding-tools.d.ts +7 -0
- package/dist/src/agent/coding-tools.d.ts.map +1 -0
- package/dist/src/agent/coding-tools.js +181 -0
- package/dist/src/agent/coding-tools.js.map +1 -0
- package/dist/src/agent/computer-use-tools.d.ts +15 -0
- package/dist/src/agent/computer-use-tools.d.ts.map +1 -0
- package/dist/src/agent/computer-use-tools.js +256 -0
- package/dist/src/agent/computer-use-tools.js.map +1 -0
- package/dist/src/agent/external-agent.controller.d.ts +41 -0
- package/dist/src/agent/external-agent.controller.d.ts.map +1 -0
- package/dist/src/agent/external-agent.controller.js +129 -0
- package/dist/src/agent/external-agent.controller.js.map +1 -0
- package/dist/src/agent/external-agent.entity.d.ts +22 -0
- package/dist/src/agent/external-agent.entity.d.ts.map +1 -0
- package/dist/src/agent/external-agent.entity.js +75 -0
- package/dist/src/agent/external-agent.entity.js.map +1 -0
- package/dist/src/agent/external-agent.service.d.ts +60 -0
- package/dist/src/agent/external-agent.service.d.ts.map +1 -0
- package/dist/src/agent/external-agent.service.js +143 -0
- package/dist/src/agent/external-agent.service.js.map +1 -0
- package/dist/src/ai.module.d.ts +22 -0
- package/dist/src/ai.module.d.ts.map +1 -0
- package/dist/src/ai.module.js +178 -0
- package/dist/src/ai.module.js.map +1 -0
- package/dist/{cli → src/cli}/adapters/antigravity.adapter.d.ts +2 -0
- package/dist/src/cli/adapters/antigravity.adapter.d.ts.map +1 -0
- package/dist/{cli → src/cli}/adapters/antigravity.adapter.js +78 -0
- package/dist/src/cli/adapters/antigravity.adapter.js.map +1 -0
- package/dist/src/cli/adapters/claude-cli.adapter.d.ts.map +1 -0
- package/dist/{cli → src/cli}/adapters/claude-cli.adapter.js +2 -0
- package/dist/src/cli/adapters/claude-cli.adapter.js.map +1 -0
- package/dist/src/cli/adapters/cli-proxy-api.adapter.d.ts +32 -0
- package/dist/src/cli/adapters/cli-proxy-api.adapter.d.ts.map +1 -0
- package/dist/src/cli/adapters/cli-proxy-api.adapter.js +239 -0
- package/dist/src/cli/adapters/cli-proxy-api.adapter.js.map +1 -0
- package/dist/src/cli/adapters/codex-cli.adapter.d.ts.map +1 -0
- package/dist/src/cli/adapters/codex-cli.adapter.js.map +1 -0
- package/dist/src/cli/adapters/detect-cli.d.ts.map +1 -0
- package/dist/src/cli/adapters/detect-cli.js.map +1 -0
- package/dist/src/cli/adapters/gemini-cli.adapter.d.ts.map +1 -0
- package/dist/src/cli/adapters/gemini-cli.adapter.js.map +1 -0
- package/dist/src/cli/adapters/index.d.ts.map +1 -0
- package/dist/src/cli/adapters/index.js.map +1 -0
- package/dist/{cli → src/cli}/cli-executor.service.d.ts +2 -0
- package/dist/src/cli/cli-executor.service.d.ts.map +1 -0
- package/dist/{cli → src/cli}/cli-executor.service.js +21 -1
- package/dist/src/cli/cli-executor.service.js.map +1 -0
- package/dist/src/cli/cli-registry.service.d.ts.map +1 -0
- package/dist/{cli → src/cli}/cli-registry.service.js +3 -1
- package/dist/src/cli/cli-registry.service.js.map +1 -0
- package/dist/{cli → src/cli}/controllers/ai-cli.controller.d.ts +4 -0
- package/dist/src/cli/controllers/ai-cli.controller.d.ts.map +1 -0
- package/dist/{cli → src/cli}/controllers/ai-cli.controller.js +33 -4
- package/dist/src/cli/controllers/ai-cli.controller.js.map +1 -0
- package/dist/{cli → src/cli}/index.d.ts +1 -1
- package/dist/src/cli/index.d.ts.map +1 -0
- package/dist/src/cli/index.js.map +1 -0
- package/dist/{cli → src/cli}/types.d.ts +23 -1
- package/dist/src/cli/types.d.ts.map +1 -0
- package/dist/{skill → src/cli}/types.js.map +1 -1
- package/dist/{cli → src/cli}/workspace.service.d.ts +11 -3
- package/dist/src/cli/workspace.service.d.ts.map +1 -0
- package/dist/{cli → src/cli}/workspace.service.js +61 -6
- package/dist/src/cli/workspace.service.js.map +1 -0
- package/dist/src/computer-use/computer-use.controller.d.ts +23 -0
- package/dist/src/computer-use/computer-use.controller.d.ts.map +1 -0
- package/dist/src/computer-use/computer-use.controller.js +87 -0
- package/dist/src/computer-use/computer-use.controller.js.map +1 -0
- package/dist/src/index.d.ts +12 -0
- package/dist/src/index.d.ts.map +1 -0
- package/dist/src/index.js +51 -0
- package/dist/src/index.js.map +1 -0
- package/dist/src/orchestrator/orchestrator-plan.entity.d.ts +31 -0
- package/dist/src/orchestrator/orchestrator-plan.entity.d.ts.map +1 -0
- package/dist/src/orchestrator/orchestrator-plan.entity.js +69 -0
- package/dist/src/orchestrator/orchestrator-plan.entity.js.map +1 -0
- package/dist/src/orchestrator/orchestrator-plan.service.d.ts +27 -0
- package/dist/src/orchestrator/orchestrator-plan.service.d.ts.map +1 -0
- package/dist/src/orchestrator/orchestrator-plan.service.js +201 -0
- package/dist/src/orchestrator/orchestrator-plan.service.js.map +1 -0
- package/dist/src/orchestrator/orchestrator.controller.d.ts +29 -0
- package/dist/src/orchestrator/orchestrator.controller.d.ts.map +1 -0
- package/dist/src/orchestrator/orchestrator.controller.js +144 -0
- package/dist/src/orchestrator/orchestrator.controller.js.map +1 -0
- package/dist/src/orchestrator/workflow-feedback.entity.d.ts +12 -0
- package/dist/src/orchestrator/workflow-feedback.entity.d.ts.map +1 -0
- package/dist/src/orchestrator/workflow-feedback.entity.js +49 -0
- package/dist/src/orchestrator/workflow-feedback.entity.js.map +1 -0
- package/dist/src/orchestrator/workflow-feedback.service.d.ts +14 -0
- package/dist/src/orchestrator/workflow-feedback.service.d.ts.map +1 -0
- package/dist/src/orchestrator/workflow-feedback.service.js +51 -0
- package/dist/src/orchestrator/workflow-feedback.service.js.map +1 -0
- package/dist/src/session/claude-code-session-manager.service.d.ts +28 -0
- package/dist/src/session/claude-code-session-manager.service.d.ts.map +1 -0
- package/dist/src/session/claude-code-session-manager.service.js +177 -0
- package/dist/src/session/claude-code-session-manager.service.js.map +1 -0
- package/dist/src/skill/controllers/skill-market.controller.d.ts +18 -0
- package/dist/src/skill/controllers/skill-market.controller.d.ts.map +1 -0
- package/dist/src/skill/controllers/skill-market.controller.js +128 -0
- package/dist/src/skill/controllers/skill-market.controller.js.map +1 -0
- package/dist/{skill → src/skill}/controllers/skill.controller.d.ts +4 -0
- package/dist/src/skill/controllers/skill.controller.d.ts.map +1 -0
- package/dist/{skill → src/skill}/controllers/skill.controller.js +31 -0
- package/dist/src/skill/controllers/skill.controller.js.map +1 -0
- package/dist/src/skill/entities/octo-skill.entity.d.ts.map +1 -0
- package/dist/src/skill/entities/octo-skill.entity.js.map +1 -0
- package/dist/{skill → src/skill}/index.d.ts +3 -1
- package/dist/src/skill/index.d.ts.map +1 -0
- package/dist/{skill → src/skill}/index.js +5 -1
- package/dist/src/skill/index.js.map +1 -0
- package/dist/src/skill/services/skill-generator.service.d.ts.map +1 -0
- package/dist/src/skill/services/skill-generator.service.js.map +1 -0
- package/dist/{skill → src/skill}/services/skill-installer.service.d.ts +27 -0
- package/dist/src/skill/services/skill-installer.service.d.ts.map +1 -0
- package/dist/{skill → src/skill}/services/skill-installer.service.js +108 -0
- package/dist/src/skill/services/skill-installer.service.js.map +1 -0
- package/dist/src/skill/services/skill-market-helpers.d.ts +25 -0
- package/dist/src/skill/services/skill-market-helpers.d.ts.map +1 -0
- package/dist/src/skill/services/skill-market-helpers.js +22 -0
- package/dist/src/skill/services/skill-market-helpers.js.map +1 -0
- package/dist/src/skill/services/skill-market.service.d.ts +60 -0
- package/dist/src/skill/services/skill-market.service.d.ts.map +1 -0
- package/dist/src/skill/services/skill-market.service.js +228 -0
- package/dist/src/skill/services/skill-market.service.js.map +1 -0
- package/dist/{skill → src/skill}/services/skill-registry.service.d.ts +1 -0
- package/dist/src/skill/services/skill-registry.service.d.ts.map +1 -0
- package/dist/{skill → src/skill}/services/skill-registry.service.js +18 -6
- package/dist/src/skill/services/skill-registry.service.js.map +1 -0
- package/dist/{skill → src/skill}/types.d.ts +32 -1
- package/dist/src/skill/types.d.ts.map +1 -0
- package/dist/{cli → src/skill}/types.js.map +1 -1
- package/dist/src/workflow-design/DocumentExtractorAgent.d.ts +47 -0
- package/dist/src/workflow-design/DocumentExtractorAgent.d.ts.map +1 -0
- package/dist/src/workflow-design/DocumentExtractorAgent.js +118 -0
- package/dist/src/workflow-design/DocumentExtractorAgent.js.map +1 -0
- package/dist/src/workflow-design/N8nNodeKnowledgeBase.d.ts +26 -0
- package/dist/src/workflow-design/N8nNodeKnowledgeBase.d.ts.map +1 -0
- package/dist/src/workflow-design/N8nNodeKnowledgeBase.js +362 -0
- package/dist/src/workflow-design/N8nNodeKnowledgeBase.js.map +1 -0
- package/dist/src/workflow-design/OctoAgentClient.d.ts +42 -0
- package/dist/src/workflow-design/OctoAgentClient.d.ts.map +1 -0
- package/dist/src/workflow-design/OctoAgentClient.js +191 -0
- package/dist/src/workflow-design/OctoAgentClient.js.map +1 -0
- package/dist/src/workflow-design/WorkflowDesignOrchestrator.d.ts +45 -0
- package/dist/src/workflow-design/WorkflowDesignOrchestrator.d.ts.map +1 -0
- package/dist/src/workflow-design/WorkflowDesignOrchestrator.js +265 -0
- package/dist/src/workflow-design/WorkflowDesignOrchestrator.js.map +1 -0
- package/dist/src/workflow-design/agents/ActionChainAgent.d.ts +9 -0
- package/dist/src/workflow-design/agents/ActionChainAgent.d.ts.map +1 -0
- package/dist/src/workflow-design/agents/ActionChainAgent.js +79 -0
- package/dist/src/workflow-design/agents/ActionChainAgent.js.map +1 -0
- package/dist/src/workflow-design/agents/CodeScriptAgent.d.ts +15 -0
- package/dist/src/workflow-design/agents/CodeScriptAgent.d.ts.map +1 -0
- package/dist/src/workflow-design/agents/CodeScriptAgent.js +96 -0
- package/dist/src/workflow-design/agents/CodeScriptAgent.js.map +1 -0
- package/dist/src/workflow-design/agents/LogicFlowAgent.d.ts +9 -0
- package/dist/src/workflow-design/agents/LogicFlowAgent.d.ts.map +1 -0
- package/dist/src/workflow-design/agents/LogicFlowAgent.js +78 -0
- package/dist/src/workflow-design/agents/LogicFlowAgent.js.map +1 -0
- package/dist/src/workflow-design/agents/TriggerAnalystAgent.d.ts +9 -0
- package/dist/src/workflow-design/agents/TriggerAnalystAgent.d.ts.map +1 -0
- package/dist/src/workflow-design/agents/TriggerAnalystAgent.js +65 -0
- package/dist/src/workflow-design/agents/TriggerAnalystAgent.js.map +1 -0
- package/dist/src/workflow-design/agents/WorkflowValidatorAgent.d.ts +9 -0
- package/dist/src/workflow-design/agents/WorkflowValidatorAgent.d.ts.map +1 -0
- package/dist/src/workflow-design/agents/WorkflowValidatorAgent.js +117 -0
- package/dist/src/workflow-design/agents/WorkflowValidatorAgent.js.map +1 -0
- package/dist/src/workflow-design/index.d.ts +12 -0
- package/dist/src/workflow-design/index.d.ts.map +1 -0
- package/dist/src/workflow-design/index.js +23 -0
- package/dist/src/workflow-design/index.js.map +1 -0
- package/dist/src/workflow-design/types.d.ts +155 -0
- package/dist/src/workflow-design/types.d.ts.map +1 -0
- package/dist/src/workflow-design/types.js +6 -0
- package/dist/src/workflow-design/types.js.map +1 -0
- package/package.json +21 -15
- package/web/components/CreateWorkspaceDialog.tsx +157 -0
- package/web/components/DirBrowser.tsx +85 -0
- package/web/components/GithubTokenDialog.tsx +152 -0
- package/web/components/LocalPathSection.tsx +91 -0
- package/web/components/MarketInstallDialog.tsx +196 -0
- package/web/components/MarketSkillCard.tsx +74 -0
- package/web/components/RepoList.tsx +97 -0
- package/web/components/SearchResultsGrid.tsx +55 -0
- package/web/components/SkillCard.tsx +4 -1
- package/web/components/ToolCard.tsx +88 -0
- package/web/components/WorkspaceCard.tsx +146 -0
- package/web/components/WorkspaceItem.tsx +65 -0
- package/web/components/WorkspaceSelect.tsx +94 -0
- package/web/index.ts +18 -1
- package/web/manifest.ts +7 -0
- package/web/messages/en-US.json +231 -7
- package/web/messages/zh-CN.json +234 -7
- package/web/pages/AgentMarketPage.tsx +318 -0
- package/web/pages/AgentRunsPage.tsx +162 -0
- package/web/pages/AgentsPage.tsx +239 -0
- package/web/pages/AiCliToolsPage.tsx +17 -128
- package/web/pages/ApprovalsPage.tsx +260 -0
- package/web/pages/OrchestratorPage.tsx +384 -0
- package/web/pages/SessionsPage.tsx +248 -0
- package/web/pages/SkillMarketPage.tsx +25 -0
- package/web/pages/WorkspaceSection.tsx +13 -113
- package/web/pages/WorkspacesPage.tsx +29 -254
- package/web/services/agent-approval.service.ts +35 -0
- package/web/services/agent-gateway.service.ts +41 -0
- package/web/services/agent-run.service.ts +48 -0
- package/web/services/agent-session.service.ts +30 -0
- package/web/services/ai-cli-service.ts +33 -0
- package/web/services/external-agent.service.ts +38 -0
- package/web/services/orchestrator.service.ts +52 -0
- package/web/services/skill-market-service.ts +88 -0
- package/web/services/skill-service.ts +12 -0
- package/dist/ai.module.d.ts +0 -4
- package/dist/ai.module.d.ts.map +0 -1
- package/dist/ai.module.js +0 -59
- package/dist/ai.module.js.map +0 -1
- package/dist/cli/adapters/antigravity.adapter.d.ts.map +0 -1
- package/dist/cli/adapters/antigravity.adapter.js.map +0 -1
- package/dist/cli/adapters/claude-cli.adapter.d.ts.map +0 -1
- package/dist/cli/adapters/claude-cli.adapter.js.map +0 -1
- package/dist/cli/adapters/codex-cli.adapter.d.ts.map +0 -1
- package/dist/cli/adapters/codex-cli.adapter.js.map +0 -1
- package/dist/cli/adapters/detect-cli.d.ts.map +0 -1
- package/dist/cli/adapters/detect-cli.js.map +0 -1
- package/dist/cli/adapters/gemini-cli.adapter.d.ts.map +0 -1
- package/dist/cli/adapters/gemini-cli.adapter.js.map +0 -1
- package/dist/cli/adapters/index.d.ts.map +0 -1
- package/dist/cli/adapters/index.js.map +0 -1
- package/dist/cli/cli-executor.service.d.ts.map +0 -1
- package/dist/cli/cli-executor.service.js.map +0 -1
- package/dist/cli/cli-registry.service.d.ts.map +0 -1
- package/dist/cli/cli-registry.service.js.map +0 -1
- package/dist/cli/controllers/ai-cli.controller.d.ts.map +0 -1
- package/dist/cli/controllers/ai-cli.controller.js.map +0 -1
- package/dist/cli/index.d.ts.map +0 -1
- package/dist/cli/index.js.map +0 -1
- package/dist/cli/types.d.ts.map +0 -1
- package/dist/cli/workspace.service.d.ts.map +0 -1
- package/dist/cli/workspace.service.js.map +0 -1
- package/dist/index.d.ts +0 -5
- package/dist/index.d.ts.map +0 -1
- package/dist/index.js +0 -23
- package/dist/index.js.map +0 -1
- package/dist/skill/controllers/skill.controller.d.ts.map +0 -1
- package/dist/skill/controllers/skill.controller.js.map +0 -1
- package/dist/skill/entities/octo-skill.entity.d.ts.map +0 -1
- package/dist/skill/entities/octo-skill.entity.js.map +0 -1
- package/dist/skill/index.d.ts.map +0 -1
- package/dist/skill/index.js.map +0 -1
- package/dist/skill/services/skill-generator.service.d.ts.map +0 -1
- package/dist/skill/services/skill-generator.service.js.map +0 -1
- package/dist/skill/services/skill-installer.service.d.ts.map +0 -1
- package/dist/skill/services/skill-installer.service.js.map +0 -1
- package/dist/skill/services/skill-registry.service.d.ts.map +0 -1
- package/dist/skill/services/skill-registry.service.js.map +0 -1
- package/dist/skill/types.d.ts.map +0 -1
- /package/dist/{cli → src/cli}/adapters/claude-cli.adapter.d.ts +0 -0
- /package/dist/{cli → src/cli}/adapters/codex-cli.adapter.d.ts +0 -0
- /package/dist/{cli → src/cli}/adapters/codex-cli.adapter.js +0 -0
- /package/dist/{cli → src/cli}/adapters/detect-cli.d.ts +0 -0
- /package/dist/{cli → src/cli}/adapters/detect-cli.js +0 -0
- /package/dist/{cli → src/cli}/adapters/gemini-cli.adapter.d.ts +0 -0
- /package/dist/{cli → src/cli}/adapters/gemini-cli.adapter.js +0 -0
- /package/dist/{cli → src/cli}/adapters/index.d.ts +0 -0
- /package/dist/{cli → src/cli}/adapters/index.js +0 -0
- /package/dist/{cli → src/cli}/cli-registry.service.d.ts +0 -0
- /package/dist/{cli → src/cli}/index.js +0 -0
- /package/dist/{cli → src/cli}/types.js +0 -0
- /package/dist/{skill → src/skill}/entities/octo-skill.entity.d.ts +0 -0
- /package/dist/{skill → src/skill}/entities/octo-skill.entity.js +0 -0
- /package/dist/{skill → src/skill}/services/skill-generator.service.d.ts +0 -0
- /package/dist/{skill → src/skill}/services/skill-generator.service.js +0 -0
- /package/dist/{skill → src/skill}/types.js +0 -0
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { useState, useCallback, useEffect } from 'react'
|
|
4
|
+
import { useTranslations } from 'next-intl'
|
|
5
|
+
import { toast } from 'sonner'
|
|
6
|
+
import { Loader2, FolderOpen } from 'lucide-react'
|
|
7
|
+
|
|
8
|
+
import { Button } from '@octo-cyber/ui/components/ui/button'
|
|
9
|
+
import { Checkbox } from '@octo-cyber/ui/components/ui/checkbox'
|
|
10
|
+
import {
|
|
11
|
+
Dialog,
|
|
12
|
+
DialogContent,
|
|
13
|
+
DialogDescription,
|
|
14
|
+
DialogFooter,
|
|
15
|
+
DialogHeader,
|
|
16
|
+
DialogTitle,
|
|
17
|
+
} from '@octo-cyber/ui/components/ui/dialog'
|
|
18
|
+
import { Label } from '@octo-cyber/ui/components/ui/label'
|
|
19
|
+
import {
|
|
20
|
+
Select,
|
|
21
|
+
SelectContent,
|
|
22
|
+
SelectItem,
|
|
23
|
+
SelectTrigger,
|
|
24
|
+
SelectValue,
|
|
25
|
+
} from '@octo-cyber/ui/components/ui/select'
|
|
26
|
+
|
|
27
|
+
import { skillMarketService } from '../services/skill-market-service'
|
|
28
|
+
import { aiCliService } from '../services/ai-cli-service'
|
|
29
|
+
import type { MarketSkillInfo, SkillTarget } from '../services/skill-market-service'
|
|
30
|
+
|
|
31
|
+
const AVAILABLE_TARGETS: { id: SkillTarget; label: string }[] = [
|
|
32
|
+
{ id: 'claude', label: 'Claude Code' },
|
|
33
|
+
{ id: 'cursor', label: 'Cursor' },
|
|
34
|
+
{ id: 'codex', label: 'Codex' },
|
|
35
|
+
{ id: 'gemini', label: 'Gemini CLI' },
|
|
36
|
+
]
|
|
37
|
+
|
|
38
|
+
interface WorkspaceOption {
|
|
39
|
+
id: string
|
|
40
|
+
name: string
|
|
41
|
+
path: string
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
interface MarketInstallDialogProps {
|
|
45
|
+
skill: MarketSkillInfo | null
|
|
46
|
+
open: boolean
|
|
47
|
+
onOpenChange: (open: boolean) => void
|
|
48
|
+
onDone: () => void
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
export default function MarketInstallDialog({
|
|
52
|
+
skill,
|
|
53
|
+
open,
|
|
54
|
+
onOpenChange,
|
|
55
|
+
onDone,
|
|
56
|
+
}: MarketInstallDialogProps) {
|
|
57
|
+
const t = useTranslations('aiSkillMarket')
|
|
58
|
+
const tc = useTranslations('common')
|
|
59
|
+
|
|
60
|
+
const [selected, setSelected] = useState<SkillTarget[]>([])
|
|
61
|
+
const [installing, setInstalling] = useState(false)
|
|
62
|
+
const [installScope, setInstallScope] = useState<'user' | 'workspace'>('user')
|
|
63
|
+
const [workspaces, setWorkspaces] = useState<WorkspaceOption[]>([])
|
|
64
|
+
const [selectedWorkspace, setSelectedWorkspace] = useState<string>('')
|
|
65
|
+
|
|
66
|
+
useEffect(() => {
|
|
67
|
+
if (open && installScope === 'workspace' && workspaces.length === 0) {
|
|
68
|
+
aiCliService.getWorkspaces().then((list) => {
|
|
69
|
+
setWorkspaces(list.map((w) => ({ id: w.id, name: w.name, path: w.path })))
|
|
70
|
+
}).catch(() => { /* ignore */ })
|
|
71
|
+
}
|
|
72
|
+
}, [open, installScope, workspaces.length])
|
|
73
|
+
|
|
74
|
+
const toggleTarget = useCallback((id: SkillTarget) => {
|
|
75
|
+
setSelected((prev) =>
|
|
76
|
+
prev.includes(id) ? prev.filter((x) => x !== id) : [...prev, id],
|
|
77
|
+
)
|
|
78
|
+
}, [])
|
|
79
|
+
|
|
80
|
+
const handleInstall = useCallback(async () => {
|
|
81
|
+
if (!skill || selected.length === 0) return
|
|
82
|
+
if (installScope === 'workspace' && !selectedWorkspace) {
|
|
83
|
+
toast.error(t('selectWorkspace'))
|
|
84
|
+
return
|
|
85
|
+
}
|
|
86
|
+
setInstalling(true)
|
|
87
|
+
try {
|
|
88
|
+
const params: Parameters<typeof skillMarketService.install>[0] = {
|
|
89
|
+
downloadUrl: skill.downloadUrl,
|
|
90
|
+
name: skill.name,
|
|
91
|
+
targets: selected,
|
|
92
|
+
}
|
|
93
|
+
if (installScope === 'workspace') {
|
|
94
|
+
params.workspacePath = selectedWorkspace
|
|
95
|
+
}
|
|
96
|
+
const result = await skillMarketService.install(params)
|
|
97
|
+
if (result.failed.length > 0) {
|
|
98
|
+
const failedNames = result.failed.map((f) => f.target).join(', ')
|
|
99
|
+
toast.error(`${t('installFailed')}: ${failedNames}`)
|
|
100
|
+
} else {
|
|
101
|
+
toast.success(t('installSuccess'))
|
|
102
|
+
}
|
|
103
|
+
onDone()
|
|
104
|
+
} catch {
|
|
105
|
+
toast.error(t('installFailed'))
|
|
106
|
+
} finally {
|
|
107
|
+
setInstalling(false)
|
|
108
|
+
}
|
|
109
|
+
}, [skill, selected, installScope, selectedWorkspace, t, onDone])
|
|
110
|
+
|
|
111
|
+
return (
|
|
112
|
+
<Dialog open={open} onOpenChange={onOpenChange}>
|
|
113
|
+
<DialogContent className="sm:max-w-md">
|
|
114
|
+
<DialogHeader>
|
|
115
|
+
<DialogTitle>{t('installTitle')}</DialogTitle>
|
|
116
|
+
<DialogDescription>
|
|
117
|
+
{t('installDesc')}
|
|
118
|
+
</DialogDescription>
|
|
119
|
+
</DialogHeader>
|
|
120
|
+
|
|
121
|
+
<div className="space-y-3 py-4">
|
|
122
|
+
<p className="text-sm font-medium">{t('selectTargets')}</p>
|
|
123
|
+
{AVAILABLE_TARGETS.map((tgt) => (
|
|
124
|
+
<div key={tgt.id} className="flex items-center space-x-3">
|
|
125
|
+
<Checkbox
|
|
126
|
+
id={`market-target-${tgt.id}`}
|
|
127
|
+
checked={selected.includes(tgt.id)}
|
|
128
|
+
onCheckedChange={() => toggleTarget(tgt.id)}
|
|
129
|
+
/>
|
|
130
|
+
<Label htmlFor={`market-target-${tgt.id}`} className="text-sm">
|
|
131
|
+
{tgt.label}
|
|
132
|
+
</Label>
|
|
133
|
+
</div>
|
|
134
|
+
))}
|
|
135
|
+
</div>
|
|
136
|
+
|
|
137
|
+
{/* Install scope: user-level or workspace-level */}
|
|
138
|
+
<div className="space-y-2">
|
|
139
|
+
<p className="text-sm font-medium">{t('installScopeLabel')}</p>
|
|
140
|
+
<div className="flex gap-2">
|
|
141
|
+
<Button
|
|
142
|
+
variant={installScope === 'user' ? 'default' : 'outline'}
|
|
143
|
+
size="sm"
|
|
144
|
+
onClick={() => setInstallScope('user')}
|
|
145
|
+
>
|
|
146
|
+
{t('userLevel')}
|
|
147
|
+
</Button>
|
|
148
|
+
<Button
|
|
149
|
+
variant={installScope === 'workspace' ? 'default' : 'outline'}
|
|
150
|
+
size="sm"
|
|
151
|
+
onClick={() => setInstallScope('workspace')}
|
|
152
|
+
>
|
|
153
|
+
<FolderOpen className="mr-1 h-3.5 w-3.5" />
|
|
154
|
+
{t('workspaceLevel')}
|
|
155
|
+
</Button>
|
|
156
|
+
</div>
|
|
157
|
+
|
|
158
|
+
{installScope === 'user' && (
|
|
159
|
+
<p className="text-xs text-muted-foreground">{t('userLevelDesc')}</p>
|
|
160
|
+
)}
|
|
161
|
+
|
|
162
|
+
{installScope === 'workspace' && (
|
|
163
|
+
<div className="space-y-2">
|
|
164
|
+
<p className="text-xs text-muted-foreground">{t('workspaceLevelDesc')}</p>
|
|
165
|
+
<Select value={selectedWorkspace} onValueChange={setSelectedWorkspace}>
|
|
166
|
+
<SelectTrigger className="w-full">
|
|
167
|
+
<SelectValue placeholder={t('selectWorkspace')} />
|
|
168
|
+
</SelectTrigger>
|
|
169
|
+
<SelectContent>
|
|
170
|
+
{workspaces.map((ws) => (
|
|
171
|
+
<SelectItem key={ws.id} value={ws.path}>
|
|
172
|
+
{ws.name} — {ws.path}
|
|
173
|
+
</SelectItem>
|
|
174
|
+
))}
|
|
175
|
+
</SelectContent>
|
|
176
|
+
</Select>
|
|
177
|
+
</div>
|
|
178
|
+
)}
|
|
179
|
+
</div>
|
|
180
|
+
|
|
181
|
+
<DialogFooter>
|
|
182
|
+
<Button variant="outline" onClick={() => onOpenChange(false)}>
|
|
183
|
+
{tc('cancel')}
|
|
184
|
+
</Button>
|
|
185
|
+
<Button
|
|
186
|
+
onClick={handleInstall}
|
|
187
|
+
disabled={installing || selected.length === 0}
|
|
188
|
+
>
|
|
189
|
+
{installing && <Loader2 className="mr-2 h-4 w-4 animate-spin" />}
|
|
190
|
+
{installing ? t('installing') : t('install')}
|
|
191
|
+
</Button>
|
|
192
|
+
</DialogFooter>
|
|
193
|
+
</DialogContent>
|
|
194
|
+
</Dialog>
|
|
195
|
+
)
|
|
196
|
+
}
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { memo } from 'react'
|
|
4
|
+
import { useTranslations } from 'next-intl'
|
|
5
|
+
import { Download, Star, ExternalLink } from 'lucide-react'
|
|
6
|
+
|
|
7
|
+
import { Badge } from '@octo-cyber/ui/components/ui/badge'
|
|
8
|
+
import { Button } from '@octo-cyber/ui/components/ui/button'
|
|
9
|
+
import {
|
|
10
|
+
Card,
|
|
11
|
+
CardContent,
|
|
12
|
+
CardDescription,
|
|
13
|
+
CardHeader,
|
|
14
|
+
CardTitle,
|
|
15
|
+
} from '@octo-cyber/ui/components/ui/card'
|
|
16
|
+
|
|
17
|
+
import type { MarketSkillInfo } from '../services/skill-market-service'
|
|
18
|
+
|
|
19
|
+
interface MarketSkillCardProps {
|
|
20
|
+
skill: MarketSkillInfo
|
|
21
|
+
onInstall: () => void
|
|
22
|
+
className?: string
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
function MarketSkillCard({
|
|
26
|
+
skill,
|
|
27
|
+
onInstall,
|
|
28
|
+
className,
|
|
29
|
+
}: MarketSkillCardProps) {
|
|
30
|
+
const t = useTranslations('aiSkillMarket')
|
|
31
|
+
|
|
32
|
+
return (
|
|
33
|
+
<Card className={`flex flex-col ${className ?? ''}`}>
|
|
34
|
+
<CardHeader className="pb-2">
|
|
35
|
+
<CardTitle className="text-base font-semibold">{skill.name}</CardTitle>
|
|
36
|
+
<CardDescription className="line-clamp-2 text-sm">
|
|
37
|
+
{skill.description}
|
|
38
|
+
</CardDescription>
|
|
39
|
+
</CardHeader>
|
|
40
|
+
|
|
41
|
+
<CardContent className="flex flex-1 flex-col justify-between gap-3">
|
|
42
|
+
<div className="flex flex-wrap items-center gap-2 text-xs text-muted-foreground">
|
|
43
|
+
<span className="flex items-center gap-1">
|
|
44
|
+
<ExternalLink className="h-3 w-3" />
|
|
45
|
+
{t('from')} {skill.repo}
|
|
46
|
+
</span>
|
|
47
|
+
{skill.stars != null && (
|
|
48
|
+
<span className="flex items-center gap-1">
|
|
49
|
+
<Star className="h-3 w-3" />
|
|
50
|
+
{skill.stars}
|
|
51
|
+
</span>
|
|
52
|
+
)}
|
|
53
|
+
{skill.category && (
|
|
54
|
+
<Badge variant="outline" className="text-xs">
|
|
55
|
+
{skill.category}
|
|
56
|
+
</Badge>
|
|
57
|
+
)}
|
|
58
|
+
</div>
|
|
59
|
+
|
|
60
|
+
<Button
|
|
61
|
+
variant="outline"
|
|
62
|
+
size="sm"
|
|
63
|
+
className="w-full"
|
|
64
|
+
onClick={onInstall}
|
|
65
|
+
>
|
|
66
|
+
<Download className="mr-2 h-4 w-4" />
|
|
67
|
+
{t('install')}
|
|
68
|
+
</Button>
|
|
69
|
+
</CardContent>
|
|
70
|
+
</Card>
|
|
71
|
+
)
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
export default memo(MarketSkillCard)
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import {
|
|
4
|
+
Loader2, Package, Star, ChevronDown, ChevronRight,
|
|
5
|
+
} from 'lucide-react'
|
|
6
|
+
import type { MarketSkillInfo, SkillRepo } from '../services/skill-market-service'
|
|
7
|
+
import MarketSkillCard from './MarketSkillCard'
|
|
8
|
+
|
|
9
|
+
interface RepoListProps {
|
|
10
|
+
repos: SkillRepo[]
|
|
11
|
+
loading: boolean
|
|
12
|
+
expandedRepo: string | null
|
|
13
|
+
repoSkills: Record<string, MarketSkillInfo[]>
|
|
14
|
+
loadingRepo: string | null
|
|
15
|
+
onToggleRepo: (owner: string, repo: string) => void
|
|
16
|
+
onInstall: (skill: MarketSkillInfo) => void
|
|
17
|
+
t: (key: string) => string
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export function RepoList({
|
|
21
|
+
repos, loading, expandedRepo, repoSkills, loadingRepo,
|
|
22
|
+
onToggleRepo, onInstall, t,
|
|
23
|
+
}: RepoListProps) {
|
|
24
|
+
if (loading) {
|
|
25
|
+
return (
|
|
26
|
+
<div className="flex items-center justify-center py-12">
|
|
27
|
+
<Loader2 className="h-6 w-6 animate-spin text-muted-foreground" />
|
|
28
|
+
</div>
|
|
29
|
+
)
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
if (repos.length === 0) {
|
|
33
|
+
return (
|
|
34
|
+
<div className="flex flex-col items-center justify-center py-12 text-muted-foreground">
|
|
35
|
+
<Package className="mb-2 h-8 w-8" />
|
|
36
|
+
<p>{t('noResults')}</p>
|
|
37
|
+
</div>
|
|
38
|
+
)
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
return (
|
|
42
|
+
<div className="space-y-2">
|
|
43
|
+
{repos.map((repo) => {
|
|
44
|
+
const key = `${repo.owner}/${repo.repo}`
|
|
45
|
+
const isExpanded = expandedRepo === key
|
|
46
|
+
const skills = repoSkills[key]
|
|
47
|
+
const isLoading = loadingRepo === key
|
|
48
|
+
|
|
49
|
+
return (
|
|
50
|
+
<div key={key} className="rounded-lg border border-border">
|
|
51
|
+
<button
|
|
52
|
+
type="button"
|
|
53
|
+
className="flex w-full items-center justify-between p-4 text-left hover:bg-muted/50"
|
|
54
|
+
onClick={() => onToggleRepo(repo.owner, repo.repo)}
|
|
55
|
+
>
|
|
56
|
+
<div className="flex flex-col gap-1">
|
|
57
|
+
<span className="font-medium">{key}</span>
|
|
58
|
+
<span className="text-sm text-muted-foreground">{repo.description}</span>
|
|
59
|
+
</div>
|
|
60
|
+
<div className="flex items-center gap-3 text-sm text-muted-foreground">
|
|
61
|
+
<span className="flex items-center gap-1">
|
|
62
|
+
<Star className="h-3.5 w-3.5" />{repo.stars}
|
|
63
|
+
</span>
|
|
64
|
+
{repo.skillCount != null && (
|
|
65
|
+
<span>{repo.skillCount} {t('skills')}</span>
|
|
66
|
+
)}
|
|
67
|
+
{isExpanded ? <ChevronDown className="h-4 w-4" /> : <ChevronRight className="h-4 w-4" />}
|
|
68
|
+
</div>
|
|
69
|
+
</button>
|
|
70
|
+
|
|
71
|
+
{isExpanded && (
|
|
72
|
+
<div className="border-t border-border p-4">
|
|
73
|
+
{isLoading ? (
|
|
74
|
+
<div className="flex items-center justify-center py-6">
|
|
75
|
+
<Loader2 className="h-5 w-5 animate-spin text-muted-foreground" />
|
|
76
|
+
</div>
|
|
77
|
+
) : skills && skills.length > 0 ? (
|
|
78
|
+
<div className="grid gap-4 md:grid-cols-2 lg:grid-cols-3">
|
|
79
|
+
{skills.map((skill) => (
|
|
80
|
+
<MarketSkillCard
|
|
81
|
+
key={`${skill.repo}/${skill.path}`}
|
|
82
|
+
skill={skill}
|
|
83
|
+
onInstall={() => onInstall(skill)}
|
|
84
|
+
/>
|
|
85
|
+
))}
|
|
86
|
+
</div>
|
|
87
|
+
) : skills ? (
|
|
88
|
+
<p className="py-4 text-center text-sm text-muted-foreground">{t('noResults')}</p>
|
|
89
|
+
) : null}
|
|
90
|
+
</div>
|
|
91
|
+
)}
|
|
92
|
+
</div>
|
|
93
|
+
)
|
|
94
|
+
})}
|
|
95
|
+
</div>
|
|
96
|
+
)
|
|
97
|
+
}
|
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { Loader2, Search, Package } from 'lucide-react'
|
|
4
|
+
import type { MarketSkillInfo } from '../services/skill-market-service'
|
|
5
|
+
import MarketSkillCard from './MarketSkillCard'
|
|
6
|
+
|
|
7
|
+
interface SearchResultsGridProps {
|
|
8
|
+
results: MarketSkillInfo[]
|
|
9
|
+
hasSearched: boolean
|
|
10
|
+
searching: boolean
|
|
11
|
+
onInstall: (skill: MarketSkillInfo) => void
|
|
12
|
+
t: (key: string) => string
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export function SearchResultsGrid({
|
|
16
|
+
results, hasSearched, searching, onInstall, t,
|
|
17
|
+
}: SearchResultsGridProps) {
|
|
18
|
+
if (searching) {
|
|
19
|
+
return (
|
|
20
|
+
<div className="flex items-center justify-center py-12">
|
|
21
|
+
<Loader2 className="h-6 w-6 animate-spin text-muted-foreground" />
|
|
22
|
+
</div>
|
|
23
|
+
)
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
if (!hasSearched) {
|
|
27
|
+
return (
|
|
28
|
+
<div className="flex flex-col items-center justify-center py-12 text-muted-foreground">
|
|
29
|
+
<Search className="mb-2 h-8 w-8" />
|
|
30
|
+
<p>{t('searchPlaceholder')}</p>
|
|
31
|
+
</div>
|
|
32
|
+
)
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
if (results.length === 0) {
|
|
36
|
+
return (
|
|
37
|
+
<div className="flex flex-col items-center justify-center py-12 text-muted-foreground">
|
|
38
|
+
<Package className="mb-2 h-8 w-8" />
|
|
39
|
+
<p>{t('noResults')}</p>
|
|
40
|
+
</div>
|
|
41
|
+
)
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
return (
|
|
45
|
+
<div className="grid gap-4 md:grid-cols-2 lg:grid-cols-3">
|
|
46
|
+
{results.map((skill) => (
|
|
47
|
+
<MarketSkillCard
|
|
48
|
+
key={`${skill.repo}/${skill.path}`}
|
|
49
|
+
skill={skill}
|
|
50
|
+
onInstall={() => onInstall(skill)}
|
|
51
|
+
/>
|
|
52
|
+
))}
|
|
53
|
+
</div>
|
|
54
|
+
)
|
|
55
|
+
}
|
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
'use client'
|
|
2
2
|
|
|
3
|
+
import { memo } from 'react'
|
|
3
4
|
import { useTranslations } from 'next-intl'
|
|
4
5
|
import {
|
|
5
6
|
Download,
|
|
@@ -39,7 +40,7 @@ interface SkillCardProps {
|
|
|
39
40
|
onDelete: () => void
|
|
40
41
|
}
|
|
41
42
|
|
|
42
|
-
|
|
43
|
+
function SkillCard({
|
|
43
44
|
skill,
|
|
44
45
|
targets,
|
|
45
46
|
onInstall,
|
|
@@ -137,3 +138,5 @@ export default function SkillCard({
|
|
|
137
138
|
</Card>
|
|
138
139
|
)
|
|
139
140
|
}
|
|
141
|
+
|
|
142
|
+
export default memo(SkillCard)
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { Loader2, RefreshCw, Play, CheckCircle2, XCircle, Terminal, Globe } from 'lucide-react'
|
|
4
|
+
import { useTranslations } from 'next-intl'
|
|
5
|
+
import { Button } from '@octo-cyber/ui/components/ui/button'
|
|
6
|
+
import { Badge } from '@octo-cyber/ui/components/ui/badge'
|
|
7
|
+
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from '@octo-cyber/ui/components/ui/card'
|
|
8
|
+
import type { CliToolInfo } from '../services/ai-cli-service'
|
|
9
|
+
|
|
10
|
+
function ToolStatusIcon({ isAvailable }: { isAvailable: boolean }) {
|
|
11
|
+
if (isAvailable) return <CheckCircle2 className="h-5 w-5 text-green-500" />
|
|
12
|
+
return <XCircle className="h-5 w-5 text-muted-foreground" />
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
export function ToolCard({
|
|
16
|
+
tool,
|
|
17
|
+
onTest,
|
|
18
|
+
onHealthCheck,
|
|
19
|
+
isChecking,
|
|
20
|
+
installedSkills,
|
|
21
|
+
}: {
|
|
22
|
+
tool: CliToolInfo
|
|
23
|
+
onTest: (tool: CliToolInfo) => void
|
|
24
|
+
onHealthCheck: (toolName: string) => void
|
|
25
|
+
isChecking: boolean
|
|
26
|
+
installedSkills: Array<{ name: string; skillDir: string }>
|
|
27
|
+
}) {
|
|
28
|
+
const t = useTranslations('aiCli')
|
|
29
|
+
const ToolIcon = tool.toolType === 'api' ? Globe : Terminal
|
|
30
|
+
|
|
31
|
+
return (
|
|
32
|
+
<Card>
|
|
33
|
+
<CardHeader className="flex flex-row items-center justify-between space-y-0 pb-2">
|
|
34
|
+
<div className="flex items-center gap-3">
|
|
35
|
+
<ToolIcon className="h-5 w-5 text-muted-foreground" />
|
|
36
|
+
<div>
|
|
37
|
+
<CardTitle className="text-base">{tool.displayName}</CardTitle>
|
|
38
|
+
<CardDescription className="text-xs">{tool.name}</CardDescription>
|
|
39
|
+
</div>
|
|
40
|
+
</div>
|
|
41
|
+
<ToolStatusIcon isAvailable={tool.available} />
|
|
42
|
+
</CardHeader>
|
|
43
|
+
<CardContent>
|
|
44
|
+
<div className="space-y-3">
|
|
45
|
+
<div className="flex items-center gap-2 text-sm">
|
|
46
|
+
<Badge variant={tool.available ? 'default' : 'secondary'}>
|
|
47
|
+
{tool.available ? t('status.available') : t('status.unavailable')}
|
|
48
|
+
</Badge>
|
|
49
|
+
<Badge variant="outline" className="text-xs">{t(`toolType.${tool.toolType}`)}</Badge>
|
|
50
|
+
{tool.version && <span className="text-muted-foreground">v{tool.version}</span>}
|
|
51
|
+
{tool.models && tool.models.length > 0 && (
|
|
52
|
+
<span className="text-muted-foreground text-xs">
|
|
53
|
+
{t('modelCount', { count: tool.models.length })}
|
|
54
|
+
</span>
|
|
55
|
+
)}
|
|
56
|
+
</div>
|
|
57
|
+
{tool.path && (
|
|
58
|
+
<p className="text-xs text-muted-foreground truncate" title={tool.path}>{tool.path}</p>
|
|
59
|
+
)}
|
|
60
|
+
{installedSkills.length > 0 && (
|
|
61
|
+
<div className="space-y-1.5 pt-1 border-t">
|
|
62
|
+
<p className="text-xs font-medium text-muted-foreground pt-1">
|
|
63
|
+
{t('installedSkills')} ({installedSkills.length})
|
|
64
|
+
</p>
|
|
65
|
+
<div className="flex flex-wrap gap-1">
|
|
66
|
+
{installedSkills.map((skill) => (
|
|
67
|
+
<Badge key={skill.name} variant="outline" className="text-xs">{skill.name}</Badge>
|
|
68
|
+
))}
|
|
69
|
+
</div>
|
|
70
|
+
</div>
|
|
71
|
+
)}
|
|
72
|
+
<div className="flex gap-2 pt-1">
|
|
73
|
+
<Button variant="outline" size="sm" onClick={() => onHealthCheck(tool.name)} disabled={isChecking}>
|
|
74
|
+
{isChecking ? <Loader2 className="mr-1 h-3 w-3 animate-spin" /> : <RefreshCw className="mr-1 h-3 w-3" />}
|
|
75
|
+
{t('actions.healthCheck')}
|
|
76
|
+
</Button>
|
|
77
|
+
{tool.available && (
|
|
78
|
+
<Button variant="outline" size="sm" onClick={() => onTest(tool)}>
|
|
79
|
+
<Play className="mr-1 h-3 w-3" />
|
|
80
|
+
{t('actions.testRun')}
|
|
81
|
+
</Button>
|
|
82
|
+
)}
|
|
83
|
+
</div>
|
|
84
|
+
</div>
|
|
85
|
+
</CardContent>
|
|
86
|
+
</Card>
|
|
87
|
+
)
|
|
88
|
+
}
|
|
@@ -0,0 +1,146 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { memo } from 'react'
|
|
4
|
+
import { useTranslations } from 'next-intl'
|
|
5
|
+
import {
|
|
6
|
+
Trash2,
|
|
7
|
+
FolderOpen,
|
|
8
|
+
Github,
|
|
9
|
+
Loader2,
|
|
10
|
+
GitPullRequest,
|
|
11
|
+
GitBranch,
|
|
12
|
+
} from 'lucide-react'
|
|
13
|
+
|
|
14
|
+
import { Button } from '@octo-cyber/ui/components/ui/button'
|
|
15
|
+
import { Badge } from '@octo-cyber/ui/components/ui/badge'
|
|
16
|
+
import {
|
|
17
|
+
Card,
|
|
18
|
+
CardContent,
|
|
19
|
+
CardHeader,
|
|
20
|
+
CardTitle,
|
|
21
|
+
} from '@octo-cyber/ui/components/ui/card'
|
|
22
|
+
import type { WorkspaceInfo } from '../services/ai-cli-service'
|
|
23
|
+
|
|
24
|
+
type InstalledSkillsMap = Record<string, Array<{ name: string; skillDir: string }>>
|
|
25
|
+
|
|
26
|
+
function formatDate(dateStr: string): string {
|
|
27
|
+
try {
|
|
28
|
+
return new Date(dateStr).toLocaleString()
|
|
29
|
+
} catch {
|
|
30
|
+
return dateStr
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
export const WorkspaceCard = memo(function WorkspaceCard({
|
|
35
|
+
ws,
|
|
36
|
+
onDelete,
|
|
37
|
+
onPull,
|
|
38
|
+
isPulling,
|
|
39
|
+
wsSkills,
|
|
40
|
+
}: {
|
|
41
|
+
ws: WorkspaceInfo
|
|
42
|
+
onDelete: (ws: WorkspaceInfo) => void
|
|
43
|
+
onPull: (id: string) => void
|
|
44
|
+
isPulling: boolean
|
|
45
|
+
wsSkills: InstalledSkillsMap
|
|
46
|
+
}) {
|
|
47
|
+
const t = useTranslations('aiCli.workspaces')
|
|
48
|
+
const tc = useTranslations('common')
|
|
49
|
+
|
|
50
|
+
const allSkills = Object.entries(wsSkills).flatMap(([target, skills]) =>
|
|
51
|
+
skills.map((s) => ({ ...s, target })),
|
|
52
|
+
)
|
|
53
|
+
|
|
54
|
+
return (
|
|
55
|
+
<Card>
|
|
56
|
+
<CardHeader className="pb-2">
|
|
57
|
+
<div className="flex items-center justify-between">
|
|
58
|
+
<div className="flex items-center gap-2">
|
|
59
|
+
{ws.type === 'github' ? (
|
|
60
|
+
<Github className="h-4 w-4 text-muted-foreground" />
|
|
61
|
+
) : (
|
|
62
|
+
<FolderOpen className="h-4 w-4 text-muted-foreground" />
|
|
63
|
+
)}
|
|
64
|
+
<CardTitle className="text-sm">
|
|
65
|
+
{ws.name || ws.id.slice(0, 8)}
|
|
66
|
+
</CardTitle>
|
|
67
|
+
</div>
|
|
68
|
+
<Badge variant={ws.type === 'github' ? 'default' : 'secondary'}>
|
|
69
|
+
{t(`type.${ws.type}`)}
|
|
70
|
+
</Badge>
|
|
71
|
+
</div>
|
|
72
|
+
</CardHeader>
|
|
73
|
+
<CardContent className="space-y-2">
|
|
74
|
+
{ws.description && (
|
|
75
|
+
<p className="text-xs text-muted-foreground line-clamp-2">
|
|
76
|
+
{ws.description}
|
|
77
|
+
</p>
|
|
78
|
+
)}
|
|
79
|
+
<p className="text-xs text-muted-foreground truncate" title={ws.path}>
|
|
80
|
+
{ws.path}
|
|
81
|
+
</p>
|
|
82
|
+
{ws.type === 'local' && ws.repoUrl && (
|
|
83
|
+
<p className="text-xs text-muted-foreground truncate" title={ws.repoUrl}>
|
|
84
|
+
<GitBranch className="inline h-3 w-3 mr-1" />
|
|
85
|
+
{ws.repoUrl}
|
|
86
|
+
{ws.branch && ` (${ws.branch})`}
|
|
87
|
+
</p>
|
|
88
|
+
)}
|
|
89
|
+
{ws.type === 'github' && ws.repoUrl && (
|
|
90
|
+
<p className="text-xs text-muted-foreground truncate" title={ws.repoUrl}>
|
|
91
|
+
{ws.repoUrl}
|
|
92
|
+
</p>
|
|
93
|
+
)}
|
|
94
|
+
{ws.type === 'github' && ws.branch && (
|
|
95
|
+
<p className="text-xs text-muted-foreground">
|
|
96
|
+
{t('branch', { branch: ws.branch })}
|
|
97
|
+
</p>
|
|
98
|
+
)}
|
|
99
|
+
<div className="flex items-center justify-between text-xs text-muted-foreground">
|
|
100
|
+
<span>{formatDate(ws.createdAt)}</span>
|
|
101
|
+
</div>
|
|
102
|
+
{allSkills.length > 0 && (
|
|
103
|
+
<div className="space-y-1.5 pt-1 border-t">
|
|
104
|
+
<p className="text-xs font-medium text-muted-foreground pt-1">
|
|
105
|
+
{t('installedSkills')} ({allSkills.length})
|
|
106
|
+
</p>
|
|
107
|
+
<div className="flex flex-wrap gap-1">
|
|
108
|
+
{allSkills.map((skill) => (
|
|
109
|
+
<Badge key={`${skill.target}-${skill.name}`} variant="outline" className="text-xs">
|
|
110
|
+
{skill.name}
|
|
111
|
+
<span className="ml-1 text-muted-foreground">({skill.target})</span>
|
|
112
|
+
</Badge>
|
|
113
|
+
))}
|
|
114
|
+
</div>
|
|
115
|
+
</div>
|
|
116
|
+
)}
|
|
117
|
+
<div className="flex justify-end gap-2 pt-1">
|
|
118
|
+
{ws.type === 'github' && (
|
|
119
|
+
<Button
|
|
120
|
+
variant="outline"
|
|
121
|
+
size="sm"
|
|
122
|
+
onClick={() => onPull(ws.id)}
|
|
123
|
+
disabled={isPulling}
|
|
124
|
+
>
|
|
125
|
+
{isPulling ? (
|
|
126
|
+
<Loader2 className="mr-1 h-3 w-3 animate-spin" />
|
|
127
|
+
) : (
|
|
128
|
+
<GitPullRequest className="mr-1 h-3 w-3" />
|
|
129
|
+
)}
|
|
130
|
+
{isPulling ? t('pulling') : t('actions.pull')}
|
|
131
|
+
</Button>
|
|
132
|
+
)}
|
|
133
|
+
<Button
|
|
134
|
+
variant="ghost"
|
|
135
|
+
size="sm"
|
|
136
|
+
className="text-destructive hover:text-destructive"
|
|
137
|
+
onClick={() => onDelete(ws)}
|
|
138
|
+
>
|
|
139
|
+
<Trash2 className="mr-1 h-3 w-3" />
|
|
140
|
+
{tc('delete')}
|
|
141
|
+
</Button>
|
|
142
|
+
</div>
|
|
143
|
+
</CardContent>
|
|
144
|
+
</Card>
|
|
145
|
+
)
|
|
146
|
+
})
|