Files
sub2api-cn-relay-manager/internal/provision/rollback_service_test.go
2026-05-23 10:55:57 +08:00

129 lines
5.4 KiB
Go

package provision
import (
"context"
"errors"
"reflect"
"strings"
"testing"
"sub2api-cn-relay-manager/internal/host/sub2api"
"sub2api-cn-relay-manager/internal/store/sqlite"
)
func TestRollbackServiceDeletesManagedResourcesInDependencyOrder(t *testing.T) {
host := &fakeHostAdapter{
managedSnapshot: sub2api.ManagedResourceSnapshot{
Groups: []sub2api.NamedResource{{ID: "group_1", Name: "DeepSeek 默认分组"}},
Channels: []sub2api.NamedResource{{ID: "channel_1", Name: "DeepSeek 默认渠道"}},
Plans: []sub2api.NamedResource{{ID: "plan_1", Name: "DeepSeek 默认套餐"}},
Accounts: []sub2api.NamedResource{{ID: "account_1", Name: "deepseek-01"}, {ID: "account_2", Name: "deepseek-02"}},
},
}
svc := NewRollbackService(host)
report, err := svc.Rollback(context.Background(), RollbackRequest{Provider: sampleProviderManifest()})
if err != nil {
t.Fatalf("Rollback() error = %v", err)
}
if report.AccountsDeleted != 2 || report.PlansDeleted != 1 || report.ChannelsDeleted != 1 || report.GroupsDeleted != 1 {
t.Fatalf("Rollback() report = %+v, want all managed resources deleted", report)
}
want := []string{"account:account_2", "account:account_1", "plan:plan_1", "channel:channel_1", "group:group_1"}
if !reflect.DeepEqual(host.deletedResources, want) {
t.Fatalf("deleted resources = %#v, want %#v", host.deletedResources, want)
}
}
func TestRollbackServiceReturnsEmptyReportWhenNoManagedResourcesExist(t *testing.T) {
host := &fakeHostAdapter{}
svc := NewRollbackService(host)
report, err := svc.Rollback(context.Background(), RollbackRequest{Provider: sampleProviderManifest()})
if err != nil {
t.Fatalf("Rollback() error = %v", err)
}
if report.AccountsDeleted != 0 || report.PlansDeleted != 0 || report.ChannelsDeleted != 0 || report.GroupsDeleted != 0 {
t.Fatalf("Rollback() report = %+v, want zero deletions", report)
}
if len(host.deletedResources) != 0 {
t.Fatalf("deleted resources = %#v, want none", host.deletedResources)
}
}
func TestRollbackServiceRollbackStoredResourcesDeletesOnlyProvidedIDs(t *testing.T) {
host := &fakeHostAdapter{}
svc := NewRollbackService(host)
report, err := svc.RollbackStoredResources(context.Background(), []sqlite.ManagedResource{
{BatchID: 2, ResourceType: "group", HostResourceID: "group_shared", ResourceName: "DeepSeek 默认分组"},
{BatchID: 2, ResourceType: "account", HostResourceID: "account_2", ResourceName: "deepseek-02"},
})
if err != nil {
t.Fatalf("RollbackStoredResources() error = %v", err)
}
if report.AccountsDeleted != 1 || report.GroupsDeleted != 1 || report.ChannelsDeleted != 0 || report.PlansDeleted != 0 {
t.Fatalf("RollbackStoredResources() report = %+v", report)
}
want := []string{"account:account_2", "group:group_shared"}
if !reflect.DeepEqual(host.deletedResources, want) {
t.Fatalf("deleted resources = %#v, want %#v", host.deletedResources, want)
}
}
func TestRollbackServiceRequiresHost(t *testing.T) {
svc := NewRollbackService(nil)
if _, err := svc.Rollback(context.Background(), RollbackRequest{Provider: sampleProviderManifest()}); err == nil || err.Error() != "rollback host is required" {
t.Fatalf("Rollback() error = %v, want rollback host is required", err)
}
if _, err := svc.RollbackStoredResources(context.Background(), nil); err == nil || err.Error() != "rollback host is required" {
t.Fatalf("RollbackStoredResources() error = %v, want rollback host is required", err)
}
}
func TestRollbackServiceCollectsDeleteErrors(t *testing.T) {
host := &fakeHostAdapter{
managedSnapshot: sub2api.ManagedResourceSnapshot{
Groups: []sub2api.NamedResource{{ID: "group_1", Name: "DeepSeek 默认分组"}},
Channels: []sub2api.NamedResource{{ID: "channel_1", Name: "DeepSeek 默认渠道"}},
Plans: []sub2api.NamedResource{{ID: "plan_1", Name: "DeepSeek 默认套餐"}},
Accounts: []sub2api.NamedResource{{ID: "account_1", Name: "deepseek-01"}, {ID: "account_2", Name: "deepseek-02"}},
},
deleteErrors: map[string]error{
"account:account_2": errors.New("account blocked"),
"channel:channel_1": errors.New("channel blocked"),
},
}
report, err := NewRollbackService(host).Rollback(context.Background(), RollbackRequest{Provider: sampleProviderManifest()})
if err == nil {
t.Fatal("Rollback() error = nil, want joined delete errors")
}
if report.AccountsDeleted != 1 || report.PlansDeleted != 1 || report.ChannelsDeleted != 0 || report.GroupsDeleted != 1 {
t.Fatalf("Rollback() report = %+v, want partial success counts", report)
}
if !strings.Contains(err.Error(), "delete account account_2") || !strings.Contains(err.Error(), "delete channel channel_1") {
t.Fatalf("Rollback() error = %v, want joined account/channel errors", err)
}
}
func TestNamedResourceSnapshotFromStoredIgnoresUnknownTypes(t *testing.T) {
snapshot := namedResourceSnapshotFromStored([]sqlite.ManagedResource{
{ResourceType: "group", HostResourceID: "group_1", ResourceName: "g1"},
{ResourceType: "mystery", HostResourceID: "mystery_1", ResourceName: "m1"},
{ResourceType: "account", HostResourceID: "account_1", ResourceName: "a1"},
})
if len(snapshot.Groups) != 1 || snapshot.Groups[0].ID != "group_1" {
t.Fatalf("snapshot.Groups = %#v, want group_1 only", snapshot.Groups)
}
if len(snapshot.Accounts) != 1 || snapshot.Accounts[0].ID != "account_1" {
t.Fatalf("snapshot.Accounts = %#v, want account_1 only", snapshot.Accounts)
}
if len(snapshot.Plans) != 0 || len(snapshot.Channels) != 0 {
t.Fatalf("snapshot = %#v, want unknown type ignored", snapshot)
}
}