服务器 > 网络 > websocket

Django3框架-(3)-[使用websocket]:使用channels实现websocket功能;简化的配置和实际使用方式

87人参与 2024-08-03 websocket

概述:

对于django使用channels实现websocket的功能,之前就写了几篇博文了。随着在项目的使用和实际维护来说,重新设置了相关处理方法。

一般来说,前后端都只维护一个全局的连接,通过携带数据来判断具体的操作,大致的业务逻辑(非群聊功能):

1、前端主动发起连接,发送了数据给后端,后端获取到数据后,解析出前端需要的是啥数据,查询出数据,返回给前端。(一次请求一次返回了)

2、部分数据变化了,后端需要主动告知前端,让前端重新查询对应的数据。(实时更新数据)

一、依赖

python=3.9.0

包:

项目结构:

项目名

二、settings.py设置

#注册channels
installed_apps = [
   	...
    'channels',  # django通过其实现websocket
]

wsgi_application = 'heartfailure.wsgi.application'

#channels使用需要添加asgi_application
asgi_application = 'heartfailure.asgi.application'

#使用channel_layers需要配置通道
channel_layers = {
    "default": {
        #1、使用内存作为通道(开发使用)
        "backend": "channels.layers.inmemorychannellayer",
        #2、使用redis(上线使用)
        # 'backend': 'channels.layers.redischannellayer',
        # 'config': {
        #     'hosts': [('localhost', 6379)],
        # },
    }
}

#####1、 cors资源跨域共享配置
cors_origin_allow_all = true
cors_allow_methods = (
    'delete',
    'get',
    'options',
    'patch',
    'post',
    'put',
    'view',
)

cors_allow_headers = (
    'xmlhttprequest',
    'x_filename',
    'accept-encoding',
    'authorization',
    'content-type',
    'dnt',
    'origin',
    'user-agent',
    'x-csrftoken',
    'x-requested-with',
    'pragma',
    'token' #请求头允许自定义的字符串
)

三、创建websocket包

概述:将所有的wesocket相关的请求都放到一个包,集中管理。

 

websocket包下创建:

1、consumers.py

from channels.generic.websocket import websocketconsumer
from channels.exceptions import stopconsumer
from asgiref.sync import async_to_sync
import time
import json
# 接收到前端的websocket请求,直接向单个发送需要数据
from apps.websocket.send_data import base_send


class alldataconsumers(websocketconsumer):
    # 统一的房间名
    room_name = 'chat_all_data'

    def connect(self):
        cls = alldataconsumers
        self.room_group_name = cls.room_name
        # 加入到房间组内, self.channel_name是当前
        async_to_sync(self.channel_layer.group_add)(
            self.room_group_name, self.channel_name
        )
        headers = self.scope['headers']
        print(headers)
        token = none
        for key, value in headers:
            if key == b'token':
                token = value.decode('utf-8')
                print(token)

            # 同意创建连接
        self.accept()

    def disconnect(self, close_code):
        print('有浏览器退出了websocket!!!!')
        # leave room group
        async_to_sync(self.channel_layer.group_discard)(
            self.room_group_name, self.channel_name
        )

    # receive message from websocket
    def receive(self, text_data=none, bytes_data=none):
        '''
        :param text_data: 接收字符串类型的数据
        :param bytes_data:  接收bytes类型的数据
        :return:
        如果是浏览器直接请求时,就单独给这个浏览器返回结果,无需给房间组内的发送数据
        '''
        try:
            text_data_json = json.loads(text_data)
            the_type = text_data_json.get('type', 'none')
        except exception as e:
            self.send(
                json.dumps({'code': 400, 'msg': '传递的数据请按照{"type":"xx","id":x,"params":{}}格式'}, ensure_ascii=false))
            self.disconnect(400)
            return

        #1、前端主动请求websocket时,拿到对应的数据,单独给该websocket返回数据
        send_data = base_send(text_data_json)
        if isinstance(send_data,dict):
            #需要给请求的前端返回数据
            self.send(json.dumps(send_data, ensure_ascii=false))
        else:
            #无需给请求的前端返回数据
            pass

        # 2、将数据发送到房间组内 (在非聊天模式无需这样操作)
        # async_to_sync(self.channel_layer.group_send)(
        #     self.room_group_name, {'type':'send_to_chrome','data':send_data}
        # )
        '''
        参数说明:
        self.room_group, 给哪个房间组发送数据,
        {'type':'send_to_chrome','data':send_data}
            send_to_chrome 是处理函数,在这里负责将房间组内的数据发送给浏览器
            send_data  要发送的数据
        '''

    # 自定义的处理房间组内的数据:实时推送就是使用这个来实现的
    def send_to_chrome(self, event):
        try:
            data = event.get('data')
            # 接收房间组广播数据,将数据发送给websocket
            self.send(json.dumps(data, ensure_ascii=false))
        except exception as e:
            print('给全局的websocket推送消息失败')


