如何判断用户的电子邮件地址是否已使用Django,allauth,rest-auth和自定义用户进行了验证

时间:2019-01-31 18:47:51

标签: django django-allauth email-confirmation

我正在将Django 2.0.10与rest-framework,rest-auth和allauth一起使用。我有一个自定义的用户模型。

我已经使用allauth视图进行了电子邮件验证。用户注册时将发送验证电子邮件。如果我单击电子邮件中的链接,则会进入带有按钮的页面,单击该按钮以单击以验证电子邮件。所有这一切都没有错误。但是我不知道这实际上是什么。用户数据中的任何字段似乎都没有变化。

我想要的行为是使用户能够注册和登录,但只能在验证电子邮件后才能向网站添加内容。

编辑:this post给出了部分答案,但没有说明如何将验证状态另存为用户的属性,以便您在加载用户数据时可以在前端对其进行检查。 / p>

settings.py

# django rest auth
ACCOUNT_AUTHENTICATION_METHOD = 'email'
ACCOUNT_EMAIL_REQUIRED = True
ACCOUNT_USERNAME_REQUIRED = False
OLD_PASSWORD_FIELD_ENABLED = True
LOGOUT_ON_PASSWORD_CHANGE = False
ACCOUNT_EMAIL_VERIFICATION = 'optional'

api / urls.py

from allauth.account.views import confirm_email

urlpatterns = [
    re_path(r'^rest-auth/registration/account-confirm-email/(?P<key>[-:\w]+)/$', confirm_email,
     name='account_confirm_email'),
...
]

users / models.py

import uuid 

from django.contrib.auth.models import AbstractUser, UserManager
from django.db import models
from django.utils.http import int_to_base36

class CustomUserManager(UserManager):
    def get_by_natural_key(self, username):
        case_insensitive_username_field = '{}__iexact'.format(self.model.USERNAME_FIELD)
        return self.get(**{case_insensitive_username_field: username})

ID_LENGTH = 12

def pkgen():
    from base64 import b32encode
    from hashlib import sha1
    from random import random

    pk = int_to_base36(uuid.uuid4().int)[:ID_LENGTH]
    return pk

class CustomUser(AbstractUser):
    objects = CustomUserManager()
    slug = models.CharField(max_length=ID_LENGTH, default=pkgen, editable=False)
    id = models.UUIDField(primary_key=True, default=uuid.uuid4, editable=False)

    def __str__(self):
        return self.email

用户登录后,如何确定他们是否已验证电子邮件地址?感谢您的帮助!

2 个答案:

答案 0 :(得分:0)

啊哈!感谢this postthis post,我想我有了答案。

电子邮件地址的状态保存在单独的表EmailAdress中,而不作为用户模型的一部分。可以在modelviewset中访问它,如下所示:

api.py

from allauth.account.admin import EmailAddress

class ListViewSet(viewsets.ModelViewSet):
    ...

    def get_queryset(self):
        # can view public lists and lists the user created
        if self.request.user.is_authenticated:
            print('is there a verified email address?')
            print(EmailAddress.objects.filter(user=self.request.user, verified=True).exists())

            ...

如果用户具有任何已验证的电子邮件地址,它将返回True。

但是,向用户添加验证状态更为有用。可以使用here中所述的信号来完成。

views.py

from allauth.account.signals import email_confirmed
from django.dispatch import receiver

@receiver(email_confirmed)
def email_confirmed_(request, email_address, **kwargs):
    user = email_address.user
    user.email_verified = True

    user.save()

现在在api.py中,您可以像这样检查:

print(self.request.user.email_verified)

如果您只有一个无法更改或删除的电子邮件地址,则此方法有效。如果您允许多个电子邮件地址,我想您需要进行更多检查并相应更新用户状态。但是我只有一个用于登录的电子邮件地址,所以我认为可以。

我认为最好将“ email_verified”作为用户个人资料的一部分,但这是一个有效的演示。

答案 1 :(得分:0)

我遇到了同样的问题,我可以使用下面的代码解决这个问题:

#Project-level folder urls.py
from django.contrib import admin
from django.urls import path, include
from allauth.account.views import ConfirmEmailView, EmailVerificationSentView 
#These ConfirmEmailView, EmailVerificationSentView are very important
#I used other allauth/ dj-rest-auth views and they didn't automatically verify the email.

urlpatterns = [
path('admin/', admin.site.urls),
path('dj-rest-auth/registration/account-confirm-email/<str:key>/',
    ConfirmEmailView.as_view()), #This is at the top because apparently won't work if below. #Integral to problem solution

path('dj-rest-auth/', include('dj_rest_auth.urls')),
path('dj-rest-auth/registration/', include('dj_rest_auth.registration.urls')),
path('api-auth', include('rest_framework.urls')),

 path('dj-rest-auth/registration/account-confirm-email/', 
  EmailVerificationSentView.as_view(),
        name='account_email_verification_sent'),#Integral to problem solution

]

上面的代码允许我使用注册网址创建新用户。之后会发送一封带有链接的电子邮件。当用户点击链接时,他们将被重定向到登录页面,他们的电子邮件现在已在数据库中得到验证。