Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Asynchronous Notification #2388

Open
wants to merge 34 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
34 commits
Select commit Hold shift + click to select a range
b923859
commit 1
HanilJain Jun 29, 2024
4a2a6dc
commit 2
HanilJain Jun 29, 2024
d1c914e
commit
HanilJain Jul 1, 2024
3ecfeaf
notification working
HanilJain Jul 1, 2024
e725cf1
redis removed
HanilJain Jul 1, 2024
5dfd532
mem channel added
HanilJain Jul 1, 2024
cee5e5a
Merge branch 'main' into async_project
HanilJain Jul 1, 2024
589d25c
poetry update
HanilJain Jul 1, 2024
d7e3ed6
shifted to website app
HanilJain Jul 1, 2024
fa53157
pre-commit
HanilJain Jul 1, 2024
be388dd
Merge branch 'main' into async_project
DonnieBLT Jul 2, 2024
df73ca7
Apply suggestions from code review
HanilJain Jul 3, 2024
4920eff
Merge branch 'main' into async_project
HanilJain Jul 3, 2024
0dbfb52
Merge branch 'main' into async_project
HanilJain Jul 4, 2024
585b69f
comments revoked
HanilJain Jul 4, 2024
dbf2162
Merge branch 'async_project' of https://github.com/HanilJain/BLT into…
HanilJain Jul 4, 2024
da192ae
Merge branch 'main' into async_project
HanilJain Jul 13, 2024
74065bd
Merge branch 'main' into async_project
HanilJain Jul 16, 2024
5f3caab
Merge branch 'main' into async_project
HanilJain Jul 16, 2024
10d4857
Merge branch 'main' into async_project
HanilJain Sep 18, 2024
50103d4
Merge branch 'main' into async_project
HanilJain Sep 27, 2024
02161d5
Merge branch 'main' into async_project
HanilJain Oct 24, 2024
51aff8c
upgrade
HanilJain Oct 28, 2024
fdd96da
dep removed 1
HanilJain Oct 28, 2024
8463320
pusher integrates
HanilJain Oct 29, 2024
48b5a48
pusher added
HanilJain Oct 29, 2024
3896e16
Merge branch 'main' into async_project
HanilJain Oct 29, 2024
7cdc3c1
pusher removed
HanilJain Oct 29, 2024
210f1fe
Merge branch 'async_project' of https://github.com/HanilJain/BLT into…
HanilJain Oct 29, 2024
74376f4
updating requirements.txt
HanilJain Oct 29, 2024
95008bb
restore .gitignore
HanilJain Oct 29, 2024
ea7ed9e
update requirements.txt
HanilJain Oct 29, 2024
ef0e6c8
removed vagrant
HanilJain Oct 30, 2024
c5adccd
Merge branch 'main' into async_project
HanilJain Oct 30, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion Dockerfile
Original file line number Diff line number Diff line change
Expand Up @@ -21,10 +21,10 @@ RUN apt-get update && apt-get install -y \

RUN pip install poetry
RUN poetry config virtualenvs.create false
RUN poetry lock --no-update
RUN poetry install

RUN python manage.py migrate
RUN python manage.py loaddata website/fixtures/initial_data.json
# RUN python manage.py collectstatic
RUN python manage.py initsuperuser

16 changes: 0 additions & 16 deletions Vagrantfile

This file was deleted.

28 changes: 28 additions & 0 deletions blt/asgi.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
"""
ASGI config for channels_celery_project project.

It exposes the ASGI callable as a module-level variable named ``application``.

For more information on this file, see
https://docs.djangoproject.com/en/3.2/howto/deployment/asgi/
"""

import os

import django

os.environ.setdefault("DJANGO_SETTINGS_MODULE", "blt.settings")
django.setup()

from channels.auth import AuthMiddlewareStack
from channels.routing import ProtocolTypeRouter, URLRouter
from django.core.asgi import get_asgi_application

from website.routing import websocket_urlpatterns

application = ProtocolTypeRouter(
{
"http": get_asgi_application(),
"websocket": AuthMiddlewareStack(URLRouter(websocket_urlpatterns)),
}
)
11 changes: 11 additions & 0 deletions blt/settings.py
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,7 @@
# Application definition

INSTALLED_APPS = (
"daphne",
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think we need this

"django.contrib.admin",
"django.contrib.auth",
"django.contrib.contenttypes",
Expand Down Expand Up @@ -198,6 +199,15 @@

WSGI_APPLICATION = "blt.wsgi.application"

ASGI_APPLICATION = "blt.asgi.application"

CHANNEL_LAYERS = {
"default": {
"BACKEND": "channels.layers.InMemoryChannelLayer",
},
}


# Database
# https://docs.djangoproject.com/en/1.9/ref/settings/#databases

Expand Down Expand Up @@ -318,6 +328,7 @@
"0.0.0.0",
"blt.owasp.org",
"." + DOMAIN_NAME_PREVIOUS,
"blt.onrender.com",
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't think we need this

]