2、send_data.py


def base_send(data:dict):
    '''
    功能:发起websocket请求时,给当前websocket返回数据
    :param data: {'type':'要操作的数据类型','id':'有id就是指定每个数据','params':{'page':'页码','page_size':'页面大小', }}
    :return:
    '''
    the_type = data.get('type')
    id = data.get('id')
    send_data = {
        'type':the_type,
        'data':'返回的数据'
    }
    #用户管理-搜索功能,用户信息是实时更新的
    if the_type == 'search_user_data':
        #前端发起websocket请求时,此类型时,无需返回数据
        return send_data

3、update.py

#channels包相关
from asgiref.sync import async_to_sync
from channels.layers import get_channel_layer

class alldataconsumersupdate:
    '''
    功能:在http视图中,给房间组=chat_all_data 推送指定的消息
    '''

    def _make_channel_layer(self,send_data):
        '''
        :param send_data: 在http视图中查询好的数据,要给房间组内所有的websocket对象发送数据
        '''
        channel_layer = get_channel_layer()
        #拿到房间组名
        group_name = 'chat_all_data'
        #给该房间组组内发送数据 注意是group_send方法
        async_to_sync(channel_layer.group_send)(
            group_name, #房间组名,给这个房间组发送数据
            {
                'type':'send_to_chrome', #处理这个房间组的消费者类必须有send_to_chrome方法
                'data':send_data   #要发送给websocket对象的数据
            }
        )
        '''
        send_to_chrome: 该房间组对应的消费者,必须存在这个函数,在这个函数中进行将数据发送给房间组所有的websocket对象
        send_data : 查询出来的数据
        '''
    #用户管理-搜索用户页面-实时更新数据,由前端自己去获取数据
    def search_user_data(self):
        send_data = {
            'type':'search_user_data',
            'page_update':1
        }
        #给房间发送数据
        self._make_channel_layer(send_data=send_data)
        return true

4、routings.py

from django.urls import path
from . import consumers

# 这个变量是存放websocket的路由
socket_urlpatterns = [
    path('socket/all/',consumers.alldataconsumers.as_asgi()),

]

四、修改settings.py同级的asgi.py文件

asgi.py

import os

from django.core.asgi import get_asgi_application

#新的模块
from channels.routing import protocoltyperouter, urlrouter
# 导入websocket的路由模块
from apps.websocket import routings

#项目名,settings.py所在的目录名
os.environ.setdefault('django_settings_module', '项目名.settings')

application = protocoltyperouter({
    # http路由走这里
    "http": get_asgi_application(),
    # chat应用下rountings模块下的路由变量socket_urlpatterns,就是存路由的列表
    "websocket": urlrouter(routings.socket_urlpatterns)
})

五、在视图函数中怎么发送websocket通知

from apps.websocket.update import websocket_update_obj #websocket推送数据的接口

#在视图函数中直接调用需要的方法就可以实现推送了
websocket_update_obj.search_user_data()

将所有的推送方法都放到一个类中,可以很方便的进行管理,后期修改时,也可以实现统一的修改。

六、启动项目

python manage.py runserver 8005

看到 asgi/channels version xxx 就说明启动成功,此时的django项目才支持websocket

七、测试

访问:easyswoole-websocket在线测试工具

(0)
打赏 微信扫一扫 微信扫一扫

您想发表意见!!点此发布评论

推荐阅读

前端如何使用WebSocket发送消息

08-03

Nginx 配置 WebSocket 代理

08-06

websocket获取实时数据的几种常见链接方式

08-06

websocket (@ServerEndpoint)基本使用指南

08-01

nginx代理webSocket链接,webSocket频繁断开重连方式

09-20

Nginx配置WebSocket代理的示例代码

10-14

猜你喜欢

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论