I’m mucking about with node.js and have discovered two ways of reading a file and sending it down the wire, once I’ve established that it exists and have sent the proper MIME type with writeHead:
// read the entire file into memory and then spit it out
fs.readFile(filename, function(err, data){
if (err) throw err;
response.write(data, 'utf8');
response.end();
});
// read and pass the file as a stream of chunks
fs.createReadStream(filename, {
'flags': 'r',
'encoding': 'binary',
'mode': 0666,
'bufferSize': 4 * 1024
}).addListener( "data", function(chunk) {
response.write(chunk, 'binary');
}).addListener( "close",function() {
response.end();
});
Am I correct in assuming that fs.createReadStream might provide a better user experience if the file in question was something large, like a video? It feels like it might be less block-ish; is this true? Are there other pros, cons, caveats, or gotchas I need to know?
A better approach, if you are just going to hook up “data” to “write()” and “close” to “end()”:
The
read.pipe(write)orsys.pump(read, write)approach has the benefit of also adding flow control. So, if the write stream cannot accept data as quickly, it’ll tell the read stream to back off, so as to minimize the amount of data getting buffered in memory.The
flags:"r"andmode:0666are implied by the fact that it is aFileReadStream. Thebinaryencoding is deprecated — if an encoding is not specified, it’ll just work with the raw data buffers.Also, you could add some other goodies that will make your file serving a whole lot slicker:
req.headers.rangeand see if it matches a string like/bytes=([0-9]+)-([0-9]+)/. If so, you want to just stream from that start to end location. (Missing number means 0 or “the end”.)304 Not Modified.if-modified-sinceheader against themtimedate on the stat object. 304 if it wasn’t modified since the date provided.Also, in general, if you can, send a
Content-Lengthheader. (You’restat-ing the file, so you should have this.)