Django WebSocket DISCONNECT / ws / chat /обби / [127.0.0.1:3022] - PullRequest
1 голос
/ 13 февраля 2020

Я хочу создать приложение чата, я следую https://channels.readthedocs.io/en/latest/tutorial/part_2.html здесь,

chat/
    __init__.py
    routing.py
    urls.py
    settings.py
    wsgi.py

я добавил этот код в свой routing.py

from channels.auth import AuthMiddlewareStack
from channels.routing import ProtocolTypeRouter, URLRouter
import chat.routing

application = ProtocolTypeRouter({
    # (http->django views is added by default)
    'websocket': AuthMiddlewareStack(
        URLRouter(
            chat.routing.websocket_urlpatterns
        )
    ),
})

в моем settings.py

ASGI_APPLICATION = 'Accounting.routing.application'
CHANNEL_LAYERS = {
    'default': {
        'BACKEND': 'channels_redis.core.RedisChannelLayer',
        'CONFIG': {
            "hosts": [('127.0.0.1', 6379)],
        },
    },
}

в моем urls.py

urlpatterns = [
    path('chat/', include('chat.urls')),
    path('admin/', admin.site.urls),
]

в моем приложении чата

chat/
    __init__.py
    consumers.py
    routing.py
    templates/
        chat/
            index.html
            room.html
    urls.py
    views.py

у меня есть consumer.py

from asgiref.sync import async_to_sync
from channels.generic.websocket import WebsocketConsumer
import json

class ChatConsumer(WebsocketConsumer):
    def connect(self):
        self.room_name = self.scope['url_route']['kwargs']['room_name']
        self.room_group_name = 'chat_%s' % self.room_name

        # Join room group
        async_to_sync(self.channel_layer.group_add)(
            self.room_group_name,
            self.channel_name
        )

        self.accept()

    def disconnect(self, close_code):
        # 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):
        text_data_json = json.loads(text_data)
        message = text_data_json['message']

        # Send message to room group
        async_to_sync(self.channel_layer.group_send)(
            self.room_group_name,
            {
                'type': 'chat_message',
                'message': message
            }
        )

    # Receive message from room group
    def chat_message(self, event):
        message = event['message']

        # Send message to WebSocket
        self.send(text_data=json.dumps({
            'message': message
        }))

это код в моем чате> routing.py

from django.urls import re_path

from . import consumers

websocket_urlpatterns = [
    re_path(r'ws/chat/(?P<room_name>\w+)/$', consumers.ChatConsumer),
]

в моем чате> views.py

from django.shortcuts import render

def index(request):
    return render(request, 'chat/index.html', {})

def room(request, room_name):
    return render(request, 'chat/room.html', {
        'room_name': room_name
    })

у меня есть urls.py в моем приложении чата

из django .urls путь импорта

from . import views
app_name = 'chat'
urlpatterns = [
    path('', views.index, name='index'),
    path('<str:room_name>/', views.room, name='room'),
]

Я следую всем направлениям, копирую, вставляю код, размещение py, все в turorial, но все равно я получаю эту ошибку

enter image description here

я что-то пропустил?

ОБНОВЛЕНИЕ

enter image description here

когда я попробовал это в моем терминале pycharm docker run -p 6379: 6379 -d redis: 2.8

enter image description here

1 Ответ

0 голосов
/ 18 февраля 2020

здесь есть некоторые проблемы.

1) вы смешиваете async и sync потребительский код.

Я предлагаю просто использовать async consumer все ваши методы, тогда должны быть async методы.

Похоже, вы вызываете get_thread из контекста syn c, но get_thread был обернут в database_sync_to_async, поэтому должен вызываться из асинхронного c контекста (и его следует ожидать).

вам также нужно await self.channel_layer.group_add

2) в вашем ThreadView у вас есть ChatMessage.objects.create здесь вы также должны отправить сообщение через группу каналов потока.

from channels.layers import get_channel_layer
channel_layer = get_channel_layer()

async_to_sync(channel_layer.group_send)(f"thread_{thread.id}", {"type": "chat.message", "text": ...})

вам также нужно сохранить сообщение чата в БД в вашем websocket_receive.

...