From 6723582b654e72a6bc112ca8957afcc9066ca506 Mon Sep 17 00:00:00 2001 From: Tomas Della Vedova Date: Sat, 29 Oct 2022 16:06:12 +0200 Subject: [PATCH] Clean up socket in case of error (#98) --- .github/workflows/build.yml | 6 +++- index.js | 2 ++ package.json | 2 +- test/hang-socket/http.js | 67 +++++++++++++++++++++++++++++++++++++ test/hang-socket/https.js | 67 +++++++++++++++++++++++++++++++++++++ test/hang-socket/runner.sh | 6 ++++ 6 files changed, 148 insertions(+), 2 deletions(-) create mode 100644 test/hang-socket/http.js create mode 100644 test/hang-socket/https.js create mode 100755 test/hang-socket/runner.sh diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 44ad1b0..b6177f4 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -27,6 +27,10 @@ jobs: run: | npm install - - name: Test + - name: Test / 1 run: | npm run test-ci + + - name: Test / 2 + run: | + ./test/hang-socket/runner.sh diff --git a/index.js b/index.js index 3484628..29913b1 100644 --- a/index.js +++ b/index.js @@ -43,6 +43,7 @@ class HttpProxyAgent extends http.Agent { if (response.statusCode === 200) { callback(null, socket) } else { + socket.destroy() callback(new Error(`Bad response: ${response.statusCode}`), null) } }) @@ -101,6 +102,7 @@ class HttpsProxyAgent extends https.Agent { const secureSocket = super.createConnection({ ...options, socket }) callback(null, secureSocket) } else { + socket.destroy() callback(new Error(`Bad response: ${response.statusCode}`), null) } }) diff --git a/package.json b/package.json index 9d4d296..9479cd1 100644 --- a/package.json +++ b/package.json @@ -13,7 +13,7 @@ "./*": "./*.js" }, "scripts": { - "test": "standard && NODE_EXTRA_CA_CERTS=test/fixtures/certs_unit_test.pem ava -v test/*.test.js && tsd", + "test": "standard && NODE_EXTRA_CA_CERTS=test/fixtures/certs_unit_test.pem ava -v test/*.test.js && NODE_EXTRA_CA_CERTS=test/fixtures/certs_unit_test.pem ./test/hang-socket/runner.sh && tsd", "test-ci": "standard && ava -v test/*.test.js && tsd" }, "engines": { diff --git a/test/hang-socket/http.js b/test/hang-socket/http.js new file mode 100644 index 0000000..db90af1 --- /dev/null +++ b/test/hang-socket/http.js @@ -0,0 +1,67 @@ +'use strict' + +const http = require('http') +const { createServer, SERVER_HOSTNAME } = require('../utils') +const { HttpProxyAgent } = require('../../') + +function request (opts) { + return new Promise((resolve, reject) => { + const req = http.request(opts, resolve) + req.on('error', reject) + req.end(opts.body) + }) +} + +const timeout = setTimeout(() => { + console.log('The http agent is not cleaning up hanging sockets') + process.exit(1) +}, 5000) + +async function run () { + const server = await createServer() + server.on('connect', (request, socket, head) => { + socket.on('end', () => { + clearTimeout(timeout) + }) + const lines = [ + 'HTTP/1.1 403 FORBIDDEN', + '', + 'Forbidden' + ] + socket.write(lines.join('\r\n')) + }) + + const agent = new HttpProxyAgent({ + keepAlive: true, + keepAliveMsecs: 1000, + maxSockets: 256, + maxFreeSockets: 256, + scheduling: 'lifo', + timeout: 500, + proxy: `http://${SERVER_HOSTNAME}:${server.address().port}` + }) + + try { + await request({ + method: 'GET', + hostname: 'www.example.com', + port: '', + path: '/', + agent + }) + console.error(new Error('Should throw')) + process.exit(1) + } catch (err) { + if (err.message !== 'Bad response: 403') { + console.error(new Error('Expected a different error')) + process.exit(1) + } + } + + server.close() +} + +run().catch(err => { + console.error(err) + process.exit(1) +}) diff --git a/test/hang-socket/https.js b/test/hang-socket/https.js new file mode 100644 index 0000000..cc57a6e --- /dev/null +++ b/test/hang-socket/https.js @@ -0,0 +1,67 @@ +'use strict' + +const https = require('https') +const { createSecureServer, SERVER_HOSTNAME } = require('../utils') +const { HttpsProxyAgent } = require('../../') + +function request (opts) { + return new Promise((resolve, reject) => { + const req = https.request(opts, resolve) + req.on('error', reject) + req.end(opts.body) + }) +} + +const timeout = setTimeout(() => { + console.log('The https agent is not cleaning up hanging sockets') + process.exit(1) +}, 5000) + +async function run () { + const server = await createSecureServer() + server.on('connect', (request, socket, head) => { + socket.on('end', () => { + clearTimeout(timeout) + }) + const lines = [ + 'HTTP/1.1 403 FORBIDDEN', + '', + 'Forbidden' + ] + socket.write(lines.join('\r\n')) + }) + + const agent = new HttpsProxyAgent({ + keepAlive: true, + keepAliveMsecs: 1000, + maxSockets: 256, + maxFreeSockets: 256, + scheduling: 'lifo', + timeout: 500, + proxy: `https://${SERVER_HOSTNAME}:${server.address().port}` + }) + + try { + await request({ + method: 'GET', + hostname: 'www.example.com', + port: '', + path: '/', + agent + }) + console.error(new Error('Should throw')) + process.exit(1) + } catch (err) { + if (err.message !== 'Bad response: 403') { + console.error(new Error('Expected a different error, got:', err.message)) + process.exit(1) + } + } + + server.close() +} + +run().catch(err => { + console.error(err) + process.exit(1) +}) diff --git a/test/hang-socket/runner.sh b/test/hang-socket/runner.sh new file mode 100755 index 0000000..c99e879 --- /dev/null +++ b/test/hang-socket/runner.sh @@ -0,0 +1,6 @@ +#!/usr/bin/env bash + +set -euo pipefail + +node test/hang-socket/http.js +node test/hang-socket/https.js