You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					201 lines
				
				4.8 KiB
			
		
		
			
		
	
	
					201 lines
				
				4.8 KiB
			| 
								 
											3 years ago
										 
									 | 
							
								'use strict';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Module dependencies.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								var url = require('url');
							 | 
						||
| 
								 | 
							
								var LRU = require('lru-cache');
							 | 
						||
| 
								 | 
							
								var Agent = require('agent-base');
							 | 
						||
| 
								 | 
							
								var inherits = require('util').inherits;
							 | 
						||
| 
								 | 
							
								var debug = require('debug')('proxy-agent');
							 | 
						||
| 
								 | 
							
								var getProxyForUrl = require('proxy-from-env').getProxyForUrl;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								var http = require('http');
							 | 
						||
| 
								 | 
							
								var https = require('https');
							 | 
						||
| 
								 | 
							
								var PacProxyAgent = require('pac-proxy-agent');
							 | 
						||
| 
								 | 
							
								var HttpProxyAgent = require('http-proxy-agent');
							 | 
						||
| 
								 | 
							
								var HttpsProxyAgent = require('https-proxy-agent');
							 | 
						||
| 
								 | 
							
								var SocksProxyAgent = require('socks-proxy-agent');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Module exports.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports = module.exports = ProxyAgent;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Number of `http.Agent` instances to cache.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * This value was arbitrarily chosen... a better
							 | 
						||
| 
								 | 
							
								 * value could be conceived with some benchmarks.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								var cacheSize = 20;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Cache for `http.Agent` instances.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.cache = new LRU(cacheSize);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Built-in proxy types.
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								exports.proxies = Object.create(null);
							 | 
						||
| 
								 | 
							
								exports.proxies.http = httpOrHttpsProxy;
							 | 
						||
| 
								 | 
							
								exports.proxies.https = httpOrHttpsProxy;
							 | 
						||
| 
								 | 
							
								exports.proxies.socks = SocksProxyAgent;
							 | 
						||
| 
								 | 
							
								exports.proxies.socks4 = SocksProxyAgent;
							 | 
						||
| 
								 | 
							
								exports.proxies.socks4a = SocksProxyAgent;
							 | 
						||
| 
								 | 
							
								exports.proxies.socks5 = SocksProxyAgent;
							 | 
						||
