如何在React前端中使用allauth和rest-auth重新发送Django确认电子邮件

10

我正在使用Django 2.0.10,结合rest framework、rest-auth和allauth,以及React前端。Rest-auth提供了使用JWT令牌认证的登录和注销功能,但我不知道如何允许用户请求重新发送验证电子邮件。

我希望用户能够登录并点击一个按钮,显示“重新发送确认电子邮件”。例如,如果他们意外删除了电子邮件,则需要能够请求另一个电子邮件。

我看到一些帖子建议可以使用allauth中的send_email_confirmation,但这需要一个CSRF令牌,而该令牌将由模板生成。我尝试按照文档中免除csrf的方法,但没有任何区别。我还尝试设置authentication_classes = () 如此建议的。以下是我的代码:

settings:

REST_FRAMEWORK = {
    'DEFAULT_AUTHENTICATION_CLASSES': [
        'rest_framework.authentication.TokenAuthentication',
    ],
}

urls.py

from users.views import EmailConfirmation

urlpatterns = [
...
url(r'^/sendconfirmationemail/', EmailConfirmation.as_view(), name='send-email-confirmation')
]

views.py

from rest_framework.views import APIView
from django.views.decorators.csrf import csrf_exempt
from django.utils.decorators import method_decorator

class EmailConfirmation(APIView):
    @method_decorator(csrf_exempt)
    authentication_classes = ()

    def post(self):
        send_email_confirmation(user=self.request.user)

当我向端点'/api/v1/rest-auth/sendconfirmationemail/'发布内容时,出现错误Forbidden。
<p>You are seeing this message because this site requires a CSRF cookie when submitting forms. This cookie is required for security reasons, to ensure that your browser is not being hijacked by third parties.</p>
<p>If you have configured your browser to disable cookies, please re-enable them, at least for this site, or for &#39;same-origin&#39; requests.</p>

编辑:我也尝试过像这样的教程中添加CSRF令牌到我的请求。但是我遇到了同样的问题。我尝试过以下方法:

function getCookie(name) {
    var cookieValue = null;
    if (document.cookie && document.cookie !== '') {
        var cookies = document.cookie.split(';');
        for (var i = 0; i < cookies.length; i++) {
            var cookie = cookies[i].trim();
            //var cookie = jQuery.trim(cookies[i]);
            // Does this cookie string begin with the name we want?
            if (cookie.substring(0, name.length + 1) === (name + '=')) {
                cookieValue = decodeURIComponent(cookie.substring(name.length + 1));
                break;
            }
        }
    }
    return cookieValue;
}

然后我使用 Fetch API 构建请求,代码如下:

curl 'http://localhost:3000/api/v1/rest-auth/sendconfirmationemail/' -H 'Authorization: Token 55c8da5de68b657cf9dafd820a7f02f997fa3d64' -H 'Origin: http://localhost:3000' -H 'Accept-Encoding: gzip, deflate, br' -H 'Accept-Language: en-GB,en-US;q=0.9,en;q=0.8' -H 'User-Agent: Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/63.0.3239.132 Safari/537.36' -H 'Content-Type: text/plain;charset=UTF-8' -H 'Accept: */*' -H 'Referer: http://localhost:3000/account' -H 'Connection: keep-alive' -H 'X-CSRFToken: YOHB6RXqpZMFIXKT31T9tAjaUsH0w3eIjaaCvbgAqmP64SWeVNd3Sz3g8nZUEmVS' --data-binary 'csrfmiddlewaretoken=YOHB6RXqpZMFIXKT31T9tAjaUsH0w3eIjaaCvbgAqmP64SWeVNd3Sz3g8nZUEmVS' --compressed

当我查看Django模板的工作示例时,我发现与表单数据一起发送的csrfmiddlewaretoken值与标头中发送的X-CSRFToken不同,我认为Django模板提供的值是盐值,并且这可能会产生差异。但是我找不到任何指示告诉我如何获取正确的值?或者我的fetch请求形式有误吗?

如果我在我的React页面中使用此表单:

<form action="api/v1/sendconfirmationemail" method="post">
<Input type="hidden" name="csrfmiddlewaretoken"  value={this.getCookie('csrftoken')} />
<button type="submit">Send</button>
</form>

