从一个uwsgi向运行Django Channels的另一个uwsgi实例分发请求

时间:2019-03-22 19:48:40

标签: django websocket redis uwsgi django-channels

我目前正在使用Django渠道进行网络套接字通信。我阅读了这篇文章,并指出我应该将该项目分成两个uwsgi实例。声明

“ Web服务器承担将正常请求分配给一个uWSGI实例,将WebSocket请求分配给另一个uWSGI实例的任务”

现在我有两个正在运行的uwsgi实例。这就是我两者的运行方式。

此uwsgi处理正常的Django网站请求

uwsgi --virtualenv /home/ec2-user/MyProjVenv --socket /home/ec2-user/MyProjVenv/MyProjWeb/site1.socket --chmod-socket=777 --buffer-size=32768 --workers=5 --master --module main.wsgi

此uwsgi处理websocket请求

uwsgi --virtualenv /home/ec2-user/MyProjVenv --http-socket /home/ec2-user/MyProjVenv/MyProjWeb/web.socket --gevent 1000 --http-websockets --workers=2 --master --chmod-socket=777  --module main.wsgi_websocket

现在,websocket uwsgi启动main.wsgi_websocket

main.wsgi_websocket 1的代码是这个

import os
import gevent.socket
import redis.connection
redis.connection.socket = gevent.socket
os.environ.update(DJANGO_SETTINGS_MODULE='main.settings')
from ws4redis.uwsgi_runserver import uWSGIWebsocketServer
application = uWSGIWebsocketServer()

现在将两个uwsgi实例分解后,我可以正常访问网站。websocket uwsgi实例也可以接收数据,但是我不确定是否将数据传递给django网站uwsgi实例,该实例基本上使用django通道和具有用于发送/接收功能的处理程序。我在这里使用Django频道,这是我在Django频道设置中指定的配置

CHANNEL_LAYERS = {
    "default": {
        "BACKEND": "asgi_redis.RedisChannelLayer",
        "CONFIG": {
            "hosts": [(redis_host, 6379)],
        },
       "ROUTING": "main.routing.channel_routing", 
    },
}

频道路由是这个

channel_routing = [
    include("chat.routing.websocket_routing", path=r"^/chat/stream"),
    include("chat.routing.custom_routing"),
]

这是我拥有的websocket_routing

websocket_routing = [
    route("websocket.connect", ws_connect),


    # Called when WebSockets get sent a data frame
    route("websocket.receive", ws_receive),


    # Called when WebSockets disconnect
    route("websocket.disconnect", ws_disconnect),
]

现在的问题是我的ws_receive从未被调用。如果我使用ipaddress:8000/chat/stream在本地开发机上进行测试,则可以很好地工作,但是我不知道为什么在使用i padress/ws/时没有调用我的接收。我确定我的其他uwsgi实例正在获取该数据,但我不知道如何确定是否将其传递给djnago端的其他uwsgi实例,如果是这样,为什么我的接收没有被调用?关于此的任何建议肯定会有所帮助

1 个答案:

答案 0 :(得分:0)

当我在这里Nginx with Daphne gives 502 Bad Gateway

看到您的其他问题时,我对此感到疑惑

拆分项目是一个好主意。我假设这两个实例在nginx之后运行(来自您上面的问题)。

那么,nginx应该决定哪个请求转到哪个实例?您可以为每个channelsdjango应用使用不同的URL路径。

示例:

对于Django应用:/whatever/whatever/...  并为channels/ws/whatever/...

让我们假设您的channels消费者实例在8000上。

将此添加到您的nginx

location /ws/ {
            proxy_pass http://0.0.0.0:8000;
            proxy_http_version 1.1;
            proxy_set_header Upgrade $http_upgrade;
            proxy_set_header Connection "upgrade";

            proxy_redirect     off;
            proxy_set_header   Host $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-Host $server_name;
        }

通过这种方式,无论您的查询是什么,如果url以/ws/开头,则该端口将被端口8000上运行的任何实例所消耗。