diff --git a/jaseci_serv/jaseci_serv/asgi.py b/jaseci_serv/jaseci_serv/asgi.py index 16006605b4..6e5984ebbd 100644 --- a/jaseci_serv/jaseci_serv/asgi.py +++ b/jaseci_serv/jaseci_serv/asgi.py @@ -8,9 +8,16 @@ """ import os - +from .socket.routing import websocket_urlpatterns from django.core.asgi import get_asgi_application +from channels.routing import ProtocolTypeRouter, URLRouter +from channels.auth import AuthMiddlewareStack os.environ.setdefault("DJANGO_SETTINGS_MODULE", "jaseci_serv.settings") -application = get_asgi_application() +application = ProtocolTypeRouter( + { + "http": get_asgi_application(), + "websocket": AuthMiddlewareStack(URLRouter(websocket_urlpatterns)), + } +) diff --git a/jaseci_serv/jaseci_serv/settings.py b/jaseci_serv/jaseci_serv/settings.py index 48c817525c..ec0bbe4261 100644 --- a/jaseci_serv/jaseci_serv/settings.py +++ b/jaseci_serv/jaseci_serv/settings.py @@ -37,6 +37,7 @@ # Application definition INSTALLED_APPS = [ + "daphne", "django.contrib.admin", "django.contrib.auth", "django.contrib.contenttypes", @@ -96,7 +97,7 @@ }, ] -WSGI_APPLICATION = "jaseci_serv.wsgi.application" +ASGI_APPLICATION = "jaseci_serv.asgi.application" # Database # https://docs.djangoproject.com/en/3.0/ref/settings/#databases diff --git a/jaseci_serv/jaseci_serv/socket/__init__.py b/jaseci_serv/jaseci_serv/socket/__init__.py new file mode 100644 index 0000000000..e69de29bb2 diff --git a/jaseci_serv/jaseci_serv/socket/consumer.py b/jaseci_serv/jaseci_serv/socket/consumer.py new file mode 100644 index 0000000000..4cd5b4da51 --- /dev/null +++ b/jaseci_serv/jaseci_serv/socket/consumer.py @@ -0,0 +1,65 @@ +from json import loads, dumps +from .event_action import authenticated_user +from channels.generic.websocket import WebsocketConsumer +from asgiref.sync import async_to_sync +from uuid import uuid4 +from jaseci_serv.base.models import lookup_global_config +from channels.layers import settings + + +class SocketConsumer(WebsocketConsumer): + def connect(self): + self.accept() + session_id = None + authenticated = False + target = self.scope["url_route"]["kwargs"]["target"] + + if target == "anonymous": + self.target = session_id = str(uuid4()) + else: + user = authenticated_user(target) + if user: + self.target = user.master.urn[9:] + authenticated = True + else: + self.target = session_id = str(uuid4()) + + async_to_sync(self.channel_layer.group_add)(self.target, self.channel_name) + self.send( + text_data=dumps( + { + "type": "connect", + "authenticated": authenticated, + "session_id": session_id, + } + ) + ) + + def receive(self, text_data=None, bytes_data=None): + data = loads(text_data) + + async_to_sync(self.channel_layer.group_send)( + self.target, {"type": "notify", "data": data} + ) + + def notify(self, data): + self.send(text_data=dumps(data)) + + +setattr( + settings, + "CHANNEL_LAYERS", + { + "default": loads( + lookup_global_config( + "CHANNEL_LAYER", + { + "BACKEND": "channels_redis.core.RedisChannelLayer", + "CONFIG": { + "hosts": [("localhost", 6379)], + }, + }, + ) + ) + }, +) diff --git a/jaseci_serv/jaseci_serv/socket/event_action.py b/jaseci_serv/jaseci_serv/socket/event_action.py new file mode 100644 index 0000000000..3a24565419 --- /dev/null +++ b/jaseci_serv/jaseci_serv/socket/event_action.py @@ -0,0 +1,39 @@ +try: + from hmac import compare_digest +except ImportError: + + def compare_digest(a, b): + return a == b + + +import binascii +from knox.crypto import hash_token +from knox.models import AuthToken +from knox.settings import CONSTANTS +from jaseci.jsorc.live_actions import jaseci_action +from channels.layers import get_channel_layer +from asgiref.sync import async_to_sync +from django.utils import timezone + + +def authenticated_user(token: str): + for auth_token in AuthToken.objects.filter( + token_key=token[: CONSTANTS.TOKEN_KEY_LENGTH] + ): + try: + digest = hash_token(token) + if ( + compare_digest(digest, auth_token.digest) + and auth_token.expiry > timezone.now() + ): + return auth_token.user + except (TypeError, binascii.Error): + pass + return None + + +@jaseci_action(act_group=["wb"]) +def notify(target: str, data: dict): + async_to_sync(get_channel_layer().group_send)( + target, {"type": "notify", "data": data} + ) diff --git a/jaseci_serv/jaseci_serv/socket/routing.py b/jaseci_serv/jaseci_serv/socket/routing.py new file mode 100644 index 0000000000..f65238fdb8 --- /dev/null +++ b/jaseci_serv/jaseci_serv/socket/routing.py @@ -0,0 +1,6 @@ +from django.urls import path +from . import consumer + +websocket_urlpatterns = [ + path(r"ws/socket-server/", consumer.SocketConsumer.as_asgi()) +] diff --git a/jaseci_serv/jaseci_serv/wsgi.py b/jaseci_serv/jaseci_serv/wsgi.py deleted file mode 100644 index 51adefd8ca..0000000000 --- a/jaseci_serv/jaseci_serv/wsgi.py +++ /dev/null @@ -1,16 +0,0 @@ -""" -WSGI config for jaseci project. - -It exposes the WSGI callable as a module-level variable named ``application``. - -For more information on this file, see -https://docs.djangoproject.com/en/3.0/howto/deployment/wsgi/ -""" - -import os - -from django.core.wsgi import get_wsgi_application - -os.environ.setdefault("DJANGO_SETTINGS_MODULE", "jaseci_serv.settings") - -application = get_wsgi_application() diff --git a/jaseci_serv/setup.py b/jaseci_serv/setup.py index b106ecd7b7..5f28de5515 100644 --- a/jaseci_serv/setup.py +++ b/jaseci_serv/setup.py @@ -48,6 +48,7 @@ def get_ver(): "dj-rest-auth[with_social]", "django-allauth>=0.52.0", "tzdata>=2022.7", + "channels[daphne]==4.0.0", ], package_data={ "": [ diff --git a/jaseci_serv/templates/examples/social_auth.html b/jaseci_serv/templates/examples/social_auth.html index 84943b8a5c..a6369d8453 100644 --- a/jaseci_serv/templates/examples/social_auth.html +++ b/jaseci_serv/templates/examples/social_auth.html @@ -145,4 +145,18 @@

Google Identity Services Authorization Token model

}); {% endif %} {% endif %} + +{% if provider == "google" %} +socket = new WebSocket(`ws://${window.location.host}/ws/socket-server/276a40aec1dffc48a25463c3e2545473b45a663364adf3a2f523b903aa254c9f`) +{% else %} +socket = new WebSocket(`ws://${window.location.host}/ws/socket-server/anonymous`) +{% endif %} +socket.onmessage = (event) => { + console.log(event) +} + +function notify-server() { + socket.send(JSON.stringify({"message": "test"})) +} + \ No newline at end of file