1357 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			1357 lines
		
	
	
		
			33 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| function _extends() {
 | ||
| 	_extends = Object.assign || function(target) {
 | ||
| 		for (var i = 1; i < arguments.length; i++) {
 | ||
| 			var source = arguments[i];
 | ||
| 
 | ||
| 			for (var 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 */
 | ||
| var formatRegExp = /%[sdj%]/g;
 | ||
| var 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(function(e) {
 | ||
| 					return typeof e === 'string';
 | ||
| 				})) {
 | ||
| 				console.warn(type, errors);
 | ||
| 			}
 | ||
| 		}
 | ||
| 	};
 | ||
| }
 | ||
| 
 | ||
| function convertFieldsError(errors) {
 | ||
| 	if (!errors || !errors.length) return null;
 | ||
| 	var fields = {};
 | ||
| 	errors.forEach(function(error) {
 | ||
| 		var field = error.field;
 | ||
| 		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];
 | ||
| 	}
 | ||
| 
 | ||
| 	var i = 1;
 | ||
| 	var f = args[0];
 | ||
| 	var len = args.length;
 | ||
| 
 | ||
| 	if (typeof f === 'function') {
 | ||
| 		return f.apply(null, args.slice(1));
 | ||
| 	}
 | ||
| 
 | ||
| 	if (typeof f === 'string') {
 | ||
| 		var str = String(f).replace(formatRegExp, function(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 (var 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) {
 | ||
| 	var results = [];
 | ||
| 	var total = 0;
 | ||
| 	var arrLength = arr.length;
 | ||
| 
 | ||
| 	function count(errors) {
 | ||
| 		results.push.apply(results, errors);
 | ||
| 		total++;
 | ||
| 
 | ||
| 		if (total === arrLength) {
 | ||
| 			callback(results);
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	arr.forEach(function(a) {
 | ||
| 		func(a, count);
 | ||
| 	});
 | ||
| }
 | ||
| 
 | ||
| function asyncSerialArray(arr, func, callback) {
 | ||
| 	var index = 0;
 | ||
| 	var arrLength = arr.length;
 | ||
| 
 | ||
| 	function next(errors) {
 | ||
| 		if (errors && errors.length) {
 | ||
| 			callback(errors);
 | ||
| 			return;
 | ||
| 		}
 | ||
| 
 | ||
| 		var original = index;
 | ||
| 		index = index + 1;
 | ||
| 
 | ||
| 		if (original < arrLength) {
 | ||
| 			func(arr[original], next);
 | ||
| 		} else {
 | ||
| 			callback([]);
 | ||
| 		}
 | ||
| 	}
 | ||
| 
 | ||
| 	next([]);
 | ||
| }
 | ||
| 
 | ||
| function flattenObjArr(objArr) {
 | ||
| 	var ret = [];
 | ||
| 	Object.keys(objArr).forEach(function(k) {
 | ||
| 		ret.push.apply(ret, objArr[k]);
 | ||
| 	});
 | ||
| 	return ret;
 | ||
| }
 | ||
| 
 | ||
| function asyncMap(objArr, option, func, callback) {
 | ||
| 	if (option.first) {
 | ||
| 		var _pending = new Promise(function(resolve, reject) {
 | ||
| 			var next = function next(errors) {
 | ||
| 				callback(errors);
 | ||
| 				return errors.length ? reject({
 | ||
| 					errors: errors,
 | ||
| 					fields: convertFieldsError(errors)
 | ||
| 				}) : resolve();
 | ||
| 			};
 | ||
| 
 | ||
| 			var flattenArr = flattenObjArr(objArr);
 | ||
| 			asyncSerialArray(flattenArr, func, next);
 | ||
| 		});
 | ||
| 
 | ||
| 		_pending["catch"](function(e) {
 | ||
| 			return e;
 | ||
| 		});
 | ||
| 
 | ||
| 		return _pending;
 | ||
| 	}
 | ||
| 
 | ||
| 	var firstFields = option.firstFields || [];
 | ||
| 
 | ||
| 	if (firstFields === true) {
 | ||
| 		firstFields = Object.keys(objArr);
 | ||
| 	}
 | ||
| 
 | ||
| 	var objArrKeys = Object.keys(objArr);
 | ||
| 	var objArrLength = objArrKeys.length;
 | ||
| 	var total = 0;
 | ||
| 	var results = [];
 | ||
| 	var pending = new Promise(function(resolve, reject) {
 | ||
| 		var 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(function(key) {
 | ||
| 			var arr = objArr[key];
 | ||
| 
 | ||
| 			if (firstFields.indexOf(key) !== -1) {
 | ||
| 				asyncSerialArray(arr, func, next);
 | ||
| 			} else {
 | ||
| 				asyncParallelArray(arr, func, next);
 | ||
| 			}
 | ||
| 		});
 | ||
| 	});
 | ||
| 	pending["catch"](function(e) {
 | ||
| 		return 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 (var s in source) {
 | ||
| 			if (source.hasOwnProperty(s)) {
 | ||
| 				var value = source[s];
 | ||
| 
 | ||
| 				if (typeof value === 'object' && typeof target[s] === 'object') {
 | ||
| 					target[s] = _extends({}, 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 */
 | ||
| 
 | ||
| var 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 types.number(value) && parseInt(value, 10) === value;
 | ||
| 	},
 | ||
| 	"float": function float(value) {
 | ||
| 		return types.number(value) && !types.integer(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;
 | ||
| 	}
 | ||
| 
 | ||
| 	var custom = ['integer', 'float', 'array', 'regexp', 'object', 'method', 'email', 'number', 'date', 'url', 'hex'];
 | ||
| 	var 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) {
 | ||
| 	var len = typeof rule.len === 'number';
 | ||
| 	var min = typeof rule.min === 'number';
 | ||
| 	var max = typeof rule.max === 'number'; // 正则匹配码点范围从U+010000一直到U+10FFFF的文字(补充平面Supplementary Plane)
 | ||
| 
 | ||
| 	var spRegexp = /[\uD800-\uDBFF][\uDC00-\uDFFF]/g;
 | ||
| 	var val = value;
 | ||
| 	var key = null;
 | ||
| 	var num = typeof value === 'number';
 | ||
| 	var str = typeof value === 'string';
 | ||
| 	var 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));
 | ||
| 	}
 | ||
| }
 | ||
| 
 | ||
| var 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') {
 | ||
| 			var _pattern = new RegExp(rule.pattern);
 | ||
| 
 | ||
| 			if (!_pattern.test(value)) {
 | ||
| 				errors.push(format(options.messages.pattern.mismatch, rule.fullField, value, rule.pattern));
 | ||
| 			}
 | ||
| 		}
 | ||
| 	}
 | ||
| }
 | ||
| 
 | ||
| var rules = {
 | ||
| 	required: required,
 | ||
| 	whitespace: whitespace,
 | ||
| 	type: type,
 | ||
| 	range: 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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);
 | ||
| }
 | ||
| 
 | ||
| var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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)) {
 | ||
| 			var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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) {
 | ||
| 	var ruleType = rule.type;
 | ||
| 	var errors = [];
 | ||
| 	var 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) {
 | ||
| 	var errors = [];
 | ||
| 	var 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);
 | ||
| }
 | ||
| 
 | ||
| var validators = {
 | ||
| 	string: string,
 | ||
| 	method: method,
 | ||
| 	number: number,
 | ||
| 	"boolean": _boolean,
 | ||
| 	regexp: regexp,
 | ||
| 	integer: integer,
 | ||
| 	"float": floatFn,
 | ||
| 	array: array,
 | ||
| 	object: object,
 | ||
| 	"enum": enumerable$1,
 | ||
| 	pattern: pattern$2,
 | ||
| 	date: date,
 | ||
| 	url: type$1,
 | ||
| 	hex: type$1,
 | ||
| 	email: type$1,
 | ||
| 	required: required$1,
 | ||
| 	any: 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() {
 | ||
| 			var cloned = JSON.parse(JSON.stringify(this));
 | ||
| 			cloned.clone = this.clone;
 | ||
| 			return cloned;
 | ||
| 		}
 | ||
| 	};
 | ||
| }
 | ||
| var 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 = {};
 | ||
| 		var z;
 | ||
| 		var 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) {
 | ||
| 		var _this = this;
 | ||
| 
 | ||
| 		if (o === void 0) {
 | ||
| 			o = {};
 | ||
| 		}
 | ||
| 
 | ||
| 		if (oc === void 0) {
 | ||
| 			oc = function oc() {};
 | ||
| 		}
 | ||
| 
 | ||
| 		var source = source_;
 | ||
| 		var options = o;
 | ||
| 		var 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) {
 | ||
| 			var i;
 | ||
| 			var errors = [];
 | ||
| 			var fields = {};
 | ||
| 
 | ||
| 			function add(e) {
 | ||
| 				if (Array.isArray(e)) {
 | ||
| 					var _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) {
 | ||
| 			var 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();
 | ||
| 		}
 | ||
| 
 | ||
| 		var arr;
 | ||
| 		var value;
 | ||
| 		var series = {};
 | ||
| 		var keys = options.keys || Object.keys(this.rules);
 | ||
| 		keys.forEach(function(z) {
 | ||
| 			arr = _this.rules[z];
 | ||
| 			value = source[z];
 | ||
| 			arr.forEach(function(r) {
 | ||
| 				var rule = r;
 | ||
| 
 | ||
| 				if (typeof rule.transform === 'function') {
 | ||
| 					if (source === source_) {
 | ||
| 						source = _extends({}, source);
 | ||
| 					}
 | ||
| 
 | ||
| 					value = source[z] = rule.transform(value);
 | ||
| 				}
 | ||
| 
 | ||
| 				if (typeof rule === 'function') {
 | ||
| 					rule = {
 | ||
| 						validator: rule
 | ||
| 					};
 | ||
| 				} else {
 | ||
| 					rule = _extends({}, 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: rule,
 | ||
| 					value: value,
 | ||
| 					source: source,
 | ||
| 					field: z
 | ||
| 				});
 | ||
| 			});
 | ||
| 		});
 | ||
| 		var errorFields = {};
 | ||
| 		return asyncMap(series, options, function(data, doIt) {
 | ||
| 			var rule = data.rule;
 | ||
| 			var 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 _extends({}, schema, {
 | ||
| 					fullField: rule.fullField + "." + key
 | ||
| 				});
 | ||
| 			}
 | ||
| 
 | ||
| 			function cb(e) {
 | ||
| 				if (e === void 0) {
 | ||
| 					e = [];
 | ||
| 				}
 | ||
| 
 | ||
| 				var 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);
 | ||
| 					}
 | ||
| 
 | ||
| 					var fieldsSchema = {};
 | ||
| 
 | ||
| 					if (rule.defaultField) {
 | ||
| 						for (var k in data.value) {
 | ||
| 							if (data.value.hasOwnProperty(k)) {
 | ||
| 								fieldsSchema[k] = rule.defaultField;
 | ||
| 							}
 | ||
| 						}
 | ||
| 					}
 | ||
| 
 | ||
| 					fieldsSchema = _extends({}, fieldsSchema, {}, data.rule.fields);
 | ||
| 
 | ||
| 					for (var f in fieldsSchema) {
 | ||
| 						if (fieldsSchema.hasOwnProperty(f)) {
 | ||
| 							var fieldSchema = Array.isArray(fieldsSchema[f]) ? fieldsSchema[f] : [fieldsSchema[f]];
 | ||
| 							fieldsSchema[f] = fieldSchema.map(addFullfield.bind(null, f));
 | ||
| 						}
 | ||
| 					}
 | ||
| 
 | ||
| 					var 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, function(errs) {
 | ||
| 						var finalErrors = [];
 | ||
| 
 | ||
| 						if (errors && errors.length) {
 | ||
| 							finalErrors.push.apply(finalErrors, errors);
 | ||
| 						}
 | ||
| 
 | ||
| 						if (errs && errs.length) {
 | ||
| 							finalErrors.push.apply(finalErrors, errs);
 | ||
| 						}
 | ||
| 
 | ||
| 						doIt(finalErrors.length ? finalErrors : null);
 | ||
| 					});
 | ||
| 				}
 | ||
| 			}
 | ||
| 
 | ||
| 			var 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(function() {
 | ||
| 					return cb();
 | ||
| 				}, function(e) {
 | ||
| 					return cb(e);
 | ||
| 				});
 | ||
| 			}
 | ||
| 		}, function(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;
 | ||
| 		}
 | ||
| 
 | ||
| 		var keys = Object.keys(rule);
 | ||
| 		var 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
 | 
