Начиная с этого очень простого примера рабочего кода:
from channels.generic.websocket import JsonWebsocketConsumer
class IsacomptaManagementFeesConsumer(JsonWebsocketConsumer):
pass
При подключении к этому потребителю веб-сокета с javascript это работает как ожидалось. Соединение установлено правильно, и я получаю следующие журналы:
WebSocket HANDSHAKING /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:38108]
WebSocket CONNECT /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:38108]
WebSocket DISCONNECT /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:38108]
Теперь, если я изменил свой код на следующий код, чтобы использовать группы:
from channels.generic.websocket import JsonWebsocketConsumer
class IsacomptaManagementFeesConsumer(JsonWebsocketConsumer):
groups = ['foobar']
Затем соединение выходит из строя. JavaScript консоль говорит мне:
Firefox can’t establish a connection to the server at ws://antoine.cocoonr.hq:3001/manager/accounting/isacompta/2020/03/management-fees.ws.
error { target: WebSocket, isTrusted: true, srcElement: WebSocket, currentTarget: WebSocket, eventPhase: 2, bubbles: false, cancelable: false, defaultPrevented: false, composed: false, timeStamp: 7804, … }
И журналы сервера выглядят так:
WebSocket HANDSHAKING /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:38128]
WebSocket DISCONNECT /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:38128]
Хотя на стороне сервера не возникает никаких исключений.
I также можно получить аналогичное поведение без использования групп. Давайте рассмотрим пример с большим рабочим кодом:
from channels.generic.websocket import JsonWebsocketConsumer
class IsacomptaManagementFeesConsumer(JsonWebsocketConsumer):
def connect(self):
print("one")
self.accept()
print("two")
self.send_json({'text': "Foobar"})
print("three")
Этот код работает нормально, соединение установлено правильно, и вот журналы сервера:
WebSocket HANDSHAKING /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:38168]
one
WebSocket CONNECT /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:38168]
two
three
WebSocket DISCONNECT /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:38168]
Но если я сделаю следующее небольшое изменение в моем коде:
from asgiref.sync import async_to_sync
from channels.generic.websocket import JsonWebsocketConsumer
class IsacomptaManagementFeesConsumer(JsonWebsocketConsumer):
def connect(self):
print("one")
self.accept()
print("two")
async_to_sync(self.channel_layer.send)(self.channel_name, {
'type': 'foobar.send',
'text': "Foobar",
})
print("three")
def foobar_send(self, event):
print("AAA")
self.send_json({'text': event['text'])
print("BBB")
Затем соединение установлено правильно, но сразу закрывается, и процесс останавливается без возможности вывести «три», а функция «foobar_send» не Выполнено либо.
WebSocket HANDSHAKING /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:38224]
one
WebSocket CONNECT /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:38224]
two
WebSocket DISCONNECT /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:38224]
Я не понимаю, почему «три» не печатается, в то время как исключение также не возникает. Это означает, что исключение вызвано на self.channel_layers.send()
и перехвачено вызывающей стороной IsacomptaManagementFeesConsumer.connect()
?
После написания последнего абзаца я решил попробовать:
import traceback
from asgiref.sync import async_to_sync
from channels.generic.websocket import JsonWebsocketConsumer
class IsacomptaManagementFeesConsumer(JsonWebsocketConsumer):
def connect(self):
print("one")
self.accept()
print("two")
try:
async_to_sync(self.channel_layer.send)(self.channel_name, {
'type': 'foobar.send',
'text': "Foobar",
})
except Exception as e:
print(e)
traceback.print_stack()
print("three")
def foobar_send(self, event):
print("AAA")
self.send_json({'text': event['text'])
print("BBB")
И это все, есть скрытая ошибка «файл не найден»:
WebSocket HANDSHAKING /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:42562]
one
WebSocket CONNECT /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:42562]
two
[Errno 2] No such file or directory
File "/usr/lib64/python3.6/threading.py", line 884, in _bootstrap
self._bootstrap_inner()
File "/usr/lib64/python3.6/threading.py", line 916, in _bootstrap_inner
self.run()
File "/usr/lib64/python3.6/threading.py", line 864, in run
self._target(*self._args, **self._kwargs)
File "/usr/lib64/python3.6/concurrent/futures/thread.py", line 69, in _worker
work_item.run()
File "/usr/lib64/python3.6/concurrent/futures/thread.py", line 56, in run
result = self.fn(*self.args, **self.kwargs)
File "/home/tony/.venvs/cocoonr/lib/python3.6/site-packages/channels/db.py", line 14, in thread_handler
return super().thread_handler(loop, *args, **kwargs)
File "/home/tony/.venvs/cocoonr/lib/python3.6/site-packages/asgiref/sync.py", line 277, in thread_handler
return func(*args, **kwargs)
File "/home/tony/.venvs/cocoonr/lib/python3.6/site-packages/channels/consumer.py", line 105, in dispatch
handler(message)
File "/home/tony/.venvs/cocoonr/lib/python3.6/site-packages/channels/generic/websocket.py", line 39, in websocket_connect
self.connect()
File "/home/tony/Workspace/cocoonr/billing/consumers.py", line 32, in connect
traceback.print_stack()
three
WebSocket DISCONNECT /manager/accounting/isacompta/2020/03/management-fees.ws [192.168.96.1:42562]
Я использую Django 3.0 с каналами 2.4.0 и channel-redis 2.4.2. Вот мой параметр CHANNEL_LAYERS:
CHANNEL_LAYERS = {
'default': {
'BACKEND': 'channels_redis.core.RedisChannelLayer',
'CONFIG': {
'hosts': [
{
'address': get_env('CHANNELS_REDIS_HOST',
default='localhost:6379'),
'password': get_env('CHANNELS_REDIS_PASSWORD',
default=None),
'db': 1,
},
],
},
},
}