344 lines
10 KiB
JavaScript
344 lines
10 KiB
JavaScript
|
'use strict';
|
||
|
|
||
|
const printString = require('./printString');
|
||
|
|
||
|
const toString = Object.prototype.toString;
|
||
|
const toISOString = Date.prototype.toISOString;
|
||
|
const errorToString = Error.prototype.toString;
|
||
|
const regExpToString = RegExp.prototype.toString;
|
||
|
const symbolToString = Symbol.prototype.toString;
|
||
|
|
||
|
const SYMBOL_REGEXP = /^Symbol\((.*)\)(.*)$/;
|
||
|
const NEWLINE_REGEXP = /\n/ig;
|
||
|
|
||
|
const getSymbols = Object.getOwnPropertySymbols || (obj => []);
|
||
|
|
||
|
function isToStringedArrayType(toStringed) {
|
||
|
return (
|
||
|
toStringed === '[object Array]' ||
|
||
|
toStringed === '[object ArrayBuffer]' ||
|
||
|
toStringed === '[object DataView]' ||
|
||
|
toStringed === '[object Float32Array]' ||
|
||
|
toStringed === '[object Float64Array]' ||
|
||
|
toStringed === '[object Int8Array]' ||
|
||
|
toStringed === '[object Int16Array]' ||
|
||
|
toStringed === '[object Int32Array]' ||
|
||
|
toStringed === '[object Uint8Array]' ||
|
||
|
toStringed === '[object Uint8ClampedArray]' ||
|
||
|
toStringed === '[object Uint16Array]' ||
|
||
|
toStringed === '[object Uint32Array]'
|
||
|
);
|
||
|
}
|
||
|
|
||
|
function printNumber(val) {
|
||
|
if (val != +val) return 'NaN';
|
||
|
const isNegativeZero = val === 0 && (1 / val) < 0;
|
||
|
return isNegativeZero ? '-0' : '' + val;
|
||
|
}
|
||
|
|
||
|
function printFunction(val) {
|
||
|
if (val.name === '') {
|
||
|
return '[Function anonymous]'
|
||
|
} else {
|
||
|
return '[Function ' + val.name + ']';
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function printSymbol(val) {
|
||
|
return symbolToString.call(val).replace(SYMBOL_REGEXP, 'Symbol($1)');
|
||
|
}
|
||
|
|
||
|
function printError(val) {
|
||
|
return '[' + errorToString.call(val) + ']';
|
||
|
}
|
||
|
|
||
|
function printBasicValue(val) {
|
||
|
if (val === true || val === false) return '' + val;
|
||
|
if (val === undefined) return 'undefined';
|
||
|
if (val === null) return 'null';
|
||
|
|
||
|
const typeOf = typeof val;
|
||
|
|
||
|
if (typeOf === 'number') return printNumber(val);
|
||
|
if (typeOf === 'string') return '"' + printString(val) + '"';
|
||
|
if (typeOf === 'function') return printFunction(val);
|
||
|
if (typeOf === 'symbol') return printSymbol(val);
|
||
|
|
||
|
const toStringed = toString.call(val);
|
||
|
|
||
|
if (toStringed === '[object WeakMap]') return 'WeakMap {}';
|
||
|
if (toStringed === '[object WeakSet]') return 'WeakSet {}';
|
||
|
if (toStringed === '[object Function]' || toStringed === '[object GeneratorFunction]') return printFunction(val, min);
|
||
|
if (toStringed === '[object Symbol]') return printSymbol(val);
|
||
|
if (toStringed === '[object Date]') return toISOString.call(val);
|
||
|
if (toStringed === '[object Error]') return printError(val);
|
||
|
if (toStringed === '[object RegExp]') return regExpToString.call(val);
|
||
|
if (toStringed === '[object Arguments]' && val.length === 0) return 'Arguments []';
|
||
|
if (isToStringedArrayType(toStringed) && val.length === 0) return val.constructor.name + ' []';
|
||
|
|
||
|
if (val instanceof Error) return printError(val);
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
function printList(list, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min) {
|
||
|
let body = '';
|
||
|
|
||
|
if (list.length) {
|
||
|
body += edgeSpacing;
|
||
|
|
||
|
const innerIndent = prevIndent + indent;
|
||
|
|
||
|
for (let i = 0; i < list.length; i++) {
|
||
|
body += innerIndent + print(list[i], indent, innerIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
|
||
|
if (i < list.length - 1) {
|
||
|
body += ',' + spacing;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
body += edgeSpacing + prevIndent;
|
||
|
}
|
||
|
|
||
|
return '[' + body + ']';
|
||
|
}
|
||
|
|
||
|
function printArguments(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min) {
|
||
|
return (min ? '' : 'Arguments ') + printList(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
}
|
||
|
|
||
|
function printArray(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min) {
|
||
|
return (min ? '' : val.constructor.name + ' ') + printList(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
}
|
||
|
|
||
|
function printMap(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min) {
|
||
|
let result = 'Map {';
|
||
|
const iterator = val.entries();
|
||
|
let current = iterator.next();
|
||
|
|
||
|
if (!current.done) {
|
||
|
result += edgeSpacing;
|
||
|
|
||
|
const innerIndent = prevIndent + indent;
|
||
|
|
||
|
while (!current.done) {
|
||
|
const key = print(current.value[0], indent, innerIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
const value = print(current.value[1], indent, innerIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
|
||
|
result += innerIndent + key + ' => ' + value;
|
||
|
|
||
|
current = iterator.next();
|
||
|
|
||
|
if (!current.done) {
|
||
|
result += ',' + spacing;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
result += edgeSpacing + prevIndent;
|
||
|
}
|
||
|
|
||
|
return result + '}';
|
||
|
}
|
||
|
|
||
|
function printObject(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min) {
|
||
|
const constructor = min ? '' : (val.constructor ? val.constructor.name + ' ' : 'Object ');
|
||
|
let result = constructor + '{';
|
||
|
let keys = Object.keys(val).sort();
|
||
|
const symbols = getSymbols(val);
|
||
|
|
||
|
if (symbols.length) {
|
||
|
keys = keys
|
||
|
.filter(key => !(typeof key === 'symbol' || toString.call(key) === '[object Symbol]'))
|
||
|
.concat(symbols);
|
||
|
}
|
||
|
|
||
|
if (keys.length) {
|
||
|
result += edgeSpacing;
|
||
|
|
||
|
const innerIndent = prevIndent + indent;
|
||
|
|
||
|
for (let i = 0; i < keys.length; i++) {
|
||
|
const key = keys[i];
|
||
|
const name = print(key, indent, innerIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
const value = print(val[key], indent, innerIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
|
||
|
result += innerIndent + name + ': ' + value;
|
||
|
|
||
|
if (i < keys.length - 1) {
|
||
|
result += ',' + spacing;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
result += edgeSpacing + prevIndent;
|
||
|
}
|
||
|
|
||
|
return result + '}';
|
||
|
}
|
||
|
|
||
|
function printSet(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min) {
|
||
|
let result = 'Set {';
|
||
|
const iterator = val.entries();
|
||
|
let current = iterator.next();
|
||
|
|
||
|
if (!current.done) {
|
||
|
result += edgeSpacing;
|
||
|
|
||
|
const innerIndent = prevIndent + indent;
|
||
|
|
||
|
while (!current.done) {
|
||
|
result += innerIndent + print(current.value[1], indent, innerIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
|
||
|
current = iterator.next();
|
||
|
|
||
|
if (!current.done) {
|
||
|
result += ',' + spacing;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
result += edgeSpacing + prevIndent;
|
||
|
}
|
||
|
|
||
|
return result + '}';
|
||
|
}
|
||
|
|
||
|
function printComplexValue(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min) {
|
||
|
refs = refs.slice();
|
||
|
if (refs.indexOf(val) > -1) {
|
||
|
return '[Circular]';
|
||
|
} else {
|
||
|
refs.push(val);
|
||
|
}
|
||
|
|
||
|
currentDepth++;
|
||
|
|
||
|
const hitMaxDepth = currentDepth > maxDepth;
|
||
|
|
||
|
if (!hitMaxDepth && val.toJSON && typeof val.toJSON === 'function') {
|
||
|
return print(val.toJSON(), indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
}
|
||
|
|
||
|
const toStringed = toString.call(val);
|
||
|
if (toStringed === '[object Arguments]') {
|
||
|
return hitMaxDepth ? '[Arguments]' : printArguments(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
} else if (isToStringedArrayType(toStringed)) {
|
||
|
return hitMaxDepth ? '[Array]' : printArray(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
} else if (toStringed === '[object Map]') {
|
||
|
return hitMaxDepth ? '[Map]' : printMap(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
} else if (toStringed === '[object Set]') {
|
||
|
return hitMaxDepth ? '[Set]' : printSet(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
} else if (typeof val === 'object') {
|
||
|
return hitMaxDepth ? '[Object]' : printObject(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function printPlugin(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min) {
|
||
|
let match = false;
|
||
|
let plugin;
|
||
|
|
||
|
for (let p = 0; p < plugins.length; p++) {
|
||
|
plugin = plugins[p];
|
||
|
|
||
|
if (plugin.test(val)) {
|
||
|
match = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (!match) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
function boundPrint(val) {
|
||
|
return print(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
}
|
||
|
|
||
|
function boundIndent(str) {
|
||
|
const indentation = prevIndent + indent;
|
||
|
return indentation + str.replace(NEWLINE_REGEXP, '\n' + indentation);
|
||
|
}
|
||
|
|
||
|
return plugin.print(val, boundPrint, boundIndent, {
|
||
|
edgeSpacing: edgeSpacing,
|
||
|
spacing: spacing
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function print(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min) {
|
||
|
const basic = printBasicValue(val);
|
||
|
if (basic) return basic;
|
||
|
|
||
|
const plugin = printPlugin(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
if (plugin) return plugin;
|
||
|
|
||
|
return printComplexValue(val, indent, prevIndent, spacing, edgeSpacing, refs, maxDepth, currentDepth, plugins, min);
|
||
|
}
|
||
|
|
||
|
const DEFAULTS = {
|
||
|
indent: 2,
|
||
|
min: false,
|
||
|
maxDepth: Infinity,
|
||
|
plugins: []
|
||
|
};
|
||
|
|
||
|
function validateOptions(opts) {
|
||
|
Object.keys(opts).forEach(key => {
|
||
|
if (!DEFAULTS.hasOwnProperty(key)) {
|
||
|
throw new Error('prettyFormat: Invalid option: ' + key);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
if (opts.min && opts.indent !== undefined && opts.indent !== 0) {
|
||
|
throw new Error('prettyFormat: Cannot run with min option and indent');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function normalizeOptions(opts) {
|
||
|
const result = {};
|
||
|
|
||
|
Object.keys(DEFAULTS).forEach(key =>
|
||
|
result[key] = opts.hasOwnProperty(key) ? opts[key] : DEFAULTS[key]
|
||
|
);
|
||
|
|
||
|
if (result.min) {
|
||
|
result.indent = 0;
|
||
|
}
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
function createIndent(indent) {
|
||
|
return new Array(indent + 1).join(' ');
|
||
|
}
|
||
|
|
||
|
function prettyFormat(val, opts) {
|
||
|
if (!opts) {
|
||
|
opts = DEFAULTS;
|
||
|
} else {
|
||
|
validateOptions(opts)
|
||
|
opts = normalizeOptions(opts);
|
||
|
}
|
||
|
|
||
|
let indent;
|
||
|
let refs;
|
||
|
const prevIndent = '';
|
||
|
const currentDepth = 0;
|
||
|
const spacing = opts.min ? ' ' : '\n';
|
||
|
const edgeSpacing = opts.min ? '' : '\n';
|
||
|
|
||
|
if (opts && opts.plugins.length) {
|
||
|
indent = createIndent(opts.indent);
|
||
|
refs = [];
|
||
|
var pluginsResult = printPlugin(val, indent, prevIndent, spacing, edgeSpacing, refs, opts.maxDepth, currentDepth, opts.plugins, opts.min);
|
||
|
if (pluginsResult) return pluginsResult;
|
||
|
}
|
||
|
|
||
|
var basicResult = printBasicValue(val);
|
||
|
if (basicResult) return basicResult;
|
||
|
|
||
|
if (!indent) indent = createIndent(opts.indent);
|
||
|
if (!refs) refs = [];
|
||
|
return printComplexValue(val, indent, prevIndent, spacing, edgeSpacing, refs, opts.maxDepth, currentDepth, opts.plugins, opts.min);
|
||
|
}
|
||
|
|
||
|
module.exports = prettyFormat;
|