295 lines
8.2 KiB
Go
295 lines
8.2 KiB
Go
package http
|
||
|
||
import (
|
||
"strings"
|
||
|
||
"tyapi-server/internal/shared/interfaces"
|
||
|
||
"github.com/gin-gonic/gin"
|
||
"github.com/go-playground/locales/zh"
|
||
ut "github.com/go-playground/universal-translator"
|
||
"github.com/go-playground/validator/v10"
|
||
zh_translations "github.com/go-playground/validator/v10/translations/zh"
|
||
)
|
||
|
||
// RequestValidatorZh 中文验证器实现
|
||
type RequestValidatorZh struct {
|
||
validator *validator.Validate
|
||
translator ut.Translator
|
||
response interfaces.ResponseBuilder
|
||
}
|
||
|
||
// NewRequestValidatorZh 创建支持中文翻译的请求验证器
|
||
func NewRequestValidatorZh(response interfaces.ResponseBuilder) interfaces.RequestValidator {
|
||
// 创建验证器实例
|
||
validate := validator.New()
|
||
|
||
// 创建中文locale
|
||
zhLocale := zh.New()
|
||
uni := ut.New(zhLocale, zhLocale)
|
||
|
||
// 获取中文翻译器
|
||
trans, _ := uni.GetTranslator("zh")
|
||
|
||
// 注册中文翻译
|
||
zh_translations.RegisterDefaultTranslations(validate, trans)
|
||
|
||
// 注册自定义验证器
|
||
registerCustomValidatorsZh(validate, trans)
|
||
|
||
return &RequestValidatorZh{
|
||
validator: validate,
|
||
translator: trans,
|
||
response: response,
|
||
}
|
||
}
|
||
|
||
// Validate 验证请求体
|
||
func (v *RequestValidatorZh) Validate(c *gin.Context, dto interface{}) error {
|
||
if err := v.validator.Struct(dto); err != nil {
|
||
validationErrors := v.formatValidationErrorsZh(err)
|
||
v.response.ValidationError(c, validationErrors)
|
||
return err
|
||
}
|
||
return nil
|
||
}
|
||
|
||
// ValidateQuery 验证查询参数
|
||
func (v *RequestValidatorZh) ValidateQuery(c *gin.Context, dto interface{}) error {
|
||
if err := c.ShouldBindQuery(dto); err != nil {
|
||
v.response.BadRequest(c, "查询参数格式错误", err.Error())
|
||
return err
|
||
}
|
||
|
||
if err := v.validator.Struct(dto); err != nil {
|
||
validationErrors := v.formatValidationErrorsZh(err)
|
||
v.response.ValidationError(c, validationErrors)
|
||
return err
|
||
}
|
||
return nil
|
||
}
|
||
|
||
// ValidateParam 验证路径参数
|
||
func (v *RequestValidatorZh) ValidateParam(c *gin.Context, dto interface{}) error {
|
||
if err := c.ShouldBindUri(dto); err != nil {
|
||
v.response.BadRequest(c, "路径参数格式错误", err.Error())
|
||
return err
|
||
}
|
||
|
||
if err := v.validator.Struct(dto); err != nil {
|
||
validationErrors := v.formatValidationErrorsZh(err)
|
||
v.response.ValidationError(c, validationErrors)
|
||
return err
|
||
}
|
||
return nil
|
||
}
|
||
|
||
// BindAndValidate 绑定并验证请求
|
||
func (v *RequestValidatorZh) BindAndValidate(c *gin.Context, dto interface{}) error {
|
||
// 绑定请求体
|
||
if err := c.ShouldBindJSON(dto); err != nil {
|
||
v.response.BadRequest(c, "请求体格式错误", err.Error())
|
||
return err
|
||
}
|
||
|
||
// 验证数据
|
||
return v.Validate(c, dto)
|
||
}
|
||
|
||
// formatValidationErrorsZh 格式化验证错误(中文翻译版)
|
||
func (v *RequestValidatorZh) formatValidationErrorsZh(err error) map[string][]string {
|
||
errors := make(map[string][]string)
|
||
|
||
if validationErrors, ok := err.(validator.ValidationErrors); ok {
|
||
for _, fieldError := range validationErrors {
|
||
fieldName := v.getFieldNameZh(fieldError)
|
||
|
||
// 首先尝试使用翻译器获取翻译后的错误消息
|
||
errorMessage := fieldError.Translate(v.translator)
|
||
|
||
// 如果翻译后的消息包含英文字段名,则替换为中文字段名
|
||
fieldDisplayName := v.getFieldDisplayName(fieldError.Field())
|
||
if fieldDisplayName != fieldError.Field() {
|
||
// 替换字段名为中文
|
||
errorMessage = strings.ReplaceAll(errorMessage, fieldError.Field(), fieldDisplayName)
|
||
}
|
||
|
||
if _, exists := errors[fieldName]; !exists {
|
||
errors[fieldName] = []string{}
|
||
}
|
||
errors[fieldName] = append(errors[fieldName], errorMessage)
|
||
}
|
||
}
|
||
|
||
return errors
|
||
}
|
||
|
||
// getFieldNameZh 获取字段名(JSON标签优先)
|
||
func (v *RequestValidatorZh) getFieldNameZh(fieldError validator.FieldError) string {
|
||
fieldName := fieldError.Field()
|
||
return v.toSnakeCase(fieldName)
|
||
}
|
||
|
||
// getFieldDisplayName 获取字段显示名称(中文)
|
||
func (v *RequestValidatorZh) getFieldDisplayName(field string) string {
|
||
fieldNames := map[string]string{
|
||
"phone": "手机号",
|
||
"password": "密码",
|
||
"confirm_password": "确认密码",
|
||
"old_password": "原密码",
|
||
"new_password": "新密码",
|
||
"confirm_new_password": "确认新密码",
|
||
"code": "验证码",
|
||
"username": "用户名",
|
||
"email": "邮箱",
|
||
"display_name": "显示名称",
|
||
"scene": "使用场景",
|
||
"Password": "密码",
|
||
"NewPassword": "新密码",
|
||
"ConfirmPassword": "确认密码",
|
||
}
|
||
|
||
if displayName, exists := fieldNames[field]; exists {
|
||
return displayName
|
||
}
|
||
return field
|
||
}
|
||
|
||
// toSnakeCase 转换为snake_case
|
||
func (v *RequestValidatorZh) toSnakeCase(str string) string {
|
||
var result strings.Builder
|
||
for i, r := range str {
|
||
if i > 0 && (r >= 'A' && r <= 'Z') {
|
||
result.WriteRune('_')
|
||
}
|
||
result.WriteRune(r)
|
||
}
|
||
return strings.ToLower(result.String())
|
||
}
|
||
|
||
// registerCustomValidatorsZh 注册自定义验证器和中文翻译
|
||
func registerCustomValidatorsZh(v *validator.Validate, trans ut.Translator) {
|
||
// 注册手机号验证器
|
||
v.RegisterValidation("phone", validatePhoneZh)
|
||
v.RegisterTranslation("phone", trans, func(ut ut.Translator) error {
|
||
return ut.Add("phone", "{0}必须是有效的手机号", true)
|
||
}, func(ut ut.Translator, fe validator.FieldError) string {
|
||
t, _ := ut.T("phone", fe.Field())
|
||
return t
|
||
})
|
||
|
||
// 注册用户名验证器
|
||
v.RegisterValidation("username", validateUsernameZh)
|
||
v.RegisterTranslation("username", trans, func(ut ut.Translator) error {
|
||
return ut.Add("username", "{0}格式不正确,只能包含字母、数字、下划线,且不能以数字开头", true)
|
||
}, func(ut ut.Translator, fe validator.FieldError) string {
|
||
t, _ := ut.T("username", fe.Field())
|
||
return t
|
||
})
|
||
|
||
// 注册密码强度验证器
|
||
v.RegisterValidation("strong_password", validateStrongPasswordZh)
|
||
v.RegisterTranslation("strong_password", trans, func(ut ut.Translator) error {
|
||
return ut.Add("strong_password", "{0}强度不足,必须包含大小写字母和数字,且不少于8位", true)
|
||
}, func(ut ut.Translator, fe validator.FieldError) string {
|
||
t, _ := ut.T("strong_password", fe.Field())
|
||
return t
|
||
})
|
||
|
||
// 自定义eqfield翻译
|
||
v.RegisterTranslation("eqfield", trans, func(ut ut.Translator) error {
|
||
return ut.Add("eqfield", "{0}必须等于{1}", true)
|
||
}, func(ut ut.Translator, fe validator.FieldError) string {
|
||
t, _ := ut.T("eqfield", fe.Field(), fe.Param())
|
||
return t
|
||
})
|
||
}
|
||
|
||
// validatePhoneZh 验证手机号
|
||
func validatePhoneZh(fl validator.FieldLevel) bool {
|
||
phone := fl.Field().String()
|
||
if phone == "" {
|
||
return true // 空值由required标签处理
|
||
}
|
||
|
||
// 中国手机号验证:11位,以1开头
|
||
if len(phone) != 11 {
|
||
return false
|
||
}
|
||
|
||
if !strings.HasPrefix(phone, "1") {
|
||
return false
|
||
}
|
||
|
||
// 检查是否全是数字
|
||
for _, r := range phone {
|
||
if r < '0' || r > '9' {
|
||
return false
|
||
}
|
||
}
|
||
|
||
return true
|
||
}
|
||
|
||
// validateUsernameZh 验证用户名
|
||
func validateUsernameZh(fl validator.FieldLevel) bool {
|
||
username := fl.Field().String()
|
||
if username == "" {
|
||
return true // 空值由required标签处理
|
||
}
|
||
|
||
// 用户名规则:3-30个字符,只能包含字母、数字、下划线,不能以数字开头
|
||
if len(username) < 3 || len(username) > 30 {
|
||
return false
|
||
}
|
||
|
||
// 不能以数字开头
|
||
if username[0] >= '0' && username[0] <= '9' {
|
||
return false
|
||
}
|
||
|
||
// 只能包含字母、数字、下划线
|
||
for _, r := range username {
|
||
if !((r >= 'a' && r <= 'z') || (r >= 'A' && r <= 'Z') || (r >= '0' && r <= '9') || r == '_') {
|
||
return false
|
||
}
|
||
}
|
||
|
||
return true
|
||
}
|
||
|
||
// validateStrongPasswordZh 验证密码强度
|
||
func validateStrongPasswordZh(fl validator.FieldLevel) bool {
|
||
password := fl.Field().String()
|
||
if password == "" {
|
||
return true // 空值由required标签处理
|
||
}
|
||
|
||
// 密码强度规则:至少8个字符,包含大小写字母、数字
|
||
if len(password) < 8 {
|
||
return false
|
||
}
|
||
|
||
hasUpper := false
|
||
hasLower := false
|
||
hasDigit := false
|
||
|
||
for _, r := range password {
|
||
switch {
|
||
case r >= 'A' && r <= 'Z':
|
||
hasUpper = true
|
||
case r >= 'a' && r <= 'z':
|
||
hasLower = true
|
||
case r >= '0' && r <= '9':
|
||
hasDigit = true
|
||
}
|
||
}
|
||
|
||
return hasUpper && hasLower && hasDigit
|
||
}
|
||
|
||
// ValidateStruct 直接验证结构体
|
||
func (v *RequestValidatorZh) ValidateStruct(dto interface{}) error {
|
||
return v.validator.Struct(dto)
|
||
}
|