如何在Django中获取用户IP地址?


287

如何在Django中获取用户的IP?

我有这样的看法:

# Create your views
from django.contrib.gis.utils import GeoIP
from django.template import  RequestContext
from django.shortcuts import render_to_response


def home(request):
  g = GeoIP()
  client_ip = request.META['REMOTE_ADDR']
  lat,long = g.lat_lon(client_ip)
  return render_to_response('home_page_tmp.html',locals())

但是我得到这个错误:

KeyError at /mypage/
    'REMOTE_ADDR'
    Request Method: GET
    Request URL:    http://mywebsite.com/mypage/
    Django Version: 1.2.4
    Exception Type: KeyError
    Exception Value:    
    'REMOTE_ADDR'
    Exception Location: /mysite/homepage/views.py in home, line 9
    Python Executable:  /usr/bin/python
    Python Version: 2.6.6
    Python Path:    ['/mysite', '/usr/local/lib/python2.6/dist-packages/flup-1.0.2-py2.6.egg', '/usr/lib/python2.6', '/usr/lib/python2.6/plat-linux2', '/usr/lib/python2.6/lib-tk', '/usr/lib/python2.6/lib-old', '/usr/lib/python2.6/lib-dynload', '/usr/local/lib/python2.6/dist-packages', '/usr/lib/python2.6/dist-packages', '/usr/lib/pymodules/python2.6']
    Server time:    Sun, 2 Jan 2011 20:42:50 -0600

2
尝试转储request.META.keys()
Martin v。Löwis

2
[“ HTTP_COOKIE”,“ SCRIPT_NAME”,“ REQUEST_METHOD”,“ PATH_INFO”,“ SERVER_PROTOCOL”,“ QUERY_STRING”,“ CONTENT_LENGTH”,“ HTTP_ACCEPT_CHARSET”,“ HTTP_USER_AGENT”,“ HTTP_CONNECTION”,“ SERVER_NAME”,“ wsgi。 ,“ SERVER_PORT”,“ wsgi.input”,“ HTTP_HOST”,“ wsgi.multithread”,“ HTTP_CACHE_CONTROL”,“ HTTP_ACCEPT”,“ wsgi.version”,“ wsgi.run_once”,“ wsgi.errors”和“ wsgi”。多进程”,“ HTTP_ACCEPT_LANGUAGE”,“ CONTENT_TYPE”,“ CSRF_COOKIE”,“ HTTP_ACCEPT_ENCODING”]
头像

2
谢谢您提出这个好问题。我的fastcgi没有传递REMOTE_ADDR元密钥。我在nginx.conf中添加了以下行,并修复了该问题:fastcgi_param REMOTE_ADDR $ remote_addr;
阿凡达

Answers:


434
def get_client_ip(request):
    x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR')
    if x_forwarded_for:
        ip = x_forwarded_for.split(',')[0]
    else:
        ip = request.META.get('REMOTE_ADDR')
    return ip

确保正确配置了反向代理(如果有)(例如,mod_rpaf为Apache安装)。

