diff --git a/lib/cors-anywhere.js b/lib/cors-anywhere.js index eb4f47c5..0805650f 100644 --- a/lib/cors-anywhere.js +++ b/lib/cors-anywhere.js @@ -3,6 +3,7 @@ 'use strict'; +var dns = require('dns'); var httpProxy = require('http-proxy'); var net = require('net'); var url = require('url'); @@ -84,7 +85,7 @@ function proxyRequest(req, res, proxy) { var proxyOptions = { changeOrigin: false, prependPath: false, - target: location, + target: location.href, headers: { host: location.host, }, @@ -132,7 +133,24 @@ function proxyRequest(req, res, proxy) { // Start proxying the request try { - proxy.web(req, res, proxyOptions); + if (!req.corsAnywhereRequestState.dnsLookup) { + // Start proxying the request + proxy.web(req, res, proxyOptions); + return; + } + var targetUrl = url.parse(proxyOptions.target); + req.corsAnywhereRequestState.dnsLookup(targetUrl.hostname, function (err, address) { + if (err) { + // TODO: Should errors just be propagated, or should we support something like + // err.statusCode, err.statusText and err.message to customize the HTTP response? + proxy.emit('error', err, req, res); + return; + } + targetUrl.host = null; // Null .host so that .hostname + .port is used. + targetUrl.hostname = address; + proxyOptions.target = url.format(targetUrl); + proxy.web(req, res, proxyOptions); + }); } catch (err) { proxy.emit('error', err, req, res); } @@ -268,6 +286,7 @@ function getHandler(options, proxy) { setHeaders: {}, // Set these request headers. corsMaxAge: 0, // If set, an Access-Control-Max-Age header with this value (in seconds) will be added. helpFile: __dirname + '/help.txt', + dnsLookup: null, }; Object.keys(corsAnywhere).forEach(function(option) { @@ -299,6 +318,7 @@ function getHandler(options, proxy) { getProxyForUrl: corsAnywhere.getProxyForUrl, maxRedirects: corsAnywhere.maxRedirects, corsMaxAge: corsAnywhere.corsMaxAge, + dnsLookup: corsAnywhere.dnsLookup, }; var cors_headers = withCORS({}, req); diff --git a/server.js b/server.js index 757ac1d0..cf1c650e 100644 --- a/server.js +++ b/server.js @@ -1,3 +1,5 @@ +var dns = require('dns'); + // Listen on a specific host via the HOST environment variable var host = process.env.HOST || '0.0.0.0'; // Listen on a specific port via the PORT environment variable @@ -44,6 +46,24 @@ cors_proxy.createServer({ // Do not add X-Forwarded-For, etc. headers, because Heroku already adds it. xfwd: false, }, + dnsLookup: function (hostname, callback) { + var excludedHostnamePrefixes = [ + '169.254.', + '127.', + '0:0:0:0:0:0:0:1', + '::1', + '10.', + '172.16.', + '192.168.', + 'fe80::10' + ]; + dns.lookup(hostname, { hints: dns.ADDRCONFIG }, (err, address, family) => { + if (address && excludedHostnamePrefixes.some(p => address.startsWith(p))) { + err = 'ExcludedAddress' + } + callback(err, address, family); + }); + }, }).listen(port, host, function() { console.log('Running CORS Anywhere on ' + host + ':' + port); });