package encrypt import ( "golang.org/x/crypto/bcrypt" "regexp" "strings" ) // ValidatePhoneNumber 校验手机号格式是否为中国大陆合法手机号。 // 去除空格后匹配以 1 开头的 11 位数字,第二位为 3~9。 // // 参数: // - phone: 需要验证的手机号字符串。 // // 返回: // - true 表示格式合法,false 表示格式不合法。 func ValidatePhoneNumber(phone string) bool { phone = strings.ReplaceAll(phone, " ", "") pattern := `^1[3-9]\d{9}$` matched, err := regexp.MatchString(pattern, phone) if err != nil { return false } return matched } // EncryptPhoneNumber 加密手机号字符串。 // 使用 bcrypt 进行加密,适合需要不可逆存储的场景。 // ⚠️ 注意:加密结果无法解密,仅能做比对用途。 // // 参数: // - phone: 待加密的手机号(需合法)。 // // 返回: // - 加密后的哈希字符串;如果手机号无效或加密失败,返回空字符串。 func EncryptPhoneNumber(phone string) string { if !ValidatePhoneNumber(phone) { return "" } hashedPhone, err := bcrypt.GenerateFromPassword([]byte(phone), bcrypt.DefaultCost) if err != nil { return "" } return string(hashedPhone) } // MaskPhoneNumber 对手机号进行脱敏处理。 // 中间四位将被替换为星号,以保护用户隐私。 // // 参数: // - phone: 原始手机号字符串(需合法)。 // // 返回: // - 脱敏后的手机号,如 138****1234;如果手机号无效,则返回原始输入。 func MaskPhoneNumber(phone string) string { if !ValidatePhoneNumber(phone) { return phone } return phone[:3] + "****" + phone[7:] } // ComparePhoneNumber 比较明文手机号与加密后的哈希是否匹配。 // 用于验证用户输入的手机号是否与加密存储的值一致。 // // 参数: // - hashedPhone: 使用 bcrypt 加密后的手机号哈希。 // - phone: 用户输入的明文手机号(需合法)。 // // 返回: // - true 表示匹配成功;false 表示不匹配或格式错误。 func ComparePhoneNumber(hashedPhone, phone string) bool { if !ValidatePhoneNumber(phone) { return false } err := bcrypt.CompareHashAndPassword([]byte(hashedPhone), []byte(phone)) return err == nil }