- 前端: 所有界面显示、i18n 文本、组件中的品牌名称 - 后端: 服务层、设置默认值、邮件模板、安装向导 - 数据库: 迁移脚本注释 - 保持功能完全一致,仅更改品牌名称 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
217 lines
7.6 KiB
Go
217 lines
7.6 KiB
Go
//go:build integration
|
||
|
||
package repository
|
||
|
||
import (
|
||
"context"
|
||
"fmt"
|
||
"strings"
|
||
"testing"
|
||
"time"
|
||
|
||
dbent "github.com/Wei-Shaw/sub2api/ent"
|
||
"github.com/Wei-Shaw/sub2api/ent/apikey"
|
||
"github.com/Wei-Shaw/sub2api/ent/schema/mixins"
|
||
"github.com/Wei-Shaw/sub2api/ent/usersubscription"
|
||
"github.com/Wei-Shaw/sub2api/internal/service"
|
||
"github.com/stretchr/testify/require"
|
||
)
|
||
|
||
func uniqueSoftDeleteValue(t *testing.T, prefix string) string {
|
||
t.Helper()
|
||
safeName := strings.NewReplacer("/", "_", " ", "_").Replace(t.Name())
|
||
return fmt.Sprintf("%s-%s", prefix, safeName)
|
||
}
|
||
|
||
func createEntUser(t *testing.T, ctx context.Context, client *dbent.Client, email string) *dbent.User {
|
||
t.Helper()
|
||
|
||
u, err := client.User.Create().
|
||
SetEmail(email).
|
||
SetPasswordHash("test-password-hash").
|
||
Save(ctx)
|
||
require.NoError(t, err, "create ent user")
|
||
return u
|
||
}
|
||
|
||
func TestEntSoftDelete_ApiKey_DefaultFilterAndSkip(t *testing.T) {
|
||
ctx := context.Background()
|
||
// 使用全局 ent client,确保软删除验证在实际持久化数据上进行。
|
||
client := testEntClient(t)
|
||
|
||
u := createEntUser(t, ctx, client, uniqueSoftDeleteValue(t, "sd-user")+"@example.com")
|
||
|
||
repo := NewApiKeyRepository(client)
|
||
key := &service.ApiKey{
|
||
UserID: u.ID,
|
||
Key: uniqueSoftDeleteValue(t, "sk-soft-delete"),
|
||
Name: "soft-delete",
|
||
Status: service.StatusActive,
|
||
}
|
||
require.NoError(t, repo.Create(ctx, key), "create api key")
|
||
|
||
require.NoError(t, repo.Delete(ctx, key.ID), "soft delete api key")
|
||
|
||
_, err := repo.GetByID(ctx, key.ID)
|
||
require.ErrorIs(t, err, service.ErrApiKeyNotFound, "deleted rows should be hidden by default")
|
||
|
||
_, err = client.ApiKey.Query().Where(apikey.IDEQ(key.ID)).Only(ctx)
|
||
require.Error(t, err, "default ent query should not see soft-deleted rows")
|
||
require.True(t, dbent.IsNotFound(err), "expected ent not-found after default soft delete filter")
|
||
|
||
got, err := client.ApiKey.Query().
|
||
Where(apikey.IDEQ(key.ID)).
|
||
Only(mixins.SkipSoftDelete(ctx))
|
||
require.NoError(t, err, "SkipSoftDelete should include soft-deleted rows")
|
||
require.NotNil(t, got.DeletedAt, "deleted_at should be set after soft delete")
|
||
}
|
||
|
||
func TestEntSoftDelete_ApiKey_DeleteIdempotent(t *testing.T) {
|
||
ctx := context.Background()
|
||
// 使用全局 ent client,避免事务回滚影响幂等性验证。
|
||
client := testEntClient(t)
|
||
|
||
u := createEntUser(t, ctx, client, uniqueSoftDeleteValue(t, "sd-user2")+"@example.com")
|
||
|
||
repo := NewApiKeyRepository(client)
|
||
key := &service.ApiKey{
|
||
UserID: u.ID,
|
||
Key: uniqueSoftDeleteValue(t, "sk-soft-delete2"),
|
||
Name: "soft-delete2",
|
||
Status: service.StatusActive,
|
||
}
|
||
require.NoError(t, repo.Create(ctx, key), "create api key")
|
||
|
||
require.NoError(t, repo.Delete(ctx, key.ID), "first delete")
|
||
require.NoError(t, repo.Delete(ctx, key.ID), "second delete should be idempotent")
|
||
}
|
||
|
||
func TestEntSoftDelete_ApiKey_HardDeleteViaSkipSoftDelete(t *testing.T) {
|
||
ctx := context.Background()
|
||
// 使用全局 ent client,确保 SkipSoftDelete 的硬删除语义可验证。
|
||
client := testEntClient(t)
|
||
|
||
u := createEntUser(t, ctx, client, uniqueSoftDeleteValue(t, "sd-user3")+"@example.com")
|
||
|
||
repo := NewApiKeyRepository(client)
|
||
key := &service.ApiKey{
|
||
UserID: u.ID,
|
||
Key: uniqueSoftDeleteValue(t, "sk-soft-delete3"),
|
||
Name: "soft-delete3",
|
||
Status: service.StatusActive,
|
||
}
|
||
require.NoError(t, repo.Create(ctx, key), "create api key")
|
||
|
||
require.NoError(t, repo.Delete(ctx, key.ID), "soft delete api key")
|
||
|
||
// Hard delete using SkipSoftDelete so the hook doesn't convert it to update-deleted_at.
|
||
_, err := client.ApiKey.Delete().Where(apikey.IDEQ(key.ID)).Exec(mixins.SkipSoftDelete(ctx))
|
||
require.NoError(t, err, "hard delete")
|
||
|
||
_, err = client.ApiKey.Query().
|
||
Where(apikey.IDEQ(key.ID)).
|
||
Only(mixins.SkipSoftDelete(ctx))
|
||
require.True(t, dbent.IsNotFound(err), "expected row to be hard deleted")
|
||
}
|
||
|
||
// --- UserSubscription 软删除测试 ---
|
||
|
||
func createEntGroup(t *testing.T, ctx context.Context, client *dbent.Client, name string) *dbent.Group {
|
||
t.Helper()
|
||
|
||
g, err := client.Group.Create().
|
||
SetName(name).
|
||
SetStatus(service.StatusActive).
|
||
Save(ctx)
|
||
require.NoError(t, err, "create ent group")
|
||
return g
|
||
}
|
||
|
||
func TestEntSoftDelete_UserSubscription_DefaultFilterAndSkip(t *testing.T) {
|
||
ctx := context.Background()
|
||
client := testEntClient(t)
|
||
|
||
u := createEntUser(t, ctx, client, uniqueSoftDeleteValue(t, "sd-sub-user")+"@example.com")
|
||
g := createEntGroup(t, ctx, client, uniqueSoftDeleteValue(t, "sd-sub-group"))
|
||
|
||
repo := NewUserSubscriptionRepository(client)
|
||
sub := &service.UserSubscription{
|
||
UserID: u.ID,
|
||
GroupID: g.ID,
|
||
Status: service.SubscriptionStatusActive,
|
||
ExpiresAt: time.Now().Add(24 * time.Hour),
|
||
}
|
||
require.NoError(t, repo.Create(ctx, sub), "create user subscription")
|
||
|
||
require.NoError(t, repo.Delete(ctx, sub.ID), "soft delete user subscription")
|
||
|
||
_, err := repo.GetByID(ctx, sub.ID)
|
||
require.Error(t, err, "deleted rows should be hidden by default")
|
||
|
||
_, err = client.UserSubscription.Query().Where(usersubscription.IDEQ(sub.ID)).Only(ctx)
|
||
require.Error(t, err, "default ent query should not see soft-deleted rows")
|
||
require.True(t, dbent.IsNotFound(err), "expected ent not-found after default soft delete filter")
|
||
|
||
got, err := client.UserSubscription.Query().
|
||
Where(usersubscription.IDEQ(sub.ID)).
|
||
Only(mixins.SkipSoftDelete(ctx))
|
||
require.NoError(t, err, "SkipSoftDelete should include soft-deleted rows")
|
||
require.NotNil(t, got.DeletedAt, "deleted_at should be set after soft delete")
|
||
}
|
||
|
||
func TestEntSoftDelete_UserSubscription_DeleteIdempotent(t *testing.T) {
|
||
ctx := context.Background()
|
||
client := testEntClient(t)
|
||
|
||
u := createEntUser(t, ctx, client, uniqueSoftDeleteValue(t, "sd-sub-user2")+"@example.com")
|
||
g := createEntGroup(t, ctx, client, uniqueSoftDeleteValue(t, "sd-sub-group2"))
|
||
|
||
repo := NewUserSubscriptionRepository(client)
|
||
sub := &service.UserSubscription{
|
||
UserID: u.ID,
|
||
GroupID: g.ID,
|
||
Status: service.SubscriptionStatusActive,
|
||
ExpiresAt: time.Now().Add(24 * time.Hour),
|
||
}
|
||
require.NoError(t, repo.Create(ctx, sub), "create user subscription")
|
||
|
||
require.NoError(t, repo.Delete(ctx, sub.ID), "first delete")
|
||
require.NoError(t, repo.Delete(ctx, sub.ID), "second delete should be idempotent")
|
||
}
|
||
|
||
func TestEntSoftDelete_UserSubscription_ListExcludesDeleted(t *testing.T) {
|
||
ctx := context.Background()
|
||
client := testEntClient(t)
|
||
|
||
u := createEntUser(t, ctx, client, uniqueSoftDeleteValue(t, "sd-sub-user3")+"@example.com")
|
||
g1 := createEntGroup(t, ctx, client, uniqueSoftDeleteValue(t, "sd-sub-group3a"))
|
||
g2 := createEntGroup(t, ctx, client, uniqueSoftDeleteValue(t, "sd-sub-group3b"))
|
||
|
||
repo := NewUserSubscriptionRepository(client)
|
||
|
||
sub1 := &service.UserSubscription{
|
||
UserID: u.ID,
|
||
GroupID: g1.ID,
|
||
Status: service.SubscriptionStatusActive,
|
||
ExpiresAt: time.Now().Add(24 * time.Hour),
|
||
}
|
||
require.NoError(t, repo.Create(ctx, sub1), "create subscription 1")
|
||
|
||
sub2 := &service.UserSubscription{
|
||
UserID: u.ID,
|
||
GroupID: g2.ID,
|
||
Status: service.SubscriptionStatusActive,
|
||
ExpiresAt: time.Now().Add(24 * time.Hour),
|
||
}
|
||
require.NoError(t, repo.Create(ctx, sub2), "create subscription 2")
|
||
|
||
// 软删除 sub1
|
||
require.NoError(t, repo.Delete(ctx, sub1.ID), "soft delete subscription 1")
|
||
|
||
// ListByUserID 应只返回未删除的订阅
|
||
subs, err := repo.ListByUserID(ctx, u.ID)
|
||
require.NoError(t, err, "ListByUserID")
|
||
require.Len(t, subs, 1, "should only return non-deleted subscriptions")
|
||
require.Equal(t, sub2.ID, subs[0].ID, "expected sub2 to be returned")
|
||
}
|