0

我有这个模型文件,其中定义了自定义用户。

class CustomUser(AbstractBaseUser,PermissionsMixin, GuardianUserMixin):
    email = models.EmailField(max_length=50,
            verbose_name='email address', unique=True)
    first_name    = models.CharField('first name', max_length=15,blank=False)
    last_name     = models.CharField('last name', max_length=15,blank=True)
    date_joined   = models.DateTimeField('date joined', auto_now_add=True)
    slug          = models.SlugField('slug', max_length=50, unique=True, null=True) 
    is_active     = models.BooleanField('active',default=True)


    objects = UserManager()

    USERNAME_FIELD = 'email'
    REQUIRED_FIELDS = []


    class Meta:
    app_label = "latiro_app"
    db_table =  "users"


        permissions = (
                ("view_user", "view User"),
                ("edit_profile", "Edit Profile"),
                )


    def get_full_name(self):
        full_name = '{0} {1}'.format(self.first_name, self.last_name)
        return full_name.strip()

    def get_short_name(self):
        return self.first_name


    def generate_user_slug(self):
        max_length = CustomUser._meta.get_field('slug').max_length
        full_name = self.get_full_name()
        slug = original = slugify(full_name)[:max_length]

        for i in itertools.count(1):
            if not CustomUser.objects.filter(slug=slug).exists():
                break

            slug = '{0}-{1}' .format(original[:max_length - len(str(i)) - 1], i)
        return slug


    def save(self, *args, **kwargs):
        if not self.slug:
            self.slug = self.generate_user_slug()
        super().save()         


def get_anonymous_user_instance(CustomUser):
    return CustomUser(first_name='Anonymous')


class Profile (models.Model):
    user            = models.OneToOneField(settings.AUTH_USER_MODEL, on_delete= models.CASCADE,
                                    null=False, verbose_name='list of users')
    phone_regex   = RegexValidator(regex=r'^\+?1?\d{9,15}')
    phone_number  = models.CharField('phone number', validators=[phone_regex], max_length=50, blank=True)
    country         = models.CharField(max_length=50,blank=True)
    province        = models.CharField(max_length=50)
    city            = models.CharField(max_length=50,blank=True)
    profile_picture = models.ImageField(upload_to='user_profile/%y/%m/%d/', blank=True)
    followers       = models.IntegerField(default=0, null=True)
    following       = models.IntegerField(default=0, null=True)

而这个信号

def create_user_profile(sender, instance, created, **kwargs):

    if kwargs.get('created', True) and not kwargs.get('raw', False):
        profile = Profile(user=instance)
            profile.save()

post_save.connect(create_user_profile,
              sender = settings.AUTH_USER_MODEL)

运行第一次迁移时,Django 将我的应用程序很好地迁移到数据库中。但是当我第二次运行迁移时,Django 会抛出这个错误:

django.db.utils.IntegrityError: (1062, "Duplicate entry '' for key 'email'")。

我正在使用 MySQL。这里出了什么问题?

更新:当我进行第二次迁移时,我发现了引发错误的代码。

我的模型文件中有这段代码:

def get_anonymous_user_instance(CustomUser):
    return CustomUser(first_name='Anonymous')

我将此功能导入到我的设置中,这样当我第一次进行迁移时,将创建匿名用户。并且它确实按预期进行。但是当我再次运行迁移时,django 尝试创建另一个匿名用户,这就是我收到此错误的原因。

当我在我的设置中注释掉这一行时:

" GUARDIAN_GET_INIT_ANONYMOUS_USER = 'latiro_app.models.get_anonymous_user_instance' "

我没有看到这个错误。

4

1 回答 1

1

email是一个独特的领域,所以你不能有两个CustomUser相同的 s e-mail。Guardian 正在从设置中的可调用对象返回的对象创建新用户GUARDIAN_GET_INIT_ANONYMOUS_USER,因此它尝试创建相同的用户并失败。您可以使用以下方法解决此问题get_or_create

def get_anonymous_guardian_user_instance(CustomUser):
    user, _ = CustomUser.objects.get_or_create(
        first_name='Anonymous',
        email='guardian@example.com',
    )
    return user

实际上有一个关于此的Guardian 问题,但不幸的是它没有解决,也不清楚它是否应该是一个问题。该文档提醒用户注意,但在我看来,syncdb关于创建匿名用户的部分(它不处理unique约束)中的可调用对象应该返回什么并不是很清楚。

于 2018-03-22T13:31:15.243 回答