299 lines
11 KiB
Python
299 lines
11 KiB
Python
#!/usr/bin/env python3
|
||
"""
|
||
Vertex AI Memory Bank Web界面
|
||
一个简单的Streamlit应用,用于通过Web界面访问和管理Memory Bank
|
||
"""
|
||
|
||
import streamlit as st
|
||
import asyncio
|
||
import sys
|
||
import os
|
||
from datetime import datetime
|
||
|
||
# 添加项目路径
|
||
sys.path.append('/Users/ben/liurenchaxin/src')
|
||
|
||
try:
|
||
from jixia.memory.factory import get_memory_backend
|
||
except ImportError as e:
|
||
st.error(f"无法导入jixia模块: {e}")
|
||
st.info("请确保已激活虚拟环境并安装了所需依赖")
|
||
st.stop()
|
||
|
||
# 页面配置
|
||
st.set_page_config(
|
||
page_title="Memory Bank 管理界面",
|
||
page_icon="🧠",
|
||
layout="wide"
|
||
)
|
||
|
||
# 标题
|
||
st.title("🧠 Vertex AI Memory Bank 管理界面")
|
||
st.markdown("---")
|
||
|
||
# 侧边栏配置
|
||
st.sidebar.header("配置")
|
||
project_id = st.sidebar.text_input("项目ID", value="inner-radius-469712-e9")
|
||
location = st.sidebar.text_input("区域", value="us-central1")
|
||
|
||
# 八仙列表
|
||
EIGHT_IMMORTALS = [
|
||
"lvdongbin", "tieguaili", "hanxiangzi", "lanzaihe",
|
||
"hesengu", "zhonghanli", "caogujiu", "hanzhongli"
|
||
]
|
||
|
||
# 缓存Memory Bank后端
|
||
@st.cache_resource
|
||
def get_memory_backend_cached():
|
||
"""获取Memory Bank后端(缓存)"""
|
||
try:
|
||
loop = asyncio.new_event_loop()
|
||
asyncio.set_event_loop(loop)
|
||
backend = loop.run_until_complete(get_memory_backend("vertex"))
|
||
return backend, loop
|
||
except Exception as e:
|
||
st.error(f"初始化Memory Bank失败: {e}")
|
||
return None, None
|
||
|
||
# 异步函数包装器
|
||
def run_async(coro):
|
||
"""运行异步函数"""
|
||
backend, loop = get_memory_backend_cached()
|
||
if backend is None:
|
||
return None
|
||
try:
|
||
return loop.run_until_complete(coro)
|
||
except Exception as e:
|
||
st.error(f"操作失败: {e}")
|
||
return None
|
||
|
||
# 主界面
|
||
tab1, tab2, tab3, tab4 = st.tabs(["📋 Memory Bank列表", "🔍 搜索记忆", "➕ 添加记忆", "📊 统计信息"])
|
||
|
||
with tab1:
|
||
st.header("Memory Bank 实例列表")
|
||
|
||
if st.button("🔄 刷新列表", key="refresh_list"):
|
||
st.rerun()
|
||
|
||
# 显示八仙Memory Bank状态
|
||
cols = st.columns(4)
|
||
for i, immortal in enumerate(EIGHT_IMMORTALS):
|
||
with cols[i % 4]:
|
||
with st.container():
|
||
st.subheader(f"🧙♂️ {immortal}")
|
||
|
||
# 检查Memory Bank状态
|
||
backend, _ = get_memory_backend_cached()
|
||
if backend:
|
||
try:
|
||
# 尝试获取agent context来验证Memory Bank存在
|
||
context = run_async(backend.get_agent_context(immortal))
|
||
if context is not None:
|
||
st.success("✅ 活跃")
|
||
|
||
# 显示记忆数量
|
||
memories = run_async(backend.search_memories(immortal, "", limit=100))
|
||
if memories:
|
||
st.info(f"📝 记忆数量: {len(memories)}")
|
||
else:
|
||
st.info("📝 记忆数量: 0")
|
||
else:
|
||
st.warning("⚠️ 未初始化")
|
||
except Exception as e:
|
||
st.error(f"❌ 错误: {str(e)[:50]}...")
|
||
else:
|
||
st.error("❌ 连接失败")
|
||
|
||
with tab2:
|
||
st.header("🔍 搜索记忆")
|
||
|
||
col1, col2 = st.columns([1, 2])
|
||
|
||
with col1:
|
||
selected_agent = st.selectbox("选择Agent", EIGHT_IMMORTALS, key="search_agent")
|
||
search_query = st.text_input("搜索关键词", placeholder="输入要搜索的内容...", key="search_query")
|
||
search_limit = st.slider("结果数量", 1, 50, 10, key="search_limit")
|
||
|
||
if st.button("🔍 搜索", key="search_button"):
|
||
if search_query:
|
||
with st.spinner("搜索中..."):
|
||
backend, _ = get_memory_backend_cached()
|
||
if backend:
|
||
memories = run_async(backend.search_memories(selected_agent, search_query, limit=search_limit))
|
||
st.session_state['search_results'] = memories
|
||
st.session_state['search_agent'] = selected_agent
|
||
st.session_state['search_query'] = search_query
|
||
else:
|
||
st.warning("请输入搜索关键词")
|
||
|
||
with col2:
|
||
st.subheader("搜索结果")
|
||
|
||
if 'search_results' in st.session_state and st.session_state['search_results']:
|
||
st.success(f"找到 {len(st.session_state['search_results'])} 条记忆")
|
||
|
||
for i, memory in enumerate(st.session_state['search_results']):
|
||
with st.expander(f"记忆 {i+1}: {memory.get('content', 'N/A')[:50]}..."):
|
||
st.write(f"**内容**: {memory.get('content', 'N/A')}")
|
||
st.write(f"**类型**: {memory.get('memory_type', 'N/A')}")
|
||
st.write(f"**时间**: {memory.get('timestamp', 'N/A')}")
|
||
if 'metadata' in memory:
|
||
st.write(f"**元数据**: {memory['metadata']}")
|
||
elif 'search_results' in st.session_state:
|
||
st.info("未找到匹配的记忆")
|
||
else:
|
||
st.info("请执行搜索以查看结果")
|
||
|
||
with tab3:
|
||
st.header("➕ 添加记忆")
|
||
|
||
col1, col2 = st.columns([1, 1])
|
||
|
||
with col1:
|
||
add_agent = st.selectbox("选择Agent", EIGHT_IMMORTALS, key="add_agent")
|
||
memory_type = st.selectbox("记忆类型", ["conversation", "preference", "knowledge", "strategy"], key="memory_type")
|
||
memory_content = st.text_area("记忆内容", placeholder="输入要添加的记忆内容...", height=150, key="memory_content")
|
||
|
||
# 可选的元数据
|
||
st.subheader("元数据(可选)")
|
||
importance = st.slider("重要性", 1, 10, 5, key="importance")
|
||
tags = st.text_input("标签(用逗号分隔)", placeholder="标签1, 标签2, 标签3", key="tags")
|
||
|
||
if st.button("➕ 添加记忆", key="add_memory_button"):
|
||
if memory_content:
|
||
with st.spinner("添加记忆中..."):
|
||
backend, _ = get_memory_backend_cached()
|
||
if backend:
|
||
# 准备元数据
|
||
metadata = {
|
||
"importance": importance,
|
||
"timestamp": datetime.now().isoformat(),
|
||
"source": "web_interface"
|
||
}
|
||
if tags:
|
||
metadata["tags"] = [tag.strip() for tag in tags.split(",")]
|
||
|
||
# 添加记忆
|
||
success = run_async(backend.add_memory(
|
||
agent_id=add_agent,
|
||
content=memory_content,
|
||
memory_type=memory_type,
|
||
metadata=metadata
|
||
))
|
||
|
||
if success:
|
||
st.success("✅ 记忆添加成功!")
|
||
# 清空输入
|
||
st.session_state['memory_content'] = ""
|
||
st.session_state['tags'] = ""
|
||
else:
|
||
st.error("❌ 添加记忆失败")
|
||
else:
|
||
st.warning("请输入记忆内容")
|
||
|
||
with col2:
|
||
st.subheader("添加记忆预览")
|
||
if memory_content:
|
||
st.info(f"**Agent**: {add_agent}")
|
||
st.info(f"**类型**: {memory_type}")
|
||
st.info(f"**内容**: {memory_content}")
|
||
st.info(f"**重要性**: {importance}/10")
|
||
if tags:
|
||
st.info(f"**标签**: {tags}")
|
||
else:
|
||
st.info("输入记忆内容以查看预览")
|
||
|
||
with tab4:
|
||
st.header("📊 统计信息")
|
||
|
||
if st.button("🔄 刷新统计", key="refresh_stats"):
|
||
st.rerun()
|
||
|
||
# 获取统计信息
|
||
backend, _ = get_memory_backend_cached()
|
||
if backend:
|
||
stats_data = []
|
||
|
||
for immortal in EIGHT_IMMORTALS:
|
||
try:
|
||
# 获取记忆数量
|
||
memories = run_async(backend.search_memories(immortal, "", limit=1000))
|
||
memory_count = len(memories) if memories else 0
|
||
|
||
# 获取agent context
|
||
context = run_async(backend.get_agent_context(immortal))
|
||
status = "活跃" if context else "未初始化"
|
||
|
||
stats_data.append({
|
||
"Agent": immortal,
|
||
"状态": status,
|
||
"记忆数量": memory_count,
|
||
"最后更新": datetime.now().strftime("%Y-%m-%d %H:%M:%S")
|
||
})
|
||
except Exception as e:
|
||
stats_data.append({
|
||
"Agent": immortal,
|
||
"状态": "错误",
|
||
"记忆数量": 0,
|
||
"最后更新": f"错误: {str(e)[:30]}..."
|
||
})
|
||
|
||
# 显示统计表格
|
||
st.dataframe(stats_data, use_container_width=True)
|
||
|
||
# 显示汇总信息
|
||
col1, col2, col3, col4 = st.columns(4)
|
||
|
||
total_agents = len(EIGHT_IMMORTALS)
|
||
active_agents = sum(1 for item in stats_data if item["状态"] == "活跃")
|
||
total_memories = sum(item["记忆数量"] for item in stats_data)
|
||
avg_memories = total_memories / total_agents if total_agents > 0 else 0
|
||
|
||
with col1:
|
||
st.metric("总Agent数", total_agents)
|
||
|
||
with col2:
|
||
st.metric("活跃Agent数", active_agents)
|
||
|
||
with col3:
|
||
st.metric("总记忆数", total_memories)
|
||
|
||
with col4:
|
||
st.metric("平均记忆数", f"{avg_memories:.1f}")
|
||
|
||
# 页脚
|
||
st.markdown("---")
|
||
st.markdown(
|
||
"""
|
||
<div style='text-align: center; color: #666;'>
|
||
🧠 Vertex AI Memory Bank Web界面 |
|
||
<a href='https://cloud.google.com/vertex-ai/generative-ai/docs/agent-engine/memory-bank/overview' target='_blank'>官方文档</a>
|
||
</div>
|
||
""",
|
||
unsafe_allow_html=True
|
||
)
|
||
|
||
# 使用说明
|
||
with st.expander("📖 使用说明"):
|
||
st.markdown("""
|
||
### 功能说明
|
||
|
||
1. **Memory Bank列表**: 查看所有八仙角色的Memory Bank状态和记忆数量
|
||
2. **搜索记忆**: 在指定Agent的记忆中搜索特定内容
|
||
3. **添加记忆**: 为Agent添加新的记忆,支持不同类型和元数据
|
||
4. **统计信息**: 查看所有Agent的统计数据和汇总信息
|
||
|
||
### 使用前准备
|
||
|
||
1. 确保已激活虚拟环境: `source venv/bin/activate`
|
||
2. 确保已设置Google Cloud认证: `gcloud auth application-default login`
|
||
3. 运行此界面: `streamlit run memory_bank_web_interface.py`
|
||
|
||
### 注意事项
|
||
|
||
- Memory Bank目前仅在us-central1区域可用
|
||
- 搜索功能支持模糊匹配
|
||
- 添加的记忆会立即生效
|
||
- 统计信息实时更新
|
||
""") |