Сбой соединения WebSocket и неожиданное закрытие сокета в каналах Django
У меня возникли проблемы с установлением и поддержанием WebSocket-соединения в моем Django-проекте с помощью Django Channels. Я создал систему уведомлений, которая использует WebSockets для передачи сообщений подключенным клиентам. Однако я сталкиваюсь с двумя ошибками:
- «WebSocket-соединение с „ws://127.0.0.1:8000/ws/notification/broadcast/“ failed»
- «Сокет чата неожиданно закрыт». Вот мой код:
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-соединение.