6

django-allauth をカスタム ユーザー モデル (サブクラス化された AbstractUser) と統合しようとしていますが、サインアップ フォームをテストすると、フィールド (date_of_birth) が null であるため整合性エラーが発生しますが、送信された値は u'1976-4- でした6'

私は新しいカスタムユーザーのものと、django-allauth を学んでいるときにクラスベースのビューを学んでいるので、何か間違ったことをしていると確信していますが、github の問題を数日読んだ後、いくつかのチュートリアル、readthedocs、およびstackoverflowの質問私はまだ自分が間違っていることを明確に理解していません(まあ、私が間違っていることの1つを知っています:あちこちでさまざまなソリューションを試しているので、間違いなくミスモッシュがあります実装)

しかし、allauth をサブクラス化された AbstractUser と統合する方法についての良い答えが見つからないので、誰かが私を啓発できるなら、本当に感謝しています。

(注 - フィクスチャを介してロードしたユーザーとしてログインすると、サイトは多かれ少なかれ機能します。したがって、django-allauth 以外の省略は省略であると想定してください。以下以外の説明が必要な場合は、喜んでお知らせします。編集)

設定.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'

ユーザーデータ/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)

ユーザーデータ/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()

templates/allauth/account/signup.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']

エラーが発生しました (投稿データとの違いに注意してください)

例外の種類: /accounts/signup/ の IntegrityError

例外値: 列 "date_of_birth" の null 値が not-null 制約に違反しています

詳細: 失敗した行には (19, pbkdf2_sha256$12000$exNVzh4QI0Rb$mCTz9Tc+TIBbD8+lIZs2B3hqjxd+qmI..., 2014-07-02 16:27:43.751428+00, f, gn, One, Person, 1@bt.co が含まれています、f、t、2014-07-02 16:27:43.751473+00、null、、null、null、null、f、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

4

1 に答える 1

9

答えは、私がまだ考えているところですが、allauth.account.adapter.DefaultAccountAdapter正しく処理されないフィールド タイプ (たとえば__getitem__、 のような属性を持たないフィールドmodels.DateField) を含むモデルを保存する場合は、実装する必要があるようです。以下のようなカスタムアダプター。

:サブクラス化された抽象ユーザー モデルは、渡されたユーザーであるため user.email = data.get('email') 、DefaultAccountAdapter クラスで使用される allauth 内部メソッドを使用するのではなく、フォーム データを直接使用することをお勧めします。

userdata/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
于 2014-07-08T01:13:29.123 に答える