代码之家  ›  专栏  ›  技术社区  ›  TM. Randy Simon

django-允许重复用户名

  •  10
  • TM. Randy Simon  · 技术社区  · 15 年前

    我正在Django的一个项目中工作,该项目要求单独的用户组 username 命名空间。

    例如,我可能有多个“组织”,并且 用户名 应该只在该组织中是唯一的。

    我知道我可以通过使用另一个包含用户名/组织ID的模型来实现这一点,但是仍然会在默认的django-auth上留下这个无用的(和必需的)字段。 User 我必须用一些东西来填充。

    我已经由自己的auth后端编写了,它根据LDAP对用户进行身份验证。然而,正如我之前提到的,我仍然面临着如何填充/忽略 用户名 默认Django用户的字段。

    是否有方法删除 用户名 对于django-auth用户?

    5 回复  |  直到 6 年前
        1
  •  8
  •   Community Keith    7 年前

    我不确定这是否正是你要找的,但我认为你可以使用类似于 this answer .

    以下代码可以工作,只要它在Django加载模型时执行的位置。

    from django.contrib.auth.models import User
    
    User._meta.get_field('username')._unique = False
    

    请注意,这不会更改 auth_user 表(如果已创建)。所以你需要做这个改变 之前 你运行同步数据库。或者,如果您不想重新创建 授权用户 表,您可以进行此更改,然后手动更改数据库表以删除约束。

        2
  •  6
  •   Srikar Appalaraju Tonetel    12 年前

    您可以做的是扩展用户模型。对于用户表,生成一个在站点中根本不会显示的用户名(例如,a_123、a_345)。

    然后创建一个扩展用户的新模型。

    class AppUser(User):
        username = models.CharField...
        organization = models.CharField...
    

    然后创建一个使用appuser而不是user对象的自定义身份验证后端。

        3
  •  1
  •   basZero    15 年前

    我个人并不需要找到解决方案,但解决这个问题的一种方法(从SaaS的角度)是在用户名前面加上组织标识符(假定为唯一的组织)。例如:subdomain.yoursite.com将等同于用户名为:subdomain_username的用户。您只需在登录到子域时编写一些业务逻辑代码,就可以将其附加到用户名上。

        4
  •  0
  •   tar    11 年前

    我正面临着同样的问题,我读了很多书(关于如何在1.5中解决这个问题),我只是想到了一个更简单的解决方案。如果您只是添加一个带有组织ID的固定长度前缀来存储用户名呢?

    即,组织ID=115,选择用户名=“john”,固定长度为6。因此,在数据库中,您存储为用户名“000115_john”。

    当您登录时,只需加入两个参数,并尝试使用Django提供的内容进行身份验证。我不确定固定长度是否是严格必要的,但是如果用户选择只有数字的用户名,就可以避免不必要的结果。

        5
  •  0
  •   Srijan Anand    6 年前

    我也遇到了这个问题。我正在做一个项目,我必须使用电子邮件和手机号码作为登录字段,但它们都不应该是唯一的,因为它们是不同类型的用户,一个用户可以有多个用户实体,而且该项目只需要一个auth用户表(hectic right!).

    所以我扩展了abstractbaseuser类,在这里我可以更改username_field属性。以下是如何:

    from django.contrib.auth.models import AbstractUser
    from django.contrib.auth.models import PermissionsMixin
    
    # Custom model for User
    class User(AbstractBaseUser, PermissionsMixin):
    
        first_name = models.CharField(max_length=100, blank=False)
        last_name = models.CharField(max_length=100, blank=True)
        password = models.CharField(max_length=255, blank=False)
        email = models.EmailField(max_length=255, blank=False)
        mobile = models.CharField(max_length=12)
        user_type = models.ForeignKey(UserTypes, on_delete=models.DO_NOTHING)
        is_approved = models.BooleanField(default=False)
    
        objects = UserManager()
        # Here's the Catch
        USERNAME_FIELD = 'id'
    
        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
    
        def email_user(self, subject, message, from_email=None, **kwargs):
            '''
            Sends an email to this User.
            '''
            send_mail(subject, message, from_email, [self.email], **kwargs)
    
        class Meta:
            db_table = 'user'
    

    是的,没错,很惊讶?用户名字段应该是唯一的字段,这是该属性的约束。我不能使用电子邮件或手机号码作为唯一字段。

    然后我创建了一个自定义管理器来删除用户名字段(参考= https://simpleisbetterthancomplex.com/tutorial/2016/07/22/how-to-extend-django-user-model.html#abstractbaseuser )

    from django.contrib.auth.base_user import BaseUserManager
    
    
    class UserManager(BaseUserManager):
        use_in_migrations = True
    
        def _create_user(self, email, password, **extra_fields):
            """
            Creates and saves a User with the given email and password.
            """
            if not email:
                raise ValueError('The given email must be set')
            email = self.normalize_email(email)
            user = self.model(email=email, **extra_fields)
            user.set_password(password)
            user.save(using=self._db)
            return user
    
        def create_user(self, email, password=None, **extra_fields):
            extra_fields.setdefault('is_superuser', False)
            return self._create_user(email, password, **extra_fields)
    
        def create_superuser(self, email, password, **extra_fields):
            extra_fields.setdefault('is_superuser', True)
    
            if extra_fields.get('is_superuser') is not True:
                raise ValueError('Superuser must have is_superuser=True.')
    
            return self._create_user(email, password, **extra_fields)
    

    这就行了。