UNPKG

string-kit

Version:

A string manipulation toolbox, featuring a string formatter (inspired by sprintf), a variable inspector (output featuring ANSI colors and HTML) and various escape functions (shell argument, regexp, html, etc).

761 lines (626 loc) 26.1 kB
/* String Kit Copyright (c) 2014 - 2021 Cédric Ronvel The MIT License (MIT) Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ /* Variable inspector. */ "use strict" ; const escape = require( './escape.js' ) ; const ansi = require( './ansi.js' ) ; const EMPTY = {} ; const TRIVIAL_CONSTRUCTOR = new Set( [ Object , Array ] ) ; /* Inspect a variable, return a string ready to be displayed with console.log(), or even as an HTML output. Options: * style: * 'none': (default) normal output suitable for console.log() or writing in a file * 'inline': like 'none', but without newlines * 'color': colorful output suitable for terminal * 'html': html output * any object: full controle, inheriting from 'none' * tab: `string` override the tab of the style * depth: depth limit, default: 3 * maxLength: length limit for strings, default: 250 * outputMaxLength: length limit for the inspect output string, default: 5000 * noFunc: do not display functions * noDescriptor: do not display descriptor information * noArrayProperty: do not display array properties * noIndex: do not display array indexes * bulletIndex: do not display array indexes, instead display a bullet: * * noType: do not display type and constructor * noTypeButConstructor: do not display type, display non-trivial constructor (not Object or Array, but all others) * enumOnly: only display enumerable properties * funcDetails: display function's details * proto: display object's prototype * sort: sort the keys * noMarkup: don't add Javascript/JSON markup: {}[]," * minimal: imply noFunc: true, noDescriptor: true, noType: true, noArrayProperty: true, enumOnly: true, proto: false and funcDetails: false. Display a minimal JSON-like output * minimalPlusConstructor: like minimal, but output non-trivial constructor * protoBlackList: `Set` of blacklisted object prototype (will not recurse inside it) * propertyBlackList: `Set` of blacklisted property names (will not even display it) * useInspect: use .inspect() method when available on an object (default to false) * useInspectPropertyBlackList: if set and if the object to be inspected has an 'inspectPropertyBlackList' property which value is a `Set`, use it like the 'propertyBlackList' option */ function inspect( options , variable ) { if ( arguments.length < 2 ) { variable = options ; options = {} ; } else if ( ! options || typeof options !== 'object' ) { options = {} ; } var runtime = { depth: 0 , ancestors: [] } ; if ( ! options.style ) { options.style = inspectStyle.none ; } else if ( typeof options.style === 'string' ) { options.style = inspectStyle[ options.style ] ; } // Too slow: //else { options.style = Object.assign( {} , inspectStyle.none , options.style ) ; } if ( options.depth === undefined ) { options.depth = 3 ; } if ( options.maxLength === undefined ) { options.maxLength = 250 ; } if ( options.outputMaxLength === undefined ) { options.outputMaxLength = 5000 ; } // /!\ nofunc is deprecated if ( options.nofunc ) { options.noFunc = true ; } if ( options.minimal ) { options.noFunc = true ; options.noDescriptor = true ; options.noType = true ; options.noArrayProperty = true ; options.enumOnly = true ; options.proto = false ; options.funcDetails = false ; } if ( options.minimalPlusConstructor ) { options.noFunc = true ; options.noDescriptor = true ; options.noTypeButConstructor = true ; options.noArrayProperty = true ; options.enumOnly = true ; options.proto = false ; options.funcDetails = false ; } var str = inspect_( runtime , options , variable ) ; if ( str.length > options.outputMaxLength ) { str = options.style.truncate( str , options.outputMaxLength ) ; } return str ; } exports.inspect = inspect ; function inspect_( runtime , options , variable ) { var i , funcName , length , proto , propertyList , isTrivialConstructor , constructor , keyIsProperty , type , pre , isArray , isFunc , specialObject , str = '' , key = '' , descriptorStr = '' , indent = '' , descriptor , nextAncestors ; // Prepare things (indentation, key, descriptor, ... ) type = typeof variable ; if ( runtime.depth ) { indent = ( options.tab ?? options.style.tab ).repeat( options.noMarkup ? runtime.depth - 1 : runtime.depth ) ; } if ( type === 'function' && options.noFunc ) { return '' ; } if ( runtime.key !== undefined ) { if ( runtime.descriptor ) { descriptorStr = [] ; if ( runtime.descriptor.error ) { descriptorStr = '[' + runtime.descriptor.error + ']' ; } else { if ( ! runtime.descriptor.configurable ) { descriptorStr.push( '-conf' ) ; } if ( ! runtime.descriptor.enumerable ) { descriptorStr.push( '-enum' ) ; } // Already displayed by runtime.forceType //if ( runtime.descriptor.get || runtime.descriptor.set ) { descriptorStr.push( 'getter/setter' ) ; } else if ( ! runtime.descriptor.writable ) { descriptorStr.push( '-w' ) ; } //if ( descriptorStr.length ) { descriptorStr = '(' + descriptorStr.join( ' ' ) + ')' ; } if ( descriptorStr.length ) { descriptorStr = descriptorStr.join( ' ' ) ; } else { descriptorStr = '' ; } } } if ( runtime.keyIsProperty ) { if ( ! options.noMarkup && keyNeedingQuotes( runtime.key ) ) { key = '"' + options.style.key( runtime.key ) + '": ' ; } else { key = options.style.key( runtime.key ) + ': ' ; } } else if ( options.bulletIndex ) { key = ( typeof options.bulletIndex === 'string' ? options.bulletIndex : '*' ) + ' ' ; } else if ( ! options.noIndex ) { key = options.style.index( runtime.key ) ; } if ( descriptorStr ) { descriptorStr = ' ' + options.style.type( descriptorStr ) ; } } pre = runtime.noPre ? '' : indent + key ; // Describe the current variable if ( variable === undefined ) { str += pre + options.style.constant( 'undefined' ) + descriptorStr + options.style.newline ; } else if ( variable === EMPTY ) { str += pre + options.style.constant( '[empty]' ) + descriptorStr + options.style.newline ; } else if ( variable === null ) { str += pre + options.style.constant( 'null' ) + descriptorStr + options.style.newline ; } else if ( variable === false ) { str += pre + options.style.constant( 'false' ) + descriptorStr + options.style.newline ; } else if ( variable === true ) { str += pre + options.style.constant( 'true' ) + descriptorStr + options.style.newline ; } else if ( type === 'number' ) { str += pre + options.style.number( variable.toString() ) + ( options.noType || options.noTypeButConstructor ? '' : ' ' + options.style.type( 'number' ) ) + descriptorStr + options.style.newline ; } else if ( type === 'string' ) { if ( variable.length > options.maxLength ) { str += pre + ( options.noMarkup ? '' : '"' ) + options.style.string( escape.control( variable.slice( 0 , options.maxLength - 1 ) ) ) + '…' + ( options.noMarkup ? '' : '"' ) + ( options.noType || options.noTypeButConstructor ? '' : ' ' + options.style.type( 'string' ) + options.style.length( '(' + variable.length + ' - TRUNCATED)' ) ) + descriptorStr + options.style.newline ; } else { str += pre + ( options.noMarkup ? '' : '"' ) + options.style.string( escape.control( variable ) ) + ( options.noMarkup ? '' : '"' ) + ( options.noType || options.noTypeButConstructor ? '' : ' ' + options.style.type( 'string' ) + options.style.length( '(' + variable.length + ')' ) ) + descriptorStr + options.style.newline ; } } else if ( Buffer.isBuffer( variable ) ) { str += pre + options.style.inspect( variable.inspect() ) + ( options.noType ? '' : ' ' + options.style.type( 'Buffer' ) + options.style.length( '(' + variable.length + ')' ) ) + descriptorStr + options.style.newline ; } else if ( type === 'object' || type === 'function' ) { funcName = length = '' ; isFunc = false ; if ( type === 'function' ) { isFunc = true ; funcName = ' ' + options.style.funcName( ( variable.name ? variable.name : '(anonymous)' ) ) ; length = options.style.length( '(' + variable.length + ')' ) ; } isArray = false ; if ( Array.isArray( variable ) ) { isArray = true ; length = options.style.length( '(' + variable.length + ')' ) ; } if ( ! variable.constructor ) { constructor = '(no constructor)' ; } else if ( ! variable.constructor.name ) { constructor = '(anonymous)' ; } else { constructor = variable.constructor.name ; } isTrivialConstructor = ! variable.constructor || TRIVIAL_CONSTRUCTOR.has( variable.constructor ) ; constructor = options.style.constructorName( constructor ) ; proto = Object.getPrototypeOf( variable ) ; str += pre ; if ( ! options.noType && ( ! options.noTypeButConstructor || ! isTrivialConstructor ) ) { if ( runtime.forceType && ! options.noType && ! options.noTypeButConstructor ) { str += options.style.type( runtime.forceType ) ; } else if ( options.noTypeButConstructor ) { str += constructor ; } else { str += constructor + funcName + length + ' ' + options.style.type( type ) + descriptorStr ; } if ( ! isFunc || options.funcDetails ) { str += ' ' ; } // if no funcDetails imply no space here } if ( isArray && options.noArrayProperty ) { propertyList = [ ... Array( variable.length ).keys() ] ; } else { propertyList = Object.getOwnPropertyNames( variable ) ; } if ( options.sort ) { propertyList.sort() ; } // Special Objects specialObject = specialObjectSubstitution( variable , runtime , options ) ; if ( options.protoBlackList && options.protoBlackList.has( proto ) ) { str += options.style.limit( '[skip]' ) + options.style.newline ; } else if ( specialObject !== undefined ) { if ( typeof specialObject === 'string' ) { str += '=> ' + specialObject + options.style.newline ; } else { str += '=> ' + inspect_( { depth: runtime.depth , ancestors: runtime.ancestors , noPre: true } , options , specialObject ) ; } } else if ( isFunc && ! options.funcDetails ) { str += options.style.newline ; } else if ( ! propertyList.length && ! options.proto ) { str += ( options.noMarkup ? '' : isArray ? '[]' : '{}' ) + options.style.newline ; } else if ( runtime.depth >= options.depth ) { str += options.style.limit( '[depth limit]' ) + options.style.newline ; } else if ( runtime.ancestors.indexOf( variable ) !== -1 ) { str += options.style.limit( '[circular]' ) + options.style.newline ; } else { /* str += options.noMarkup ? ( isArray && options.noIndex && ! runtime.keyIsProperty ? '' : options.style.newline ) : ( isArray ? '[' : '{' ) + options.style.newline ; //*/ //* str += ( options.noMarkup ? '' : isArray ? '[' : '{' ) + options.style.newline ; //*/ // Do not use .concat() here, it doesn't works as expected with arrays... nextAncestors = runtime.ancestors.slice() ; nextAncestors.push( variable ) ; for ( i = 0 ; i < propertyList.length && str.length < options.outputMaxLength ; i ++ ) { if ( ! isArray && ( ( options.propertyBlackList && options.propertyBlackList.has( propertyList[ i ] ) ) || ( options.useInspectPropertyBlackList && ( variable.inspectPropertyBlackList instanceof Set ) && variable.inspectPropertyBlackList.has( propertyList[ i ] ) ) ) ) { //str += options.style.limit( '[skip]' ) + options.style.newline ; continue ; } if ( isArray && options.noArrayProperty && ! ( propertyList[ i ] in variable ) ) { // Hole in the array (sparse array, item deleted, ...) str += inspect_( { depth: runtime.depth + 1 , ancestors: nextAncestors , key: propertyList[ i ] , keyIsProperty: false } , options , EMPTY ) ; } else { try { descriptor = Object.getOwnPropertyDescriptor( variable , propertyList[ i ] ) ; // Note: descriptor can be undefined, this happens when the object is a Proxy with a bad implementation: // it reports that key (Object.keys()) but doesn't give the descriptor for it. if ( descriptor && ! descriptor.enumerable && options.enumOnly ) { continue ; } keyIsProperty = ! isArray || ! descriptor.enumerable || isNaN( propertyList[ i ] ) ; if ( ! options.noDescriptor && descriptor && ( descriptor.get || descriptor.set ) ) { str += inspect_( { depth: runtime.depth + 1 , ancestors: nextAncestors , key: propertyList[ i ] , keyIsProperty: keyIsProperty , descriptor: descriptor , forceType: 'getter/setter' } , options , { get: descriptor.get , set: descriptor.set } ) ; } else { str += inspect_( { depth: runtime.depth + 1 , ancestors: nextAncestors , key: propertyList[ i ] , keyIsProperty: keyIsProperty , descriptor: options.noDescriptor ? undefined : descriptor || { error: "Bad Proxy Descriptor" } } , options , variable[ propertyList[ i ] ] ) ; } } catch ( error ) { str += inspect_( { depth: runtime.depth + 1 , ancestors: nextAncestors , key: propertyList[ i ] , keyIsProperty: keyIsProperty , descriptor: options.noDescriptor ? undefined : descriptor } , options , error ) ; } } if ( i < propertyList.length - 1 ) { str += options.style.comma ; } } if ( options.proto ) { str += inspect_( { depth: runtime.depth + 1 , ancestors: nextAncestors , key: '__proto__' , keyIsProperty: true } , options , proto ) ; } str += options.noMarkup ? '' : indent + ( isArray ? ']' : '}' ) + options.style.newline ; } } // Finalizing if ( runtime.depth === 0 ) { if ( options.style.trim ) { str = str.trim() ; } if ( options.style === 'html' ) { str = escape.html( str ) ; } } return str ; } function keyNeedingQuotes( key ) { if ( ! key.length ) { return true ; } return false ; } var promiseStates = [ 'pending' , 'fulfilled' , 'rejected' ] ; // Some special object are better written down when substituted by something else function specialObjectSubstitution( object , runtime , options ) { if ( typeof object.constructor !== 'function' ) { // Some objects have no constructor, e.g.: Object.create(null) //console.error( object ) ; return ; } if ( object instanceof String ) { return object.toString() ; } if ( object instanceof RegExp ) { return object.toString() ; } if ( object instanceof Date ) { return object.toString() + ' [' + object.getTime() + ']' ; } if ( typeof Set === 'function' && object instanceof Set ) { // This is an ES6 'Set' Object return Array.from( object ) ; } if ( typeof Map === 'function' && object instanceof Map ) { // This is an ES6 'Map' Object return Array.from( object ) ; } if ( object instanceof Promise ) { if ( process && process.binding && process.binding( 'util' ) && process.binding( 'util' ).getPromiseDetails ) { let details = process.binding( 'util' ).getPromiseDetails( object ) ; let state = promiseStates[ details[ 0 ] ] ; let str = 'Promise <' + state + '>' ; if ( state === 'fulfilled' ) { str += ' ' + inspect_( { depth: runtime.depth , ancestors: runtime.ancestors , noPre: true } , options , details[ 1 ] ) ; } else if ( state === 'rejected' ) { if ( details[ 1 ] instanceof Error ) { str += ' ' + inspectError( { style: options.style , noErrorStack: true } , details[ 1 ] ) ; } else { str += ' ' + inspect_( { depth: runtime.depth , ancestors: runtime.ancestors , noPre: true } , options , details[ 1 ] ) ; } } return str ; } } if ( object._bsontype ) { // This is a MongoDB ObjectID, rather boring to display in its original form // due to esoteric characters that confuse both the user and the terminal displaying it. // Substitute it to its string representation return object.toString() ; } if ( options.useInspect && typeof object.inspect === 'function' ) { return object.inspect() ; } return ; } /* Options: noErrorStack: set to true if the stack should not be displayed */ function inspectError( options , error ) { var str = '' , stack , type , code ; if ( arguments.length < 2 ) { error = options ; options = {} ; } else if ( ! options || typeof options !== 'object' ) { options = {} ; } if ( ! options.style ) { options.style = inspectStyle.none ; } else if ( typeof options.style === 'string' ) { options.style = inspectStyle[ options.style ] ; } if ( ! ( error instanceof Error ) ) { str += '[not an Error] ' ; if ( typeof error === 'string' ) { let maxLength = 5000 ; if ( error.length > maxLength ) { str += options.style.errorMessage( escape.control( error.slice( 0 , maxLength - 1 ) , true ) ) + '…' + options.style.length( '(' + error.length + ' - TRUNCATED)' ) + options.style.newline ; } else { str += options.style.errorMessage( escape.control( error , true ) ) + options.style.newline ; } return str ; } else if ( ! error || typeof error !== 'object' || ! error.name || typeof error.name !== 'string' || ! error.message || typeof error.message !== 'string' ) { str += inspect( options , error ) ; return str ; } // It's an object, but it's compatible with Error, so we can move on... } if ( error.stack && ! options.noErrorStack ) { stack = inspectStack( options , error.stack ) ; } type = error.type || error.constructor.name ; code = error.code || error.name || error.errno || error.number ; str += options.style.errorType( type ) + ( code ? ' [' + options.style.errorType( code ) + ']' : '' ) + ': ' ; str += options.style.errorMessage( error.message ) + '\n' ; if ( stack ) { str += options.style.errorStack( stack ) + '\n' ; } if ( error.from ) { str += options.style.newline + options.style.errorFromMessage( 'From error:' ) + options.style.newline + inspectError( options , error.from ) ; } return str ; } exports.inspectError = inspectError ; function inspectStack( options , stack ) { if ( arguments.length < 2 ) { stack = options ; options = {} ; } else if ( ! options || typeof options !== 'object' ) { options = {} ; } if ( ! options.style ) { options.style = inspectStyle.none ; } else if ( typeof options.style === 'string' ) { options.style = inspectStyle[ options.style ] ; } if ( ! stack ) { return ; } if ( ( options.browser || process.browser ) && stack.indexOf( '@' ) !== -1 ) { // Assume a Firefox-compatible stack-trace here... stack = stack .replace( /[</]*(?=@)/g , '' ) // Firefox output some WTF </</</</< stuff in its stack trace -- removing that .replace( /^\s*([^@]*)\s*@\s*([^\n]*)(?::([0-9]+):([0-9]+))?$/mg , ( matches , method , file , line , column ) => { return options.style.errorStack( ' at ' ) + ( method ? options.style.errorStackMethod( method ) + ' ' : '' ) + options.style.errorStack( '(' ) + ( file ? options.style.errorStackFile( file ) : options.style.errorStack( 'unknown' ) ) + ( line ? options.style.errorStack( ':' ) + options.style.errorStackLine( line ) : '' ) + ( column ? options.style.errorStack( ':' ) + options.style.errorStackColumn( column ) : '' ) + options.style.errorStack( ')' ) ; } ) ; } else { stack = stack.replace( /^[^\n]*\n/ , '' ) ; stack = stack.replace( /^\s*(at)\s+(?:(?:(async|new)\s+)?([^\s:()[\]\n]+(?:\([^)]+\))?)\s)?(?:\[as ([^\s:()[\]\n]+)\]\s)?(?:\(?([^:()[\]\n]+):([0-9]+):([0-9]+)\)?)?$/mg , ( matches , at , keyword , method , as , file , line , column ) => { return options.style.errorStack( ' at ' ) + ( keyword ? options.style.errorStackKeyword( keyword ) + ' ' : '' ) + ( method ? options.style.errorStackMethod( method ) + ' ' : '' ) + ( as ? options.style.errorStack( '[as ' ) + options.style.errorStackMethodAs( as ) + options.style.errorStack( '] ' ) : '' ) + options.style.errorStack( '(' ) + ( file ? options.style.errorStackFile( file ) : options.style.errorStack( 'unknown' ) ) + ( line ? options.style.errorStack( ':' ) + options.style.errorStackLine( line ) : '' ) + ( column ? options.style.errorStack( ':' ) + options.style.errorStackColumn( column ) : '' ) + options.style.errorStack( ')' ) ; } ) ; } return stack ; } exports.inspectStack = inspectStack ; // Inspect's styles var inspectStyle = {} ; var inspectStyleNoop = str => str ; inspectStyle.none = { trim: false , tab: ' ' , newline: '\n' , comma: '' , limit: inspectStyleNoop , type: str => '<' + str + '>' , constant: inspectStyleNoop , funcName: inspectStyleNoop , constructorName: str => '<' + str + '>' , length: inspectStyleNoop , key: inspectStyleNoop , index: str => '[' + str + '] ' , number: inspectStyleNoop , inspect: inspectStyleNoop , string: inspectStyleNoop , errorType: inspectStyleNoop , errorMessage: inspectStyleNoop , errorStack: inspectStyleNoop , errorStackKeyword: inspectStyleNoop , errorStackMethod: inspectStyleNoop , errorStackMethodAs: inspectStyleNoop , errorStackFile: inspectStyleNoop , errorStackLine: inspectStyleNoop , errorStackColumn: inspectStyleNoop , errorFromMessage: inspectStyleNoop , truncate: ( str , maxLength ) => str.slice( 0 , maxLength - 1 ) + '…' } ; inspectStyle.inline = Object.assign( {} , inspectStyle.none , { trim: true , tab: '' , newline: ' ' , comma: ', ' , length: () => '' , index: () => '' //type: () => '' , } ) ; inspectStyle.color = Object.assign( {} , inspectStyle.none , { limit: str => ansi.bold + ansi.brightRed + str + ansi.reset , type: str => ansi.italic + ansi.brightBlack + str + ansi.reset , constant: str => ansi.cyan + str + ansi.reset , funcName: str => ansi.italic + ansi.magenta + str + ansi.reset , constructorName: str => ansi.magenta + str + ansi.reset , length: str => ansi.italic + ansi.brightBlack + str + ansi.reset , key: str => ansi.green + str + ansi.reset , index: str => ansi.blue + '[' + str + ']' + ansi.reset + ' ' , number: str => ansi.cyan + str + ansi.reset , inspect: str => ansi.cyan + str + ansi.reset , string: str => ansi.blue + str + ansi.reset , errorType: str => ansi.red + ansi.bold + str + ansi.reset , errorMessage: str => ansi.red + ansi.italic + str + ansi.reset , errorStack: str => ansi.brightBlack + str + ansi.reset , errorStackKeyword: str => ansi.italic + ansi.bold + str + ansi.reset , errorStackMethod: str => ansi.brightYellow + str + ansi.reset , errorStackMethodAs: str => ansi.yellow + str + ansi.reset , errorStackFile: str => ansi.brightCyan + str + ansi.reset , errorStackLine: str => ansi.blue + str + ansi.reset , errorStackColumn: str => ansi.magenta + str + ansi.reset , errorFromMessage: str => ansi.yellow + ansi.underline + str + ansi.reset , truncate: ( str , maxLength ) => { var trail = ansi.gray + '…' + ansi.reset ; str = str.slice( 0 , maxLength - trail.length ) ; // Search for an ansi escape sequence at the end, that could be truncated. // The longest one is '\x1b[107m': 6 characters. var lastEscape = str.lastIndexOf( '\x1b' ) ; if ( lastEscape >= str.length - 6 ) { str = str.slice( 0 , lastEscape ) ; } return str + trail ; } } ) ; inspectStyle.html = Object.assign( {} , inspectStyle.none , { tab: '&nbsp;&nbsp;&nbsp;&nbsp;' , newline: '<br />' , limit: str => '<span style="color:red">' + str + '</span>' , type: str => '<i style="color:gray">' + str + '</i>' , constant: str => '<span style="color:cyan">' + str + '</span>' , funcName: str => '<i style="color:magenta">' + str + '</i>' , constructorName: str => '<span style="color:magenta">' + str + '</span>' , length: str => '<i style="color:gray">' + str + '</i>' , key: str => '<span style="color:green">' + str + '</span>' , index: str => '<span style="color:blue">[' + str + ']</span> ' , number: str => '<span style="color:cyan">' + str + '</span>' , inspect: str => '<span style="color:cyan">' + str + '</span>' , string: str => '<span style="color:blue">' + str + '</span>' , errorType: str => '<span style="color:red">' + str + '</span>' , errorMessage: str => '<span style="color:red">' + str + '</span>' , errorStack: str => '<span style="color:gray">' + str + '</span>' , errorStackKeyword: str => '<i>' + str + '</i>' , errorStackMethod: str => '<span style="color:yellow">' + str + '</span>' , errorStackMethodAs: str => '<span style="color:yellow">' + str + '</span>' , errorStackFile: str => '<span style="color:cyan">' + str + '</span>' , errorStackLine: str => '<span style="color:blue">' + str + '</span>' , errorStackColumn: str => '<span style="color:gray">' + str + '</span>' , errorFromMessage: str => '<span style="color:yellow">' + str + '</span>' } ) ;