Django频道用户名发送者

时间:2020-09-07 19:37:40

标签: django django-channels

我是django频道的新手,我遵循了文档教程,并建立了django聊天室,到目前为止,我可以发送和接收消息,但是发件人的问题不明,我试图发送用户名,但它不起作用,我在服务器上打印了用户名,但在前面我得到了实际的登录用户名?我很困惑吗?

consumer.py:

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
        self.user = self.scope["user"]
        # 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):
        text_data_json = json.loads(text_data)
        message = text_data_json['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):
        message = event['message']
        print(self.user.username)

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

room.html:

<!-- chat/templates/chat/room.html -->
<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8"/>
    <title>Chat Room</title>

<script src="https://ajax.googleapis.com/ajax/libs/jquery/3.5.1/jquery.min.js"></script>
</head>
<body>
<div style="width:800px; margin:0 auto;">username :
     <a  id="username" href="">{{ user.get_username }}</a>
     <textarea id="chat-log" cols="100" rows="20"></textarea><br/>
    <input id="chat-message-input" type="text" size="75"/><br/>
    <input id="chat-message-submit" type="button" value="Send"/>
</div>

</body>
<script>
    var roomName = {{ room_name_json }};

    var chatSocket = new WebSocket(
        'ws://' + window.location.host +
        '/ws/chat/' + roomName + '/');

    chatSocket.onmessage = function(e) {
        var data = JSON.parse(e.data);
        var message = data['message'];
        console.log(e.data)

        document.querySelector('#chat-log').value += ( message + '\n');
    };

    chatSocket.onclose = function(e) {
        console.error('Chat socket closed unexpectedly');
    };

    document.querySelector('#chat-message-input').focus();
    document.querySelector('#chat-message-input').onkeyup = function(e) {
        if (e.keyCode === 13) {  // enter, return
            document.querySelector('#chat-message-submit').click();
        }
    };

    document.querySelector('#chat-message-submit').onclick = function(e) {
        var messageInputDom = document.querySelector('#chat-message-input');
        var message = messageInputDom.value;
        chatSocket.send(JSON.stringify({
            'message': message
        }));

        messageInputDom.value = '';
    };
</script>
</html>

2 个答案:

答案 0 :(得分:3)

我正在研究相同的问题,并测试了适用于我的解决方案。我将登录的用户名附加到邮件中。

# Receive message from WebSocket
async def receive(self, text_data):
        
    username = self.scope["user"].username # store logged-in user from WebSocket scope
    text_data_json = json.loads(text_data)
    message = text_data_json['message']
    message = (username + ': ' + message) # add to message here
    
    # 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']
    
    # Send message to WebSocket
    await self.send(text_data=json.dumps({
        'message': message,
        "username": username,
    }))

答案 1 :(得分:1)

用户连接时,他获得了ChatConsumer的单独实例,并且您设置了self.user。如果他从其他渠道收到消息,则self.user不会改变,仍然是您为他设置的消息。因此,要显示发件人的用户名,您必须将其与以下消息一起发送:

    ...
    # Receive message from WebSocket
    async def receive(self, text_data):
        text_data_json = json.loads(text_data)
        message = text_data_json['message']

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

    # Receive message from room group
    async def chat_message(self, event):
        message = event['message']
        username = event['username']
        print(username)

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

请注意,ChatConsumer的单独实例运行方法receivechat_message,因此这些方法中的self.user是不同的。虽然receive方法是在发送者的通道中执行的,但是chat_message方法是在接收者的通道中执行的,在这种情况下,接收者的通道也将包括发送者,因为他与接收者属于同一组。 例如,要查看您收到的消息是否是您发送的,可以在chat_message方法中执行以下操作:

if event['username'] == self.user.username:
    ...