rdev/internal/adapter/redis/provisioner_test.go
jordan c59d348040 chore: prepare for composable monorepo template implementation
This commit captures the current state before implementing the composable
monorepo template system. Key changes included:

Infrastructure:
- Add CockroachDB provisioner adapter for database provisioning
- Add Redis provisioner adapter for cache provisioning
- Add build events system with PostgreSQL storage
- Add WebSocket endpoint for real-time build progress

Code agent improvements:
- Fix Claude Code adapter to use default allowed tools instead of dangerously-skip-permissions
- Add context-aware stream closing for cancellation support
- Improve parser tests for edge cases

Build system:
- Add build event constants and metrics
- Remove deprecated git_operations.go (replaced by pod_git_operations.go)
- Add rollback logic for multi-step provisioning operations

Documentation:
- Add composable-monorepo feature documentation
- Add DNS/Cloudflare service documentation
- Update deployment and troubleshooting guides

Cookbooks:
- Add fullstack-app cookbook
- Refactor landing-test with shared library

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-31 11:39:28 -07:00

139 lines
3.3 KiB
Go

package redis
import (
"context"
"os"
"testing"
"time"
"log/slog"
)
// Integration tests - require REDIS_TEST_URL env var
// Example: REDIS_TEST_URL=redis://:password@localhost:6379 go test ./internal/adapter/redis/...
func TestProvisioner_Integration(t *testing.T) {
redisURL := os.Getenv("REDIS_TEST_URL")
if redisURL == "" {
t.Skip("REDIS_TEST_URL not set, skipping integration test")
}
// Parse URL for config (simplified, assumes redis://:password@host:port format)
// In real tests, use a proper URL parser
cfg := Config{
Host: "localhost",
Port: 6379,
Password: "",
KeyPrefix: "test:",
}
logger := slog.New(slog.NewTextHandler(os.Stdout, &slog.HandlerOptions{Level: slog.LevelDebug}))
prov, err := NewProvisioner(cfg, logger)
if err != nil {
t.Fatalf("failed to create provisioner: %v", err)
}
defer prov.Close()
ctx, cancel := context.WithTimeout(context.Background(), 30*time.Second)
defer cancel()
projectID := "test-project-" + time.Now().Format("20060102150405")
// Test CreateProjectCache
t.Run("CreateProjectCache", func(t *testing.T) {
creds, err := prov.CreateProjectCache(ctx, projectID)
if err != nil {
t.Fatalf("CreateProjectCache failed: %v", err)
}
if creds.ProjectID != projectID {
t.Errorf("ProjectID = %q, want %q", creds.ProjectID, projectID)
}
if creds.Username == "" {
t.Error("Username is empty")
}
if creds.URL == "" {
t.Error("URL is empty")
}
if creds.Prefix == "" {
t.Error("Prefix is empty")
}
t.Logf("Created cache: username=%s prefix=%s", creds.Username, creds.Prefix)
})
// Test GetProjectCache
t.Run("GetProjectCache", func(t *testing.T) {
creds, err := prov.GetProjectCache(ctx, projectID)
if err != nil {
t.Fatalf("GetProjectCache failed: %v", err)
}
if creds == nil {
t.Fatal("GetProjectCache returned nil")
}
if creds.Username == "" {
t.Error("Username is empty")
}
})
// Test DeleteProjectCache
t.Run("DeleteProjectCache", func(t *testing.T) {
err := prov.DeleteProjectCache(ctx, projectID, true)
if err != nil {
t.Fatalf("DeleteProjectCache failed: %v", err)
}
// Verify user is deleted
creds, err := prov.GetProjectCache(ctx, projectID)
if err != nil {
t.Fatalf("GetProjectCache after delete failed: %v", err)
}
if creds != nil {
t.Error("User still exists after delete")
}
})
}
func TestUsernameFor(t *testing.T) {
p := &Provisioner{keyPrefix: "project:"}
tests := []struct {
projectID string
want string
}{
{"my-app", "proj-my-app"},
{"my_app", "proj-my-app"}, // underscore converted to hyphen
{"MyApp123", "proj-MyApp123"},
{"app.name", "proj-app-name"}, // dot converted to hyphen
}
for _, tt := range tests {
t.Run(tt.projectID, func(t *testing.T) {
got := p.usernameFor(tt.projectID)
if got != tt.want {
t.Errorf("usernameFor(%q) = %q, want %q", tt.projectID, got, tt.want)
}
})
}
}
func TestPrefixFor(t *testing.T) {
p := &Provisioner{keyPrefix: "project:"}
tests := []struct {
projectID string
want string
}{
{"my-app", "project:my-app:"},
{"app123", "project:app123:"},
}
for _, tt := range tests {
t.Run(tt.projectID, func(t *testing.T) {
got := p.prefixFor(tt.projectID)
if got != tt.want {
t.Errorf("prefixFor(%q) = %q, want %q", tt.projectID, got, tt.want)
}
})
}
}