You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1343 lines
38 KiB
1343 lines
38 KiB
function _extends() {
|
|
_extends = Object.assign || function (target) {
|
|
for (let i = 1; i < arguments.length; i++) {
|
|
const source = arguments[i]
|
|
|
|
for (const key in source) {
|
|
if (Object.prototype.hasOwnProperty.call(source, key)) {
|
|
target[key] = source[key]
|
|
}
|
|
}
|
|
}
|
|
|
|
return target
|
|
}
|
|
|
|
return _extends.apply(this, arguments)
|
|
}
|
|
|
|
/* eslint no-console:0 */
|
|
const formatRegExp = /%[sdj%]/g
|
|
let warning = function warning() {} // don't print warning message when in production env or node runtime
|
|
|
|
if (typeof process !== 'undefined' && process.env && process.env.NODE_ENV !== 'production' && typeof window
|
|
!== 'undefined' && typeof document !== 'undefined') {
|
|
warning = function warning(type, errors) {
|
|
if (typeof console !== 'undefined' && console.warn) {
|
|
if (errors.every((e) => typeof e === 'string')) {
|
|
console.warn(type, errors)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function convertFieldsError(errors) {
|
|
if (!errors || !errors.length) return null
|
|
const fields = {}
|
|
errors.forEach((error) => {
|
|
const { field } = error
|
|
fields[field] = fields[field] || []
|
|
fields[field].push(error)
|
|
})
|
|
return fields
|
|
}
|
|
|
|
function format() {
|
|
for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {
|
|
args[_key] = arguments[_key]
|
|
}
|
|
|
|
let i = 1
|
|
const f = args[0]
|
|
const len = args.length
|
|
|
|
if (typeof f === 'function') {
|
|
return f.apply(null, args.slice(1))
|
|
}
|
|
|
|
if (typeof f === 'string') {
|
|
let str = String(f).replace(formatRegExp, (x) => {
|
|
if (x === '%%') {
|
|
return '%'
|
|
}
|
|
|
|
if (i >= len) {
|
|
return x
|
|
}
|
|
|
|
switch (x) {
|
|
case '%s':
|
|
return String(args[i++])
|
|
|
|
case '%d':
|
|
return Number(args[i++])
|
|
|
|
case '%j':
|
|
try {
|
|
return JSON.stringify(args[i++])
|
|
} catch (_) {
|
|
return '[Circular]'
|
|
}
|
|
|
|
break
|
|
|
|
default:
|
|
return x
|
|
}
|
|
})
|
|
|
|
for (let arg = args[i]; i < len; arg = args[++i]) {
|
|
str += ` ${arg}`
|
|
}
|
|
|
|
return str
|
|
}
|
|
|
|
return f
|
|
}
|
|
|
|
function isNativeStringType(type) {
|
|
return type === 'string' || type === 'url' || type === 'hex' || type === 'email' || type === 'pattern'
|
|
}
|
|
|
|
function isEmptyValue(value, type) {
|
|
if (value === undefined || value === null) {
|
|
return true
|
|
}
|
|
|
|
if (type === 'array' && Array.isArray(value) && !value.length) {
|
|
return true
|
|
}
|
|
|
|
if (isNativeStringType(type) && typeof value === 'string' && !value) {
|
|
return true
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
function asyncParallelArray(arr, func, callback) {
|
|
const results = []
|
|
let total = 0
|
|
const arrLength = arr.length
|
|
|
|
function count(errors) {
|
|
results.push.apply(results, errors)
|
|
total++
|
|
|
|
if (total === arrLength) {
|
|
callback(results)
|
|
}
|
|
}
|
|
|
|
arr.forEach((a) => {
|
|
func(a, count)
|
|
})
|
|
}
|
|
|
|
function asyncSerialArray(arr, func, callback) {
|
|
let index = 0
|
|
const arrLength = arr.length
|
|
|
|
function next(errors) {
|
|
if (errors && errors.length) {
|
|
callback(errors)
|
|
return
|
|
}
|
|
|
|
const original = index
|
|
index += 1
|
|
|
|
if (original < arrLength) {
|
|
func(arr[original], next)
|
|
} else {
|
|
callback([])
|
|
}
|
|
}
|
|
|
|
next([])
|
|
}
|
|
|
|
function flattenObjArr(objArr) {
|
|
const ret = []
|
|
Object.keys(objArr).forEach((k) => {
|
|
ret.push.apply(ret, objArr[k])
|
|
})
|
|
return ret
|
|
}
|
|
|
|
function asyncMap(objArr, option, func, callback) {
|
|
if (option.first) {
|
|
const _pending = new Promise((resolve, reject) => {
|
|
const next = function next(errors) {
|
|
callback(errors)
|
|
return errors.length ? reject({
|
|
errors,
|
|
fields: convertFieldsError(errors)
|
|
}) : resolve()
|
|
}
|
|
|
|
const flattenArr = flattenObjArr(objArr)
|
|
asyncSerialArray(flattenArr, func, next)
|
|
})
|
|
|
|
_pending.catch((e) => e)
|
|
|
|
return _pending
|
|
}
|
|
|
|
let firstFields = option.firstFields || []
|
|
|
|
if (firstFields === true) {
|
|
firstFields = Object.keys(objArr)
|
|
}
|
|
|
|
const objArrKeys = Object.keys(objArr)
|
|
const objArrLength = objArrKeys.length
|
|
let total = 0
|
|
const results = []
|
|
const pending = new Promise((resolve, reject) => {
|
|
const next = function next(errors) {
|
|
results.push.apply(results, errors)
|
|
total++
|
|
|
|
if (total === objArrLength) {
|
|
callback(results)
|
|
return results.length ? reject({
|
|
errors: results,
|
|
fields: convertFieldsError(results)
|
|
}) : resolve()
|
|
}
|
|
}
|
|
|
|
if (!objArrKeys.length) {
|
|
callback(results)
|
|
resolve()
|
|
}
|
|
|
|
objArrKeys.forEach((key) => {
|
|
const arr = objArr[key]
|
|
|
|
if (firstFields.indexOf(key) !== -1) {
|
|
asyncSerialArray(arr, func, next)
|
|
} else {
|
|
asyncParallelArray(arr, func, next)
|
|
}
|
|
})
|
|
})
|
|
pending.catch((e) => e)
|
|
return pending
|
|
}
|
|
|
|
function complementError(rule) {
|
|
return function (oe) {
|
|
if (oe && oe.message) {
|
|
oe.field = oe.field || rule.fullField
|
|
return oe
|
|
}
|
|
|
|
return {
|
|
message: typeof oe === 'function' ? oe() : oe,
|
|
field: oe.field || rule.fullField
|
|
}
|
|
}
|
|
}
|
|
|
|
function deepMerge(target, source) {
|
|
if (source) {
|
|
for (const s in source) {
|
|
if (source.hasOwnProperty(s)) {
|
|
const value = source[s]
|
|
|
|
if (typeof value === 'object' && typeof target[s] === 'object') {
|
|
target[s] = { ...target[s], ...value }
|
|
} else {
|
|
target[s] = value
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return target
|
|
}
|
|
|
|
/**
|
|
* Rule for validating required fields.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param source The source object being validated.
|
|
* @param errors An array of errors that this rule may add
|
|
* validation errors to.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function required(rule, value, source, errors, options, type) {
|
|
if (rule.required && (!source.hasOwnProperty(rule.field) || isEmptyValue(value, type || rule.type))) {
|
|
errors.push(format(options.messages.required, rule.fullField))
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Rule for validating whitespace.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param source The source object being validated.
|
|
* @param errors An array of errors that this rule may add
|
|
* validation errors to.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function whitespace(rule, value, source, errors, options) {
|
|
if (/^\s+$/.test(value) || value === '') {
|
|
errors.push(format(options.messages.whitespace, rule.fullField))
|
|
}
|
|
}
|
|
|
|
/* eslint max-len:0 */
|
|
|
|
const pattern = {
|
|
// http://emailregex.com/
|
|
email: /^(([^<>()\[\]\\.,;:\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,}))$/,
|
|
url: new RegExp(
|
|
'^(?!mailto:)(?:(?:http|https|ftp)://|//)(?:\\S+(?::\\S*)?@)?(?:(?:(?:[1-9]\\d?|1\\d\\d|2[01]\\d|22[0-3])(?:\\.(?:1?\\d{1,2}|2[0-4]\\d|25[0-5])){2}(?:\\.(?:[0-9]\\d?|1\\d\\d|2[0-4]\\d|25[0-4]))|(?:(?:[a-z\\u00a1-\\uffff0-9]+-*)*[a-z\\u00a1-\\uffff0-9]+)(?:\\.(?:[a-z\\u00a1-\\uffff0-9]+-*)*[a-z\\u00a1-\\uffff0-9]+)*(?:\\.(?:[a-z\\u00a1-\\uffff]{2,})))|localhost)(?::\\d{2,5})?(?:(/|\\?|#)[^\\s]*)?$',
|
|
'i'
|
|
),
|
|
hex: /^#?([a-f0-9]{6}|[a-f0-9]{3})$/i
|
|
}
|
|
var types = {
|
|
integer: function integer(value) {
|
|
return /^(-)?\d+$/.test(value);
|
|
},
|
|
float: function float(value) {
|
|
return /^(-)?\d+(\.\d+)?$/.test(value);
|
|
},
|
|
array: function array(value) {
|
|
return Array.isArray(value)
|
|
},
|
|
regexp: function regexp(value) {
|
|
if (value instanceof RegExp) {
|
|
return true
|
|
}
|
|
|
|
try {
|
|
return !!new RegExp(value)
|
|
} catch (e) {
|
|
return false
|
|
}
|
|
},
|
|
date: function date(value) {
|
|
return typeof value.getTime === 'function' && typeof value.getMonth === 'function' && typeof value.getYear
|
|
=== 'function'
|
|
},
|
|
number: function number(value) {
|
|
if (isNaN(value)) {
|
|
return false
|
|
}
|
|
|
|
// 修改源码,将字符串数值先转为数值
|
|
return typeof +value === 'number'
|
|
},
|
|
object: function object(value) {
|
|
return typeof value === 'object' && !types.array(value)
|
|
},
|
|
method: function method(value) {
|
|
return typeof value === 'function'
|
|
},
|
|
email: function email(value) {
|
|
return typeof value === 'string' && !!value.match(pattern.email) && value.length < 255
|
|
},
|
|
url: function url(value) {
|
|
return typeof value === 'string' && !!value.match(pattern.url)
|
|
},
|
|
hex: function hex(value) {
|
|
return typeof value === 'string' && !!value.match(pattern.hex)
|
|
}
|
|
}
|
|
/**
|
|
* Rule for validating the type of a value.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param source The source object being validated.
|
|
* @param errors An array of errors that this rule may add
|
|
* validation errors to.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function type(rule, value, source, errors, options) {
|
|
if (rule.required && value === undefined) {
|
|
required(rule, value, source, errors, options)
|
|
return
|
|
}
|
|
|
|
const custom = ['integer', 'float', 'array', 'regexp', 'object', 'method', 'email', 'number', 'date', 'url', 'hex']
|
|
const ruleType = rule.type
|
|
|
|
if (custom.indexOf(ruleType) > -1) {
|
|
if (!types[ruleType](value)) {
|
|
errors.push(format(options.messages.types[ruleType], rule.fullField, rule.type))
|
|
} // straight typeof check
|
|
} else if (ruleType && typeof value !== rule.type) {
|
|
errors.push(format(options.messages.types[ruleType], rule.fullField, rule.type))
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Rule for validating minimum and maximum allowed values.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param source The source object being validated.
|
|
* @param errors An array of errors that this rule may add
|
|
* validation errors to.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function range(rule, value, source, errors, options) {
|
|
const len = typeof rule.len === 'number'
|
|
const min = typeof rule.min === 'number'
|
|
const max = typeof rule.max === 'number' // 正则匹配码点范围从U+010000一直到U+10FFFF的文字(补充平面Supplementary Plane)
|
|
|
|
const spRegexp = /[\uD800-\uDBFF][\uDC00-\uDFFF]/g
|
|
let val = value
|
|
let key = null
|
|
const num = typeof value === 'number'
|
|
const str = typeof value === 'string'
|
|
const arr = Array.isArray(value)
|
|
|
|
if (num) {
|
|
key = 'number'
|
|
} else if (str) {
|
|
key = 'string'
|
|
} else if (arr) {
|
|
key = 'array'
|
|
} // if the value is not of a supported type for range validation
|
|
// the validation rule rule should use the
|
|
// type property to also test for a particular type
|
|
|
|
if (!key) {
|
|
return false
|
|
}
|
|
|
|
if (arr) {
|
|
val = value.length
|
|
}
|
|
|
|
if (str) {
|
|
// 处理码点大于U+010000的文字length属性不准确的bug,如"𠮷𠮷𠮷".lenght !== 3
|
|
val = value.replace(spRegexp, '_').length
|
|
}
|
|
|
|
if (len) {
|
|
if (val !== rule.len) {
|
|
errors.push(format(options.messages[key].len, rule.fullField, rule.len))
|
|
}
|
|
} else if (min && !max && val < rule.min) {
|
|
errors.push(format(options.messages[key].min, rule.fullField, rule.min))
|
|
} else if (max && !min && val > rule.max) {
|
|
errors.push(format(options.messages[key].max, rule.fullField, rule.max))
|
|
} else if (min && max && (val < rule.min || val > rule.max)) {
|
|
errors.push(format(options.messages[key].range, rule.fullField, rule.min, rule.max))
|
|
}
|
|
}
|
|
|
|
const ENUM = 'enum'
|
|
/**
|
|
* Rule for validating a value exists in an enumerable list.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param source The source object being validated.
|
|
* @param errors An array of errors that this rule may add
|
|
* validation errors to.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function enumerable(rule, value, source, errors, options) {
|
|
rule[ENUM] = Array.isArray(rule[ENUM]) ? rule[ENUM] : []
|
|
|
|
if (rule[ENUM].indexOf(value) === -1) {
|
|
errors.push(format(options.messages[ENUM], rule.fullField, rule[ENUM].join(', ')))
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Rule for validating a regular expression pattern.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param source The source object being validated.
|
|
* @param errors An array of errors that this rule may add
|
|
* validation errors to.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function pattern$1(rule, value, source, errors, options) {
|
|
if (rule.pattern) {
|
|
if (rule.pattern instanceof RegExp) {
|
|
// if a RegExp instance is passed, reset `lastIndex` in case its `global`
|
|
// flag is accidentally set to `true`, which in a validation scenario
|
|
// is not necessary and the result might be misleading
|
|
rule.pattern.lastIndex = 0
|
|
|
|
if (!rule.pattern.test(value)) {
|
|
errors.push(format(options.messages.pattern.mismatch, rule.fullField, value, rule.pattern))
|
|
}
|
|
} else if (typeof rule.pattern === 'string') {
|
|
const _pattern = new RegExp(rule.pattern)
|
|
|
|
if (!_pattern.test(value)) {
|
|
errors.push(format(options.messages.pattern.mismatch, rule.fullField, value, rule.pattern))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const rules = {
|
|
required,
|
|
whitespace,
|
|
type,
|
|
range,
|
|
enum: enumerable,
|
|
pattern: pattern$1
|
|
}
|
|
|
|
/**
|
|
* Performs validation for string types.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param callback The callback function.
|
|
* @param source The source object being validated.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function string(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value, 'string') && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options, 'string')
|
|
|
|
if (!isEmptyValue(value, 'string')) {
|
|
rules.type(rule, value, source, errors, options)
|
|
rules.range(rule, value, source, errors, options)
|
|
rules.pattern(rule, value, source, errors, options)
|
|
|
|
if (rule.whitespace === true) {
|
|
rules.whitespace(rule, value, source, errors, options)
|
|
}
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
/**
|
|
* Validates a function.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param callback The callback function.
|
|
* @param source The source object being validated.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function method(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value) && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options)
|
|
|
|
if (value !== undefined) {
|
|
rules.type(rule, value, source, errors, options)
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
/**
|
|
* Validates a number.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param callback The callback function.
|
|
* @param source The source object being validated.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function number(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (value === '') {
|
|
value = undefined
|
|
}
|
|
|
|
if (isEmptyValue(value) && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options)
|
|
|
|
if (value !== undefined) {
|
|
rules.type(rule, value, source, errors, options)
|
|
rules.range(rule, value, source, errors, options)
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
/**
|
|
* Validates a boolean.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param callback The callback function.
|
|
* @param source The source object being validated.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function _boolean(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value) && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options)
|
|
|
|
if (value !== undefined) {
|
|
rules.type(rule, value, source, errors, options)
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
/**
|
|
* Validates the regular expression type.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param callback The callback function.
|
|
* @param source The source object being validated.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function regexp(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value) && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options)
|
|
|
|
if (!isEmptyValue(value)) {
|
|
rules.type(rule, value, source, errors, options)
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
/**
|
|
* Validates a number is an integer.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param callback The callback function.
|
|
* @param source The source object being validated.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function integer(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value) && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options)
|
|
|
|
if (value !== undefined) {
|
|
rules.type(rule, value, source, errors, options)
|
|
rules.range(rule, value, source, errors, options)
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
/**
|
|
* Validates a number is a floating point number.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param callback The callback function.
|
|
* @param source The source object being validated.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function floatFn(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value) && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options)
|
|
|
|
if (value !== undefined) {
|
|
rules.type(rule, value, source, errors, options)
|
|
rules.range(rule, value, source, errors, options)
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
/**
|
|
* Validates an array.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param callback The callback function.
|
|
* @param source The source object being validated.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function array(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value, 'array') && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options, 'array')
|
|
|
|
if (!isEmptyValue(value, 'array')) {
|
|
rules.type(rule, value, source, errors, options)
|
|
rules.range(rule, value, source, errors, options)
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
/**
|
|
* Validates an object.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param callback The callback function.
|
|
* @param source The source object being validated.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function object(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value) && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options)
|
|
|
|
if (value !== undefined) {
|
|
rules.type(rule, value, source, errors, options)
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
const ENUM$1 = 'enum'
|
|
/**
|
|
* Validates an enumerable list.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param callback The callback function.
|
|
* @param source The source object being validated.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function enumerable$1(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value) && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options)
|
|
|
|
if (value !== undefined) {
|
|
rules[ENUM$1](rule, value, source, errors, options)
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
/**
|
|
* Validates a regular expression pattern.
|
|
*
|
|
* Performs validation when a rule only contains
|
|
* a pattern property but is not declared as a string type.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param callback The callback function.
|
|
* @param source The source object being validated.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function pattern$2(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value, 'string') && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options)
|
|
|
|
if (!isEmptyValue(value, 'string')) {
|
|
rules.pattern(rule, value, source, errors, options)
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
function date(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value) && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options)
|
|
|
|
if (!isEmptyValue(value)) {
|
|
let dateObject
|
|
|
|
if (typeof value === 'number') {
|
|
dateObject = new Date(value)
|
|
} else {
|
|
dateObject = value
|
|
}
|
|
|
|
rules.type(rule, dateObject, source, errors, options)
|
|
|
|
if (dateObject) {
|
|
rules.range(rule, dateObject.getTime(), source, errors, options)
|
|
}
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
function required$1(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const type = Array.isArray(value) ? 'array' : typeof value
|
|
rules.required(rule, value, source, errors, options, type)
|
|
callback(errors)
|
|
}
|
|
|
|
function type$1(rule, value, callback, source, options) {
|
|
const ruleType = rule.type
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value, ruleType) && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options, ruleType)
|
|
|
|
if (!isEmptyValue(value, ruleType)) {
|
|
rules.type(rule, value, source, errors, options)
|
|
}
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
/**
|
|
* Performs validation for any type.
|
|
*
|
|
* @param rule The validation rule.
|
|
* @param value The value of the field on the source object.
|
|
* @param callback The callback function.
|
|
* @param source The source object being validated.
|
|
* @param options The validation options.
|
|
* @param options.messages The validation messages.
|
|
*/
|
|
|
|
function any(rule, value, callback, source, options) {
|
|
const errors = []
|
|
const validate = rule.required || !rule.required && source.hasOwnProperty(rule.field)
|
|
|
|
if (validate) {
|
|
if (isEmptyValue(value) && !rule.required) {
|
|
return callback()
|
|
}
|
|
|
|
rules.required(rule, value, source, errors, options)
|
|
}
|
|
|
|
callback(errors)
|
|
}
|
|
|
|
const validators = {
|
|
string,
|
|
method,
|
|
number,
|
|
boolean: _boolean,
|
|
regexp,
|
|
integer,
|
|
float: floatFn,
|
|
array,
|
|
object,
|
|
enum: enumerable$1,
|
|
pattern: pattern$2,
|
|
date,
|
|
url: type$1,
|
|
hex: type$1,
|
|
email: type$1,
|
|
required: required$1,
|
|
any
|
|
}
|
|
|
|
function newMessages() {
|
|
return {
|
|
default: 'Validation error on field %s',
|
|
required: '%s is required',
|
|
enum: '%s must be one of %s',
|
|
whitespace: '%s cannot be empty',
|
|
date: {
|
|
format: '%s date %s is invalid for format %s',
|
|
parse: '%s date could not be parsed, %s is invalid ',
|
|
invalid: '%s date %s is invalid'
|
|
},
|
|
types: {
|
|
string: '%s is not a %s',
|
|
method: '%s is not a %s (function)',
|
|
array: '%s is not an %s',
|
|
object: '%s is not an %s',
|
|
number: '%s is not a %s',
|
|
date: '%s is not a %s',
|
|
boolean: '%s is not a %s',
|
|
integer: '%s is not an %s',
|
|
float: '%s is not a %s',
|
|
regexp: '%s is not a valid %s',
|
|
email: '%s is not a valid %s',
|
|
url: '%s is not a valid %s',
|
|
hex: '%s is not a valid %s'
|
|
},
|
|
string: {
|
|
len: '%s must be exactly %s characters',
|
|
min: '%s must be at least %s characters',
|
|
max: '%s cannot be longer than %s characters',
|
|
range: '%s must be between %s and %s characters'
|
|
},
|
|
number: {
|
|
len: '%s must equal %s',
|
|
min: '%s cannot be less than %s',
|
|
max: '%s cannot be greater than %s',
|
|
range: '%s must be between %s and %s'
|
|
},
|
|
array: {
|
|
len: '%s must be exactly %s in length',
|
|
min: '%s cannot be less than %s in length',
|
|
max: '%s cannot be greater than %s in length',
|
|
range: '%s must be between %s and %s in length'
|
|
},
|
|
pattern: {
|
|
mismatch: '%s value %s does not match pattern %s'
|
|
},
|
|
clone: function clone() {
|
|
const cloned = JSON.parse(JSON.stringify(this))
|
|
cloned.clone = this.clone
|
|
return cloned
|
|
}
|
|
}
|
|
}
|
|
const messages = newMessages()
|
|
|
|
/**
|
|
* Encapsulates a validation schema.
|
|
*
|
|
* @param descriptor An object declaring validation rules
|
|
* for this schema.
|
|
*/
|
|
|
|
function Schema(descriptor) {
|
|
this.rules = null
|
|
this._messages = messages
|
|
this.define(descriptor)
|
|
}
|
|
|
|
Schema.prototype = {
|
|
messages: function messages(_messages) {
|
|
if (_messages) {
|
|
this._messages = deepMerge(newMessages(), _messages)
|
|
}
|
|
|
|
return this._messages
|
|
},
|
|
define: function define(rules) {
|
|
if (!rules) {
|
|
throw new Error('Cannot configure a schema with no rules')
|
|
}
|
|
|
|
if (typeof rules !== 'object' || Array.isArray(rules)) {
|
|
throw new Error('Rules must be an object')
|
|
}
|
|
|
|
this.rules = {}
|
|
let z
|
|
let item
|
|
|
|
for (z in rules) {
|
|
if (rules.hasOwnProperty(z)) {
|
|
item = rules[z]
|
|
this.rules[z] = Array.isArray(item) ? item : [item]
|
|
}
|
|
}
|
|
},
|
|
validate: function validate(source_, o, oc) {
|
|
const _this = this
|
|
|
|
if (o === void 0) {
|
|
o = {}
|
|
}
|
|
|
|
if (oc === void 0) {
|
|
oc = function oc() {}
|
|
}
|
|
|
|
let source = source_
|
|
let options = o
|
|
let callback = oc
|
|
|
|
if (typeof options === 'function') {
|
|
callback = options
|
|
options = {}
|
|
}
|
|
|
|
if (!this.rules || Object.keys(this.rules).length === 0) {
|
|
if (callback) {
|
|
callback()
|
|
}
|
|
|
|
return Promise.resolve()
|
|
}
|
|
|
|
function complete(results) {
|
|
let i
|
|
let errors = []
|
|
let fields = {}
|
|
|
|
function add(e) {
|
|
if (Array.isArray(e)) {
|
|
let _errors
|
|
|
|
errors = (_errors = errors).concat.apply(_errors, e)
|
|
} else {
|
|
errors.push(e)
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < results.length; i++) {
|
|
add(results[i])
|
|
}
|
|
|
|
if (!errors.length) {
|
|
errors = null
|
|
fields = null
|
|
} else {
|
|
fields = convertFieldsError(errors)
|
|
}
|
|
|
|
callback(errors, fields)
|
|
}
|
|
|
|
if (options.messages) {
|
|
let messages$1 = this.messages()
|
|
|
|
if (messages$1 === messages) {
|
|
messages$1 = newMessages()
|
|
}
|
|
|
|
deepMerge(messages$1, options.messages)
|
|
options.messages = messages$1
|
|
} else {
|
|
options.messages = this.messages()
|
|
}
|
|
|
|
let arr
|
|
let value
|
|
const series = {}
|
|
const keys = options.keys || Object.keys(this.rules)
|
|
keys.forEach((z) => {
|
|
arr = _this.rules[z]
|
|
value = source[z]
|
|
arr.forEach((r) => {
|
|
let rule = r
|
|
|
|
if (typeof rule.transform === 'function') {
|
|
if (source === source_) {
|
|
source = { ...source }
|
|
}
|
|
|
|
value = source[z] = rule.transform(value)
|
|
}
|
|
|
|
if (typeof rule === 'function') {
|
|
rule = {
|
|
validator: rule
|
|
}
|
|
} else {
|
|
rule = { ...rule }
|
|
}
|
|
|
|
rule.validator = _this.getValidationMethod(rule)
|
|
rule.field = z
|
|
rule.fullField = rule.fullField || z
|
|
rule.type = _this.getType(rule)
|
|
|
|
if (!rule.validator) {
|
|
return
|
|
}
|
|
|
|
series[z] = series[z] || []
|
|
series[z].push({
|
|
rule,
|
|
value,
|
|
source,
|
|
field: z
|
|
})
|
|
})
|
|
})
|
|
const errorFields = {}
|
|
return asyncMap(series, options, (data, doIt) => {
|
|
const { rule } = data
|
|
let deep = (rule.type === 'object' || rule.type === 'array') && (typeof rule.fields === 'object' || typeof rule.defaultField
|
|
=== 'object')
|
|
deep = deep && (rule.required || !rule.required && data.value)
|
|
rule.field = data.field
|
|
|
|
function addFullfield(key, schema) {
|
|
return { ...schema, fullField: `${rule.fullField}.${key}` }
|
|
}
|
|
|
|
function cb(e) {
|
|
if (e === void 0) {
|
|
e = []
|
|
}
|
|
|
|
let errors = e
|
|
|
|
if (!Array.isArray(errors)) {
|
|
errors = [errors]
|
|
}
|
|
|
|
if (!options.suppressWarning && errors.length) {
|
|
Schema.warning('async-validator:', errors)
|
|
}
|
|
|
|
if (errors.length && rule.message) {
|
|
errors = [].concat(rule.message)
|
|
}
|
|
|
|
errors = errors.map(complementError(rule))
|
|
|
|
if (options.first && errors.length) {
|
|
errorFields[rule.field] = 1
|
|
return doIt(errors)
|
|
}
|
|
|
|
if (!deep) {
|
|
doIt(errors)
|
|
} else {
|
|
// if rule is required but the target object
|
|
// does not exist fail at the rule level and don't
|
|
// go deeper
|
|
if (rule.required && !data.value) {
|
|
if (rule.message) {
|
|
errors = [].concat(rule.message).map(complementError(rule))
|
|
} else if (options.error) {
|
|
errors = [options.error(rule, format(options.messages.required, rule.field))]
|
|
} else {
|
|
errors = []
|
|
}
|
|
|
|
return doIt(errors)
|
|
}
|
|
|
|
let fieldsSchema = {}
|
|
|
|
if (rule.defaultField) {
|
|
for (const k in data.value) {
|
|
if (data.value.hasOwnProperty(k)) {
|
|
fieldsSchema[k] = rule.defaultField
|
|
}
|
|
}
|
|
}
|
|
|
|
fieldsSchema = { ...fieldsSchema, ...data.rule.fields }
|
|
|
|
for (const f in fieldsSchema) {
|
|
if (fieldsSchema.hasOwnProperty(f)) {
|
|
const fieldSchema = Array.isArray(fieldsSchema[f]) ? fieldsSchema[f] : [fieldsSchema[f]]
|
|
fieldsSchema[f] = fieldSchema.map(addFullfield.bind(null, f))
|
|
}
|
|
}
|
|
|
|
const schema = new Schema(fieldsSchema)
|
|
schema.messages(options.messages)
|
|
|
|
if (data.rule.options) {
|
|
data.rule.options.messages = options.messages
|
|
data.rule.options.error = options.error
|
|
}
|
|
|
|
schema.validate(data.value, data.rule.options || options, (errs) => {
|
|
const finalErrors = []
|
|
|
|
if (errors && errors.length) {
|
|
finalErrors.push.apply(finalErrors, errors)
|
|
}
|
|
|
|
if (errs && errs.length) {
|
|
finalErrors.push.apply(finalErrors, errs)
|
|
}
|
|
|
|
doIt(finalErrors.length ? finalErrors : null)
|
|
})
|
|
}
|
|
}
|
|
|
|
let res
|
|
|
|
if (rule.asyncValidator) {
|
|
res = rule.asyncValidator(rule, data.value, cb, data.source, options)
|
|
} else if (rule.validator) {
|
|
res = rule.validator(rule, data.value, cb, data.source, options)
|
|
|
|
if (res === true) {
|
|
cb()
|
|
} else if (res === false) {
|
|
cb(rule.message || `${rule.field} fails`)
|
|
} else if (res instanceof Array) {
|
|
cb(res)
|
|
} else if (res instanceof Error) {
|
|
cb(res.message)
|
|
}
|
|
}
|
|
|
|
if (res && res.then) {
|
|
res.then(() => cb(), (e) => cb(e))
|
|
}
|
|
}, (results) => {
|
|
complete(results)
|
|
})
|
|
},
|
|
getType: function getType(rule) {
|
|
if (rule.type === undefined && rule.pattern instanceof RegExp) {
|
|
rule.type = 'pattern'
|
|
}
|
|
|
|
if (typeof rule.validator !== 'function' && rule.type && !validators.hasOwnProperty(rule.type)) {
|
|
throw new Error(format('Unknown rule type %s', rule.type))
|
|
}
|
|
|
|
return rule.type || 'string'
|
|
},
|
|
getValidationMethod: function getValidationMethod(rule) {
|
|
if (typeof rule.validator === 'function') {
|
|
return rule.validator
|
|
}
|
|
|
|
const keys = Object.keys(rule)
|
|
const messageIndex = keys.indexOf('message')
|
|
|
|
if (messageIndex !== -1) {
|
|
keys.splice(messageIndex, 1)
|
|
}
|
|
|
|
if (keys.length === 1 && keys[0] === 'required') {
|
|
return validators.required
|
|
}
|
|
|
|
return validators[this.getType(rule)] || false
|
|
}
|
|
}
|
|
|
|
Schema.register = function register(type, validator) {
|
|
if (typeof validator !== 'function') {
|
|
throw new Error('Cannot register a validator by type, validator is not a function')
|
|
}
|
|
|
|
validators[type] = validator
|
|
}
|
|
|
|
Schema.warning = warning
|
|
Schema.messages = messages
|
|
|
|
export default Schema
|
|
// # sourceMappingURL=index.js.map
|