使用nginx反向代理后如何在后台web应用中获取用户ip

使用nginx反向代理后如何在后台web应用中获取用户ip

原创 2017年05月01日 15:02:44



  • 问题背景

    在实际应用中,我们可能需要获取用户的ip地址,比如做异地登陆的判断,或者统计ip访问次数等,通常情况下我们使用request.RemoteAddr就可以获取到客户端ip,但是当我们使用了nginx作为反向代理后,使用request.RemoteAddr获取到的就一直是nginx服务器的ip的地址,那这时应该怎么办?

    原理解释

    经过反向代理后,由于在客户端和web服务器之间增加了中间层,因此web服务器无法直接拿到客户端的ip,通过$remote_addr变量拿到的将是反向代理服务器的ip地址。

    当你使用了nginx反向服务器后,在web端使用request.RemoteAddr(本质上就是获取$remote_addr),取得的是nginx的地址,即$remote_addr变量中封装的是nginx的地址,当然是没法获得用户的真实ip的,但是,nginx是可以获得用户的真实ip的,也就是说nginx使用$remote_addr变量时获得的是用户的真实ip,如果我们想要在web端获得用户的真实ip,就必须在nginx这里作一个赋值操作,如下:
    proxy_set_header  X-Real_IP  $remote_addr;
    其中这个X-Real_IP是一个自定义的变量名,名字可以随意取,这样做完之后,用户的真实ip就被放在X-Real_IP这个变量里了,然后,在web端可以这样获取:
    request.Header.Get("X-Real_IP")


    这里我们将nginx里的相关变量解释一下,通常我们会看到有这样一些配置

    1. server {  
    2.     listen       80;  
    3.     server_name  localhost;  
    4.     #charset koi8-r;  
    5.     #access_log  logs/host.access.log  main;  
    6.     location /{  
    7.         root   html;  
    8.         index  index.html index.htm;  
    9.         proxy_pass                  http://backend;   
    10.         proxy_redirect              off;  
    11.         proxy_set_header            Host $host;  
    12.         proxy_set_header            X-real-ip $remote_addr;  
    13.         proxy_set_header            X-Forwarded-For $proxy_add_x_forwarded_for;  
    14.         # proxy_set_header            X-Forwarded-For $http_x_forwarded_for;  
    15.     }  
    16. }  
    1. proxy_set_header  X-Real_IP  $remote_addr;
    这句话之前已经解释过,有了这句就可以在web服务器端获得用户的真实ip
    但是,实际上要获得用户的真实ip,不是只有这一个方法,下面我们继续看。
     
    2.  proxy_set_header  X-Forwarded-For  $proxy_add_x_forwarded_for;
    我们先看看这里有个X-Forwarded-For变量,这是一个squid开发的,用于识别通过HTTP代理或负载平衡器原始IP一个连接到Web服务器的客户机地址的非rfc标准,如果有做X-Forwarded-For设置的话,每次经过proxy转发都会有记录,格式就是client1, proxy1, proxy2,以逗号隔开各个地址,由于他是非rfc标准,所以默认是没有的,需要强制添加,在默认情况下经过proxy转发的请求,在后端看来远程地址都是proxy端的ip 。也就是说在默认情况下我们使用request.Header.Get("X-Forwarded-For")获取不到用户的ip,如果我们想要通过这个变量获得用户的ip,我们需要自己在nginx添加如下配置:
    proxy_set_header  X-Forwarded-For  $proxy_add_x_forwarded_for;
    意思是增加一个$proxy_add_x_forwarded_for到X-Forwarded-For里去,注意是增加,而不是覆盖,当然由于默认的X-Forwarded-For值是空的,所以我们总感觉X-Forwarded-For的值就等于$proxy_add_x_forwarded_for的值,实际上当你搭建两台nginx在不同的ip上,并且都使用了这段配置,那你会发现在web服务器端通过request.Header.Get("X-Forwarded-For")获得的将会是客户端ip和第一台nginx的ip。
     
    那么$proxy_add_x_forwarded_for又是什么?
    $proxy_add_x_forwarded_for变量包含客户端请求头中的"X-Forwarded-For",与$remote_addr两部分,他们之间用逗号分开。
    举个例子,有一个web应用,在它之前通过了两个nginx转发,即用户访问该web通过两台nginx。
    在第一台nginx中,使用
    proxy_set_header   X-Forwarded-For  $proxy_add_x_forwarded_for;
    现在的$proxy_add_x_forwarded_for变量的"X-Forwarded-For"部分是空的,所以只有$remote_addr,而$remote_addr的值是用户的ip,于是赋值以后,X-Forwarded-For变量的值就是用户的真实的ip地址了。
     
    到了第二台nginx,使用
    proxy_set_header   X-Forwarded-For   $proxy_add_x_forwarded_for;
    现在的$proxy_add_x_forwarded_for变量,X-Forwarded-For部分包含的是用户的真实ip,$remote_addr部分的值是上一台nginx的ip地址,于是通过这个赋值以后现在的X-Forwarded-For的值就变成了“用户的真实ip,第一台nginx的ip”。
     
    最后我们看到还有一个$http_x_forwarded_for变量,这个变量就是X-Forwarded-For,由于之前我们说了,默认的这个X-Forwarded-For是为空的,所以当我们直接使用proxy_set_header   X-Forwarded-For  $http_x_forwarded_for时会发现,web服务器端使用request.Header.Get("X-Forwarded-For")获得的值是null。如果想要通过request.Header.Get("X-Forwarded-For")获得用户ip,就必须先使用proxy_set_header   X-Forwarded-For  $proxy_add_x_forwarded_for;这样就可以获得用户真实ip。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值