Python Django:在数据库save()上从服务器向客户端发送消息

问题描述 投票:2回答:1

我想在保存模型时通知客户端。我首先在post_save上创建了一个django信号。

@receiver(post_save, sender=Scooter)
async def scooter_post_update(sender, instance, created, **kwargs):
    # Notify client here

接下来,我从django-channels创建了AsyncConsumer类,并提供了其路由。

// routing.py
    application = ProtocolTypeRouter({
        # Empty for now (http->django views is added by default)
        'websocket': AllowedHostsOriginValidator(
            AuthMiddlewareStack(
                URLRouter(
                    [
                        path('scooters/', ScootersUpdateConsumer)
                    ]
               )
        )
    )
})

// consumers.py
    class ScootersUpdateConsumer(AsyncConsumer):
        async def websocket_connect(self, event):
            print("Connected!", event)
            await self.send({
                "type": "websocket.accept"
            })
        async def send_message(self):
            await self.send({
                "type": "websocket.send",
                'text': 'Oy, mate!'
            })
        async def websocket_receive(self, event):
            print("Receive!", event)
        async def websocket_disconnect(self, event):
            print("Disconnected!", event)

现在我的问题是如何从scooter_post_update()方法调用send_message()。

python django websocket django-channels django-signals
1个回答
1
投票

这些步骤非常简单。您必须获取通道层,然后只发送带有设置为type键的消息作为您的监听方法名称:

    import channels
    from asgiref.sync import async_to_sync

    @receiver(post_save, sender=Scooter)
    def scooter_post_update(sender, instance, created, **kwargs):
        channel_layer = channels.layers.get_channel_layer()
        async_to_sync(channel_layer.send)(
            {"type": "send_message", "data": data}
        )

以及您要通过渠道发送的其他任何内容。

请注意,您传递的所有数据必须必须是可序列化的,因此您必须注意,所有对象都已预先序列化。

您传递给send方法的词典的必需部分是type键(如前所述),它必须包含将在使用者上调用的方法名称。

此外,您可以使用组,因此可以向一组收听者广播消息:

    import channels
    from asgiref.sync import async_to_sync

    @receiver(post_save, sender=Scooter)
    def scooter_post_update(sender, instance, created, **kwargs):
        channel_layer = channels.layers.get_channel_layer()
        async_to_sync(channel_layer.group_send)(
            "group_name", {"type": "send_message", "data": data}
        )

在消费者方面:

    class ScootersUpdateConsumer(AsyncConsumer):
        async def websocket_connect(self, event):
            await self.channel_layer.group_add("group_name", self.channel_name)
            await self.send({
                "type": "websocket.accept"
            })

请注意,在两种情况下,您都使用async_to_sync包装器,当您从同步作用域调用异步代码时应使用该包装器。

© www.soinside.com 2019 - 2024. All rights reserved.