// Limited implementation of python % string operator, supports only %s and %r for now // (other formats are not used here, but may appear in custom templates) 'use strict' const { inspect } = require('util') module.exports = function sub(pattern, ...values) { let regex = /%(?:(%)|(-)?(\*)?(?:\((\w+)\))?([A-Za-z]))/g let result = pattern.replace(regex, function (_, is_literal, is_left_align, is_padded, name, format) { if (is_literal) return '%' let padded_count = 0 if (is_padded) { if (values.length === 0) throw new TypeError('not enough arguments for format string') padded_count = values.shift() if (!Number.isInteger(padded_count)) throw new TypeError('* wants int') } let str if (name !== undefined) { let dict = values[0] if (typeof dict !== 'object' || dict === null) throw new TypeError('format requires a mapping') if (!(name in dict)) throw new TypeError(`no such key: '${name}'`) str = dict[name] } else { if (values.length === 0) throw new TypeError('not enough arguments for format string') str = values.shift() } switch (format) { case 's': str = String(str) break case 'r': str = inspect(str) break case 'd': case 'i': if (typeof str !== 'number') { throw new TypeError(`%${format} format: a number is required, not ${typeof str}`) } str = String(str.toFixed(0)) break default: throw new TypeError(`unsupported format character '${format}'`) } if (padded_count > 0) { return is_left_align ? str.padEnd(padded_count) : str.padStart(padded_count) } else { return str } }) if (values.length) { if (values.length === 1 && typeof values[0] === 'object' && values[0] !== null) { // mapping } else { throw new TypeError('not all arguments converted during string formatting') } } return result }