我正在做一个网站,用户可以登录和下载文件,使用Flask微框架(基于Werkzeug),它使用Python(在我的情况下是2.6)。

我需要在用户登录时获得用户的IP地址(用于登录目的)。 有人知道怎么做吗?肯定有一种方法可以用Python来实现吗?


当前回答

下面的代码总是给出客户端的公共IP(而不是代理背后的私有IP)。

from flask import request

if request.environ.get('HTTP_X_FORWARDED_FOR') is None:
    print(request.environ['REMOTE_ADDR'])
else:
    print(request.environ['HTTP_X_FORWARDED_FOR']) # if behind a proxy

其他回答

Httpbin.org使用这个方法:

return jsonify(origin=request.headers.get('X-Forwarded-For', request.remote_addr))

代理可能会让这有点棘手,如果你正在使用ProxyFix (Flask docs),一定要检查一下。看一下request。在您特定的环境中环境。使用nginx,我有时会这样做:

from flask import request   
request.environ.get('HTTP_X_REAL_IP', request.remote_addr)   

当代理(如nginx)转发地址时,它们通常在请求头中包含原始IP。

更新参见flask-security实现。同样,在实现ProxyFix之前,请查看有关ProxyFix的文档。您的解决方案可能根据您的特定环境而有所不同。

用户的IP地址可以使用以下片段检索:

from flask import request
print(request.remote_addr)

如果你使用Gunicorn和Nginx环境,那么下面的代码模板适合你。

addr_ip4 = request.remote_addr

下面是最简单的解决方案,以及如何在生产中使用。

from flask import Flask, request
from werkzeug.middleware.proxy_fix import ProxyFix
app = Flask(__name__)
# Set environment from any X-Forwarded-For headers if proxy is configured properly
app.wsgi_app = ProxyFix(app.wsgi_app, x_host=1)

@app.before_request
def before_process():
   ip_address = request.remote_addr

添加include proxy_params到/etc/nginx/sites-available/$project。

  location / {
    proxy_pass http://unix:$project_dir/gunicorn.sock;
    include proxy_params;
  }

include proxy_params转发ProxyFix解析的以下报头。

$ sudo cat /etc/nginx/proxy_params 
proxy_set_header Host $http_host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Proto $scheme;