package service import ( "encoding/json" "reflect" "testing" "github.com/alicebob/miniredis/v2" "github.com/ez-api/ez-api/internal/model" "github.com/ez-api/foundation/contract" "github.com/redis/go-redis/v9" ) func TestSyncProvider_WritesSnapshotAndRouting(t *testing.T) { goldenRaw := contract.ProviderSnapshotJSON() var golden map[string]any if err := json.Unmarshal(goldenRaw, &golden); err != nil { t.Fatalf("parse golden json: %v", err) } mr := miniredis.RunT(t) rdb := redis.NewClient(&redis.Options{Addr: mr.Addr()}) svc := NewSyncService(rdb) p := &model.Provider{ Name: "p1", Type: "vertex-express", Group: "default", Models: "gemini-3-pro-preview", Status: "active", AutoBan: true, GoogleProject: "", GoogleLocation: "global", } p.ID = 42 if err := svc.SyncProvider(p); err != nil { t.Fatalf("SyncProvider: %v", err) } raw := mr.HGet("config:providers", "42") if raw == "" { t.Fatalf("expected config:providers hash entry") } var snap map[string]any if err := json.Unmarshal([]byte(raw), &snap); err != nil { t.Fatalf("invalid snapshot json: %v", err) } for k, v := range golden { if !reflect.DeepEqual(snap[k], v) { t.Fatalf("snapshot mismatch for %q: got=%#v want=%#v", k, snap[k], v) } } routeKey := "route:group:default:gemini-3-pro-preview" if !mr.Exists(routeKey) { t.Fatalf("expected routing key %q to exist", routeKey) } ok, err := mr.SIsMember(routeKey, "42") if err != nil { t.Fatalf("SIsMember: %v", err) } if !ok { t.Fatalf("expected provider id 42 in routing set %q", routeKey) } } func TestSyncKey_WritesTokenID(t *testing.T) { mr := miniredis.RunT(t) rdb := redis.NewClient(&redis.Options{Addr: mr.Addr()}) svc := NewSyncService(rdb) k := &model.Key{ TokenHash: "hash", MasterID: 1, IssuedAtEpoch: 1, Status: "active", Group: "default", Scopes: "chat:write", DefaultNamespace: "default", Namespaces: "default", } k.ID = 123 if err := svc.SyncKey(k); err != nil { t.Fatalf("SyncKey: %v", err) } if got := mr.HGet("auth:token:hash", "id"); got != "123" { t.Fatalf("expected auth:token:hash.id=123, got %q", got) } } func TestSyncProviderDelete_RemovesSnapshotAndRouting(t *testing.T) { mr := miniredis.RunT(t) rdb := redis.NewClient(&redis.Options{Addr: mr.Addr()}) svc := NewSyncService(rdb) p := &model.Provider{ Name: "p1", Type: "openai", Group: "default", Models: "gpt-4o-mini,gpt-4o", Status: "active", } p.ID = 7 if err := svc.SyncProvider(p); err != nil { t.Fatalf("SyncProvider: %v", err) } if err := svc.SyncProviderDelete(p); err != nil { t.Fatalf("SyncProviderDelete: %v", err) } if got := mr.HGet("config:providers", "7"); got != "" { t.Fatalf("expected provider snapshot removed, got %q", got) } if ok, _ := mr.SIsMember("route:group:default:gpt-4o-mini", "7"); ok { t.Fatalf("expected provider removed from route set") } if ok, _ := mr.SIsMember("route:group:default:gpt-4o", "7"); ok { t.Fatalf("expected provider removed from route set") } }