Node.js how to pipe gzip stream to writestream

10,368

Solution 1

my example:

async function run(){

var zlib = require('zlib');
var fs = require("fs");

var gz = zlib.createGzip();
gz.on('error', function(err){    console.log(err.stack);   });
// gz.on('finish', function() {    console.log("finished compression, now need to finish writing...");   });

var f = fs.createWriteStream('test.txt.gz');
f.on('error', function(err){    console.log(err.stack);       });
f.on('finish', function() {    console.log("Write success.");   });

gz.pipe(f);

// if stream.write returns false need to wait for drain,
// for example using await for a promise in an async function, or maybe run the callback on drain
if(!gz.write('test','utf-8'))
  await new Promise( (resolve,reject)=> gz.once('drain', resolve) )

gz.end();

console.log("Compress zip file success.");

}

run();

the code in the question had 3 problems

  • pipe returns its argument, not the called-on object
  • need to end the stream, so it will write the data to output
  • better to set up events before they are used

// needded for code in question to run:
function dosomething(){

}
var gz_path="test.txt.gz"
var data="test"


//corrected code

var fs = require("fs");
var zlib = require('zlib');

// pipe retuns its argument, the folowing is correct:
// var f = zlib.createGzip().pipe(fs.createWriteStream(gz_path));

var gz = zlib.createGzip();

gz.pipe(fs.createWriteStream(gz_path));

// it helps to setup events before they fire  
gz.on('error', function(err){
    console.log("wtf error", err);
});

gz.on('finish', function(){
    console.log("write stream is done");
    dosomething();
});

gz.write(data);

// need to end stream for data to be written
gz.end()

Solution 2

import { createReadStream, createWriteStream } from 'fs'
import { createGzip } from 'zlib'

const filename = yourFile
// opens the file as a readable stream
createReadStream(filename)
   .pipe(createGzip())
   .pipe(createWriteStream(`${filename}.gz`))
   .on('finish', () => console.log('File compressed'))

Solution 3

try

var zlib = require('zlib');
var stream = require('stream');
var util = require('util');
var fs = require('fs');

var gz = zlib.createGzip();

function StringifyStream(){
    stream.Transform.call(this);

    this._readableState.objectMode = false;
    this._writableState.objectMode = true;
}
util.inherits(StringifyStream, stream.Transform);

StringifyStream.prototype._transform = function(obj, encoding, cb){
    this.push(JSON.stringify(obj));
    cb();
};


var data = "some data in here";

var rs = new stream.Readable({ objectMode: true });
rs.push(data);
rs.push(null);


rs.pipe(new StringifyStream())
  .pipe(gz)
  .pipe(fs.createWriteStream('test.gz'))
  .on('error', function(err){
    console.log("wtf error", err);
  })
  .on('finish', function(){
  console.log("write stream is done");
  // dosomething();
});
Share:
10,368
kevzettler
Author by

kevzettler

web developer by day lead singer in punk band by night

Updated on July 26, 2022

Comments

  • kevzettler
    kevzettler almost 2 years

    I can't seem to get this to work. I want to write some data to a gzip stream and then pipe the gzip stream to a file write stream. I want to then call a function when the files done writing. I currently have:

      var gz = zlib.createGzip()
                   .pipe(fs.createWriteStream(gz_path));
    
      gz.write(data);
    
      gz.on('error', function(err){
        console.log("wtf error", err);
      });
    
      gz.on('finish', function(){
        console.log("write stream is done");
        dosomething();
      });
    

    The finish event or the error event is never called.

    • user568109
      user568109 over 9 years
      Try closing the gz stream with end() after finished writing.
  • kevzettler
    kevzettler over 9 years
    No luck. Would errors form the fs.createWriteStream bubble up to the gz.on error?
  • Kamrul
    Kamrul over 9 years
    hi sorry, i did not realised that you have not converted the stream properly. Please try the code now.
  • h-kippo
    h-kippo over 3 years
    it's worth noting that gz can be used only once. After it has been closed, no more data can be written to it. Therefore it's a often better to initialize gz during stream creation, not once for the whole file