| 
								 | 
							
								exports.proxies.socks5h = SocksProxyAgent;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								PacProxyAgent.protocols.forEach(function (protocol) {
							 | 
						||
| 
								 | 
							
								  exports.proxies['pac+' + protocol] = PacProxyAgent;
							 | 
						||
| 
								 | 
							
								});
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								function httpOrHttps(opts, secureEndpoint) {
							 | 
						||
| 
								 | 
							
								  if (secureEndpoint) {
							 | 
						||
| 
								 | 
							
								    // HTTPS
							 | 
						||
| 
								 | 
							
								    return https.globalAgent;
							 | 
						||
| 
								 | 
							
								  } else {
							 | 
						||
| 
								 | 
							
								    // HTTP
							 | 
						||
| 
								 | 
							
								    return http.globalAgent;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								function httpOrHttpsProxy(opts, secureEndpoint) {
							 | 
						||
| 
								 | 
							
								  if (secureEndpoint) {
							 | 
						||
| 
								 | 
							
								    // HTTPS
							 | 
						||
| 
								 | 
							
								    return new HttpsProxyAgent(opts);
							 | 
						||
| 
								 | 
							
								  } else {
							 | 
						||
| 
								 | 
							
								    // HTTP
							 | 
						||
| 
								 | 
							
								    return new HttpProxyAgent(opts);
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								function mapOptsToProxy(opts) {
							 | 
						||
| 
								 | 
							
								  // NO_PROXY case
							 | 
						||
| 
								 | 
							
								  if (!opts) {
							 | 
						||
| 
								 | 
							
								    return {
							 | 
						||
| 
								 | 
							
								      uri: 'no proxy',
							 | 
						||
| 
								 | 
							
								      fn: httpOrHttps
							 | 
						||
| 
								 | 
							
								    };
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if ('string' == typeof opts) opts = url.parse(opts);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  var proxies;
							 | 
						||
| 
								 | 
							
								  if (opts.proxies) {
							 | 
						||
| 
								 | 
							
								    proxies = Object.assign({}, exports.proxies, opts.proxies);
							 | 
						||
| 
								 | 
							
								  } else {
							 | 
						||
| 
								 | 
							
								    proxies = exports.proxies;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // get the requested proxy "protocol"
							 | 
						||
| 
								 | 
							
								  var protocol = opts.protocol;
							 | 
						||
| 
								 | 
							
								  if (!protocol) {
							 | 
						||
| 
								 | 
							
								    throw new TypeError('You must specify a "protocol" for the ' +
							 | 
						||
| 
								 | 
							
								                        'proxy type (' + Object.keys(proxies).join(', ') + ')');
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // strip the trailing ":" if present
							 | 
						||
| 
								 | 
							
								  if (':' == protocol[protocol.length - 1]) {
							 | 
						||
| 
								 | 
							
								    protocol = protocol.substring(0, protocol.length - 1);
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // get the proxy `http.Agent` creation function
							 | 
						||
| 
								 | 
							
								  var proxyFn = proxies[protocol];
							 | 
						||
| 
								 | 
							
								  if ('function' != typeof proxyFn) {
							 | 
						||
| 
								 | 
							
								    throw new TypeError('unsupported proxy protocol: "' + protocol + '"');
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // format the proxy info back into a URI, since an opts object
							 | 
						||
| 
								 | 
							
								  // could have been passed in originally. This generated URI is used
							 | 
						||
| 
								 | 
							
								  // as part of the "key" for the LRU cache
							 | 
						||
| 
								 | 
							
								  return {
							 | 
						||
| 
								 | 
							
								    opts: opts,
							 | 
						||
| 
								 | 
							
								    uri: url.format({
							 | 
						||
| 
								 | 
							
								      protocol: protocol + ':',
							 | 
						||
| 
								 | 
							
								      slashes: true,
							 | 
						||
| 
								 | 
							
								      auth: opts.auth,
							 | 
						||
| 
								 | 
							
								      hostname: opts.hostname || opts.host,
							 | 
						||
| 
								 | 
							
								      port: opts.port
							 | 
						||
| 
								 | 
							
								    }),
							 | 
						||
| 
								 | 
							
								    fn: proxyFn,
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Attempts to get an `http.Agent` instance based off of the given proxy URI
							 | 
						||
| 
								 | 
							
								 * information, and the `secure` flag.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * An LRU cache is used, to prevent unnecessary creation of proxy
							 | 
						||
| 
								 | 
							
								 * `http.Agent` instances.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @param {String} uri proxy url
							 | 
						||
| 
								 | 
							
								 * @param {Boolean} secure true if this is for an HTTPS request, false for HTTP
							 | 
						||
| 
								 | 
							
								 * @return {http.Agent}
							 | 
						||
| 
								 | 
							
								 * @api public
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								function ProxyAgent (opts) {
							 | 
						||
| 
								 | 
							
								  if (!(this instanceof ProxyAgent)) return new ProxyAgent(opts);
							 | 
						||
| 
								 | 
							
								  debug('creating new ProxyAgent instance: %o', opts);
							 | 
						||
| 
								 | 
							
								  Agent.call(this);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (opts) {
							 | 
						||
| 
								 | 
							
								    var proxy = mapOptsToProxy(opts);
							 | 
						||
| 
								 | 
							
								    this.proxy = proxy.opts;
							 | 
						||
| 
								 | 
							
								    this.proxyUri = proxy.uri;
							 | 
						||
| 
								 | 
							
								    this.proxyFn = proxy.fn;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								inherits(ProxyAgent, Agent);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								ProxyAgent.prototype.callback = function(req, opts, fn) {
							 | 
						||
| 
								 | 
							
								  var proxyOpts = this.proxy;
							 | 
						||
| 
								 | 
							
								  var proxyUri = this.proxyUri;
							 | 
						||
| 
								 | 
							
								  var proxyFn = this.proxyFn;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // if we did not instantiate with a proxy, set one per request
							 | 
						||
| 
								 | 
							
								  if (!proxyOpts) {
							 | 
						||
| 
								 | 
							
								    var urlOpts = getProxyForUrl(opts);
							 | 
						||
| 
								 | 
							
								    var proxy = mapOptsToProxy(urlOpts, opts);
							 | 
						||
| 
								 | 
							
								    proxyOpts = proxy.opts;
							 | 
						||
| 
								 | 
							
								    proxyUri = proxy.uri;
							 | 
						||
| 
								 | 
							
								    proxyFn = proxy.fn;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // create the "key" for the LRU cache
							 | 
						||
| 
								 | 
							
								  var key = proxyUri;
							 | 
						||
| 
								 | 
							
								  if (opts.secureEndpoint) key += ' secure';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // attempt to get a cached `http.Agent` instance first
							 | 
						||
| 
								 | 
							
								  var agent = exports.cache.get(key);
							 | 
						||
| 
								 | 
							
								  if (!agent) {
							 | 
						||
| 
								 | 
							
								    // get an `http.Agent` instance from protocol-specific agent function
							 | 
						||
| 
								 | 
							
								    agent = proxyFn(proxyOpts, opts.secureEndpoint);
							 | 
						||
| 
								 | 
							
								    if (agent) {
							 | 
						||
| 
								 | 
							
								      exports.cache.set(key, agent);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  } else {
							 | 
						||
| 
								 | 
							
								    debug('cache hit with key: %o', key);
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  if (!proxyOpts) {
							 | 
						||
| 
								 | 
							
								    agent.addRequest(req, opts);
							 | 
						||
| 
								 | 
							
								  } else {
							 | 
						||
| 
								 | 
							
								    // XXX: agent.callback() is an agent-base-ism
							 | 
						||
| 
								 | 
							
								    agent.callback(req, opts)
							 | 
						||
| 
								 | 
							
								      .then(function(socket) { fn(null, socket); })
							 | 
						||
| 
								 | 
							
								      .catch(function(error) { fn(error); });
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								}
							 |