138 lines
3.5 KiB
Go
138 lines
3.5 KiB
Go
package authlogic
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"regexp"
|
|
"strings"
|
|
"tianyuan-api/apps/user/internal/model"
|
|
|
|
"github.com/zeromicro/go-zero/core/stores/redis"
|
|
"github.com/zeromicro/go-zero/core/stores/sqlc"
|
|
|
|
"tianyuan-api/apps/user/internal/svc"
|
|
"tianyuan-api/apps/user/user"
|
|
|
|
"github.com/zeromicro/go-zero/core/logx"
|
|
)
|
|
|
|
type RegisterUserLogic struct {
|
|
ctx context.Context
|
|
svcCtx *svc.ServiceContext
|
|
logx.Logger
|
|
}
|
|
|
|
func NewRegisterUserLogic(ctx context.Context, svcCtx *svc.ServiceContext) *RegisterUserLogic {
|
|
return &RegisterUserLogic{
|
|
ctx: ctx,
|
|
svcCtx: svcCtx,
|
|
Logger: logx.WithContext(ctx),
|
|
}
|
|
}
|
|
|
|
// 注册接口
|
|
func (l *RegisterUserLogic) RegisterUser(in *user.RegisterReq) (*user.EmptyResponse, error) {
|
|
// 检查密码是否一致
|
|
if in.Password != in.ConfirmPassword {
|
|
return nil, errors.New("密码不一致")
|
|
}
|
|
// 检查密码强度
|
|
if err := checkPasswordStrength(in.Password); err != nil {
|
|
return nil, err
|
|
}
|
|
// 校验手机号码
|
|
err := validatePhoneNumber(in.Phone)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
// 从 Redis 获取验证码
|
|
savedCode, err := l.svcCtx.Redis.Get(fmt.Sprintf("register:%s", in.Phone))
|
|
if err != nil {
|
|
if errors.Is(err, redis.Nil) {
|
|
return nil, errors.New("验证码已过期")
|
|
}
|
|
return nil, err
|
|
}
|
|
|
|
// 验证码不匹配
|
|
if savedCode != in.Code {
|
|
return nil, errors.New("验证码不正确")
|
|
}
|
|
// 检查用户名是否已经存在
|
|
existingUser, err := l.svcCtx.UserModel.FindOneByUsername(l.ctx, in.Username)
|
|
if err != nil && err != sqlc.ErrNotFound {
|
|
// 如果发生其他错误,返回错误
|
|
return nil, err
|
|
}
|
|
if existingUser != nil {
|
|
// 用户名已经存在,返回错误
|
|
return nil, errors.New("用户名已存在,请选择其他用户名")
|
|
}
|
|
|
|
// 检查手机号是否已经存在
|
|
existingPhone, err := l.svcCtx.UserModel.FindOneByPhone(l.ctx, in.Phone)
|
|
if err != nil && err != sqlc.ErrNotFound {
|
|
// 如果发生其他错误,返回错误
|
|
return nil, err
|
|
}
|
|
if existingPhone != nil {
|
|
// 用户名已经存在,返回错误
|
|
return nil, errors.New("手机号码已存在,请选择其他用户名")
|
|
}
|
|
// 加密密码
|
|
hashedPassword := hashPassword(in.Password)
|
|
|
|
// 构建 Users 结构体
|
|
users := &model.Users{
|
|
Username: in.Username,
|
|
Password: hashedPassword,
|
|
Phone: in.Phone,
|
|
AuthStatus: "unverified",
|
|
}
|
|
|
|
// 调用 Insert 方法插入用户数据
|
|
_, err = l.svcCtx.UserModel.Insert(l.ctx, users)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &user.EmptyResponse{}, nil
|
|
}
|
|
|
|
// 密码强度检查
|
|
func checkPasswordStrength(password string) error {
|
|
// 检查密码长度是否不少于8位
|
|
if len(password) < 8 {
|
|
return errors.New("密码长度不能少于8位")
|
|
}
|
|
|
|
// 检查密码是否为简单重复的字符(如"11111111" 或 "aaaaaaaa"等)
|
|
firstChar := password[0]
|
|
if strings.Count(password, string(firstChar)) == len(password) {
|
|
return errors.New("密码不能是重复的字符")
|
|
}
|
|
|
|
// 正则表达式:密码必须包含数字或字母,不能是全符号
|
|
var passwordRegex = `^[A-Za-z0-9]+$`
|
|
match, _ := regexp.MatchString(passwordRegex, password)
|
|
if !match {
|
|
return errors.New("密码只能包含字母和数字")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// 校验手机号码的函数
|
|
func validatePhoneNumber(phone string) error {
|
|
// 定义正则表达式,匹配中国大陆的手机号格式
|
|
var phoneRegex = `^1[3-9]\d{9}$`
|
|
|
|
// 检查手机号是否匹配正则表达式
|
|
match, _ := regexp.MatchString(phoneRegex, phone)
|
|
if !match {
|
|
return errors.New("手机号码格式不正确")
|
|
}
|
|
return nil
|
|
}
|