Подтвердить что ты не робот

Регистрация Django rest-auth allauth с электронной почтой, именем и фамилией, без имени пользователя

Я использую django-rest-auth и allauth для входа и регистрации в моем приложении django. Я не написал ни одной отдельной отдельной строчки кода для входа или регистрации. Регистрация прошла успешно с помощью электронной почты и предоставленного пароля.

Я не использую имя пользователя для аутентификации, а не электронную почту.

В моем просматриваемом api для регистрации я получаю следующее:

введите описание изображения здесь

Наряду с этими полями я хочу иметь first_name и last_name (в таблице по умолчанию для auth_user были эти столбцы), так что у моего вновь созданного auth_user также есть эти поля, заданные вместе с адресом электронной почты и хешированным паролем.

Как я могу это достичь? Эта просматриваемая форма не так важна, но возможность сохранения first_name и last_name - это то, что мне нужно в первую очередь.

4b9b3361

Ответ 1

  1. Убедитесь, что в вашем файле settings.py есть ACCOUNT_USERNAME_REQUIRED = False.

  2. Для first_name и last_name вам нужно написать пользовательский RegisterSerializer (https://github.com/Tivix/django-rest-auth/blob/master/rest_auth/registration/serializers.py#L166)

вот пример кода для serializers.py

from allauth.account import app_settings as allauth_settings
from allauth.utils import email_address_exists
from allauth.account.adapter import get_adapter
from allauth.account.utils import setup_user_email

class RegisterSerializer(serializers.Serializer):
    email = serializers.EmailField(required=allauth_settings.EMAIL_REQUIRED)
    first_name = serializers.CharField(required=True, write_only=True)
    last_name = serializers.CharField(required=True, write_only=True)
    password1 = serializers.CharField(required=True, write_only=True)
    password2 = serializers.CharField(required=True, write_only=True)

    def validate_email(self, email):
        email = get_adapter().clean_email(email)
        if allauth_settings.UNIQUE_EMAIL:
            if email and email_address_exists(email):
                raise serializers.ValidationError(
                    _("A user is already registered with this e-mail address."))
        return email

    def validate_password1(self, password):
        return get_adapter().clean_password(password)

    def validate(self, data):
        if data['password1'] != data['password2']:
            raise serializers.ValidationError(
                _("The two password fields didn't match."))
        return data

    def get_cleaned_data(self):
        return {
            'first_name': self.validated_data.get('first_name', ''),
            'last_name': self.validated_data.get('last_name', ''),
            'password1': self.validated_data.get('password1', ''),
            'email': self.validated_data.get('email', ''),
        }

    def save(self, request):
        adapter = get_adapter()
        user = adapter.new_user(request)
        self.cleaned_data = self.get_cleaned_data()
        adapter.save_user(request, user, self)
        setup_user_email(request, user, [])
        user.profile.save()
        return user
  1. В settings.py обязательно добавьте ссылку на новый сериализатор.

    REST_AUTH_REGISTER_SERIALIZERS = {
            'REGISTER_SERIALIZER': 'path.to.RegisterSerializer',
    }
    

Ответ 2

Вы также можете просто перезаписать метод custom_signup на RegisterSerializer, который предназначен для этой цели.

from rest_auth.registration.serializers import RegisterSerializer
from rest_auth.registration.views import RegisterView
from rest_framework import serializers


class NameRegistrationSerializer(RegisterSerializer):

  first_name = serializers.CharField(required=False)
  last_name = serializers.CharField(required=False)

  def custom_signup(self, request, user):
    user.first_name = self.validated_data.get('first_name', '')
    user.last_name = self.validated_data.get('last_name', '')
    user.save(update_fields=['first_name', 'last_name'])


class NameRegistrationView(RegisterView):
  serializer_class = NameRegistrationSerializer

Затем используйте следующее в urls.py

url(r'^rest-auth/registration/name-registration/$', NameRegistrationView.as_view(), name="rest_name_register")

или установите REGISTER_SERIALIZER в settings.py

REST_AUTH_REGISTER_SERIALIZERS = {
    'REGISTER_SERIALIZER': 'path.to.RegisterSerializer',
}

Ответ 3

Более элегантным решением было бы унаследовать от RegisterSerializer и расширить по мере необходимости.

class MyRegisterSerializer(RegisterSerializer):
    first_name = serializers.CharField(required=True, write_only=True)
    last_name = serializers.CharField(required=True, write_only=True)

    def get_cleaned_data(self):
        return {
            'first_name': self.validated_data.get('first_name', ''),
            'last_name': self.validated_data.get('last_name', ''),
            'password1': self.validated_data.get('password1', ''),
            'email': self.validated_data.get('email', ''),
        }

    def save(self, request):
        adapter = get_adapter()
        user = adapter.new_user(request)
        self.cleaned_data = self.get_cleaned_data()
        adapter.save_user(request, user, self)
        setup_user_email(request, user, [])
        user.save()
        return user