Home Reference Source Test

src/Rule/index.js

  1. const ErrorCollector = require('./ErrorCollector');
  2. const { getErrorFromObject, getErrorFromFunctionOrString } = require('./util');
  3. const { TEST_FUNCTIONS, OPTIONAL } = require('../testFunctions');
  4. const { AND, OR, isObject } = require('./../util');
  5.  
  6. const OPERATORS = {
  7. '&': AND,
  8. '|': OR,
  9. };
  10.  
  11. /**
  12. * The Rule class validates only one value
  13. * once a rule is created it can be used multiple times
  14. */
  15. class Rule {
  16. /**
  17. *
  18. * @param {String|Object} obj the rule object it describes a the test that are ran by the Rule
  19. * @param {String} error the error returned when the tested input is not correct
  20. */
  21. constructor(obj, error) {
  22. if (typeof obj === 'string' || obj instanceof String) {
  23. this.rule = { type: obj };
  24. } else {
  25. this.rule = obj;
  26. }
  27. this.error = error;
  28. this.errorCollector = new ErrorCollector();
  29. this.testEntryObject();
  30. }
  31.  
  32. /**
  33. *
  34. * @param {any} val the value to be tested
  35. * @param {Object|String} obj the error object or string thats showed on error
  36. * @param {String} path the path to the tested value this is used when
  37. * using validator to keep track of the prop value ex: obj.min
  38. *
  39. * @return {boolean}
  40. */
  41.  
  42. test(val, obj, path) {
  43. this.errorCollector.clear();
  44. const types = this.getTypes();
  45. const operators = this.getRuleOperators();
  46. let ret = this.testOneRule(val, obj, types[0], path);
  47.  
  48. for (let i = 1; i < types.length; i += 1) {
  49. const operator = operators[i] || operators[i - 1];
  50. ret = operator(ret, this.testOneRule(val, obj, types[i], path));
  51. }
  52. return ret;
  53. }
  54.  
  55. /**
  56. * converts array from string if multiple types given in type
  57. * its the case for exemple int|float
  58. * @private
  59. * @return {[String]}
  60. */
  61.  
  62. getTypes() {
  63. return this.rule.type.split(/[&|]/);
  64. }
  65.  
  66. /**
  67. * Returns a list of the operators when multiple types given
  68. * its the case for example int|float
  69. * @private
  70. * @returns {[String]}
  71. */
  72. getRuleOperators() {
  73. const ret = [];
  74. const operators = this.rule.type.match(/[&|]/g) || '&';
  75. for (let i = 0; i < operators.length; i += 1) {
  76. ret.push(OPERATORS[operators[i]]);
  77. }
  78. return ret;
  79. }
  80.  
  81. /**
  82. * @private
  83. * @param val value to be tested
  84. * @param {Object} obj error object
  85. * @param {String} type the type from getTypes()
  86. * @param {String} path the path to the value if Validator is used
  87. *
  88. * @returns {boolean}
  89. */
  90. testOneRule(val, obj, type, path) {
  91. if (Rule.TEST_FUNCTIONS[type].optional(val, this.rule.optional, obj) === true) {
  92. return true;
  93. }
  94.  
  95. const keys = Object.keys(this.rule);
  96. keys.sort((key) => {
  97. if (key === 'type') return -1;
  98. return 0;
  99. });
  100.  
  101. for (let i = 0; i < keys.length; i += 1) {
  102. const key = keys[i];
  103. const testFunction = Rule.TEST_FUNCTIONS[type][key];
  104.  
  105. if (testFunction(val, this.rule[key], obj) === false && testFunction !== OPTIONAL) {
  106. this.errorCollector.collect(this.getError(path, val, key));
  107. return false;
  108. }
  109. }
  110. return true;
  111. }
  112.  
  113. /**
  114. * Tests the validity of the constructor object
  115. * thows an error if the object is invalid
  116. */
  117.  
  118. testEntryObject() {
  119. if (!this.rule.type) {
  120. throw Error('`type` is required');
  121. }
  122. const types = this.getTypes();
  123. types.forEach((type) => {
  124. this.testEntryObjectOneType(type);
  125. });
  126. }
  127.  
  128. /**
  129. * Tests the validity of the constructor object
  130. * thows an error if the object is invalid
  131. * tests if all the keys are valid
  132. */
  133.  
  134. testEntryObjectOneType(type) {
  135. const keys = Object.keys(this.rule);
  136. for (let i = 0; i < keys.length; i += 1) {
  137. const key = keys[i];
  138. if (!Rule.TEST_FUNCTIONS[type]) {
  139. throw Error(`The \`${type}\` type doesn't exist`);
  140. }
  141. if (!Rule.TEST_FUNCTIONS[type][key]) {
  142. throw new Error(`\`${type}\` doesn't have "${key}" test!`);
  143. }
  144. }
  145. }
  146.  
  147. /**
  148. * returns a list of errors if they are present
  149. * @return {[String]}
  150. */
  151.  
  152. getError(path, value, key) {
  153. if (isObject(this.error)) {
  154. return getErrorFromObject(this.error, path, value, key);
  155. }
  156. return getErrorFromFunctionOrString(this.error, path, value);
  157. }
  158.  
  159. /**
  160. * Add custom rule to the Rule class
  161. * @param {String} name the name of the rule
  162. * @param {Function} rule the validation function
  163. */
  164. static addCustom(name, rule) {
  165. Rule.TEST_FUNCTIONS[name] = rule;
  166. Rule.TEST_FUNCTIONS[name].optional = OPTIONAL;
  167. }
  168. }
  169.  
  170. Rule.TEST_FUNCTIONS = TEST_FUNCTIONS;
  171.  
  172. module.exports = Rule;