为 django 1.5 自定义用户模型子类化 django-registration 1.0 表单
Posted
技术标签:
【中文标题】为 django 1.5 自定义用户模型子类化 django-registration 1.0 表单【英文标题】:Subclassing django-registration 1.0 forms for django 1.5 custom user models 【发布时间】:2013-07-28 08:26:11 【问题描述】:django-registration 1.0 现在支持 django 1.5 自定义用户模型。 django-registration 文档只有以下关于它的常见问题解答:
我正在使用 Django 1.5 和自定义用户模型;我该怎么做?
虽然提供的两个内置后端 django-registration 都假设 Django 的默认用户模型,基础 视图类故意与用户模型无关。简单的子类 它们,并为您的自定义用户模型实现逻辑。
我不确定我需要子类化哪些视图以及它们应该包含哪些内容。我还注意到 django-registration 中的 ProfileManager
仍然假定一个单独的用户名字段。
在我的具体情况下,我删除了“用户名”字段,添加了“显示名称”,并将“电子邮件”作为识别字段:
class MyUser(AbstractBaseUser, PermissionsMixin):
email = models.EmailField(
verbose_name="Email Address",
max_length=384,
unique=True,
db_index=True,)
display_name = models.CharField(max_length=128, blank=True)
date_joined = models.DateTimeField(default=timezone.now)
is_active = models.BooleanField(default=True)
is_admin = models.BooleanField(default=False)
objects = MyUserManager()
USERNAME_FIELD = 'email'
def get_full_name(self):
return self.email
def get_short_name(self):
return self.email
def __unicode__(self):
return self.email
def has_perm(self, perm, obj=None):
return True
def has_module_perms(self, app_label):
return True
@property
def is_staff(self):
return self.is_admin
在没有子类化任何 django-registration 类的情况下,注册表单的默认呈现从 User
中提取字段,而不是 MyUser
。
我看过下面的 SO 线程 django-registration app and Django 1.5 custom user model,但没有帮助。
更新
我注意到 RegistrationForm 使用“用户名”字段进行硬编码。常见问题解答只提到子类化后端,所以我不确定这里的意图是什么。我也应该对表单进行子类化吗?
【问题讨论】:
请看一下这个提交 - bitbucket.org/LinnTroll/django-registration-1.5/commits/… !!为此,您需要继承DefaultBackend
、RegistrationFormFromUserModel
和 RegistrationManager
。
你在settings.py中把AUTH_USER_MODEL改成新的自定义用户模型了吗?
你最后做了什么?我正在考虑仅处理此问题,但明确定义注册表单中的字段,而不是仅使用 form
模板标签。不理想,但我认为这可能是最简单的选择。
我刚刚对特定应用程序执行了查找和替换,并用我的自定义用户模型和字段替换了所有实例。不优雅,但我发现它相当有效
@jon 你找到解决方案了吗?我有类似的问题。你可以发布你所做的工作吗?谢谢
【参考方案1】:
有些部分绝对不兼容 Django 1.5:https://bitbucket.org/ubernostrum/django-registration/src/8f242e35ef7c004e035e54b4bb093c32bf77c29f/registration/forms.py?at=default#cl-48
class RegistrationForm(forms.Form):
# ...
def clean_username(self):
# ...
# The line below needs fixing
existing = User.objects.filter(username__iexact=self.cleaned_data['username'])
if existing.exists():
raise forms.ValidationError(_("A user with that username already exists."))
else:
return self.cleaned_data['username']
因此,除非这些方法被更改和/或您将它们子类化,否则它不会起作用。
对于您的具体情况,此注册表应该可以解决问题:
from registration import forms as registration_forms
from django.contrib import auth
class RegistrationForm(registration_forms.RegistrationForm):
def clean_username(self):
'''
Validate that the username is alphanumeric and is not already
in use.
'''
User = auth.get_user_model()
existing = User.objects.filter(display_name__iexact=self.cleaned_data['username'])
if existing.exists():
raise forms.ValidationError(_("A user with that name already exists."))
else:
return self.cleaned_data['username']
除了模型上的自定义属性:
class MyUser(AbstractBaseUser, PermissionsMixin):
# ...
def get_username(self):
return self.display_name
def set_username(self, username):
self.display_name = username
def del_username(self):
del self.display_name
username = property(get_username, set_username, del_username)
【讨论】:
以上是关于为 django 1.5 自定义用户模型子类化 django-registration 1.0 表单的主要内容,如果未能解决你的问题,请参考以下文章
无法在 Django 1.5 中使用自定义用户模型创建超级用户
在 Django 1.5 自定义用户模型中使用电子邮件作为用户名字段导致 FieldError