199 lines
5.8 KiB
Go
199 lines
5.8 KiB
Go
package repository
|
|
|
|
import (
|
|
"context"
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/service"
|
|
|
|
"github.com/Wei-Shaw/sub2api/internal/model"
|
|
"github.com/Wei-Shaw/sub2api/internal/pkg/pagination"
|
|
|
|
"gorm.io/gorm"
|
|
"gorm.io/gorm/clause"
|
|
)
|
|
|
|
type groupRepository struct {
|
|
db *gorm.DB
|
|
}
|
|
|
|
func NewGroupRepository(db *gorm.DB) service.GroupRepository {
|
|
return &groupRepository{db: db}
|
|
}
|
|
|
|
func (r *groupRepository) Create(ctx context.Context, group *model.Group) error {
|
|
err := r.db.WithContext(ctx).Create(group).Error
|
|
return translatePersistenceError(err, nil, service.ErrGroupExists)
|
|
}
|
|
|
|
func (r *groupRepository) GetByID(ctx context.Context, id int64) (*model.Group, error) {
|
|
var group model.Group
|
|
err := r.db.WithContext(ctx).First(&group, id).Error
|
|
if err != nil {
|
|
return nil, translatePersistenceError(err, service.ErrGroupNotFound, nil)
|
|
}
|
|
return &group, nil
|
|
}
|
|
|
|
func (r *groupRepository) Update(ctx context.Context, group *model.Group) error {
|
|
return r.db.WithContext(ctx).Save(group).Error
|
|
}
|
|
|
|
func (r *groupRepository) Delete(ctx context.Context, id int64) error {
|
|
return r.db.WithContext(ctx).Delete(&model.Group{}, id).Error
|
|
}
|
|
|
|
func (r *groupRepository) List(ctx context.Context, params pagination.PaginationParams) ([]model.Group, *pagination.PaginationResult, error) {
|
|
return r.ListWithFilters(ctx, params, "", "", nil)
|
|
}
|
|
|
|
// ListWithFilters lists groups with optional filtering by platform, status, and is_exclusive
|
|
func (r *groupRepository) ListWithFilters(ctx context.Context, params pagination.PaginationParams, platform, status string, isExclusive *bool) ([]model.Group, *pagination.PaginationResult, error) {
|
|
var groups []model.Group
|
|
var total int64
|
|
|
|
db := r.db.WithContext(ctx).Model(&model.Group{})
|
|
|
|
// Apply filters
|
|
if platform != "" {
|
|
db = db.Where("platform = ?", platform)
|
|
}
|
|
if status != "" {
|
|
db = db.Where("status = ?", status)
|
|
}
|
|
if isExclusive != nil {
|
|
db = db.Where("is_exclusive = ?", *isExclusive)
|
|
}
|
|
|
|
if err := db.Count(&total).Error; err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
if err := db.Offset(params.Offset()).Limit(params.Limit()).Order("id ASC").Find(&groups).Error; err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
// 获取每个分组的账号数量
|
|
for i := range groups {
|
|
count, _ := r.GetAccountCount(ctx, groups[i].ID)
|
|
groups[i].AccountCount = count
|
|
}
|
|
|
|
pages := int(total) / params.Limit()
|
|
if int(total)%params.Limit() > 0 {
|
|
pages++
|
|
}
|
|
|
|
return groups, &pagination.PaginationResult{
|
|
Total: total,
|
|
Page: params.Page,
|
|
PageSize: params.Limit(),
|
|
Pages: pages,
|
|
}, nil
|
|
}
|
|
|
|
func (r *groupRepository) ListActive(ctx context.Context) ([]model.Group, error) {
|
|
var groups []model.Group
|
|
err := r.db.WithContext(ctx).Where("status = ?", model.StatusActive).Order("id ASC").Find(&groups).Error
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
// 获取每个分组的账号数量
|
|
for i := range groups {
|
|
count, _ := r.GetAccountCount(ctx, groups[i].ID)
|
|
groups[i].AccountCount = count
|
|
}
|
|
return groups, nil
|
|
}
|
|
|
|
func (r *groupRepository) ListActiveByPlatform(ctx context.Context, platform string) ([]model.Group, error) {
|
|
var groups []model.Group
|
|
err := r.db.WithContext(ctx).Where("status = ? AND platform = ?", model.StatusActive, platform).Order("id ASC").Find(&groups).Error
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
// 获取每个分组的账号数量
|
|
for i := range groups {
|
|
count, _ := r.GetAccountCount(ctx, groups[i].ID)
|
|
groups[i].AccountCount = count
|
|
}
|
|
return groups, nil
|
|
}
|
|
|
|
func (r *groupRepository) ExistsByName(ctx context.Context, name string) (bool, error) {
|
|
var count int64
|
|
err := r.db.WithContext(ctx).Model(&model.Group{}).Where("name = ?", name).Count(&count).Error
|
|
return count > 0, err
|
|
}
|
|
|
|
func (r *groupRepository) GetAccountCount(ctx context.Context, groupID int64) (int64, error) {
|
|
var count int64
|
|
err := r.db.WithContext(ctx).Model(&model.AccountGroup{}).Where("group_id = ?", groupID).Count(&count).Error
|
|
return count, err
|
|
}
|
|
|
|
// DeleteAccountGroupsByGroupID 删除分组与账号的关联关系
|
|
func (r *groupRepository) DeleteAccountGroupsByGroupID(ctx context.Context, groupID int64) (int64, error) {
|
|
result := r.db.WithContext(ctx).Where("group_id = ?", groupID).Delete(&model.AccountGroup{})
|
|
return result.RowsAffected, result.Error
|
|
}
|
|
|
|
func (r *groupRepository) DeleteCascade(ctx context.Context, id int64) ([]int64, error) {
|
|
group, err := r.GetByID(ctx, id)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var affectedUserIDs []int64
|
|
if group.IsSubscriptionType() {
|
|
var subscriptions []model.UserSubscription
|
|
if err := r.db.WithContext(ctx).
|
|
Model(&model.UserSubscription{}).
|
|
Where("group_id = ?", id).
|
|
Select("user_id").
|
|
Find(&subscriptions).Error; err != nil {
|
|
return nil, err
|
|
}
|
|
for _, sub := range subscriptions {
|
|
affectedUserIDs = append(affectedUserIDs, sub.UserID)
|
|
}
|
|
}
|
|
|
|
err = r.db.WithContext(ctx).Transaction(func(tx *gorm.DB) error {
|
|
// 1. 删除订阅类型分组的订阅记录
|
|
if group.IsSubscriptionType() {
|
|
if err := tx.Where("group_id = ?", id).Delete(&model.UserSubscription{}).Error; err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
// 2. 将 api_keys 中绑定该分组的 group_id 设为 nil
|
|
if err := tx.Model(&model.ApiKey{}).Where("group_id = ?", id).Update("group_id", nil).Error; err != nil {
|
|
return err
|
|
}
|
|
|
|
// 3. 从 users.allowed_groups 数组中移除该分组 ID
|
|
if err := tx.Model(&model.User{}).
|
|
Where("? = ANY(allowed_groups)", id).
|
|
Update("allowed_groups", gorm.Expr("array_remove(allowed_groups, ?)", id)).Error; err != nil {
|
|
return err
|
|
}
|
|
|
|
// 4. 删除 account_groups 中间表的数据
|
|
if err := tx.Where("group_id = ?", id).Delete(&model.AccountGroup{}).Error; err != nil {
|
|
return err
|
|
}
|
|
|
|
// 5. 删除分组本身(带锁,避免并发写)
|
|
if err := tx.Clauses(clause.Locking{Strength: "UPDATE"}).Delete(&model.Group{}, id).Error; err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return affectedUserIDs, nil
|
|
}
|