From 1b720aa80202dd5ac6e8823a197ae6d530ac4b70 Mon Sep 17 00:00:00 2001 From: Robert Nagy Date: Wed, 21 Aug 2019 22:03:11 +0200 Subject: [PATCH] http: free listeners on free sockets Reduced memory usage by ensuring free sockets don't have extra listeners while in the pool. PR-URL: https://github.com/nodejs/node/pull/29259 Reviewed-By: Anna Henningsen Reviewed-By: Matteo Collina Reviewed-By: James M Snell Reviewed-By: Rich Trott Reviewed-By: Ruben Bridgewater --- lib/_http_client.js | 2 ++ test/parallel/test-http-agent-keepalive.js | 18 +++++++++++++++--- 2 files changed, 17 insertions(+), 3 deletions(-) diff --git a/lib/_http_client.js b/lib/_http_client.js index 796216611519ab..e84400377d8c1a 100644 --- a/lib/_http_client.js +++ b/lib/_http_client.js @@ -618,6 +618,8 @@ function responseKeepAlive(req) { } socket.removeListener('close', socketCloseListener); socket.removeListener('error', socketErrorListener); + socket.removeListener('data', socketOnData); + socket.removeListener('end', socketOnEnd); socket.once('error', freeSocketErrorListener); // There are cases where _handle === null. Avoid those. Passing null to // nextTick() will call getDefaultTriggerAsyncId() to retrieve the id. diff --git a/test/parallel/test-http-agent-keepalive.js b/test/parallel/test-http-agent-keepalive.js index 0a85b30b0a42ce..a1f902fab091e1 100644 --- a/test/parallel/test-http-agent-keepalive.js +++ b/test/parallel/test-http-agent-keepalive.js @@ -140,9 +140,21 @@ server.listen(0, common.mustCall(() => { // Check for listener leaks when reusing sockets. function checkListeners(socket) { - assert.strictEqual(socket.listenerCount('data'), 1); - assert.strictEqual(socket.listenerCount('drain'), 1); + const callback = common.mustCall(() => { + if (!socket.destroyed) { + assert.strictEqual(socket.listenerCount('data'), 0); + assert.strictEqual(socket.listenerCount('drain'), 0); + // Sockets have freeSocketErrorListener. + assert.strictEqual(socket.listenerCount('error'), 1); + // Sockets have onReadableStreamEnd. + assert.strictEqual(socket.listenerCount('end'), 1); + } + + socket.off('free', callback); + socket.off('close', callback); + }); assert.strictEqual(socket.listenerCount('error'), 1); - // Sockets have onReadableStreamEnd. assert.strictEqual(socket.listenerCount('end'), 2); + socket.once('free', callback); + socket.once('close', callback); }