510 lines
15 KiB
JavaScript
510 lines
15 KiB
JavaScript
let message = ""
|
||
/**
|
||
* ! !!!!!!
|
||
* 以下为凶残的cron表达式验证,胆小肾虚及心脏病者慎入!!!
|
||
* 不听劝告者后果自负T T
|
||
* !!!!!!!
|
||
* cron表达式为秒,分,时,日,月,周,年
|
||
* 判断正误方法:错误的话返回错误信息,正确的话返回true
|
||
* @param {string | true}cronExpression crom表达式
|
||
*/
|
||
export default function cronValidate (cronExpression) {
|
||
// 先将cron表达式进行分割
|
||
let cronParams = cronExpression.split(" ")
|
||
// 判断cron表达式是否具有该具有的属性长度,没有年份的长度为6,带年份的长度为7,其他情况都是错误的
|
||
if (cronParams.length < 6 || cronParams.length > 7) {
|
||
return "cron表达式需要输入6-7位参数,请重新输入"
|
||
} else {
|
||
// 日和周必须有一个为?,或者全为*
|
||
if ((cronParams[3] == "?" && cronParams[5] != "?") || (cronParams[5] == "?" && cronParams[3] != "?") || (cronParams[3] == "*" && cronParams[5] == "*")) {
|
||
// 检查第一位的秒是否正确
|
||
message = checkSecondsField(cronParams[0])
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
|
||
// 检查第二位的分是否正确
|
||
message = checkMinutesField(cronParams[1])
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
|
||
// 检查第三位的时是否正确
|
||
message = checkHoursField(cronParams[2])
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
|
||
// 检查第四位的日是否正确
|
||
message = checkDayOfMonthField(cronParams[3])
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
|
||
// 检查第五位的月是否正确
|
||
message = checkMonthsField(cronParams[4])
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
|
||
// 检查第6位的周是否正确
|
||
message = checkDayOfWeekField(cronParams[5])
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
|
||
// 检查第七位的年是否正确
|
||
if (cronParams.length > 6) {
|
||
message = checkYearField(cronParams[6])
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
}
|
||
|
||
return true
|
||
} else {
|
||
return "指定日时周必须设为不指定(?),指定周时日必须设为不指定(?)"
|
||
}
|
||
}
|
||
}
|
||
|
||
// 检查秒的函数方法
|
||
/**
|
||
*
|
||
* @param secondsField
|
||
*/
|
||
function checkSecondsField (secondsField) {
|
||
return checkField(secondsField, 0, 59, "秒")
|
||
}
|
||
|
||
// 检查分的函数方法
|
||
/**
|
||
*
|
||
* @param minutesField
|
||
*/
|
||
function checkMinutesField (minutesField) {
|
||
return checkField(minutesField, 0, 59, "分")
|
||
}
|
||
|
||
// 检查小时的函数方法
|
||
/**
|
||
*
|
||
* @param hoursField
|
||
*/
|
||
function checkHoursField (hoursField) {
|
||
return checkField(hoursField, 0, 23, "时")
|
||
}
|
||
|
||
// 检查日期的函数方法
|
||
/**
|
||
*
|
||
* @param dayOfMonthField
|
||
*/
|
||
function checkDayOfMonthField (dayOfMonthField) {
|
||
if (dayOfMonthField == "?") {
|
||
return true
|
||
}
|
||
if (dayOfMonthField.indexOf("L") >= 0) {
|
||
return checkFieldWithLetter(dayOfMonthField, "L", 1, 7, "日")
|
||
} else if (dayOfMonthField.indexOf("W") >= 0) {
|
||
return checkFieldWithLetter(dayOfMonthField, "W", 1, 31, "日")
|
||
} else if (dayOfMonthField.indexOf("C") >= 0) {
|
||
return checkFieldWithLetter(dayOfMonthField, "C", 1, 31, "日")
|
||
}
|
||
return checkField(dayOfMonthField, 1, 31, "日")
|
||
}
|
||
|
||
// 检查月份的函数方法
|
||
/**
|
||
*
|
||
* @param monthsField
|
||
*/
|
||
function checkMonthsField (monthsField) {
|
||
// 月份简写处理
|
||
if (monthsField != "*") {
|
||
monthsField = monthsField.replace("JAN", "1")
|
||
monthsField = monthsField.replace("FEB", "2")
|
||
monthsField = monthsField.replace("MAR", "3")
|
||
monthsField = monthsField.replace("APR", "4")
|
||
monthsField = monthsField.replace("MAY", "5")
|
||
monthsField = monthsField.replace("JUN", "6")
|
||
monthsField = monthsField.replace("JUL", "7")
|
||
monthsField = monthsField.replace("AUG", "8")
|
||
monthsField = monthsField.replace("SEP", "9")
|
||
monthsField = monthsField.replace("OCT", "10")
|
||
monthsField = monthsField.replace("NOV", "11")
|
||
monthsField = monthsField.replace("DEC", "12")
|
||
return checkField(monthsField, 1, 12, "月份")
|
||
} else {
|
||
return true
|
||
}
|
||
}
|
||
|
||
// 星期验证
|
||
/**
|
||
*
|
||
* @param dayOfWeekField
|
||
*/
|
||
function checkDayOfWeekField (dayOfWeekField) {
|
||
dayOfWeekField = dayOfWeekField.replace("SUN", "1")
|
||
dayOfWeekField = dayOfWeekField.replace("MON", "2")
|
||
dayOfWeekField = dayOfWeekField.replace("TUE", "3")
|
||
dayOfWeekField = dayOfWeekField.replace("WED", "4")
|
||
dayOfWeekField = dayOfWeekField.replace("THU", "5")
|
||
dayOfWeekField = dayOfWeekField.replace("FRI", "6")
|
||
dayOfWeekField = dayOfWeekField.replace("SAT", "7")
|
||
if (dayOfWeekField == "?") {
|
||
return true
|
||
}
|
||
if (dayOfWeekField.indexOf("L") >= 0) {
|
||
return checkFieldWithLetterWeek(dayOfWeekField, "L", 1, 7, "星期")
|
||
} else if (dayOfWeekField.indexOf("C") >= 0) {
|
||
return checkFieldWithLetterWeek(dayOfWeekField, "C", 1, 7, "星期")
|
||
} else if (dayOfWeekField.indexOf("#") >= 0) {
|
||
return checkFieldWithLetterWeek(dayOfWeekField, "#", 1, 7, "星期")
|
||
} else {
|
||
return checkField(dayOfWeekField, 1, 7, "星期")
|
||
}
|
||
}
|
||
|
||
// 检查年份的函数方法
|
||
/**
|
||
*
|
||
* @param yearField
|
||
*/
|
||
function checkYearField (yearField) {
|
||
return checkField(yearField, 1970, 2099, "年的")
|
||
}
|
||
|
||
// 通用的检查值的大小范围的方法( - , / *)
|
||
/**
|
||
*
|
||
* @param value
|
||
* @param minimal
|
||
* @param maximal
|
||
* @param attribute
|
||
*/
|
||
function checkField (value, minimal, maximal, attribute) {
|
||
// 校验值中是否有“-”,如果有“-”的话,下标会>0
|
||
if (value.indexOf("-") > -1) {
|
||
return checkRangeAndCycle(value, minimal, maximal, attribute)
|
||
} else if (value.indexOf(",") > -1) {
|
||
// 校验值中是否有“,”,如果有“,”的话,下标会>0
|
||
return checkListField(value, minimal, maximal, attribute)
|
||
} else if (value.indexOf("/") > -1) {
|
||
// 校验值中是否有“/”,如果有“/”的话,下标会>0
|
||
return checkIncrementField(value, minimal, maximal, attribute)
|
||
} else if (value == "*") {
|
||
// 校验值是否为“*”
|
||
return true
|
||
} else {
|
||
// 校验单独的数字,英文字母,以及各种神奇的符号等...
|
||
return checkIntValue(value, minimal, maximal, true, attribute)
|
||
}
|
||
}
|
||
|
||
// 检测是否是整数以及是否在范围内,参数:检测的值,下限,上限,是否检查端点,检查的属性
|
||
/**
|
||
*
|
||
* @param value
|
||
* @param minimal
|
||
* @param maximal
|
||
* @param checkExtremity
|
||
* @param attribute
|
||
*/
|
||
function checkIntValue (value, minimal, maximal, checkExtremity, attribute) {
|
||
try {
|
||
// 用10进制犯法来进行整数转换
|
||
let val = parseInt(value, 10)
|
||
if (value == val) {
|
||
if (checkExtremity) {
|
||
if (val < minimal || val > maximal) {
|
||
return (attribute + "的参数取值范围必须在" + minimal + "-" + maximal + "之间")
|
||
}
|
||
return true
|
||
}
|
||
return true
|
||
}
|
||
return (attribute + "的参数存在非法字符,必须为整数或允许的大写英文")
|
||
} catch (e) {
|
||
return (attribute + "的参数有非法字符,必须是整数~")
|
||
}
|
||
}
|
||
// 检验枚举类型的参数是否正确
|
||
/**
|
||
*
|
||
* @param value
|
||
* @param minimal
|
||
* @param maximal
|
||
* @param attribute
|
||
*/
|
||
function checkListField (value, minimal, maximal, attribute) {
|
||
let st = value.split(",")
|
||
let values = new Array(st.length)
|
||
// 计算枚举的数字在数组中中出现的次数,出现一次为没有重复的。
|
||
let count = 0
|
||
for (let j = 0; j < st.length; j++) {
|
||
values[j] = st[j]
|
||
}
|
||
// 判断枚举类型的值是否重复
|
||
for (let i = 0; i < values.length; i++) {
|
||
// 判断枚举的值是否在范围内
|
||
message = checkIntValue(values[i], minimal, maximal, true, attribute)
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
count = 0
|
||
for (let j = 0; j < values.length; j++) {
|
||
if (values[i] == values[j]) {
|
||
count++
|
||
}
|
||
if (count > 1) {
|
||
return (attribute + "中的参数重复")
|
||
}
|
||
}
|
||
}
|
||
let previousValue = -1
|
||
// 判断枚举的值是否排序正确
|
||
for (let i = 0; i < values.length; i++) {
|
||
let currentValue = values[i]
|
||
try {
|
||
let val = parseInt(currentValue, 10)
|
||
if (val < previousValue) {
|
||
return (attribute + "的参数应该从小到大")
|
||
} else {
|
||
previousValue = val
|
||
}
|
||
} catch (e) {
|
||
// 前面验证过了,这边的代码不可能跑到
|
||
return ("这段提示用不到")
|
||
}
|
||
}
|
||
return true
|
||
}
|
||
|
||
// 检验循环
|
||
/**
|
||
*
|
||
* @param value
|
||
* @param minimal
|
||
* @param maximal
|
||
* @param attribute
|
||
*/
|
||
function checkIncrementField (value, minimal, maximal, attribute) {
|
||
if (value.split("/").length > 2) {
|
||
return (attribute + "中的参数只能有一个'/'")
|
||
}
|
||
let start = value.substring(0, value.indexOf("/"))
|
||
let increment = value.substring(value.indexOf("/") + 1)
|
||
if (start != "*") {
|
||
// 检验前值是否正确
|
||
message = checkIntValue(start, minimal, maximal, true, attribute)
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
// 检验后值是否正确
|
||
message = checkIntValue(increment, minimal, maximal, true, attribute)
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
return true
|
||
} else {
|
||
// 检验后值是否正确
|
||
return checkIntValue(increment, minimal, maximal, false, attribute)
|
||
}
|
||
}
|
||
|
||
// 检验范围
|
||
/**
|
||
*
|
||
* @param params
|
||
* @param minimal
|
||
* @param maximal
|
||
* @param attribute
|
||
*/
|
||
function checkRangeAndCycle (params, minimal, maximal, attribute) {
|
||
// 校验“-”符号是否只有一个
|
||
if (params.split("-").length > 2) {
|
||
return (attribute + "中的参数只能有一个'-'")
|
||
}
|
||
let value = null
|
||
let cycle = null
|
||
// 检验范围内是否有嵌套周期
|
||
if (params.indexOf("/") > -1) {
|
||
// 校验“/”符号是否只有一个
|
||
if (params.split("/").length > 2) {
|
||
return (attribute + "中的参数只能有一个'/'")
|
||
}
|
||
value = params.split("/")[0]
|
||
cycle = params.split("/")[1]
|
||
// 判断循环的参数是否正确
|
||
message = checkIntValue(cycle, minimal, maximal, true, attribute)
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
} else {
|
||
value = params
|
||
}
|
||
let startValue = value.substring(0, value.indexOf("-"))
|
||
let endValue = value.substring(value.indexOf("-") + 1)
|
||
// 判断参数范围的第一个值是否正确
|
||
message = checkIntValue(startValue, minimal, maximal, true, attribute)
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
// 判断参数范围的第二个值是否正确
|
||
message = checkIntValue(endValue, minimal, maximal, true, attribute)
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
// 判断参数的范围前值是否小于后值
|
||
try {
|
||
let startVal = parseInt(startValue, 10)
|
||
let endVal = parseInt(endValue, 10)
|
||
if (endVal < startVal) {
|
||
return (attribute + "的取值范围错误,前值必须小于后值")
|
||
}
|
||
if ((endVal - startVal) < parseInt(cycle, 10)) {
|
||
return (attribute + "的取值范围内的循环无意义")
|
||
}
|
||
return true
|
||
} catch (e) {
|
||
// 用不到这行代码的
|
||
return (attribute + "的参数有非法字符,必须是整数")
|
||
}
|
||
}
|
||
|
||
// 检查日中的特殊字符
|
||
/**
|
||
*
|
||
* @param value
|
||
* @param letter
|
||
* @param minimalBefore
|
||
* @param maximalBefore
|
||
* @param attribute
|
||
*/
|
||
function checkFieldWithLetter (value, letter, minimalBefore, maximalBefore, attribute) {
|
||
// 判断是否只有一个字母
|
||
for (let i = 0; i < value.length; i++) {
|
||
let count = 0
|
||
if (value.charAt(i) == letter) {
|
||
count++
|
||
}
|
||
if (count > 1) {
|
||
return (attribute + "的值的" + letter + "字母只能有一个")
|
||
}
|
||
}
|
||
// 校验L
|
||
if (letter == "L") {
|
||
if (value == "LW") {
|
||
return true
|
||
}
|
||
if (value == "L") {
|
||
return true
|
||
}
|
||
if (value.endsWith("LW") && value.length > 2) {
|
||
return (attribute + "中的参数,最后的LW前面不能有任何字母参数")
|
||
}
|
||
if (!value.endsWith("L")) {
|
||
return (attribute + "中的参数,L字母后面不能有W以外的字符、数字等")
|
||
} else {
|
||
let num = value.substring(0, value.indexOf(letter))
|
||
return checkIntValue(num, minimalBefore, maximalBefore, true, attribute)
|
||
}
|
||
}
|
||
|
||
// 校验W
|
||
if (letter == "W") {
|
||
if (!value.endsWith("W")) {
|
||
return (attribute + "中的参数的W必须作为结尾")
|
||
} else {
|
||
if (value == "W") {
|
||
return (attribute + "中的参数的W前面必须有数字")
|
||
}
|
||
let num = value.substring(0, value.indexOf(letter))
|
||
return checkIntValue(num, minimalBefore, maximalBefore, true, attribute)
|
||
}
|
||
}
|
||
|
||
if (letter == "C") {
|
||
if (!value.endsWith("C")) {
|
||
return (attribute + "中的参数的C必须作为结尾")
|
||
} else {
|
||
if (value == "C") {
|
||
return (attribute + "中的参数的C前面必须有数字")
|
||
}
|
||
let num = value.substring(0, value.indexOf(letter))
|
||
return checkIntValue(num, minimalBefore, maximalBefore, true, attribute)
|
||
}
|
||
}
|
||
}
|
||
|
||
// 检查星期中的特殊字符
|
||
/**
|
||
*
|
||
* @param value
|
||
* @param letter
|
||
* @param minimalBefore
|
||
* @param maximalBefore
|
||
* @param attribute
|
||
*/
|
||
function checkFieldWithLetterWeek (value, letter, minimalBefore, maximalBefore, attribute) {
|
||
// 判断是否只有一个字母
|
||
for (let i = 0; i < value.length; i++) {
|
||
let count = 0
|
||
if (value.charAt(i) == letter) {
|
||
count++
|
||
}
|
||
if (count > 1) {
|
||
return (attribute + "的值的" + letter + "字母只能有一个")
|
||
}
|
||
}
|
||
// 校验L
|
||
if (letter == "L") {
|
||
if (value == "L") {
|
||
return true
|
||
}
|
||
if (!value.endsWith("L")) {
|
||
return (attribute + "中的参数,L字母必须是最后一位")
|
||
} else {
|
||
let num = value.substring(0, value.indexOf(letter))
|
||
return checkIntValue(num, minimalBefore, maximalBefore, true, attribute)
|
||
}
|
||
}
|
||
|
||
if (letter == "C") {
|
||
if (!value.endsWith("C")) {
|
||
return (attribute + "中的参数的C必须作为结尾")
|
||
} else {
|
||
if (value == "C") {
|
||
return (attribute + "中的参数的C前面必须有数字")
|
||
}
|
||
let num = value.substring(0, value.indexOf(letter))
|
||
return checkIntValue(num, minimalBefore, maximalBefore, true, attribute)
|
||
}
|
||
}
|
||
|
||
if (letter == "#") {
|
||
if (value == "#") {
|
||
return (attribute + "中的#前后必须有整数")
|
||
}
|
||
if (value.charAt(0) == letter) {
|
||
return (attribute + "中的#前面必须有整数")
|
||
}
|
||
if (value.endsWith("#")) {
|
||
return (attribute + "中的#后面必须有整数")
|
||
}
|
||
let num1 = value.substring(0, value.indexOf(letter))
|
||
let num2 = value.substring(value.indexOf(letter) + 1, value.length)
|
||
message = checkIntValue(num1, 1, 4, true, (attribute + "的#前面"))
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
message = checkIntValue(num2, minimalBefore, maximalBefore, true, (attribute + "的#后面"))
|
||
if (message != true) {
|
||
return message
|
||
}
|
||
return true
|
||
}
|
||
}
|