temp
This commit is contained in:
		| @@ -21,19 +21,7 @@ type EnterpriseInfo struct { | ||||
| 	UnifiedSocialCode string `gorm:"type:varchar(50);not null;index" json:"unified_social_code" comment:"统一社会信用代码"` | ||||
| 	LegalPersonName   string `gorm:"type:varchar(100);not null" json:"legal_person_name" comment:"法定代表人姓名"` | ||||
| 	LegalPersonID     string `gorm:"type:varchar(50);not null" json:"legal_person_id" comment:"法定代表人身份证号"` | ||||
|  | ||||
| 	// 认证状态 - 各环节的验证结果 | ||||
| 	IsOCRVerified    bool   `gorm:"default:false" json:"is_ocr_verified" comment:"OCR验证是否通过"` | ||||
| 	IsFaceVerified   bool   `gorm:"default:false" json:"is_face_verified" comment:"人脸识别是否通过"` | ||||
| 	IsCertified      bool   `gorm:"default:false" json:"is_certified" comment:"是否已完成认证"` | ||||
| 	VerificationData string `gorm:"type:text" json:"verification_data,omitempty" comment:"验证数据(JSON格式)"` | ||||
|  | ||||
| 	// OCR识别结果 - 从营业执照中自动识别的信息 | ||||
| 	OCRRawData    string  `gorm:"type:text" json:"ocr_raw_data,omitempty" comment:"OCR原始返回数据(JSON格式)"` | ||||
| 	OCRConfidence float64 `gorm:"type:decimal(5,2)" json:"ocr_confidence,omitempty" comment:"OCR识别置信度(0-1)"` | ||||
|  | ||||
| 	// 认证完成时间 | ||||
| 	CertifiedAt *time.Time `json:"certified_at,omitempty" comment:"认证完成时间"` | ||||
| 	LegalPersonPhone  string `gorm:"type:varchar(50);not null" json:"legal_person_phone" comment:"法定代表人手机号"` | ||||
|  | ||||
| 	// 时间戳字段 | ||||
| 	CreatedAt time.Time      `gorm:"autoCreateTime" json:"created_at" comment:"创建时间"` | ||||
| @@ -70,34 +58,6 @@ func (e *EnterpriseInfo) Validate() error { | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // IsFullyVerified 检查是否已完成所有验证 | ||||
| func (e *EnterpriseInfo) IsFullyVerified() bool { | ||||
| 	return e.IsOCRVerified && e.IsFaceVerified && e.IsCertified | ||||
| } | ||||
|  | ||||
| // UpdateOCRVerification 更新OCR验证状态 | ||||
| func (e *EnterpriseInfo) UpdateOCRVerification(isVerified bool, rawData string, confidence float64) { | ||||
| 	e.IsOCRVerified = isVerified | ||||
| 	e.OCRRawData = rawData | ||||
| 	e.OCRConfidence = confidence | ||||
| } | ||||
|  | ||||
| // UpdateFaceVerification 更新人脸识别验证状态 | ||||
| func (e *EnterpriseInfo) UpdateFaceVerification(isVerified bool) { | ||||
| 	e.IsFaceVerified = isVerified | ||||
| } | ||||
|  | ||||
| // CompleteCertification 完成认证 | ||||
| func (e *EnterpriseInfo) CompleteCertification() { | ||||
| 	e.IsCertified = true | ||||
| 	now := time.Now() | ||||
| 	e.CertifiedAt = &now | ||||
| } | ||||
|  | ||||
| // IsReadOnly 检查企业信息是否只读(认证完成后不可修改) | ||||
| func (e *EnterpriseInfo) IsReadOnly() bool { | ||||
| 	return e.IsCertified | ||||
| } | ||||
|  | ||||
| // BeforeCreate GORM钩子:创建前自动生成UUID | ||||
| func (e *EnterpriseInfo) BeforeCreate(tx *gorm.DB) error { | ||||
|   | ||||
| @@ -41,6 +41,7 @@ const ( | ||||
| 	SMSSceneResetPassword  SMSScene = "reset_password"  // 重置密码 - 忘记密码重置 | ||||
| 	SMSSceneBind           SMSScene = "bind"            // 绑定手机号 - 绑定新手机号 | ||||
| 	SMSSceneUnbind         SMSScene = "unbind"          // 解绑手机号 - 解绑当前手机号 | ||||
| 	SMSSceneCertification  SMSScene = "certification"   // 企业认证 - 企业入驻认证 | ||||
| ) | ||||
|  | ||||
| // BeforeCreate GORM钩子:创建前自动生成UUID | ||||
| @@ -195,6 +196,7 @@ func (s *SMSCode) IsSceneValid() bool { | ||||
| 		SMSSceneResetPassword, | ||||
| 		SMSSceneBind, | ||||
| 		SMSSceneUnbind, | ||||
| 		SMSSceneCertification, | ||||
| 	} | ||||
|  | ||||
| 	for _, scene := range validScenes { | ||||
| @@ -214,6 +216,7 @@ func (s *SMSCode) GetSceneName() string { | ||||
| 		SMSSceneResetPassword:  "重置密码", | ||||
| 		SMSSceneBind:           "绑定手机号", | ||||
| 		SMSSceneUnbind:         "解绑手机号", | ||||
| 		SMSSceneCertification:  "企业认证", | ||||
| 	} | ||||
|  | ||||
| 	if name, exists := sceneNames[s.Scene]; exists { | ||||
| @@ -283,6 +286,7 @@ func IsValidScene(scene SMSScene) bool { | ||||
| 		SMSSceneResetPassword, | ||||
| 		SMSSceneBind, | ||||
| 		SMSSceneUnbind, | ||||
| 		SMSSceneCertification, | ||||
| 	} | ||||
|  | ||||
| 	for _, validScene := range validScenes { | ||||
| @@ -302,6 +306,7 @@ func GetSceneName(scene SMSScene) string { | ||||
| 		SMSSceneResetPassword:  "重置密码", | ||||
| 		SMSSceneBind:           "绑定手机号", | ||||
| 		SMSSceneUnbind:         "解绑手机号", | ||||
| 		SMSSceneCertification:  "企业认证", | ||||
| 	} | ||||
|  | ||||
| 	if name, exists := sceneNames[scene]; exists { | ||||
|   | ||||
| @@ -1,7 +1,6 @@ | ||||
| package entities | ||||
|  | ||||
| import ( | ||||
| 	"errors" | ||||
| 	"fmt" | ||||
| 	"regexp" | ||||
| 	"time" | ||||
| @@ -11,6 +10,14 @@ import ( | ||||
| 	"gorm.io/gorm" | ||||
| ) | ||||
|  | ||||
| // UserType 用户类型枚举 | ||||
| type UserType string | ||||
|  | ||||
| const ( | ||||
| 	UserTypeNormal UserType = "user"  // 普通用户 | ||||
| 	UserTypeAdmin  UserType = "admin" // 管理员 | ||||
| ) | ||||
|  | ||||
| // User 用户实体 | ||||
| // 系统用户的核心信息,提供基础的账户管理功能 | ||||
| // 支持手机号登录,密码加密存储,实现Entity接口便于统一管理 | ||||
| @@ -20,6 +27,16 @@ type User struct { | ||||
| 	Phone    string `gorm:"uniqueIndex;type:varchar(20);not null" json:"phone" comment:"手机号码(登录账号)"` | ||||
| 	Password string `gorm:"type:varchar(255);not null" json:"-" comment:"登录密码(加密存储,不返回前端)"` | ||||
|  | ||||
| 	// 用户类型和基本信息 | ||||
| 	UserType string `gorm:"type:varchar(20);not null;default:'user'" json:"user_type" comment:"用户类型(user/admin)"` | ||||
| 	Username string `gorm:"type:varchar(100)" json:"username" comment:"用户名(管理员专用)"` | ||||
|  | ||||
| 	// 管理员特有字段 | ||||
| 	Active      bool       `gorm:"default:true" json:"is_active" comment:"账户是否激活"` | ||||
| 	LastLoginAt *time.Time `json:"last_login_at" comment:"最后登录时间"` | ||||
| 	LoginCount  int        `gorm:"default:0" json:"login_count" comment:"登录次数统计"` | ||||
| 	Permissions string     `gorm:"type:text" json:"permissions" comment:"权限列表(JSON格式存储)"` | ||||
|  | ||||
| 	// 时间戳字段 | ||||
| 	CreatedAt time.Time      `gorm:"autoCreateTime" json:"created_at" comment:"创建时间"` | ||||
| 	UpdatedAt time.Time      `gorm:"autoUpdateTime" json:"updated_at" comment:"更新时间"` | ||||
| @@ -78,6 +95,42 @@ func (u *User) Validate() error { | ||||
|  | ||||
| // ================ 业务方法 ================ | ||||
|  | ||||
| // IsAdmin 检查是否为管理员 | ||||
| func (u *User) IsAdmin() bool { | ||||
| 	return u.UserType == string(UserTypeAdmin) | ||||
| } | ||||
|  | ||||
| // IsNormalUser 检查是否为普通用户 | ||||
| func (u *User) IsNormalUser() bool { | ||||
| 	return u.UserType == string(UserTypeNormal) | ||||
| } | ||||
|  | ||||
| // SetUserType 设置用户类型 | ||||
| func (u *User) SetUserType(userType UserType) { | ||||
| 	u.UserType = string(userType) | ||||
| } | ||||
|  | ||||
| // UpdateLastLoginAt 更新最后登录时间 | ||||
| func (u *User) UpdateLastLoginAt() { | ||||
| 	now := time.Now() | ||||
| 	u.LastLoginAt = &now | ||||
| } | ||||
|  | ||||
| // IncrementLoginCount 增加登录次数 | ||||
| func (u *User) IncrementLoginCount() { | ||||
| 	u.LoginCount++ | ||||
| } | ||||
|  | ||||
| // Activate 激活用户账户 | ||||
| func (u *User) Activate() { | ||||
| 	u.Active = true | ||||
| } | ||||
|  | ||||
| // Deactivate 停用用户账户 | ||||
| func (u *User) Deactivate() { | ||||
| 	u.Active = false | ||||
| } | ||||
|  | ||||
| // ChangePassword 修改密码 | ||||
| // 验证旧密码,检查新密码强度,更新密码 | ||||
| func (u *User) ChangePassword(oldPassword, newPassword, confirmPassword string) error { | ||||
| @@ -168,6 +221,11 @@ func (u *User) CanLogin() bool { | ||||
| 		return false | ||||
| 	} | ||||
|  | ||||
| 	// 如果是管理员,检查是否激活 | ||||
| 	if u.IsAdmin() && !u.Active { | ||||
| 		return false | ||||
| 	} | ||||
|  | ||||
| 	return true | ||||
| } | ||||
|  | ||||
| @@ -205,7 +263,7 @@ func (u *User) GetMaskedPhone() string { | ||||
| 	return u.Phone[:3] + "****" + u.Phone[len(u.Phone)-4:] | ||||
| } | ||||
|  | ||||
| // ================ 私有辅助方法 ================ | ||||
| // ================ 私有方法 ================ | ||||
|  | ||||
| // hashPassword 加密密码 | ||||
| func (u *User) hashPassword(password string) (string, error) { | ||||
| @@ -218,61 +276,30 @@ func (u *User) hashPassword(password string) (string, error) { | ||||
|  | ||||
| // validatePasswordStrength 验证密码强度 | ||||
| func (u *User) validatePasswordStrength(password string) error { | ||||
| 	if len(password) < 8 { | ||||
| 		return NewValidationError("密码长度至少8位") | ||||
| 	if len(password) < 6 { | ||||
| 		return NewValidationError("密码长度不能少于6位") | ||||
| 	} | ||||
|  | ||||
| 	if len(password) > 128 { | ||||
| 		return NewValidationError("密码长度不能超过128位") | ||||
| 	if len(password) > 20 { | ||||
| 		return NewValidationError("密码长度不能超过20位") | ||||
| 	} | ||||
|  | ||||
| 	// 检查是否包含数字 | ||||
| 	hasDigit := regexp.MustCompile(`[0-9]`).MatchString(password) | ||||
| 	if !hasDigit { | ||||
| 		return NewValidationError("密码必须包含数字") | ||||
| 	} | ||||
|  | ||||
| 	// 检查是否包含字母 | ||||
| 	hasLetter := regexp.MustCompile(`[a-zA-Z]`).MatchString(password) | ||||
| 	if !hasLetter { | ||||
| 		return NewValidationError("密码必须包含字母") | ||||
| 	} | ||||
|  | ||||
| 	// 检查是否包含特殊字符(可选,可以根据需求调整) | ||||
| 	hasSpecial := regexp.MustCompile(`[!@#$%^&*()_+\-=\[\]{};':"\\|,.<>\/?]`).MatchString(password) | ||||
| 	if !hasSpecial { | ||||
| 		return NewValidationError("密码必须包含特殊字符") | ||||
| 	} | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| // ================ 静态工具方法 ================ | ||||
|  | ||||
| // IsValidPhoneFormat 验证手机号格式(静态方法) | ||||
| // IsValidPhoneFormat 验证手机号格式 | ||||
| func IsValidPhoneFormat(phone string) bool { | ||||
| 	if phone == "" { | ||||
| 		return false | ||||
| 	} | ||||
| 	// 中国手机号验证(11位数字,以1开头) | ||||
| 	pattern := `^1[3-9]\d{9}$` | ||||
| 	matched, _ := regexp.MatchString(pattern, phone) | ||||
| 	return matched | ||||
| } | ||||
|  | ||||
| // NewUser 创建新用户(工厂方法) | ||||
| // NewUser 创建新用户 | ||||
| func NewUser(phone, password string) (*User, error) { | ||||
| 	user := &User{ | ||||
| 		ID:    "", // 由数据库或调用方设置 | ||||
| 		Phone: phone, | ||||
| 		Phone:    phone, | ||||
| 		UserType: string(UserTypeNormal), // 默认为普通用户 | ||||
| 		Active:   true, | ||||
| 	} | ||||
|  | ||||
| 	// 验证手机号 | ||||
| 	if err := user.SetPhone(phone); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	// 设置密码 | ||||
| 	if err := user.SetPassword(password); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
| @@ -280,41 +307,60 @@ func NewUser(phone, password string) (*User, error) { | ||||
| 	return user, nil | ||||
| } | ||||
|  | ||||
| // TableName 指定表名 | ||||
| // NewAdminUser 创建新管理员用户 | ||||
| func NewAdminUser(phone, password, username string) (*User, error) { | ||||
| 	user := &User{ | ||||
| 		Phone:    phone, | ||||
| 		Username: username, | ||||
| 		UserType: string(UserTypeAdmin), | ||||
| 		Active:   true, | ||||
| 	} | ||||
|  | ||||
| 	if err := user.SetPassword(password); err != nil { | ||||
| 		return nil, err | ||||
| 	} | ||||
|  | ||||
| 	return user, nil | ||||
| } | ||||
|  | ||||
| // TableName 指定数据库表名 | ||||
| func (User) TableName() string { | ||||
| 	return "users" | ||||
| } | ||||
|  | ||||
| // ValidationError 验证错误 | ||||
| // 自定义验证错误类型,提供结构化的错误信息 | ||||
| // ================ 错误处理 ================ | ||||
|  | ||||
| type ValidationError struct { | ||||
| 	Message string | ||||
| } | ||||
|  | ||||
| // Error 实现error接口 | ||||
| func (e *ValidationError) Error() string { | ||||
| 	return e.Message | ||||
| } | ||||
|  | ||||
| // NewValidationError 创建新的验证错误 | ||||
| // 工厂方法,用于创建验证错误实例 | ||||
| func NewValidationError(message string) *ValidationError { | ||||
| 	return &ValidationError{Message: message} | ||||
| } | ||||
|  | ||||
| // IsValidationError 检查是否为验证错误 | ||||
| func IsValidationError(err error) bool { | ||||
| 	var validationErr *ValidationError | ||||
| 	return errors.As(err, &validationErr) | ||||
| 	_, ok := err.(*ValidationError) | ||||
| 	return ok | ||||
| } | ||||
|  | ||||
| // UserCache 用户缓存结构体 | ||||
| // 专门用于缓存序列化,包含Password字段 | ||||
| // ================ 缓存相关 ================ | ||||
|  | ||||
| type UserCache struct { | ||||
| 	// 基础标识 | ||||
| 	ID       string `json:"id" comment:"用户唯一标识"` | ||||
| 	Phone    string `json:"phone" comment:"手机号码(登录账号)"` | ||||
| 	Password string `json:"password" comment:"登录密码(加密存储)"` | ||||
| 	UserType string `json:"user_type" comment:"用户类型"` | ||||
| 	Username string `json:"username" comment:"用户名"` | ||||
|  | ||||
| 	Active      bool       `gorm:"default:true" json:"is_active" comment:"账户是否激活"` | ||||
| 	LastLoginAt *time.Time `json:"last_login_at" comment:"最后登录时间"` | ||||
| 	LoginCount  int        `gorm:"default:0" json:"login_count" comment:"登录次数统计"` | ||||
| 	Permissions string     `gorm:"type:text" json:"permissions" comment:"权限列表(JSON格式存储)"` | ||||
|  | ||||
| 	// 时间戳字段 | ||||
| 	CreatedAt time.Time      `json:"created_at" comment:"创建时间"` | ||||
| @@ -322,24 +368,38 @@ type UserCache struct { | ||||
| 	DeletedAt gorm.DeletedAt `json:"deleted_at" comment:"软删除时间"` | ||||
| } | ||||
|  | ||||
| // ToCache 转换为缓存结构体 | ||||
| // ToCache 转换为缓存结构 | ||||
| func (u *User) ToCache() *UserCache { | ||||
| 	return &UserCache{ | ||||
| 		ID:        u.ID, | ||||
| 		Phone:     u.Phone, | ||||
| 		Password:  u.Password, | ||||
| 		UserType:  u.UserType, | ||||
| 		Username:  u.Username, | ||||
| 		CreatedAt: u.CreatedAt, | ||||
| 		UpdatedAt: u.UpdatedAt, | ||||
| 		DeletedAt: u.DeletedAt, | ||||
| 		// 补充所有字段 | ||||
| 		// 管理员特有字段 | ||||
| 		Active:      u.Active, | ||||
| 		LastLoginAt: u.LastLoginAt, | ||||
| 		LoginCount:  u.LoginCount, | ||||
| 		Permissions: u.Permissions, | ||||
| 	} | ||||
| } | ||||
|  | ||||
| // FromCache 从缓存结构体转换 | ||||
| // FromCache 从缓存结构恢复 | ||||
| func (u *User) FromCache(cache *UserCache) { | ||||
| 	u.ID = cache.ID | ||||
| 	u.Phone = cache.Phone | ||||
| 	u.Password = cache.Password | ||||
| 	u.UserType = cache.UserType | ||||
| 	u.Username = cache.Username | ||||
| 	u.CreatedAt = cache.CreatedAt | ||||
| 	u.UpdatedAt = cache.UpdatedAt | ||||
| 	u.DeletedAt = cache.DeletedAt | ||||
| 	u.Active = cache.Active | ||||
| 	u.LastLoginAt = cache.LastLoginAt | ||||
| 	u.LoginCount = cache.LoginCount | ||||
| 	u.Permissions = cache.Permissions | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user