feat: 🚀 增强会话管理与缓存机制 * 会话 ID 空值兜底,若 `conversation_id` 为空时自动生成 UUID * 在响应头写入 `X-Conversation-ID`,供前端使用,保持同一会话状态 perf: ⚡ 会话状态缓存优化 * 当缓存未命中但 DB 已确认/创建会话后,调用 `SetConversationStatus` 回写 Redis * 缓存写回失败时记录日志,不中断聊天主流程,确保业务流畅性 fix: 🐛 修复历史消息顺序问题与编译错误 * 修复历史消息顺序问题,保证返回的 N 条历史消息按时间正序喂给模型 * 通过反转 `created_at desc` 查询结果的切片,确保模型输入顺序正确 * 修复 `fmt.Errorf` 参数不匹配问题,修正编译错误 * 整理 `agent-cache.go` 为标准 UTF-8 编码,避免 Go 编译报错 `invalid UTF-8 encoding` feat: 🛠️ 独立构建 MCP 服务器 * 使用 `Codex` 构建独立于后端的 MCP 服务器,简化与 Codex 的协作 * 通过该服务器方便 Codex 直接测试和查看 Redis 与 MySQL 中的数据
54 lines
1.1 KiB
Go
54 lines
1.1 KiB
Go
package tools
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"sort"
|
|
)
|
|
|
|
type Tool interface {
|
|
Name() string
|
|
Description() string
|
|
InputSchema() map[string]any
|
|
Execute(ctx context.Context, args map[string]any) (map[string]any, error)
|
|
}
|
|
|
|
type Registry struct {
|
|
tools map[string]Tool
|
|
}
|
|
|
|
func NewRegistry(toolList ...Tool) (*Registry, error) {
|
|
r := &Registry{tools: make(map[string]Tool, len(toolList))}
|
|
for _, t := range toolList {
|
|
name := t.Name()
|
|
if _, exists := r.tools[name]; exists {
|
|
return nil, fmt.Errorf("duplicated tool name: %s", name)
|
|
}
|
|
r.tools[name] = t
|
|
}
|
|
return r, nil
|
|
}
|
|
|
|
func (r *Registry) Find(name string) (Tool, bool) {
|
|
t, ok := r.tools[name]
|
|
return t, ok
|
|
}
|
|
|
|
func (r *Registry) List() []map[string]any {
|
|
out := make([]map[string]any, 0, len(r.tools))
|
|
names := make([]string, 0, len(r.tools))
|
|
for name := range r.tools {
|
|
names = append(names, name)
|
|
}
|
|
sort.Strings(names)
|
|
for _, name := range names {
|
|
t := r.tools[name]
|
|
out = append(out, map[string]any{
|
|
"name": t.Name(),
|
|
"description": t.Description(),
|
|
"inputSchema": t.InputSchema(),
|
|
})
|
|
}
|
|
return out
|
|
}
|