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