package auth import ( "context" "database/sql" "encoding/hex" "fmt" "qnc-server/app/main/api/internal/service" "qnc-server/app/main/api/internal/svc" "qnc-server/app/main/api/internal/types" "qnc-server/app/main/model" "qnc-server/common/xerr" "qnc-server/pkg/lzkit/crypto" "github.com/bytedance/sonic" "github.com/pkg/errors" "github.com/redis/go-redis/v9" "github.com/zeromicro/go-zero/core/logx" "github.com/zeromicro/go-zero/core/stores/sqlx" ) type SmsAuthorizationLogic struct { logx.Logger ctx context.Context svcCtx *svc.ServiceContext } func NewSmsAuthorizationLogic(ctx context.Context, svcCtx *svc.ServiceContext) *SmsAuthorizationLogic { return &SmsAuthorizationLogic{ Logger: logx.WithContext(ctx), ctx: ctx, svcCtx: svcCtx, } } func (l *SmsAuthorizationLogic) SmsAuthorization(req *types.SmsAuthorizationReq) (resp *types.SmsAuthorizationResp, err error) { order, err := l.svcCtx.OrderModel.FindOneByOrderNo(l.ctx, req.OrderNo) if err != nil { return nil, errors.Wrapf(xerr.NewErrCode(xerr.DB_ERROR), "查询订单失败: %v", err) } if order.Status != "paid" { return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "订单未支付") } authorization, err := l.svcCtx.AuthorizationModel.FindOneByOrderId(l.ctx, order.Id) if err != nil { return nil, errors.Wrapf(xerr.NewErrCode(xerr.DB_ERROR), "查询授权信息失败: %v", err) } if authorization.Status != "pending" { return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "授权信息状态不正确") } key, decodeErr := hex.DecodeString(l.svcCtx.Config.Encrypt.SecretKey) if decodeErr != nil { return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "查询授权信息失败: %v", decodeErr) } name, err := crypto.AesDecrypt(authorization.TargetName, key) if err != nil { return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "解密姓名失败: %v", err) } idCard, err := crypto.AesDecrypt(authorization.TargetIdcard, key) if err != nil { return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "解密身份证号失败: %v", err) } threeFactorVerificationResp, err := l.svcCtx.VerificationService.ThreeFactorVerification(service.ThreeFactorVerificationRequest{ Mobile: req.Mobile, Name: string(name), IDCard: string(idCard), }) if err != nil { return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "三要素验证失败: %v", err) } if !threeFactorVerificationResp.Passed { return nil, errors.Wrapf(xerr.NewErrMsg("该手机号码不是被查询人的手机号"), "三要素验证失败") } secretKey := l.svcCtx.Config.Encrypt.SecretKey encryptedMobile, err := crypto.EncryptMobile(req.Mobile, secretKey) if err != nil { return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "加密手机号失败: %v", err) } // 检查手机号是否在一分钟内已发送过验证码 redisKey := fmt.Sprintf("%s:%s", "authorization", encryptedMobile) cacheCode, err := l.svcCtx.Redis.Get(redisKey) if err != nil { if errors.Is(err, redis.Nil) { return nil, errors.Wrapf(xerr.NewErrMsg("验证码已过期"), "验证码过期: %s", encryptedMobile) } return nil, errors.Wrapf(xerr.NewErrCode(xerr.DB_ERROR), "读取验证码redis缓存失败, mobile: %s, err: %+v", encryptedMobile, err) } if cacheCode != req.Code { return nil, errors.Wrapf(xerr.NewErrMsg("验证码不正确"), "验证码不正确: %s", encryptedMobile) } err = l.svcCtx.AuthorizationModel.Trans(l.ctx, func(ctx context.Context, session sqlx.Session) error { order, err := l.svcCtx.OrderModel.FindOne(ctx, authorization.OrderId) if err != nil { return errors.Wrapf(err, "查询订单失败") } redisKey := fmt.Sprintf(types.QueryCacheKey, order.UserId, order.OrderNo) cache, cacheErr := l.svcCtx.Redis.GetCtx(ctx, redisKey) if cacheErr != nil { return fmt.Errorf("获取缓存内容失败: %+v", cacheErr) } var data types.QueryCacheLoad err = sonic.Unmarshal([]byte(cache), &data) if err != nil { return fmt.Errorf("解析缓存内容失败: %+v", err) } // 插入新queryModel query := &model.Query{ OrderId: authorization.OrderId, UserId: authorization.UserId, ProductId: order.ProductId, QueryParams: data.Params, QueryState: "pending", } _, insertQueryErr := l.svcCtx.QueryModel.Insert(ctx, session, query) if insertQueryErr != nil { return errors.Wrapf(insertQueryErr, "保存查询失败") } authorization.GrantType = sql.NullString{ String: model.AuthorizationGrantTypeSms, Valid: true, } // 更新主授权状态 authorization.Status = model.AuthorizationStatusSuccess _, err = l.svcCtx.AuthorizationModel.Update(ctx, session, authorization) if err != nil { return errors.Wrapf(err, "更新授权状态失败") } if asyncErr := l.svcCtx.AsynqService.SendQueryTask(authorization.OrderId); asyncErr != nil { logx.Errorf("异步任务调度失败: %v", asyncErr) return asyncErr } return nil }) if err != nil { return nil, errors.Wrapf(xerr.NewErrCode(xerr.SERVER_COMMON_ERROR), "更新授权信息失败: %v", err) } return &types.SmsAuthorizationResp{ OrderID: authorization.OrderId, Passed: true, }, nil }