Skip to content

Commit d6a90b7

Browse files
committed
http: emit 'drain' on OutgoingMessage only after buffers drain
Previously, socketOnDrain could be invoked synchronously from _flushOutput (via _onPendingData -> updateOutgoingData) while the bytes just handed to the socket were still buffered and while outputSize had not yet been reset on the OutgoingMessage. The 'drain' event fired even though res.writableLength was non-zero, breaking the invariant a user would reasonably expect after `while (!res.write(...));`. Gate the emission in socketOnDrain on msg.writableLength === 0 (which also covers outputSize + chunked buffer + socket.writableLength), and apply the same check in OutgoingMessage._flush so that 'drain' is only emitted when the response is genuinely drained. The socket's own 'drain' event will otherwise propagate through socketOnDrain when the socket buffer actually empties. Signed-off-by: Robert Nagy <[email protected]> Assisted-by: Claude Opus 4.6 (1M context) <[email protected]>
1 parent 2030fd3 commit d6a90b7

3 files changed

Lines changed: 67 additions & 3 deletions

File tree

lib/_http_outgoing.js

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -1139,12 +1139,12 @@ OutgoingMessage.prototype._flush = function _flush() {
11391139

11401140
if (socket?.writable) {
11411141
// There might be remaining data in this.output; write it out
1142-
const ret = this._flushOutput(socket);
1142+
this._flushOutput(socket);
11431143

11441144
if (this.finished) {
11451145
// This is a queue to the server or client to bring in the next this.
11461146
this._finish();
1147-
} else if (ret && this[kNeedDrain]) {
1147+
} else if (this[kNeedDrain] && this.writableLength === 0) {
11481148
this[kNeedDrain] = false;
11491149
this.emit('drain');
11501150
}

lib/_http_server.js

Lines changed: 7 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -817,7 +817,13 @@ function socketOnDrain(socket, state) {
817817
}
818818

819819
const msg = socket._httpMessage;
820-
if (msg && !msg.finished && msg[kNeedDrain]) {
820+
// Only emit 'drain' once the message has no data pending anywhere, so that
821+
// msg.writableLength === 0 when the event fires. socketOnDrain is called
822+
// synchronously from updateOutgoingData during _flushOutput, at which point
823+
// the bytes we just handed to the socket (or the stale outputSize) mean
824+
// the message is not actually drained yet - we wait for the socket's
825+
// own 'drain' event instead.
826+
if (msg && !msg.finished && msg[kNeedDrain] && msg.writableLength === 0) {
821827
msg[kNeedDrain] = false;
822828
msg.emit('drain');
823829
}
Lines changed: 58 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
'use strict';
2+
// Regression test: when a pipelined ServerResponse (whose writes were
3+
// buffered in outputData while the socket belonged to a previous response)
4+
// is finally assigned its socket and flushed, 'drain' must not be emitted
5+
// until the socket's own buffer has actually drained. Previously,
6+
// socketOnDrain was called synchronously from _flushOutput via _onPendingData
7+
// and emitted 'drain' even though the bytes we just wrote were still sitting
8+
// in the socket's writable buffer, so res.writableLength was non-zero.
9+
10+
const common = require('../common');
11+
const http = require('http');
12+
const net = require('net');
13+
const assert = require('assert');
14+
15+
let step = 0;
16+
17+
const server = http.createServer(common.mustCall((req, res) => {
18+
step++;
19+
20+
if (step === 1) {
21+
// Keep the first response open briefly so the second is queued with
22+
// res.socket === null.
23+
res.writeHead(200, { 'Content-Type': 'text/plain' });
24+
setTimeout(() => res.end('ok'), 50);
25+
return;
26+
}
27+
28+
// Second (pipelined) response - queued in state.outgoing, no socket yet.
29+
assert.strictEqual(res.socket, null);
30+
31+
res.writeHead(200, { 'Content-Type': 'text/plain' });
32+
33+
// Write past the response's highWaterMark so res.write() returns false
34+
// and kNeedDrain is set. Data is buffered in outputData.
35+
const chunk = Buffer.alloc(16 * 1024, 'x');
36+
while (res.write(chunk));
37+
assert.strictEqual(res.writableNeedDrain, true);
38+
39+
res.on('drain', common.mustCall(() => {
40+
assert.strictEqual(
41+
res.writableLength, 0,
42+
`'drain' fired with writableLength=${res.writableLength}`,
43+
);
44+
res.end();
45+
server.close();
46+
}));
47+
}, 2));
48+
49+
server.listen(0, common.mustCall(function() {
50+
const port = this.address().port;
51+
const client = net.connect(port);
52+
client.write(
53+
`GET /1 HTTP/1.1\r\nHost: localhost:${port}\r\n\r\n` +
54+
`GET /2 HTTP/1.1\r\nHost: localhost:${port}\r\n\r\n`,
55+
);
56+
client.resume();
57+
client.on('error', () => {});
58+
}));

0 commit comments

Comments
 (0)