一尘不染

如何从Django中的contrib.auth使模型用户中的电子邮件字段唯一

django

我需要contrib.auth通过确保电子邮件字段条目唯一来修补标准的用户模型:

User._meta.fields[4].unique = True

代码在哪里最好做到这一点?

我想避免使用数字字段[4]。最好使用用户fields [‘email’],但字段不是字典,而是列表。

另一个想法可能是打开一个新票证,然后在其中上传带有新参数的补丁程序settings.py

AUTH_USER_EMAIL_UNIQUE = True

关于在Django用户模型中实现电子邮件地址唯一性的最正确方法有何建议?


阅读 483

收藏
2020-03-27

共1个答案

一尘不染

你的代码将无效,因为字段实例的属性是只读的。我担心这可能比你想的要复杂一些。

如果仅使用表单创建User实例,则可以定义一个强制执行以下行为的自定义ModelForm:

from django import forms
from django.contrib.auth.models import User

class UserForm(forms.ModelForm):
    class Meta:
        model = User

    def clean_email(self):
        email = self.cleaned_data.get('email')
        username = self.cleaned_data.get('username')
        if email and User.objects.filter(email=email).exclude(username=username).exists():
            raise forms.ValidationError(u'Email addresses must be unique.')
        return email

然后,只需在需要创建新用户的地方使用此表单即可。

顺便说一句,你可以用来Model._meta.get_field('field_name')按名称而不是按位置获取字段。因此,例如:

# The following lines are equivalent
User._meta.fields[4]
User._meta.get_field('email')

更新
Django文档建议你clean对跨多个表单字段的所有验证使用该方法,因为在所有<FIELD>.clean<FIELD>_clean方法之后调用该方法。这意味着你(大部分)可以依靠cleaned_data内部存在的字段值clean。

由于表单字段是按照声明的顺序进行验证的,因此我认为可以偶尔将多字段验证放入<FIELD>_clean方法中,只要所讨论的字段出现在它依赖的所有其他字段之后。我这样做是为了使所有验证错误都与字段本身(而不是表单)相关联。

2020-03-27