util.js 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. 'use strict';
  2. module.exports = {
  3. copy: copy,
  4. checkDataType: checkDataType,
  5. checkDataTypes: checkDataTypes,
  6. coerceToTypes: coerceToTypes,
  7. toHash: toHash,
  8. getProperty: getProperty,
  9. escapeQuotes: escapeQuotes,
  10. ucs2length: require('./ucs2length'),
  11. varOccurences: varOccurences,
  12. varReplace: varReplace,
  13. cleanUpCode: cleanUpCode,
  14. finalCleanUpCode: finalCleanUpCode,
  15. schemaHasRules: schemaHasRules,
  16. schemaHasRulesExcept: schemaHasRulesExcept,
  17. toQuotedString: toQuotedString,
  18. getPathExpr: getPathExpr,
  19. getPath: getPath,
  20. getData: getData,
  21. unescapeFragment: unescapeFragment,
  22. escapeFragment: escapeFragment,
  23. escapeJsonPointer: escapeJsonPointer
  24. };
  25. function copy(o, to) {
  26. to = to || {};
  27. for (var key in o) to[key] = o[key];
  28. return to;
  29. }
  30. function checkDataType(dataType, data, negate) {
  31. var EQUAL = negate ? ' !== ' : ' === '
  32. , AND = negate ? ' || ' : ' && '
  33. , OK = negate ? '!' : ''
  34. , NOT = negate ? '' : '!';
  35. switch (dataType) {
  36. case 'null': return data + EQUAL + 'null';
  37. case 'array': return OK + 'Array.isArray(' + data + ')';
  38. case 'object': return '(' + OK + data + AND +
  39. 'typeof ' + data + EQUAL + '"object"' + AND +
  40. NOT + 'Array.isArray(' + data + '))';
  41. case 'integer': return '(typeof ' + data + EQUAL + '"number"' + AND +
  42. NOT + '(' + data + ' % 1)' +
  43. AND + data + EQUAL + data + ')';
  44. default: return 'typeof ' + data + EQUAL + '"' + dataType + '"';
  45. }
  46. }
  47. function checkDataTypes(dataTypes, data) {
  48. switch (dataTypes.length) {
  49. case 1: return checkDataType(dataTypes[0], data, true);
  50. default:
  51. var code = '';
  52. var types = toHash(dataTypes);
  53. if (types.array && types.object) {
  54. code = types.null ? '(': '(!' + data + ' || ';
  55. code += 'typeof ' + data + ' !== "object")';
  56. delete types.null;
  57. delete types.array;
  58. delete types.object;
  59. }
  60. if (types.number) delete types.integer;
  61. for (var t in types)
  62. code += (code ? ' && ' : '' ) + checkDataType(t, data, true);
  63. return code;
  64. }
  65. }
  66. var COERCE_TO_TYPES = toHash([ 'string', 'number', 'integer', 'boolean', 'null' ]);
  67. function coerceToTypes(optionCoerceTypes, dataTypes) {
  68. if (Array.isArray(dataTypes)) {
  69. var types = [];
  70. for (var i=0; i<dataTypes.length; i++) {
  71. var t = dataTypes[i];
  72. if (COERCE_TO_TYPES[t]) types[types.length] = t;
  73. else if (optionCoerceTypes === 'array' && t === 'array') types[types.length] = t;
  74. }
  75. if (types.length) return types;
  76. } else if (COERCE_TO_TYPES[dataTypes]) {
  77. return [dataTypes];
  78. } else if (optionCoerceTypes === 'array' && dataTypes === 'array') {
  79. return ['array'];
  80. }
  81. }
  82. function toHash(arr) {
  83. var hash = {};
  84. for (var i=0; i<arr.length; i++) hash[arr[i]] = true;
  85. return hash;
  86. }
  87. var IDENTIFIER = /^[a-z$_][a-z$_0-9]*$/i;
  88. var SINGLE_QUOTE = /'|\\/g;
  89. function getProperty(key) {
  90. return typeof key == 'number'
  91. ? '[' + key + ']'
  92. : IDENTIFIER.test(key)
  93. ? '.' + key
  94. : "['" + escapeQuotes(key) + "']";
  95. }
  96. function escapeQuotes(str) {
  97. return str.replace(SINGLE_QUOTE, '\\$&')
  98. .replace(/\n/g, '\\n')
  99. .replace(/\r/g, '\\r')
  100. .replace(/\f/g, '\\f')
  101. .replace(/\t/g, '\\t');
  102. }
  103. function varOccurences(str, dataVar) {
  104. dataVar += '[^0-9]';
  105. var matches = str.match(new RegExp(dataVar, 'g'));
  106. return matches ? matches.length : 0;
  107. }
  108. function varReplace(str, dataVar, expr) {
  109. dataVar += '([^0-9])';
  110. expr = expr.replace(/\$/g, '$$$$');
  111. return str.replace(new RegExp(dataVar, 'g'), expr + '$1');
  112. }
  113. var EMPTY_ELSE = /else\s*{\s*}/g
  114. , EMPTY_IF_NO_ELSE = /if\s*\([^)]+\)\s*\{\s*\}(?!\s*else)/g
  115. , EMPTY_IF_WITH_ELSE = /if\s*\(([^)]+)\)\s*\{\s*\}\s*else(?!\s*if)/g;
  116. function cleanUpCode(out) {
  117. return out.replace(EMPTY_ELSE, '')
  118. .replace(EMPTY_IF_NO_ELSE, '')
  119. .replace(EMPTY_IF_WITH_ELSE, 'if (!($1))');
  120. }
  121. var ERRORS_REGEXP = /[^v\.]errors/g
  122. , REMOVE_ERRORS = /var errors = 0;|var vErrors = null;|validate.errors = vErrors;/g
  123. , REMOVE_ERRORS_ASYNC = /var errors = 0;|var vErrors = null;/g
  124. , RETURN_VALID = 'return errors === 0;'
  125. , RETURN_TRUE = 'validate.errors = null; return true;'
  126. , RETURN_ASYNC = /if \(errors === 0\) return data;\s*else throw new ValidationError\(vErrors\);/
  127. , RETURN_DATA_ASYNC = 'return data;'
  128. , ROOTDATA_REGEXP = /[^A-Za-z_$]rootData[^A-Za-z0-9_$]/g
  129. , REMOVE_ROOTDATA = /if \(rootData === undefined\) rootData = data;/;
  130. function finalCleanUpCode(out, async) {
  131. var matches = out.match(ERRORS_REGEXP);
  132. if (matches && matches.length == 2) {
  133. out = async
  134. ? out.replace(REMOVE_ERRORS_ASYNC, '')
  135. .replace(RETURN_ASYNC, RETURN_DATA_ASYNC)
  136. : out.replace(REMOVE_ERRORS, '')
  137. .replace(RETURN_VALID, RETURN_TRUE);
  138. }
  139. matches = out.match(ROOTDATA_REGEXP);
  140. if (!matches || matches.length !== 3) return out;
  141. return out.replace(REMOVE_ROOTDATA, '');
  142. }
  143. function schemaHasRules(schema, rules) {
  144. if (typeof schema == 'boolean') return !schema;
  145. for (var key in schema) if (rules[key]) return true;
  146. }
  147. function schemaHasRulesExcept(schema, rules, exceptKeyword) {
  148. if (typeof schema == 'boolean') return !schema && exceptKeyword != 'not';
  149. for (var key in schema) if (key != exceptKeyword && rules[key]) return true;
  150. }
  151. function toQuotedString(str) {
  152. return '\'' + escapeQuotes(str) + '\'';
  153. }
  154. function getPathExpr(currentPath, expr, jsonPointers, isNumber) {
  155. var path = jsonPointers // false by default
  156. ? '\'/\' + ' + expr + (isNumber ? '' : '.replace(/~/g, \'~0\').replace(/\\//g, \'~1\')')
  157. : (isNumber ? '\'[\' + ' + expr + ' + \']\'' : '\'[\\\'\' + ' + expr + ' + \'\\\']\'');
  158. return joinPaths(currentPath, path);
  159. }
  160. function getPath(currentPath, prop, jsonPointers) {
  161. var path = jsonPointers // false by default
  162. ? toQuotedString('/' + escapeJsonPointer(prop))
  163. : toQuotedString(getProperty(prop));
  164. return joinPaths(currentPath, path);
  165. }
  166. var JSON_POINTER = /^\/(?:[^~]|~0|~1)*$/;
  167. var RELATIVE_JSON_POINTER = /^([0-9]+)(#|\/(?:[^~]|~0|~1)*)?$/;
  168. function getData($data, lvl, paths) {
  169. var up, jsonPointer, data, matches;
  170. if ($data === '') return 'rootData';
  171. if ($data[0] == '/') {
  172. if (!JSON_POINTER.test($data)) throw new Error('Invalid JSON-pointer: ' + $data);
  173. jsonPointer = $data;
  174. data = 'rootData';
  175. } else {
  176. matches = $data.match(RELATIVE_JSON_POINTER);
  177. if (!matches) throw new Error('Invalid JSON-pointer: ' + $data);
  178. up = +matches[1];
  179. jsonPointer = matches[2];
  180. if (jsonPointer == '#') {
  181. if (up >= lvl) throw new Error('Cannot access property/index ' + up + ' levels up, current level is ' + lvl);
  182. return paths[lvl - up];
  183. }
  184. if (up > lvl) throw new Error('Cannot access data ' + up + ' levels up, current level is ' + lvl);
  185. data = 'data' + ((lvl - up) || '');
  186. if (!jsonPointer) return data;
  187. }
  188. var expr = data;
  189. var segments = jsonPointer.split('/');
  190. for (var i=0; i<segments.length; i++) {
  191. var segment = segments[i];
  192. if (segment) {
  193. data += getProperty(unescapeJsonPointer(segment));
  194. expr += ' && ' + data;
  195. }
  196. }
  197. return expr;
  198. }
  199. function joinPaths (a, b) {
  200. if (a == '""') return b;
  201. return (a + ' + ' + b).replace(/' \+ '/g, '');
  202. }
  203. function unescapeFragment(str) {
  204. return unescapeJsonPointer(decodeURIComponent(str));
  205. }
  206. function escapeFragment(str) {
  207. return encodeURIComponent(escapeJsonPointer(str));
  208. }
  209. function escapeJsonPointer(str) {
  210. return str.replace(/~/g, '~0').replace(/\//g, '~1');
  211. }
  212. function unescapeJsonPointer(str) {
  213. return str.replace(/~1/g, '/').replace(/~0/g, '~');
  214. }