212 lines
5.8 KiB
JavaScript
212 lines
5.8 KiB
JavaScript
import { CONFIG } from '@/utils/configure'
|
||
import passwordConfig from '@/utils/passwordConfig'
|
||
|
||
/**
|
||
* @param {string} path
|
||
* @returns {Boolean}
|
||
*/
|
||
export function isExternal(path) {
|
||
return /^(https?:|mailto:|tel:)/.test(path)
|
||
}
|
||
|
||
/**
|
||
* @param {string} str
|
||
* @returns {Boolean}
|
||
*/
|
||
export function validUsername(str) {
|
||
const valid_map = ['admin', 'editor']
|
||
return valid_map.indexOf(str.trim()) >= 0
|
||
}
|
||
|
||
/**
|
||
* @param {string} url
|
||
* @returns {Boolean}
|
||
*/
|
||
export function validURL(url) {
|
||
const reg = /^(https?|ftp):\/\/([a-zA-Z0-9.-]+(:[a-zA-Z0-9.&%$-]+)*@)*((25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9][0-9]?)(\.(25[0-5]|2[0-4][0-9]|1[0-9]{2}|[1-9]?[0-9])){3}|([a-zA-Z0-9-]+\.)*[a-zA-Z0-9-]+\.(com|edu|gov|int|mil|net|org|biz|arpa|info|name|pro|aero|coop|museum|[a-zA-Z]{2}))(:[0-9]+)*(\/($|[a-zA-Z0-9.,?'\\+&%$#=~_-]+))*$/
|
||
return reg.test(url)
|
||
}
|
||
|
||
/**
|
||
* @param {string} str
|
||
* @returns {Boolean}
|
||
*/
|
||
export function validLowerCase(str) {
|
||
const reg = /^[a-z]+$/
|
||
return reg.test(str)
|
||
}
|
||
|
||
/**
|
||
* @param {string} str
|
||
* @returns {Boolean}
|
||
*/
|
||
export function validUpperCase(str) {
|
||
const reg = /^[A-Z]+$/
|
||
return reg.test(str)
|
||
}
|
||
|
||
/**
|
||
* @param {string} str
|
||
* @returns {Boolean}
|
||
*/
|
||
export function validAlphabets(str) {
|
||
const reg = /^[A-Za-z]+$/
|
||
return reg.test(str)
|
||
}
|
||
|
||
/**
|
||
* @param {string} email
|
||
* @returns {Boolean}
|
||
*/
|
||
export function validEmail(email) {
|
||
const reg = /^(([^<>()\[\]\\.,;:\s@"]+(\.[^<>()\[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/
|
||
return reg.test(email)
|
||
}
|
||
|
||
/**
|
||
* @param {string} str
|
||
* @returns {Boolean}
|
||
*/
|
||
export function isString(str) {
|
||
return typeof str === 'string' || str instanceof String
|
||
}
|
||
|
||
/**
|
||
* @param {Array} arg
|
||
* @returns {Boolean}
|
||
*/
|
||
export function isArray(arg) {
|
||
if (typeof Array.isArray === 'undefined') {
|
||
return Object.prototype.toString.call(arg) === '[object Array]'
|
||
}
|
||
return Array.isArray(arg)
|
||
}
|
||
|
||
/**
|
||
* 密码的正则表达式 最少8个字符,最多20个字符,至少一个字母,一个数字和一个特殊字符:
|
||
* @param {string} password
|
||
* @returns {Boolean}
|
||
*/
|
||
export function validPwd(value) {
|
||
const reg = /^(?=.*[A-Za-z])(?=.*\d)(?=.*[@$!%*#?&])[A-Za-z\d@$!%*#?&]{8,20}$/
|
||
return reg.test(value)
|
||
}
|
||
|
||
/**
|
||
* 弱:长度至少为8个字符。
|
||
* 一般:长度至少为8个字符,并包含至少一种字符类型。
|
||
* 强:长度至少为8个字符,并包含至少两种字符类型。
|
||
* 非常强:长度至少为8个字符,并包含所有四种字符类型。
|
||
* @param rule
|
||
* @param value
|
||
* @param callback
|
||
* @returns {*}
|
||
*/
|
||
|
||
export function validatePassword(rule, value, callback) {
|
||
if (!value) {
|
||
return callback(new Error('请输入密码'))
|
||
}
|
||
|
||
const lengthRegex = /^.{8,20}$/
|
||
const uppercaseRegex = /[A-Z]/
|
||
const lowercaseRegex = /[a-z]/
|
||
const digitRegex = /\d/
|
||
const specialCharRegex = /[!@#$%^&*(),.?":{}|<>]/
|
||
|
||
if (!lengthRegex.test(value)) {
|
||
return callback(new Error('密码长度必须为8到20位'))
|
||
}
|
||
|
||
const checks = [
|
||
{ regex: uppercaseRegex, message: '必须包含至少一个大写字母' },
|
||
{ regex: lowercaseRegex, message: '必须包含至少一个小写字母' },
|
||
{ regex: digitRegex, message: '必须包含至少一个数字' },
|
||
{ regex: specialCharRegex, message: '必须包含至少一个特殊字符' }
|
||
]
|
||
|
||
let passedChecks = checks.filter(check => check.regex.test(value)).length
|
||
|
||
let requiredChecks
|
||
switch (CONFIG.STRENGTH) {
|
||
case 'weak':
|
||
requiredChecks = 1
|
||
break
|
||
case 'medium':
|
||
requiredChecks = 2
|
||
break
|
||
case 'strong':
|
||
requiredChecks = 3
|
||
break
|
||
case 'very-strong':
|
||
requiredChecks = 4
|
||
break
|
||
default:
|
||
return callback(new Error('请选择有效的密码强度'))
|
||
}
|
||
|
||
if (passedChecks < requiredChecks) {
|
||
return callback(new Error(`密码至少包含 ${requiredChecks} 类字符(大写字母,小写字母,数字,特殊字符)`))
|
||
}
|
||
callback()
|
||
}
|
||
|
||
export function validateNewPassword(rule, value, callback) {
|
||
// 使用配置文件中的策略进行验证
|
||
|
||
// 1. 检查密码长度
|
||
if (value.length < passwordConfig.minLength || value.length > passwordConfig.maxLength) {
|
||
callback(new Error('密码长度应为' + passwordConfig.minLength + '至' + passwordConfig.maxLength + '位!'))
|
||
return
|
||
}
|
||
|
||
// 2. 检查密码复杂度
|
||
const hasUpperCase = /[A-Z]/.test(value)
|
||
const hasLowerCase = /[a-z]/.test(value)
|
||
const hasDigit = /\d/.test(value)
|
||
const hasSpecialChar = /[!@#$%^&*(),.?":{}|<>]/.test(value)
|
||
|
||
if (passwordConfig.requireUpperCase && !hasUpperCase) {
|
||
callback(new Error('密码必须包含大写字母!'))
|
||
return
|
||
}
|
||
if (passwordConfig.requireLowerCase && !hasLowerCase) {
|
||
callback(new Error('密码必须包含小写字母!'))
|
||
return
|
||
}
|
||
if (passwordConfig.requireDigit && !hasDigit) {
|
||
callback(new Error('密码必须包含数字!'))
|
||
return
|
||
}
|
||
if (passwordConfig.requireSpecialChar && !hasSpecialChar) {
|
||
callback(new Error('密码必须包含特殊字符!'))
|
||
return
|
||
}
|
||
// 3. 检查是否包含弱密码
|
||
for (const weakPwd of passwordConfig.weakPasswords) {
|
||
if (value.includes(weakPwd)) {
|
||
callback(new Error(`密码包含常见的弱密码片段: ${weakPwd}`))
|
||
return
|
||
}
|
||
}
|
||
// 4. 检查是否包含超过规定数量的连续字符
|
||
if (passwordConfig.restrictConsecutiveChars && containsConsecutiveCharacters(value, passwordConfig.maxConsecutiveChars)) {
|
||
callback(new Error(`密码不能包含超过${passwordConfig.maxConsecutiveChars}位连续字符!`))
|
||
return
|
||
}
|
||
callback() // 验证成功
|
||
}
|
||
|
||
function containsConsecutiveCharacters(password, maxConsecutive) {
|
||
let count = 1
|
||
for (let i = 1; i < password.length; i++) {
|
||
if (password[i] === password[i - 1]) {
|
||
count++
|
||
if (count > maxConsecutive) return true
|
||
} else {
|
||
count = 1
|
||
}
|
||
}
|
||
return false
|
||
}
|