基础架构
This commit is contained in:
		| @@ -0,0 +1,22 @@ | ||||
| package repositories | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"tyapi-server/internal/domains/user/entities" | ||||
| 	"tyapi-server/internal/shared/interfaces" | ||||
| ) | ||||
|  | ||||
| // EnterpriseInfoRepository 企业信息仓储接口 | ||||
| type EnterpriseInfoRepository interface { | ||||
| 	interfaces.Repository[entities.EnterpriseInfo] | ||||
|  | ||||
| 	// 基础查询 | ||||
| 	GetByUserID(ctx context.Context, userID string) (*entities.EnterpriseInfo, error) | ||||
| 	GetByUnifiedSocialCode(ctx context.Context, unifiedSocialCode string) (*entities.EnterpriseInfo, error) | ||||
|  | ||||
| 	// 业务操作 | ||||
| 	CheckUnifiedSocialCodeExists(ctx context.Context, unifiedSocialCode string, excludeUserID string) (bool, error) | ||||
| 	UpdateVerificationStatus(ctx context.Context, userID string, isOCRVerified, isFaceVerified, isCertified bool) error | ||||
| 	UpdateOCRData(ctx context.Context, userID string, rawData string, confidence float64) error | ||||
| 	CompleteCertification(ctx context.Context, userID string) error | ||||
| } | ||||
							
								
								
									
										21
									
								
								internal/domains/user/repositories/queries/user_queries.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								internal/domains/user/repositories/queries/user_queries.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| package queries | ||||
