| /* jshint mocha:true */ |
| 'use strict'; |
| |
| var assert = require('assert'); |
| var parseUrl = require('url').parse; |
| |
| var getProxyForUrl = require('./').getProxyForUrl; |
| |
| // Runs the callback with process.env temporarily set to env. |
| function runWithEnv(env, callback) { |
| var originalEnv = process.env; |
| process.env = env; |
| try { |
| callback(); |
| } finally { |
| process.env = originalEnv; |
| } |
| } |
| |
| // Defines a test case that checks whether getProxyForUrl(input) === expected. |
| function testProxyUrl(env, expected, input) { |
| assert(typeof env === 'object' && env !== null); |
| // Copy object to make sure that the in param does not get modified between |
| // the call of this function and the use of it below. |
| env = JSON.parse(JSON.stringify(env)); |
| |
| var title = 'getProxyForUrl(' + JSON.stringify(input) + ')' + |
| ' === ' + JSON.stringify(expected); |
| |
| // Save call stack for later use. |
| var stack = {}; |
| Error.captureStackTrace(stack, testProxyUrl); |
| // Only use the last stack frame because that shows where this function is |
| // called, and that is sufficient for our purpose. No need to flood the logs |
| // with an uninteresting stack trace. |
| stack = stack.stack.split('\n', 2)[1]; |
| |
| it(title, function() { |
| var actual; |
| runWithEnv(env, function() { |
| actual = getProxyForUrl(input); |
| }); |
| if (expected === actual) { |
| return; // Good! |
| } |
| try { |
| assert.strictEqual(expected, actual); // Create a formatted error message. |
| // Should not happen because previously we determined expected !== actual. |
| throw new Error('assert.strictEqual passed. This is impossible!'); |
| } catch (e) { |
| // Use the original stack trace, so we can see a helpful line number. |
| e.stack = e.message + stack; |
| throw e; |
| } |
| }); |
| } |
| |
| describe('getProxyForUrl', function() { |
| describe('No proxy variables', function() { |
| var env = {}; |
| testProxyUrl(env, '', 'http://example.com'); |
| testProxyUrl(env, '', 'https://example.com'); |
| testProxyUrl(env, '', 'ftp://example.com'); |
| }); |
| |
| describe('Invalid URLs', function() { |
| var env = {}; |
| env.ALL_PROXY = 'http://unexpected.proxy'; |
| testProxyUrl(env, '', 'bogus'); |
| testProxyUrl(env, '', '//example.com'); |
| testProxyUrl(env, '', '://example.com'); |
| testProxyUrl(env, '', '://'); |
| testProxyUrl(env, '', '/path'); |
| testProxyUrl(env, '', ''); |
| testProxyUrl(env, '', 'http:'); |
| testProxyUrl(env, '', 'http:/'); |
| testProxyUrl(env, '', 'http://'); |
| testProxyUrl(env, '', 'prototype://'); |
| testProxyUrl(env, '', 'hasOwnProperty://'); |
| testProxyUrl(env, '', '__proto__://'); |
| testProxyUrl(env, '', undefined); |
| testProxyUrl(env, '', null); |
| testProxyUrl(env, '', {}); |
| testProxyUrl(env, '', {host: 'x', protocol: 1}); |
| testProxyUrl(env, '', {host: 1, protocol: 'x'}); |
| }); |
| |
| describe('http_proxy and HTTP_PROXY', function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://http-proxy'; |
| |
| testProxyUrl(env, '', 'https://example'); |
| testProxyUrl(env, 'http://http-proxy', 'http://example'); |
| testProxyUrl(env, 'http://http-proxy', parseUrl('http://example')); |
| |
| // jscs:disable requireCamelCaseOrUpperCaseIdentifiers |
| env.http_proxy = 'http://priority'; |
| // jscs:enable requireCamelCaseOrUpperCaseIdentifiers |
| testProxyUrl(env, 'http://priority', 'http://example'); |
| }); |
| |
| describe('http_proxy with non-sensical value', function() { |
| var env = {}; |
| // Crazy values should be passed as-is. It is the responsibility of the |
| // one who launches the application that the value makes sense. |
| // TODO: Should we be stricter and perform validation? |
| env.HTTP_PROXY = 'Crazy \n!() { ::// }'; |
| testProxyUrl(env, 'Crazy \n!() { ::// }', 'http://wow'); |
| |
| // The implementation assumes that the HTTP_PROXY environment variable is |
| // somewhat reasonable, and if the scheme is missing, it is added. |
| // Garbage in, garbage out some would say... |
| env.HTTP_PROXY = 'crazy without colon slash slash'; |
| testProxyUrl(env, 'http://crazy without colon slash slash', 'http://wow'); |
| }); |
| |
| describe('https_proxy and HTTPS_PROXY', function() { |
| var env = {}; |
| // Assert that there is no fall back to http_proxy |
| env.HTTP_PROXY = 'http://unexpected.proxy'; |
| testProxyUrl(env, '', 'https://example'); |
| |
| env.HTTPS_PROXY = 'http://https-proxy'; |
| testProxyUrl(env, 'http://https-proxy', 'https://example'); |
| |
| // jscs:disable requireCamelCaseOrUpperCaseIdentifiers |
| env.https_proxy = 'http://priority'; |
| // jscs:enable requireCamelCaseOrUpperCaseIdentifiers |
| testProxyUrl(env, 'http://priority', 'https://example'); |
| }); |
| |
| describe('ftp_proxy', function() { |
| var env = {}; |
| // Something else than http_proxy / https, as a sanity check. |
| env.FTP_PROXY = 'http://ftp-proxy'; |
| |
| testProxyUrl(env, 'http://ftp-proxy', 'ftp://example'); |
| testProxyUrl(env, '', 'ftps://example'); |
| }); |
| |
| describe('all_proxy', function() { |
| var env = {}; |
| env.ALL_PROXY = 'http://catch-all'; |
| testProxyUrl(env, 'http://catch-all', 'https://example'); |
| |
| // jscs:disable requireCamelCaseOrUpperCaseIdentifiers |
| env.all_proxy = 'http://priority'; |
| // jscs:enable requireCamelCaseOrUpperCaseIdentifiers |
| testProxyUrl(env, 'http://priority', 'https://example'); |
| }); |
| |
| describe('all_proxy without scheme', function() { |
| var env = {}; |
| env.ALL_PROXY = 'noscheme'; |
| testProxyUrl(env, 'http://noscheme', 'http://example'); |
| testProxyUrl(env, 'https://noscheme', 'https://example'); |
| |
| // The module does not impose restrictions on the scheme. |
| testProxyUrl(env, 'bogus-scheme://noscheme', 'bogus-scheme://example'); |
| |
| // But the URL should still be valid. |
| testProxyUrl(env, '', 'bogus'); |
| }); |
| |
| describe('no_proxy empty', function() { |
| var env = {}; |
| env.HTTPS_PROXY = 'http://proxy'; |
| |
| // NO_PROXY set but empty. |
| env.NO_PROXY = ''; |
| testProxyUrl(env, 'http://proxy', 'https://example'); |
| |
| // No entries in NO_PROXY (comma). |
| env.NO_PROXY = ','; |
| testProxyUrl(env, 'http://proxy', 'https://example'); |
| |
| // No entries in NO_PROXY (whitespace). |
| env.NO_PROXY = ' '; |
| testProxyUrl(env, 'http://proxy', 'https://example'); |
| |
| // No entries in NO_PROXY (multiple whitespace / commas). |
| env.NO_PROXY = ',\t,,,\n, ,\r'; |
| testProxyUrl(env, 'http://proxy', 'https://example'); |
| }); |
| |
| describe('no_proxy=example (single host)', function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://proxy'; |
| |
| env.NO_PROXY = 'example'; |
| testProxyUrl(env, '', 'http://example'); |
| testProxyUrl(env, '', 'http://example:80'); |
| testProxyUrl(env, '', 'http://example:0'); |
| testProxyUrl(env, '', 'http://example:1337'); |
| testProxyUrl(env, 'http://proxy', 'http://sub.example'); |
| testProxyUrl(env, 'http://proxy', 'http://prefexample'); |
| testProxyUrl(env, 'http://proxy', 'http://example.no'); |
| testProxyUrl(env, 'http://proxy', 'http://a.b.example'); |
| testProxyUrl(env, 'http://proxy', 'http://host/example'); |
| }); |
| |
| describe('no_proxy=sub.example (subdomain)', function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://proxy'; |
| |
| env.NO_PROXY = 'sub.example'; |
| testProxyUrl(env, 'http://proxy', 'http://example'); |
| testProxyUrl(env, 'http://proxy', 'http://example:80'); |
| testProxyUrl(env, 'http://proxy', 'http://example:0'); |
| testProxyUrl(env, 'http://proxy', 'http://example:1337'); |
| testProxyUrl(env, '', 'http://sub.example'); |
| testProxyUrl(env, 'http://proxy', 'http://no.sub.example'); |
| testProxyUrl(env, 'http://proxy', 'http://sub-example'); |
| testProxyUrl(env, 'http://proxy', 'http://example.sub'); |
| }); |
| |
| describe('no_proxy=example:80 (host + port)', function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://proxy'; |
| |
| env.NO_PROXY = 'example:80'; |
| testProxyUrl(env, '', 'http://example'); |
| testProxyUrl(env, '', 'http://example:80'); |
| testProxyUrl(env, '', 'http://example:0'); |
| testProxyUrl(env, 'http://proxy', 'http://example:1337'); |
| testProxyUrl(env, 'http://proxy', 'http://sub.example'); |
| testProxyUrl(env, 'http://proxy', 'http://prefexample'); |
| testProxyUrl(env, 'http://proxy', 'http://example.no'); |
| testProxyUrl(env, 'http://proxy', 'http://a.b.example'); |
| }); |
| |
| describe('no_proxy=.example (host suffix)', function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://proxy'; |
| |
| env.NO_PROXY = '.example'; |
| testProxyUrl(env, 'http://proxy', 'http://example'); |
| testProxyUrl(env, 'http://proxy', 'http://example:80'); |
| testProxyUrl(env, 'http://proxy', 'http://example:1337'); |
| testProxyUrl(env, '', 'http://sub.example'); |
| testProxyUrl(env, '', 'http://sub.example:80'); |
| testProxyUrl(env, '', 'http://sub.example:1337'); |
| testProxyUrl(env, 'http://proxy', 'http://prefexample'); |
| testProxyUrl(env, 'http://proxy', 'http://example.no'); |
| testProxyUrl(env, '', 'http://a.b.example'); |
| }); |
| |
| describe('no_proxy=*', function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://proxy'; |
| env.NO_PROXY = '*'; |
| testProxyUrl(env, '', 'http://example.com'); |
| }); |
| |
| describe('no_proxy=*.example (host suffix with *.)', function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://proxy'; |
| |
| env.NO_PROXY = '*.example'; |
| testProxyUrl(env, 'http://proxy', 'http://example'); |
| testProxyUrl(env, 'http://proxy', 'http://example:80'); |
| testProxyUrl(env, 'http://proxy', 'http://example:1337'); |
| testProxyUrl(env, '', 'http://sub.example'); |
| testProxyUrl(env, '', 'http://sub.example:80'); |
| testProxyUrl(env, '', 'http://sub.example:1337'); |
| testProxyUrl(env, 'http://proxy', 'http://prefexample'); |
| testProxyUrl(env, 'http://proxy', 'http://example.no'); |
| testProxyUrl(env, '', 'http://a.b.example'); |
| }); |
| |
| describe('no_proxy=*example (substring suffix)', function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://proxy'; |
| |
| env.NO_PROXY = '*example'; |
| testProxyUrl(env, '', 'http://example'); |
| testProxyUrl(env, '', 'http://example:80'); |
| testProxyUrl(env, '', 'http://example:1337'); |
| testProxyUrl(env, '', 'http://sub.example'); |
| testProxyUrl(env, '', 'http://sub.example:80'); |
| testProxyUrl(env, '', 'http://sub.example:1337'); |
| testProxyUrl(env, '', 'http://prefexample'); |
| testProxyUrl(env, '', 'http://a.b.example'); |
| testProxyUrl(env, 'http://proxy', 'http://example.no'); |
| testProxyUrl(env, 'http://proxy', 'http://host/example'); |
| }); |
| |
| describe('no_proxy=.*example (arbitrary wildcards are NOT supported)', |
| function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://proxy'; |
| |
| env.NO_PROXY = '.*example'; |
| testProxyUrl(env, 'http://proxy', 'http://example'); |
| testProxyUrl(env, 'http://proxy', 'http://sub.example'); |
| testProxyUrl(env, 'http://proxy', 'http://sub.example'); |
| testProxyUrl(env, 'http://proxy', 'http://prefexample'); |
| testProxyUrl(env, 'http://proxy', 'http://x.prefexample'); |
| testProxyUrl(env, 'http://proxy', 'http://a.b.example'); |
| }); |
| |
| describe('no_proxy=[::1],[::2]:80,10.0.0.1,10.0.0.2:80 (IP addresses)', |
| function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://proxy'; |
| |
| env.NO_PROXY = '[::1],[::2]:80,10.0.0.1,10.0.0.2:80'; |
| testProxyUrl(env, '', 'http://[::1]/'); |
| testProxyUrl(env, '', 'http://[::1]:80/'); |
| testProxyUrl(env, '', 'http://[::1]:1337/'); |
| |
| testProxyUrl(env, '', 'http://[::2]/'); |
| testProxyUrl(env, '', 'http://[::2]:80/'); |
| testProxyUrl(env, 'http://proxy', 'http://[::2]:1337/'); |
| |
| testProxyUrl(env, '', 'http://10.0.0.1/'); |
| testProxyUrl(env, '', 'http://10.0.0.1:80/'); |
| testProxyUrl(env, '', 'http://10.0.0.1:1337/'); |
| |
| testProxyUrl(env, '', 'http://10.0.0.2/'); |
| testProxyUrl(env, '', 'http://10.0.0.2:80/'); |
| testProxyUrl(env, 'http://proxy', 'http://10.0.0.2:1337/'); |
| }); |
| |
| describe('no_proxy=127.0.0.1/32 (CIDR is NOT supported)', function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://proxy'; |
| |
| env.NO_PROXY = '127.0.0.1/32'; |
| testProxyUrl(env, 'http://proxy', 'http://127.0.0.1'); |
| testProxyUrl(env, 'http://proxy', 'http://127.0.0.1/32'); |
| }); |
| |
| describe('no_proxy=127.0.0.1 does NOT match localhost', function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://proxy'; |
| |
| env.NO_PROXY = '127.0.0.1'; |
| testProxyUrl(env, '', 'http://127.0.0.1'); |
| // We're not performing DNS queries, so this shouldn't match. |
| testProxyUrl(env, 'http://proxy', 'http://localhost'); |
| }); |
| |
| describe('no_proxy with protocols that have a default port', function() { |
| var env = {}; |
| env.WS_PROXY = 'http://ws'; |
| env.WSS_PROXY = 'http://wss'; |
| env.HTTP_PROXY = 'http://http'; |
| env.HTTPS_PROXY = 'http://https'; |
| env.GOPHER_PROXY = 'http://gopher'; |
| env.FTP_PROXY = 'http://ftp'; |
| env.ALL_PROXY = 'http://all'; |
| |
| env.NO_PROXY = 'xxx:21,xxx:70,xxx:80,xxx:443'; |
| |
| testProxyUrl(env, '', 'http://xxx'); |
| testProxyUrl(env, '', 'http://xxx:80'); |
| testProxyUrl(env, 'http://http', 'http://xxx:1337'); |
| |
| testProxyUrl(env, '', 'ws://xxx'); |
| testProxyUrl(env, '', 'ws://xxx:80'); |
| testProxyUrl(env, 'http://ws', 'ws://xxx:1337'); |
| |
| testProxyUrl(env, '', 'https://xxx'); |
| testProxyUrl(env, '', 'https://xxx:443'); |
| testProxyUrl(env, 'http://https', 'https://xxx:1337'); |
| |
| testProxyUrl(env, '', 'wss://xxx'); |
| testProxyUrl(env, '', 'wss://xxx:443'); |
| testProxyUrl(env, 'http://wss', 'wss://xxx:1337'); |
| |
| testProxyUrl(env, '', 'gopher://xxx'); |
| testProxyUrl(env, '', 'gopher://xxx:70'); |
| testProxyUrl(env, 'http://gopher', 'gopher://xxx:1337'); |
| |
| testProxyUrl(env, '', 'ftp://xxx'); |
| testProxyUrl(env, '', 'ftp://xxx:21'); |
| testProxyUrl(env, 'http://ftp', 'ftp://xxx:1337'); |
| }); |
| |
| describe('no_proxy should not be case-sensitive', function() { |
| var env = {}; |
| env.HTTP_PROXY = 'http://proxy'; |
| env.NO_PROXY = 'XXX,YYY,ZzZ'; |
| |
| testProxyUrl(env, '', 'http://xxx'); |
| testProxyUrl(env, '', 'http://XXX'); |
| testProxyUrl(env, '', 'http://yyy'); |
| testProxyUrl(env, '', 'http://YYY'); |
| testProxyUrl(env, '', 'http://ZzZ'); |
| testProxyUrl(env, '', 'http://zZz'); |
| }); |
| }); |