当我提交时,出现错误“方法不允许(POST):/ api / v1 / sendconfirmationemail”。来自此请求的cURL是:
curl 'http://localhost:3000/api/v1/sendconfirmationemail' -H 'Cookie: csrftoken=YOHB6RXqpZMFIXKT31T9tAjaUsH0w3eIjaaCvbgAqmP64SWeVNd3Sz3g8nZUEmVS; sessionid=uslpdgd5npa6wyk2oqpwkhj79xaen7nw' -H 'Origin: http://localhost:3000' -H 'Accept-Encoding: gzip, deflate, br' -H 'Accept-Language: en-GB,en-US;q=0.9,en;q=0.8' -H 'Upgrade-Insecure-Requests: 1' -H 'User-Agent: Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/63.0.3239.132 Safari/537.36' -H 'Content-Type: application/x-www-form-urlencoded' -H 'Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8' -H 'Cache-Control: max-age=0' -H 'Referer: http://localhost:3000/account' -H 'Connection: keep-alive' --data 'csrfmiddlewaretoken=YOHB6RXqpZMFIXKT31T9tAjaUsH0w3eIjaaCvbgAqmP64SWeVNd3Sz3g8nZUEmVS' --compressed

我该如何在React前端请求重新发送电子邮件?


可能是重复问题 https://dev59.com/IVgR5IYBdhLWcg3wHqXB - Thomas Myers
我认为这不是同样的问题,因为我正在使用令牌身份验证。那个问题使用oauth2_provider.ext.rest_framework.OAuth2Authentication。 - Little Brain
我尝试使用@method_decorator(csrf_exempt)来修饰def post(self, request)函数,但是这个方法也没有起作用。 - Little Brain
登录基于Django的登录,而不是rest api。https://www.django-rest-framework.org/api-guide/authentication/#sessionauthentication - Thomas Myers
我想我终于弄清楚了 - CSRF错误会在URL错误时显示,因此您的API调用没有指向有效的Rest Framework URL。 - Little Brain
显示剩余5条评论
1个回答

11

我认为我的问题部分原因在于,如果 fetch 请求具有不正确的 URL 并且因此没有指向有效的 rest-framework URL,则服务器似乎会显示“Forbidden (CSRF cookie not set.)”错误。这让我陷入了追踪 CSRF 问题而不是仔细检查我的 URL 的困境中。

我希望这能帮助其他人。以下是我的可工作代码:

users/views.py

from allauth.account.signals import email_confirmed
from django.dispatch import receiver
from rest_framework import generics
from rest_framework.permissions import IsAuthenticated

from allauth.account.utils import send_email_confirmation
from rest_framework.views import APIView

from . import models
from . import serializers

from rest_framework.authentication import TokenAuthentication
from rest_framework import status
from rest_framework.response import Response

class UserListView(generics.ListCreateAPIView):
    queryset = models.CustomUser.objects.all()
    serializer_class = serializers.UserSerializer
    authentication_classes = (TokenAuthentication,)

# when the email is confirmed, set a field on the user
# so the UI can check whether to show the "Resend confirmation email" button
@receiver(email_confirmed)
def email_confirmed_(request, email_address, **kwargs):
    user = email_address.user
    user.email_verified = True

    user.save()

# request a new confirmation email
class EmailConfirmation(APIView):
    permission_classes = [IsAuthenticated] 

    def post(self, request):
        if request.user.email_verified:
            return Response({'message': 'Email already verified'}, status=status.HTTP_201_CREATED)

        send_email_confirmation(request, request.user)
        return Response({'message': 'Email confirmation sent'}, status=status.HTTP_201_CREATED)

api/urls.py

from users.views import EmailConfirmation

urlpatterns = [
...
    path('sendconfirmationemail/', EmailConfirmation.as_view(), name='send-email-confirmation')
]

我正在使用JavaScript fetch()发送一个POST请求,并在头部添加身份验证令牌,代码如下:

-H 'Authorization: Token 98254e6004e4a28b9d8cf61e7a7a9ee2fc61009a' 

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接