Couldn't implement real-time user-to-user chat app in Django REST framework
I am newbie in web development and I working on one project and in this project i have to create real-time user-to-user chat application using django rest framework. I have implemented this in django itself using channels package (created consumers, routing, templates, js files and some other stuff), but I am stuck now, don't know what to do. The goal is that I have to make an API in DRF and Flutter devs should link it.
Here is my models.py
class Thread(models.Model):
first_person = models.ForeignKey(User, on_delete=models.CASCADE, null=True,
blank=True, related_name='thread_first_person')
second_person = models.ForeignKey(User, on_delete=models.CASCADE, null=True,
blank=True, related_name='thread_second_person')
updated = models.DateTimeField(auto_now=True)
timestamp = models.DateTimeField(auto_now_add=True)
objects = ThreadManager()
class Meta:
unique_together = ['first_person', 'second_person']
class ChatMessage(models.Model):
thread = models.ForeignKey(Thread, null=True, blank=True, on_delete=models.CASCADE, related_name='chatmessage_thread')
user = models.ForeignKey(User, on_delete=models.CASCADE)
message = models.TextField()
timestamp = models.DateTimeField(auto_now_add=True)
consumers.py
import json
from channels.consumer import AsyncConsumer
from channels.db import database_sync_to_async
from django.contrib.auth import get_user_model
from chat.models import Thread, ChatMessage
User = get_user_model()
class ChatConsumer(AsyncConsumer):
async def websocket_connect(self, event):
print('connected', event)
user = self.scope['user']
chat_room = f'user_chatroom_{user.id}'
self.chat_room = chat_room
await self.channel_layer.group_add(
chat_room,
self.channel_name
)
await self.send({
'type': 'websocket.accept'
})
async def websocket_receive(self, event):
print('receive', event)
received_data = json.loads(event['text'])
msg = received_data.get('message')
sent_by_id = received_data.get('sent_by')
send_to_id = received_data.get('send_to')
thread_id = received_data.get('thread_id')
if not msg:
print('Error:: empty message')
return False
sent_by_user = await self.get_user_object(sent_by_id)
send_to_user = await self.get_user_object(send_to_id)
thread_obj = await self.get_thread(thread_id)
if not sent_by_user:
print('Error:: sent by user is incorrect')
if not send_to_user:
print('Error:: send to user is incorrect')
if not thread_obj:
print('Error:: Thread id is incorrect')
await self.create_chat_message(thread_obj, sent_by_user, msg)
other_user_chat_room = f'user_chatroom_{send_to_id}'
self_user = self.scope['user']
response = {
'message': msg,
'sent_by': self_user.id,
'thread_id': thread_id
}
await self.channel_layer.group_send(
other_user_chat_room,
{
'type': 'chat_message',
'text': json.dumps(response)
}
)
await self.channel_layer.group_send(
self.chat_room,
{
'type': 'chat_message',
'text': json.dumps(response)
}
)
async def websocket_disconnect(self, event):
print('disconnect', event)
async def chat_message(self, event):
print('chat_message', event)
await self.send({
'type': 'websocket.send',
'text': event['text']
})
@database_sync_to_async
def get_user_object(self, user_id):
qs = User.objects.filter(id=user_id)
if qs.exists():
obj = qs.first()
else:
obj = None
return obj
@database_sync_to_async
def get_thread(self, thread_id):
qs = Thread.objects.filter(id=thread_id)
if qs.exists():
obj = qs.first()
else:
obj = None
return obj
@database_sync_to_async
def create_chat_message(self, thread, user, msg):
ChatMessage.objects.create(thread=thread, user=user, message=msg)
routing.py
from django.urls import path
from . import consumers
websocket_urlpatterns = [
path('chat/', consumers.ChatConsumer.as_asgi()),
]
import os
from django.core.asgi import get_asgi_application
from channels.routing import ProtocolTypeRouter, URLRouter
from channels.auth import AuthMiddlewareStack
import chat.routing
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'myproject.settings')
application = ProtocolTypeRouter({
'http': get_asgi_application(),
'websocket': AuthMiddlewareStack(
URLRouter(
chat.routing.websocket_urlpatterns
)
)
})
messages.js
let input_message = $('#input-message')
let message_body = $('.msg_card_body')
let send_message_form = $('#send-message-form')
const USER_ID = $('#logged-in-user').val()
let loc = window.location
let wsStart = 'ws://'
if(loc.protocol === 'https') {
wsStart = 'wss://'
}
let endpoint = wsStart + loc.host + loc.pathname
var socket = new WebSocket(endpoint)
socket.onopen = async function(e){
console.log('open', e)
send_message_form.on('submit', function (e){
e.preventDefault()
let message = input_message.val()
let send_to = get_active_other_user_id()
let thread_id = get_active_thread_id()
let data = {
'message': message,
'sent_by': USER_ID,
'send_to': send_to,
'thread_id': thread_id
}
data = JSON.stringify(data)
socket.send(data)
$(this)[0].reset()
})
}
socket.onmessage = async function(e){
console.log('message', e)
let data = JSON.parse(e.data)
let message = data['message']
let sent_by_id = data['sent_by']
let thread_id = data['thread_id']
newMessage(message, sent_by_id, thread_id)
}
socket.onerror = async function(e){
console.log('error', e)
}
socket.onclose = async function(e){
console.log('close', e)
}
function newMessage(message, sent_by_id, thread_id) {
if ($.trim(message) === '') {
return false;
}
let message_element;
let chat_id = 'chat_' + thread_id
if(sent_by_id == USER_ID){
message_element = `
<div class="d-flex mb-4 replied">
<div class="msg_cotainer_send">
${message}
<span class="msg_time_send">8:55 AM, Today</span>
</div>
<div class="img_cont_msg">
<img src="" class="rounded-circle user_img_msg">
</div>
</div>
`
}
else{
message_element = `
<div class="d-flex mb-4 received">
<div class="img_cont_msg">
<img src="https://static.turbosquid.com/Preview/001292/481/WV/_D.jpg" class="rounded-circle user_img_msg">
</div>
<div class="msg_cotainer">
${message}
<span class="msg_time">8:40 AM, Today</span>
</div>
</div>
`
}
let message_body = $('.messages-wrapper[chat-id="' + chat_id + '"] .msg_card_body')
message_body.append($(message_element))
message_body.animate({
scrollTop: $(document).height()
}, 100);
input_message.val(null);
}
$('.contact-li').on('click', function (){
$('.contacts .actiive').removeClass('active')
$(this).addClass('active')
// message wrappers
let chat_id = $(this).attr('chat-id')
$('.messages-wrapper.is_active').removeClass('is_active')
$('.messages-wrapper[chat-id="' + chat_id +'"]').addClass('is_active')
})
function get_active_other_user_id(){
let other_user_id = $('.messages-wrapper.is_active').attr('other-user-id')
other_user_id = $.trim(other_user_id)
return other_user_id
}
function get_active_thread_id(){
let chat_id = $('.messages-wrapper.is_active').attr('chat-id')
let thread_id = chat_id.replace('chat_', '')
return thread_id
}
Summary: How I can skip front side (js, templates) and built not regular Django project but make an API of user-to-user chat app using DRF