mirror of
https://github.com/Tivix/django-rest-auth.git
synced 2025-06-30 17:43:04 +03:00
now also accepting authorization codes from social login (e.g. facebook)
This commit is contained in:
parent
8ea935ef40
commit
74f2ffec7f
|
@ -73,3 +73,4 @@ Basing on example from installation section :doc:`Installation </installation>`
|
||||||
- /rest-auth/facebook/ (POST)
|
- /rest-auth/facebook/ (POST)
|
||||||
|
|
||||||
- access_token
|
- access_token
|
||||||
|
- code
|
||||||
|
|
|
@ -10,7 +10,8 @@ except ImportError:
|
||||||
|
|
||||||
|
|
||||||
class SocialLoginSerializer(serializers.Serializer):
|
class SocialLoginSerializer(serializers.Serializer):
|
||||||
access_token = serializers.CharField(required=True)
|
access_token = serializers.CharField(required=False)
|
||||||
|
code = serializers.CharField(required=False)
|
||||||
|
|
||||||
def _get_request(self):
|
def _get_request(self):
|
||||||
request = self.context.get('request')
|
request = self.context.get('request')
|
||||||
|
@ -33,7 +34,6 @@ class SocialLoginSerializer(serializers.Serializer):
|
||||||
return social_login
|
return social_login
|
||||||
|
|
||||||
def validate(self, attrs):
|
def validate(self, attrs):
|
||||||
access_token = attrs.get('access_token')
|
|
||||||
view = self.context.get('view')
|
view = self.context.get('view')
|
||||||
request = self._get_request()
|
request = self._get_request()
|
||||||
|
|
||||||
|
@ -48,6 +48,44 @@ class SocialLoginSerializer(serializers.Serializer):
|
||||||
|
|
||||||
adapter = adapter_class()
|
adapter = adapter_class()
|
||||||
app = adapter.get_provider().get_app(request)
|
app = adapter.get_provider().get_app(request)
|
||||||
|
|
||||||
|
# More info on code vs access_token
|
||||||
|
# http://stackoverflow.com/questions/8666316/facebook-oauth-2-0-code-and-token
|
||||||
|
|
||||||
|
# Case 1: We received the access_token
|
||||||
|
if('access_token' in attrs):
|
||||||
|
access_token = attrs.get('access_token')
|
||||||
|
|
||||||
|
# Case 2: We received the authorization code
|
||||||
|
elif('code' in attrs):
|
||||||
|
self.callback_url = getattr(view, 'callback_url', None)
|
||||||
|
self.client_class = getattr(view, 'client_class', None)
|
||||||
|
|
||||||
|
if not self.callback_url:
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
'Define callback_url in view'
|
||||||
|
)
|
||||||
|
if not self.client_class:
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
'Define client_class in view'
|
||||||
|
)
|
||||||
|
|
||||||
|
code = attrs.get('code')
|
||||||
|
|
||||||
|
provider = self.adapter.get_provider()
|
||||||
|
scope = provider.get_scope(request)
|
||||||
|
client = self.client_class(
|
||||||
|
request,
|
||||||
|
app.client_id,
|
||||||
|
app.secret,
|
||||||
|
self.adapter.access_token_method,
|
||||||
|
self.adapter.access_token_url,
|
||||||
|
self.callback_url,
|
||||||
|
scope
|
||||||
|
)
|
||||||
|
token = client.get_access_token(code)
|
||||||
|
access_token = token['access_token']
|
||||||
|
|
||||||
token = adapter.parse_token({'access_token': access_token})
|
token = adapter.parse_token({'access_token': access_token})
|
||||||
token.app = app
|
token.app = app
|
||||||
|
|
||||||
|
|
|
@ -3,21 +3,33 @@ from rest_framework.views import APIView
|
||||||
from rest_framework.response import Response
|
from rest_framework.response import Response
|
||||||
from rest_framework.permissions import AllowAny
|
from rest_framework.permissions import AllowAny
|
||||||
from rest_framework import status
|
from rest_framework import status
|
||||||
|
from rest_framework.authtoken.models import Token
|
||||||
|
|
||||||
from allauth.account.views import SignupView, ConfirmEmailView
|
from allauth.account.views import SignupView, ConfirmEmailView
|
||||||
from allauth.account.utils import complete_signup
|
from allauth.account.utils import complete_signup
|
||||||
from allauth.account import app_settings
|
from allauth.account import app_settings
|
||||||
|
|
||||||
from rest_auth.app_settings import UserDetailsSerializer
|
from rest_auth.app_settings import UserDetailsSerializer, TokenSerializer
|
||||||
from rest_auth.registration.serializers import SocialLoginSerializer
|
from rest_auth.registration.serializers import SocialLoginSerializer
|
||||||
from rest_auth.views import Login
|
from rest_auth.views import Login
|
||||||
|
|
||||||
|
|
||||||
class Register(APIView, SignupView):
|
class Register(APIView, SignupView):
|
||||||
|
"""
|
||||||
|
Accepts the credentials and creates a new user
|
||||||
|
if user does not exist already
|
||||||
|
Return the REST Token if the credentials are valid and authenticated.
|
||||||
|
Calls allauth complete_signup method
|
||||||
|
|
||||||
|
Accept the following POST parameters: username, email, password
|
||||||
|
Return the REST Framework Token Object's key.
|
||||||
|
"""
|
||||||
|
|
||||||
permission_classes = (AllowAny,)
|
permission_classes = (AllowAny,)
|
||||||
user_serializer_class = UserDetailsSerializer
|
# user_serializer_class = UserDetailsSerializer
|
||||||
allowed_methods = ('POST', 'OPTIONS', 'HEAD')
|
allowed_methods = ('POST', 'OPTIONS', 'HEAD')
|
||||||
|
token_model = Token
|
||||||
|
serializer_class = TokenSerializer
|
||||||
|
|
||||||
def get(self, *args, **kwargs):
|
def get(self, *args, **kwargs):
|
||||||
return Response({}, status=status.HTTP_405_METHOD_NOT_ALLOWED)
|
return Response({}, status=status.HTTP_405_METHOD_NOT_ALLOWED)
|
||||||
|
@ -27,6 +39,9 @@ class Register(APIView, SignupView):
|
||||||
|
|
||||||
def form_valid(self, form):
|
def form_valid(self, form):
|
||||||
self.user = form.save(self.request)
|
self.user = form.save(self.request)
|
||||||
|
self.token, created = self.token_model.objects.get_or_create(
|
||||||
|
user=self.user
|
||||||
|
)
|
||||||
if isinstance(self.request, HttpRequest):
|
if isinstance(self.request, HttpRequest):
|
||||||
request = self.request
|
request = self.request
|
||||||
else:
|
else:
|
||||||
|
@ -47,7 +62,8 @@ class Register(APIView, SignupView):
|
||||||
return self.get_response_with_errors()
|
return self.get_response_with_errors()
|
||||||
|
|
||||||
def get_response(self):
|
def get_response(self):
|
||||||
serializer = self.user_serializer_class(instance=self.user)
|
# serializer = self.user_serializer_class(instance=self.user)
|
||||||
|
serializer = self.serializer_class(instance=self.token)
|
||||||
return Response(serializer.data, status=status.HTTP_201_CREATED)
|
return Response(serializer.data, status=status.HTTP_201_CREATED)
|
||||||
|
|
||||||
def get_response_with_errors(self):
|
def get_response_with_errors(self):
|
||||||
|
@ -72,11 +88,25 @@ class VerifyEmail(APIView, ConfirmEmailView):
|
||||||
class SocialLogin(Login):
|
class SocialLogin(Login):
|
||||||
"""
|
"""
|
||||||
class used for social authentications
|
class used for social authentications
|
||||||
example usage for facebook
|
example usage for facebook with access_token
|
||||||
|
-------------
|
||||||
from allauth.socialaccount.providers.facebook.views import FacebookOAuth2Adapter
|
from allauth.socialaccount.providers.facebook.views import FacebookOAuth2Adapter
|
||||||
|
|
||||||
class FacebookLogin(SocialLogin):
|
class FacebookLogin(SocialLogin):
|
||||||
adapter_class = FacebookOAuth2Adapter
|
adapter_class = FacebookOAuth2Adapter
|
||||||
|
-------------
|
||||||
|
|
||||||
|
example usage for facebook with code
|
||||||
|
|
||||||
|
-------------
|
||||||
|
from allauth.socialaccount.providers.facebook.views import FacebookOAuth2Adapter
|
||||||
|
from allauth.socialaccount.providers.oauth2.client import OAuth2Client
|
||||||
|
|
||||||
|
class FacebookLogin(SocialLogin):
|
||||||
|
adapter_class = FacebookOAuth2Adapter
|
||||||
|
client_class = OAuth2Client
|
||||||
|
callback_url = 'localhost:8000'
|
||||||
|
-------------
|
||||||
"""
|
"""
|
||||||
|
|
||||||
serializer_class = SocialLoginSerializer
|
serializer_class = SocialLoginSerializer
|
||||||
|
|
Loading…
Reference in New Issue
Block a user