Websocket, Angular 2 and JSON Web token Authentication

后端 未结 3 2120
死守一世寂寞
死守一世寂寞 2021-01-31 18:58

My Angular 2 app (coded in typescript) has a simple authentication scheme:

  • User logs in:
  • Server returns JSON Web Token (JWT) abc123...
  • <
3条回答
  •  爱一瞬间的悲伤
    2021-01-31 19:30

    Use djangorestframework-jwt to generated your JWTs, and the following Django-Channels 2 middleware.

    The token can be set via the djangorestframework-jwt http APIs, and it will also be sent for WebSocket connections if JWT_AUTH_COOKIE is defined.

    settings.py

    JWT_AUTH = {
        'JWT_AUTH_COOKIE': 'JWT',     # the cookie will also be sent on WebSocket connections
    }
    

    routing.py:

    from channels.routing import ProtocolTypeRouter, URLRouter
    from django.urls import path
    from json_token_auth import JsonTokenAuthMiddlewareStack
    from yourapp.consumers import SocketCostumer
    
    application = ProtocolTypeRouter({
        "websocket": JsonTokenAuthMiddlewareStack(
            URLRouter([
                path("socket/", SocketCostumer),
            ]),
        ),
    
    })
    

    json_token_auth.py

    from http import cookies
    
    from channels.auth import AuthMiddlewareStack
    from django.contrib.auth.models import AnonymousUser
    from django.db import close_old_connections
    from rest_framework_jwt.authentication import BaseJSONWebTokenAuthentication
    
    
    class JsonWebTokenAuthenticationFromScope(BaseJSONWebTokenAuthentication):
        """
        Extracts the JWT from a channel scope (instead of an http request)
        """
    
        def get_jwt_value(self, scope):
            try:
                cookie = next(x for x in scope['headers'] if x[0].decode('utf-8') == 'cookie')[1].decode('utf-8')
                return cookies.SimpleCookie(cookie)['JWT'].value
            except:
                return None
    
    
    class JsonTokenAuthMiddleware(BaseJSONWebTokenAuthentication):
        """
        Token authorization middleware for Django Channels 2
        """
    
        def __init__(self, inner):
            self.inner = inner
    
        def __call__(self, scope):
    
            try:
                # Close old database connections to prevent usage of timed out connections
                close_old_connections()
    
                user, jwt_value = JsonWebTokenAuthenticationFromScope().authenticate(scope)
                scope['user'] = user
            except:
                scope['user'] = AnonymousUser()
    
            return self.inner(scope)
    
    
    def JsonTokenAuthMiddlewareStack(inner):
        return JsonTokenAuthMiddleware(AuthMiddlewareStack(inner))
    
    

提交回复
热议问题