注意:上面使用的第一X-Forwarded-For,但您可能要使用最后一项(例如,在Heroku的情况下:在Heroku上获取客户端的真实IP地址

然后将请求作为参数传递给它;

get_client_ip(request)

8
调用ip = get_client_ip(request)您的视图函数。
扬琴科2011年

4
真正的客户端IP地址不是HTTP_X_FORWARDED_FOR中的第一个,而是最后一个(请参阅Wikipedia页面)
2013

5
@jujule这是不正确的。格式通常为X-Forwarded-For: client, proxy1, proxy2。因此,第一个地址是客户的地址。
迈克尔瀑布

51
此功能很危险。通过许多设置,恶意用户很容易导致此功能返回他们想要的任何地址(而不是其真实地址)。参见esd.io/blog/flask-apps-heroku-real-ip-spoofing.html
Eli

8
从Django文档“依靠REMOTE_ADDR或类似的值被广泛地已知为最坏的实践”(djangoproject.com/weblog/2009/jul/28/security/#secondary-issue
牛头犬

209

您可以使用支持Python 23并处理IPv4IPv6的django-ipware

安装:

pip install django-ipware

简单用法:

# In a view or a middleware where the `request` object is available

from ipware import get_client_ip
ip, is_routable = get_client_ip(request)
if ip is None:
    # Unable to get the client's IP address
else:
    # We got the client's IP address
    if is_routable:
        # The client's IP address is publicly routable on the Internet
    else:
        # The client's IP address is private

# Order of precedence is (Public, Private, Loopback, None)

高级用法:

  • 自定义标头-ipware的自定义请求标头可以查看:

    i, r = get_client_ip(request, request_header_order=['X_FORWARDED_FOR'])
    i, r = get_client_ip(request, request_header_order=['X_FORWARDED_FOR', 'REMOTE_ADDR'])
  • 代理计数-Django服务器位于固定数量的代理之后:

    i, r = get_client_ip(request, proxy_count=1)
  • 受信任的代理-Django服务器位于一个或多个已知和受信任的代理之后:

    i, r = get_client_ip(request, proxy_trusted_ips=('177.2.2.2'))
    
    # For multiple proxies, simply add them to the list
    i, r = get_client_ip(request, proxy_trusted_ips=('177.2.2.2', '177.3.3.3'))
    
    # For proxies with fixed sub-domain and dynamic IP addresses, use partial pattern
    i, r = get_client_ip(request, proxy_trusted_ips=('177.2.', '177.3.'))

注意:请阅读此通知


17
看一下它的源代码。它可以处理此处其他答案所确定的所有复杂情况。
HostedMetrics.com 2014年

5
Thx @Heliodor-是的,对于普通的用例,我已经使该模块非常简单,对于复杂的用例,我已经使该模块非常灵活。最少,您需要在滚动自己的页面之前先查看其github页面。
un33k 2014年

3
注意,默认情况下,django-ipware的设置不安全!任何人都可以传递其他变量之一,您的站点将记录该IP。始终将其设置IPWARE_META_PRECEDENCE_LIST为您使用的变量,或使用诸如pypi.python.org/pypi/WsgiUnproxy之
vdboor 2015年

@vdboor您能详细说明一下吗?我在存储库中找不到IPWARE_META_PRECEDENCE_LIST。
Monolith

2
@ThaJay请注意,从2.0.0版开始,您应该使用get_client_ip()get_real_ip已弃用,并将在3.0中删除。
Un33k

77

Alexander的回答很好,但是缺乏代理处理功能,这些代理有时会在HTTP_X_FORWARDED_FOR标头中返回多个IP。

真实IP通常位于列表的末尾,如此处所述:http : //en.wikipedia.org/wiki/X-Forwarded-For

该解决方案是对Alexander代码的简单修改:

def get_client_ip(request):
    x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR')
    if x_forwarded_for:
        ip = x_forwarded_for.split(',')[-1].strip()
    else:
        ip = request.META.get('REMOTE_ADDR')
    return ip

5
是的,ip在列表的开头。这是错误的。
皮克勒2011年

4
实际上,如果用户位于代理之后,您将获得用户的内部IP地址,即RFC 1918地址。在大多数情况下,这根本不是很理想。此解决方案着重于获取客户端的外部IP地址(代理地址),这是最右边的地址。
Sævar

2
谢谢。通常,当我向我请求密钥时,request.META我会包含一个默认值,因为标头经常会让人误解:request.META.get('REMOTE_ADDR', None)
Carl G

2
@CarlG您的代码更加透明,但是get方法是从django.utils.datastructures.MultiValueDict继承的,默认值为None。但是,如果您实际上希望将默认值设置为None,则一定要包含默认值。
Sævar

2
除非您在请求到达您的第一台服务器时清理X-Forwarded-For,否则该列表中的第一个值是用户提供的。恶意用户可以轻易地欺骗他们想要的任何IP地址。您想要的地址是任何服务器之前的第一个IP,不一定是列表中的第一个IP。
伊莱

12

我想提出对扬琴科答案的改进。

我不采用X_FORWARDED_FOR列表中的第一个IP,而是采用第一个不知道内部IP的IP,因为某些路由器不遵守该协议,您可以将内部IP视为列表的第一个值。

PRIVATE_IPS_PREFIX = ('10.', '172.', '192.', )

def get_client_ip(request):
    """get the client ip from the request
    """
    remote_address = request.META.get('REMOTE_ADDR')
    # set the default value of the ip to be the REMOTE_ADDR if available
    # else None
    ip = remote_address
    # try to get the first non-proxy ip (not a private ip) from the
    # HTTP_X_FORWARDED_FOR
    x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR')
    if x_forwarded_for:
        proxies = x_forwarded_for.split(',')
        # remove the private ips from the beginning
        while (len(proxies) > 0 and
                proxies[0].startswith(PRIVATE_IPS_PREFIX)):
            proxies.pop(0)
        # take the first ip which is not a private one (of a proxy)
        if len(proxies) > 0:
            ip = proxies[0]

    return ip

希望这对遇到同样问题的Google同事有所帮助。


在检查HTTP_X_FORWARDED_FOR字段之前,此代码不会检查REMOTE_ADDR的IP是否为私有,因为它可能应该也(“ 127.0.0.1”或“ 127.”应该与IPv6等效项一起放在PRIVATE_IPS_PREFIX中。)
Rasmus Kaj

1
从技术上讲,这些前缀(172、192)不一定表示私有地址。
maniexx

2
分配给专用网络的地址范围是:172.16.0.0–172.31.255.255(16个“ B类”网络),192.168.0.0–192.168.255.255(1个“ B类”网络)和10.0.0.0–10.255.255.255(1 “ A类”或256个“ B类”网络)。
tzot

is_valid_ip没有定义
Prosenjit

7

这是完成此任务的简短说明:

request.META.get('HTTP_X_FORWARDED_FOR', request.META.get('REMOTE_ADDR', '')).split(',')[0].strip()

3
如果它们都返回None,那么您将得到一个错误。
Gourav Chawla

6

最简单的解决方案(如果您使用的是fastcgi + nignx)是itgorilla评论的内容:

谢谢您提出这个好问题。我的fastcgi没有传递REMOTE_ADDR元密钥。我在nginx.conf中添加了以下行,并修复了该问题:fastcgi_param REMOTE_ADDR $ remote_addr; – itgorilla

附:我添加这个答案只是为了使他的解决方案更明显。


1
Nginx(反向代理)和gunicorn的可比解决方案是什么?proxy_set_header REMOTE_ADDR $remote_addr;添加到nginx.conf并不能缓解问题。
哈桑·拜格

6

不再困惑在最新版本的Django中,明确提到客户端的Ip地址可从以下位置获得:

request.META.get("REMOTE_ADDR")

有关更多信息,请检查Django Docs


5

就我而言,上述方法均无效,因此我必须检查uwsgi+django源代码并在nginx中传递静态参数,并查看原因/方式,以下是我发现的内容。

环境信息:
python版本:2.7.5
Django版本:(1, 6, 6, 'final', 0)
nginx版本:nginx/1.6.0
uwsgi:2.0.7

环保设置信息:
nginx作为反向代理,在端口80 uwsgi处作为上游unix套接字侦听,最终将响应请求

Django配置信息:

USE_X_FORWARDED_HOST = True # with or without this line does not matter

Nginx的配置:

uwsgi_param      X-Real-IP              $remote_addr;
// uwsgi_param   X-Forwarded-For        $proxy_add_x_forwarded_for;
// uwsgi_param   HTTP_X_FORWARDED_FOR   $proxy_add_x_forwarded_for;

// hardcode for testing
uwsgi_param      X-Forwarded-For        "10.10.10.10";
uwsgi_param      HTTP_X_FORWARDED_FOR   "20.20.20.20";

在Django应用中获取所有参数:

X-Forwarded-For :       10.10.10.10
HTTP_X_FORWARDED_FOR :  20.20.20.20

结论:

因此,基本上,您必须在nginx中指定完全相同的字段/参数名称,并使用 request.META[field/param]在Django应用中。

现在,您可以决定是添加中间件(拦截器)还是仅HTTP_X_FORWARDED_FOR在某些视图中进行解析。


2

最初从Django中删除功能的原因是最终无法信任标头。原因是容易欺骗。例如,建议的配置Nginx反向代理的方法是:

add_header X-Forwarded-For $proxy_add_x_forwarded_for;
add_header X-Real-Ip       $remote_addr;

当您这样做时:

curl -H 'X-Forwarded-For: 8.8.8.8, 192.168.1.2' http://192.168.1.3/

您在myhost.com中的nginx将继续发送:

X-Forwarded-For: 8.8.8.8, 192.168.1.2, 192.168.1.3

X-Real-IP如果您盲目按照说明进行操作它将是第一个先前的代理的IP。

如果信任您的用户是个问题,您可以尝试以下方法django-xffhttps : //pypi.python.org/pypi/django-xff/


1

我在上面的答案中也缺少代理。我get_ip_address_from_requestdjango_easy_timezones使用

from easy_timezones.utils import get_ip_address_from_request, is_valid_ip, is_local_ip
ip = get_ip_address_from_request(request)
try:
    if is_valid_ip(ip):
        geoip_record = IpRange.objects.by_ip(ip)
except IpRange.DoesNotExist:
    return None

这是method get_ip_address_from_request,IPv4和IPv6就绪的:

def get_ip_address_from_request(request):
    """ Makes the best attempt to get the client's real IP or return the loopback """
    PRIVATE_IPS_PREFIX = ('10.', '172.', '192.', '127.')
    ip_address = ''
    x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR', '')
    if x_forwarded_for and ',' not in x_forwarded_for:
        if not x_forwarded_for.startswith(PRIVATE_IPS_PREFIX) and is_valid_ip(x_forwarded_for):
            ip_address = x_forwarded_for.strip()
    else:
        ips = [ip.strip() for ip in x_forwarded_for.split(',')]
        for ip in ips:
            if ip.startswith(PRIVATE_IPS_PREFIX):
                continue
            elif not is_valid_ip(ip):
                continue
            else:
                ip_address = ip
                break
    if not ip_address:
        x_real_ip = request.META.get('HTTP_X_REAL_IP', '')
        if x_real_ip:
            if not x_real_ip.startswith(PRIVATE_IPS_PREFIX) and is_valid_ip(x_real_ip):
                ip_address = x_real_ip.strip()
    if not ip_address:
        remote_addr = request.META.get('REMOTE_ADDR', '')
        if remote_addr:
            if not remote_addr.startswith(PRIVATE_IPS_PREFIX) and is_valid_ip(remote_addr):
                ip_address = remote_addr.strip()
    if not ip_address:
        ip_address = '127.0.0.1'
    return ip_address

0

在django.VERSION(2,1,1,'final',0)请求处理程序中

sock=request._stream.stream.raw._sock
#<socket.socket fd=1236, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0, laddr=('192.168.1.111', 8000), raddr=('192.168.1.111', 64725)>
client_ip,port=sock.getpeername()

如果您两次调用上面的代码,您可能会得到

AttributeError(“'_ io.BytesIO'对象没有属性'stream'”,)

AttributeError(“'LimitedStream'对象没有属性'raw'”)

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.