184 lines
7.0 KiB
JavaScript
Executable File
184 lines
7.0 KiB
JavaScript
Executable File
#!/usr/bin/env node
|
|
|
|
'use strict';
|
|
|
|
const { promisify } = require('util');
|
|
const { createReadStream, createWriteStream, readFile: readFileOrig, writeFile: writeFileOrig } = require('fs');
|
|
const os = require('os');
|
|
const { isAbsolute, join } = require('path');
|
|
const program = require('commander');
|
|
const pkg = require('../package');
|
|
const json2csv = require('../lib/json2csv');
|
|
const parseNdJson = require('./utils/parseNdjson');
|
|
const TablePrinter = require('./utils/TablePrinter');
|
|
|
|
const readFile = promisify(readFileOrig);
|
|
const writeFile = promisify(writeFileOrig);
|
|
|
|
const { unwind, flatten } = json2csv.transforms;
|
|
const JSON2CSVParser = json2csv.Parser;
|
|
const Json2csvTransform = json2csv.Transform;
|
|
|
|
program
|
|
.version(pkg.version)
|
|
.option('-i, --input <input>', 'Path and name of the incoming json file. Defaults to stdin.')
|
|
.option('-o, --output <output>', 'Path and name of the resulting csv file. Defaults to stdout.')
|
|
.option('-c, --config <path>', 'Specify a file with a valid JSON configuration.')
|
|
.option('-n, --ndjson', 'Treat the input as NewLine-Delimited JSON.')
|
|
.option('-s, --no-streaming', 'Process the whole JSON array in memory instead of doing it line by line.')
|
|
.option('-f, --fields <fields>', 'List of fields to process. Defaults to field auto-detection.')
|
|
.option('-v, --default-value <defaultValue>', 'Default value to use for missing fields.')
|
|
.option('-q, --quote <quote>', 'Character(s) to use as quote mark. Defaults to \'"\'.')
|
|
.option('-Q, --escaped-quote <escapedQuote>', 'Character(s) to use as a escaped quote. Defaults to a double `quote`, \'""\'.')
|
|
.option('-d, --delimiter <delimiter>', 'Character(s) to use as delimiter. Defaults to \',\'.', ',')
|
|
.option('-e, --eol <eol>', 'Character(s) to use as End-of-Line for separating rows. Defaults to \'\\n\'.', os.EOL)
|
|
.option('-E, --excel-strings','Wraps string data to force Excel to interpret it as string even if it contains a number.')
|
|
.option('-H, --no-header', 'Disable the column name header.')
|
|
.option('-a, --include-empty-rows', 'Includes empty rows in the resulting CSV output.')
|
|
.option('-b, --with-bom', 'Includes BOM character at the beginning of the CSV.')
|
|
.option('-p, --pretty', 'Print output as a pretty table. Use only when printing to console.')
|
|
// Built-in transforms
|
|
.option('--unwind [paths]', 'Creates multiple rows from a single JSON document similar to MongoDB unwind.')
|
|
.option('--unwind-blank', 'When unwinding, blank out instead of repeating data. Defaults to false.', false)
|
|
.option('--flatten-objects', 'Flatten nested objects. Defaults to false.', false)
|
|
.option('--flatten-arrays', 'Flatten nested arrays. Defaults to false.', false)
|
|
.option('--flatten-separator <separator>', 'Flattened keys separator. Defaults to \'.\'.', '.')
|
|
.parse(process.argv);
|
|
|
|
function makePathAbsolute(filePath) {
|
|
return (filePath && !isAbsolute(filePath))
|
|
? join(process.cwd(), filePath)
|
|
: filePath;
|
|
}
|
|
|
|
program.input = makePathAbsolute(program.input);
|
|
program.output = makePathAbsolute(program.output);
|
|
program.config = makePathAbsolute(program.config);
|
|
|
|
// don't fail if piped to e.g. head
|
|
/* istanbul ignore next */
|
|
process.stdout.on('error', (error) => {
|
|
if (error.code === 'EPIPE') process.exit(1);
|
|
});
|
|
|
|
function getInputStream(inputPath) {
|
|
if (inputPath) return createReadStream(inputPath, { encoding: 'utf8' });
|
|
|
|
process.stdin.resume();
|
|
process.stdin.setEncoding('utf8');
|
|
return process.stdin;
|
|
}
|
|
|
|
function getOutputStream(outputPath, config) {
|
|
if (outputPath) return createWriteStream(outputPath, { encoding: 'utf8' });
|
|
if (config.pretty) return new TablePrinter(config).writeStream();
|
|
return process.stdout;
|
|
}
|
|
|
|
async function getInput(inputPath, ndjson) {
|
|
if (!inputPath) return getInputFromStdin();
|
|
if (ndjson) return parseNdJson(await readFile(inputPath, 'utf8'));
|
|
return require(inputPath);
|
|
}
|
|
|
|
async function getInputFromStdin() {
|
|
return new Promise((resolve, reject) => {
|
|
process.stdin.resume();
|
|
process.stdin.setEncoding('utf8');
|
|
|
|
let inputData = '';
|
|
process.stdin.on('data', chunk => (inputData += chunk));
|
|
/* istanbul ignore next */
|
|
process.stdin.on('error', err => reject(new Error('Could not read from stdin', err)));
|
|
process.stdin.on('end', () => {
|
|
try {
|
|
resolve(program.ndjson ? parseNdJson(inputData) : JSON.parse(inputData));
|
|
} catch (err) {
|
|
reject(new Error('Invalid data received from stdin', err));
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
async function processOutput(outputPath, csv, config) {
|
|
if (!outputPath) {
|
|
// eslint-disable-next-line no-console
|
|
config.pretty ? (new TablePrinter(config)).printCSV(csv) : console.log(csv);
|
|
return;
|
|
}
|
|
|
|
await writeFile(outputPath, csv);
|
|
}
|
|
|
|
async function processInMemory(config, opts) {
|
|
const input = await getInput(program.input, config.ndjson);
|
|
const output = new JSON2CSVParser(opts).parse(input);
|
|
await processOutput(program.output, output, config);
|
|
}
|
|
|
|
async function processStream(config, opts) {
|
|
const input = getInputStream(program.input);
|
|
const transform = new Json2csvTransform(opts);
|
|
const output = getOutputStream(program.output, config);
|
|
|
|
await new Promise((resolve, reject) => {
|
|
input.pipe(transform).pipe(output);
|
|
input.on('error', reject);
|
|
transform.on('error', reject);
|
|
output.on('error', reject)
|
|
.on('finish', resolve);
|
|
});
|
|
}
|
|
|
|
(async (program) => {
|
|
try {
|
|
const config = Object.assign({}, program.config ? require(program.config) : {}, program);
|
|
|
|
const transforms = [];
|
|
if (config.unwind) {
|
|
transforms.push(unwind({
|
|
paths: config.unwind === true ? undefined : config.unwind.split(','),
|
|
blankOut: config.unwindBlank
|
|
}));
|
|
}
|
|
|
|
if (config.flattenObjects || config.flattenArrays) {
|
|
transforms.push(flatten({
|
|
objects: config.flattenObjects,
|
|
arrays: config.flattenArrays,
|
|
separator: config.flattenSeparator
|
|
}));
|
|
}
|
|
|
|
const opts = {
|
|
transforms,
|
|
fields: config.fields
|
|
? (Array.isArray(config.fields) ? config.fields : config.fields.split(','))
|
|
: config.fields,
|
|
defaultValue: config.defaultValue,
|
|
quote: config.quote,
|
|
escapedQuote: config.escapedQuote,
|
|
delimiter: config.delimiter,
|
|
eol: config.eol,
|
|
excelStrings: config.excelStrings,
|
|
header: config.header,
|
|
includeEmptyRows: config.includeEmptyRows,
|
|
withBOM: config.withBom
|
|
};
|
|
|
|
await (config.streaming ? processStream : processInMemory)(config, opts);
|
|
} catch(err) {
|
|
let processedError = err;
|
|
if (program.input && err.message.includes(program.input)) {
|
|
processedError = new Error(`Invalid input file. (${err.message})`);
|
|
} else if (program.output && err.message.includes(program.output)) {
|
|
processedError = new Error(`Invalid output file. (${err.message})`);
|
|
} else if (program.config && err.message.includes(program.config)) {
|
|
processedError = new Error(`Invalid config file. (${err.message})`);
|
|
}
|
|
// eslint-disable-next-line no-console
|
|
console.error(processedError);
|
|
process.exit(1);
|
|
}
|
|
})(program);
|