======
This is a fork of https://github.com/daspawn/xlsx-stream-reader#readme
I upgraded some packages and added extra config options to allow you to choose your tmp folder and ignore sheets based on their number.
this is usefull for AWS LAMBDA where /tmp only has 512mb, so you can mount an EFS volume to /mnt/tmp and use that directory with unlimited space.
Memory efficinet minimalist streaming XLSX reader that can handle piped streams as input. Events are emmited while reading the stream.
Example
More examples can be found if example
folder
var stream = new XlsxStreamReader({
verbose: false,
formatting: false
})
Options
Key | Default Value | Description |
---|---|---|
verbose | true | throw additional exceptions, if false - then pass empty string in that places |
formatting | true | should cells with combined formats be formatted or not |
saxTrim | true | whether or not to trim text and comment nodes |
tmpOptions | {} | an options object to be passed to the "tmp" package, eg. to use a folder other than OS tmp directory { tmpdir: "/mnt/tmp" } |
sheetNumbers | [] | Array with the sheet number to process. This prevents writing to /tmp the sheets what are not needed. Eg. if [1,4] only events for sheets 1 and 4 will be returned. Use only if you're sure the xlsx file will have its internal naming compatible with the order of the sheets. This is not guaranteed by xlsx spec but seems to be followed by excel and most libs that write xlsx, eg. sheet1.xml might represent sheet 3 but 99.99% of the time it will be sheet 1. |
enableCompression | false | If true it used gzip when writing the sheets to /tmp its unclear if this improves or worsens performance, it definaetly requires more cpu/memory, but also less IO to disk. |
const XlsxStreamReader = require("xlsx-stream-reader");
var workBookReader = new XlsxStreamReader();
workBookReader.on('error', function (error) {
throw(error);
});
workBookReader.on('sharedStrings', function () {
// do not need to do anything with these,
// cached and used when processing worksheets
console.log(workBookReader.workBookSharedStrings);
});
workBookReader.on('styles', function () {
// do not need to do anything with these
// but not currently handled in any other way
console.log(workBookReader.workBookStyles);
});
workBookReader.on('worksheet', function (workSheetReader) {
if (workSheetReader.id > 1){
// we only want first sheet
workSheetReader.skip();
return;
}
// print worksheet name
console.log(workSheetReader.name);
// if we do not listen for rows we will only get end event
// and have infor about the sheet like row count
workSheetReader.on('row', function (row) {
if (row.attributes.r == 1){
// do something with row 1 like save as column names
}else{
// second param to forEach colNum is very important as
// null columns are not defined in the array, ie sparse array
row.values.forEach(function(rowVal, colNum){
// do something with row values
});
}
});
workSheetReader.on('end', function () {
console.log(workSheetReader.rowCount);
});
// call process after registering handlers
workSheetReader.process();
});
workBookReader.on('end', function () {
// end of workbook reached
});
fs.createReadStream(fileName).pipe(workBookReader);
Beta Warning
This module is currently in use on a live internal business system for product management. That being said this should still be considered beta. More usage and input from users will be needed due to the numerous differences/incompatibilities/flukes I have already run into with XLSX files.
Limitations
The row reader currently returns stored values for formulas (these are normally available) and does not calculate the formula itself. As time permits the row handler will be more capable but was enough for currrent purposes (loading values from large worksheets fast)
Inspiration
Need a simple XLSX file streaming reader to handle large excel sheets but only one available/compatible was by guyonroche/exceljs. The stream reader module at the time was unfinished/unusable and rewrite attempts exposed column shifting I could not solve
More Information
Events are emmited as pertinent parts of the workbook and worksheet are receieved in the stream. Theoretically you could pause the input stream if events are being receieved too fast but this has not been tested
Events can potentially (even though I have not seen it) be receieved out of order,
if you receive a worksheet end event while still receieving rows be sure to make sure
your number of rows receieved equals the workSheetReader.rowCount
Theoretically you could process an excel sheet as it is being uploaded, depending on the sheet type, but untried (I encountered some XLSX files that have a different zip format that requires having the entire file to read the archive contents properly), but still probably better to save temp first and read streasm from there.
Currently if the zip archive does not have the shared strings at the begining of the archive then the input stream for each sheet is pied into a temp file until the shared string are encountered and processed, then re-read the temp worksheets with the shared strings.
API Information
Create a new XlsxStreamReader object (workBookReader). After attaching handlers you
can pipe()
your input stream into the reader to begin processing
error
{Error Object}
Emitted if there was an error in processing (may not catch all errors, some may be thrown depending on where the error happened)
Emmitted once the XLSX zip parser has closed and all sheets have been processed
After the workbook shared strings have been parsed this event is emmited. Shared strings
are available via array workBookReader.workBookSharedStrings
.
After the workbook styles have been parsed this event is emmited. Styles are available
via array workBookReader.workBookStyles
workSheetReader
{Object} XlsxStreamReaderWorkSheet object
Emmitted when a worksheet is reached. The sheet number is availble via
{Number} workSheetReader.id
. You can either process or skip at this point,
but you must do one for the processing to the next sheet to continue/finish.
Once event is recieved you can attach worksheet on handlers (end, row) then you
would workSheetReader.process()
. If you do not want to process a sheet and instead
want to skip entirely, you would workSheetReader.skip()
without attaching any handlers.
Emmitted once the end of the worksheet has been reached. The row count is
available via {Number} workSheetReader.rowCount
row
{Object} Row object
Emmitted on every row encountered in the worksheet. for more details on what is in the row object attributes, see the Row class on MSDN.
For example:
row.values
: sparse array containing all cell valuesrow.formulas
: sparse array containing all cell formulasrow.attributes.r
: row indexrow.attributes.ht
: Row height measured in point sizerow.attributes.customFormat
: '1' if the row style should be applied.row.attributes.hidden
: '1' if the row is hidden
References
Used Modules
Authors
Written by Brian Taber and Kirill Husyatin
License