Nginx和Flask-socketio Websockets:活着但不是消息?

我一直在让Nginx与Python Flask-socketio库(基于gevent)很好地玩。 目前,由于我们正在积极开发,我试图让Nginx只是作为代理工作。 对于发送页面,我可以通过直接运行flask-socketio应用程序或通过运行gunicorn得到这个工作。 一个顺手:websocket消息传递似乎没有工作。 页面成功托pipe和显示。 但是,当我尝试使用websockets,他们不工作。 他们还活着,websocket认为它是连接的,但他们不会发送消息。 如果我删除了Nginx的代理,他们的工作。 当我尝试发送消息时,Firefox给了我这个错误:

Firefox无法在ws:///socket.io/1/websocket/build立与服务器的连接。

Web地址是服务器所在的位置,唯一的ID就是一堆随机数字。 这似乎足以保持连接生活(例如,客户端认为它是连接),但不能通过websocket发送消息。 我不得不认为这个问题与代理的某个部分有关,但是在debugging问题时可能会遇到很大的麻烦(部分原因是这是我第一次使用Flask-socketIO和nginx)。 我用于nginx的configuration文件是:

user <user name>; ## This is set to the user name for the remote SSH session worker_processes 5; events { worker_connections 1024; ## Default: 1024 } http { default_type application/octet-stream; log_format main '$remote_addr - $remote_user [$time_local] $status ' '"$request" $body_bytes_sent "$http_referer" ' '"$http_user_agent" "$http_x_forwarded_for"'; sendfile on; server_names_hash_bucket_size 128; # this seems to be required for some vhosts server { listen 80; server_name _; location / { proxy_pass http://localhost:8000; proxy_http_version 1.1; proxy_set_header Upgrade $http_upgrade; proxy_set_header Connection "upgrade"; proxy_set_header Host $host; } } } 

我把configuration文件作为一个通用的例子和一个websocket特定的汞合金,但试图摆弄它没有解决这个问题。 另外,当我在wsgi模式下使用werkzeug Proxy_Fix调用Flask app.wsgi_app时, 我已经尝试过,没有,没有用,但是。 如果有人有一些洞察力,我会全神贯注。

我设法解决这个问题。 这些问题并不是特定于flask-socketio,但是它们是针对Ubuntu,NginX和gevent-socketio的。 存在两个重要的问题:

  1. Ubuntu 12.04有一个真正古老的nginx版本(稳定版本1.1.19 vs 1.6.x)。 为什么? 谁知道。 我们知道的是,这个版本不支持任何有用的websockets,因为1.3.13是最早应该使用的。
  2. 默认情况下,gevent-socketio希望你的套接字位于/socket.io位置。 您可以升级整个HTTP连接,但我有一些麻烦,使之正常工作(尤其是我把SSL投入混合后)。
  3. 我修复了#1,但是在摆弄它的时候,我用nginx和apt-get安装了… Ubuntu上默认的nginx版本。 然后,我为什么事情比以前更糟糕地神秘地混淆了。 许多.conf文件勇敢地在这场战斗中失去了生命。

如果试图在这个配置中调试websocket,我会推荐以下步骤:

  1. 通过'nginx -v'检查你的nginx版本。 如果它小于1.4,则升级它。
  2. 检查你的nginx.conf设置。 您需要确保连接升级。
  3. 检查您的服务器IP和端口是否与您的nginx.conf反向代理相匹配。
  4. 检查你的客户端(例如socketio.js)是否使用正确的协议连接到正确的位置和端口。
  5. 检查您的阻止端口。 我在EC2上,所以你必须手动打开80(HTTP)和443(SSL / HTTPS)。

刚刚检查了所有这些东西,就有外卖。

  1. 在Ubuntu上升级到最新的稳定的nginx版本( full ref )可以通过:

     sudo apt-get install python-software-properties sudo apt-get install software-properties-common sudo add-apt-repository ppa:nginx/stable sudo apt-get update sudo apt-get install nginx 

    在像Windows这样的系统中,您可以使用安装程序,并且不太可能得到错误的版本。

  2. 许多配置文件可能会令人困惑,因为nginx在2013年正式添加套接字,使得早期的解决方法配置过时。 现有的配置文件并不涵盖nginx,gevent-socketio和SSL的所有基础,而是将它们分开( Nginx Tutorial , Gevent-socketio , Node.js和SSL )。 使用flask-socketio(包装gevent-socketio)和SSL的nginx 1.6的配置文件是:

     user <user account, probably optional>; worker_processes 2; error_log /var/log/nginx/error.log; pid /var/run/nginx.pid; events { worker_connections 1024; } http { include mime.types; default_type application/octet-stream; access_log /var/log/nginx/access.log; sendfile on; # tcp_nopush on; keepalive_timeout 3; # tcp_nodelay on; # gzip on; client_max_body_size 20m; index index.html; map $http_upgrade $connection_upgrade { default upgrade; '' close; } server { # listn on 80 and 443 listen 80 default; listen 443 ssl; (only needed if you want SSL/HTTPS) server_name <your server name here, optional unless you use SSL>; # SSL Certificate (only needed if you want SSL/HTTPS) ssl_certificate <file location for your unified .crt file>; ssl_certificate_key <file location for your .key file>; # Optional: Redirect all non-SSL traffic to SSL. (if you want ONLY SSL/HTTPS) # if ($ssl_protocol = "") { # rewrite ^ https://$host$request_uri? permanent; # } # Split off basic traffic to backends location / { proxy_pass http://localhost:8081; # 127.0.0.1 is preferred, actually. proxy_redirect off; } location /socket.io { proxy_pass http://127.0.0.1:8081/socket.io; # 127.0.0.1 is preferred, actually. proxy_redirect off; proxy_buffering off; # Optional proxy_http_version 1.1; proxy_set_header Upgrade $http_upgrade; proxy_set_header Connection $connection_upgrade; } } } 
  3. 检查你的Flask-socketio使用正确的端口很容易。 这足以与上面的工作:

     from flask import Flask, render_template, session, request, abort import flask.ext.socketio FLASK_CORE_APP = Flask(__name__) FLASK_CORE_APP.config['SECRET_KEY'] = '12345' # Luggage combination SOCKET_IO_CORE = flask.ext.socketio.SocketIO(FLASK_CORE_APP) @FLASK_CORE_APP.route('/') def index(): return render_template('index.html') @SOCKET_IO_CORE.on('message') def receive_message(message): return "Echo: %s"%(message,) SOCKET_IO_CORE.run(FLASK_CORE_APP, host=127.0.0.1, port=8081) 
  4. 对于像socketio.js这样的客户端来说,连接应该很简单。 例如:

     <script type="text/javascript" src="//cdnjs.cloudflare.com/ajax/libs/socket.io/0.9.16/socket.io.min.js"></script> <script type="text/javascript"> var url = window.location.protocol + document.domain + ':' + location.port, socket = io.connect(url); socket.on('message', alert); io.emit("message", "Test") </script> 
  5. 打开端口实际上更多的是服务器故障或超级用户的问题,因为它将取决于你的防火墙很多。 对于Amazon EC2,请参阅此处 。

  6. 如果尝试所有这些都行不通,那就哭吧。 然后返回到列表的顶部。 因为你可能只是不小心重新安装了旧版本的nginx。