# Static files (CSS, JavaScript, Images)
Expand Down
2 changes: 2 additions & 0 deletions blt/urls.py
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@
BugHuntApiViewsetV2,
CompanyViewSet,
DomainViewSet,
FetchNotificationApiView,
FlagIssueApiView,
InviteFriendApiViewset,
IssueViewSet,
Expand Down Expand Up @@ -366,6 +367,7 @@
InviteFriendApiViewset.as_view(),
name="api_invite_friend",
),
path("api/notification/", FetchNotificationApiView.as_view(), name="notification"),
re_path(r"^scoreboard/$", ScoreboardView.as_view(), name="scoreboard"),
re_path(r"^issue/$", IssueCreate.as_view(), name="issue"),
re_path(
Expand Down
4 changes: 2 additions & 2 deletions docker-compose.yml
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,9 @@ version: "3"

services:
app:
command: "poetry run python manage.py runserver 0.0.0.0:8000"
command: "poetry run uvicorn blt.asgi:application --host 0.0.0.0 --port 8000"
build: .
volumes:
- .:/blt
ports:
- "8000:8000"
- "8000:8000"
3,856 changes: 1,793 additions & 2,063 deletions poetry.lock

Large diffs are not rendered by default.

15 changes: 3 additions & 12 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,6 @@ authors = ["OWASP BLT <[email protected]>"]
license = "AGPLv3"

[tool.poetry.dependencies]

python = "3.11.2"
python-dotenv = "^1.0.1"
toml = "^0.10.2"
Expand All @@ -21,7 +20,6 @@ django-import-export = "^4.1.0"
django-annoying = "^0.10.7"
dj-rest-auth = "^5.0.2"
tweepy = "^4.8.0"
Unidecode = "^1.3.8"
user-agents = "^2.2.0"
whitenoise = "^6.8.2"
django-debug-toolbar = "^4.4.6"
Expand All @@ -32,8 +30,6 @@ boto = "^2.49.0"
django-cors-headers = "^4.6.0"
protobuf = "^4.25.3"
django-storages = { extras = ["google"], version = "^1.14.3" }
django-timedeltafield = "^0.7.10"
EasyProcess = "^1.1"
"giturlparse.py" = "^0.0.5"
gunicorn = "^23.0.0"
oauthlib = "^3.2.0"
Expand All @@ -51,33 +47,28 @@ djangorestframework = "^3.15.2"
cffi = "^1.17.1"
django-mdeditor = "^0.1.20"
django-tz-detect = "^0.4.0"
django-tellme = "^0.7.3"
django-bootstrap-datepicker-plus = "^5.0.5"
django-star-ratings = "^0.9.2"
stripe = "^8.4.0"
django-environ = "^0.11.2"
django-humanize = "^0.1.2"
drf-yasg = "^1.21.8"
django-simple-captcha = "^0.6.0"
django-filter = "^24.3"
webdriver-manager = "^4.0.2"
pillow = "^10.4.0"
chromedriver-autoinstaller = "^0.6.4"
sentry-sdk = "^2.17.0"
bitcash = "^1.0.2"
pydantic = "^2.7.3"
pydantic_core = "^2.18.4"
channels = {version = "^4.1.0", extras = ["extras", "daphne"]}
uvicorn = "^0.30.1"
langchain = "^0.2.16"
langchain-community = "^0.2.17"
langchain-core = "^0.2.40"
langchain-openai = "^0.1.25"
unstructured = "^0.16.3"
Markdown = "^3.6"
faiss-cpu = "^1.8.0"
openai = "^1.52.2"
psutil = "^5.9.8"
python-bitcoinrpc = "^1.0"
sendgrid = "^6.11.0"
pusher = "^3.3.2"

[tool.poetry.group.dev.dependencies]
black = "^24.8.0"
Expand Down
66 changes: 66 additions & 0 deletions requirements.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
python-dotenv==1.0.1
toml==0.10.2
Django==5.1.1
dj-database-url==2.3.0
django-allauth==0.61.1
beautifulsoup4==4.12.3
colorthief==0.2.1
django-email-obfuscator==0.1.5
django-gravatar2==1.4.5
django-import-export==4.1.0
django-annoying==0.10.7
dj-rest-auth==5.0.2
tweepy==4.8.0
user-agents==2.2.0
whitenoise==6.7.0
django-debug-toolbar==4.4.6
selenium==4.24.0
pylibmc==1.6.1
psycopg2-binary==2.9.10
django-cors-headers==4.5.0
protobuf==4.25.3
django-storages[google]==1.14.3
giturlparse.py==0.0.5
gunicorn==23.0.0
oauthlib==3.2.0
parse==1.20.2
python-openid==2.2.5
pytesseract==0.3.13
pytz==2024.1
PyVirtualDisplay==3.0
requests==2.32.3
requests-oauthlib==1.3.1
six==1.16.0
tablib==3.2.0
ua-parser==0.18.0
djangorestframework==3.15.2
cffi==1.17.1
django-mdeditor==0.1.20
django-tz-detect==0.4.0
django-star-ratings==0.9.2
stripe==8.4.0
django-environ==0.11.2
django-humanize==0.1.2
drf-yasg==1.21.8
django-simple-captcha==0.6.0
django-filter==24.3
pillow==10.4.0
chromedriver-autoinstaller==0.6.4
sentry-sdk==2.17.0
bitcash==1.0.2
channels[extras,daphne]==4.1.0
uvicorn==0.30.1
langchain==0.2.16
langchain-community==0.2.17
langchain-core==0.2.40
langchain-openai==0.1.25
Markdown==3.6
openai==1.52.2
python-bitcoinrpc==1.0
sendgrid==6.11.0

# Development dependencies
black==24.8.0
isort==5.13.2
ruff==0.7.0
pre-commit==3.8.0
1 change: 0 additions & 1 deletion vagrant/requirements.txt

This file was deleted.

18 changes: 0 additions & 18 deletions vagrant/setup.sh

This file was deleted.

2 changes: 2 additions & 0 deletions website/admin.py
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@
Issue,
IssueScreenshot,
Monitor,
Notification,
Payment,
Points,
Project,
Expand Down Expand Up @@ -413,6 +414,7 @@ class TagAdmin(admin.ModelAdmin):
admin.site.register(Payment, PaymentAdmin)
admin.site.register(IssueScreenshot, IssueScreenshotAdmin)
admin.site.register(HuntPrize)
admin.site.register(Notification)
admin.site.register(ChatBotLog, ChatBotLogAdmin)
admin.site.register(Blocked, BlockedAdmin)
admin.site.register(Suggestion, SuggestionAdmin)
Expand Down
29 changes: 29 additions & 0 deletions website/api/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@
InviteFriend,
Issue,
IssueScreenshot,
Notification,
Points,
Project,
Tag,
Expand Down Expand Up @@ -701,6 +702,34 @@ class CompanyViewSet(viewsets.ModelViewSet):
http_method_names = ("get", "post", "put")


from asgiref.sync import async_to_sync
from channels.layers import get_channel_layer


class FetchNotificationApiView(APIView):
def get(self, request, *args, **kwargs):
user_id = request.query_params.get("user_id")
if user_id is None or not user_id.isdigit():
return Response("Invalid User Id, ID should be integer", status=400)
user_exists = User.objects.filter(id=user_id).exists()
if not user_exists:
return Response("User Does Not Exist", status=400)

notification = Notification.objects.filter(user__id=user_id).all()
messages = [n.message for n in notification]
notification_id = [n.id for n in notification]
channel_layer = get_channel_layer()
async_to_sync(channel_layer.group_send)(
f"notification_{user_id}",
{
"type": "send_notification",
"notification_id": notification_id,
"message": messages,
},
)
return Response("OK")


class ContributorViewSet(viewsets.ModelViewSet):
queryset = Contributor.objects.all()
serializer_class = ContributorSerializer
Expand Down
33 changes: 33 additions & 0 deletions website/consumers.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
import json

from asgiref.sync import sync_to_async
from channels.generic.websocket import AsyncWebsocketConsumer

from website.models import Notification


class NotificationConsumer(AsyncWebsocketConsumer):
async def connect(self):
self.room_name = self.scope["url_route"]["kwargs"]["room_name"]
self.room_group_name = "notification_%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)

async def receive(self, text_data):
data = json.loads(text_data)
notification_id = data["notification_id"]
notification = await sync_to_async(Notification.objects.get)(id=notification_id)
await sync_to_async(notification.delete)()

# Receive message from room group
async def send_notification(self, event):
message = event
# Send message to WebSocket
await self.send(text_data=json.dumps(message))
4 changes: 2 additions & 2 deletions website/fixtures/initial_data.json
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@
"model": "website.domain",
"pk": 55,
"fields": {
"name": "example.com",
"name": "example1.com",
"url": "http://example.com",
"logo": "",
"webshot": "",
Expand Down Expand Up @@ -45,7 +45,7 @@
"model": "website.domain",
"pk": 99,
"fields": {
"name": "owasp.org",
"name": "owasp",
"url": "http://owasp.org",
"logo": "",
"webshot": "",
Expand Down
Loading
Loading