| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | package services | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"time" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"go.uber.org/zap" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"tyapi-server/internal/config" | 
					
						
							|  |  |  | 	"tyapi-server/internal/domains/user/entities" | 
					
						
							|  |  |  | 	"tyapi-server/internal/domains/user/repositories" | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	"tyapi-server/internal/infrastructure/external/sms" | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	"tyapi-server/internal/shared/interfaces" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SMSCodeService 短信验证码服务 | 
					
						
							|  |  |  | type SMSCodeService struct { | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	repo      repositories.SMSCodeRepository | 
					
						
							|  |  |  | 	smsClient *sms.AliSMSService | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	cache     interfaces.CacheService | 
					
						
							|  |  |  | 	config    config.SMSConfig | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	appConfig config.AppConfig | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	logger    *zap.Logger | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // NewSMSCodeService 创建短信验证码服务 | 
					
						
							|  |  |  | func NewSMSCodeService( | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	repo repositories.SMSCodeRepository, | 
					
						
							|  |  |  | 	smsClient *sms.AliSMSService, | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	cache interfaces.CacheService, | 
					
						
							|  |  |  | 	config config.SMSConfig, | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	appConfig config.AppConfig, | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	logger *zap.Logger, | 
					
						
							|  |  |  | ) *SMSCodeService { | 
					
						
							|  |  |  | 	return &SMSCodeService{ | 
					
						
							|  |  |  | 		repo:      repo, | 
					
						
							|  |  |  | 		smsClient: smsClient, | 
					
						
							|  |  |  | 		cache:     cache, | 
					
						
							|  |  |  | 		config:    config, | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 		appConfig: appConfig, | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 		logger:    logger, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SendCode 发送验证码 | 
					
						
							|  |  |  | func (s *SMSCodeService) SendCode(ctx context.Context, phone string, scene entities.SMSScene, clientIP, userAgent string) error { | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	// 1. 生成验证码 | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	code := s.smsClient.GenerateCode(s.config.CodeLength) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	// 2. 使用工厂方法创建SMS验证码记录 | 
					
						
							| 
									
										
										
										
											2025-07-11 21:05:58 +08:00
										 |  |  | 	smsCode, err := entities.NewSMSCode(phone, code, scene, s.config.ExpireTime, clientIP, userAgent) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return fmt.Errorf("创建验证码记录失败: %w", err) | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	// 4. 保存验证码 | 
					
						
							|  |  |  | 	*smsCode, err = s.repo.Create(ctx, *smsCode) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 		s.logger.Error("保存短信验证码失败", | 
					
						
							| 
									
										
										
										
											2025-07-11 21:05:58 +08:00
										 |  |  | 			zap.String("phone", smsCode.GetMaskedPhone()), | 
					
						
							|  |  |  | 			zap.String("scene", smsCode.GetSceneName()), | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 			zap.Error(err)) | 
					
						
							|  |  |  | 		return fmt.Errorf("保存验证码失败: %w", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	// 5. 发送短信 | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	if err := s.smsClient.SendVerificationCode(ctx, phone, code); err != nil { | 
					
						
							|  |  |  | 		// 记录发送失败但不删除验证码记录,让其自然过期 | 
					
						
							|  |  |  | 		s.logger.Error("发送短信验证码失败", | 
					
						
							| 
									
										
										
										
											2025-07-11 21:05:58 +08:00
										 |  |  | 			zap.String("phone", smsCode.GetMaskedPhone()), | 
					
						
							|  |  |  | 			zap.String("code", smsCode.GetMaskedCode()), | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 			zap.Error(err)) | 
					
						
							|  |  |  | 		return fmt.Errorf("短信发送失败: %w", err) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	// 6. 更新发送记录缓存 | 
					
						
							|  |  |  | 	s.updateSendRecord(ctx, phone, scene) | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	s.logger.Info("短信验证码发送成功", | 
					
						
							| 
									
										
										
										
											2025-07-11 21:05:58 +08:00
										 |  |  | 		zap.String("phone", smsCode.GetMaskedPhone()), | 
					
						
							|  |  |  | 		zap.String("scene", smsCode.GetSceneName()), | 
					
						
							|  |  |  | 		zap.String("remaining_time", smsCode.GetRemainingTime().String())) | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // VerifyCode 验证验证码 | 
					
						
							|  |  |  | func (s *SMSCodeService) VerifyCode(ctx context.Context, phone, code string, scene entities.SMSScene) error { | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	// 开发模式下跳过验证码校验 | 
					
						
							|  |  |  | 	if s.appConfig.IsDevelopment() { | 
					
						
							|  |  |  | 		s.logger.Info("开发模式:验证码校验已跳过", | 
					
						
							|  |  |  | 			zap.String("phone", phone), | 
					
						
							|  |  |  | 			zap.String("scene", string(scene)), | 
					
						
							|  |  |  | 			zap.String("code", code)) | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-11 21:05:58 +08:00
										 |  |  | 	// 1. 根据手机号和场景获取有效的验证码记录 | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	smsCode, err := s.repo.GetValidByPhoneAndScene(ctx, phone, scene) | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return fmt.Errorf("验证码无效或已过期") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	// 2. 检查场景是否匹配 | 
					
						
							|  |  |  | 	if smsCode.Scene != scene { | 
					
						
							|  |  |  | 		return fmt.Errorf("验证码错误或已过期") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 3. 使用实体的验证方法 | 
					
						
							| 
									
										
										
										
											2025-07-11 21:05:58 +08:00
										 |  |  | 	if err := smsCode.VerifyCode(code); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	// 4. 保存更新后的验证码状态 | 
					
						
							|  |  |  | 	if err := s.repo.Update(ctx, *smsCode); err != nil { | 
					
						
							| 
									
										
										
										
											2025-07-11 21:05:58 +08:00
										 |  |  | 		s.logger.Error("更新验证码状态失败", | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 			zap.String("code_id", smsCode.ID), | 
					
						
							|  |  |  | 			zap.Error(err)) | 
					
						
							|  |  |  | 		return fmt.Errorf("验证码状态更新失败") | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	s.logger.Info("短信验证码验证成功", | 
					
						
							| 
									
										
										
										
											2025-07-11 21:05:58 +08:00
										 |  |  | 		zap.String("phone", smsCode.GetMaskedPhone()), | 
					
						
							|  |  |  | 		zap.String("scene", smsCode.GetSceneName())) | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-11 21:05:58 +08:00
										 |  |  | // CanResendCode 检查是否可以重新发送验证码 | 
					
						
							|  |  |  | func (s *SMSCodeService) CanResendCode(ctx context.Context, phone string, scene entities.SMSScene) (bool, error) { | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	// 1. 获取最近的验证码记录(按场景) | 
					
						
							|  |  |  | 	recentCode, err := s.repo.GetValidByPhoneAndScene(ctx, phone, scene) | 
					
						
							| 
									
										
										
										
											2025-07-11 21:05:58 +08:00
										 |  |  | 	if err != nil { | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 		// 如果没有该场景的记录,可以发送 | 
					
						
							| 
									
										
										
										
											2025-07-11 21:05:58 +08:00
										 |  |  | 		return true, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 2. 使用实体的方法检查是否可以重新发送 | 
					
						
							|  |  |  | 	canResend := recentCode.CanResend(s.config.RateLimit.MinInterval) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 3. 记录检查结果 | 
					
						
							|  |  |  | 	if !canResend { | 
					
						
							|  |  |  | 		remainingTime := s.config.RateLimit.MinInterval - time.Since(recentCode.CreatedAt) | 
					
						
							|  |  |  | 		s.logger.Info("验证码发送频率限制", | 
					
						
							|  |  |  | 			zap.String("phone", recentCode.GetMaskedPhone()), | 
					
						
							|  |  |  | 			zap.String("scene", recentCode.GetSceneName()), | 
					
						
							|  |  |  | 			zap.Duration("remaining_wait_time", remainingTime)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return canResend, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // GetCodeStatus 获取验证码状态信息 | 
					
						
							|  |  |  | func (s *SMSCodeService) GetCodeStatus(ctx context.Context, phone string, scene entities.SMSScene) (map[string]interface{}, error) { | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	// 1. 获取最近的验证码记录(按场景) | 
					
						
							|  |  |  | 	recentCode, err := s.repo.GetValidByPhoneAndScene(ctx, phone, scene) | 
					
						
							| 
									
										
										
										
											2025-07-11 21:05:58 +08:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return map[string]interface{}{ | 
					
						
							|  |  |  | 			"has_code": false, | 
					
						
							|  |  |  | 			"message":  "没有找到验证码记录", | 
					
						
							|  |  |  | 		}, nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 2. 构建状态信息 | 
					
						
							|  |  |  | 	status := map[string]interface{}{ | 
					
						
							|  |  |  | 		"has_code":       true, | 
					
						
							|  |  |  | 		"is_valid":       recentCode.IsValid(), | 
					
						
							|  |  |  | 		"is_expired":     recentCode.IsExpired(), | 
					
						
							|  |  |  | 		"is_used":        recentCode.Used, | 
					
						
							|  |  |  | 		"remaining_time": recentCode.GetRemainingTime().String(), | 
					
						
							|  |  |  | 		"scene":          recentCode.GetSceneName(), | 
					
						
							|  |  |  | 		"can_resend":     recentCode.CanResend(s.config.RateLimit.MinInterval), | 
					
						
							|  |  |  | 		"created_at":     recentCode.CreatedAt, | 
					
						
							|  |  |  | 		"security_info":  recentCode.GetSecurityInfo(), | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return status, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | // checkRateLimit 检查发送频率限制 | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | func (s *SMSCodeService) CheckRateLimit(ctx context.Context, phone string, scene entities.SMSScene) error { | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	now := time.Now() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 检查最小发送间隔 | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	lastSentKey := fmt.Sprintf("sms:last_sent:%s:%s", scene, phone) | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	var lastSent time.Time | 
					
						
							|  |  |  | 	if err := s.cache.Get(ctx, lastSentKey, &lastSent); err == nil { | 
					
						
							|  |  |  | 		if now.Sub(lastSent) < s.config.RateLimit.MinInterval { | 
					
						
							|  |  |  | 			return fmt.Errorf("请等待 %v 后再试", s.config.RateLimit.MinInterval) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 检查每小时发送限制 | 
					
						
							|  |  |  | 	hourlyKey := fmt.Sprintf("sms:hourly:%s:%s", phone, now.Format("2006010215")) | 
					
						
							|  |  |  | 	var hourlyCount int | 
					
						
							|  |  |  | 	if err := s.cache.Get(ctx, hourlyKey, &hourlyCount); err == nil { | 
					
						
							|  |  |  | 		if hourlyCount >= s.config.RateLimit.HourlyLimit { | 
					
						
							|  |  |  | 			return fmt.Errorf("每小时最多发送 %d 条短信", s.config.RateLimit.HourlyLimit) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 检查每日发送限制 | 
					
						
							|  |  |  | 	dailyKey := fmt.Sprintf("sms:daily:%s:%s", phone, now.Format("20060102")) | 
					
						
							|  |  |  | 	var dailyCount int | 
					
						
							|  |  |  | 	if err := s.cache.Get(ctx, dailyKey, &dailyCount); err == nil { | 
					
						
							|  |  |  | 		if dailyCount >= s.config.RateLimit.DailyLimit { | 
					
						
							|  |  |  | 			return fmt.Errorf("每日最多发送 %d 条短信", s.config.RateLimit.DailyLimit) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // updateSendRecord 更新发送记录 | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | func (s *SMSCodeService) updateSendRecord(ctx context.Context, phone string, scene entities.SMSScene) { | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	now := time.Now() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 更新最后发送时间 | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	lastSentKey := fmt.Sprintf("sms:last_sent:%s:%s", scene, phone) | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | 	s.cache.Set(ctx, lastSentKey, now, s.config.RateLimit.MinInterval) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 更新每小时计数 | 
					
						
							|  |  |  | 	hourlyKey := fmt.Sprintf("sms:hourly:%s:%s", phone, now.Format("2006010215")) | 
					
						
							|  |  |  | 	var hourlyCount int | 
					
						
							|  |  |  | 	if err := s.cache.Get(ctx, hourlyKey, &hourlyCount); err == nil { | 
					
						
							|  |  |  | 		s.cache.Set(ctx, hourlyKey, hourlyCount+1, time.Hour) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		s.cache.Set(ctx, hourlyKey, 1, time.Hour) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// 更新每日计数 | 
					
						
							|  |  |  | 	dailyKey := fmt.Sprintf("sms:daily:%s:%s", phone, now.Format("20060102")) | 
					
						
							|  |  |  | 	var dailyCount int | 
					
						
							|  |  |  | 	if err := s.cache.Get(ctx, dailyKey, &dailyCount); err == nil { | 
					
						
							|  |  |  | 		s.cache.Set(ctx, dailyKey, dailyCount+1, 24*time.Hour) | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		s.cache.Set(ctx, dailyKey, 1, 24*time.Hour) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // CleanExpiredCodes 清理过期验证码 | 
					
						
							|  |  |  | func (s *SMSCodeService) CleanExpiredCodes(ctx context.Context) error { | 
					
						
							| 
									
										
										
										
											2025-07-13 16:36:20 +08:00
										 |  |  | 	return s.repo.DeleteBatch(ctx, []string{}) | 
					
						
							| 
									
										
										
										
											2025-07-02 16:17:59 +08:00
										 |  |  | } |