|  | ||||
| // ListUsersQuery 用户列表查询参数 | ||||
| type ListUsersQuery struct { | ||||
| 	Page      int    `json:"page"` | ||||
| 	PageSize  int    `json:"page_size"` | ||||
| 	Phone     string `json:"phone"` | ||||
| 	StartDate string `json:"start_date"` | ||||
| 	EndDate   string `json:"end_date"` | ||||
| } | ||||
|  | ||||
| // ListSMSCodesQuery 短信验证码列表查询参数 | ||||
| type ListSMSCodesQuery struct { | ||||
| 	Page      int    `json:"page"` | ||||
| 	PageSize  int    `json:"page_size"` | ||||
| 	Phone     string `json:"phone"` | ||||
| 	Purpose   string `json:"purpose"` | ||||
| 	Status    string `json:"status"` | ||||
| 	StartDate string `json:"start_date"` | ||||
| 	EndDate   string `json:"end_date"` | ||||
| } | ||||
| @@ -1,148 +0,0 @@ | ||||
| package repositories | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"fmt" | ||||
| 	"time" | ||||
|  | ||||
| 	"go.uber.org/zap" | ||||
| 	"gorm.io/gorm" | ||||
|  | ||||
| 	"tyapi-server/internal/domains/user/entities" | ||||
| 	"tyapi-server/internal/shared/interfaces" | ||||
| ) | ||||
|  | ||||
| // SMSCodeRepository 短信验证码仓储 | ||||
| type SMSCodeRepository struct { | ||||
| 	db     *gorm.DB | ||||
| 	cache  interfaces.CacheService | ||||
| 	logger *zap.Logger | ||||
| } | ||||
|  | ||||
| // NewSMSCodeRepository 创建短信验证码仓储 | ||||
| func NewSMSCodeRepository(db *gorm.DB, cache interfaces.CacheService, logger *zap.Logger) *SMSCodeRepository { | ||||
| 	return &SMSCodeRepository{ | ||||
| 		db:     db, | ||||
| 		cache:  cache, | ||||
| 		logger: logger, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Create 创建短信验证码记录 | ||||
| func (r *SMSCodeRepository) Create(ctx context.Context, smsCode *entities.SMSCode) error { | ||||
| 	if err := r.db.WithContext(ctx).Create(smsCode).Error; err != nil { | ||||
| 		r.logger.Error("创建短信验证码失败", zap.Error(err)) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// 缓存验证码 | ||||
| 	cacheKey := r.buildCacheKey(smsCode.Phone, smsCode.Scene) | ||||
| 	r.cache.Set(ctx, cacheKey, smsCode, 5*time.Minute) | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // GetValidCode 获取有效的验证码 | ||||
| func (r *SMSCodeRepository) GetValidCode(ctx context.Context, phone string, scene entities.SMSScene) (*entities.SMSCode, error) { | ||||
| 	// 先从缓存查找 | ||||
| 	cacheKey := r.buildCacheKey(phone, scene) | ||||
| 	var smsCode entities.SMSCode | ||||
| 	if err := r.cache.Get(ctx, cacheKey, &smsCode); err == nil { | ||||
| 		return &smsCode, nil | ||||
| 	} | ||||
|  | ||||
| 	// 从数据库查找最新的有效验证码 | ||||
| 	if err := r.db.WithContext(ctx). | ||||
| 		Where("phone = ? AND scene = ? AND expires_at > ? AND used_at IS NULL", | ||||
| 			phone, scene, time.Now()). | ||||
| 		Order("created_at DESC"). | ||||
| 		First(&smsCode).Error; err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// 缓存结果 | ||||
| 	r.cache.Set(ctx, cacheKey, &smsCode, 5*time.Minute) | ||||
|  | ||||
| 	return &smsCode, nil | ||||
| } | ||||
|  | ||||
| // MarkAsUsed 标记验证码为已使用 | ||||
| func (r *SMSCodeRepository) MarkAsUsed(ctx context.Context, id string) error { | ||||
| 	now := time.Now() | ||||
| 	if err := r.db.WithContext(ctx). | ||||
| 		Model(&entities.SMSCode{}). | ||||
| 		Where("id = ?", id). | ||||
| 		Update("used_at", now).Error; err != nil { | ||||
| 		r.logger.Error("标记验证码为已使用失败", zap.Error(err)) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	r.logger.Info("验证码已标记为使用", zap.String("code_id", id)) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Update 更新验证码记录 | ||||
| func (r *SMSCodeRepository) Update(ctx context.Context, smsCode *entities.SMSCode) error { | ||||
| 	if err := r.db.WithContext(ctx).Save(smsCode).Error; err != nil { | ||||
| 		r.logger.Error("更新验证码记录失败", zap.Error(err)) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// 更新缓存 | ||||
| 	cacheKey := r.buildCacheKey(smsCode.Phone, smsCode.Scene) | ||||
| 	r.cache.Set(ctx, cacheKey, smsCode, 5*time.Minute) | ||||
|  | ||||
| 	r.logger.Info("验证码记录更新成功", zap.String("code_id", smsCode.ID)) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // GetRecentCode 获取最近的验证码记录(不限制有效性) | ||||
| func (r *SMSCodeRepository) GetRecentCode(ctx context.Context, phone string, scene entities.SMSScene) (*entities.SMSCode, error) { | ||||
| 	var smsCode entities.SMSCode | ||||
| 	if err := r.db.WithContext(ctx). | ||||
| 		Where("phone = ? AND scene = ?", phone, scene). | ||||
| 		Order("created_at DESC"). | ||||
| 		First(&smsCode).Error; err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return &smsCode, nil | ||||
| } | ||||
|  | ||||
| // CleanupExpired 清理过期的验证码 | ||||
| func (r *SMSCodeRepository) CleanupExpired(ctx context.Context) error { | ||||
| 	result := r.db.WithContext(ctx). | ||||
| 		Where("expires_at < ?", time.Now()). | ||||
| 		Delete(&entities.SMSCode{}) | ||||
|  | ||||
| 	if result.Error != nil { | ||||
| 		r.logger.Error("清理过期验证码失败", zap.Error(result.Error)) | ||||
| 		return result.Error | ||||
| 	} | ||||
|  | ||||
| 	if result.RowsAffected > 0 { | ||||
| 		r.logger.Info("清理过期验证码完成", zap.Int64("count", result.RowsAffected)) | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // CountRecentCodes 统计最近发送的验证码数量 | ||||
| func (r *SMSCodeRepository) CountRecentCodes(ctx context.Context, phone string, scene entities.SMSScene, duration time.Duration) (int64, error) { | ||||
| 	var count int64 | ||||
| 	if err := r.db.WithContext(ctx). | ||||
| 		Model(&entities.SMSCode{}). | ||||
| 		Where("phone = ? AND scene = ? AND created_at > ?", | ||||
| 			phone, scene, time.Now().Add(-duration)). | ||||
| 		Count(&count).Error; err != nil { | ||||
| 		r.logger.Error("统计最近验证码数量失败", zap.Error(err)) | ||||
| 		return 0, err | ||||
| 	} | ||||
|  | ||||
| 	return count, nil | ||||
| } | ||||
|  | ||||
| // buildCacheKey 构建缓存键 | ||||
| func (r *SMSCodeRepository) buildCacheKey(phone string, scene entities.SMSScene) string { | ||||
| 	return fmt.Sprintf("sms_code:%s:%s", phone, string(scene)) | ||||
| } | ||||
| @@ -1,220 +0,0 @@ | ||||
| package repositories | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"time" | ||||
|  | ||||
| 	"go.uber.org/zap" | ||||
| 	"gorm.io/gorm" | ||||
|  | ||||
| 	"tyapi-server/internal/domains/user/entities" | ||||
| 	"tyapi-server/internal/shared/interfaces" | ||||
| ) | ||||
|  | ||||
| // 定义错误常量 | ||||
| var ( | ||||
| 	// ErrUserNotFound 用户不存在错误 | ||||
| 	ErrUserNotFound = errors.New("用户不存在") | ||||
| ) | ||||
|  | ||||
| // UserRepository 用户仓储实现 | ||||
| type UserRepository struct { | ||||
| 	db     *gorm.DB | ||||
| 	cache  interfaces.CacheService | ||||
| 	logger *zap.Logger | ||||
| } | ||||
|  | ||||
| // NewUserRepository 创建用户仓储 | ||||
| func NewUserRepository(db *gorm.DB, cache interfaces.CacheService, logger *zap.Logger) *UserRepository { | ||||
| 	return &UserRepository{ | ||||
| 		db:     db, | ||||
| 		cache:  cache, | ||||
| 		logger: logger, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // Create 创建用户 | ||||
| func (r *UserRepository) Create(ctx context.Context, user *entities.User) error { | ||||
| 	if err := r.db.WithContext(ctx).Create(user).Error; err != nil { | ||||
| 		r.logger.Error("创建用户失败", zap.Error(err)) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// 清除相关缓存 | ||||
| 	r.deleteCacheByPhone(ctx, user.Phone) | ||||
|  | ||||
| 	r.logger.Info("用户创建成功", zap.String("user_id", user.ID)) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // GetByID 根据ID获取用户 | ||||
| func (r *UserRepository) GetByID(ctx context.Context, id string) (*entities.User, error) { | ||||
| 	// 尝试从缓存获取 | ||||
| 	cacheKey := fmt.Sprintf("user:id:%s", id) | ||||
| 	var user entities.User | ||||
| 	if err := r.cache.Get(ctx, cacheKey, &user); err == nil { | ||||
| 		return &user, nil | ||||
| 	} | ||||
|  | ||||
| 	// 从数据库查询 | ||||
| 	if err := r.db.WithContext(ctx).Where("id = ?", id).First(&user).Error; err != nil { | ||||
| 		if errors.Is(err, gorm.ErrRecordNotFound) { | ||||
| 			return nil, ErrUserNotFound | ||||
| 		} | ||||
| 		r.logger.Error("根据ID查询用户失败", zap.Error(err)) | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// 缓存结果 | ||||
| 	r.cache.Set(ctx, cacheKey, &user, 10*time.Minute) | ||||
|  | ||||
| 	return &user, nil | ||||
| } | ||||
|  | ||||
| // FindByPhone 根据手机号查找用户 | ||||
| func (r *UserRepository) FindByPhone(ctx context.Context, phone string) (*entities.User, error) { | ||||
| 	// 尝试从缓存获取 | ||||
| 	cacheKey := fmt.Sprintf("user:phone:%s", phone) | ||||
| 	var user entities.User | ||||
| 	if err := r.cache.Get(ctx, cacheKey, &user); err == nil { | ||||
| 		return &user, nil | ||||
| 	} | ||||
|  | ||||
| 	// 从数据库查询 | ||||
| 	if err := r.db.WithContext(ctx).Where("phone = ?", phone).First(&user).Error; err != nil { | ||||
| 		if errors.Is(err, gorm.ErrRecordNotFound) { | ||||
| 			return nil, ErrUserNotFound | ||||
| 		} | ||||
| 		r.logger.Error("根据手机号查询用户失败", zap.Error(err)) | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// 缓存结果 | ||||
| 	r.cache.Set(ctx, cacheKey, &user, 10*time.Minute) | ||||
|  | ||||
| 	return &user, nil | ||||
| } | ||||
|  | ||||
| // Update 更新用户 | ||||
| func (r *UserRepository) Update(ctx context.Context, user *entities.User) error { | ||||
| 	if err := r.db.WithContext(ctx).Save(user).Error; err != nil { | ||||
| 		r.logger.Error("更新用户失败", zap.Error(err)) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// 清除相关缓存 | ||||
| 	r.deleteCacheByID(ctx, user.ID) | ||||
| 	r.deleteCacheByPhone(ctx, user.Phone) | ||||
|  | ||||
| 	r.logger.Info("用户更新成功", zap.String("user_id", user.ID)) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Delete 删除用户 | ||||
| func (r *UserRepository) Delete(ctx context.Context, id string) error { | ||||
| 	// 先获取用户信息用于清除缓存 | ||||
| 	user, err := r.GetByID(ctx, id) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if err := r.db.WithContext(ctx).Delete(&entities.User{}, "id = ?", id).Error; err != nil { | ||||
| 		r.logger.Error("删除用户失败", zap.Error(err)) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// 清除相关缓存 | ||||
| 	r.deleteCacheByID(ctx, id) | ||||
| 	r.deleteCacheByPhone(ctx, user.Phone) | ||||
|  | ||||
| 	r.logger.Info("用户删除成功", zap.String("user_id", id)) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // SoftDelete 软删除用户 | ||||
| func (r *UserRepository) SoftDelete(ctx context.Context, id string) error { | ||||
| 	// 先获取用户信息用于清除缓存 | ||||
| 	user, err := r.GetByID(ctx, id) | ||||
| 	if err != nil { | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	if err := r.db.WithContext(ctx).Delete(&entities.User{}, "id = ?", id).Error; err != nil { | ||||
| 		r.logger.Error("软删除用户失败", zap.Error(err)) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// 清除相关缓存 | ||||
| 	r.deleteCacheByID(ctx, id) | ||||
| 	r.deleteCacheByPhone(ctx, user.Phone) | ||||
|  | ||||
| 	r.logger.Info("用户软删除成功", zap.String("user_id", id)) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // Restore 恢复软删除的用户 | ||||
| func (r *UserRepository) Restore(ctx context.Context, id string) error { | ||||
| 	if err := r.db.WithContext(ctx).Unscoped().Model(&entities.User{}).Where("id = ?", id).Update("deleted_at", nil).Error; err != nil { | ||||
| 		r.logger.Error("恢复用户失败", zap.Error(err)) | ||||
| 		return err | ||||
| 	} | ||||
|  | ||||
| 	// 清除相关缓存 | ||||
| 	r.deleteCacheByID(ctx, id) | ||||
|  | ||||
| 	r.logger.Info("用户恢复成功", zap.String("user_id", id)) | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // List 分页获取用户列表 | ||||
| func (r *UserRepository) List(ctx context.Context, offset, limit int) ([]*entities.User, error) { | ||||
| 	var users []*entities.User | ||||
| 	if err := r.db.WithContext(ctx).Offset(offset).Limit(limit).Find(&users).Error; err != nil { | ||||
| 		r.logger.Error("查询用户列表失败", zap.Error(err)) | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return users, nil | ||||
| } | ||||
|  | ||||
| // Count 获取用户总数 | ||||
| func (r *UserRepository) Count(ctx context.Context) (int64, error) { | ||||
| 	var count int64 | ||||
| 	if err := r.db.WithContext(ctx).Model(&entities.User{}).Count(&count).Error; err != nil { | ||||
| 		r.logger.Error("统计用户数量失败", zap.Error(err)) | ||||
| 		return 0, err | ||||
| 	} | ||||
|  | ||||
| 	return count, nil | ||||
| } | ||||
|  | ||||
| // ExistsByPhone 检查手机号是否存在 | ||||
| func (r *UserRepository) ExistsByPhone(ctx context.Context, phone string) (bool, error) { | ||||
| 	var count int64 | ||||
| 	if err := r.db.WithContext(ctx).Model(&entities.User{}).Where("phone = ?", phone).Count(&count).Error; err != nil { | ||||
| 		r.logger.Error("检查手机号是否存在失败", zap.Error(err)) | ||||
| 		return false, err | ||||
| 	} | ||||
|  | ||||
| 	return count > 0, nil | ||||
| } | ||||
|  | ||||
| // 私有辅助方法 | ||||
|  | ||||
| // deleteCacheByID 根据ID删除缓存 | ||||
| func (r *UserRepository) deleteCacheByID(ctx context.Context, id string) { | ||||
| 	cacheKey := fmt.Sprintf("user:id:%s", id) | ||||
| 	if err := r.cache.Delete(ctx, cacheKey); err != nil { | ||||
| 		r.logger.Warn("删除用户ID缓存失败", zap.String("cache_key", cacheKey), zap.Error(err)) | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // deleteCacheByPhone 根据手机号删除缓存 | ||||
| func (r *UserRepository) deleteCacheByPhone(ctx context.Context, phone string) { | ||||
| 	cacheKey := fmt.Sprintf("user:phone:%s", phone) | ||||
| 	if err := r.cache.Delete(ctx, cacheKey); err != nil { | ||||
| 		r.logger.Warn("删除用户手机号缓存失败", zap.String("cache_key", cacheKey), zap.Error(err)) | ||||
| 	} | ||||
| } | ||||
| @@ -0,0 +1,71 @@ | ||||
| package repositories | ||||
|  | ||||
| import ( | ||||
| 	"context" | ||||
| 	"tyapi-server/internal/domains/user/entities" | ||||
| 	"tyapi-server/internal/domains/user/repositories/queries" | ||||
| 	"tyapi-server/internal/shared/interfaces" | ||||
| ) | ||||
|  | ||||
| // UserStats 用户统计信息 | ||||
| type UserStats struct { | ||||
| 	TotalUsers         int64 | ||||
| 	ActiveUsers        int64 | ||||
| 	TodayRegistrations int64 | ||||
| 	TodayLogins        int64 | ||||
| } | ||||
|  | ||||
| // UserRepository 用户仓储接口 | ||||
| type UserRepository interface { | ||||
| 	interfaces.Repository[entities.User] | ||||
|  | ||||
| 	// 基础查询 - 直接使用实体 | ||||
| 	GetByPhone(ctx context.Context, phone string) (*entities.User, error) | ||||
|  | ||||
| 	// 复杂查询 - 使用查询参数 | ||||
| 	ListUsers(ctx context.Context, query *queries.ListUsersQuery) ([]*entities.User, int64, error) | ||||
|  | ||||
| 	// 业务操作 | ||||
| 	ValidateUser(ctx context.Context, phone, password string) (*entities.User, error) | ||||
| 	UpdateLastLogin(ctx context.Context, userID string) error | ||||
| 	UpdatePassword(ctx context.Context, userID string, newPassword string) error | ||||
| 	CheckPassword(ctx context.Context, userID string, password string) (bool, error) | ||||
| 	ActivateUser(ctx context.Context, userID string) error | ||||
| 	DeactivateUser(ctx context.Context, userID string) error | ||||
|  | ||||
| 	// 统计信息 | ||||
| 	GetStats(ctx context.Context) (*UserStats, error) | ||||
| 	GetStatsByDateRange(ctx context.Context, startDate, endDate string) (*UserStats, error) | ||||
| } | ||||
|  | ||||
| // SMSCodeRepository 短信验证码仓储接口 | ||||
| type SMSCodeRepository interface { | ||||
| 	interfaces.Repository[entities.SMSCode] | ||||
|  | ||||
| 	// 基础查询 - 直接使用实体 | ||||
| 	GetByPhone(ctx context.Context, phone string) (*entities.SMSCode, error) | ||||
| 	GetLatestByPhone(ctx context.Context, phone string) (*entities.SMSCode, error) | ||||
| 	GetValidByPhone(ctx context.Context, phone string) (*entities.SMSCode, error) | ||||
| 	GetValidByPhoneAndScene(ctx context.Context, phone string, scene entities.SMSScene) (*entities.SMSCode, error) | ||||
|  | ||||
| 	// 复杂查询 - 使用查询参数 | ||||
| 	ListSMSCodes(ctx context.Context, query *queries.ListSMSCodesQuery) ([]*entities.SMSCode, int64, error) | ||||
|  | ||||
| 	// 业务操作 | ||||
| 	CreateCode(ctx context.Context, phone string, code string, purpose string) (entities.SMSCode, error) | ||||
| 	ValidateCode(ctx context.Context, phone string, code string, purpose string) (bool, error) | ||||
| 	InvalidateCode(ctx context.Context, phone string) error | ||||
| 	CheckSendFrequency(ctx context.Context, phone string, purpose string) (bool, error) | ||||
| 	GetTodaySendCount(ctx context.Context, phone string) (int64, error) | ||||
|  | ||||
| 	// 统计信息 | ||||
| 	GetCodeStats(ctx context.Context, phone string, days int) (*SMSCodeStats, error) | ||||
| } | ||||
|  | ||||
| // SMSCodeStats 短信验证码统计信息 | ||||
| type SMSCodeStats struct { | ||||
| 	TotalSent      int64 | ||||
| 	TotalValidated int64 | ||||
| 	SuccessRate    float64 | ||||
| 	TodaySent      int64 | ||||
| } | ||||
		Reference in New Issue
	
	Block a user