11 work units built in parallel and merged: Agent handlers (Phase 2): - P2.2 Deploy: pull images, stop/remove/run containers, update registry - P2.3 Lifecycle: stop/start/restart with desired_state tracking - P2.4 Status: list (registry), live check (runtime), get status (drift+events) - P2.5 Sync: receive desired state, reconcile unmanaged containers - P2.6 File transfer: push/pull scoped to /srv/<service>/, path validation - P2.7 Adopt: match <service>-* containers, derive component names - P2.8 Monitor: continuous watch loop, drift/flap alerting, event pruning - P2.9 Snapshot: VACUUM INTO database backup command CLI commands (Phase 3): - P3.2 Login, P3.3 Deploy, P3.4 Stop/Start/Restart - P3.5 List/Ps/Status, P3.6 Sync, P3.7 Adopt - P3.8 Service show/edit/export, P3.9 Push/Pull, P3.10 Node list/add/remove Deployment artifacts (Phase 4): - Systemd units (agent service + backup timer) - Example configs (CLI + agent) - Install script (idempotent) All packages: build, vet, lint (0 issues), test (all pass). Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
83 lines
1.5 KiB
Go
83 lines
1.5 KiB
Go
package agent
|
|
|
|
import (
|
|
"testing"
|
|
)
|
|
|
|
func TestValidatePath(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
service string
|
|
path string
|
|
want string
|
|
wantErr bool
|
|
}{
|
|
{
|
|
name: "valid simple path",
|
|
service: "mcr",
|
|
path: "mcr.toml",
|
|
want: "/srv/mcr/mcr.toml",
|
|
},
|
|
{
|
|
name: "valid nested path",
|
|
service: "mcr",
|
|
path: "certs/cert.pem",
|
|
want: "/srv/mcr/certs/cert.pem",
|
|
},
|
|
{
|
|
name: "reject traversal",
|
|
service: "mcr",
|
|
path: "../etc/passwd",
|
|
wantErr: true,
|
|
},
|
|
{
|
|
name: "reject absolute path",
|
|
service: "mcr",
|
|
path: "/etc/passwd",
|
|
wantErr: true,
|
|
},
|
|
{
|
|
name: "reject empty service",
|
|
service: "",
|
|
path: "mcr.toml",
|
|
wantErr: true,
|
|
},
|
|
{
|
|
name: "reject empty path",
|
|
service: "mcr",
|
|
path: "",
|
|
wantErr: true,
|
|
},
|
|
{
|
|
name: "reject double dot in middle",
|
|
service: "mcr",
|
|
path: "certs/../../etc/passwd",
|
|
wantErr: true,
|
|
},
|
|
{
|
|
name: "valid deeply nested",
|
|
service: "metacrypt",
|
|
path: "data/keys/primary.key",
|
|
want: "/srv/metacrypt/data/keys/primary.key",
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
got, err := validatePath(tt.service, tt.path)
|
|
if tt.wantErr {
|
|
if err == nil {
|
|
t.Fatalf("expected error, got path %q", got)
|
|
}
|
|
return
|
|
}
|
|
if err != nil {
|
|
t.Fatalf("unexpected error: %v", err)
|
|
}
|
|
if got != tt.want {
|
|
t.Errorf("got %q, want %q", got, tt.want)
|
|
}
|
|
})
|
|
}
|
|
}
|