[d24f17c] | 1 | 'use strict';
|
---|
| 2 |
|
---|
| 3 | var jsonStringify = (typeof JSON !== 'undefined' ? JSON : require('jsonify')).stringify;
|
---|
| 4 |
|
---|
| 5 | var isArray = require('isarray');
|
---|
| 6 | var objectKeys = require('object-keys');
|
---|
| 7 | var callBind = require('call-bind');
|
---|
| 8 | var callBound = require('call-bind/callBound');
|
---|
| 9 |
|
---|
| 10 | var $join = callBound('Array.prototype.join');
|
---|
| 11 | var $push = callBound('Array.prototype.push');
|
---|
| 12 |
|
---|
| 13 | var strRepeat = function repeat(n, char) {
|
---|
| 14 | var str = '';
|
---|
| 15 | for (var i = 0; i < n; i += 1) {
|
---|
| 16 | str += char;
|
---|
| 17 | }
|
---|
| 18 | return str;
|
---|
| 19 | };
|
---|
| 20 |
|
---|
| 21 | var defaultReplacer = function (parent, key, value) { return value; };
|
---|
| 22 |
|
---|
| 23 | module.exports = function stableStringify(obj) {
|
---|
| 24 | var opts = arguments.length > 1 ? arguments[1] : void undefined;
|
---|
| 25 | var space = (opts && opts.space) || '';
|
---|
| 26 | if (typeof space === 'number') { space = strRepeat(space, ' '); }
|
---|
| 27 | var cycles = !!opts && typeof opts.cycles === 'boolean' && opts.cycles;
|
---|
| 28 | var replacer = opts && opts.replacer ? callBind(opts.replacer) : defaultReplacer;
|
---|
| 29 |
|
---|
| 30 | var cmpOpt = typeof opts === 'function' ? opts : opts && opts.cmp;
|
---|
| 31 | var cmp = cmpOpt && function (node) {
|
---|
| 32 | var get = cmpOpt.length > 2 && function get(k) { return node[k]; };
|
---|
| 33 | return function (a, b) {
|
---|
| 34 | return cmpOpt(
|
---|
| 35 | { key: a, value: node[a] },
|
---|
| 36 | { key: b, value: node[b] },
|
---|
| 37 | get ? { __proto__: null, get: get } : void undefined
|
---|
| 38 | );
|
---|
| 39 | };
|
---|
| 40 | };
|
---|
| 41 |
|
---|
| 42 | var seen = [];
|
---|
| 43 | return (function stringify(parent, key, node, level) {
|
---|
| 44 | var indent = space ? '\n' + strRepeat(level, space) : '';
|
---|
| 45 | var colonSeparator = space ? ': ' : ':';
|
---|
| 46 |
|
---|
| 47 | if (node && node.toJSON && typeof node.toJSON === 'function') {
|
---|
| 48 | node = node.toJSON();
|
---|
| 49 | }
|
---|
| 50 |
|
---|
| 51 | node = replacer(parent, key, node);
|
---|
| 52 |
|
---|
| 53 | if (node === undefined) {
|
---|
| 54 | return;
|
---|
| 55 | }
|
---|
| 56 | if (typeof node !== 'object' || node === null) {
|
---|
| 57 | return jsonStringify(node);
|
---|
| 58 | }
|
---|
| 59 | if (isArray(node)) {
|
---|
| 60 | var out = [];
|
---|
| 61 | for (var i = 0; i < node.length; i++) {
|
---|
| 62 | var item = stringify(node, i, node[i], level + 1) || jsonStringify(null);
|
---|
| 63 | $push(out, indent + space + item);
|
---|
| 64 | }
|
---|
| 65 | return '[' + $join(out, ',') + indent + ']';
|
---|
| 66 | }
|
---|
| 67 |
|
---|
| 68 | if (seen.indexOf(node) !== -1) {
|
---|
| 69 | if (cycles) { return jsonStringify('__cycle__'); }
|
---|
| 70 | throw new TypeError('Converting circular structure to JSON');
|
---|
| 71 | } else { $push(seen, node); }
|
---|
| 72 |
|
---|
| 73 | var keys = objectKeys(node).sort(cmp && cmp(node));
|
---|
| 74 | var out = [];
|
---|
| 75 | for (var i = 0; i < keys.length; i++) {
|
---|
| 76 | var key = keys[i];
|
---|
| 77 | var value = stringify(node, key, node[key], level + 1);
|
---|
| 78 |
|
---|
| 79 | if (!value) { continue; }
|
---|
| 80 |
|
---|
| 81 | var keyValue = jsonStringify(key)
|
---|
| 82 | + colonSeparator
|
---|
| 83 | + value;
|
---|
| 84 |
|
---|
| 85 | $push(out, indent + space + keyValue);
|
---|
| 86 | }
|
---|
| 87 | seen.splice(seen.indexOf(node), 1);
|
---|
| 88 | return '{' + $join(out, ',') + indent + '}';
|
---|
| 89 |
|
---|
| 90 | }({ '': obj }, '', obj, 0));
|
---|
| 91 | };
|
---|