django에서 사용자 IP 주소를 어떻게 얻습니까? client_ip = request.META[‘REMOTE_ADDR’]

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



답변

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)


답변

Python 2 & 3 을 지원 하고 IPv4 & IPv6을 처리 하는 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.'))

참고 : 이주의 사항을 읽으십시오 .


답변

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


답변

yanchenko의 답변에 개선을 제안하고 싶습니다.

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 직원에게 도움이되기를 바랍니다.


답변

이 작업을 수행하는 짧은 하나의 라이너가 있습니다.

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


답변

가장 간단한 해결책 (fastcgi + nignx를 사용하는 경우)은 itgorilla가 다음과 같이 언급 한 것입니다.

이 위대한 질문에 감사드립니다. 내 fastcgi가 REMOTE_ADDR 메타 키를 전달하지 못했습니다. nginx.conf에 아래 줄을 추가하고 문제를 해결했습니다. fastcgi_param REMOTE_ADDR $ remote_addr; –이고 릴라

추신 : 나는 그의 해결책을 더 잘 보이게하기 위해이 대답을 추가했습니다.


답변

더 이상 혼동하지 않아도 최신 버전의 Django에서는 클라이언트의 IP 주소를

request.META.get("REMOTE_ADDR")

자세한 내용은 Django Docs를 확인하십시오.