Skip to content

Commit f9ef1ff

Browse files
author
Mateusz Sikora
committed
Merge branch 'philippeluickx-development'
2 parents 7f86c4c + 388314f commit f9ef1ff

File tree

11 files changed

+199
-60
lines changed

11 files changed

+199
-60
lines changed

docs/api_endpoints.rst

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -73,3 +73,4 @@ Basing on example from installation section :doc:`Installation </installation>`
7373
- /rest-auth/facebook/ (POST)
7474

7575
- access_token
76+
- code

docs/configuration.rst

Lines changed: 6 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -6,17 +6,17 @@ Configuration
66
You can define your custom serializers for each endpoint without overriding urls and views by adding ``REST_AUTH_SERIALIZERS`` dictionary in your django settings.
77
Possible key values:
88

9-
- LOGIN_SERIALIZER - serializer class in ``rest_auth.views.Login``, default value ``rest_auth.serializers.LoginSerializer``
9+
- LOGIN_SERIALIZER - serializer class in ``rest_auth.views.LoginView``, default value ``rest_auth.serializers.LoginSerializer``
1010

11-
- TOKEN_SERIALIZER - response for successful authentication in ``rest_auth.views.Login``, default value ``rest_auth.serializers.TokenSerializer``
11+
- TOKEN_SERIALIZER - response for successful authentication in ``rest_auth.views.LoginView``, default value ``rest_auth.serializers.TokenSerializer``
1212

13-
- USER_DETAILS_SERIALIZER - serializer class in ``rest_auth.views.UserDetails``, default value ``rest_auth.serializers.UserDetailsSerializer``
13+
- USER_DETAILS_SERIALIZER - serializer class in ``rest_auth.views.UserDetailsView``, default value ``rest_auth.serializers.UserDetailsSerializer``
1414

15-
- PASSWORD_RESET_SERIALIZER - serializer class in ``rest_auth.views.PasswordReset``, default value ``rest_auth.serializers.PasswordResetSerializer``
15+
- PASSWORD_RESET_SERIALIZER - serializer class in ``rest_auth.views.PasswordResetView``, default value ``rest_auth.serializers.PasswordResetSerializer``
1616

17-
- PASSWORD_RESET_CONFIRM_SERIALIZER - serializer class in ``rest_auth.views.PasswordResetConfirm``, default value ``rest_auth.serializers.PasswordResetConfirmSerializer``
17+
- PASSWORD_RESET_CONFIRM_SERIALIZER - serializer class in ``rest_auth.views.PasswordResetConfirmView``, default value ``rest_auth.serializers.PasswordResetConfirmSerializer``
1818

19-
- PASSWORD_CHANGE_SERIALIZER - serializer class in ``rest_auth.views.PasswordChange``, default value ``rest_auth.serializers.PasswordChangeSerializer``
19+
- PASSWORD_CHANGE_SERIALIZER - serializer class in ``rest_auth.views.PasswordChangeView``, default value ``rest_auth.serializers.PasswordChangeSerializer``
2020

2121

2222
Example configuration:

docs/faq.rst

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,7 @@ FAQ
3131
# custom fields for user
3232
company_name = models.CharField(max_length=100)
3333
34-
To allow update user details within one request send to rest_auth.views.UserDetails view, create serializer like this:
34+
To allow update user details within one request send to rest_auth.views.UserDetailsView view, create serializer like this:
3535

3636
.. code-block:: python
3737

docs/installation.rst

Lines changed: 3 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -91,14 +91,14 @@ Using ``django-allauth``, ``django-rest-auth`` provides helpful class for creati
9191

9292
3. Add Social Application in django admin panel
9393

94-
4. Create new view as a subclass of ``rest_auth.registration.views.SocialLogin`` with ``FacebookOAuth2Adapter`` adapter as an attribute:
94+
4. Create new view as a subclass of ``rest_auth.registration.views.SocialLoginView`` with ``FacebookOAuth2Adapter`` adapter as an attribute:
9595

9696
.. code-block:: python
9797
9898
from allauth.socialaccount.providers.facebook.views import FacebookOAuth2Adapter
99-
from rest_auth.registration.views import SocialLogin
99+
from rest_auth.registration.views import SocialLoginView
100100
101-
class FacebookLogin(SocialLogin):
101+
class FacebookLogin(SocialLoginView):
102102
adapter_class = FacebookOAuth2Adapter
103103
104104
5. Create url for FacebookLogin view:

