stringify.js 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  1. 'use strict';
  2. var utils = require('./utils');
  3. var formats = require('./formats');
  4. var has = Object.prototype.hasOwnProperty;
  5. var arrayPrefixGenerators = {
  6. brackets: function brackets(prefix) {
  7. return prefix + '[]';
  8. },
  9. comma: 'comma',
  10. indices: function indices(prefix, key) {
  11. return prefix + '[' + key + ']';
  12. },
  13. repeat: function repeat(prefix) {
  14. return prefix;
  15. }
  16. };
  17. var isArray = Array.isArray;
  18. var split = String.prototype.split;
  19. var push = Array.prototype.push;
  20. var pushToArray = function (arr, valueOrArray) {
  21. push.apply(arr, isArray(valueOrArray) ? valueOrArray : [valueOrArray]);
  22. };
  23. var toISO = Date.prototype.toISOString;
  24. var defaultFormat = formats['default'];
  25. var defaults = {
  26. addQueryPrefix: false,
  27. allowDots: false,
  28. charset: 'utf-8',
  29. charsetSentinel: false,
  30. delimiter: '&',
  31. encode: true,
  32. encoder: utils.encode,
  33. encodeValuesOnly: false,
  34. format: defaultFormat,
  35. formatter: formats.formatters[defaultFormat],
  36. // deprecated
  37. indices: false,
  38. serializeDate: function serializeDate(date) {
  39. return toISO.call(date);
  40. },
  41. skipNulls: false,
  42. strictNullHandling: false
  43. };
  44. var isNonNullishPrimitive = function isNonNullishPrimitive(v) {
  45. return typeof v === 'string'
  46. || typeof v === 'number'
  47. || typeof v === 'boolean'
  48. || typeof v === 'symbol'
  49. || typeof v === 'bigint';
  50. };
  51. var stringify = function stringify(
  52. object,
  53. prefix,
  54. generateArrayPrefix,
  55. strictNullHandling,
  56. skipNulls,
  57. encoder,
  58. filter,
  59. sort,
  60. allowDots,
  61. serializeDate,
  62. format,
  63. formatter,
  64. encodeValuesOnly,
  65. charset
  66. ) {
  67. var obj = object;
  68. if (typeof filter === 'function') {
  69. obj = filter(prefix, obj);
  70. } else if (obj instanceof Date) {
  71. obj = serializeDate(obj);
  72. } else if (generateArrayPrefix === 'comma' && isArray(obj)) {
  73. obj = utils.maybeMap(obj, function (value) {
  74. if (value instanceof Date) {
  75. return serializeDate(value);
  76. }
  77. return value;
  78. });
  79. }
  80. if (obj === null) {
  81. if (strictNullHandling) {
  82. return encoder && !encodeValuesOnly ? encoder(prefix, defaults.encoder, charset, 'key', format) : prefix;
  83. }
  84. obj = '';
  85. }
  86. if (isNonNullishPrimitive(obj) || utils.isBuffer(obj)) {
  87. if (encoder) {
  88. var keyValue = encodeValuesOnly ? prefix : encoder(prefix, defaults.encoder, charset, 'key', format);
  89. if (generateArrayPrefix === 'comma' && encodeValuesOnly) {
  90. var valuesArray = split.call(String(obj), ',');
  91. var valuesJoined = '';
  92. for (var i = 0; i < valuesArray.length; ++i) {
  93. valuesJoined += (i === 0 ? '' : ',') + formatter(encoder(valuesArray[i], defaults.encoder, charset, 'value', format));
  94. }
  95. return [formatter(keyValue) + '=' + valuesJoined];
  96. }
  97. return [formatter(keyValue) + '=' + formatter(encoder(obj, defaults.encoder, charset, 'value', format))];
  98. }
  99. return [formatter(prefix) + '=' + formatter(String(obj))];
  100. }
  101. var values = [];
  102. if (typeof obj === 'undefined') {
  103. return values;
  104. }
  105. var objKeys;
  106. if (generateArrayPrefix === 'comma' && isArray(obj)) {
  107. // we need to join elements in
  108. objKeys = [{ value: obj.length > 0 ? obj.join(',') || null : void undefined }];
  109. } else if (isArray(filter)) {
  110. objKeys = filter;
  111. } else {
  112. var keys = Object.keys(obj);
  113. objKeys = sort ? keys.sort(sort) : keys;
  114. }
  115. for (var j = 0; j < objKeys.length; ++j) {
  116. var key = objKeys[j];
  117. var value = typeof key === 'object' && typeof key.value !== 'undefined' ? key.value : obj[key];
  118. if (skipNulls && value === null) {
  119. continue;
  120. }
  121. var keyPrefix = isArray(obj)
  122. ? typeof generateArrayPrefix === 'function' ? generateArrayPrefix(prefix, key) : prefix
  123. : prefix + (allowDots ? '.' + key : '[' + key + ']');
  124. pushToArray(values, stringify(
  125. value,
  126. keyPrefix,
  127. generateArrayPrefix,
  128. strictNullHandling,
  129. skipNulls,
  130. encoder,
  131. filter,
  132. sort,
  133. allowDots,
  134. serializeDate,
  135. format,
  136. formatter,
  137. encodeValuesOnly,
  138. charset
  139. ));
  140. }
  141. return values;
  142. };
  143. var normalizeStringifyOptions = function normalizeStringifyOptions(opts) {
  144. if (!opts) {
  145. return defaults;
  146. }
  147. if (opts.encoder !== null && typeof opts.encoder !== 'undefined' && typeof opts.encoder !== 'function') {
  148. throw new TypeError('Encoder has to be a function.');
  149. }
  150. var charset = opts.charset || defaults.charset;
  151. if (typeof opts.charset !== 'undefined' && opts.charset !== 'utf-8' && opts.charset !== 'iso-8859-1') {
  152. throw new TypeError('The charset option must be either utf-8, iso-8859-1, or undefined');
  153. }
  154. var format = formats['default'];
  155. if (typeof opts.format !== 'undefined') {
  156. if (!has.call(formats.formatters, opts.format)) {
  157. throw new TypeError('Unknown format option provided.');
  158. }
  159. format = opts.format;
  160. }
  161. var formatter = formats.formatters[format];
  162. var filter = defaults.filter;
  163. if (typeof opts.filter === 'function' || isArray(opts.filter)) {
  164. filter = opts.filter;
  165. }
  166. return {
  167. addQueryPrefix: typeof opts.addQueryPrefix === 'boolean' ? opts.addQueryPrefix : defaults.addQueryPrefix,
  168. allowDots: typeof opts.allowDots === 'undefined' ? defaults.allowDots : !!opts.allowDots,
  169. charset: charset,
  170. charsetSentinel: typeof opts.charsetSentinel === 'boolean' ? opts.charsetSentinel : defaults.charsetSentinel,
  171. delimiter: typeof opts.delimiter === 'undefined' ? defaults.delimiter : opts.delimiter,
  172. encode: typeof opts.encode === 'boolean' ? opts.encode : defaults.encode,
  173. encoder: typeof opts.encoder === 'function' ? opts.encoder : defaults.encoder,
  174. encodeValuesOnly: typeof opts.encodeValuesOnly === 'boolean' ? opts.encodeValuesOnly : defaults.encodeValuesOnly,
  175. filter: filter,
  176. format: format,
  177. formatter: formatter,
  178. serializeDate: typeof opts.serializeDate === 'function' ? opts.serializeDate : defaults.serializeDate,
  179. skipNulls: typeof opts.skipNulls === 'boolean' ? opts.skipNulls : defaults.skipNulls,
  180. sort: typeof opts.sort === 'function' ? opts.sort : null,
  181. strictNullHandling: typeof opts.strictNullHandling === 'boolean' ? opts.strictNullHandling : defaults.strictNullHandling
  182. };
  183. };
  184. module.exports = function (object, opts) {
  185. var obj = object;
  186. var options = normalizeStringifyOptions(opts);
  187. var objKeys;
  188. var filter;
  189. if (typeof options.filter === 'function') {
  190. filter = options.filter;
  191. obj = filter('', obj);
  192. } else if (isArray(options.filter)) {
  193. filter = options.filter;
  194. objKeys = filter;
  195. }
  196. var keys = [];
  197. if (typeof obj !== 'object' || obj === null) {
  198. return '';
  199. }
  200. var arrayFormat;
  201. if (opts && opts.arrayFormat in arrayPrefixGenerators) {
  202. arrayFormat = opts.arrayFormat;
  203. } else if (opts && 'indices' in opts) {
  204. arrayFormat = opts.indices ? 'indices' : 'repeat';
  205. } else {
  206. arrayFormat = 'indices';
  207. }
  208. var generateArrayPrefix = arrayPrefixGenerators[arrayFormat];
  209. if (!objKeys) {
  210. objKeys = Object.keys(obj);
  211. }
  212. if (options.sort) {
  213. objKeys.sort(options.sort);
  214. }
  215. for (var i = 0; i < objKeys.length; ++i) {
  216. var key = objKeys[i];
  217. if (options.skipNulls && obj[key] === null) {
  218. continue;
  219. }
  220. pushToArray(keys, stringify(
  221. obj[key],
  222. key,
  223. generateArrayPrefix,
  224. options.strictNullHandling,
  225. options.skipNulls,
  226. options.encode ? options.encoder : null,
  227. options.filter,
  228. options.sort,
  229. options.allowDots,
  230. options.serializeDate,
  231. options.format,
  232. options.formatter,
  233. options.encodeValuesOnly,
  234. options.charset
  235. ));
  236. }
  237. var joined = keys.join(options.delimiter);
  238. var prefix = options.addQueryPrefix === true ? '?' : '';
  239. if (options.charsetSentinel) {
  240. if (options.charset === 'iso-8859-1') {
  241. // encodeURIComponent('&#10003;'), the "numeric entity" representation of a checkmark
  242. prefix += 'utf8=%26%2310003%3B&';
  243. } else {
  244. // encodeURIComponent('✓')
  245. prefix += 'utf8=%E2%9C%93&';
  246. }
  247. }
  248. return joined.length > 0 ? prefix + joined : '';
  249. };