【问题标题】:Django channels: Save messages to databaseDjango 频道:将消息保存到数据库
【发布时间】:2021-01-19 03:43:38
【问题描述】:

我是频道的新手,我按照他们的官方文档做了一个聊天室应用程序。现在我正在尝试保存聊天消息。我所知道的是我可以创建一个模型,但我知道如何将它从 consumer.py 保存到我的数据库中。我在消息中添加了用户名。一点帮助将不胜感激。

我的 Consumers.py:

import json
from channels.generic.websocket import AsyncWebsocketConsumer

class ChatConsumer(AsyncWebsocketConsumer):
    async def connect(self):
        self.room_name = self.scope['url_route']['kwargs']['room_name']
        self.room_group_name = 'chat_%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 WebSocket
    async def receive(self, text_data):
        username = self.scope["user"].first_name
        name = self.scope['user'].username
        text_data_json = json.loads(text_data)
        message = text_data_json['message']
        message = (username + '(' + name + ')' + ':\n' + message)

        # Send message to room group
        await self.channel_layer.group_send(
            self.room_group_name,
            {   
                'type': 'chat_message',
                'message': message
            }
        )

    # Receive message from room group
    async def chat_message(self, event):
        username = self.scope["user"].username
        message = event['message']
        name = self.scope["user"].username

        # Send message to WebSocket
        await self.send(text_data=json.dumps({
            'message': message,
            "username": username,
            "name": name
        }))

我保存消息的模型:

class Message(models.Model):
    author = models.ForeignKey(User, related_name='messages', on_delete=models.CASCADE)
    context = models.TextField()
    timestamp = models.DateTimeField(auto_now_add=True)

我也尝试按照本教程进行操作: https://www.youtube.com/watch?v=xrKKRRC518Y

【问题讨论】:

  • 你解决了吗?你能帮我找到你找到的解决方案吗?

标签: django django-models chat django-channels channels


【解决方案1】:

对于仍有此问题的用户:

由于使用的 WebSocket 是AsyncWebsocketConsumer,我们必须使用database_sync_to_async为什么会这样?因为我们不想让打开数据库连接

你会这样做:

@database_sync_to_async
def create_chat(self, msg, sender):
    new_msg = Message.objects.create(sender=sender, msg=msg)
    new_msg.save()

这是一个简单的例子来演示如何成功使用database_sync_to_async

import json

from channels.db import database_sync_to_async
from channels.generic.websocket import AsyncWebsocketConsumer

from .models import Message


class ChatConsumer(AsyncWebsocketConsumer):
    @database_sync_to_async
    def create_chat(self, msg, sender):
        new_msg = Message.objects.create(sender=sender, msg=msg)
        new_msg.save()
        return new_msg

    async def connect(self):
        self.room_name = self.scope['url_route']['kwargs']['room_name']
        self.room_group_name = 'chat_%s' % self.room_name
        await self.channel_layer.group_add(self.room_group_name, self.channel_name)
        await self.accept()

    async def disconnect(self, close_code):
        await self.channel_layer.group_discard(self.room_group_name, self.channel_name)

    async def receive(self, text_data):
        text_data_json = json.loads(text_data)
        message = text_data_json['message']
        sender = text_data_json['sender']
        await self.channel_layer.group_send(self.room_group_name, {
            'type': 'chat_message',
            'message': message,
            'sender': sender
        })

    async def chat_message(self, event):
        message = event['message']
        sender = event['sender']
        new_msg = await self.create_chat(sender, message)  # It is necessary to await creation of messages
        await self.send(text_data=json.dumps({
            'message': new_msg.message,
            'sender': new_msg.sender
        }))

【讨论】:

    【解决方案2】:

    定期创建消息有什么问题?

    message = Message.objects.create(context=message, author=self.scope['user'])
    

    【讨论】:

    • 哦,我没试过。我相信我应该把它放在接收函数中吧?
    • 我想是的,是的,不是超级经验丰富的渠道,但取决于您在哪里执行此操作,您可能会收到重复的消息。所以你需要尝试一下。
    • 但这会出错,因为它不能在异步上下文中使用
    • 你知道怎么做吗?我有完全相同的问题:/
    猜你喜欢
    • 1970-01-01
    • 2020-10-10
    • 1970-01-01
    • 2021-11-22
    • 2020-07-16
    • 2015-03-11
    • 1970-01-01
    • 2012-08-30
    • 2017-12-01
    相关资源
    最近更新 更多