feat: add safe agent writeback flow
This commit is contained in:
@@ -1,12 +1,14 @@
|
||||
package server
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"database/sql"
|
||||
"encoding/json"
|
||||
"net/http"
|
||||
"net/http/httptest"
|
||||
"os"
|
||||
"path/filepath"
|
||||
"strings"
|
||||
"testing"
|
||||
|
||||
"codex-agent-manager/internal/app"
|
||||
@@ -199,3 +201,113 @@ func TestReadOnlyEndpointsRejectUnsupportedMethods(t *testing.T) {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestAgentValidateEndpointReturnsDiffAndRejectsUnsupportedMethods(t *testing.T) {
|
||||
root := t.TempDir()
|
||||
agentsDir := filepath.Join(root, "agents")
|
||||
if err := os.MkdirAll(agentsDir, 0o755); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if err := os.WriteFile(filepath.Join(agentsDir, "backend.toml"), []byte(`name = "旧名称"`+"\n"), 0o644); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
req := httptest.NewRequest(http.MethodPost, "/api/agents/backend/validate", bytes.NewBufferString(`{"content":"name = \"新名称\"\n"}`))
|
||||
rec := httptest.NewRecorder()
|
||||
New(app.Config{CodexHome: root, HTTPAddr: "127.0.0.1:0"}).ServeHTTP(rec, req)
|
||||
|
||||
if rec.Code != http.StatusOK {
|
||||
t.Fatalf("status = %d, body = %s", rec.Code, rec.Body.String())
|
||||
}
|
||||
var body struct {
|
||||
Valid bool `json:"valid"`
|
||||
Diff string `json:"diff"`
|
||||
CurrentHash string `json:"currentHash"`
|
||||
TargetPath string `json:"targetPath"`
|
||||
}
|
||||
if err := json.Unmarshal(rec.Body.Bytes(), &body); err != nil {
|
||||
t.Fatalf("invalid json: %v", err)
|
||||
}
|
||||
if !body.Valid || body.CurrentHash == "" || body.TargetPath == "" || !strings.Contains(body.Diff, "新名称") {
|
||||
t.Fatalf("unexpected validate body: %#v", body)
|
||||
}
|
||||
|
||||
req = httptest.NewRequest(http.MethodGet, "/api/agents/backend/validate", nil)
|
||||
rec = httptest.NewRecorder()
|
||||
New(app.Config{CodexHome: root, HTTPAddr: "127.0.0.1:0"}).ServeHTTP(rec, req)
|
||||
if rec.Code != http.StatusMethodNotAllowed {
|
||||
t.Fatalf("GET validate status = %d, want %d", rec.Code, http.StatusMethodNotAllowed)
|
||||
}
|
||||
}
|
||||
|
||||
func TestAgentValidateEndpointReturnsBadRequestForInvalidBody(t *testing.T) {
|
||||
req := httptest.NewRequest(http.MethodPost, "/api/agents/backend/validate", bytes.NewBufferString(`{`))
|
||||
rec := httptest.NewRecorder()
|
||||
New(app.Config{CodexHome: t.TempDir(), HTTPAddr: "127.0.0.1:0"}).ServeHTTP(rec, req)
|
||||
|
||||
if rec.Code != http.StatusBadRequest {
|
||||
t.Fatalf("status = %d, body = %s", rec.Code, rec.Body.String())
|
||||
}
|
||||
}
|
||||
|
||||
func TestAgentWriteEndpointCreatesBackupAndRejectsConflicts(t *testing.T) {
|
||||
root := t.TempDir()
|
||||
agentsDir := filepath.Join(root, "agents")
|
||||
target := filepath.Join(agentsDir, "backend.toml")
|
||||
if err := os.MkdirAll(agentsDir, 0o755); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if err := os.WriteFile(target, []byte(`name = "旧名称"`+"\n"), 0o644); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
handler := New(app.Config{CodexHome: root, HTTPAddr: "127.0.0.1:0"})
|
||||
validateReq := httptest.NewRequest(http.MethodPost, "/api/agents/backend/validate", bytes.NewBufferString(`{"content":"name = \"新名称\"\n"}`))
|
||||
validateRec := httptest.NewRecorder()
|
||||
handler.ServeHTTP(validateRec, validateReq)
|
||||
if validateRec.Code != http.StatusOK {
|
||||
t.Fatalf("validate status = %d, body = %s", validateRec.Code, validateRec.Body.String())
|
||||
}
|
||||
var validation struct {
|
||||
CurrentHash string `json:"currentHash"`
|
||||
}
|
||||
if err := json.Unmarshal(validateRec.Body.Bytes(), &validation); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
|
||||
writeBody := `{"content":"name = \"新名称\"\n","expectedHash":"` + validation.CurrentHash + `"}`
|
||||
writeReq := httptest.NewRequest(http.MethodPost, "/api/agents/backend/write", bytes.NewBufferString(writeBody))
|
||||
writeRec := httptest.NewRecorder()
|
||||
handler.ServeHTTP(writeRec, writeReq)
|
||||
|
||||
if writeRec.Code != http.StatusOK {
|
||||
t.Fatalf("write status = %d, body = %s", writeRec.Code, writeRec.Body.String())
|
||||
}
|
||||
var written struct {
|
||||
Status string `json:"status"`
|
||||
TargetPath string `json:"targetPath"`
|
||||
BackupPath string `json:"backupPath"`
|
||||
}
|
||||
if err := json.Unmarshal(writeRec.Body.Bytes(), &written); err != nil {
|
||||
t.Fatal(err)
|
||||
}
|
||||
if written.Status != "written" || written.TargetPath != target || written.BackupPath == "" {
|
||||
t.Fatalf("unexpected write body: %#v", written)
|
||||
}
|
||||
if data, err := os.ReadFile(target); err != nil || string(data) != `name = "新名称"`+"\n" {
|
||||
t.Fatalf("target content = %q, err = %v", string(data), err)
|
||||
}
|
||||
|
||||
conflictReq := httptest.NewRequest(http.MethodPost, "/api/agents/backend/write", bytes.NewBufferString(writeBody))
|
||||
conflictRec := httptest.NewRecorder()
|
||||
handler.ServeHTTP(conflictRec, conflictReq)
|
||||
if conflictRec.Code != http.StatusConflict {
|
||||
t.Fatalf("conflict status = %d, want %d, body = %s", conflictRec.Code, http.StatusConflict, conflictRec.Body.String())
|
||||
}
|
||||
|
||||
getReq := httptest.NewRequest(http.MethodGet, "/api/agents/backend/write", nil)
|
||||
getRec := httptest.NewRecorder()
|
||||
handler.ServeHTTP(getRec, getReq)
|
||||
if getRec.Code != http.StatusMethodNotAllowed {
|
||||
t.Fatalf("GET write status = %d, want %d", getRec.Code, http.StatusMethodNotAllowed)
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user