Browse Source

fs: pass err to callback if buffer is too big

In fs.readFile, if an encoding is specified and toString fails, do not
throw an error. Instead, pass the error to the callback.

Fixes: https://github.com/nodejs/node/issues/2767
PR-URL: https://github.com/nodejs/node/pull/3485
Reviewed-By: James M Snell <jasnell@gmail.com>
Reviewed-By: Trevor Norris <trev.norris@gmail.com>
process-exit-stdio-flushing
Evan Lucas 9 years ago
parent
commit
b6207906c4
  1. 18
      lib/fs.js
  2. 52
      test/parallel/test-fs-readfile-tostring-fail.js

18
lib/fs.js

@ -395,12 +395,24 @@ function readFileAfterClose(err) {
else else
buffer = context.buffer; buffer = context.buffer;
if (context.encoding) if (err) return callback(err, buffer);
buffer = buffer.toString(context.encoding);
callback(err, buffer); if (context.encoding) {
return tryToString(buffer, context.encoding, callback);
} }
callback(null, buffer);
}
function tryToString(buf, encoding, callback) {
var e;
try {
buf = buf.toString(encoding);
} catch (err) {
e = err;
}
callback(e, buf);
}
fs.readFileSync = function(path, options) { fs.readFileSync = function(path, options) {
if (!options) { if (!options) {

52
test/parallel/test-fs-readfile-tostring-fail.js

@ -0,0 +1,52 @@
'use strict';
const common = require('../common');
const assert = require('assert');
const fs = require('fs');
const path = require('path');
const kStringMaxLength = process.binding('buffer').kStringMaxLength;
common.refreshTmpDir();
const file = path.join(common.tmpDir, 'toobig.txt');
const stream = fs.createWriteStream(file, {
flags: 'a'
});
const size = kStringMaxLength / 200;
const a = new Buffer(size).fill('a');
for (var i = 0; i < 201; i++) {
stream.write(a);
}
stream.end();
stream.on('finish', common.mustCall(function() {
// make sure that the toString does not throw an error
fs.readFile(file, 'utf8', common.mustCall(function(err, buf) {
assert.ok(err instanceof Error);
assert.strictEqual('toString failed', err.message);
}));
}));
function destroy() {
try {
fs.unlinkSync(file);
} catch (err) {
// it may not exist
}
}
process.on('exit', destroy);
process.on('SIGINT', function() {
destroy();
process.exit();
});
// To make sure we don't leave a very large file
// on test machines in the event this test fails.
process.on('uncaughtException', function(err) {
destroy();
throw err;
});
Loading…
Cancel
Save