Как получить IP-адрес пользователя в django?
Как получить IP-адрес пользователя в django?
У меня есть вид вот так:
# 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
Ответы
Ответ 1
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
, но вы можете использовать элемент last (например, в случае Heroku: Получить реальный IP-адрес клиента на Heroku)
А затем просто передайте запрос в качестве аргумента для него;
get_client_ip(request)
Ответ 2
Вы можете использовать django-ipware, который поддерживает Python 2 & 3 и обрабатывает IPv4 и IPv6.
Установка:
pip install django-ipware
Простое использование:
Получить IP-адрес клиента.
# 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 IP address
else:
# We got the client IP address
if is_routable:
# The client IP address is publicly routable on the Internet
else:
# The client IP address is private
# Order of precedence is (Public, Private, Loopback, None)
Расширенное использование:
Custom Header - пользовательский заголовок запроса для 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.'))
Примечание: прочитайте это уведомление.
Ответ 3
Ответ Alexander велик, но не хватает обработки прокси-серверов, которые иногда возвращают несколько IP-адресов в заголовок HTTP_X_FORWARDED_FOR.
Реальный IP обычно находится в конце списка, как описано здесь: http://en.wikipedia.org/wiki/X-Forwarded-For
Решение - простая модификация кода Александра:
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
Ответ 4
Я хотел бы предложить улучшение ответа Янченко.
Вместо того, чтобы принимать первый ip в списке X_FORWARDED_FOR, я беру первый, который не является известным внутренним ip, поскольку некоторые маршрутизаторы не уважают протокол, и вы можете видеть внутреннее ips как первое значение списка.
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, которые имеют ту же проблему.
Ответ 5
Самое простое решение (если вы используете fastcgi + nignx) - это то, что комментирует itgorilla:
Спасибо за этот замечательный вопрос. Мой fastcgi не передавал мета-ключ REMOTE_ADDR. Я добавил строку ниже в nginx.conf и исправил проблему: fastcgi_param REMOTE_ADDR $remote_addr; - itgorilla
Ps: Я добавил этот ответ, чтобы сделать его решение более заметным.
Ответ 6
В моем случае ничего из этого не работает, поэтому я должен проверить исходный код uwsgi
+ django
и передать статический параметр в nginx и посмотреть, почему/как, а ниже - то, что я нашел.
Информация о Env:
версия python: 2.7.5
Версия Django: (1, 6, 6, 'final', 0)
Версия nginx: nginx/1.6.0
uwsgi: 2.0.7
Информация о настройке Env:
nginx как обратный прокси-сервер, прослушивающий порт 80
uwsgi как восходящий сокет unix, будет отвечать на запрос в конце концов
Информация о конфигурации Django:
USE_X_FORWARDED_HOST = True # with or without this line does not matter
nginx config:
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
в определенных представлениях.
Ответ 7
Вот короткий вкладыш для этого:
request.META.get('HTTP_X_FORWARDED_FOR', request.META.get('REMOTE_ADDR', '')).split(',')[0].strip()
Ответ 8
Причина, по которой функциональность была удалена из 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/
Ваш nginx в myhost.com отправит вперед:
X-Forwarded-For: 8.8.8.8, 192.168.1.2, 192.168.1.3
X-Real-IP
будет IP-адресом первого предыдущего прокси-сервера, если вы будете следовать инструкциям вслепую.
Если вам доверяют, кто ваши пользователи, вы можете попробовать что-то вроде django-xff
: https://pypi.python.org/pypi/django-xff/
Ответ 9
Я также пропустил прокси в ответе выше. Я использовал get_ip_address_from_request
из django_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
И вот метод get_ip_address_from_request
, IPv4 и IPv6 готовы:
def get_ip_address_from_request(request):
""" Makes the best attempt to get the client 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
Ответ 10
request.META.get('REMOTE_ADDR')
здесь будет достаточно.
Ответ 11
В 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 "не имеет атрибута" поток "",)
AttributeError ("Объект" LimitedStream "не имеет атрибута" raw "")
Ответ 12
Самое простое решение этого:
from ipaddress import ip_address
а затем используйте его как:
print(get_client_ip(request))