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.

validate.js 12KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488
  1. var pattern = {
  2. email: /^\S+?@\S+?\.\S+?$/,
  3. idcard: /^[1-9]\d{5}(18|19|([23]\d))\d{2}((0[1-9])|(10|11|12))(([0-2][1-9])|10|20|30|31)\d{3}[0-9Xx]$/,
  4. url: new RegExp(
  5. "^(?!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]*)?$",
  6. 'i')
  7. };
  8. const FORMAT_MAPPING = {
  9. "int": 'integer',
  10. "bool": 'boolean',
  11. "double": 'number',
  12. "long": 'number',
  13. "password": 'string'
  14. // "fileurls": 'array'
  15. }
  16. function formatMessage(args, resources = '') {
  17. var defaultMessage = ['label']
  18. defaultMessage.forEach((item) => {
  19. if (args[item] === undefined) {
  20. args[item] = ''
  21. }
  22. })
  23. let str = resources
  24. for (let key in args) {
  25. let reg = new RegExp('{' + key + '}')
  26. str = str.replace(reg, args[key])
  27. }
  28. return str
  29. }
  30. function isEmptyValue(value, type) {
  31. if (value === undefined || value === null) {
  32. return true;
  33. }
  34. if (typeof value === 'string' && !value) {
  35. return true;
  36. }
  37. if (Array.isArray(value) && !value.length) {
  38. return true;
  39. }
  40. if (type === 'object' && !Object.keys(value).length) {
  41. return true;
  42. }
  43. return false;
  44. }
  45. const types = {
  46. integer(value) {
  47. return types.number(value) && parseInt(value, 10) === value;
  48. },
  49. string(value) {
  50. return typeof value === 'string';
  51. },
  52. number(value) {
  53. if (isNaN(value)) {
  54. return false;
  55. }
  56. return typeof value === 'number';
  57. },
  58. "boolean": function(value) {
  59. return typeof value === 'boolean';
  60. },
  61. "float": function(value) {
  62. return types.number(value) && !types.integer(value);
  63. },
  64. array(value) {
  65. return Array.isArray(value);
  66. },
  67. object(value) {
  68. return typeof value === 'object' && !types.array(value);
  69. },
  70. date(value) {
  71. return value instanceof Date;
  72. },
  73. timestamp(value) {
  74. if (!this.integer(value) || Math.abs(value).toString().length > 16) {
  75. return false
  76. }
  77. return true;
  78. },
  79. file(value) {
  80. return typeof value.url === 'string';
  81. },
  82. email(value) {
  83. return typeof value === 'string' && !!value.match(pattern.email) && value.length < 255;
  84. },
  85. url(value) {
  86. return typeof value === 'string' && !!value.match(pattern.url);
  87. },
  88. pattern(reg, value) {
  89. try {
  90. return new RegExp(reg).test(value);
  91. } catch (e) {
  92. console.log('输出内容',e)
  93. return false;
  94. }
  95. },
  96. method(value) {
  97. return typeof value === 'function';
  98. },
  99. idcard(value) {
  100. return typeof value === 'string' && !!value.match(pattern.idcard);
  101. },
  102. 'url-https'(value) {
  103. return this.url(value) && value.startsWith('https://');
  104. },
  105. 'url-scheme'(value) {
  106. return value.startsWith('://');
  107. },
  108. 'url-web'(value) {
  109. return false;
  110. }
  111. }
  112. class RuleValidator {
  113. constructor(message) {
  114. this._message = message
  115. }
  116. async validateRule(fieldKey, fieldValue, value, data, allData) {
  117. var result = null
  118. let rules = fieldValue.rules
  119. let hasRequired = rules.findIndex((item) => {
  120. return item.required
  121. })
  122. if (hasRequired < 0) {
  123. if (value === null || value === undefined) {
  124. return result
  125. }
  126. if (typeof value === 'string' && !value.length) {
  127. return result
  128. }
  129. }
  130. var message = this._message
  131. if (rules === undefined) {
  132. return message['default']
  133. }
  134. for (var i = 0; i < rules.length; i++) {
  135. let rule = rules[i]
  136. let vt = this._getValidateType(rule)
  137. Object.assign(rule, {
  138. label: fieldValue.label || `["${fieldKey}"]`
  139. })
  140. if (RuleValidatorHelper[vt]) {
  141. result = RuleValidatorHelper[vt](rule, value, message)
  142. if (result != null) {
  143. break
  144. }
  145. }
  146. if (rule.validateExpr) {
  147. let now = Date.now()
  148. let resultExpr = rule.validateExpr(value, allData, now)
  149. if (resultExpr === false) {
  150. result = this._getMessage(rule, rule.errorMessage || this._message['default'])
  151. break
  152. }
  153. }
  154. if (rule.validateFunction) {
  155. result = await this.validateFunction(rule, value, data, allData, vt)
  156. if (result !== null) {
  157. break
  158. }
  159. }
  160. }
  161. if (result !== null) {
  162. result = message.TAG + result
  163. }
  164. return result
  165. }
  166. async validateFunction(rule, value, data, allData, vt) {
  167. let result = null
  168. try {
  169. let callbackMessage = null
  170. const res = await rule.validateFunction(rule, value, allData || data, (message) => {
  171. callbackMessage = message
  172. })
  173. if (callbackMessage || (typeof res === 'string' && res) || res === false) {
  174. result = this._getMessage(rule, callbackMessage || res, vt)
  175. }
  176. } catch (e) {
  177. console.log('输出内容',e)
  178. result = this._getMessage(rule, e.message, vt)
  179. }
  180. return result
  181. }
  182. _getMessage(rule, message, vt) {
  183. return formatMessage(rule, message || rule.errorMessage || this._message[vt] || message['default'])
  184. }
  185. _getValidateType(rule) {
  186. var result = ''
  187. if (rule.required) {
  188. result = 'required'
  189. } else if (rule.format) {
  190. result = 'format'
  191. } else if (rule.arrayType) {
  192. result = 'arrayTypeFormat'
  193. } else if (rule.range) {
  194. result = 'range'
  195. } else if (rule.maximum !== undefined || rule.minimum !== undefined) {
  196. result = 'rangeNumber'
  197. } else if (rule.maxLength !== undefined || rule.minLength !== undefined) {
  198. result = 'rangeLength'
  199. } else if (rule.pattern) {
  200. result = 'pattern'
  201. } else if (rule.validateFunction) {
  202. result = 'validateFunction'
  203. }
  204. return result
  205. }
  206. }
  207. const RuleValidatorHelper = {
  208. required(rule, value, message) {
  209. if (rule.required && isEmptyValue(value, rule.format || typeof value)) {
  210. return formatMessage(rule, rule.errorMessage || message.required);
  211. }
  212. return null
  213. },
  214. range(rule, value, message) {
  215. const {
  216. range,
  217. errorMessage
  218. } = rule;
  219. let list = new Array(range.length);
  220. for (let i = 0; i < range.length; i++) {
  221. const item = range[i];
  222. if (types.object(item) && item.value !== undefined) {
  223. list[i] = item.value;
  224. } else {
  225. list[i] = item;
  226. }
  227. }
  228. let result = false
  229. if (Array.isArray(value)) {
  230. result = (new Set(value.concat(list)).size === list.length);
  231. } else {
  232. if (list.indexOf(value) > -1) {
  233. result = true;
  234. }
  235. }
  236. if (!result) {
  237. return formatMessage(rule, errorMessage || message['enum']);
  238. }
  239. return null
  240. },
  241. rangeNumber(rule, value, message) {
  242. if (!types.number(value)) {
  243. return formatMessage(rule, rule.errorMessage || message.pattern.mismatch);
  244. }
  245. let {
  246. minimum,
  247. maximum,
  248. exclusiveMinimum,
  249. exclusiveMaximum
  250. } = rule;
  251. let min = exclusiveMinimum ? value <= minimum : value < minimum;
  252. let max = exclusiveMaximum ? value >= maximum : value > maximum;
  253. if (minimum !== undefined && min) {
  254. return formatMessage(rule, rule.errorMessage || message['number'][exclusiveMinimum ?
  255. 'exclusiveMinimum' : 'minimum'
  256. ])
  257. } else if (maximum !== undefined && max) {
  258. return formatMessage(rule, rule.errorMessage || message['number'][exclusiveMaximum ?
  259. 'exclusiveMaximum' : 'maximum'
  260. ])
  261. } else if (minimum !== undefined && maximum !== undefined && (min || max)) {
  262. return formatMessage(rule, rule.errorMessage || message['number'].range)
  263. }
  264. return null
  265. },
  266. rangeLength(rule, value, message) {
  267. if (!types.string(value) && !types.array(value)) {
  268. return formatMessage(rule, rule.errorMessage || message.pattern.mismatch);
  269. }
  270. let min = rule.minLength;
  271. let max = rule.maxLength;
  272. let val = value.length;
  273. if (min !== undefined && val < min) {
  274. return formatMessage(rule, rule.errorMessage || message['length'].minLength)
  275. } else if (max !== undefined && val > max) {
  276. return formatMessage(rule, rule.errorMessage || message['length'].maxLength)
  277. } else if (min !== undefined && max !== undefined && (val < min || val > max)) {
  278. return formatMessage(rule, rule.errorMessage || message['length'].range)
  279. }
  280. return null
  281. },
  282. pattern(rule, value, message) {
  283. if (!types['pattern'](rule.pattern, value)) {
  284. return formatMessage(rule, rule.errorMessage || message.pattern.mismatch);
  285. }
  286. return null
  287. },
  288. format(rule, value, message) {
  289. var customTypes = Object.keys(types);
  290. var format = FORMAT_MAPPING[rule.format] ? FORMAT_MAPPING[rule.format] : (rule.format || rule.arrayType);
  291. if (customTypes.indexOf(format) > -1) {
  292. if (!types[format](value)) {
  293. return formatMessage(rule, rule.errorMessage || message.typeError);
  294. }
  295. }
  296. return null
  297. },
  298. arrayTypeFormat(rule, value, message) {
  299. if (!Array.isArray(value)) {
  300. return formatMessage(rule, rule.errorMessage || message.typeError);
  301. }
  302. for (let i = 0; i < value.length; i++) {
  303. const element = value[i];
  304. let formatResult = this.format(rule, element, message)
  305. if (formatResult !== null) {
  306. return formatResult
  307. }
  308. }
  309. return null
  310. }
  311. }
  312. class SchemaValidator extends RuleValidator {
  313. constructor(schema, options) {
  314. super(SchemaValidator.message);
  315. this._schema = schema
  316. this._options = options || null
  317. }
  318. updateSchema(schema) {
  319. this._schema = schema
  320. }
  321. async validate(data, allData) {
  322. let result = this._checkFieldInSchema(data)
  323. if (!result) {
  324. result = await this.invokeValidate(data, false, allData)
  325. }
  326. return result.length ? result[0] : null
  327. }
  328. async validateAll(data, allData) {
  329. let result = this._checkFieldInSchema(data)
  330. if (!result) {
  331. result = await this.invokeValidate(data, true, allData)
  332. }
  333. return result
  334. }
  335. async validateUpdate(data, allData) {
  336. let result = this._checkFieldInSchema(data)
  337. if (!result) {
  338. result = await this.invokeValidateUpdate(data, false, allData)
  339. }
  340. return result.length ? result[0] : null
  341. }
  342. async invokeValidate(data, all, allData) {
  343. let result = []
  344. let schema = this._schema
  345. for (let key in schema) {
  346. let value = schema[key]
  347. let errorMessage = await this.validateRule(key, value, data[key], data, allData)
  348. if (errorMessage != null) {
  349. result.push({
  350. key,
  351. errorMessage
  352. })
  353. if (!all) break
  354. }
  355. }
  356. return result
  357. }
  358. async invokeValidateUpdate(data, all, allData) {
  359. let result = []
  360. for (let key in data) {
  361. let errorMessage = await this.validateRule(key, this._schema[key], data[key], data, allData)
  362. if (errorMessage != null) {
  363. result.push({
  364. key,
  365. errorMessage
  366. })
  367. if (!all) break
  368. }
  369. }
  370. return result
  371. }
  372. _checkFieldInSchema(data) {
  373. var keys = Object.keys(data)
  374. var keys2 = Object.keys(this._schema)
  375. if (new Set(keys.concat(keys2)).size === keys2.length) {
  376. return ''
  377. }
  378. var noExistFields = keys.filter((key) => {
  379. return keys2.indexOf(key) < 0;
  380. })
  381. var errorMessage = formatMessage({
  382. field: JSON.stringify(noExistFields)
  383. }, SchemaValidator.message.TAG + SchemaValidator.message['defaultInvalid'])
  384. return [{
  385. key: 'invalid',
  386. errorMessage
  387. }]
  388. }
  389. }
  390. function Message() {
  391. return {
  392. TAG: "",
  393. default: '验证错误',
  394. defaultInvalid: '提交的字段{field}在数据库中并不存在',
  395. validateFunction: '验证无效',
  396. required: '{label}必填',
  397. 'enum': '{label}超出范围',
  398. timestamp: '{label}格式无效',
  399. whitespace: '{label}不能为空',
  400. typeError: '{label}类型无效',
  401. date: {
  402. format: '{label}日期{value}格式无效',
  403. parse: '{label}日期无法解析,{value}无效',
  404. invalid: '{label}日期{value}无效'
  405. },
  406. length: {
  407. minLength: '{label}长度不能少于{minLength}',
  408. maxLength: '{label}长度不能超过{maxLength}',
  409. range: '{label}必须介于{minLength}和{maxLength}之间'
  410. },
  411. number: {
  412. minimum: '{label}不能小于{minimum}',
  413. maximum: '{label}不能大于{maximum}',
  414. exclusiveMinimum: '{label}不能小于等于{minimum}',
  415. exclusiveMaximum: '{label}不能大于等于{maximum}',
  416. range: '{label}必须介于{minimum}and{maximum}之间'
  417. },
  418. pattern: {
  419. mismatch: '{label}格式不匹配'
  420. }
  421. };
  422. }
  423. SchemaValidator.message = new Message();
  424. export default SchemaValidator