2012-05-15 66 views
3

我正在嘗試構建自定義模型管理器,但遇到了錯誤。代碼如下所示:Django自定義模型管理器數據庫錯誤

class LookupManager(models.Manager): 
    def get_options(self, *args, **kwargs): 
     return [(t.key, t.value) \ 
       for t in Lookup.objects.filter(group=args[0].upper())] 


class Lookup(models.Model): 
    group = models.CharField(max_length=1) 
    key = models.CharField(max_length=1) 
    value = models.CharField(max_length=128) 
    objects = LookupManager() 

(我曾與get_options發揮各地比較使用super()等方式來過濾結果很多)

當我運行syncdb,我得到以下錯誤(ops_lookup感相應的表):

django.db.utils.DatabaseError: no such table: ops_lookup 

我注意到,如果我改變了經理返回[],而不是一個過濾器,然後syncdb作品。另外,如果我運行了syncdb並且所有表都存在,那麼將代碼更改爲上面的代碼,它也可以工作。

我該如何讓Django第一次運行syncdb時不要期望這張表存在?

更新 通過回溯追溯之後,我意識到發生了什麼事。查找表旨在包含填充其他表中某些列的選項的值。我認爲會發生什麼情況是,當創建其他表時,管理器會被調用,看起來,在創建查找表之前會發生這種情況。

有沒有辦法強迫Django的先創建查找表(短重命名呢?)

+0

請使用--traceback運行syncdb,併發布完整的回溯。 – jpic

回答

0

發生了什麼事是你想在模塊加載時訪問數據庫。例如:

class MyModel(models.Model): 
    name = models.CharField(max_length=255) 


class OtherModel(models.Model): 
    some_field = models.CharField(
     max_length=255, 
     # Next line fails on syncdb because the database table hasn't been created yet 
     # but the model is being queried during module load time (during class definition) 
     choices=[(o.pk, o.name) for o in MyModel.objects.all()] 
    ) 

這相當於你在做什麼,因爲,如你所說,你使用的管理器方法(及物動詞)生成其它型號的選擇。

用生成器表達式替換列表理解將返回一個可迭代的值,但不會評估過濾後的查詢集,直到第一次迭代。因此,這將解決上面的例子:使用

choices=((o.pk, o.name) for o in MyModel.objects.all()) 

你的榜樣,那就是:

class LookupManager(models.Manager): 
    def get_options(self, *args, **kwargs): 
     return ((t.key, t.value) for t in Lookup.objects.filter(group=args[0].upper())) 

(注意使用()代替[])(外這是創建生成器表達式的語法。