Skip to content

Commit dded482

Browse files
Karl BöhlmarkFishrock123
Karl Böhlmark
authored andcommitted
http: remove stale timeout listeners
In order to prevent a memory leak when using keep alive, ensure that the timeout listener for the request is removed when the response has ended. PR-URL: #9440 Reviewed-By: Evan Lucas <evanlucas@me.com> Reviewed-By: James M Snell <jasnell@gmail.com> Reviewed-By: Ben Noordhuis <info@bnoordhuis.nl> Reviewed-By: Colin Ihrig <cjihrig@gmail.com>
1 parent 53e8e96 commit dded482

File tree

2 files changed

+51
-1
lines changed

2 files changed

+51
-1
lines changed

lib/_http_client.js

+7-1
Original file line numberDiff line numberDiff line change
@@ -562,7 +562,13 @@ function tickOnSocket(req, socket) {
562562
socket.on('close', socketCloseListener);
563563

564564
if (req.timeout) {
565-
socket.once('timeout', () => req.emit('timeout'));
565+
const emitRequestTimeout = () => req.emit('timeout');
566+
socket.once('timeout', emitRequestTimeout);
567+
req.once('response', (res) => {
568+
res.once('end', () => {
569+
socket.removeListener('timeout', emitRequestTimeout);
570+
});
571+
});
566572
}
567573
req.emit('socket', socket);
568574
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
'use strict';
2+
const common = require('../common');
3+
const http = require('http');
4+
const assert = require('assert');
5+
6+
const agent = new http.Agent({ keepAlive: true });
7+
8+
const server = http.createServer((req, res) => {
9+
res.end('');
10+
});
11+
12+
const options = {
13+
agent,
14+
method: 'GET',
15+
port: undefined,
16+
host: common.localhostIPv4,
17+
path: '/',
18+
timeout: common.platformTimeout(100)
19+
};
20+
21+
server.listen(0, options.host, common.mustCall(() => {
22+
options.port = server.address().port;
23+
doRequest(common.mustCall((numListeners) => {
24+
assert.strictEqual(numListeners, 1);
25+
doRequest(common.mustCall((numListeners) => {
26+
assert.strictEqual(numListeners, 1);
27+
server.close();
28+
agent.destroy();
29+
}));
30+
}));
31+
}));
32+
33+
function doRequest(cb) {
34+
http.request(options, common.mustCall((response) => {
35+
const sockets = agent.sockets[`${options.host}:${options.port}:`];
36+
assert.strictEqual(sockets.length, 1);
37+
const socket = sockets[0];
38+
const numListeners = socket.listeners('timeout').length;
39+
response.resume();
40+
response.once('end', common.mustCall(() => {
41+
process.nextTick(cb, numListeners);
42+
}));
43+
})).end();
44+
}

0 commit comments

Comments
 (0)