stream: Don't stop reading on zero-length decoded output

Fixes regression introduced in 7e1cf84c9e
pull/24504/head
isaacs 2013-01-31 13:33:37 -08:00
parent 7e1cf84c9e
commit a6c18472cd
2 changed files with 88 additions and 55 deletions

View File

@ -291,9 +291,6 @@ function onread(stream, er, chunk) {
return;
}
if (state.decoder)
chunk = state.decoder.write(chunk);
// at this point, if we got a zero-length buffer or string,
// and we're not in object-mode, then there's really no point
// continuing. it means that there is nothing to read right
@ -305,6 +302,9 @@ function onread(stream, er, chunk) {
0 === chunk.length)
return;
if (state.decoder)
chunk = state.decoder.write(chunk);
// update the buffer info.
state.length += state.objectMode ? 1 : chunk.length;
state.buffer.push(chunk);

View File

@ -24,62 +24,95 @@ var assert = require('assert');
var Readable = require('stream').Readable;
var r = new Readable();
test1();
test2();
// should not end when we get a Buffer(0) or '' as the _read result
// that just means that there is *temporarily* no data, but to go
// ahead and try again later.
//
// note that this is very unusual. it only works for crypto streams
// because the other side of the stream will call read(0) to cycle
// data through openssl. that's why we set the timeouts to call
// r.read(0) again later, otherwise there is no more work being done
// and the process just exits.
function test1() {
var r = new Readable();
var buf = new Buffer(5);
buf.fill('x');
var reads = 5;
r._read = function(n, cb) {
switch (reads--) {
case 0:
return cb(null, null); // EOF
case 1:
return cb(null, buf);
case 2:
setTimeout(r.read.bind(r, 0), 10);
return cb(null, new Buffer(0)); // Not-EOF!
case 3:
setTimeout(r.read.bind(r, 0), 10);
return process.nextTick(function() {
return cb(null, new Buffer(0));
});
case 4:
setTimeout(r.read.bind(r, 0), 10);
return setTimeout(function() {
return cb(null, new Buffer(0));
});
case 5:
return setTimeout(function() {
// should not end when we get a Buffer(0) or '' as the _read result
// that just means that there is *temporarily* no data, but to go
// ahead and try again later.
//
// note that this is very unusual. it only works for crypto streams
// because the other side of the stream will call read(0) to cycle
// data through openssl. that's why we set the timeouts to call
// r.read(0) again later, otherwise there is no more work being done
// and the process just exits.
var buf = new Buffer(5);
buf.fill('x');
var reads = 5;
r._read = function(n, cb) {
switch (reads--) {
case 0:
return cb(null, null); // EOF
case 1:
return cb(null, buf);
});
default:
throw new Error('unreachable');
case 2:
setTimeout(r.read.bind(r, 0), 10);
return cb(null, new Buffer(0)); // Not-EOF!
case 3:
setTimeout(r.read.bind(r, 0), 10);
return process.nextTick(function() {
return cb(null, new Buffer(0));
});
case 4:
setTimeout(r.read.bind(r, 0), 10);
return setTimeout(function() {
return cb(null, new Buffer(0));
});
case 5:
return setTimeout(function() {
return cb(null, buf);
});
default:
throw new Error('unreachable');
}
};
var results = [];
function flow() {
var chunk;
while (null !== (chunk = r.read()))
results.push(chunk + '');
}
};
r.on('readable', flow);
r.on('end', function() {
results.push('EOF');
});
flow();
var results = [];
function flow() {
var chunk;
while (null !== (chunk = r.read()))
results.push(chunk + '');
process.on('exit', function() {
assert.deepEqual(results, [ 'xxxxx', 'xxxxx', 'EOF' ]);
console.log('ok');
});
}
r.on('readable', flow);
r.on('end', function() {
results.push('EOF');
});
flow();
process.on('exit', function() {
assert.deepEqual(results, [ 'xxxxx', 'xxxxx', 'EOF' ]);
console.log('ok');
});
function test2() {
var r = new Readable({ encoding: 'base64' });
var reads = 5;
r._read = function(n, cb) {
if (!reads--)
return cb(null, null); // EOF
else
return cb(null, new Buffer('x'));
};
var results = [];
function flow() {
var chunk;
while (null !== (chunk = r.read()))
results.push(chunk + '');
}
r.on('readable', flow);
r.on('end', function() {
results.push('EOF');
});
flow();
process.on('exit', function() {
assert.deepEqual(results, [ 'eHh4', 'eHg=', 'EOF' ]);
console.log('ok');
});
}