418 lines
No EOL
57 KiB
JavaScript
418 lines
No EOL
57 KiB
JavaScript
"use strict";
|
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
};
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
var Converter_1 = require("../src/Converter");
|
|
var src_1 = __importDefault(require("../src"));
|
|
var assert = require("assert");
|
|
var fs = require("fs");
|
|
var sandbox = require("sinon").sandbox.create();
|
|
var file = __dirname + "/data/testData";
|
|
var trailCommaData = __dirname + "/data/trailingComma";
|
|
describe("CSV Converter", function () {
|
|
afterEach(function () {
|
|
sandbox.restore();
|
|
});
|
|
it("should create new instance of csv", function () {
|
|
var obj = new Converter_1.Converter();
|
|
assert(obj);
|
|
});
|
|
it("should read from a stream", function (done) {
|
|
var obj = new Converter_1.Converter();
|
|
var stream = fs.createReadStream(file);
|
|
obj.then(function (obj) {
|
|
assert.equal(obj.length, 2);
|
|
done();
|
|
});
|
|
stream.pipe(obj);
|
|
});
|
|
it("should call onNext once a row is parsed.", function (done) {
|
|
var obj = new Converter_1.Converter();
|
|
var stream = fs.createReadStream(file);
|
|
var called = false;
|
|
obj.subscribe(function (resultRow) {
|
|
assert(resultRow);
|
|
called = true;
|
|
});
|
|
obj.on("done", function () {
|
|
assert(called);
|
|
done();
|
|
});
|
|
stream.pipe(obj);
|
|
});
|
|
it("should emit end_parsed message once it is finished.", function (done) {
|
|
var obj = new Converter_1.Converter();
|
|
obj.then(function (result) {
|
|
assert(result);
|
|
assert(result.length === 2);
|
|
assert(result[0].date);
|
|
assert(result[0].employee);
|
|
assert(result[0].employee.name);
|
|
assert(result[0].employee.age);
|
|
assert(result[0].employee.number);
|
|
assert(result[0].employee.key.length === 2);
|
|
assert(result[0].address.length === 2);
|
|
done();
|
|
});
|
|
fs.createReadStream(file).pipe(obj);
|
|
});
|
|
it("should handle traling comma gracefully", function (done) {
|
|
var stream = fs.createReadStream(trailCommaData);
|
|
var obj = new Converter_1.Converter();
|
|
obj.then(function (result) {
|
|
assert(result);
|
|
assert(result.length > 0);
|
|
done();
|
|
});
|
|
stream.pipe(obj);
|
|
});
|
|
it("should handle comma in column which is surrounded by qoutes", function (done) {
|
|
var testData = __dirname + "/data/dataWithComma";
|
|
var rs = fs.createReadStream(testData);
|
|
var obj = new Converter_1.Converter({
|
|
"quote": "#"
|
|
});
|
|
obj.then(function (result) {
|
|
assert(result[0].col1 === "\"Mini. Sectt");
|
|
assert.equal(result[3].col2, "125001,fenvkdsf");
|
|
// console.log(result);
|
|
done();
|
|
});
|
|
rs.pipe(obj);
|
|
});
|
|
it("should be able to convert a csv to column array data", function (done) {
|
|
var columArrData = __dirname + "/data/columnArray";
|
|
var rs = fs.createReadStream(columArrData);
|
|
var result = {};
|
|
var csvConverter = new Converter_1.Converter();
|
|
//end_parsed will be emitted once parsing finished
|
|
csvConverter.then(function () {
|
|
assert(result.TIMESTAMP.length === 5);
|
|
done();
|
|
});
|
|
//record_parsed will be emitted each time a row has been parsed.
|
|
csvConverter.subscribe(function (resultRow, rowIndex) {
|
|
for (var key in resultRow) {
|
|
if (resultRow.hasOwnProperty(key)) {
|
|
if (!result[key] || !(result[key] instanceof Array)) {
|
|
result[key] = [];
|
|
}
|
|
result[key][rowIndex] = resultRow[key];
|
|
}
|
|
}
|
|
});
|
|
rs.pipe(csvConverter);
|
|
});
|
|
it("should be able to convert csv string directly", function (done) {
|
|
var testData = __dirname + "/data/testData";
|
|
var data = fs.readFileSync(testData).toString();
|
|
var csvConverter = new Converter_1.Converter();
|
|
//end_parsed will be emitted once parsing finished
|
|
csvConverter.then(function (jsonObj) {
|
|
assert.equal(jsonObj.length, 2);
|
|
});
|
|
csvConverter.fromString(data).then(function (jsonObj) {
|
|
assert(jsonObj.length === 2);
|
|
done();
|
|
});
|
|
});
|
|
it("should be able to convert csv string with error", function (done) {
|
|
var testData = __dirname + "/data/dataWithUnclosedQuotes";
|
|
var data = fs.readFileSync(testData).toString();
|
|
var csvConverter = new Converter_1.Converter();
|
|
csvConverter.fromString(data).then(undefined, function (err) {
|
|
assert(err);
|
|
assert.equal(err.err, "unclosed_quote");
|
|
done();
|
|
});
|
|
});
|
|
it("should be able to convert csv string without callback provided", function (done) {
|
|
var testData = __dirname + "/data/testData";
|
|
var data = fs.readFileSync(testData).toString();
|
|
var csvConverter = new Converter_1.Converter();
|
|
//end_parsed will be emitted once parsing finished
|
|
csvConverter.then(function (jsonObj) {
|
|
assert(jsonObj.length === 2);
|
|
done();
|
|
});
|
|
csvConverter.fromString(data);
|
|
});
|
|
it("should be able to handle columns with double quotes", function (done) {
|
|
var testData = __dirname + "/data/dataWithQoutes";
|
|
var data = fs.readFileSync(testData).toString();
|
|
var csvConverter = new Converter_1.Converter();
|
|
csvConverter.fromString(data).then(function (jsonObj) {
|
|
assert(jsonObj[0].TIMESTAMP === '13954264"22', JSON.stringify(jsonObj[0].TIMESTAMP));
|
|
assert(jsonObj[1].TIMESTAMP === 'abc, def, ccc', JSON.stringify(jsonObj[1].TIMESTAMP));
|
|
done();
|
|
});
|
|
});
|
|
it("should be able to handle columns with two double quotes", function (done) {
|
|
var testData = __dirname + "/data/twodoublequotes";
|
|
var data = fs.readFileSync(testData).toString();
|
|
var csvConverter = new Converter_1.Converter();
|
|
csvConverter.fromString(data).then(function (jsonObj) {
|
|
assert.equal(jsonObj[0].title, "\"");
|
|
assert.equal(jsonObj[0].data, "xyabcde");
|
|
assert.equal(jsonObj[0].uuid, "fejal\"eifa");
|
|
assert.equal(jsonObj[0].fieldA, "bnej\"\"falkfe");
|
|
assert.equal(jsonObj[0].fieldB, "\"eisjfes\"");
|
|
done();
|
|
});
|
|
});
|
|
it("should handle empty csv file", function (done) {
|
|
var testData = __dirname + "/data/emptyFile";
|
|
var rs = fs.createReadStream(testData);
|
|
var csvConverter = new Converter_1.Converter();
|
|
csvConverter.then(function (jsonObj) {
|
|
assert(jsonObj.length === 0);
|
|
done();
|
|
});
|
|
rs.pipe(csvConverter);
|
|
});
|
|
it("should parse large csv file", function (done) {
|
|
var testData = __dirname + "/data/large-csv-sample.csv";
|
|
var rs = fs.createReadStream(testData);
|
|
var csvConverter = new Converter_1.Converter();
|
|
var count = 0;
|
|
csvConverter.subscribe(function () {
|
|
//console.log(arguments);
|
|
count++;
|
|
});
|
|
csvConverter.then(function () {
|
|
assert(count === 5290);
|
|
done();
|
|
});
|
|
rs.pipe(csvConverter);
|
|
});
|
|
it("should parse data and covert to specific types", function (done) {
|
|
var testData = __dirname + "/data/dataWithType";
|
|
var rs = fs.createReadStream(testData);
|
|
var csvConverter = new Converter_1.Converter({
|
|
checkType: true,
|
|
colParser: {
|
|
"column6": "string",
|
|
"column7": "string"
|
|
}
|
|
});
|
|
csvConverter.subscribe(function (d) {
|
|
assert(typeof d.column1 === "number");
|
|
assert(typeof d.column2 === "string");
|
|
assert.equal(d["colume4"], "someinvaliddate");
|
|
assert(d.column5.hello === "world");
|
|
assert(d.column6 === '{"hello":"world"}');
|
|
assert(d.column7 === "1234");
|
|
assert(d.column8 === "abcd");
|
|
assert(d.column9 === true);
|
|
assert(d.column10[0] === 23);
|
|
assert(d.column10[1] === 31);
|
|
assert(d.column11[0].hello === "world");
|
|
assert(d["name#!"] === false);
|
|
});
|
|
csvConverter.on("done", function () {
|
|
done();
|
|
});
|
|
rs.pipe(csvConverter);
|
|
});
|
|
it("should turn off field type check", function (done) {
|
|
var testData = __dirname + "/data/dataWithType";
|
|
var rs = fs.createReadStream(testData);
|
|
var csvConverter = new Converter_1.Converter({
|
|
checkType: false
|
|
});
|
|
csvConverter.subscribe(function (d) {
|
|
assert(typeof d.column1 === "string");
|
|
assert(typeof d.column2 === "string");
|
|
assert(d["column3"] === "2012-01-01");
|
|
assert(d["colume4"] === "someinvaliddate");
|
|
assert(d.column5 === '{"hello":"world"}');
|
|
assert.equal(d["column6"], '{"hello":"world"}');
|
|
assert(d["column7"] === "1234");
|
|
assert(d["column8"] === "abcd");
|
|
assert(d.column9 === "true");
|
|
assert(d.column10[0] === "23");
|
|
assert(d.column10[1] === "31");
|
|
assert(d["name#!"] === 'false');
|
|
});
|
|
csvConverter.then(function () {
|
|
done();
|
|
});
|
|
rs.pipe(csvConverter);
|
|
});
|
|
it("should emit data event correctly", function (done) {
|
|
var testData = __dirname + "/data/large-csv-sample.csv";
|
|
var csvConverter = new Converter_1.Converter({});
|
|
var count = 0;
|
|
csvConverter.on("data", function (d) {
|
|
count++;
|
|
});
|
|
csvConverter.on("end", function () {
|
|
assert.equal(count, 5290);
|
|
done();
|
|
});
|
|
var rs = fs.createReadStream(testData);
|
|
rs.pipe(csvConverter);
|
|
});
|
|
it("should process column with linebreaks", function (done) {
|
|
var testData = __dirname + "/data/lineBreak";
|
|
var rs = fs.createReadStream(testData);
|
|
var csvConverter = new Converter_1.Converter({
|
|
checkType: true
|
|
});
|
|
csvConverter.subscribe(function (d) {
|
|
assert(d.Period === 13);
|
|
assert(d["Apparent age"] === "Unknown");
|
|
done();
|
|
});
|
|
rs.pipe(csvConverter);
|
|
});
|
|
it("be able to ignore empty columns", function (done) {
|
|
var testData = __dirname + "/data/dataIgnoreEmpty";
|
|
var rs = fs.createReadStream(testData);
|
|
var st = rs.pipe(src_1.default({ ignoreEmpty: true }));
|
|
st.then(function (res) {
|
|
var j = res[0];
|
|
assert(res.length === 3);
|
|
assert(j.col2.length === 2);
|
|
assert(j.col2[1] === "d3");
|
|
assert(j.col4.col3 === undefined);
|
|
assert(j.col4.col5 === "world");
|
|
assert(res[1].col1 === "d2");
|
|
assert(res[2].col1 === "d4");
|
|
done();
|
|
});
|
|
});
|
|
it("should allow no header", function (done) {
|
|
var testData = __dirname + "/data/noheadercsv";
|
|
var rs = fs.createReadStream(testData);
|
|
var st = rs.pipe(new Converter_1.Converter({ noheader: true }));
|
|
st.then(function (res) {
|
|
var j = res[0];
|
|
assert(res.length === 5);
|
|
assert(j.field1 === "CC102-PDMI-001");
|
|
assert(j.field2 === "eClass_5.1.3");
|
|
done();
|
|
});
|
|
});
|
|
it("should allow customised header", function (done) {
|
|
var testData = __dirname + "/data/noheadercsv";
|
|
var rs = fs.createReadStream(testData);
|
|
var st = rs.pipe(new Converter_1.Converter({
|
|
noheader: true,
|
|
headers: ["a", "b"]
|
|
}));
|
|
st.then(function (res) {
|
|
var j = res[0];
|
|
assert(res.length === 5);
|
|
assert(j.a === "CC102-PDMI-001");
|
|
assert(j.b === "eClass_5.1.3");
|
|
assert(j.field3 === "10/3/2014");
|
|
done();
|
|
});
|
|
});
|
|
it("should allow customised header to override existing header", function (done) {
|
|
var testData = __dirname + "/data/complexJSONCSV";
|
|
var rs = fs.createReadStream(testData);
|
|
var st = rs.pipe(new Converter_1.Converter({
|
|
headers: []
|
|
}));
|
|
st.then(function (res) {
|
|
var j = res[0];
|
|
assert(res.length === 2);
|
|
assert(j.field1 === "Food Factory");
|
|
assert(j.field2 === "Oscar");
|
|
done();
|
|
});
|
|
});
|
|
it("should handle when there is an empty string", function (done) {
|
|
var testData = __dirname + "/data/dataWithEmptyString";
|
|
var rs = fs.createReadStream(testData);
|
|
var st = rs.pipe(new Converter_1.Converter({
|
|
noheader: true,
|
|
headers: ["a", "b", "c"],
|
|
checkType: true
|
|
}));
|
|
st.then(function (res) {
|
|
var j = res[0];
|
|
// assert(res.length===2);
|
|
assert(j.a === "green");
|
|
assert(j.b === 40);
|
|
assert.equal(j.c, "");
|
|
done();
|
|
});
|
|
});
|
|
it("should detect eol correctly when first chunk is smaller than header row length", function (done) {
|
|
var testData = __dirname + "/data/dataNoTrimCRLF";
|
|
var rs = fs.createReadStream(testData, { highWaterMark: 3 });
|
|
var st = rs.pipe(new Converter_1.Converter({
|
|
trim: false
|
|
}));
|
|
st.then(function (res) {
|
|
var j = res[0];
|
|
assert(res.length === 2);
|
|
assert(j.name === "joe");
|
|
assert(j.age === "20");
|
|
assert.equal(res[1].name, "sam");
|
|
assert.equal(res[1].age, "30");
|
|
done();
|
|
});
|
|
});
|
|
it("should detect eol correctly when first chunk ends in middle of CRLF line break", function (done) {
|
|
var testData = __dirname + "/data/dataNoTrimCRLF";
|
|
var rs = fs.createReadStream(testData, { highWaterMark: 9 });
|
|
var st = rs.pipe(new Converter_1.Converter({
|
|
trim: false
|
|
}));
|
|
st.then(function (res) {
|
|
var j = res[0];
|
|
assert(res.length === 2);
|
|
assert(j.name === "joe");
|
|
assert(j.age === "20");
|
|
assert.equal(res[1].name, "sam");
|
|
assert.equal(res[1].age, "30");
|
|
done();
|
|
});
|
|
});
|
|
it("should emit eol event when line ending is detected as CRLF", function (done) {
|
|
var testData = __dirname + "/data/dataNoTrimCRLF";
|
|
var rs = fs.createReadStream(testData);
|
|
var st = rs.pipe(new Converter_1.Converter());
|
|
var eolCallback = sandbox.spy(function (eol) {
|
|
assert.equal(eol, "\r\n");
|
|
});
|
|
st.on("eol", eolCallback);
|
|
st.then(function () {
|
|
assert.equal(eolCallback.callCount, 1, 'should emit eol event once');
|
|
done();
|
|
});
|
|
});
|
|
it("should emit eol event when line ending is detected as LF", function (done) {
|
|
var testData = __dirname + "/data/columnArray";
|
|
var rs = fs.createReadStream(testData);
|
|
var st = rs.pipe(new Converter_1.Converter());
|
|
var eolCallback = sandbox.spy(function (eol) {
|
|
assert.equal(eol, "\n");
|
|
});
|
|
st.on("eol", eolCallback);
|
|
st.then(function () {
|
|
assert.equal(eolCallback.callCount, 1, 'should emit eol event once');
|
|
done();
|
|
});
|
|
});
|
|
it("should remove the Byte Order Mark (BOM) from input", function (done) {
|
|
var testData = __dirname + "/data/dataNoTrimBOM";
|
|
var rs = fs.createReadStream(testData);
|
|
var st = rs.pipe(new Converter_1.Converter({
|
|
trim: false
|
|
}));
|
|
st.then(function (res) {
|
|
var j = res[0];
|
|
assert(res.length === 2);
|
|
assert(j.name === "joe");
|
|
assert(j.age === "20");
|
|
done();
|
|
});
|
|
});
|
|
});
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"/Users/kxiang/work/projects/csv2json/test/testCSVConverter.ts","sources":["/Users/kxiang/work/projects/csv2json/test/testCSVConverter.ts"],"names":[],"mappings":";;;;;AAAA,8CAA2C;AAC3C,+CAAyB;AACzB,IAAI,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC,CAAC;AAC/B,IAAI,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;AACvB,IAAI,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC;AAChD,IAAI,IAAI,GAAG,SAAS,GAAG,gBAAgB,CAAC;AACxC,IAAI,cAAc,GAAG,SAAS,GAAG,qBAAqB,CAAC;AACvD,QAAQ,CAAC,eAAe,EAAE;IACxB,SAAS,CAAC;QACR,OAAO,CAAC,OAAO,EAAE,CAAC;IACpB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,mCAAmC,EAAE;QACtC,IAAI,GAAG,GAAG,IAAI,qBAAS,EAAE,CAAC;QAC1B,MAAM,CAAC,GAAG,CAAC,CAAC;IACd,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,2BAA2B,EAAE,UAAU,IAAI;QAC5C,IAAI,GAAG,GAAG,IAAI,qBAAS,EAAE,CAAC;QAC1B,IAAI,MAAM,GAAG,EAAE,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC;QACvC,GAAG,CAAC,IAAI,CAAC,UAAU,GAAG;YACpB,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;YAC5B,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QACH,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACnB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0CAA0C,EAAE,UAAU,IAAI;QAC3D,IAAI,GAAG,GAAG,IAAI,qBAAS,EAAE,CAAC;QAC1B,IAAI,MAAM,GAAG,EAAE,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC;QACvC,IAAI,MAAM,GAAG,KAAK,CAAC;QACnB,GAAG,CAAC,SAAS,CAAC,UAAU,SAAS;YAC/B,MAAM,CAAC,SAAS,CAAC,CAAC;YAClB,MAAM,GAAG,IAAI,CAAC;QAChB,CAAC,CAAC,CAAC;QACH,GAAG,CAAC,EAAE,CAAC,MAAM,EAAE;YACb,MAAM,CAAC,MAAM,CAAC,CAAC;YACf,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QACH,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACnB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,qDAAqD,EAAE,UAAU,IAAI;QACtE,IAAI,GAAG,GAAG,IAAI,qBAAS,EAAE,CAAC;QAC1B,GAAG,CAAC,IAAI,CAAC,UAAU,MAAM;YACvB,MAAM,CAAC,MAAM,CAAC,CAAC;YACf,MAAM,CAAC,MAAM,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YAC5B,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;YACvB,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;YAC3B,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;YAChC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;YAC/B,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;YAClC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YAC5C,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YACvC,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACtC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wCAAwC,EAAE,UAAU,IAAI;QACzD,IAAI,MAAM,GAAG,EAAE,CAAC,gBAAgB,CAAC,cAAc,CAAC,CAAC;QACjD,IAAI,GAAG,GAAG,IAAI,qBAAS,EAAE,CAAC;QAC1B,GAAG,CAAC,IAAI,CAAC,UAAU,MAAM;YACvB,MAAM,CAAC,MAAM,CAAC,CAAC;YACf,MAAM,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;YAC1B,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QACH,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACnB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6DAA6D,EAAE,UAAU,IAAI;QAC9E,IAAI,QAAQ,GAAG,SAAS,GAAG,qBAAqB,CAAC;QACjD,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,GAAG,GAAG,IAAI,qBAAS,CAAC;YACtB,OAAO,EAAE,GAAG;SACb,CAAC,CAAC;QACH,GAAG,CAAC,IAAI,CAAC,UAAU,MAAM;YACvB,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,IAAI,KAAK,eAAe,CAAC,CAAC;YAC3C,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,iBAAiB,CAAC,CAAC;YAChD,uBAAuB;YACvB,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IACf,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,sDAAsD,EAAE,UAAU,IAAI;QACvE,IAAI,YAAY,GAAG,SAAS,GAAG,mBAAmB,CAAC;QACnD,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,YAAY,CAAC,CAAC;QAC3C,IAAI,MAAM,GAAO,EAAE,CAAC;QACpB,IAAI,YAAY,GAAG,IAAI,qBAAS,EAAE,CAAC;QACnC,kDAAkD;QAClD,YAAY,CAAC,IAAI,CAAC;YAChB,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YACtC,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QAEH,gEAAgE;QAChE,YAAY,CAAC,SAAS,CAAC,UAAU,SAAS,EAAE,QAAQ;YAClD,KAAK,IAAI,GAAG,IAAI,SAAS,EAAE;gBACzB,IAAI,SAAS,CAAC,cAAc,CAAC,GAAG,CAAC,EAAE;oBACjC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,MAAM,CAAC,GAAG,CAAC,YAAY,KAAK,CAAC,EAAE;wBACnD,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC;qBAClB;oBACD,MAAM,CAAC,GAAG,CAAC,CAAC,QAAQ,CAAC,GAAG,SAAS,CAAC,GAAG,CAAC,CAAC;iBACxC;aACF;QACH,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IACxB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,+CAA+C,EAAE,UAAU,IAAI;QAChE,IAAI,QAAQ,GAAG,SAAS,GAAG,gBAAgB,CAAC;QAC5C,IAAI,IAAI,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC,QAAQ,EAAE,CAAC;QAChD,IAAI,YAAY,GAAG,IAAI,qBAAS,EAAE,CAAC;QACnC,kDAAkD;QAClD,YAAY,CAAC,IAAI,CAAC,UAAU,OAAO;YACjC,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;QAClC,CAAC,CAAC,CAAC;QACH,YAAY,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,UAAU,OAAO;YAClD,MAAM,CAAC,OAAO,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YAC7B,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,iDAAiD,EAAE,UAAU,IAAI;QAClE,IAAI,QAAQ,GAAG,SAAS,GAAG,8BAA8B,CAAC;QAC1D,IAAI,IAAI,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC,QAAQ,EAAE,CAAC;QAChD,IAAI,YAAY,GAAG,IAAI,qBAAS,EAAE,CAAC;QACnC,YAAY,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,SAAS,EAAE,UAAU,GAAG;YACzD,MAAM,CAAC,GAAG,CAAC,CAAC;YACZ,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,EAAE,gBAAgB,CAAC,CAAC;YACxC,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gEAAgE,EAAE,UAAU,IAAI;QACjF,IAAI,QAAQ,GAAG,SAAS,GAAG,gBAAgB,CAAC;QAC5C,IAAI,IAAI,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC,QAAQ,EAAE,CAAC;QAChD,IAAI,YAAY,GAAG,IAAI,qBAAS,EAAE,CAAC;QACnC,kDAAkD;QAClD,YAAY,CAAC,IAAI,CAAC,UAAU,OAAO;YACjC,MAAM,CAAC,OAAO,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YAC7B,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QACH,YAAY,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;IAChC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,qDAAqD,EAAE,UAAU,IAAI;QACtE,IAAI,QAAQ,GAAG,SAAS,GAAG,sBAAsB,CAAC;QAClD,IAAI,IAAI,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC,QAAQ,EAAE,CAAC;QAChD,IAAI,YAAY,GAAG,IAAI,qBAAS,EAAE,CAAC;QACnC,YAAY,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,UAAU,OAAO;YAClD,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,SAAS,KAAK,aAAa,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC;YAErF,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,SAAS,KAAK,eAAe,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC;YACvF,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,yDAAyD,EAAE,UAAU,IAAI;QAC1E,IAAI,QAAQ,GAAG,SAAS,GAAG,uBAAuB,CAAC;QACnD,IAAI,IAAI,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC,QAAQ,EAAE,CAAC;QAChD,IAAI,YAAY,GAAG,IAAI,qBAAS,EAAE,CAAC;QACnC,YAAY,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,UAAU,OAAO;YAClD,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;YACrC,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;YACzC,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,aAAa,CAAC,CAAC;YAC7C,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,gBAAgB,CAAC,CAAC;YAClD,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,MAAM,EAAE,aAAa,CAAC,CAAC;YAC/C,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8BAA8B,EAAE,UAAU,IAAI;QAC/C,IAAI,QAAQ,GAAG,SAAS,GAAG,iBAAiB,CAAC;QAC7C,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,YAAY,GAAG,IAAI,qBAAS,EAAE,CAAC;QACnC,YAAY,CAAC,IAAI,CAAC,UAAU,OAAO;YACjC,MAAM,CAAC,OAAO,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YAC7B,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IACxB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6BAA6B,EAAE,UAAU,IAAI;QAC9C,IAAI,QAAQ,GAAG,SAAS,GAAG,4BAA4B,CAAC;QACxD,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,YAAY,GAAG,IAAI,qBAAS,EAAE,CAAC;QACnC,IAAI,KAAK,GAAG,CAAC,CAAC;QACd,YAAY,CAAC,SAAS,CAAC;YACrB,yBAAyB;YACzB,KAAK,EAAE,CAAC;QACV,CAAC,CAAC,CAAC;QACH,YAAY,CAAC,IAAI,CAAC;YAChB,MAAM,CAAC,KAAK,KAAK,IAAI,CAAC,CAAC;YACvB,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IACxB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gDAAgD,EAAE,UAAU,IAAI;QACjE,IAAI,QAAQ,GAAG,SAAS,GAAG,oBAAoB,CAAC;QAChD,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,YAAY,GAAG,IAAI,qBAAS,CAAC;YAC/B,SAAS,EAAE,IAAI;YACf,SAAS,EAAE;gBACT,SAAS,EAAE,QAAQ;gBACnB,SAAS,EAAE,QAAQ;aACpB;SACF,CAAC,CAAC;QACH,YAAY,CAAC,SAAS,CAAC,UAAU,CAAC;YAChC,MAAM,CAAC,OAAO,CAAC,CAAC,OAAO,KAAK,QAAQ,CAAC,CAAC;YACtC,MAAM,CAAC,OAAO,CAAC,CAAC,OAAO,KAAK,QAAQ,CAAC,CAAC;YACtC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,EAAE,iBAAiB,CAAC,CAAC;YAC9C,MAAM,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,KAAK,OAAO,CAAC,CAAC;YACpC,MAAM,CAAC,CAAC,CAAC,OAAO,KAAK,mBAAmB,CAAC,CAAC;YAC1C,MAAM,CAAC,CAAC,CAAC,OAAO,KAAK,MAAM,CAAC,CAAC;YAC7B,MAAM,CAAC,CAAC,CAAC,OAAO,KAAK,MAAM,CAAC,CAAC;YAC7B,MAAM,CAAC,CAAC,CAAC,OAAO,KAAK,IAAI,CAAC,CAAC;YAC3B,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;YAC7B,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;YAC7B,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,KAAK,KAAK,OAAO,CAAC,CAAC;YACxC,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,KAAK,CAAC,CAAC;QAChC,CAAC,CAAC,CAAC;QACH,YAAY,CAAC,EAAE,CAAC,MAAM,EAAE;YACtB,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IACxB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,kCAAkC,EAAE,UAAU,IAAI;QACnD,IAAI,QAAQ,GAAG,SAAS,GAAG,oBAAoB,CAAC;QAChD,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,YAAY,GAAG,IAAI,qBAAS,CAAC;YAC/B,SAAS,EAAE,KAAK;SACjB,CAAC,CAAC;QACH,YAAY,CAAC,SAAS,CAAC,UAAU,CAAC;YAChC,MAAM,CAAC,OAAO,CAAC,CAAC,OAAO,KAAK,QAAQ,CAAC,CAAC;YACtC,MAAM,CAAC,OAAO,CAAC,CAAC,OAAO,KAAK,QAAQ,CAAC,CAAC;YACtC,MAAM,CAAC,CAAC,CAAC,SAAS,CAAC,KAAK,YAAY,CAAC,CAAC;YACtC,MAAM,CAAC,CAAC,CAAC,SAAS,CAAC,KAAK,iBAAiB,CAAC,CAAC;YAC3C,MAAM,CAAC,CAAC,CAAC,OAAO,KAAK,mBAAmB,CAAC,CAAC;YAC1C,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,SAAS,CAAC,EAAE,mBAAmB,CAAC,CAAC;YAChD,MAAM,CAAC,CAAC,CAAC,SAAS,CAAC,KAAK,MAAM,CAAC,CAAC;YAChC,MAAM,CAAC,CAAC,CAAC,SAAS,CAAC,KAAK,MAAM,CAAC,CAAC;YAChC,MAAM,CAAC,CAAC,CAAC,OAAO,KAAK,MAAM,CAAC,CAAC;YAC7B,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC;YAC/B,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC;YAC/B,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,OAAO,CAAC,CAAC;QAClC,CAAC,CAAC,CAAC;QACH,YAAY,CAAC,IAAI,CAAC;YAChB,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IACxB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,kCAAkC,EAAE,UAAU,IAAI;QACnD,IAAI,QAAQ,GAAG,SAAS,GAAG,4BAA4B,CAAC;QAExD,IAAI,YAAY,GAAG,IAAI,qBAAS,CAAC,EAChC,CAAC,CAAC;QACH,IAAI,KAAK,GAAG,CAAC,CAAC;QACd,YAAY,CAAC,EAAE,CAAC,MAAM,EAAE,UAAU,CAAC;YACjC,KAAK,EAAE,CAAC;QACV,CAAC,CAAC,CAAC;QACH,YAAY,CAAC,EAAE,CAAC,KAAK,EAAE;YACrB,MAAM,CAAC,KAAK,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC;YAC1B,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QACH,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IACxB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,uCAAuC,EAAE,UAAU,IAAI;QACxD,IAAI,QAAQ,GAAG,SAAS,GAAG,iBAAiB,CAAC;QAC7C,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,YAAY,GAAG,IAAI,qBAAS,CAAC;YAC/B,SAAS,EAAE,IAAI;SAChB,CAAC,CAAC;QACH,YAAY,CAAC,SAAS,CAAC,UAAU,CAAC;YAChC,MAAM,CAAC,CAAC,CAAC,MAAM,KAAK,EAAE,CAAC,CAAC;YACxB,MAAM,CAAC,CAAC,CAAC,cAAc,CAAC,KAAK,SAAS,CAAC,CAAC;YACxC,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;IACxB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,iCAAiC,EAAE,UAAU,IAAI;QAClD,IAAI,QAAQ,GAAG,SAAS,GAAG,uBAAuB,CAAC;QACnD,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,aAAG,CAAC,EAAE,WAAW,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;QAC7C,EAAE,CAAC,IAAI,CAAC,UAAU,GAAG;YACnB,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;YACf,MAAM,CAAC,GAAG,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YACzB,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YAC5B,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC;YAC3B,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,KAAK,SAAS,CAAC,CAAC;YAClC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,KAAK,OAAO,CAAC,CAAC;YAChC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,KAAK,IAAI,CAAC,CAAC;YAC7B,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,KAAK,IAAI,CAAC,CAAC;YAC7B,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,wBAAwB,EAAE,UAAU,IAAI;QACzC,IAAI,QAAQ,GAAG,SAAS,GAAG,mBAAmB,CAAC;QAC/C,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,qBAAS,CAAC,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC;QACpD,EAAE,CAAC,IAAI,CAAC,UAAU,GAAG;YACnB,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;YACf,MAAM,CAAC,GAAG,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YACzB,MAAM,CAAC,CAAC,CAAC,MAAM,KAAK,gBAAgB,CAAC,CAAC;YACtC,MAAM,CAAC,CAAC,CAAC,MAAM,KAAK,cAAc,CAAC,CAAC;YACpC,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gCAAgC,EAAE,UAAU,IAAI;QACjD,IAAI,QAAQ,GAAG,SAAS,GAAG,mBAAmB,CAAC;QAC/C,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,qBAAS,CAAC;YAC7B,QAAQ,EAAE,IAAI;YACd,OAAO,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC;SACpB,CAAC,CAAC,CAAC;QACJ,EAAE,CAAC,IAAI,CAAC,UAAU,GAAG;YACnB,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;YACf,MAAM,CAAC,GAAG,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YACzB,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,gBAAgB,CAAC,CAAC;YACjC,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,cAAc,CAAC,CAAC;YAC/B,MAAM,CAAC,CAAC,CAAC,MAAM,KAAK,WAAW,CAAC,CAAC;YACjC,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4DAA4D,EAAE,UAAU,IAAI;QAC7E,IAAI,QAAQ,GAAG,SAAS,GAAG,sBAAsB,CAAC;QAClD,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,qBAAS,CAAC;YAC7B,OAAO,EAAE,EAAE;SACZ,CAAC,CAAC,CAAC;QACJ,EAAE,CAAC,IAAI,CAAC,UAAU,GAAG;YACnB,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;YACf,MAAM,CAAC,GAAG,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YACzB,MAAM,CAAC,CAAC,CAAC,MAAM,KAAK,cAAc,CAAC,CAAC;YACpC,MAAM,CAAC,CAAC,CAAC,MAAM,KAAK,OAAO,CAAC,CAAC;YAC7B,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,6CAA6C,EAAE,UAAU,IAAI;QAC9D,IAAI,QAAQ,GAAG,SAAS,GAAG,2BAA2B,CAAC;QACvD,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,qBAAS,CAAC;YAC7B,QAAQ,EAAE,IAAI;YACd,OAAO,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC;YACxB,SAAS,EAAE,IAAI;SAChB,CAAC,CAAC,CAAC;QACJ,EAAE,CAAC,IAAI,CAAC,UAAU,GAAG;YACnB,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;YAEf,0BAA0B;YAC1B,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,OAAO,CAAC,CAAC;YACxB,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;YACnB,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YACtB,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gFAAgF,EAAE,UAAU,IAAI;QACjG,IAAI,QAAQ,GAAG,SAAS,GAAG,sBAAsB,CAAC;QAClD,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,EAAE,EAAE,aAAa,EAAE,CAAC,EAAE,CAAC,CAAC;QAE7D,IAAI,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,qBAAS,CAAC;YAC7B,IAAI,EAAE,KAAK;SACZ,CAAC,CAAC,CAAC;QACJ,EAAE,CAAC,IAAI,CAAC,UAAU,GAAG;YACnB,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;YACf,MAAM,CAAC,GAAG,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YACzB,MAAM,CAAC,CAAC,CAAC,IAAI,KAAK,KAAK,CAAC,CAAC;YACzB,MAAM,CAAC,CAAC,CAAC,GAAG,KAAK,IAAI,CAAC,CAAC;YACvB,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;YACjC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;YAC/B,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,gFAAgF,EAAE,UAAU,IAAI;QACjG,IAAI,QAAQ,GAAG,SAAS,GAAG,sBAAsB,CAAC;QAClD,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,EAAE,EAAE,aAAa,EAAE,CAAC,EAAE,CAAC,CAAC;QAE7D,IAAI,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,qBAAS,CAAC;YAC7B,IAAI,EAAE,KAAK;SACZ,CAAC,CAAC,CAAC;QACJ,EAAE,CAAC,IAAI,CAAC,UAAU,GAAG;YACnB,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;YACf,MAAM,CAAC,GAAG,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC;YACzB,MAAM,CAAC,CAAC,CAAC,IAAI,KAAK,KAAK,CAAC,CAAC;YACzB,MAAM,CAAC,CAAC,CAAC,GAAG,KAAK,IAAI,CAAC,CAAC;YACvB,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;YACjC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;YAC/B,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,4DAA4D,EAAE,UAAU,IAAI;QAC7E,IAAI,QAAQ,GAAG,SAAS,GAAG,sBAAsB,CAAC;QAClD,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QAEvC,IAAI,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,qBAAS,EAAE,CAAC,CAAC;QAClC,IAAI,WAAW,GAAG,OAAO,CAAC,GAAG,CAAC,UAAU,GAAG;YACzC,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;QAC5B,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,EAAE,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC;QAC1B,EAAE,CAAC,IAAI,CAAC;YACN,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,SAAS,EAAE,CAAC,EAAE,4BAA4B,CAAC,CAAC;YACrE,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0DAA0D,EAAE,UAAU,IAAI;QAC3E,IAAI,QAAQ,GAAG,SAAS,GAAG,mBAAmB,CAAC;QAC/C,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QAEvC,IAAI,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,qBAAS,EAAE,CAAC,CAAC;QAClC,IAAI,WAAW,GAAG,OAAO,CAAC,GAAG,CAAC,UAAU,GAAG;YACzC,MAAM,CAAC,KAAK,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;QAC1B,CAAC,CAAC,CAAC;QACH,EAAE,CAAC,EAAE,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC;QAC1B,EAAE,CAAC,IAAI,CAAC;YACN,MAAM,CAAC,KAAK,CAAC,WAAW,CAAC,SAAS,EAAE,CAAC,EAAE,4BAA4B,CAAC,CAAC;YACrE,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAA;IACJ,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,oDAAoD,EAAE,UAAU,IAAI;QACrE,IAAI,QAAQ,GAAG,SAAS,GAAG,qBAAqB,CAAC;QACjD,IAAI,EAAE,GAAG,EAAE,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,qBAAS,CAAC;YAC7B,IAAI,EAAE,KAAK;SACZ,CAAC,CAAC,CAAC;QACJ,EAAE,CAAC,IAAI,CAAE,UAAU,GAAG;YACpB,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC;YAEf,MAAM,CAAC,GAAG,CAAC,MAAM,KAAG,CAAC,CAAC,CAAC;YACvB,MAAM,CAAC,CAAC,CAAC,IAAI,KAAK,KAAK,CAAC,CAAC;YACzB,MAAM,CAAC,CAAC,CAAC,GAAG,KAAK,IAAI,CAAC,CAAC;YACvB,IAAI,EAAE,CAAC;QACT,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC","sourcesContent":["import {Converter} from \"../src/Converter\";\nimport csv from \"../src\";\nvar assert = require(\"assert\");\nvar fs = require(\"fs\");\nvar sandbox = require(\"sinon\").sandbox.create();\nvar file = __dirname + \"/data/testData\";\nvar trailCommaData = __dirname + \"/data/trailingComma\";\ndescribe(\"CSV Converter\", function () {\n  afterEach(function () {\n    sandbox.restore();\n  });\n\n  it(\"should create new instance of csv\", function () {\n    var obj = new Converter();\n    assert(obj);\n  });\n\n  it(\"should read from a stream\", function (done) {\n    var obj = new Converter();\n    var stream = fs.createReadStream(file);\n    obj.then(function (obj) {\n      assert.equal(obj.length, 2);\n      done();\n    });\n    stream.pipe(obj);\n  });\n\n  it(\"should call onNext once a row is parsed.\", function (done) {\n    var obj = new Converter();\n    var stream = fs.createReadStream(file);\n    var called = false;\n    obj.subscribe(function (resultRow) {\n      assert(resultRow);\n      called = true;\n    });\n    obj.on(\"done\", function () {\n      assert(called);\n      done();\n    });\n    stream.pipe(obj);\n  });\n\n  it(\"should emit end_parsed message once it is finished.\", function (done) {\n    var obj = new Converter();\n    obj.then(function (result) {\n      assert(result);\n      assert(result.length === 2);\n      assert(result[0].date);\n      assert(result[0].employee);\n      assert(result[0].employee.name);\n      assert(result[0].employee.age);\n      assert(result[0].employee.number);\n      assert(result[0].employee.key.length === 2);\n      assert(result[0].address.length === 2);\n      done();\n    });\n    fs.createReadStream(file).pipe(obj);\n  });\n\n  it(\"should handle traling comma gracefully\", function (done) {\n    var stream = fs.createReadStream(trailCommaData);\n    var obj = new Converter();\n    obj.then(function (result) {\n      assert(result);\n      assert(result.length > 0);\n      done();\n    });\n    stream.pipe(obj);\n  });\n\n  it(\"should handle comma in column which is surrounded by qoutes\", function (done) {\n    var testData = __dirname + \"/data/dataWithComma\";\n    var rs = fs.createReadStream(testData);\n    var obj = new Converter({\n      \"quote\": \"#\"\n    });\n    obj.then(function (result) {\n      assert(result[0].col1 === \"\\\"Mini. Sectt\");\n      assert.equal(result[3].col2, \"125001,fenvkdsf\");\n      // console.log(result);\n      done();\n    });\n    rs.pipe(obj);\n  });\n\n  it(\"should be able to convert a csv to column array data\", function (done) {\n    var columArrData = __dirname + \"/data/columnArray\";\n    var rs = fs.createReadStream(columArrData);\n    var result:any = {};\n    var csvConverter = new Converter();\n    //end_parsed will be emitted once parsing finished\n    csvConverter.then(function () {\n      assert(result.TIMESTAMP.length === 5);\n      done();\n    });\n\n    //record_parsed will be emitted each time a row has been parsed.\n    csvConverter.subscribe(function (resultRow, rowIndex) {\n      for (var key in resultRow) {\n        if (resultRow.hasOwnProperty(key)) {\n          if (!result[key] || !(result[key] instanceof Array)) {\n            result[key] = [];\n          }\n          result[key][rowIndex] = resultRow[key];\n        }\n      }\n    });\n    rs.pipe(csvConverter);\n  });\n\n  it(\"should be able to convert csv string directly\", function (done) {\n    var testData = __dirname + \"/data/testData\";\n    var data = fs.readFileSync(testData).toString();\n    var csvConverter = new Converter();\n    //end_parsed will be emitted once parsing finished\n    csvConverter.then(function (jsonObj) {\n      assert.equal(jsonObj.length, 2);\n    });\n    csvConverter.fromString(data).then(function (jsonObj) {\n      assert(jsonObj.length === 2);\n      done();\n    });\n  });\n\n  it(\"should be able to convert csv string with error\", function (done) {\n    var testData = __dirname + \"/data/dataWithUnclosedQuotes\";\n    var data = fs.readFileSync(testData).toString();\n    var csvConverter = new Converter();\n    csvConverter.fromString(data).then(undefined, function (err) {\n      assert(err);\n      assert.equal(err.err, \"unclosed_quote\");\n      done();\n    });\n  });\n\n  it(\"should be able to convert csv string without callback provided\", function (done) {\n    var testData = __dirname + \"/data/testData\";\n    var data = fs.readFileSync(testData).toString();\n    var csvConverter = new Converter();\n    //end_parsed will be emitted once parsing finished\n    csvConverter.then(function (jsonObj) {\n      assert(jsonObj.length === 2);\n      done();\n    });\n    csvConverter.fromString(data);\n  });\n\n  it(\"should be able to handle columns with double quotes\", function (done) {\n    var testData = __dirname + \"/data/dataWithQoutes\";\n    var data = fs.readFileSync(testData).toString();\n    var csvConverter = new Converter();\n    csvConverter.fromString(data).then(function (jsonObj) {\n      assert(jsonObj[0].TIMESTAMP === '13954264\"22', JSON.stringify(jsonObj[0].TIMESTAMP));\n\n      assert(jsonObj[1].TIMESTAMP === 'abc, def, ccc', JSON.stringify(jsonObj[1].TIMESTAMP));\n      done();\n    });\n  });\n\n  it(\"should be able to handle columns with two double quotes\", function (done) {\n    var testData = __dirname + \"/data/twodoublequotes\";\n    var data = fs.readFileSync(testData).toString();\n    var csvConverter = new Converter();\n    csvConverter.fromString(data).then(function (jsonObj) {\n      assert.equal(jsonObj[0].title, \"\\\"\");\n      assert.equal(jsonObj[0].data, \"xyabcde\");\n      assert.equal(jsonObj[0].uuid, \"fejal\\\"eifa\");\n      assert.equal(jsonObj[0].fieldA, \"bnej\\\"\\\"falkfe\");\n      assert.equal(jsonObj[0].fieldB, \"\\\"eisjfes\\\"\");\n      done();\n    });\n  });\n\n  it(\"should handle empty csv file\", function (done) {\n    var testData = __dirname + \"/data/emptyFile\";\n    var rs = fs.createReadStream(testData);\n    var csvConverter = new Converter();\n    csvConverter.then(function (jsonObj) {\n      assert(jsonObj.length === 0);\n      done();\n    });\n    rs.pipe(csvConverter);\n  });\n\n  it(\"should parse large csv file\", function (done) {\n    var testData = __dirname + \"/data/large-csv-sample.csv\";\n    var rs = fs.createReadStream(testData);\n    var csvConverter = new Converter();\n    var count = 0;\n    csvConverter.subscribe(function () {\n      //console.log(arguments);\n      count++;\n    });\n    csvConverter.then(function () {\n      assert(count === 5290);\n      done();\n    });\n    rs.pipe(csvConverter);\n  });\n\n  it(\"should parse data and covert to specific types\", function (done) {\n    var testData = __dirname + \"/data/dataWithType\";\n    var rs = fs.createReadStream(testData);\n    var csvConverter = new Converter({\n      checkType: true,\n      colParser: {\n        \"column6\": \"string\",\n        \"column7\": \"string\"\n      }\n    });\n    csvConverter.subscribe(function (d) {\n      assert(typeof d.column1 === \"number\");\n      assert(typeof d.column2 === \"string\");\n      assert.equal(d[\"colume4\"], \"someinvaliddate\");\n      assert(d.column5.hello === \"world\");\n      assert(d.column6 === '{\"hello\":\"world\"}');\n      assert(d.column7 === \"1234\");\n      assert(d.column8 === \"abcd\");\n      assert(d.column9 === true);\n      assert(d.column10[0] === 23);\n      assert(d.column10[1] === 31);\n      assert(d.column11[0].hello === \"world\");\n      assert(d[\"name#!\"] === false);\n    });\n    csvConverter.on(\"done\", function () {\n      done();\n    });\n    rs.pipe(csvConverter);\n  });\n\n  it(\"should turn off field type check\", function (done) {\n    var testData = __dirname + \"/data/dataWithType\";\n    var rs = fs.createReadStream(testData);\n    var csvConverter = new Converter({\n      checkType: false\n    });\n    csvConverter.subscribe(function (d) {\n      assert(typeof d.column1 === \"string\");\n      assert(typeof d.column2 === \"string\");\n      assert(d[\"column3\"] === \"2012-01-01\");\n      assert(d[\"colume4\"] === \"someinvaliddate\");\n      assert(d.column5 === '{\"hello\":\"world\"}');\n      assert.equal(d[\"column6\"], '{\"hello\":\"world\"}');\n      assert(d[\"column7\"] === \"1234\");\n      assert(d[\"column8\"] === \"abcd\");\n      assert(d.column9 === \"true\");\n      assert(d.column10[0] === \"23\");\n      assert(d.column10[1] === \"31\");\n      assert(d[\"name#!\"] === 'false');\n    });\n    csvConverter.then(function () {\n      done();\n    });\n    rs.pipe(csvConverter);\n  });\n\n  it(\"should emit data event correctly\", function (done) {\n    var testData = __dirname + \"/data/large-csv-sample.csv\";\n\n    var csvConverter = new Converter({\n    });\n    var count = 0;\n    csvConverter.on(\"data\", function (d) {\n      count++;\n    });\n    csvConverter.on(\"end\", function () {\n      assert.equal(count, 5290);\n      done();\n    });\n    var rs = fs.createReadStream(testData);\n    rs.pipe(csvConverter);\n  });\n\n  it(\"should process column with linebreaks\", function (done) {\n    var testData = __dirname + \"/data/lineBreak\";\n    var rs = fs.createReadStream(testData);\n    var csvConverter = new Converter({\n      checkType: true\n    });\n    csvConverter.subscribe(function (d) {\n      assert(d.Period === 13);\n      assert(d[\"Apparent age\"] === \"Unknown\");\n      done();\n    });\n    rs.pipe(csvConverter);\n  });\n\n  it(\"be able to ignore empty columns\", function (done) {\n    var testData = __dirname + \"/data/dataIgnoreEmpty\";\n    var rs = fs.createReadStream(testData);\n    var st = rs.pipe(csv({ ignoreEmpty: true }));\n    st.then(function (res) {\n      var j = res[0];\n      assert(res.length === 3);\n      assert(j.col2.length === 2);\n      assert(j.col2[1] === \"d3\");\n      assert(j.col4.col3 === undefined);\n      assert(j.col4.col5 === \"world\");\n      assert(res[1].col1 === \"d2\");\n      assert(res[2].col1 === \"d4\");\n      done();\n    });\n  });\n\n  it(\"should allow no header\", function (done) {\n    var testData = __dirname + \"/data/noheadercsv\";\n    var rs = fs.createReadStream(testData);\n    var st = rs.pipe(new Converter({ noheader: true }));\n    st.then(function (res) {\n      var j = res[0];\n      assert(res.length === 5);\n      assert(j.field1 === \"CC102-PDMI-001\");\n      assert(j.field2 === \"eClass_5.1.3\");\n      done();\n    });\n  });\n\n  it(\"should allow customised header\", function (done) {\n    var testData = __dirname + \"/data/noheadercsv\";\n    var rs = fs.createReadStream(testData);\n    var st = rs.pipe(new Converter({\n      noheader: true,\n      headers: [\"a\", \"b\"]\n    }));\n    st.then(function (res) {\n      var j = res[0];\n      assert(res.length === 5);\n      assert(j.a === \"CC102-PDMI-001\");\n      assert(j.b === \"eClass_5.1.3\");\n      assert(j.field3 === \"10/3/2014\");\n      done();\n    });\n  });\n\n  it(\"should allow customised header to override existing header\", function (done) {\n    var testData = __dirname + \"/data/complexJSONCSV\";\n    var rs = fs.createReadStream(testData);\n    var st = rs.pipe(new Converter({\n      headers: []\n    }));\n    st.then(function (res) {\n      var j = res[0];\n      assert(res.length === 2);\n      assert(j.field1 === \"Food Factory\");\n      assert(j.field2 === \"Oscar\");\n      done();\n    });\n  });\n\n  it(\"should handle when there is an empty string\", function (done) {\n    var testData = __dirname + \"/data/dataWithEmptyString\";\n    var rs = fs.createReadStream(testData);\n    var st = rs.pipe(new Converter({\n      noheader: true,\n      headers: [\"a\", \"b\", \"c\"],\n      checkType: true\n    }));\n    st.then(function (res) {\n      var j = res[0];\n\n      // assert(res.length===2);\n      assert(j.a === \"green\");\n      assert(j.b === 40);\n      assert.equal(j.c, \"\");\n      done();\n    });\n  });\n\n  it(\"should detect eol correctly when first chunk is smaller than header row length\", function (done) {\n    var testData = __dirname + \"/data/dataNoTrimCRLF\";\n    var rs = fs.createReadStream(testData, { highWaterMark: 3 });\n\n    var st = rs.pipe(new Converter({\n      trim: false\n    }));\n    st.then(function (res) {\n      var j = res[0];\n      assert(res.length === 2);\n      assert(j.name === \"joe\");\n      assert(j.age === \"20\");\n      assert.equal(res[1].name, \"sam\");\n      assert.equal(res[1].age, \"30\");\n      done();\n    });\n  });\n\n  it(\"should detect eol correctly when first chunk ends in middle of CRLF line break\", function (done) {\n    var testData = __dirname + \"/data/dataNoTrimCRLF\";\n    var rs = fs.createReadStream(testData, { highWaterMark: 9 });\n\n    var st = rs.pipe(new Converter({\n      trim: false\n    }));\n    st.then(function (res) {\n      var j = res[0];\n      assert(res.length === 2);\n      assert(j.name === \"joe\");\n      assert(j.age === \"20\");\n      assert.equal(res[1].name, \"sam\");\n      assert.equal(res[1].age, \"30\");\n      done();\n    });\n  });\n\n  it(\"should emit eol event when line ending is detected as CRLF\", function (done) {\n    var testData = __dirname + \"/data/dataNoTrimCRLF\";\n    var rs = fs.createReadStream(testData);\n\n    var st = rs.pipe(new Converter());\n    var eolCallback = sandbox.spy(function (eol) {\n      assert.equal(eol, \"\\r\\n\");\n    });\n    st.on(\"eol\", eolCallback);\n    st.then(function () {\n      assert.equal(eolCallback.callCount, 1, 'should emit eol event once');\n      done();\n    })\n  });\n\n  it(\"should emit eol event when line ending is detected as LF\", function (done) {\n    var testData = __dirname + \"/data/columnArray\";\n    var rs = fs.createReadStream(testData);\n\n    var st = rs.pipe(new Converter());\n    var eolCallback = sandbox.spy(function (eol) {\n      assert.equal(eol, \"\\n\");\n    });\n    st.on(\"eol\", eolCallback);\n    st.then(function () {\n      assert.equal(eolCallback.callCount, 1, 'should emit eol event once');\n      done();\n    })\n  });\n\n  it(\"should remove the Byte Order Mark (BOM) from input\", function (done) {\n    var testData = __dirname + \"/data/dataNoTrimBOM\";\n    var rs = fs.createReadStream(testData);\n    var st = rs.pipe(new Converter({\n      trim: false\n    }));\n    st.then( function (res) {\n      var j = res[0];\n\n      assert(res.length===2);\n      assert(j.name === \"joe\");\n      assert(j.age === \"20\");\n      done();\n    });\n  });\n});\n"]}
|