1
0
mirror of https://github.com/bigchaindb/js-bigchaindb-driver.git synced 2025-01-07 12:24:07 +01:00
js-bigchaindb-driver/src/format_text.js

98 lines
3.6 KiB
JavaScript
Raw Normal View History

2017-06-12 16:57:29 +02:00
import { sprintf } from 'sprintf-js'
2017-04-26 15:58:19 +02:00
// Regexes taken from or inspired by sprintf-js
const Regex = {
2017-06-12 17:19:09 +02:00
TEMPLATE_LITERAL: /\${([^)]+?)}/g,
2017-04-26 15:58:19 +02:00
KEY: /^([a-z_][a-z_\d]*)/i,
KEY_ACCESS: /^\.([a-z_][a-z_\d]*)/i,
INDEX_ACCESS: /^\[(\d+)\]/
2017-06-12 16:57:29 +02:00
}
2017-04-26 15:58:19 +02:00
/**
* imported from https://github.com/bigchaindb/js-utility-belt/
*
* Formats strings similarly to C's sprintf, with the addition of '${...}' formats.
*
* Makes a first pass replacing '${...}' formats before passing the expanded string and other
* arguments to sprintf-js. For more information on what sprintf can do, see
* https://github.com/alexei/sprintf.js.
*
* Examples:
* formatText('Hi there ${dimi}!', { dimi: 'Dimi' })
* => 'Hi there Dimi!'
*
* formatText('${database} is %(status)s', { database: 'BigchainDB', status: 'big' })
* => 'BigchainDB is big'
*
* Like sprintf-js, string interpolation for keywords and indexes is supported too:
* formatText('Berlin is best known for its ${berlin.topKnownFor[0].name}', {
* berlin: {
* topKnownFor: [{
* name: 'Currywurst'
* }, ...
* ]
* }
* })
* => 'Berlin is best known for its Currywurst'
*/
export default function formatText(s, ...argv) {
2017-06-12 16:57:29 +02:00
let expandedFormatStr = s
2017-04-26 15:58:19 +02:00
// Try to replace formats of the form '${...}' if named replacement fields are used
if (s && argv.length === 1 && typeof argv[0] === 'object') {
2017-06-12 16:57:29 +02:00
const templateSpecObj = argv[0]
2017-04-26 15:58:19 +02:00
expandedFormatStr = s.replace(Regex.TEMPLATE_LITERAL, (match, replacement) => {
2017-06-12 16:57:29 +02:00
let interpolationLeft = replacement
2017-04-26 15:58:19 +02:00
/**
* Interpolation algorithm inspired by sprintf-js.
*
* Goes through the replacement string getting the left-most key or index to interpolate
* on each pass. `value` at each step holds the last interpolation result, `curMatch` is
* the current property match, and `interpolationLeft` is the portion of the replacement
* string still to be interpolated.
*
* It's useful to note that RegExp.exec() returns with an array holding:
* [0]: Full string matched
* [1+]: Matching groups
*
* And that in the regexes defined, the first matching group always corresponds to the
* property matched.
*/
2017-06-12 16:57:29 +02:00
let value
let curMatch = Regex.KEY.exec(interpolationLeft)
2017-04-26 15:58:19 +02:00
if (curMatch !== null) {
2017-06-12 16:57:29 +02:00
value = templateSpecObj[curMatch[1]]
2017-04-26 15:58:19 +02:00
// Assigning in the conditionals here makes the code less bloated
/* eslint-disable no-cond-assign */
while ((interpolationLeft = interpolationLeft.substring(curMatch[0].length)) &&
value != null) {
if ((curMatch = Regex.KEY_ACCESS.exec(interpolationLeft))) {
2017-06-12 16:57:29 +02:00
value = value[curMatch[1]]
2017-04-26 15:58:19 +02:00
} else if ((curMatch = Regex.INDEX_ACCESS.exec(interpolationLeft))) {
2017-06-12 16:57:29 +02:00
value = value[curMatch[1]]
2017-04-26 15:58:19 +02:00
} else {
2017-06-12 16:57:29 +02:00
break
2017-04-26 15:58:19 +02:00
}
}
/* eslint-enable no-cond-assign */
}
// If there's anything left to interpolate by the end then we've failed to interpolate
// the entire replacement string.
if (interpolationLeft.length) {
throw new SyntaxError(
`[formatText] failed to parse named argument key: ${replacement}`
2017-06-12 16:57:29 +02:00
)
2017-04-26 15:58:19 +02:00
}
2017-06-12 16:57:29 +02:00
return value
})
2017-04-26 15:58:19 +02:00
}
2017-06-12 16:57:29 +02:00
return sprintf(expandedFormatStr, ...argv)
}