将现有模型迁移到Django内置用户模型

时间:2013-01-11 19:48:29

标签: python django django-models django-users

我正在运行Django 1.4.2项目。我有一个类User(models.Model)的数据库,现在想要更改它,以便我使用来自django.contrib.auth.models的内置模型User。我可以简单地使用我现有的代码,只是让自己的用户模型成为Django内置的一个子类吗?这是现在的模型。

class User(models.Model):
    u_id = models.AutoField(primary_key=True)
    #password = models.CharField(max_length=765)
    old_password = models.CharField(max_length=765, null=True, blank=True)
    birthday = models.DateField(null=True, blank=True)
    school_year = models.CharField(max_length=765, blank=True)
    gender = models.CharField(max_length=18, blank=True)
    #email = models.CharField(max_length=765, blank=True)
    profile_pic = models.CharField(max_length=765, blank=True)
    is_cloudinary_pic = models.BooleanField(default=False)
    fb_id = models.CharField(max_length=765, blank=True)
    phone_num = models.CharField(max_length=765, blank=True)
    name = models.CharField(max_length=765, blank=True)
    confirmcode = models.CharField(max_length=765, blank=True)
    status = models.CharField(max_length=765, blank=True)
    netid = models.CharField(max_length=765, blank=True)
    favorites = models.ManyToManyField('listings.Listing', blank=True, related_name='favorites')
    stripe = models.CharField(max_length=765, blank=True)
    rating = models.IntegerField(null=True, blank=True)
    preferences = models.CharField(max_length=765, blank=True)
    b_notification = models.CharField(max_length=765, blank=True)
    l_notification = models.CharField(max_length=765, blank=True)
    address = models.CharField(max_length=765, blank=True)
    city = models.CharField(max_length=765, blank=True)
    state = models.CharField(max_length=6, blank=True)
    zip = models.CharField(max_length=15, blank=True)
    emailprefs = models.CharField(max_length=30, blank=True)
    exp_month = models.CharField(max_length=6, blank=True)
    exp_year = models.CharField(max_length=12, blank=True)
    last4 = models.CharField(max_length=12, blank=True)
    c_type = models.CharField(max_length=765, blank=True)
    user_type = models.CharField(max_length=10, blank=True)
    contract = models.ForeignKey('contracts.Contract', null=True,blank=True, default=None, on_delete=models.SET_NULL)
    last_listing_booked = models.ForeignKey('listings.Listing', null=True,blank=True, default=None, on_delete=models.SET_NULL, related_name='last_listing_booked')
    last_locked_on = models.IntegerField(null=True, blank=True)
    waitlist_number = models.IntegerField(null=True, blank=True)
    waitlist_listing = models.ForeignKey('listings.Listing', null=True, blank=True, on_delete=models.SET_NULL, related_name='waitlist_listing')
    uploaded_contract = models.FileField(upload_to=contract_file_name, max_length=765, null=True, blank=True, default=None, storage=FileSystemStorage(location=os.path.join(MEDIA_ROOT, 'contracts')))

当我在使我的模型成为Django User模型的子类后尝试使用Django South进行迁移时,我被要求为user_ptr设置一个值,这是一个我找不到任何信息的字段。

一般来说,简单地以这种方式进行子类化是不安全的吗?我不能丢失我已经拥有的所有用户数据并重新开始。请注意,密码和电子邮件字段已被注释掉,因为它们与Django的那些字段版本冲突。

2 个答案:

答案 0 :(得分:4)

如果可以选择升级到Django 1.5,只需提供use your own User model,只要它提供所有必填字段。

user_ptr是一个隐式的OneToOne模型,它将您的对象与您派生的类相关联。最简单的方法可能是在以下步骤中手动创建迁移:

  1. 添加user_ptr字段WITH null = True

  2. 迭代所有User对象并创建相应的auth.User对象。如果您不确定如何执行此操作,可以阅读South文档中的section on data migrations

  3. 禁止null user_ptr字段

  4. 如果您使用的散列方案与Django的内置方法不兼容,您可能需要做的另一件事是实现自己的密码。这不是很难做到,并在another question中解决。

答案 1 :(得分:4)

如果使用Django 1.4,则应创建UserProfile模型,而不是django.contrib.auth.models.User的子类。直接扩展Django的User类会导致内部django.auth机制出现各种问题。

您可以在http://www.turnkeylinux.org/blog/django-profile了解有关如何在django中实施用户个人资料的详情。

Django 1.5 configurable user model another answerVinay Pai提到了{{3}}。