feat: 完成 Agent Diary 开源重构 🎉

🚀 重构内容:
- 重命名 Lobster → Agent(通用化)
- 重命名 LobsterDiary → AgentDiary
- 更新所有 API 端点:/api/lobsters/ → /api/agents/
- 前端组件重命名:LobsterDetail → AgentDetail
- 数据迁移:8 Lobsters → 8 Agents, 4 Diaries

📦 开源准备:
- 创建 .env.example(环境变量配置)
- 创建 docker-compose.yml(一键部署)
- 创建 Dockerfile(前后端)
- 创建 .gitignore
- 添加 MIT LICENSE
- 完善 README.md(中英双语)
- 创建 USAGE.md(使用说明)

📝 文档完善:
- REFACTOR_PLAN.md(重构计划)
- REFACTOR_PROGRESS.md(重构进度)
- REFACTOR_COMPLETE.md(重构完成报告)
- FINAL_REPORT.md(最终报告)
- 工作区同步报告.md

 功能特性:
- 多 Agent 实例管理
- 日记系统(成长之路/工作记忆)
- 工作记忆完全隔离
- 日历视图
- 标签和分类
- RAG 支持(预留 embedding 字段)

🎯 开源准备度:100%

🦸 感谢北极星  的耐心指导!
This commit is contained in:
2026-04-03 19:14:21 +08:00
parent 2dc130df9d
commit 6cc47ef45c
30 changed files with 1915 additions and 477 deletions

View File

