A toolkit of RxJS operators to handle CSV formatted data (RFC 4180).
- Parses and generates CSV formatted text
- CSV format support (fully compliant with RFC 4180)
- fields delimiter:
,
- quote:
"
- line breaks:
CRLF
(default) ,LF
,CR
(parsing only)
- fields delimiter:
- Simple API (a handful of RxJS operators)
- Modular design to perfectly fulfill custom requirements
- Compatible with standard RxJS 6 data processing
- ES6 module, typings available
npm i csv-rxjs-kit
The package consists of several simple RxJS operators specific to CSV data processing (text parsing/generating, CSV header manipulation, CSV data records validation) and some utility functions.
All those CSV operators are intended to combine with standard RxJS operators to build up required functionality.
The main data type for the operators is the CSV record:
type csvRecord = string[];
The very first CSV record in a data stream might be a header record. The header record consists of field names (instead of field data) for the following CSV records. The header record can be used to construct Objects with well-named properties or (obviously) to create a CSV text file with a header. One can remove and add an arbitrary header to the data stream.
Empty CSV records are prohibited in CSV files but operators csvParse()
and csvStringify()
allow them. If one needs to get rid of them after parsing or before stringify, they can use csvDropEmpty()
operator.
The package provides basic CSV records verification functionality. RFC 4180 requires all CSV records to be the same length. The validation ensures this and throws an error by default. There are options to change the default validation behaviour to erase, ignore or repair invalid records.
The package provides several specific operators to produce, handle and consume CSV records.
-
Operators to produce CSV records:
csvParse()
parses CSV formatted textcsvConvert()
(withcsvExtractor
helper) extracts data from Object's propertiescsvFromArray()
convert arrays of any data
-
Operators for CSV records conversion to other data formats:
csvStringify()
produces CSV formatted textcsvConvert()
(withcsvBuilder
helper) creates Objects
-
Operators to handle CSV records:
csvDropEmpty()
removes empty records from the records streamcsvDropHeader()
removes the header (first) record from the records streamcsvInjectHeader()
insert a new header record to the data streamcsvValidateRecord()
(withcsvRecordValidator
helper) validates (modifies) records
Every CSV record is an array of strings so one can also use standard RxJS operators to process CSV records.
The most dependent on RFC 4180 operators in the package are csvParse()
and csvStringify()
. Other operators are intended to handle the CSV record stream and implement required application functionality. They can be replaced by combinations of standard RxJS operators.
The package provides some functions to handle CSV records.
- Validation functions:
csvJustifier()
returns helper function, that adjusts record's length or removes invalid records.
- Processing functions:
csvPropNames()
returns Object's property names as a header record.csvPropValues()
returns helper function, that creates a record for Object's properties values.csvAssembler()
returns helper function, that creates an Object with properties initialized by a record's values.
// input CSV text data (with headers)
const data$: Observable<string> = ...;
data$.pipe(
csvParse(), // convert to CSV records
csvDropEmpty(), // remove empty CSV records
csvValidateRecord(true, csvJustifier()), // validate records
csvConvert(true, csvObjAssemble()), // convert CSV records to objects
map((objects) => JSON.stringify(objects)) // generate JSON
) // <- Observable<string>, JSON formatted input CSV records
const data$: Observable<unknown[]> = ...; // input arrays
const names: string[] = ...; // field names
data$.pipe(
csvFromArray(), // don't need this for string arrays
csvInjectHeader(names), // inject field names
csvStringify({ delimiter: '\n', last_break: true }) // generate CSV
) // <- Observable<string>, CSV formatted text
const data$: Observable<string> = ...; // input CSV text data
data$.pipe(
csvParse(),
map((rec) => {
const sum = rec.reduce((acc, v) => +v + acc, 0);
rec.unshift(sum);
return rec;
}),
csvStringify()
) // <- Observable<string>, CSV formatted text