package user import ( "context" "errors" "fmt" "github.com/perfect-panel/server/pkg/cache" "github.com/redis/go-redis/v9" "gorm.io/gorm" ) var ( cacheUserIdPrefix = "cache:user:id:" cacheUserEmailPrefix = "cache:user:email:" ) var _ Model = (*customUserModel)(nil) type ( Model interface { userModel customUserLogicModel } userModel interface { Insert(ctx context.Context, data *User, tx ...*gorm.DB) error FindOne(ctx context.Context, id int64) (*User, error) Update(ctx context.Context, data *User, tx ...*gorm.DB) error Delete(ctx context.Context, id int64, tx ...*gorm.DB) error Transaction(ctx context.Context, fn func(db *gorm.DB) error) error } customUserModel struct { *defaultUserModel } defaultUserModel struct { cache.CachedConn table string } ) func newUserModel(db *gorm.DB, c *redis.Client) *defaultUserModel { return &defaultUserModel{ CachedConn: cache.NewConn(db, c), table: "`user`", } } func (m *defaultUserModel) batchGetCacheKeys(users ...*User) []string { var keys []string for _, user := range users { keys = append(keys, user.GetCacheKeys()...) } return keys } func (m *defaultUserModel) getCacheKeys(data *User) []string { if data == nil { return []string{} } return data.GetCacheKeys() } func (m *defaultUserModel) clearUserCache(ctx context.Context, data ...*User) error { return m.ClearUserCache(ctx, data...) } func (m *defaultUserModel) FindOneByEmail(ctx context.Context, email string) (*User, error) { var user User key := fmt.Sprintf("%s%v", cacheUserEmailPrefix, email) err := m.QueryCtx(ctx, &user, key, func(conn *gorm.DB, v interface{}) error { var data AuthMethods if err := conn.Model(&AuthMethods{}).Where("`auth_type` = 'email' AND `auth_identifier` = ?", email).First(&data).Error; err != nil { return err } return conn.Model(&User{}).Where("`id` = ?", data.UserId).Preload("UserDevices").Preload("AuthMethods").First(v).Error }) return &user, err } func (m *defaultUserModel) Insert(ctx context.Context, data *User, tx ...*gorm.DB) error { err := m.ExecCtx(ctx, func(conn *gorm.DB) error { if len(tx) > 0 { conn = tx[0] } return conn.Create(&data).Error }, m.getCacheKeys(data)...) return err } func (m *defaultUserModel) FindOne(ctx context.Context, id int64) (*User, error) { userIdKey := fmt.Sprintf("%s%v", cacheUserIdPrefix, id) var resp User err := m.QueryCtx(ctx, &resp, userIdKey, func(conn *gorm.DB, v interface{}) error { return conn.Model(&User{}).Where("`id` = ?", id).Preload("UserDevices").Preload("AuthMethods").First(&resp).Error }) return &resp, err } func (m *defaultUserModel) Update(ctx context.Context, data *User, tx ...*gorm.DB) error { old, err := m.FindOne(ctx, data.Id) if err != nil && !errors.Is(err, gorm.ErrRecordNotFound) { return err } err = m.ExecCtx(ctx, func(conn *gorm.DB) error { if len(tx) > 0 { conn = tx[0] } return conn.Save(data).Error }, m.getCacheKeys(old)...) return err } func (m *defaultUserModel) Delete(ctx context.Context, id int64, tx ...*gorm.DB) error { data, err := m.FindOne(ctx, id) if err != nil { if errors.Is(err, gorm.ErrRecordNotFound) { return nil } return err } // 使用批量相关缓存清理,包含所有相关数据的缓存 defer func() { if clearErr := m.BatchClearRelatedCache(ctx, data); clearErr != nil { // 记录清理缓存错误,但不阻断删除操作 } }() return m.TransactCtx(ctx, func(db *gorm.DB) error { if len(tx) > 0 { db = tx[0] } // 删除用户相关的所有数据 if err := db.Model(&User{}).Where("`id` = ?", id).Delete(&User{}).Error; err != nil { return err } if err := db.Model(&AuthMethods{}).Where("`user_id` = ?", id).Delete(&AuthMethods{}).Error; err != nil { return err } if err := db.Model(&Subscribe{}).Where("`user_id` = ?", id).Delete(&Subscribe{}).Error; err != nil { return err } if err := db.Model(&BalanceLog{}).Where("`user_id` = ?", id).Delete(&BalanceLog{}).Error; err != nil { return err } if err := db.Model(&GiftAmountLog{}).Where("`user_id` = ?", id).Delete(&GiftAmountLog{}).Error; err != nil { return err } if err := db.Model(&LoginLog{}).Where("`user_id` = ?", id).Delete(&LoginLog{}).Error; err != nil { return err } if err := db.Model(&SubscribeLog{}).Where("`user_id` = ?", id).Delete(&SubscribeLog{}).Error; err != nil { return err } if err := db.Model(&Device{}).Where("`user_id` = ?", id).Delete(&Device{}).Error; err != nil { return err } if err := db.Model(&CommissionLog{}).Where("`user_id` = ?", id).Delete(&CommissionLog{}).Error; err != nil { return err } return nil }) } func (m *defaultUserModel) Transaction(ctx context.Context, fn func(db *gorm.DB) error) error { return m.TransactCtx(ctx, fn) }