-
Notifications
You must be signed in to change notification settings - Fork 2
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #116 from gcivil-nyu-org/added-chat-functionality
added chat functionality
- Loading branch information
Showing
19 changed files
with
358 additions
and
4 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
from channels.routing import ProtocolTypeRouter, URLRouter | ||
from chat import routing as core_routing | ||
|
||
application = ProtocolTypeRouter( | ||
{"websocket": URLRouter(core_routing.websocket_urlpatterns)} | ||
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Empty file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
# Register your models here. | ||
from django.contrib import admin | ||
|
||
from .models import Message | ||
|
||
admin.site.register(Message) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
from django.apps import AppConfig | ||
|
||
|
||
class ChatConfig(AppConfig): | ||
default_auto_field = "django.db.models.BigAutoField" | ||
name = "chat" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,48 @@ | ||
# consumers.py | ||
|
||
import json | ||
from channels.generic.websocket import AsyncWebsocketConsumer | ||
from .models import Message | ||
from asgiref.sync import sync_to_async | ||
from django.contrib.auth.models import User # <-- Add this import | ||
|
||
|
||
class ChatConsumer(AsyncWebsocketConsumer): | ||
async def connect(self): | ||
self.room_name = self.scope["url_route"]["kwargs"]["room_name"] | ||
self.room_group_name = f"chat_{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) | ||
|
||
async def receive(self, text_data): | ||
data = json.loads(text_data) | ||
message = data["message"] | ||
sender_id = data["sender_id"] | ||
recipient_id = data["recipient_id"] | ||
|
||
await self.save_message(sender_id, recipient_id, message) | ||
|
||
# Send message to room group | ||
await self.channel_layer.group_send( | ||
self.room_group_name, {"type": "chat.message", "message": message} | ||
) | ||
|
||
@sync_to_async | ||
def save_message(self, sender_id, recipient_id, message): | ||
sender = User.objects.get(id=sender_id) | ||
recipient = User.objects.get(id=recipient_id) | ||
|
||
Message.objects.create(sender=sender, recipient=recipient, content=message) | ||
|
||
# Receive message from room group | ||
async def chat_message(self, event): | ||
message = event["message"] | ||
|
||
# Send message to WebSocket | ||
await self.send(text_data=json.dumps({"message": message})) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,9 @@ | ||
# chat/forms.py | ||
|
||
from django import forms | ||
|
||
|
||
class MessageForm(forms.Form): | ||
message = forms.CharField( | ||
widget=forms.TextInput(attrs={"placeholder": "Type your message..."}) | ||
) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,48 @@ | ||
# Generated by Django 4.2.5 on 2023-11-12 01:24 | ||
|
||
from django.conf import settings | ||
from django.db import migrations, models | ||
import django.db.models.deletion | ||
|
||
|
||
class Migration(migrations.Migration): | ||
initial = True | ||
|
||
dependencies = [ | ||
migrations.swappable_dependency(settings.AUTH_USER_MODEL), | ||
] | ||
|
||
operations = [ | ||
migrations.CreateModel( | ||
name="Message", | ||
fields=[ | ||
( | ||
"id", | ||
models.BigAutoField( | ||
auto_created=True, | ||
primary_key=True, | ||
serialize=False, | ||
verbose_name="ID", | ||
), | ||
), | ||
("content", models.TextField()), | ||
("timestamp", models.DateTimeField(auto_now_add=True)), | ||
( | ||
"recipient", | ||
models.ForeignKey( | ||
on_delete=django.db.models.deletion.CASCADE, | ||
related_name="received_messages", | ||
to=settings.AUTH_USER_MODEL, | ||
), | ||
), | ||
( | ||
"sender", | ||
models.ForeignKey( | ||
on_delete=django.db.models.deletion.CASCADE, | ||
related_name="sent_messages", | ||
to=settings.AUTH_USER_MODEL, | ||
), | ||
), | ||
], | ||
), | ||
] |
Empty file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,18 @@ | ||
# chat/models.py | ||
|
||
from django.db import models | ||
from django.contrib.auth.models import User | ||
|
||
|
||
class Message(models.Model): | ||
sender = models.ForeignKey( | ||
User, on_delete=models.CASCADE, related_name="sent_messages" | ||
) | ||
recipient = models.ForeignKey( | ||
User, on_delete=models.CASCADE, related_name="received_messages" | ||
) | ||
content = models.TextField() | ||
timestamp = models.DateTimeField(auto_now_add=True) | ||
|
||
def __str__(self): | ||
return f"{self.sender.username} to {self.recipient.username}: {self.content}" |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,7 @@ | ||
from django.urls import path | ||
|
||
from . import consumers | ||
|
||
websocket_urlpatterns = [ | ||
path("ws/chat/<str:room_name>/", consumers.ChatConsumer.as_asgi()), | ||
] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,52 @@ | ||
<!-- chat/chat.html --> | ||
|
||
<!DOCTYPE html> | ||
<html lang="en"> | ||
<head> | ||
<meta charset="UTF-8"> | ||
<meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||
<title>Chat with {{ recipient.username }}</title> | ||
</head> | ||
<body> | ||
<h2>Chat with {{ recipient.username }}</h2> | ||
<div id="chat-log"> | ||
{% for message in messages %} | ||
<p>{{ message.sender.username }}: {{ message.content }}</p> | ||
{% endfor %} | ||
</div> | ||
<form id="chat-form" method="post" action=""> | ||
{% csrf_token %} | ||
{{ form.message }} | ||
<button type="submit">Send</button> | ||
</form> | ||
|
||
<script> | ||
// WebSocket connection setup | ||
const chatSocket = new WebSocket( | ||
'ws://' + window.location.host + '/ws/chat/{{ recipient.id }}/' | ||
); | ||
|
||
// Handle incoming messages from WebSocket | ||
chatSocket.onmessage = function(e) { | ||
const data = JSON.parse(e.data); | ||
document.querySelector('#chat-log').innerHTML += '<p>' + data.message + '</p>'; | ||
}; | ||
|
||
// Handle form submission | ||
document.querySelector('#chat-form').addEventListener('submit', function(e) { | ||
e.preventDefault(); | ||
const messageInput = document.querySelector('#id_message'); | ||
const message = messageInput.value.trim(); | ||
|
||
if (message) { | ||
chatSocket.send(JSON.stringify({ | ||
'message': message, | ||
'sender_id': {{ request.user.id }}, | ||
'recipient_id': {{ recipient.id }} | ||
})); | ||
messageInput.value = ''; | ||
} | ||
}); | ||
</script> | ||
</body> | ||
</html> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,69 @@ | ||
# chat/tests.py | ||
|
||
from django.test import TestCase | ||
from django.contrib.auth.models import User | ||
from django.urls import reverse | ||
from .models import Message | ||
|
||
|
||
class ChatViewTests(TestCase): | ||
def setUp(self): | ||
# Create users for testing | ||
self.user1 = User.objects.create_user(username="user1", password="password1") | ||
self.user2 = User.objects.create_user(username="user2", password="password2") | ||
|
||
# Create a message for testing | ||
self.message = Message.objects.create( | ||
sender=self.user1, recipient=self.user2, content="Test message" | ||
) | ||
|
||
# URL for the chat view | ||
self.url = reverse( | ||
"chats:chat-with-user", kwargs={"recipient_id": self.user2.id} | ||
) | ||
|
||
def test_chat_view_requires_authentication(self): | ||
# Ensure the view requires authentication | ||
response = self.client.get(self.url) | ||
self.assertEqual(response.status_code, 302) # Redirect to login page | ||
|
||
def test_chat_view_accessible_after_authentication(self): | ||
# Log in user1 | ||
self.client.login(username="user1", password="password1") | ||
|
||
# Ensure the view is accessible after authentication | ||
response = self.client.get(self.url) | ||
self.assertEqual(response.status_code, 200) | ||
|
||
def test_chat_view_displays_messages(self): | ||
# Log in user1 | ||
self.client.login(username="user1", password="password1") | ||
|
||
# Ensure the view displays messages | ||
response = self.client.get(self.url) | ||
self.assertContains(response, "Test message") | ||
|
||
def test_chat_post_form_creates_message(self): | ||
# Log in user1 | ||
self.client.login(username="user1", password="password1") | ||
|
||
# Ensure posting the form creates a new message | ||
response = self.client.post(self.url, {"message": "New test message"}) | ||
self.assertEqual(response.status_code, 302) # Redirect after successful post | ||
|
||
# Verify that the new message exists in the database | ||
new_message = Message.objects.get(content="New test message") | ||
self.assertEqual(new_message.sender, self.user1) | ||
self.assertEqual(new_message.recipient, self.user2) | ||
|
||
def test_chat_post_form_redirects_correctly(self): | ||
# Log in user1 | ||
self.client.login(username="user1", password="password1") | ||
|
||
# Ensure posting the form redirects to the correct chat page | ||
response = self.client.post(self.url, {"message": "Another message"}) | ||
self.assertEqual(response.status_code, 302) # Redirect after successful post | ||
expected_redirect_url = reverse( | ||
"chats:chat-with-user", kwargs={"recipient_id": self.user2.id} | ||
) | ||
self.assertRedirects(response, expected_redirect_url) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
from django.urls import path | ||
from . import views | ||
|
||
app_name = "chats" | ||
urlpatterns = [ | ||
path("<str:recipient_id>/", views.chatting, name="chat-with-user"), | ||
# Add other chat app URLs here | ||
] |
Oops, something went wrong.