metaforecast/node_modules/json2csv/bin/json2csv.js
2021-01-12 13:43:41 +01:00

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);