2017-02-12 17 views
0

我想定义一个通用模型,该模型可以在任何应用程序中使用。以下是我的通用模型的定义: -如何在django中定义通用模型的循环依赖关系

User = settings.AUTH_USER_MODEL 
# This mixin is used to track the user, who has created this object. 
class OwnedModel(models.Model): 
     owner   = ForeignKey(to=User, null=True, editable=False) 

     class Meta: 
      abstract = True 

# AreaOrVillage extending this model, will also get owner field, which 
# refer to User. 
class AreaOrVillage(OwnedModel): 

    area_village_name = LocationIdentifierField(name="areaOrVillageName", verbose_name="Area/Village:", blank=False, null=False) 
    area_village_code = LocationCodeField(name="areaOrVillageCode", verbose_name="Area/Village Code", null=True, blank=True) 
    zipcode    = ZipCodeField(verbose_name="ZipCode", null=False, blank=False) 
    # It's another model, referred by this model 
    tehsil    = models.ForeignKey(to=Tehsil, related_name="areas_or_villages") 

    class Meta: 
     verbose_name_plural = "Areas/Villages" 
     unique_together  = ("tehsil", "areaOrVillageName") 
     ordering   = ("tehsil", "areaOrVillageName") 
     app_label   = 'location_app' 

    def __str__(self): 
     text    = self.to_str() 
     return "{}{}{}".format(text, ", " if text and self.tehsil else "", 
           self.tehsil.__str__() if self.tehsil else "") 

    def to_str(self): 
     return "%s" %(self.areaOrVillageName or self.areaOrVillageCode) 

现在,在我的Django项目,我已经覆盖默认的用户模型如下(其中利用通用模型的,上面定义的): -

class PSS_User_Type(ModelBase): 

    def __new__(cls, name, bases, dictattrs): 
     user_type = super(PSS_User_Type, cls).__new__(cls, name, bases, dictattrs) 

     OVER_RIDDEN_ATTR = "is_staff" 
     NEW_ATTR   = "is_devteam_member" 

     if hasattr(user_type, OVER_RIDDEN_ATTR): 
      setattr(user_type, NEW_ATTR, 
        getattr(user_type, OVER_RIDDEN_ATTR)) 
      delattr(user_type, OVER_RIDDEN_ATTR) 

      setattr(user_type, "__getattr__", PSS_User_Type.__getattr__) 

     def __getattr__(self, attrname): 
      effective_attr = NEW_ATTR if attrname == OVER_RIDDEN_ATTR else attrname 
      return super(cls, self).__getattr__(attrname) 

     return user_type 

class PSS_User(LoggableModel, OwnedModel, ResolvableModelMixin, AbstractUser): 

    __metaclass__ = PSS_User_Type 

    REQUIRED_FIELDS = ['password', 'first_name', 'last_name', 
         'village', 'gotra'] 

    # TODO :- Add phone/mobile no details 

    dob = DateField(null=True, blank=True) 
    age = SmallIntegerField(null=True, blank=True, validators=[AgeValidator()]) 
    is_alive = BooleanField(null=False, blank=False, default=True) 
    gender = GenderField(null=False, blank=False) 
    user_village = ForeignKey(to=AreaOrVillage, null=False, related_name="gram_vasi") 
现在

,它引入循环依赖的问题,因为通用模型依赖于用户,以跟踪他们的创造者,用户定义还利用通用模型的,由于到运行针对数据库的迁移是给问题

django.db.utils.ProgrammingError: relation "users_pss_user" does not exist 

有没有更好的方法来解决这个问题?

在此先感谢。

回答

0

中的to领域ForeignKeycan also be a string

If you need to create a relationship on a model that has not yet been defined, you can use the name of the model, rather than the model object itself:

(强调)

例如,从您的代码,你可以做到以下几点:

user_village = ForeignKey(to='AreaOrVillage', null=False, related_name="gram_vasi") 

使用这种语法在您所有的ForeignKey字段中,您将避免任何循环引用。

+0

它给我的错误:users.PSS_User.user_village :(fields.E300)字段定义与模型'AreaOrVillage',它是未安装,或抽象的关系。 –

+0

您是否使用过这种语法来处理** Foreign **的所有**字段? – 2ps

+0

是的,我已经使用了这个外键的语法。顺便说一句,应用程序的名称也包括在内,所以,我已经将'appname.modelname'设置为ForeignKey()的参数 –