DefinitelyTyped/types/json2csv/json2csv-tests.ts
2020-04-06 10:57:43 -07:00

222 lines
6.6 KiB
TypeScript

import { Transform as NodeTransform } from 'stream';
import { createReadStream, createWriteStream } from 'fs';
import json2csv, { AsyncParser, parse, Parser, parseAsync, Transform, transforms } from 'json2csv';
const { flatten, unwind } = transforms;
let s: string;
let obj: object;
interface ExampleObj {
str?: string;
num?: number;
obj?: object;
}
/**************
* Public API *
**************/
parse({});
parse([]);
parse({}, {});
new Parser();
const parser: Parser<ExampleObj> = new Parser({});
s = parser.parse({ str: '', num: 1, obj: {} });
parser.parse([]);
const transform: Transform<ExampleObj> = new Transform<ExampleObj>({ quote: '' });
const nodeTransform: NodeTransform = transform;
// Tests using examples from Readme
interface Car {
car: string;
price: number;
}
const opts: json2csv.Options<Car> = {
fields: [
// Supports pathname -> pathvalue
'simplepath', // equivalent to {value:'simplepath'}
'path.to.value', // also equivalent to {value:'path.to.value'}
// Supports label -> simple path
{
label: 'some label', // (optional, column will be labeled 'path.to.something' if not defined)
value: 'path.to.something', // data.path.to.something
default: 'NULL' // default if value is not found (optional, overrides `defaultValue` for column)
},
// Supports label -> derived value
{
label: 'some label', // Soptional, column will be labeled with the function name or empty if the function is anonymous)
value: (row: Car, field: json2csv.FieldValueCallbackInfo) => {
if (field) {
return (row as any)[field.label].toLowerCase() || field.default;
}
},
default: 'NULL', // default if value function returns null or undefined
},
// Supports label -> derived value
{
value: (row: Car) => row.car
},
// Supports label -> derived value
{
value: (row: Car) => `"${row.car}"`,
},
]
};
const data = [{
car: 'VW Beetle',
price: 1395
}, {
car: 'VW T1',
price: 1357
}];
// Test for Synchronous Parser
try {
const parser = new Parser(opts);
const csv = parser.parse({ car: '', price: 1 });
console.log(csv);
} catch (err) {
console.error(err);
}
// Test convenience method "parse"
try {
const csv = parse(data, opts);
console.log(csv);
} catch (err) {
console.error(err);
}
// Test for Asynchronous Parser
const transformOpts = { highWaterMark: 8192 };
const asyncParser = new AsyncParser(opts, transformOpts);
let csv = '';
asyncParser.processor
.on('data', chunk => (csv += chunk.toString()))
.on('end', () => console.log(csv))
.on('error', err => console.error(err));
// Test for transform events
asyncParser.transform
.on('header', header => console.log(header))
.on('line', line => console.log(line))
.on('error', err => console.log(err));
asyncParser.input.push(data); // This data might come from an HTTP request, etc.
asyncParser.input.push(null); // Sending `null` to a stream signal that no more data is expected and ends it.
const input = createReadStream('/path/to/input', { encoding: 'utf8' });
const output = createWriteStream('path/to/output', { encoding: 'utf8' });
asyncParser.fromInput(input).promise()
.then(csv => console.log(csv))
.catch(err => console.error(err));
asyncParser.fromInput(input).promise(false)
.then(() => console.log('Ready'))
.catch(err => console.error(err));
asyncParser.fromInput(input).toOutput(output);
// Test convenience method "parseAsync" with object input
parseAsync(data, opts)
.then(csv => console.log(csv))
.catch(err => console.error(err));
// Test convenience method "parseAsync" with stream input
parseAsync(input, opts)
.then(csv => console.log(csv))
.catch(err => console.error(err));
// Test transforms
function myTransform(input: object) {
const transformed = { ...input, id: 1 };
return transformed;
}
const tranformsOpts: json2csv.Options<Car> = {
transforms: [
unwind(),
unwind({}),
unwind({ paths: ['path'] }),
unwind({ blankOut: true }),
unwind({ paths: ['path'], blankOut: true }),
flatten(),
flatten({}),
flatten({ objects: true }),
flatten({ arrays: true }),
flatten({ separator: '-' }),
flatten({ objects: true, arrays: true, separator: '-' }),
(input) => input,
(_) => ({ a: 1, b: 2 }),
myTransform,
]
};
/********************
* Internal Methods *
********************/
class ParserExt extends Parser<ExampleObj> {
constructor() {
super();
// Parser methods
obj = this.preprocessData({});
obj = this.preprocessData({ str: '', num: 1, obj: {} });
obj = this.preprocessData([]);
s = this.processData([]);
// JSON2CSVBase methods
let opts: json2csv.Options<ExampleObj>;
opts = this.preprocessOpts();
opts = this.preprocessOpts(opts);
s = this.getHeader();
obj = this.preprocessRow({});
obj = this.preprocessRow({ str: '', num: 1, obj: {} });
s = this.processRow({});
s = this.processRow({ str: '', num: 1, obj: {} });
s = this.processCell({}, { label: 'test', value: (row: object, field: json2csv.FieldValueCallbackInfo) => 'string' });
s = this.processValue(undefined);
s = this.processValue(null);
s = this.processValue(1);
s = this.processValue('test');
s = this.processValue(new Date());
s = this.processValue({});
s = this.processValue([]);
}
}
class TransformExt extends Transform<ExampleObj> {
constructor() {
super();
// Transform methods
this.pushLine({});
this.pushLine({ str: '', num: 1, obj: {} });
// JSON2CSVBase methods
let opts: json2csv.Options<ExampleObj>;
opts = this.preprocessOpts();
opts = this.preprocessOpts(opts);
s = this.getHeader();
obj = this.preprocessRow({});
obj = this.preprocessRow({ str: '', num: 1, obj: {} });
s = this.processRow({});
s = this.processRow({ str: '', num: 1, obj: {} });
s = this.processCell({}, { label: 'test', value: (row: object, field: json2csv.FieldValueCallbackInfo) => 'string' });
s = this.processValue(undefined);
s = this.processValue(null);
s = this.processValue(1);
s = this.processValue('test');
s = this.processValue(new Date());
s = this.processValue({});
s = this.processValue([]);
}
}