在 Uwsgi 和 Nginx 上使用 SocketIO 的 Flask
Flask with SocketIO on Uwsgi and Nginx
我正在开发使用 WebSockets 与前端通信的 Flask 应用程序。它托管在 Amazon EC2 上,位于 nginx 后面,由 uwsgi 提供服务器。
这是我用来服务然后应用程序的 uwsgi 配置:
[uwsgi]
plugins=python3,logfile
chdir=/srv/myapp/
master=true
home=/srv/myapp/.venv
module=application
callable=flask_app
uid=uwsgi
gid=myapp
socket=/srv/myapp/uwsgi.sock
chown-socket=uwsgi:myapp
chmod-socket=660
logto = /srv/myapp/logs/uwsgi.log
for-readline = /srv/myapp/.vars
env = %(_)
endfor =
以及涵盖 socketio 端点的 nginx 配置摘录:
location /socket.io/ {
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;
proxy_http_version 1.1;
proxy_buffering off;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "Upgrade";
proxy_pass http://unix:///srv/myapp/uwsgi.sock;
}
没有 websockets 一切正常。在本地(我使用的是 Windows)一切也都很完美 - 我只是在客户端添加了 transport: ['websockets', 'polling']
以确保选择正确的协议。
我在本地 运行 按照 Flask-SocketIO 文档中的建议,我还安装了 eventlet
(我不知道为什么,但在 Windows x64 gevent
包装器在 Werkzeug
开发服务器上运行不佳。
部署应用程序后,我在浏览器中只看到错误 websocket.js:112 WebSocket connection to 'ws://myapp.com/socket.io/?EIO=3&transport=websocket' failed: Error during WebSocket handshake: Unexpected response code: 502
。在服务器端,我在 uwsgi 日志中有以下内容:invalid request block size: 21573 (max 4096)...skip
。增加缓冲区大小没有任何改变。
在 nginx 日志中我有:*413 upstream prematurely closed connection while reading response header from upstream, client: 171.6.248.10, server: localhost,, request: "GET /socket.io/?EIO=3&transport=websocket HTTP/1.1", upstream: "http://unix:///srv/myapp/uwsgi.sock:/socket.io/?EIO=3&transport=websocket", host: "myapp.com"
我试图将这些行添加到 uwsgi.ini:
gevent=1000
http-websockets=true
没有任何成功
它是什么以及如何解决?
这不是我问题的实际答案,只是一些信息
我的应用程序使用 Python 3,RedHat 提供来自 EPEL 的 Python3.4。尽管如此,UWSGI 只能使用 gevent
与 WebSockets 一起工作。所以,正如@Miguel 提到的,只有一种方法——使用长轮询。
解决方案
最后,我将 appserver 更改为开箱即用的 WebSockets 服务的 Gunicorn。
我正在开发使用 WebSockets 与前端通信的 Flask 应用程序。它托管在 Amazon EC2 上,位于 nginx 后面,由 uwsgi 提供服务器。
这是我用来服务然后应用程序的 uwsgi 配置:
[uwsgi]
plugins=python3,logfile
chdir=/srv/myapp/
master=true
home=/srv/myapp/.venv
module=application
callable=flask_app
uid=uwsgi
gid=myapp
socket=/srv/myapp/uwsgi.sock
chown-socket=uwsgi:myapp
chmod-socket=660
logto = /srv/myapp/logs/uwsgi.log
for-readline = /srv/myapp/.vars
env = %(_)
endfor =
以及涵盖 socketio 端点的 nginx 配置摘录:
location /socket.io/ {
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;
proxy_http_version 1.1;
proxy_buffering off;
proxy_set_header Upgrade $http_upgrade;
proxy_set_header Connection "Upgrade";
proxy_pass http://unix:///srv/myapp/uwsgi.sock;
}
没有 websockets 一切正常。在本地(我使用的是 Windows)一切也都很完美 - 我只是在客户端添加了 transport: ['websockets', 'polling']
以确保选择正确的协议。
我在本地 运行 按照 Flask-SocketIO 文档中的建议,我还安装了 eventlet
(我不知道为什么,但在 Windows x64 gevent
包装器在 Werkzeug
开发服务器上运行不佳。
部署应用程序后,我在浏览器中只看到错误 websocket.js:112 WebSocket connection to 'ws://myapp.com/socket.io/?EIO=3&transport=websocket' failed: Error during WebSocket handshake: Unexpected response code: 502
。在服务器端,我在 uwsgi 日志中有以下内容:invalid request block size: 21573 (max 4096)...skip
。增加缓冲区大小没有任何改变。
在 nginx 日志中我有:*413 upstream prematurely closed connection while reading response header from upstream, client: 171.6.248.10, server: localhost,, request: "GET /socket.io/?EIO=3&transport=websocket HTTP/1.1", upstream: "http://unix:///srv/myapp/uwsgi.sock:/socket.io/?EIO=3&transport=websocket", host: "myapp.com"
我试图将这些行添加到 uwsgi.ini:
gevent=1000
http-websockets=true
没有任何成功
它是什么以及如何解决?
这不是我问题的实际答案,只是一些信息
我的应用程序使用 Python 3,RedHat 提供来自 EPEL 的 Python3.4。尽管如此,UWSGI 只能使用 gevent
与 WebSockets 一起工作。所以,正如@Miguel 提到的,只有一种方法——使用长轮询。
解决方案
最后,我将 appserver 更改为开箱即用的 WebSockets 服务的 Gunicorn。