rest_auth/registration/serializers.py

Lines changed: 75 additions & 16 deletions
Original file line numberDiff line numberDiff line change
@@ -1,40 +1,99 @@
11
from django.http import HttpRequest
22
from rest_framework import serializers
33
from requests.exceptions import HTTPError
4-
from allauth.socialaccount.helpers import complete_social_login
4+
# Import is needed only if we are using social login, in which
5+
# case the allauth.socialaccount will be declared
6+
try:
7+
from allauth.socialaccount.helpers import complete_social_login
8+
except ImportError:
9+
pass
510

611

712
class SocialLoginSerializer(serializers.Serializer):
13+
access_token = serializers.CharField(required=False)
14+
code = serializers.CharField(required=False)
815

9-
access_token = serializers.CharField(required=True)
10-
11-
def validate(self, attrs):
12-
access_token = attrs.get('access_token')
13-
view = self.context.get('view')
16+
def _get_request(self):
1417
request = self.context.get('request')
1518
if not isinstance(request, HttpRequest):
1619
request = request._request
20+
return request
21+
22+
def get_social_login(self, adapter, app, token, response):
23+
"""
24+
25+
:param adapter: allauth.socialaccount Adapter subclass. Usually OAuthAdapter or Auth2Adapter
26+
:param app: `allauth.socialaccount.SocialApp` instance
27+
:param token: `allauth.socialaccount.SocialToken` instance
28+
:param response: Provider's response for OAuth1. Not used in the
29+
:return: :return: A populated instance of the `allauth.socialaccount.SocialLoginView` instance
30+
"""
31+
request = self._get_request()
32+
social_login = adapter.complete_login(request, app, token, response=response)
33+
social_login.token = token
34+
return social_login
35+
36+
def validate(self, attrs):
37+
view = self.context.get('view')
38+
request = self._get_request()
1739

1840
if not view:
1941
raise serializers.ValidationError(
2042
'View is not defined, pass it as a context variable'
2143
)
2244

23-
self.adapter_class = getattr(view, 'adapter_class', None)
24-
25-
if not self.adapter_class:
45+
adapter_class = getattr(view, 'adapter_class', None)
46+
if not adapter_class:
2647
raise serializers.ValidationError('Define adapter_class in view')
2748

28-
self.adapter = self.adapter_class()
29-
app = self.adapter.get_provider().get_app(request)
30-
token = self.adapter.parse_token({'access_token': access_token})
49+
adapter = adapter_class()
50+
app = adapter.get_provider().get_app(request)
51+
52+
# More info on code vs access_token
53+
# http://stackoverflow.com/questions/8666316/facebook-oauth-2-0-code-and-token
54+
55+
# Case 1: We received the access_token
56+
if('access_token' in attrs):
57+
access_token = attrs.get('access_token')
58+
59+
# Case 2: We received the authorization code
60+
elif('code' in attrs):
61+
self.callback_url = getattr(view, 'callback_url', None)
62+
self.client_class = getattr(view, 'client_class', None)
63+
64+
if not self.callback_url:
65+
raise serializers.ValidationError(
66+
'Define callback_url in view'
67+
)
68+
if not self.client_class:
69+
raise serializers.ValidationError(
70+
'Define client_class in view'
71+
)
72+
73+
code = attrs.get('code')
74+
75+
provider = adapter.get_provider()
76+
scope = provider.get_scope(request)
77+
client = self.client_class(
78+
request,
79+
app.client_id,
80+
app.secret,
81+
adapter.access_token_method,
82+
adapter.access_token_url,
83+
self.callback_url,
84+
scope
85+
)
86+
token = client.get_access_token(code)
87+
access_token = token['access_token']
88+
89+
else:
90+
raise serializers.ValidationError('Incorrect input. access_token or code is required.')
91+
92+
token = adapter.parse_token({'access_token': access_token})
3193
token.app = app
3294

3395
try:
34-
login = self.adapter.complete_login(request, app, token,
35-
response=access_token)
36-
37-
login.token = token
96+
login = self.get_social_login(adapter, app, token, access_token)
3897
complete_social_login(request, login)
3998
except HTTPError:
4099
raise serializers.ValidationError('Incorrect value')

rest_auth/registration/urls.py

