ai-code-battle/cmd/acb-evolver/internal/live/r2_test.go
jedarden 1523c52e0a Add R2 upload for live evolution observatory (Phase 10)
- Add R2 client module (cmd/acb-evolver/internal/live/r2.go) with
  S3-compatible uploads to Cloudflare R2
- UploadLiveJSON() uploads evolution state to evolution/live.json
  with Cache-Control: max-age=10 for near-real-time updates
- Add -r2 and -r2-only flags to live-export subcommand
- Add tests for R2 config validation and credential handling
- Update frontend to fetch live data from R2 URL instead of Pages

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-29 04:53:35 -04:00

116 lines
2.4 KiB
Go

// Package live provides R2 upload for the evolution live.json feed.
package live
import (
"strings"
"testing"
)
func TestR2ConfigFromEnv(t *testing.T) {
// Test with no env vars
cfg := R2ConfigFromEnv()
if cfg.AccessKey != "" {
t.Error("expected empty AccessKey")
}
if cfg.HasCredentials() {
t.Error("expected no credentials without env vars")
}
}
func TestR2ConfigHasCredentials(t *testing.T) {
tests := []struct {
name string
accessKey string
secretKey string
endpoint string
bucket string
want bool
}{
{
name: "all set",
accessKey: "key",
secretKey: "secret",
endpoint: "https://example.com",
bucket: "bucket",
want: true,
},
{
name: "missing access key",
accessKey: "",
secretKey: "secret",
endpoint: "https://example.com",
bucket: "bucket",
want: false,
},
{
name: "missing secret key",
accessKey: "key",
secretKey: "",
endpoint: "https://example.com",
bucket: "bucket",
want: false,
},
{
name: "missing endpoint",
accessKey: "key",
secretKey: "secret",
endpoint: "",
bucket: "bucket",
want: false,
},
{
name: "missing bucket",
accessKey: "key",
secretKey: "secret",
endpoint: "https://example.com",
bucket: "",
want: false,
},
{
name: "all empty",
accessKey: "",
secretKey: "",
endpoint: "",
bucket: "",
want: false,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
cfg := &R2Config{
AccessKey: tt.accessKey,
SecretKey: tt.secretKey,
Endpoint: tt.endpoint,
Bucket: tt.bucket,
}
if got := cfg.HasCredentials(); got != tt.want {
t.Errorf("HasCredentials() = %v, want %v", got, tt.want)
}
})
}
}
func TestNewR2Client_MissingCredentials(t *testing.T) {
cfg := &R2Config{} // no credentials
_, err := NewR2Client(cfg)
if err == nil {
t.Error("expected error for missing credentials")
}
if !strings.Contains(err.Error(), "credentials not configured") {
t.Errorf("unexpected error: %v", err)
}
}
func TestGetEnvOrDefault(t *testing.T) {
// Test with env var set
t.Setenv("TEST_VAR", "value")
if got := getEnvOrDefault("TEST_VAR", "default"); got != "value" {
t.Errorf("got %q, want %q", got, "value")
}
// Test with env var not set
if got := getEnvOrDefault("NONEXISTENT_VAR", "default"); got != "default" {
t.Errorf("got %q, want %q", got, "default")
}
}