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
						
					
					
				'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); });
 | 
						|
  }
 | 
						|
}
 |