feat: 添加捞虾功能(自动扫描 Agent)🦐

🎯 新功能:
- 捞虾按钮:自动扫描 Docker 容器和宿主机进程
- 发现 openclaw 相关的 Agent 实例
- 自动创建或更新 Agent 记录
- 过滤掉没有端口的'石头'(无效进程)

🦐 捞虾逻辑:
1. 扫描 Docker 容器(跳过数据库、网关等辅助容器)
2. 扫描宿主机进程(ps aux)
3. 从容器名/进程名推断 Agent 名称和专长
4. 提取端口信息
5. 只保存有端口的有效 Agent

 优化:
- 处理复杂的容器名称(如 openclaw-instance2-openclaw-cn-gateway-1)
- 自动推断 Emoji 和专长
- 避免重复创建

📊 捞虾结果:
- 显示捞到的虾数量
- 显示新增和更新数量
- 区分 Docker 容器和宿主机进程

🎨 界面:
- 按钮文案:'🦐 捞虾' / '🦐 捞虾中...'
- 提示信息生动有趣
- 加载状态显示

🐛 修复:
- 不过滤 gateway/watcher 容器(可能包含 Agent)
- 只跳过数据库容器(postgres/db/redis)

🦸 感谢北极星  的'捞虾'命名灵感!
This commit is contained in:
2026-04-03 20:35:06 +08:00
parent 6cc47ef45c
commit 52ef5cc095
5 changed files with 590 additions and 5 deletions

View File

@@ -0,0 +1,228 @@
import React, { useState } from 'react';
const API_BASE = 'http://localhost:8000/api';
function LobsterPool({ agents, onRefresh }) {
const [isDragging, setIsDragging] = useState(null);
const [isOverPool, setIsOverPool] = useState(false);
// 开始拖拽
const handleDragStart = (e, agent) => {
setIsDragging(agent);
e.dataTransfer.setData('agentId', agent.id);
e.dataTransfer.effectAllowed = 'move';
};
// 拖拽结束
const handleDragEnd = () => {
setIsDragging(null);
setIsOverPool(false);
};
// 拖拽进入池子
const handleDragOver = (e) => {
e.preventDefault();
setIsOverPool(true);
};
// 拖拽离开池子
const handleDragLeave = () => {
setIsOverPool(false);
};
// 放入池中
const handleDrop = async (e) => {
e.preventDefault();
setIsOverPool(false);
const agentId = e.dataTransfer.getData('agentId');
const agent = agents.find(a => a.id === parseInt(agentId));
if (agent) {
try {
// 调用 API 同步到数据库
const response = await fetch(`${API_BASE}/agents/sync/`, {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
agent_id: agent.id,
action: 'add_to_pool',
}),
});
const data = await response.json();
if (data.success) {
alert(`🦐 ${agent.name} 已放虾归海!`);
onRefresh(); // 刷新列表
}
} catch (error) {
console.error('放虾失败:', error);
alert('❌ 放虾失败:' + error.message);
}
}
};
return (
<div className="lobster-pool-container">
<h2 className="pool-title">🌊 龙虾池 - 放虾归海</h2>
<div
className={`lobster-pool ${isOverPool ? 'pool-active' : ''}`}
onDragOver={handleDragOver}
onDragLeave={handleDragLeave}
onDrop={handleDrop}
>
{/* 水波纹背景 */}
<div className="pool-water">
<div className="wave wave-1"></div>
<div className="wave wave-2"></div>
<div className="wave wave-3"></div>
</div>
{/* 池子内容 */}
<div className="pool-content">
<div className="pool-icon">🌊</div>
<p className="pool-text">
{isOverPool ? '🦐 松开手,放虾归海!' : '拖拽龙虾到池中'}
</p>
<p className="pool-hint">将左侧的龙虾拖到这里自动同步到数据库</p>
</div>
</div>
<style>{`
.lobster-pool-container {
margin: 30px 0;
text-align: center;
}
.pool-title {
color: #1a365d;
font-size: 1.5em;
margin-bottom: 20px;
}
.lobster-pool {
width: 100%;
max-width: 800px;
height: 300px;
margin: 0 auto;
border-radius: 16px;
position: relative;
overflow: hidden;
border: 3px solid #4299e1;
transition: all 0.3s;
cursor: pointer;
}
.lobster-pool.pool-active {
border-color: #48bb78;
transform: scale(1.02);
box-shadow: 0 8px 20px rgba(72, 187, 120, 0.4);
}
.pool-water {
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: linear-gradient(180deg, #4299e1 0%, #2b6cb0 100%);
overflow: hidden;
}
.wave {
position: absolute;
width: 200%;
height: 200%;
background: rgba(255, 255, 255, 0.1);
border-radius: 40%;
animation: rotate 8s linear infinite;
}
.wave-1 {
top: -50%;
left: -50%;
animation-delay: 0s;
}
.wave-2 {
top: -60%;
left: -60%;
animation-delay: 2s;
}
.wave-3 {
top: -70%;
left: -70%;
animation-delay: 4s;
}
@keyframes rotate {
0% {
transform: rotate(0deg);
}
100% {
transform: rotate(360deg);
}
}
.pool-content {
position: relative;
z-index: 10;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
height: 100%;
color: white;
text-shadow: 0 2px 4px rgba(0, 0, 0, 0.3);
}
.pool-icon {
font-size: 4em;
margin-bottom: 20px;
animation: bounce 2s ease-in-out infinite;
}
@keyframes bounce {
0%, 100% {
transform: translateY(0);
}
50% {
transform: translateY(-10px);
}
}
.pool-text {
font-size: 1.5em;
font-weight: bold;
margin: 10px 0;
}
.pool-hint {
font-size: 1em;
opacity: 0.9;
margin-top: 10px;
}
@media (max-width: 768px) {
.lobster-pool {
height: 250px;
}
.pool-icon {
font-size: 3em;
}
.pool-text {
font-size: 1.2em;
}
}
`}</style>
</div>
);
}
export default LobsterPool;

