Skip to content

Commit 85b10b3

Browse files
authored
[proxy-agent] Lazily load agents (#327)
1 parent ad68f63 commit 85b10b3

File tree

2 files changed

+42
-28
lines changed

2 files changed

+42
-28
lines changed

.changeset/late-berries-rule.md

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'proxy-agent': minor
3+
---
4+
5+
Lazily load agents in proxy-agent

packages/proxy-agent/src/index.ts

+37-28
Original file line numberDiff line numberDiff line change
@@ -5,47 +5,57 @@ import LRUCache from 'lru-cache';
55
import { Agent, AgentConnectOpts } from 'agent-base';
66
import createDebug from 'debug';
77
import { getProxyForUrl as envGetProxyForUrl } from 'proxy-from-env';
8-
import { PacProxyAgent, PacProxyAgentOptions } from 'pac-proxy-agent';
9-
import { HttpProxyAgent, HttpProxyAgentOptions } from 'http-proxy-agent';
10-
import { HttpsProxyAgent, HttpsProxyAgentOptions } from 'https-proxy-agent';
11-
import { SocksProxyAgent, SocksProxyAgentOptions } from 'socks-proxy-agent';
8+
import type { PacProxyAgent, PacProxyAgentOptions } from 'pac-proxy-agent';
9+
import type { HttpProxyAgent, HttpProxyAgentOptions } from 'http-proxy-agent';
10+
import type { HttpsProxyAgent, HttpsProxyAgentOptions } from 'https-proxy-agent';
11+
import type { SocksProxyAgent, SocksProxyAgentOptions } from 'socks-proxy-agent';
1212

1313
const debug = createDebug('proxy-agent');
1414

15-
const PROTOCOLS = [
16-
...HttpProxyAgent.protocols,
17-
...SocksProxyAgent.protocols,
18-
...PacProxyAgent.protocols,
19-
] as const;
15+
type ValidProtocol =
16+
| (typeof HttpProxyAgent.protocols)[number]
17+
| (typeof HttpsProxyAgent.protocols)[number]
18+
| (typeof SocksProxyAgent.protocols)[number]
19+
| (typeof PacProxyAgent.protocols)[number];
2020

21-
type ValidProtocol = (typeof PROTOCOLS)[number];
22-
23-
type AgentConstructor = new (...args: never[]) => Agent;
21+
type AgentConstructor = new (proxy: string, proxyAgentOptions?: ProxyAgentOptions) => Agent;
2422

2523
type GetProxyForUrlCallback = (url: string) => string | Promise<string>;
2624

25+
/**
26+
* Shorthands for built-in supported types.
27+
* Lazily loaded since some of these imports can be quite expensive
28+
* (in particular, pac-proxy-agent).
29+
*/
30+
const wellKnownAgents = {
31+
http: async () => (await import('http-proxy-agent')).HttpProxyAgent,
32+
https: async () => (await import('https-proxy-agent')).HttpsProxyAgent,
33+
socks: async () => (await import('socks-proxy-agent')).SocksProxyAgent,
34+
pac: async () => (await import('pac-proxy-agent')).PacProxyAgent,
35+
} as const;
36+
2737
/**
2838
* Supported proxy types.
2939
*/
3040
export const proxies: {
31-
[P in ValidProtocol]: [AgentConstructor, AgentConstructor];
41+
[P in ValidProtocol]: [() => Promise<AgentConstructor>, () => Promise<AgentConstructor>];
3242
} = {
33-
http: [HttpProxyAgent, HttpsProxyAgent],
34-
https: [HttpProxyAgent, HttpsProxyAgent],
35-
socks: [SocksProxyAgent, SocksProxyAgent],
36-
socks4: [SocksProxyAgent, SocksProxyAgent],
37-
socks4a: [SocksProxyAgent, SocksProxyAgent],
38-
socks5: [SocksProxyAgent, SocksProxyAgent],
39-
socks5h: [SocksProxyAgent, SocksProxyAgent],
40-
'pac+data': [PacProxyAgent, PacProxyAgent],
41-
'pac+file': [PacProxyAgent, PacProxyAgent],
42-
'pac+ftp': [PacProxyAgent, PacProxyAgent],
43-
'pac+http': [PacProxyAgent, PacProxyAgent],
44-
'pac+https': [PacProxyAgent, PacProxyAgent],
43+
http: [wellKnownAgents.http, wellKnownAgents.https],
44+
https: [wellKnownAgents.http, wellKnownAgents.https],
45+
socks: [wellKnownAgents.socks, wellKnownAgents.socks],
46+
socks4: [wellKnownAgents.socks, wellKnownAgents.socks],
47+
socks4a: [wellKnownAgents.socks, wellKnownAgents.socks],
48+
socks5: [wellKnownAgents.socks, wellKnownAgents.socks],
49+
socks5h: [wellKnownAgents.socks, wellKnownAgents.socks],
50+
'pac+data': [wellKnownAgents.pac, wellKnownAgents.pac],
51+
'pac+file': [wellKnownAgents.pac, wellKnownAgents.pac],
52+
'pac+ftp': [wellKnownAgents.pac, wellKnownAgents.pac],
53+
'pac+http': [wellKnownAgents.pac, wellKnownAgents.pac],
54+
'pac+https': [wellKnownAgents.pac, wellKnownAgents.pac],
4555
};
4656

4757
function isValidProtocol(v: string): v is ValidProtocol {
48-
return (PROTOCOLS as readonly string[]).includes(v);
58+
return Object.keys(proxies).includes(v);
4959
}
5060

5161
export type ProxyAgentOptions = HttpProxyAgentOptions<''> &
@@ -138,8 +148,7 @@ export class ProxyAgent extends Agent {
138148
throw new Error(`Unsupported protocol for proxy URL: ${proxy}`);
139149
}
140150
const ctor =
141-
proxies[proxyProto][secureEndpoint || isWebSocket ? 1 : 0];
142-
// @ts-expect-error meh…
151+
await proxies[proxyProto][secureEndpoint || isWebSocket ? 1 : 0]();
143152
agent = new ctor(proxy, this.connectOpts);
144153
this.cache.set(cacheKey, agent);
145154
} else {

0 commit comments

Comments
 (0)