Spaces:
Running
Running
// 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) | |
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 | |
} | |