mirror of
https://github.com/YspCoder/clawgo.git
synced 2026-05-04 17:57:38 +08:00
feat: add subagent profiles, memory namespaces, and webui management
This commit is contained in:
116
pkg/tools/subagent_profile_test.go
Normal file
116
pkg/tools/subagent_profile_test.go
Normal file
@@ -0,0 +1,116 @@
|
||||
package tools
|
||||
|
||||
import (
|
||||
"context"
|
||||
"strings"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestSubagentProfileStoreNormalization(t *testing.T) {
|
||||
t.Parallel()
|
||||
|
||||
store := NewSubagentProfileStore(t.TempDir())
|
||||
saved, err := store.Upsert(SubagentProfile{
|
||||
AgentID: "Coder Agent",
|
||||
Name: " ",
|
||||
Role: "coding",
|
||||
MemoryNamespace: "My Namespace",
|
||||
ToolAllowlist: []string{" Read_File ", "memory_search", "READ_FILE"},
|
||||
Status: "ACTIVE",
|
||||
})
|
||||
if err != nil {
|
||||
t.Fatalf("upsert failed: %v", err)
|
||||
}
|
||||
|
||||
if saved.AgentID != "coder-agent" {
|
||||
t.Fatalf("unexpected agent_id: %s", saved.AgentID)
|
||||
}
|
||||
if saved.Name != "coder-agent" {
|
||||
t.Fatalf("unexpected default name: %s", saved.Name)
|
||||
}
|
||||
if saved.MemoryNamespace != "my-namespace" {
|
||||
t.Fatalf("unexpected memory namespace: %s", saved.MemoryNamespace)
|
||||
}
|
||||
if len(saved.ToolAllowlist) != 2 {
|
||||
t.Fatalf("unexpected allowlist size: %d (%v)", len(saved.ToolAllowlist), saved.ToolAllowlist)
|
||||
}
|
||||
for _, tool := range saved.ToolAllowlist {
|
||||
if tool != strings.ToLower(tool) {
|
||||
t.Fatalf("tool allowlist should be lowercase, got: %s", tool)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestSubagentManagerSpawnRejectsDisabledProfile(t *testing.T) {
|
||||
t.Parallel()
|
||||
|
||||
workspace := t.TempDir()
|
||||
manager := NewSubagentManager(nil, workspace, nil, nil)
|
||||
manager.SetRunFunc(func(ctx context.Context, task *SubagentTask) (string, error) {
|
||||
return "ok", nil
|
||||
})
|
||||
store := manager.ProfileStore()
|
||||
if store == nil {
|
||||
t.Fatalf("expected profile store to be available")
|
||||
}
|
||||
if _, err := store.Upsert(SubagentProfile{
|
||||
AgentID: "writer",
|
||||
Status: "disabled",
|
||||
}); err != nil {
|
||||
t.Fatalf("failed to seed profile: %v", err)
|
||||
}
|
||||
|
||||
_, err := manager.Spawn(context.Background(), SubagentSpawnOptions{
|
||||
Task: "Write docs",
|
||||
AgentID: "writer",
|
||||
OriginChannel: "cli",
|
||||
OriginChatID: "direct",
|
||||
})
|
||||
if err == nil {
|
||||
t.Fatalf("expected disabled profile to block spawn")
|
||||
}
|
||||
}
|
||||
|
||||
func TestSubagentManagerSpawnResolvesProfileByRole(t *testing.T) {
|
||||
t.Parallel()
|
||||
|
||||
workspace := t.TempDir()
|
||||
manager := NewSubagentManager(nil, workspace, nil, nil)
|
||||
store := manager.ProfileStore()
|
||||
if store == nil {
|
||||
t.Fatalf("expected profile store to be available")
|
||||
}
|
||||
if _, err := store.Upsert(SubagentProfile{
|
||||
AgentID: "coder",
|
||||
Role: "coding",
|
||||
Status: "active",
|
||||
ToolAllowlist: []string{"read_file"},
|
||||
}); err != nil {
|
||||
t.Fatalf("failed to seed profile: %v", err)
|
||||
}
|
||||
|
||||
_, err := manager.Spawn(context.Background(), SubagentSpawnOptions{
|
||||
Task: "Implement feature",
|
||||
Role: "coding",
|
||||
OriginChannel: "cli",
|
||||
OriginChatID: "direct",
|
||||
})
|
||||
if err != nil {
|
||||
t.Fatalf("spawn failed: %v", err)
|
||||
}
|
||||
|
||||
tasks := manager.ListTasks()
|
||||
if len(tasks) != 1 {
|
||||
t.Fatalf("expected one task, got %d", len(tasks))
|
||||
}
|
||||
task := tasks[0]
|
||||
if task.AgentID != "coder" {
|
||||
t.Fatalf("expected agent_id to resolve to profile agent_id 'coder', got: %s", task.AgentID)
|
||||
}
|
||||
if task.Role != "coding" {
|
||||
t.Fatalf("expected task role to remain 'coding', got: %s", task.Role)
|
||||
}
|
||||
if len(task.ToolAllowlist) != 1 || task.ToolAllowlist[0] != "read_file" {
|
||||
t.Fatalf("expected allowlist from profile, got: %v", task.ToolAllowlist)
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user