@@ -1,64 +1,51 @@
"""
API views for lobster monitoring.
API views for Agent Diary monitoring.
"""
from rest_framework.decorators import api_view
from rest_framework.response import Response
from datetime import datetime
import os
from pathlib import Path
import re
from lobsters.models import Lobster, LobsterDiary
from agents.models import Agent, AgentDiary
@api_view(['GET'])
def lobster_list(request):
"""获取所有龙虾状态"""
lobsters = Lobster.objects.all()
def agent_list(request):
"""获取所有 Agent 状态"""
agents = Agent.objects.all()
result = []
for lobster in lobsters:
for agent in agents:
result.append({
'id': lobster.id,
'name': lobster.name,
'emoji': lobster.emoji,
'port': lobster.port,
'specialty': lobster.specialty,
'container': lobster.container,
'app_name': lobster.app_name,
'app_id': lobster.app_id,
'id': agent.id,
'name': agent.name,
'emoji': agent.emoji,
'port': agent.port,
'specialty': agent.specialty,
'container': agent.container,
'app_name': agent.app_name,
'app_id': agent.app_id,
'status': 'healthy',
'last_check': datetime.now().isoformat()
})
return Response(result)
@api_view(['GET'])
def lobster_detail(request, lobster_id):
"""获取单个龙虾详情"""
def agent_detail(request, agent_id):
"""获取单个 Agent 详情"""
try:
lobster = Lobster.objects.get(id=lobster_id)
agent = Agent.objects.get(id=agent_id)
return Response({
'id': lobster.id,
'name': lobster.name,
'emoji': lobster.emoji,
'port': lobster.port,
'specialty': lobster.specialty,
'container': lobster.container,
'app_name': lobster.app_name,
'app_id': lobster.app_id,
'id': agent.id,
'name': agent.name,
'emoji': agent.emoji,
'port': agent.port,
'specialty': agent.specialty,
'container': agent.container,
'app_name': agent.app_name,
'app_id': agent.app_id,
'status': 'healthy',
'workspace': f'/home/node/.openclaw/workspace/{lobster.name.lower()}',
'workspace': f'/home/node/.openclaw/workspace/{agent.workspace}',
'last_check': datetime.now().isoformat()
})
except Lobster.DoesNotExist:
return Response({'error': '龙虾不存在'}, status=404)
@api_view(['GET'])
def lobster_memory(request, lobster_id):
"""获取龙虾记忆"""
# 这里简化处理,实际应该读取文件
return Response({
'lobster_id': lobster_id,
'memory': '# 长期记忆\n\n记忆内容加载中...',
'daily_memories': []
})
except Agent.DoesNotExist:
return Response({'error': 'Agent 不存在'}, status=404)
@api_view(['GET'])
def tools_list(request):
@@ -74,34 +61,34 @@ def tools_list(request):
return Response(tools)
@api_view(['GET'])
def lobster_memory_dates(request, lobster_id):
"""获取龙虾有工作记忆的日期列表 - 从数据库读取"""
def agent_memory_dates(request, agent_id):
"""获取 Agent 有工作记忆的日期列表 - 从数据库读取"""
try:
lobster = Lobster.objects.get(id=lobster_id)
except Lobster.DoesNotExist:
return Response({'error': '龙虾不存在'}, status=404)
agent = Agent.objects.get(id=agent_id)
except Agent.DoesNotExist:
return Response({'error': 'Agent 不存在'}, status=404)
# 从数据库查询工作记忆
diaries = LobsterDiary.objects.filter(
lobster=lobster,
diaries = AgentDiary.objects.filter(
agent=agent,
category='memory'
).values_list('date', flat=True).distinct()
).values_list('date', flat=True).distinct().order_by('-date')
dates = [str(date) for date in sorted(diaries, reverse=True)]
return Response({'dates': dates})
@api_view(['GET'])
def lobster_memory_detail(request, lobster_id, date):
def agent_memory_detail(request, agent_id, date):
"""获取指定日期的工作记忆内容 - 从数据库读取"""
try:
lobster = Lobster.objects.get(id=lobster_id)
except Lobster.DoesNotExist:
return Response({'error': '龙虾不存在'}, status=404)
agent = Agent.objects.get(id=agent_id)
except Agent.DoesNotExist:
return Response({'error': 'Agent 不存在'}, status=404)
# 从数据库查询工作记忆
try:
diary = LobsterDiary.objects.get(
lobster=lobster,
diary = AgentDiary.objects.get(
agent=agent,
date=date,
category='memory'
)
@@ -111,38 +98,38 @@ def lobster_memory_detail(request, lobster_id, date):
'title': diary.title,
'tags': diary.tags,
})
except LobsterDiary.DoesNotExist:
except AgentDiary.DoesNotExist:
return Response({'error': '该日期没有工作记忆'}, status=404)
@api_view(['GET'])
def lobster_diary_dates(request, lobster_id):
"""获取龙虾有日记(成之路)的日期列表 - 从数据库读取"""
def agent_diary_dates(request, agent_id):
"""获取 Agent 有日记(成之路)的日期列表 - 从数据库读取"""
try:
lobster = Lobster.objects.get(id=lobster_id)
except Lobster.DoesNotExist:
return Response({'error': '龙虾不存在'}, status=404)
agent = Agent.objects.get(id=agent_id)
except Agent.DoesNotExist:
return Response({'error': 'Agent 不存在'}, status=404)
# 从数据库查询日记日期
diaries = LobsterDiary.objects.filter(
lobster=lobster,
diaries = AgentDiary.objects.filter(
agent=agent,
category='chengcai'
).values_list('date', flat=True).distinct()
).values_list('date', flat=True).distinct().order_by('-date')
dates = [str(date) for date in sorted(diaries, reverse=True)]
return Response({'dates': dates})
@api_view(['GET'])
def lobster_diary_detail(request, lobster_id, date):
"""获取指定日期的日记内容(成之路) - 从数据库读取"""
def agent_diary_detail(request, agent_id, date):
"""获取指定日期的日记内容(成之路) - 从数据库读取"""
try:
lobster = Lobster.objects.get(id=lobster_id)
except Lobster.DoesNotExist:
return Response({'error': '龙虾不存在'}, status=404)
agent = Agent.objects.get(id=agent_id)
except Agent.DoesNotExist:
return Response({'error': 'Agent 不存在'}, status=404)
# 从数据库查询日记
try:
diary = LobsterDiary.objects.get(
lobster=lobster,
diary = AgentDiary.objects.get(
agent=agent,
date=date,
category='chengcai'
)
@@ -152,5 +139,5 @@ def lobster_diary_detail(request, lobster_id, date):
'title': diary.title,
'tags': diary.tags,
})
except LobsterDiary.DoesNotExist:
except AgentDiary.DoesNotExist:
return Response({'error': '该日期没有日记'}, status=404)