TL; DR
想要此流程:
ws://...
websocket client 1 <-----------> websocket client 2
^
|
server (send messages via views)
所以我有以下内容:
views.py
def alarm(request):
layer = get_channel_layer()
async_to_sync(layer.group_send)('events', {
'type': 'events.alarm',
'content': 'triggered'
})
return HttpResponse('<p>Done</p>')
consumers.py
class EventConsumer(JsonWebsocketConsumer):
def connect(self):
print('inside EventConsumer connect()')
async_to_sync(self.channel_layer.group_add)(
'events',
self.channel_name
)
self.accept()
def disconnect(self, close_code):
print('inside EventConsumer disconnect()')
print("Closed websocket with code: ", close_code)
async_to_sync(self.channel_layer.group_discard)(
'events',
self.channel_name
)
self.close()
def receive_json(self, content, **kwargs):
print('inside EventConsumer receive_json()')
print("Received event: {}".format(content))
self.send_json(content)
def events_alarm(self, event):
print('inside EventConsumer events_alarm()')
self.send_json(
{
'type': 'events.alarm',
'content': event['content']
}
)
在routing.py
中application = ProtocolTypeRouter({
'websocket': AllowedHostsOriginValidator(
AuthMiddlewareStack(
URLRouter(
chat.routing.websocket_urlpatterns,
)
)
),
})
其中的websocket_urlpatterns
websocket_urlpatterns = [
url(r'^ws/chat/(?P<room_name>[^/]+)/$', consumers.ChatConsumer),
url(r'^ws/event/$', consumers.EventConsumer),
]
urls.py
urlpatterns = [
url(r'^alarm/$', alarm, name='alarm'),
]
当我呼叫/alarm/
时,仅发出HTTP请求,并且消息未发送到websocket
以下是日志:
[2018/09/26 18:59:12] HTTP GET /alarm/ 200 [0.07, 127.0.0.1:60124]
我的目的是使django视图发送到组(用例是服务器将通知发送到组中所有连接的成员)。
我在这里缺少什么设置。
我正在运行Django频道2.1.3,以redis作为后端。 CHANNELS_LAYERS等已全部设置。
参考链接:
编辑: 我可以从视图中使用websocket-client发送消息
from websocket import create_connection
ws = create_connection("ws://url")
ws.send("Hello, World")
但是可以不使用上面的内容发送(不要创建连接)吗?
源代码:chat-app
答案 0 :(得分:4)
@Matthaus Woolard的概念很清晰。
这就是问题所在:
当我尝试从django视图发送消息时,客户端已断开连接。这是由于服务器在代码更改后重新启动而发生的。我刷新了浏览器,它开始工作。
严重错误
所以总结一下:
在使用同步消费者的情况下,在connect()中添加以下内容:
async_to_sync(self.channel_layer.group_add)('events', self.channel_name)
或在异步使用方的情况下添加此示例:
await self.channel_layer.group_add('events', self.channel_name)
创建如下视图:
def alarm(request):
layer = get_channel_layer()
async_to_sync(layer.group_send)('events', {
'type': 'events.alarm',
'content': 'triggered'
})
return HttpResponse('<p>Done</p>')