Files
Cyrene/backend/ai-core/internal/host/sandbox_test.go
T
AskaEth 91c9ee4b2d fix: 修复 AI 回复无法送达发送者 + 重复消息 + action角色泄露 + OS环境支持
广播逻辑重构:
- AI 回复 (stream_start/response/stream_segments/multi_message/stream_end) 改用 broadcastToUser 发送给所有客户端
- 用户消息回显保持 broadcastToUserExcept 排除发送者

消息去重与角色修复:
- CacheMessage(user) 移至回复生成后,避免本轮 LLM 调用出现重复用户消息
- action 角色消息在 DB 存储时映射为 assistant,DeepSeek 等模型不支持自定义角色
- stream_end defer 机制确保错误路径也会终止客户端思考指示器

OS 完整环境支持:
- host 包重构为 HostBackend 接口 + Direct/WSL/Docker 三种后端
- 新增 os_exec/os_file/os_system 工具供 AI 在完整 Linux 环境中自由操作

其他:
- 视觉模型注入 + 图片预处理后清空 Images 避免传给 Chat 模型
- 图片 URL 相对路径→绝对 URL 转换
- DevTools 链路追踪页面 + 重启修复
- 记忆搜索模糊匹配增强
- 后台思考定时调度支持
- 管理后台页面 (模型配置/用户管理等)
- docs/api 更新广播机制说明

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-29 12:46:17 +08:00

134 lines
3.5 KiB
Go

package host
import (
"context"
"os"
"path/filepath"
"testing"
"time"
)
func TestSandboxExec(t *testing.T) {
cfg := DefaultSandboxConfig()
cfg.AllowedDirs = []string{os.TempDir()}
sandbox := NewSandbox(cfg)
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
defer cancel()
result, err := sandbox.Exec(ctx, "echo hello cyrene", os.TempDir(), 5*time.Second)
if err != nil {
t.Fatalf("exec failed: %v", err)
}
if result.ExitCode != 0 {
t.Fatalf("unexpected exit code: %d, stderr=%s", result.ExitCode, result.Stderr)
}
if result.Stdout == "" {
t.Fatal("expected output, got empty")
}
t.Logf("exec OK: stdout=%q, duration=%s", result.Stdout, result.Duration)
}
func TestSandboxBlockedCommand(t *testing.T) {
cfg := DefaultSandboxConfig()
sandbox := NewSandbox(cfg)
ctx := context.Background()
_, err := sandbox.Exec(ctx, "rm -rf /", os.TempDir(), 5*time.Second)
if err == nil {
t.Fatal("expected 'rm' to be blocked")
}
t.Logf("blocked command OK: %v", err)
}
func TestSandboxTimeout(t *testing.T) {
cfg := DefaultSandboxConfig()
cfg.AllowedCommands = append(cfg.AllowedCommands, "sleep")
sandbox := NewSandbox(cfg)
ctx := context.Background()
result, err := sandbox.Exec(ctx, "sleep 10", os.TempDir(), 1*time.Second)
if err == nil {
t.Fatal("expected timeout error")
}
if !result.TimedOut {
t.Fatal("expected TimedOut=true")
}
t.Logf("timeout OK: exit=%d, timed_out=%v", result.ExitCode, result.TimedOut)
}
func TestManagerFileOps(t *testing.T) {
cfg := DefaultSandboxConfig()
tmpDir := os.TempDir()
cfg.AllowedDirs = []string{tmpDir}
sandbox := NewSandbox(cfg)
mgr := NewManager(NewDirectBackend(sandbox))
mgr.SetAllowedDirs([]string{tmpDir})
testPath := filepath.Join(tmpDir, "cyrene-test-file.txt")
err := mgr.WriteFile(testPath, "Hello from Cyrene host manager!", 1024*1024)
if err != nil {
t.Fatalf("write failed: %v", err)
}
defer os.Remove(testPath)
content, err := mgr.ReadFile(testPath, 1024*1024)
if err != nil {
t.Fatalf("read failed: %v", err)
}
if content != "Hello from Cyrene host manager!" {
t.Fatalf("content mismatch: %q", content)
}
t.Logf("file read/write OK: %q", content)
entries, err := mgr.ListDir(tmpDir)
if err != nil {
t.Fatalf("listdir failed: %v", err)
}
found := false
for _, e := range entries {
if e.Name == "cyrene-test-file.txt" {
found = true
break
}
}
if !found {
t.Fatal("expected test file in directory listing")
}
t.Logf("listdir OK: %d entries", len(entries))
}
func TestManagerSystemInfo(t *testing.T) {
cfg := DefaultSandboxConfig()
sandbox := NewSandbox(cfg)
mgr := NewManager(NewDirectBackend(sandbox))
info := mgr.SystemInfo()
if info["hostname"] == nil || info["hostname"] == "" {
t.Fatal("expected hostname in system info")
}
if info["os"] == nil || info["os"] == "" {
t.Fatal("expected os in system info")
}
if info["arch"] == nil || info["arch"] == "" {
t.Fatal("expected arch in system info")
}
t.Logf("system info OK: os=%v arch=%v num_cpu=%v", info["os"], info["arch"], info["num_cpu"])
}
func TestPathValidation(t *testing.T) {
cfg := DefaultSandboxConfig()
cfg.AllowedDirs = []string{os.TempDir()}
sandbox := NewSandbox(cfg)
mgr := NewManager(NewDirectBackend(sandbox))
mgr.SetAllowedDirs([]string{os.TempDir()})
// Should fail: access outside allowed dirs
_, err := mgr.ReadFile("/etc/passwd", 1024)
if err == nil {
t.Fatal("expected path validation to block /etc/passwd")
}
t.Logf("path validation OK: blocked access to /etc/passwd")
}