mirror of https://github.com/antonmedv/fx
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
69 lines
1.4 KiB
JavaScript
69 lines
1.4 KiB
JavaScript
7 years ago
|
#!/usr/bin/env node
|
||
|
'use strict';
|
||
|
const meow = require('meow')
|
||
|
const stdin = require('get-stdin')
|
||
|
const cardinal = require('cardinal')
|
||
|
const theme = require('cardinal/themes/tomorrow-night')
|
||
|
|
||
|
const cli = meow(`
|
||
|
Usage
|
||
|
$ fx [code ...]
|
||
|
|
||
|
Examples
|
||
|
$ echo '{"key": "value"}' | fx "x => x.key"
|
||
|
"value"
|
||
|
|
||
|
$ echo '[1,2,3]' | fx "this.map(x => x * 2)"
|
||
|
[2, 4, 6]
|
||
|
|
||
|
$ echo '{"items": ["one", "two"]}' | fx "this.items" "this[1]"
|
||
|
"two"
|
||
|
|
||
|
$ echo '{"count": 0}' | fx "{...this, count: 1}"
|
||
|
{"count": 1}
|
||
|
`)
|
||
|
|
||
|
|
||
|
|
||
|
const highlight = process.stdout.isTTY ? cardinal.highlight : x => x
|
||
|
|
||
|
async function main() {
|
||
|
const text = await stdin()
|
||
|
|
||
|
if (text === '') {
|
||
|
cli.showHelp()
|
||
|
}
|
||
|
|
||
|
const json = JSON.parse(text)
|
||
|
const result = cli.input.reduce(reduce, json)
|
||
|
|
||
|
if (typeof result === 'undefined') {
|
||
|
console.log(undefined)
|
||
|
} else {
|
||
|
const text = JSON.stringify(result, null, 4)
|
||
|
console.log(highlight(text, {theme}))
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function reduce(json, code) {
|
||
|
if (/^\w+\s*=>/.test(code)) {
|
||
|
const fx = eval(code)
|
||
|
return fx(json)
|
||
|
} else if (/yield/.test(code)) {
|
||
|
const fx = eval(`
|
||
|
function fn() {
|
||
|
const gen = (function*(){
|
||
|
${code.replace(/\\\n/g, '')}
|
||
|
}).call(this)
|
||
|
return [...gen]
|
||
|
}; fn
|
||
|
`)
|
||
|
return fx.call(json)
|
||
|
} else {
|
||
|
const fx = eval(`function fn() { return ${code} }; fn`)
|
||
|
return fx.call(json)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
main()
|