为了账号安全,请及时绑定邮箱和手机立即绑定

关于Nodejs里代理设置的问题

解决一个历史遗留问题:

博客上线后第一篇关于Nodejs的博客就是——Nodejs绑定域名与反向代理;是关于Nodejs的代理设置,到现在为止线上几个二级小站也还是在使用那个http-proxy模块,却怎么也没想到,这些小站的一个共同的遗留问题也源自这个代理模块;问题在于:用已知的各种方式获取到的访问者的ip永远为127.0.0.1;拜托不要提os模块了,os可以获取本站ipv4的值,除此值以外,根本没有第二个ipv4的字段好嘛!

比如以博客为主站,cdn为二级站点,代理模块是这样的,像下面这样,问题就在其中:

var http = require('http'), 
    httpProxy = require('http-proxy'),
    proxy = httpProxy.createProxyServer({});

proxy.on('error', function (err, req, res) {
 res.writeHead(500, {
 'Content-Type': 'text/plain'
 });
 res.end('Something went wrong.');
});
var server = require('http').createServer(function(req, res) {
    var host = req.headers.host;
    switch (host){
        case 'cdn.famanoder.cn':
            proxy.web(req, res, { target: 'http://localhost:4000' });
        break;
        case 'www.famanoder.cn':
            proxy.web(req, res, { target: 'http://localhost:3000' });
        break;
        default:
            proxy.web(req, res, { target: 'http://localhost:3000' });
    }
});
server.listen(80);

这里是新建了一个server来监听80端口,然后根据所有请求的host,将请求代理到目标站点;如果访问首页http://famanoder.com那么将会被代理到服务器上的3000端口的站点,也就是说,我只是自认为我访问首页就是直接访问3000端口,而其实是通过一个额外的server代理过去的,所以我在所有站点里获取到的ip永远为127.0.0.1,因为代理过来后域从famanoder.com变为了localhost;

从上面的代理设置,可以看出两个问题:

1、这里新建的一个server是多余的;

2、不能将所有站点包括在代理设置内;

2.5、抱歉,请原谅我这个不完美的小孩;

既然这个server是多余的,那么先将其去掉;

既然不能将所有站点包括在代理设置内,那么就留下博客做主站监听80端口;

那么就应该在主站里分发二级站点了,事实也证明主站里监听80端口后,通过express的req.ip就能获取到访问者的ip了,同上,二级站点现在是被代理转发的,所以依然拿不到真实的访问者的ip;

既然主站现在可以拿到ip,二级站点又是由主站分发出去的,那么在转发之前通过设置转发请求的header,二级站点从header上不就可以拿到从主站传过来的ip了吗!

在express里,现在可以将这个代理设置作为一个中间件来对待,将访问者的ip由请求header带过去:

app.use(function(req,res,next){
  console.log(req.ip);
    var proxy = httpProxy.createProxyServer({
      headers:{
        'x-forward-ip':req.ip.match(/([\w\.]+)/g)[1]
      }
    });
    proxy.on('error', function (err, req, res) {
     res.writeHead(500, {
     'Content-Type': 'text/plain'
     });
     res.end('Something went wrong.');
    });
    var host = req.headers.host;
     switch (host){
      case 'cdn.famanoder.cn': 
        proxy.web(req, res, { target: 'http://localhost:4000' });
      break;
       default:  
        next();//主站不再被转发,next()是必需的!!!
     }
});

如上:添加了代理设置的headers项,二级站点将由req.header('x-forward-ip')来获取访问者的ip;

用http-proxy解决跨域问题:

如果a.com要请求b.com的接口,涉及到跨域问题,除了jsonp之外,通过http-proxy设置代理,可以直接请求a.com,然后由proxy将a.com转发到b.com实现跨域;

case 'a.com':
    proxy.web(req, res, { target: 'http://b.com' });
break;

因为跨域问题只存在于前端,后端根本不涉及到跨域问题;现在感觉http-proxy是通过http.request来实现的转发,可能就是切换host;

原文来自:花满楼(http://famanoder.com/bokes

点击查看更多内容
1人点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消