Skip to content

Commit

Permalink
Allow SOCKS proxies to be chained
Browse files Browse the repository at this point in the history
  • Loading branch information
codetheweb committed Nov 18, 2023
1 parent b133295 commit 76b52a8
Show file tree
Hide file tree
Showing 2 changed files with 100 additions and 26 deletions.
83 changes: 61 additions & 22 deletions packages/socks-proxy-agent/src/index.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,9 @@
import { SocksClient, SocksProxy, SocksClientOptions } from 'socks';
import {
SocksClient,
SocksProxy,
SocksClientOptions,
SocksClientChainOptions,
} from 'socks';
import { Agent, AgentConnectOpts } from 'agent-base';
import createDebug from 'debug';
import * as dns from 'dns';
Expand Down Expand Up @@ -87,18 +92,31 @@ export class SocksProxyAgent extends Agent {
'socks5h',
] as const;

readonly shouldLookup: boolean;
readonly proxy: SocksProxy;
readonly shouldLookup!: boolean;
readonly proxies: SocksProxy[];
timeout: number | null;

constructor(uri: string | URL, opts?: SocksProxyAgentOptions) {
constructor(
uri: string | URL | string[] | URL[],
opts?: SocksProxyAgentOptions
) {
super(opts);

const url = typeof uri === 'string' ? new URL(uri) : uri;
const { proxy, lookup } = parseSocksURL(url);
const uriList = Array.isArray(uri) ? uri : [uri];

if (uriList.length === 0) {
throw new Error('At least one proxy server URI must be specified.');
}

this.proxies = [];
for (const [i, uri] of uriList.entries()) {
const { proxy, lookup } = parseSocksURL(new URL(uri.toString()));
this.proxies.push(proxy);
if (i === 0) {
this.shouldLookup = lookup;
}
}

this.shouldLookup = lookup;
this.proxy = proxy;
this.timeout = opts?.timeout ?? null;
}

Expand All @@ -110,7 +128,7 @@ export class SocksProxyAgent extends Agent {
req: http.ClientRequest,
opts: AgentConnectOpts
): Promise<net.Socket> {
const { shouldLookup, proxy, timeout } = this;
const { shouldLookup, proxies, timeout } = this;

if (!opts.host) {
throw new Error('No `host` defined!');
Expand All @@ -133,25 +151,46 @@ export class SocksProxyAgent extends Agent {
});
}

const socksOpts: SocksClientOptions = {
proxy,
destination: {
host,
port: typeof port === 'number' ? port : parseInt(port, 10),
},
command: 'connect',
timeout: timeout ?? undefined,
};

let socket: net.Socket;
const cleanup = (tlsSocket?: tls.TLSSocket) => {
req.destroy();
socket.destroy();
if (tlsSocket) tlsSocket.destroy();
};

debug('Creating socks proxy connection: %o', socksOpts);
const { socket } = await SocksClient.createConnection(socksOpts);
debug('Successfully created socks proxy connection');
if (proxies.length === 1) {
const socksOpts: SocksClientOptions = {
proxy: proxies[0],
destination: {
host,
port: typeof port === 'number' ? port : parseInt(port, 10),
},
command: 'connect',
timeout: timeout ?? undefined,
};

debug('Creating socks proxy connection: %o', socksOpts);
const connection = await SocksClient.createConnection(socksOpts);
socket = connection.socket;
debug('Successfully created socks proxy connection');
} else {
const socksOpts: SocksClientChainOptions = {
proxies: proxies,
destination: {
host,
port: typeof port === 'number' ? port : parseInt(port, 10),
},
command: 'connect',
timeout: timeout ?? undefined,
};

debug('Creating chained socks proxy connection: %o', socksOpts);
const connection = await SocksClient.createConnectionChain(
socksOpts
);
socket = connection.socket;
debug('Successfully created chained socks proxy connection');
}

if (timeout !== null) {
socket.setTimeout(timeout);
Expand Down
43 changes: 39 additions & 4 deletions packages/socks-proxy-agent/test/test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -68,13 +68,13 @@ describe('SocksProxyAgent', () => {
describe('constructor', () => {
it('should accept a "string" proxy argument', () => {
const agent = new SocksProxyAgent(socksServerUrl.href);
assert.equal(socksServerUrl.hostname, agent.proxy.host);
assert.equal(+socksServerUrl.port, agent.proxy.port);
assert.equal(socksServerUrl.hostname, agent.proxies[0].host);
assert.equal(+socksServerUrl.port, agent.proxies[0].port);
});
it('should accept a `new URL()` result object argument', () => {
const agent = new SocksProxyAgent(socksServerUrl);
assert.equal(socksServerUrl.hostname, agent.proxy.host);
assert.equal(+socksServerUrl.port, agent.proxy.port);
assert.equal(socksServerUrl.hostname, agent.proxies[0].host);
assert.equal(+socksServerUrl.port, agent.proxies[0].port);
});
it('should respect `timeout` option during connection to socks server', async () => {
const agent = new SocksProxyAgent(socksServerUrl, { timeout: 1 });
Expand Down Expand Up @@ -107,6 +107,41 @@ describe('SocksProxyAgent', () => {
const body = await json(res);
assert.equal('bar', body.foo);
});

it('should work against an HTTP endpoint with multiple SOCKS proxies', async () => {
const secondSocksServer = socks.createServer(function (
// @ts-expect-error no types for `socksv5`
_info,
// @ts-expect-error no types for `socksv5`
accept
) {
accept();
});
await listen(secondSocksServer);
const port = secondSocksServer.address().port;
const secondSocksServerUrl = new URL(`socks://127.0.0.1:${port}`);
secondSocksServer.useAuth(socks.auth.None());

httpServer.once('request', function (req, res) {
assert.equal('/foo', req.url);
res.statusCode = 404;
res.end(JSON.stringify(req.headers));
});

const res = await req(new URL('/foo', httpServerUrl), {
agent: new SocksProxyAgent([
socksServerUrl,
secondSocksServerUrl,
]),
headers: { foo: 'bar' },
});
assert.equal(404, res.statusCode);

const body = await json(res);
assert.equal('bar', body.foo);

secondSocksServer.close();
});
});

describe('"https" module', () => {
Expand Down

0 comments on commit 76b52a8

Please sign in to comment.