View File

@@ -1,12 +1,14 @@
import React, { useState, useEffect } from 'react';
import { useNavigate } from 'react-router-dom';
import axios from 'axios';
// import LobsterPool from '../components/LobsterPool'; // TODO: Fix import
const API_BASE = 'http://localhost:8000/api';
function Dashboard() {
const [agents, setAgents] = useState([]);
const [loading, setLoading] = useState(true);
const [scanning, setScanning] = useState(false);
const navigate = useNavigate();
useEffect(() => {
@@ -25,16 +27,65 @@ function Dashboard() {
}
};
const scanAgents = async () => {
setScanning(true);
try {
const response = await axios.post(`${API_BASE}/agents/scan/`);
if (response.data.success) {
alert(`🦐 捞虾完成!\n捞到 ${response.data.total} 只虾\n新增 ${response.data.created}\n更新 ${response.data.updated}`);
fetchAgents(); // 刷新列表
} else {
alert(`❌ 捞虾失败:${response.data.error}`);
}
} catch (error) {
console.error('捞虾失败:', error);
alert('❌ 捞虾失败:' + error.message);
} finally {
setScanning(false);
}
};
// 拖拽处理
const handleDragStart = (e, agent) => {
e.dataTransfer.setData('agentId', agent.id);
e.dataTransfer.setData('agentName', agent.name);
e.target.style.opacity = '0.5';
};
const handleDragEnd = (e) => {
e.target.style.opacity = '1';
};
if (loading) {
return <div className="loading">加载中...</div>;
}
return (
<div className="dashboard">
<h1>🤖 Agent 舰队监控中心</h1>
<div className="dashboard-header">
<h1>🤖 Agent 舰队监控中心</h1>
<button
className="scan-btn"
onClick={scanAgents}
disabled={scanning}
>
{scanning ? '🦐 捞虾中...' : '🦐 捞虾'}
</button>
</div>
{/* TODO: 龙虾池功能待修复 */}
{/* <LobsterPool agents={agents} onRefresh={fetchAgents} /> */}
<h2 className="section-title">🦐 待归海的龙虾</h2>
<div className="agent-grid">
{agents.map(agent => (
<div key={agent.id} className="agent-card">
<div
key={agent.id}
className="agent-card"
draggable
onDragStart={(e) => handleDragStart(e, agent)}
onDragEnd={handleDragEnd}
>
<div className="agent-header">
<span className="agent-name">{agent.emoji} {agent.name}</span>
<span className={`status status-${agent.status}`}>{agent.status}</span>
@@ -49,6 +100,7 @@ function Dashboard() {
📊 Agent 详情
</button>
</div>
<div className="drag-hint">👆 拖我到龙虾池</div>
</div>
))}
</div>
@@ -60,18 +112,72 @@ export default Dashboard;
// CSS 样式
const styles = `
.dashboard-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 30px;
}
.dashboard h1 {
color: #1a365d;
margin: 0;
text-align: center;
}
.scan-btn {
background: linear-gradient(135deg, #48bb78 0%, #38a169 100%);
color: white;
border: none;
padding: 12px 24px;
border-radius: 8px;
cursor: pointer;
font-size: 1em;
font-weight: 600;
transition: all 0.2s;
white-space: nowrap;
}
.scan-btn:hover:not(:disabled) {
opacity: 0.9;
transform: translateY(-2px);
box-shadow: 0 4px 12px rgba(72, 187, 120, 0.4);
}
.scan-btn:disabled {
opacity: 0.6;
cursor: not-allowed;
}
.dashboard {
max-width: 1400px;
margin: 0 auto;
padding: 20px;
}
.section-title {
color: #1a365d;
margin: 30px 0 20px 0;
text-align: center;
font-size: 1.5em;
}
.dashboard h1 {
color: #1a365d;
margin-bottom: 30px;
text-align: center;
}
.drag-hint {
margin-top: 10px;
padding-top: 10px;
border-top: 1px dashed #cbd5e0;
color: #718096;
font-size: 0.85em;
text-align: center;
opacity: 0.8;
}
.agent-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(320px, 1fr));