agent/pkg/agentic/dispatch_extra_test.go
Snider e0a6b33729 test: issue tracking + broadcast with Forge mocks — agentic 72.3%
Add httptest mocks for startIssueTracking/stopIssueTracking with Forge,
broadcastStart/broadcastComplete with Core IPC.

Co-Authored-By: Virgil <virgil@lethean.io>
2026-03-25 01:15:19 +00:00

401 lines
12 KiB
Go

// SPDX-License-Identifier: EUPL-1.2
package agentic
import (
"encoding/json"
"net/http"
"net/http/httptest"
"os"
"path/filepath"
"testing"
"time"
core "dappco.re/go/core"
"dappco.re/go/core/forge"
"github.com/stretchr/testify/assert"
)
// --- agentOutputFile ---
func TestAgentOutputFile_Good(t *testing.T) {
assert.Contains(t, agentOutputFile("/ws", "codex"), ".meta/agent-codex.log")
assert.Contains(t, agentOutputFile("/ws", "claude:opus"), ".meta/agent-claude.log")
assert.Contains(t, agentOutputFile("/ws", "gemini:flash"), ".meta/agent-gemini.log")
}
// --- detectFinalStatus ---
func TestDetectFinalStatus_Good_Completed(t *testing.T) {
dir := t.TempDir()
status, question := detectFinalStatus(dir, 0, "completed")
assert.Equal(t, "completed", status)
assert.Empty(t, question)
}
func TestDetectFinalStatus_Good_Blocked(t *testing.T) {
dir := t.TempDir()
os.WriteFile(filepath.Join(dir, "BLOCKED.md"), []byte("Need API key for external service"), 0o644)
status, question := detectFinalStatus(dir, 0, "completed")
assert.Equal(t, "blocked", status)
assert.Equal(t, "Need API key for external service", question)
}
func TestDetectFinalStatus_Good_BlockedEmpty(t *testing.T) {
dir := t.TempDir()
// BLOCKED.md exists but is empty — should NOT be treated as blocked
os.WriteFile(filepath.Join(dir, "BLOCKED.md"), []byte(" \n "), 0o644)
status, _ := detectFinalStatus(dir, 0, "completed")
assert.Equal(t, "completed", status)
}
func TestDetectFinalStatus_Good_FailedExitCode(t *testing.T) {
dir := t.TempDir()
status, question := detectFinalStatus(dir, 1, "completed")
assert.Equal(t, "failed", status)
assert.Contains(t, question, "code 1")
}
func TestDetectFinalStatus_Good_FailedKilled(t *testing.T) {
dir := t.TempDir()
status, _ := detectFinalStatus(dir, 0, "killed")
assert.Equal(t, "failed", status)
}
func TestDetectFinalStatus_Good_FailedStatus(t *testing.T) {
dir := t.TempDir()
status, _ := detectFinalStatus(dir, 0, "failed")
assert.Equal(t, "failed", status)
}
func TestDetectFinalStatus_Good_BlockedTakesPrecedence(t *testing.T) {
dir := t.TempDir()
// Agent wrote BLOCKED.md AND exited non-zero — blocked takes precedence
os.WriteFile(filepath.Join(dir, "BLOCKED.md"), []byte("Need help"), 0o644)
status, question := detectFinalStatus(dir, 1, "failed")
assert.Equal(t, "blocked", status)
assert.Equal(t, "Need help", question)
}
// --- trackFailureRate ---
func TestTrackFailureRate_Good_SuccessResetsCount(t *testing.T) {
s := &PrepSubsystem{
backoff: make(map[string]time.Time),
failCount: map[string]int{"codex": 2},
}
triggered := s.trackFailureRate("codex", "completed", time.Now().Add(-10*time.Second))
assert.False(t, triggered)
assert.Equal(t, 0, s.failCount["codex"])
}
func TestTrackFailureRate_Good_SlowFailureResetsCount(t *testing.T) {
s := &PrepSubsystem{
backoff: make(map[string]time.Time),
failCount: map[string]int{"codex": 2},
}
// Started 5 minutes ago = slow failure
triggered := s.trackFailureRate("codex", "failed", time.Now().Add(-5*time.Minute))
assert.False(t, triggered)
assert.Equal(t, 0, s.failCount["codex"])
}
func TestTrackFailureRate_Good_FastFailureIncrementsCount(t *testing.T) {
s := &PrepSubsystem{
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
// Started 10 seconds ago = fast failure
triggered := s.trackFailureRate("codex", "failed", time.Now().Add(-10*time.Second))
assert.False(t, triggered)
assert.Equal(t, 1, s.failCount["codex"])
}
func TestTrackFailureRate_Good_ThreeFailsTriggersBackoff(t *testing.T) {
s := &PrepSubsystem{
backoff: make(map[string]time.Time),
failCount: map[string]int{"codex": 2}, // already 2 fast failures
}
triggered := s.trackFailureRate("codex", "failed", time.Now().Add(-10*time.Second))
assert.True(t, triggered)
assert.True(t, time.Now().Before(s.backoff["codex"]))
}
func TestTrackFailureRate_Good_ModelVariantUsesPool(t *testing.T) {
s := &PrepSubsystem{
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
s.trackFailureRate("codex:gpt-5.4", "failed", time.Now().Add(-10*time.Second))
assert.Equal(t, 1, s.failCount["codex"], "should track by base agent pool")
}
// --- startIssueTracking / stopIssueTracking ---
func TestStartIssueTracking_Good_NoForge(t *testing.T) {
s := &PrepSubsystem{
forge: nil,
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
s.startIssueTracking(t.TempDir())
}
func TestStopIssueTracking_Good_NoForge(t *testing.T) {
s := &PrepSubsystem{
forge: nil,
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
s.stopIssueTracking(t.TempDir())
}
func TestStartIssueTracking_Good_NoIssue(t *testing.T) {
dir := t.TempDir()
st := &WorkspaceStatus{Status: "running", Repo: "test"}
data, _ := json.Marshal(st)
os.WriteFile(filepath.Join(dir, "status.json"), data, 0o644)
s := &PrepSubsystem{
forge: nil,
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
s.startIssueTracking(dir)
}
func TestStartIssueTracking_Good_NoStatusFile(t *testing.T) {
s := &PrepSubsystem{
forge: nil,
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
// No status.json — should return early
s.startIssueTracking(t.TempDir())
}
func TestStartIssueTracking_Good_WithForge(t *testing.T) {
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
w.WriteHeader(201) // Forge stopwatch start returns 201
}))
t.Cleanup(srv.Close)
dir := t.TempDir()
st := &WorkspaceStatus{Status: "running", Repo: "go-io", Org: "core", Issue: 15}
data, _ := json.Marshal(st)
os.WriteFile(filepath.Join(dir, "status.json"), data, 0o644)
s := &PrepSubsystem{
forge: forge.NewForge(srv.URL, "test-token"),
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
s.startIssueTracking(dir)
}
func TestStopIssueTracking_Good_WithForge(t *testing.T) {
srv := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
w.WriteHeader(204)
}))
t.Cleanup(srv.Close)
dir := t.TempDir()
st := &WorkspaceStatus{Status: "completed", Repo: "go-io", Issue: 10}
data, _ := json.Marshal(st)
os.WriteFile(filepath.Join(dir, "status.json"), data, 0o644)
s := &PrepSubsystem{
forge: forge.NewForge(srv.URL, "test-token"),
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
s.stopIssueTracking(dir)
}
// --- broadcastStart / broadcastComplete ---
func TestBroadcastStart_Good_NoCore(t *testing.T) {
dir := t.TempDir()
t.Setenv("CORE_WORKSPACE", dir)
s := &PrepSubsystem{
core: nil,
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
s.broadcastStart("codex", dir)
}
func TestBroadcastStart_Good_WithCore(t *testing.T) {
root := t.TempDir()
t.Setenv("CORE_WORKSPACE", root)
wsDir := filepath.Join(root, "workspace", "ws-test")
os.MkdirAll(wsDir, 0o755)
st := &WorkspaceStatus{Repo: "go-io", Agent: "codex"}
data, _ := json.Marshal(st)
os.WriteFile(filepath.Join(wsDir, "status.json"), data, 0o644)
c := core.New()
s := &PrepSubsystem{
core: c,
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
s.broadcastStart("codex", wsDir)
}
func TestBroadcastComplete_Good_NoCore(t *testing.T) {
dir := t.TempDir()
t.Setenv("CORE_WORKSPACE", dir)
s := &PrepSubsystem{
core: nil,
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
s.broadcastComplete("codex", dir, "completed")
}
func TestBroadcastComplete_Good_WithCore(t *testing.T) {
root := t.TempDir()
t.Setenv("CORE_WORKSPACE", root)
wsDir := filepath.Join(root, "workspace", "ws-test")
os.MkdirAll(wsDir, 0o755)
st := &WorkspaceStatus{Repo: "go-io", Agent: "codex"}
data, _ := json.Marshal(st)
os.WriteFile(filepath.Join(wsDir, "status.json"), data, 0o644)
c := core.New()
s := &PrepSubsystem{
core: c,
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
s.broadcastComplete("codex", wsDir, "completed")
}
// --- onAgentComplete ---
func TestOnAgentComplete_Good_Completed(t *testing.T) {
root := t.TempDir()
t.Setenv("CORE_WORKSPACE", root)
wsDir := filepath.Join(root, "ws-test")
repoDir := filepath.Join(wsDir, "repo")
metaDir := filepath.Join(wsDir, ".meta")
os.MkdirAll(repoDir, 0o755)
os.MkdirAll(metaDir, 0o755)
// Write initial status
st := &WorkspaceStatus{Status: "running", Repo: "go-io", Agent: "codex", StartedAt: time.Now()}
data, _ := json.Marshal(st)
os.WriteFile(filepath.Join(wsDir, "status.json"), data, 0o644)
s := &PrepSubsystem{
core: nil,
forge: nil,
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
outputFile := filepath.Join(metaDir, "agent-codex.log")
s.onAgentComplete("codex", wsDir, outputFile, 0, "completed", "test output")
// Verify status was updated
updated, err := ReadStatus(wsDir)
assert.NoError(t, err)
assert.Equal(t, "completed", updated.Status)
assert.Equal(t, 0, updated.PID)
assert.Empty(t, updated.Question)
// Verify output was written
content, _ := os.ReadFile(outputFile)
assert.Equal(t, "test output", string(content))
}
func TestOnAgentComplete_Good_Failed(t *testing.T) {
root := t.TempDir()
t.Setenv("CORE_WORKSPACE", root)
wsDir := filepath.Join(root, "ws-fail")
repoDir := filepath.Join(wsDir, "repo")
metaDir := filepath.Join(wsDir, ".meta")
os.MkdirAll(repoDir, 0o755)
os.MkdirAll(metaDir, 0o755)
st := &WorkspaceStatus{Status: "running", Repo: "go-io", Agent: "codex", StartedAt: time.Now()}
data, _ := json.Marshal(st)
os.WriteFile(filepath.Join(wsDir, "status.json"), data, 0o644)
s := &PrepSubsystem{
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
s.onAgentComplete("codex", wsDir, filepath.Join(metaDir, "agent-codex.log"), 1, "failed", "error output")
updated, _ := ReadStatus(wsDir)
assert.Equal(t, "failed", updated.Status)
assert.Contains(t, updated.Question, "code 1")
}
func TestOnAgentComplete_Good_Blocked(t *testing.T) {
root := t.TempDir()
t.Setenv("CORE_WORKSPACE", root)
wsDir := filepath.Join(root, "ws-blocked")
repoDir := filepath.Join(wsDir, "repo")
metaDir := filepath.Join(wsDir, ".meta")
os.MkdirAll(repoDir, 0o755)
os.MkdirAll(metaDir, 0o755)
// Create BLOCKED.md
os.WriteFile(filepath.Join(repoDir, "BLOCKED.md"), []byte("Need credentials"), 0o644)
st := &WorkspaceStatus{Status: "running", Repo: "go-io", Agent: "codex", StartedAt: time.Now()}
data, _ := json.Marshal(st)
os.WriteFile(filepath.Join(wsDir, "status.json"), data, 0o644)
s := &PrepSubsystem{
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
s.onAgentComplete("codex", wsDir, filepath.Join(metaDir, "agent-codex.log"), 0, "completed", "")
updated, _ := ReadStatus(wsDir)
assert.Equal(t, "blocked", updated.Status)
assert.Equal(t, "Need credentials", updated.Question)
}
func TestOnAgentComplete_Good_EmptyOutput(t *testing.T) {
root := t.TempDir()
t.Setenv("CORE_WORKSPACE", root)
wsDir := filepath.Join(root, "ws-empty")
repoDir := filepath.Join(wsDir, "repo")
metaDir := filepath.Join(wsDir, ".meta")
os.MkdirAll(repoDir, 0o755)
os.MkdirAll(metaDir, 0o755)
st := &WorkspaceStatus{Status: "running", Repo: "test", Agent: "codex", StartedAt: time.Now()}
data, _ := json.Marshal(st)
os.WriteFile(filepath.Join(wsDir, "status.json"), data, 0o644)
s := &PrepSubsystem{
backoff: make(map[string]time.Time),
failCount: make(map[string]int),
}
outputFile := filepath.Join(metaDir, "agent-codex.log")
s.onAgentComplete("codex", wsDir, outputFile, 0, "completed", "")
// Output file should NOT be created for empty output
_, err := os.Stat(outputFile)
assert.True(t, os.IsNotExist(err))
}