mirror of
https://github.com/YspCoder/clawgo.git
synced 2026-04-13 05:37:29 +08:00
190 lines
7.2 KiB
Go
190 lines
7.2 KiB
Go
package agent
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/YspCoder/clawgo/pkg/ekg"
|
|
)
|
|
|
|
func TestSummarizePlannedTaskProgressBodyPreservesUsefulLines(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
body := "subagent 已写入 config.json。\npath: /root/.clawgo/config.json\nagent_id: tester\nrole: testing\ndisplay_name: Test Agent\ntool_allowlist: [filesystem shell]\nrouting_keywords: [test qa]\nsystem_prompt_file: agents/tester/AGENT.md"
|
|
out := summarizePlannedTaskProgressBody(body, 6, 320)
|
|
|
|
if !strings.Contains(out, "subagent 已写入 config.json。") {
|
|
t.Fatalf("expected title line, got:\n%s", out)
|
|
}
|
|
if !strings.Contains(out, "agent_id: tester") {
|
|
t.Fatalf("expected agent id line, got:\n%s", out)
|
|
}
|
|
if strings.Contains(out, "subagent 已写入 config.json。 path:") {
|
|
t.Fatalf("expected multi-line formatting, got:\n%s", out)
|
|
}
|
|
}
|
|
|
|
func TestEKGHintForTaskRequiresStrongMatchAndStaysCompact(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
workspace := t.TempDir()
|
|
memoryDir := filepath.Join(workspace, "memory")
|
|
if err := os.MkdirAll(memoryDir, 0o755); err != nil {
|
|
t.Fatalf("mkdir memory: %v", err)
|
|
}
|
|
logText := "open /srv/app/config.yaml: permission denied after deploy 42"
|
|
taskAudit := []string{
|
|
fmt.Sprintf(`{"task_id":"task-1","status":"error","source":"planner","channel":"chat","input_preview":"check nginx logs quickly","log":"%s"}`, logText),
|
|
fmt.Sprintf(`{"task_id":"task-2","status":"error","source":"planner","channel":"chat","input_preview":"deploy config service restart on cluster","log":"%s"}`, logText),
|
|
}
|
|
if err := os.WriteFile(filepath.Join(memoryDir, "task-audit.jsonl"), []byte(strings.Join(taskAudit, "\n")+"\n"), 0o644); err != nil {
|
|
t.Fatalf("write task audit: %v", err)
|
|
}
|
|
errSig := ekg.NormalizeErrorSignature(logText)
|
|
ekgEvents := []string{
|
|
fmt.Sprintf(`{"task_id":"task-2","status":"error","errsig":"%s","log":"%s"}`, errSig, logText),
|
|
fmt.Sprintf(`{"task_id":"task-2","status":"error","errsig":"%s","log":"%s"}`, errSig, logText),
|
|
fmt.Sprintf(`{"task_id":"task-2","status":"error","errsig":"%s","log":"%s"}`, errSig, logText),
|
|
}
|
|
if err := os.WriteFile(filepath.Join(memoryDir, "ekg-events.jsonl"), []byte(strings.Join(ekgEvents, "\n")+"\n"), 0o644); err != nil {
|
|
t.Fatalf("write ekg events: %v", err)
|
|
}
|
|
|
|
al := &AgentLoop{workspace: workspace, ekg: ekg.New(workspace)}
|
|
hint := al.ekgHintForTask(plannedTask{Content: "deploy config service restart after rollout"})
|
|
if hint == "" {
|
|
t.Fatalf("expected compact ekg hint")
|
|
}
|
|
if !strings.Contains(hint, "repeat_errsig=") || !strings.Contains(hint, "backoff=300s") {
|
|
t.Fatalf("expected compact fields, got: %s", hint)
|
|
}
|
|
if strings.Contains(strings.ToLower(hint), "last error") || strings.Contains(hint, logText) {
|
|
t.Fatalf("expected raw error log to be omitted, got: %s", hint)
|
|
}
|
|
}
|
|
|
|
func TestEKGHintForTaskSkipsWeakTaskMatch(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
workspace := t.TempDir()
|
|
memoryDir := filepath.Join(workspace, "memory")
|
|
if err := os.MkdirAll(memoryDir, 0o755); err != nil {
|
|
t.Fatalf("mkdir memory: %v", err)
|
|
}
|
|
logText := "dial tcp 10.0.0.8:443: i/o timeout"
|
|
taskAudit := `{"task_id":"task-3","status":"error","source":"planner","channel":"chat","input_preview":"investigate cache timeout","log":"dial tcp 10.0.0.8:443: i/o timeout"}`
|
|
if err := os.WriteFile(filepath.Join(memoryDir, "task-audit.jsonl"), []byte(taskAudit+"\n"), 0o644); err != nil {
|
|
t.Fatalf("write task audit: %v", err)
|
|
}
|
|
errSig := ekg.NormalizeErrorSignature(logText)
|
|
ekgEvents := []string{
|
|
fmt.Sprintf(`{"task_id":"task-3","status":"error","errsig":"%s","log":"%s"}`, errSig, logText),
|
|
fmt.Sprintf(`{"task_id":"task-3","status":"error","errsig":"%s","log":"%s"}`, errSig, logText),
|
|
fmt.Sprintf(`{"task_id":"task-3","status":"error","errsig":"%s","log":"%s"}`, errSig, logText),
|
|
}
|
|
if err := os.WriteFile(filepath.Join(memoryDir, "ekg-events.jsonl"), []byte(strings.Join(ekgEvents, "\n")+"\n"), 0o644); err != nil {
|
|
t.Fatalf("write ekg events: %v", err)
|
|
}
|
|
|
|
al := &AgentLoop{workspace: workspace, ekg: ekg.New(workspace)}
|
|
hint := al.ekgHintForTask(plannedTask{Content: "cache rebuild"})
|
|
if hint != "" {
|
|
t.Fatalf("expected weak match to skip ekg hint, got: %s", hint)
|
|
}
|
|
}
|
|
|
|
func TestCompactMemoryHintDropsVerboseScaffolding(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
raw := "Found 1 memories for 'deploy config' (namespace=main):\n\nSource: memory/2026-03-10.md#L12-L18\nDeploy must restart config service after updating the file.\nValidate permissions before rollout.\n\n"
|
|
got := compactMemoryHint(raw, 120)
|
|
if strings.Contains(strings.ToLower(got), "found 1 memories") {
|
|
t.Fatalf("expected summary header removed, got: %s", got)
|
|
}
|
|
if !strings.Contains(got, "src=memory/2026-03-10.md#L12-L18") {
|
|
t.Fatalf("expected source to remain compactly, got: %s", got)
|
|
}
|
|
if !strings.Contains(got, "Deploy must restart config service") {
|
|
t.Fatalf("expected main snippet retained, got: %s", got)
|
|
}
|
|
}
|
|
|
|
func TestDedupeTaskPromptHintsDropsRepeatedContext(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
seen := map[string]struct{}{}
|
|
first := taskPromptHints{ekg: "repeat_errsig=x; backoff=300s", memory: "src=memory/a.md#L1-L2 | restart service"}
|
|
second := taskPromptHints{ekg: "repeat_errsig=x; backoff=300s", memory: "src=memory/a.md#L1-L2 | restart service"}
|
|
|
|
dedupeTaskPromptHints(&first, seen)
|
|
dedupeTaskPromptHints(&second, seen)
|
|
|
|
if first.ekg == "" || first.memory == "" {
|
|
t.Fatalf("expected first hint set to remain: %+v", first)
|
|
}
|
|
if second.ekg != "" || second.memory != "" {
|
|
t.Fatalf("expected repeated hints removed: %+v", second)
|
|
}
|
|
}
|
|
|
|
func TestRenderTaskPromptWithHintsKeepsCompactShape(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
got := renderTaskPromptWithHints("deploy config service", taskPromptHints{
|
|
ekg: "repeat_errsig=perm; backoff=300s",
|
|
memory: "src=memory/x.md#L1-L2 | restart after change",
|
|
})
|
|
if !strings.Contains(got, "Task Context:\nEKG: repeat_errsig=perm; backoff=300s\nMemory: src=memory/x.md#L1-L2 | restart after change\nTask:\ndeploy config service") {
|
|
t.Fatalf("unexpected prompt shape: %s", got)
|
|
}
|
|
}
|
|
|
|
func TestBuildPlannedTaskPromptTracksExtraChars(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
prompt := buildPlannedTaskPrompt("deploy config service", taskPromptHints{
|
|
ekg: "repeat_errsig=perm; backoff=300s",
|
|
memory: "src=memory/x.md#L1-L2 | restart after change",
|
|
})
|
|
if prompt.content == "" {
|
|
t.Fatalf("expected prompt content")
|
|
}
|
|
if prompt.extraChars <= 0 || prompt.ekgChars <= 0 || prompt.memoryChars <= 0 {
|
|
t.Fatalf("expected prompt stats populated: %+v", prompt)
|
|
}
|
|
}
|
|
|
|
func TestApplyPromptBudgetPrefersEKGOverMemory(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
hints := taskPromptHints{
|
|
ekg: "repeat_errsig=perm; backoff=300s",
|
|
memory: "src=memory/x.md#L1-L2 | restart after change",
|
|
}
|
|
remaining := estimateHintChars(hints) - len(hints.memory)
|
|
applyPromptBudget(&hints, &remaining)
|
|
if hints.ekg == "" {
|
|
t.Fatalf("expected ekg retained")
|
|
}
|
|
if hints.memory != "" {
|
|
t.Fatalf("expected memory dropped under budget pressure: %+v", hints)
|
|
}
|
|
}
|
|
|
|
func TestApplyPromptBudgetDropsAllWhenBudgetExhausted(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
hints := taskPromptHints{
|
|
ekg: "repeat_errsig=perm; backoff=300s",
|
|
memory: "src=memory/x.md#L1-L2 | restart after change",
|
|
}
|
|
remaining := 8
|
|
applyPromptBudget(&hints, &remaining)
|
|
if hints.ekg != "" || hints.memory != "" {
|
|
t.Fatalf("expected all hints removed under tiny budget: %+v", hints)
|
|
}
|
|
}
|