Source: formatter.js

/**
 * @namespace formatter
 * @memberof module:kdljs
 */

import { validateDocument } from './validator.js'
import { Identifier } from './parser/tokens.js'
import { bannedIdentifiers } from './parser/base.js'

/* eslint-disable no-control-regex */
const linespace = /^[\r\n\u0085\u000C\u2028\u2029]$/
const nonAscii = /^[^\x00-\x7F]$/
const nonPrintableAscii = /^[\x00-\x19\x7F]$/
/* eslint-enable no-control-regex */
const commonEscapes = {
  '\n': '\\n',
  '\r': '\\r',
  '\t': '\\t',
  '\\': '\\\\',
  '"': '\\"',
  '\x08': '\\b',
  '\x0C': '\\f'
}

const identifierPattern = new RegExp('^(' + Identifier.PATTERN.source + ')$')

/**
 * @access private
 * @memberof module:kdljs.formatter
 * @param {string} char
 * @param {module:kdljs/formatter.ProcessedOptions} options - Formatting options
 * @return {boolean}
 */
function shouldEscapeChar (value, options) {
  const charCode = value.charCodeAt(0)

  if (options.escapes[charCode]) {
    return true
  } else if (value === '"' || value === '\\') {
    return true
  } else if (options.escapeCommon && value in commonEscapes) {
    return true
  } else if (options.escapeNonPrintableAscii && nonPrintableAscii.test(value)) {
    return true
  } else if (options.escapeLinespace && linespace.test(value)) {
    return true
  } else if (options.escapeNonAscii && nonAscii.test(value)) {
    return true
  } else {
    return false
  }
}

/**
 * @access private
 * @memberof module:kdljs.formatter
 * @param {string} char
 * @param {module:kdljs/formatter.ProcessedOptions} options - Formatting options
 * @return {string}
 */
function formatChar (value, options) {
  if (!shouldEscapeChar(value, options)) {
    return value
  } else if (value in commonEscapes) {
    return commonEscapes[value]
  } else {
    return `\\u{${value.charCodeAt(0).toString(16)}}`
  }
}

/**
 * @access private
 * @memberof module:kdljs.formatter
 * @param {string} value
 * @param {module:kdljs/formatter.ProcessedOptions} options - Formatting options
 * @return {string}
 */
function formatString (value, options) {
  if (options.preferIdentifierString && identifierPattern.test(value) && !bannedIdentifiers.has(value)) {
    return value
  }

  return `"${[...value].map(char => formatChar(char, options)).join('')}"`
}

/**
 * @access private
 * @memberof module:kdljs.formatter
 * @param {string} value
 * @param {module:kdljs/formatter.ProcessedOptions} options - Formatting options
 * @return {string}
 */
function formatIdentifier (value, options) {
  return formatString(value, { ...options, preferIdentifierString: true })
}

/**
 * @access private
 * @memberof module:kdljs.formatter
 * @param {number} value
 * @param {module:kdljs/formatter.ProcessedOptions} options - Formatting options
 * @return {string}
 */
function formatNumber (value, options) {
  if (!Number.isFinite(value)) {
    if (value === Number.POSITIVE_INFINITY) {
      return '#inf'
    } else if (value === Number.NEGATIVE_INFINITY) {
      return '#-inf'
    } else {
      return '#nan'
    }
  }

  if (options.exponentChar === 'E') {
    return value.toString().toUpperCase()
  } else {
    return value.toString()
  }
}

/**
 * @access private
 * @memberof module:kdljs.formatter
 * @param {string} [value] - KDL tag
 * @param {module:kdljs/formatter.ProcessedOptions} options - Formatting options
 * @return {string}
 */
function formatTag (value, options) {
  return value === undefined ? '' : '(' + formatIdentifier(value, options) + ')'
}

/**
 * @access private
 * @memberof module:kdljs.formatter
 * @param {module:kdljs~Value} value - KDL value
 * @param {string} [tag] - KDL tag
 * @param {module:kdljs/formatter.ProcessedOptions} options - Formatting options
 * @return {string}
 */
function formatValue (value, tag, options) {
  if (typeof value === 'string') {
    return formatTag(tag, options) + formatString(value, options)
  } else if (typeof value === 'number') {
    return formatTag(tag, options) + formatNumber(value, options)
  } else {
    return formatTag(tag, options) + '#' + value
  }
}

/**
 * @access private
 * @memberof module:kdljs.formatter
 * @param {string} key
 * @param {module:kdljs~Value} value - KDL value
 * @param {string} [tag] - KDL tag
 * @param {module:kdljs/formatter.ProcessedOptions} options - Formatting options
 * @return {string}
 */
function formatProperty (key, value, tag, options) {
  return formatIdentifier(key, options) + '=' + formatValue(value, tag, options)
}

