django-allauth: пользовательский пользователь генерирует IntegrityError в /accounts / signup/ (пользовательские поля обнуляются или теряются)
Я пытаюсь интегрировать django-allauth с пользовательской моделью пользователя (подкласс AbstractUser, но когда я тестирую форму регистрации, я получаю ошибку целостности из-за того, что поле (date_of_birth) равно null, но представленное значение было u '1976-4-6'
Я изучаю новый пользовательский материал, а также представления на основе классов, поскольку я изучаю django-allauth, поэтому я уверен, что делаю что-то неправильно, но после нескольких дней чтения вопросов github, нескольких учебников, readthedocs и вопросы stackoverflow у меня до сих пор нет четкого представления о том, что я делаю неправильно (ну, я знаю одну вещь, которую я делаю неправильно: пробую разные решения здесь и там, поэтому у меня определенно есть Мисс-мош реализаций)
Но я не могу найти хорошего ответа о том, как интегрировать allauth с подклассом AbstractUser, поэтому, если кто-то может просветить меня, я буду очень признателен.(Примечание-сайт более или менее работает, когда я регистрируюсь как пользователь, которого я загрузил через светильники, поэтому, пожалуйста предположим, что не-django-allauth пропуски являются пропусками - если вам нужно разъяснение по чему-то не ниже, я с радостью отредактирую)
Settings.py
AUTH_USER_MODEL = 'userdata.CtrackUser'
ACCOUNT_AUTHENTICATION_METHOD = 'username_email'
ACCOUNT_SIGNUP_FORM_CLASS = 'userdata.forms.SignupForm'
LOGIN_REDIRECT_URL = '/profile'
SOCIALACCOUNT_QUERY_EMAIL = True
SOCIALACCOUNT_AUTO_SIGNUP = False
ACCOUNT_USER_MODEL_USERNAME_FIELD = 'username'
Userdata/models.py
class CtrackUser(AbstractUser):
date_of_birth = models.DateField(help_text='YYYY-MM-DD format')
gender = models.CharField(max_length=2,
choices=settings.GENDER_CHOICES, blank=True)
race = models.CharField(max_length=2, choices=settings.RACE_CHOICES, null=True, blank=True)
condition = models.ForeignKey(Condition, null=True, blank=True)
location = models.CharField(max_length=255, null=True, blank=True)
my_symptoms = models.ManyToManyField(Symptom)
is_admin = models.BooleanField(default=False)
created = models.DateTimeField(auto_now_add=True)
modified = models.DateTimeField(auto_now=True)
Userdata/forms.py
from django import forms
from django.conf import settings
from django.contrib.auth import get_user_model
from allauth.account.forms import SetPasswordField, PasswordField
from crispy_forms.helper import FormHelper
from crispy_forms.layout import Submit
from medical.models import Condition
class SignupForm(forms.Form):
email = forms.EmailField(required=True,)
username = forms.CharField(max_length=80,required=True,)
password1 = SetPasswordField()
password2 = PasswordField()
first_name = forms.CharField(max_length=100,required=False,)
last_name = forms.CharField(max_length=100, required=False,)
date_of_birth = forms.DateField()
gender = forms.TypedChoiceField(
choices=settings.GENDER_CHOICES,
widget=forms.Select(attrs={'class': 'input-lg'}),
required=False,)
race = forms.TypedChoiceField(
choices=settings.RACE_CHOICES,
widget=forms.Select(attrs={'class': 'input-lg'}),
required=False,)
location = forms.CharField(max_length=255,required=False,)
condition = forms.ModelChoiceField(
queryset=Condition.objects.all(),
widget=forms.Select(attrs={'class': 'input-lg'}),
empty_label='Select condition (optional)'
)
class Meta:
model = get_user_model() # use this function for swapping user model
fields = ('email', 'username', 'password1', 'password2', 'first_name', 'last_name',
'date_of_birth', 'gender', 'race', 'location', 'condition', 'confirmation_key',)
def __init__(self, *args, **kwargs):
super(SignupForm, self).__init__(*args, **kwargs)
self.helper = FormHelper()
self.helper.form_id = 'signup_form'
self.helper.label_class = 'col-xs-6'
self.helper.field_class = 'col-xs-12'
self.helper.form_method = 'post'
self.helper.form_action = 'accounts_signup'
self.helper.add_input(Submit('submit', 'Sign up'))
def signup(self, request, user, model):
user.username = self.cleaned_data['username']
user.first_name = self.cleaned_data['first_name']
user.last_name = self.cleaned_data['last_name']
model.date_of_birth = self.cleaned_data['date_of_birth']
model.gender = self.cleaned_data['gender']
model.race = self.cleaned_data['race']
model.location = self.cleaned_data['location']
model.condition = self.cleaned_data['condition']
model.save()
user.save()
Шаблоны / allauth / учетная запись / Регистрация.html
<form id="signup_form" method="post" action="{% url 'account_signup' %}" class="form-inline">
{% csrf_token %}
{% crispy form %}
{% if redirect_field_value %}
<input type="hidden" name="{{ redirect_field_name }}" value="{{ redirect_field_value }}" />
{% endif %}
{# <div class="form-actions">#}
{# <button class="btn btn-primary" type="submit">Sign Up</button>#}
{# </div>#}
</form>
Данные
u'condition' [u'1']
u'confirmation_key' [u'']
u'date_of_birth' [u'1976-4-6']
u'email' [u'1@bt.co']
u'first_name' [u'One']
u'gender' [u'']
u'last_name' [u'Person']
u'location' [u''] u'password1' [u'123456']
u'password2' [u'123456']
u'race' [u'']
u'submit' [u'Sign up']
u'username' [u'gn']
Генерируемая ошибка (Примечание отличие от данных post)
Тип исключения: IntegrityError at / accounts / signup /
Значение исключения: нулевое значение в столбец "date_of_birth" нарушает ограничение not-null
Деталь: неудачная строка содержит (19, pbkdf2_sha256$12000$exNVzh4QI0Rb$mCTz9Tc+TIBbD8+lIZs2B3hqjxd+qmI..., 2014-07-02 16:27:43.751428+00, ф, Н., один, человек, 1@bt.co, ф, т, 2014-07-02 16:27:43.751473+00, нулевой, нуля , нуль, нуль, Ф, 2014-07-02 16:27:43.833267+00, 2014-07-02 16:27:43.83329+00).
Полная трассировка здесь: https://gist.githubusercontent.com/hanleybrand/ee260b53dfb404f5055a/raw/3325dc746120c4f7521b9b976abce45dd7d71a77/gistfile1.txt
1 ответ:
Ответ - который я все еще выясняю-кажется, что если вы сохраняете модель, содержащую типы полей, которые
allauth.account.adapter.DefaultAccountAdapter
не обрабатываются правильно (например, любое поле, в котором отсутствует атрибут__getitem__
, напримерmodels.DateField
), необходимо реализовать пользовательский адаптер, как показано ниже.Примечание: Ваша подклассовая абстрактная модель пользователя-это пользователь, который передается, поэтому рекомендуется использовать данные формы непосредственно, как
user.email = data.get('email')
вместо того, чтобы использовать внутренний allauth методы, используемые в классе DefaultAccountAdapterUserdata/adapter.py
class AccountAdapter(DefaultAccountAdapter): def save_user(self, request, user, form, commit=False): data = form.cleaned_data user.email = data.get('email') user.username = data.get('username') # all your custom fields user.date_of_birth = data.get('date_of_birth') user.gender = data.get('gender') if 'password1' in data: user.set_password(data["password1"]) else: user.set_unusable_password() self.populate_username(request, user) if commit: user.save() return user