在Django 1.5中将电子邮件设置为用户名

5
我正在阅读以下文档:https://docs.djangoproject.com/en/1.5/topics/auth/customizing/#substituting-a-custom-user-model 因此,在我的 settings.py 文件中,我添加了以下内容:
AUTH_USER_MODEL = 'membership.User'

在我的会员应用程序models.py文件中,我有以下内容:

from django.contrib.auth.models import AbstractBaseUser

class User(AbstractBaseUser):
    USERNAME_FIELD = 'email'

运行 python manage.py syncdb 命令时出现以下错误:
FieldDoesNotExist: User has no field named 'email'

我查看了AbstractBaseUser类源代码,可以在这里看到该字段已经被定义: https://github.com/django/django/blob/master/django/contrib/auth/models.py#L359。有何不妥?
2个回答

18

AbstractBaseUser 没有 email 字段,而 AbstractUser 有。

如果你想将电子邮件用作唯一标识符,则需要从 AbstractBaseUser 派生子类,并定义带有 unique=True 的 email 字段,同时编写其他功能,例如该模型的 Manager

from django.contrib.auth.models import AbstractBaseUser, BaseUserManager,\
    PermissionsMixin
from django.db import models
from django.utils.translation import ugettext_lazy as _
from django.utils import timezone
from django.utils.http import urlquote


class CustomUserManager(BaseUserManager):

    def create_user(self, email, password=None, **extra_fields):
        """
        Creates and saves a User with the given email and password.
        """
        now = timezone.now()
        if not email:
            raise ValueError('The given email must be set')
        email = CustomUserManager.normalize_email(email)
        user = self.model(email=email,
                          is_staff=False, is_active=True, is_superuser=False,
                          last_login=now, date_joined=now, **extra_fields)

        user.set_password(password)
        user.save(using=self._db)
        return user

    def create_superuser(self, email, password, **extra_fields):
        u = self.create_user(email, password, **extra_fields)
        u.is_staff = True
        u.is_active = True
        u.is_superuser = True
        u.save(using=self._db)
        return u


class User(AbstractBaseUser, PermissionsMixin):
    email = models.EmailField(_('email address'), unique=True)
    first_name = models.CharField(_('first name'), max_length=30, blank=True)
    last_name = models.CharField(_('last name'), max_length=30, blank=True)
    is_staff = models.BooleanField(_('staff status'), default=False,
        help_text=_('Designates whether the user can log into this admin '
                    'site.'))
    is_active = models.BooleanField(_('active'), default=True,
        help_text=_('Designates whether this user should be treated as '
                    'active. Unselect this instead of deleting accounts.'))
    date_joined = models.DateTimeField(_('date joined'), default=timezone.now)

    objects = CustomUserManager()

    USERNAME_FIELD = 'email'

    class Meta:
        verbose_name = _('user')
        verbose_name_plural = _('users')

    def get_absolute_url(self):
        return "/users/%s/" % urlquote(self.pk)

    def get_full_name(self):
        """
        Returns the first_name plus the last_name, with a space in between.
        """
        full_name = '%s %s' % (self.first_name, self.last_name)
        return full_name.strip()

    def get_short_name(self):
        "Returns the short name for the user."
        return self.first_name

    # define here other needed methods
    # Look at django.contrib.auth.models.AbstractUser

另外,您可能希望将此用户添加到管理员页面。查看UserAdmin,并重新定义其以与使用电子邮件字段作为唯一标识符的新用户模型兼容。


在从AbstractUser继承后,我遇到了"FieldError: Local field 'email' in class 'User' clashes with field of similar name from base class 'AbstractUser'"的问题。 - Adam
已更新答案。首先我要提到的是,AbstractBaseUser 没有电子邮件字段,但问题中写道它有。 - stalk

4

很不幸,django.contrib.auth中没有任何你可以简单地继承的模型,来得到一个替代用户名的电子邮件地址,并且与其他django.contrib.auth的东西(例如组)很好地配合使用的模型。

  1. 最简单的方法是从django.contrib.auth中复制models.pyadmin.pyforms.py,到处删除用户名,然后把电子邮件地址替换为它。我已经这样做了,并在一些客户项目中成功使用。

  2. 我已经将其放在Github和PyPI上,所以您可以通过以下方式安装:

pip install django-libtech-emailuser

并检查 GitHub上的使用说明


在Django 1.5中运行得非常好。我按照您在Github上的说明安装了您的应用程序,然后只需将我的项目中任何一个实例的from django.contrib.auth.models import User替换为from emailuser.models import EmailUser,并将任何引用User的地方替换为EmailUser。不确定为什么在Django中使用电子邮件作为用户名的常见做法除了这些1.5的新功能之外还有什么不同,但它确实存在。感谢您的分享。 - Banjer

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接