UNPKG

xpm

Version:

The xPack project manager command line tool

433 lines (363 loc) 15.8 kB
/* eslint max-statements:0 */ 'use strict'; import {describe, it} from 'node:test'; import assert from 'node:assert'; import {parse as parseUrl} from 'node:url'; import {getProxyForUrl} from 'proxy-from-env'; // 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); it(title, function() { var actual; runWithEnv(env, function() { actual = getProxyForUrl(input); }); assert.strictEqual(actual, expected); }); } 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, '', 'http://abc\x00/'); testProxyUrl(env, '', undefined); testProxyUrl(env, '', null); testProxyUrl(env, '', {}); testProxyUrl(env, '', {host: 'x', protocol: 1}); testProxyUrl(env, '', {host: 1, protocol: 'x'}); describe('difference between url.parse and WHATWG URL', function() { // Node 24 and later raise the following warning when url.parse is used: // // (node:11623) [DEP0169] DeprecationWarning: `url.parse()` behavior is // not standardized and prone to errors that have security implications. // Use the WHATWG URL API instead. CVEs are not issued for `url.parse()` // vulnerabilities. // // The above refers to https://hackerone.com/reports/678487 which shows // that a bare percentage sign is parsed inconsistently: // - `url.parse` splits hosts. // - WHATWG `URL` constructor raised an error. // // This test case shows the difference. // // For comparison: // - curl (8.17.0) refuses to connect: // $ http_proxy=http://localhost:1337 curl http://bad% // curl:(3) URL rejected: Bad hostname // - wget (GNU wget 1.25.0) passes "bad%" as Host header: // $ http_proxy=http://localhost:1337 wget http://bad% // (nc -l 1337 receives request with "bad% as Host header) // - Python (3.13.11) passes "bad%" as Host header: // $ http_proxy=http://localhost:1337 python3 -c \ // 'import urllib.request;urllib.request.urlopen("http://bad%")' // (nc -l 1337 receives request with "bad% as Host header) // A canonical URL does not have a single "%". var badUrl = 'http://bad%'; // proxy-from-env@1.1.0 and earlier accepted bad URLs: testProxyUrl(env, 'http://unexpected.proxy', parseUrl(badUrl)); // Sanity check: WHATWG URL constructor rejects badUrl. assert(!URL.canParse(badUrl)); // Verify current proxy-from-env behavior. Should reject without throwing. testProxyUrl(env, '', badUrl); }); }); 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')); // eslint-disable-next-line camelcase env.http_proxy = 'http://priority'; 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'); // eslint-disable-next-line camelcase env.https_proxy = 'http://priority'; 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'); // eslint-disable-next-line camelcase env.all_proxy = 'http://priority'; 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'); }); // Up until proxy-from-env@1.1.0, proxy-from-env had undocumented support for // specifying proxies through npm_config_ prefixes. The historical reasons // for them are no longer relevant: // https://github.com/Rob--W/proxy-from-env/issues/13#issuecomment-3150256653 describe('NPM proxy configuration', function() { describe('npm_config_*_proxy variables are unsupported', function() { var env = {}; // eslint-disable-next-line camelcase env.npm_config_http_proxy = 'http://http-proxy'; // eslint-disable-next-line camelcase env.npm_config_https_proxy = 'http://https-proxy'; // eslint-disable-next-line camelcase env.npm_config_proxy = 'http://unexpected-proxy'; testProxyUrl(env, '', 'http://example'); testProxyUrl(env, '', 'https://example'); }); }); });