Lines changed: 3 additions & 3 deletions
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,12 @@
11
from django.views.generic import TemplateView
22
from django.conf.urls import patterns, url
33

4-
from .views import Register, VerifyEmail
4+
from .views import RegisterView, VerifyEmailView
55

66
urlpatterns = patterns(
77
'',
8-
url(r'^$', Register.as_view(), name='rest_register'),
9-
url(r'^verify-email/$', VerifyEmail.as_view(), name='rest_verify_email'),
8+
url(r'^$', RegisterView.as_view(), name='rest_register'),
9+
url(r'^verify-email/$', VerifyEmailView.as_view(), name='rest_verify_email'),
1010

1111
# This url is used by django-allauth and empty TemplateView is
1212
# defined just to allow reverse() call inside app, for example when email

rest_auth/registration/views.py

Lines changed: 38 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -3,21 +3,32 @@
33
from rest_framework.response import Response
44
from rest_framework.permissions import AllowAny
55
from rest_framework import status
6+
from rest_framework.authtoken.models import Token
67

78
from allauth.account.views import SignupView, ConfirmEmailView
89
from allauth.account.utils import complete_signup
910
from allauth.account import app_settings
1011

11-
from rest_auth.app_settings import UserDetailsSerializer
12+
from rest_auth.app_settings import TokenSerializer
1213
from rest_auth.registration.serializers import SocialLoginSerializer
13-
from rest_auth.views import Login
14+
from rest_auth.views import LoginView
1415

1516

16-
class Register(APIView, SignupView):
17+
class RegisterView(APIView, SignupView):
18+
"""
19+
Accepts the credentials and creates a new user
20+
if user does not exist already
21+
Return the REST Token if the credentials are valid and authenticated.
22+
Calls allauth complete_signup method
23+
24+
Accept the following POST parameters: username, email, password
25+
Return the REST Framework Token Object's key.
26+
"""
1727

1828
permission_classes = (AllowAny,)
19-
user_serializer_class = UserDetailsSerializer
2029
allowed_methods = ('POST', 'OPTIONS', 'HEAD')
30+
token_model = Token
31+
serializer_class = TokenSerializer
2132

2233
def get(self, *args, **kwargs):
2334
return Response({}, status=status.HTTP_405_METHOD_NOT_ALLOWED)
@@ -27,6 +38,9 @@ def put(self, *args, **kwargs):
2738

2839
def form_valid(self, form):
2940
self.user = form.save(self.request)
41+
self.token, created = self.token_model.objects.get_or_create(
42+
user=self.user
43+
)
3044
if isinstance(self.request, HttpRequest):
3145
request = self.request
3246
else:
@@ -47,14 +61,15 @@ def post(self, request, *args, **kwargs):
4761
return self.get_response_with_errors()
4862

4963
def get_response(self):
50-
serializer = self.user_serializer_class(instance=self.user)
64+
# serializer = self.user_serializer_class(instance=self.user)
65+
serializer = self.serializer_class(instance=self.token)
5166
return Response(serializer.data, status=status.HTTP_201_CREATED)
5267

5368
def get_response_with_errors(self):
5469
return Response(self.form.errors, status=status.HTTP_400_BAD_REQUEST)
5570

5671

57-
class VerifyEmail(APIView, ConfirmEmailView):
72+
class VerifyEmailView(APIView, ConfirmEmailView):
5873

5974
permission_classes = (AllowAny,)
6075
allowed_methods = ('POST', 'OPTIONS', 'HEAD')
@@ -69,14 +84,28 @@ def post(self, request, *args, **kwargs):
6984
return Response({'message': 'ok'}, status=status.HTTP_200_OK)
7085

7186

72-
class SocialLogin(Login):
87+
class SocialLoginView(LoginView):
7388
"""
7489
class used for social authentications
75-
example usage for facebook
90+
example usage for facebook with access_token
91+
-------------
92+
from allauth.socialaccount.providers.facebook.views import FacebookOAuth2Adapter
7693
94+
class FacebookLogin(SocialLoginView):
95+
adapter_class = FacebookOAuth2Adapter
96+
-------------
97+
98+
example usage for facebook with code
99+
100+
-------------
77101
from allauth.socialaccount.providers.facebook.views import FacebookOAuth2Adapter
78-
class FacebookLogin(SocialLogin):
102+
from allauth.socialaccount.providers.oauth2.client import OAuth2Client
103+
104+
class FacebookLogin(SocialLoginView):
79105
adapter_class = FacebookOAuth2Adapter
106+
client_class = OAuth2Client
107+
callback_url = 'localhost:8000'
108+
-------------
80109
"""
81110

82111
serializer_class = SocialLoginSerializer

rest_auth/serializers.py

Lines changed: 56 additions & 6 deletions
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
from django.contrib.auth import get_user_model
1+
from django.contrib.auth import get_user_model, authenticate
22
from django.conf import settings
33
from django.contrib.auth.forms import PasswordResetForm, SetPasswordForm
44
try:
@@ -7,24 +7,74 @@
77
# make compatible with django 1.5
88
from django.utils.http import base36_to_int as uid_decoder
99
from django.contrib.auth.tokens import default_token_generator
10+
from django.utils.translation import ugettext_lazy as _
1011

11-
from rest_framework import serializers
12+
from rest_framework import serializers, exceptions
1213
from rest_framework.authtoken.models import Token
13-
from rest_framework.authtoken.serializers import AuthTokenSerializer
1414
from rest_framework.exceptions import ValidationError
1515

1616

17-
class LoginSerializer(AuthTokenSerializer):
17+
class LoginSerializer(serializers.Serializer):
18+
username = serializers.CharField(required=False)
19+
email = serializers.EmailField(required=False)
20+
password = serializers.CharField(style={'input_type': 'password'})
1821

1922
def validate(self, attrs):
20-
attrs = super(LoginSerializer, self).validate(attrs)
23+
username = attrs.get('username')
24+
email = attrs.get('email')
25+
password = attrs.get('password')
26+
27+
if 'allauth' in settings.INSTALLED_APPS:
28+
from allauth.account import app_settings
29+
# Authentication through email
30+
if app_settings.AUTHENTICATION_METHOD == app_settings.AuthenticationMethod.EMAIL:
31+
if email and password:
32+
user = authenticate(email=email, password=password)
33+
else:
34+
msg = _('Must include "email" and "password".')
35+
raise exceptions.ValidationError(msg)
36+
# Authentication through username
37+
elif app_settings.AUTHENTICATION_METHOD == app_settings.AuthenticationMethod.USERNAME:
38+
if username and password:
39+
user = authenticate(username=username, password=password)
40+
else:
41+
msg = _('Must include "username" and "password".')
42+
raise exceptions.ValidationError(msg)
43+
# Authentication through either username or email
44+
else:
45+
if email and password:
46+
user = authenticate(email=email, password=password)
47+
elif username and password:
48+
user = authenticate(username=username, password=password)
49+
else:
50+
msg = _('Must include either "username" or "email" and "password".')
51+
raise exceptions.ValidationError(msg)
52+
53+
elif username and password:
54+
user = authenticate(username=username, password=password)
55+
56+
else:
57+
msg = _('Must include "username" and "password".')
58+
raise exceptions.ValidationError(msg)
59+
60+
# Did we get back an active user?
61+
if user:
62+
if not user.is_active:
63+
msg = _('User account is disabled.')
64+
raise exceptions.ValidationError(msg)
65+
else:
66+
msg = _('Unable to log in with provided credentials.')
67+
raise exceptions.ValidationError(msg)
68+
69+
# If required, is the email verified?
2170
if 'rest_auth.registration' in settings.INSTALLED_APPS:
2271
from allauth.account import app_settings
2372
if app_settings.EMAIL_VERIFICATION == app_settings.EmailVerificationMethod.MANDATORY:
24-
user = attrs['user']
2573
email_address = user.emailaddress_set.get(email=user.email)
2674
if not email_address.verified:
2775
raise serializers.ValidationError('E-mail is not verified.')
76+
77+
attrs['user'] = user
2878
return attrs
2979

3080

rest_auth/test_urls.py

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -5,10 +5,10 @@
55
from allauth.socialaccount.providers.facebook.views import FacebookOAuth2Adapter
66

77
from rest_auth.urls import urlpatterns
8-
from rest_auth.registration.views import SocialLogin
8+
from rest_auth.registration.views import SocialLoginView
99

1010

11-
class FacebookLogin(SocialLogin):
11+
class FacebookLogin(SocialLoginView):
1212
adapter_class = FacebookOAuth2Adapter
1313

1414
urlpatterns += patterns(

0 commit comments

Comments
 (0)