// 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; };