Files
gastown/internal/cmd/audit_test.go
Steve Yegge 38fe3cbd32 Add gt audit command for provenance queries (gt-6r18e.8)
New command that queries work history across multiple sources:
- Git commits authored by an actor
- Beads created/closed by an actor
- Town log events (spawn, done, handoff, etc.)
- Activity feed events

Supports --actor for filtering, --since for time range, --json for
machine-readable output, and -n/--limit for result count.

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2025-12-30 16:43:39 -08:00

151 lines
3.5 KiB
Go

package cmd
import (
"testing"
"time"
)
func TestParseDuration(t *testing.T) {
tests := []struct {
input string
expected time.Duration
wantErr bool
}{
{"1h", time.Hour, false},
{"30m", 30 * time.Minute, false},
{"24h", 24 * time.Hour, false},
{"1d", 24 * time.Hour, false},
{"7d", 7 * 24 * time.Hour, false},
{"2s", 2 * time.Second, false},
{"invalid", 0, true},
}
for _, tt := range tests {
t.Run(tt.input, func(t *testing.T) {
got, err := parseDuration(tt.input)
if tt.wantErr {
if err == nil {
t.Errorf("parseDuration(%q) expected error, got nil", tt.input)
}
return
}
if err != nil {
t.Errorf("parseDuration(%q) unexpected error: %v", tt.input, err)
return
}
if got != tt.expected {
t.Errorf("parseDuration(%q) = %v, want %v", tt.input, got, tt.expected)
}
})
}
}
func TestExtractAuthorName(t *testing.T) {
tests := []struct {
input string
expected string
}{
{"gastown/crew/joe", "joe"},
{"gastown/polecats/toast", "toast"},
{"mayor", "mayor"},
{"gastown/witness", "witness"},
{"", ""},
}
for _, tt := range tests {
t.Run(tt.input, func(t *testing.T) {
got := extractAuthorName(tt.input)
if got != tt.expected {
t.Errorf("extractAuthorName(%q) = %q, want %q", tt.input, got, tt.expected)
}
})
}
}
func TestMatchesActor(t *testing.T) {
tests := []struct {
name string
actor string
expected bool
}{
// Exact matches
{"joe", "joe", true},
{"Joe", "joe", true}, // Case insensitive
{"JOE", "joe", true},
// Actor as path, name as simple name
{"joe", "gastown/crew/joe", true},
{"Joe", "gastown/crew/joe", true},
// Partial matches
{"joe-session1", "joe", true},
{"gastown-joe", "joe", true},
// Non-matches
{"bob", "joe", false},
{"", "joe", false},
{"witness", "gastown/crew/joe", false},
}
for _, tt := range tests {
t.Run(tt.name+"_"+tt.actor, func(t *testing.T) {
got := matchesActor(tt.name, tt.actor)
if got != tt.expected {
t.Errorf("matchesActor(%q, %q) = %v, want %v", tt.name, tt.actor, got, tt.expected)
}
})
}
}
func TestParseBeadsTimestamp(t *testing.T) {
tests := []struct {
input string
expected string // Format: "2006-01-02 15:04"
isZero bool
}{
{"2025-12-30T16:19:00Z", "2025-12-30 16:19", false},
{"2025-12-30 16:19", "2025-12-30 16:19", false},
{"2025-12-30", "2025-12-30 00:00", false},
{"invalid", "", true},
{"", "", true},
}
for _, tt := range tests {
t.Run(tt.input, func(t *testing.T) {
got := parseBeadsTimestamp(tt.input)
if tt.isZero {
if !got.IsZero() {
t.Errorf("parseBeadsTimestamp(%q) expected zero time, got %v", tt.input, got)
}
return
}
gotStr := got.Format("2006-01-02 15:04")
if gotStr != tt.expected {
t.Errorf("parseBeadsTimestamp(%q) = %q, want %q", tt.input, gotStr, tt.expected)
}
})
}
}
func TestFormatSource(t *testing.T) {
// Just verify it doesn't panic and returns non-empty strings
sources := []string{"git", "beads", "townlog", "events", "unknown"}
for _, s := range sources {
result := formatSource(s)
if result == "" {
t.Errorf("formatSource(%q) returned empty string", s)
}
}
}
func TestFormatType(t *testing.T) {
// Just verify it doesn't panic and returns non-empty strings
types := []string{"commit", "bead_created", "bead_closed", "spawn", "done", "handoff", "crash", "kill", "merged", "merge_failed", "unknown"}
for _, typ := range types {
result := formatType(typ)
if result == "" {
t.Errorf("formatType(%q) returned empty string", typ)
}
}
}