Сбой соединения WebSocket и неожиданное закрытие сокета в каналах Django

У меня возникли проблемы с установлением и поддержанием WebSocket-соединения в моем Django-проекте с помощью Django Channels. Я создал систему уведомлений, которая использует WebSockets для передачи сообщений подключенным клиентам. Однако я сталкиваюсь с двумя ошибками:

  1. «WebSocket-соединение с „ws://127.0.0.1:8000/ws/notification/broadcast/“ failed»
  2. «Сокет чата неожиданно закрыт». Вот мой код:

routing.py:

from django.urls import re_path  
from . import consumers  
  
websocket_urlpatterns = [  
   re_path(r'ws/notification/(?P<room_name>\w+)/$', consumers.NotificationConsumer.as_asgi()),  
]

views.py:

def notification_view(request):  
   return render(request,'person/notification.html',{'room_name': "broadcast"})

settings.py:

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

consumer_context_processor.py:

from notification_app.models import BroadcastNotification  
def notifications(request):  
   allnotifications = BroadcastNotification.objects.all()  
   return {'notifications': allnotifications}

models.py:

class BroadcastNotification(models.Model):  
   message = models.TextField()  
   notification_image=models.ImageField(upload_to="notification",default="notification.jpg",blank=True,null=True)  
   notification_link = models.URLField(max_length=10000, help_text="Add a valid URL",blank=True,null=True)  
   broadcast_on = models.DateTimeField()  
   sent = models.BooleanField(default=False)  
  
   class Meta:  
      ordering = ['-broadcast_on']  
  
@receiver(post_save, sender=BroadcastNotification)  
def notification_handler(sender, instance, created, **kwargs):  
   # call group_send function directly to send notificatoions or you can create a dynamic task in celery beat  
   if created:  
      schedule, created = CrontabSchedule.objects.get_or_create(hour = instance.broadcast_on.hour, minute = instance.broadcast_on.minute, day_of_month = instance.broadcast_on.day, month_of_year = instance.broadcast_on.month)  
      task = PeriodicTask.objects.create(crontab=schedule, name="broadcast-notification-"+str(instance.id), task="notifications_app.tasks.broadcast_notification", args=json.dumps((instance.id,)))

task.py:

@shared_task(bind = True)  
def broadcast_notification(self, data):  
   print(data)  
   try:  
      notification = BroadcastNotification.objects.filter(id = int(data))  
      if len(notification)>0:  
        notification = notification.first()  
        channel_layer = get_channel_layer()  
        loop = asyncio.new_event_loop()  
        asyncio.set_event_loop(loop)  
        loop.run_until_complete(channel_layer.group_send(  
           "notification_broadcast",  
           {  
              'type': 'send_notification',  
              'message': json.dumps(notification.message),  
           }))  
        notification.sent = True  
        notification.save()  
        return 'Done'  
  
      else:  
        self.update_state(  
           state = 'FAILURE',  
           meta = {'exe': "Not Found"}  
        )  
  
        raise Ignore()  
  
   except:  
      self.update_state(  
           state = 'FAILURE',  
           meta = {  
                'exe': "Failed"  
                # 'exc_type': type(ex).__name__,  
                # 'exc_message': traceback.format_exc().split('\n')  
                # 'custom': '...'  
              }  
        )  
  
      raise Ignore()

Фронтенд (JavaScript):

{{ room_name|json_script:"room-name" }}  
   <script>  
      const roomName = JSON.parse(document.getElementById('room-name').textContent);  
  
      const notificationSocket = new WebSocket(  
        'ws://'  
        + window.location.host  
        + '/ws/notification/'  
        + roomName  
        + '/'  
      );  
  
      notificationSocket.onmessage = function(e) {  
        const data = JSON.parse(e.data);  
        //document.querySelector('#chat-log').value += (data.message + '\n');  
        console.log(data);  
        document.getElementById("notifications-dropdown").innerHTML = "<li class='dropdown-item'>" + data + "</li><hr class='dropdown-divider'>" + document.getElementById("notifications-dropdown").innerHTML;  
        document.getElementById("notification-badge").innerHTML = parseInt(document.getElementById("notification-badge").innerHTML) + 1;  
      };  
  
      notificationSocket.onclose = function(e) {  
        console.error('Chat socket closed unexpectedly');  
      };  
   </script>

consumers.py:

class NotificationConsumer(AsyncWebsocketConsumer):  
   async def connect(self):  
      self.room_name = self.scope['url_route']['kwargs']['room_name']  
      self.room_group_name = 'notification_%s' % self.room_name  
  
      # Join room group  
      await self.channel_layer.group_add(  
        self.room_group_name,  
        self.channel_name  
      )  
  
      await self.accept()  
  
   async def disconnect(self, close_code):  
      # Leave room group  
      await self.channel_layer.group_discard(  
        self.room_group_name,  
        self.channel_name  
      )  
  
  
   # Receive message from room group  
   async def send_notification(self, event):  
      message = json.loads(event['message'])  
  
      # Send message to WebSocket  
      await self.send(text_data=json.dumps(message))

Я пытался следовать официальной документации и примерам Django Channels, но все равно столкнулся с этими проблемами. Может ли кто-нибудь помочь мне определить проблему и предложить решение?

Проверьте конфигурацию URL-адреса WebSocket Соединение WebSocket не удается:

WebSocket connection to 'ws://127.0.0.1:8000/ws/notification/broadcast/'  
failed
Problem: You're using ws://127.0.0.1:8000/ws/notification/broadcast/, but your routing pattern in routing.py has a regular expression that requires a room_name parameter, so the URL should actually look like:

``
ws://127.0.0.1:8000/ws/notification/broadcast/

Это означает, что вы передаете параметр room_name как часть URL-адреса WebSocket, и он должен соответствовать шаблону URL-адреса в файле routing.py. Скорее всего, это приводит к сбою соединения WebSocket.

Решение: Убедитесь, что имя комнаты правильно передается из фронтенда. Проверьте, правильно ли вы передаете имя_комнаты при создании URL-адреса WebSocket:

const notificationSocket = new WebSocket(
  'ws://' + window.location.host + '/ws/notification/' + roomName + '/'
);

Убедитесь, что roomName правильно задается во фронтенде, и оно должно соответствовать тому, что ожидает ваш routing.py. Судя по вашему текущему коду, похоже, что вы устанавливаете room_name в контексте notification_view как «broadcast», что правильно, но вы должны убедиться, что оно передается в WebSocket-соединение.

Вернуться на верх