/**
 * @access private
 * @memberof module:kdljs.formatter
 * @param {module:kdljs~Node} node - KDL node
 * @param {module:kdljs/formatter.ProcessedOptions} options - Formatting options
 * @param {number} indent
 * @return {string}
 */
function formatNode (node, options, indent) {
  let values = node.values
  if (!options.printNullArgs) {
    values = values.filter(value => value !== null)
  }

  let properties = Object.keys(node.properties).sort()
  if (!options.printNullProps) {
    properties = properties.filter(key => node.properties[key] !== null)
  }

  const currentIndent = options.indentChar.repeat(options.indent * indent)
  const parts = [
    currentIndent + formatTag(node.tags.name, options) + formatIdentifier(node.name, options),
    ...values.map((value, index) => formatValue(value, node.tags.values[index], options)),
    ...properties.map(key => formatProperty(key, node.properties[key], node.tags.properties[key], options))
  ]

  if (node.children.length) {
    indent++
    parts.push('{' + options.newline + formatDocument(node.children, options, indent) + options.newline + currentIndent + '}')
    indent--
  } else if (options.printEmptyChildren) {
    parts.push('{}')
  }

  return options.requireSemicolons ? parts.join(' ') + ';' : parts.join(' ')
}

/**
 * @access private
 * @memberof module:kdljs.formatter
 * @param {module:kdljs~Document} doc - KDL document
 * @param {module:kdljs/formatter.ProcessedOptions} options - Formatting options
 * @param {number} indent
 * @return {string}
 */
function formatDocument (doc, options, indent) {
  const nodes = doc.map(node => formatNode(node, options, indent))

  return nodes.join(options.newline)
}

/**
 * @access private
 * @typedef ProcessedOptions
 * @memberof module:kdljs.formatter
 * @type {Object}
 * @property {Object<number,boolean>} escapes
 * @property {boolean} requireSemicolons
 * @property {boolean} escapeNonAscii
 * @property {boolean} escapeNonPrintableAscii
 * @property {boolean} escapeCommon
 * @property {boolean} escapeLinespace
 * @property {string} newline
 * @property {number} indent
 * @property {string} indentChar
 * @property {string} exponentChar
 * @property {string} preferIdentifierString
 * @property {boolean} printEmptyChildren
 * @property {boolean} printNullArgs
 * @property {boolean} printNullProps
 */

/**
 * @access private
 * @memberof module:kdljs.formatter
 * @param {module:kdljs/formatter.Options} options - Formatting options
 * @return {module:kdljs/formatter.ProcessedOptions}
 */
function processOptions (options) {
  return {
    escapes: {},
    requireSemicolons: false,
    escapeNonAscii: false,
    escapeNonPrintableAscii: true,
    escapeCommon: true,
    escapeLinespace: true,
    newline: '\n',
    indent: 4,
    indentChar: ' ',
    exponentChar: 'E',
    preferIdentifierString: false,
    printEmptyChildren: false,
    printNullArgs: true,
    printNullProps: true,
    ...options
  }
}

/**
 * @typedef Options
 * @memberof module:kdljs.formatter
 * @type {Object}
 * @property {Object<number,boolean>} [escapes={}] - A map of which characters to escape (by decimal codepoint)
 * @property {boolean} [requireSemicolons=false] - Whether to print semicolons after each node
 * @property {boolean} [escapeNonAscii=false] - Whether to escape any non-ASCII characters
 * @property {boolean} [escapeNonPrintableAscii=true] - Whether to escape non-printable ASCII characters
 * @property {boolean} [escapeCommon=true] - Whether to escape common characters (i.e. those with single-character escape codes)
 * @property {boolean} [escapeLinespace=true] - Whether to escape linespace characters
 * @property {string} [newline='\n'] - The newline character
 * @property {number} [indent=4] - The number of characters (from `indentChar`) to indent each level with
 * @property {string} [indentChar=' '] - What character to indent with
 * @property {string} [exponentChar='E'] - What character to use for the exponent in floats (`e` or `E`)
 * @property {boolean} [preferIdentifierString=false] - Whether to prefer identifier-style strings
 * @property {boolean} [printEmptyChildren=false] - Whether to print empty brackets if a node has no children
 * @property {boolean} [printNullArgs=true] - Whether to print `null` values
 * @property {boolean} [printNullProps=true] - Whether to print properties with value `null`
 */

/**
 * @function format
 * @memberof module:kdljs.formatter
 * @param {module:kdljs~Document} doc - Input KDL document
 * @param {module:kdljs.formatter.Options} [options={}] - Formatting options
 * @return {string} formatted KDL file
 */
export function format (doc, options) {
  if (!validateDocument(doc)) {
    throw new TypeError('Invalid KDL document')
  }

  const processsedOptions = processOptions(options || {})

  return formatDocument(doc, processsedOptions, 0) + processsedOptions.newline
}