Django动态模型属性

时间:2015-12-11 15:24:20

标签: python django dynamic properties translation

我试图将动态模型属性添加到Django模型中,但它以某种方式无法工作并最终覆盖属性。

我想要实现的目标

我希望能够编写mymodel.title_de,而django内部必须查询数据库并获得正确的翻译。

简化示例代码(models.py)

class Translation(models.Model):
    def get_prototype(self, model, cache_id, default, lang=None):
        if not lang:
            lang = get_language()

        cached = cache.get('%s_%s_%s' % (cache_id, self.id, lang))
        if not cached:
            try:
                cached = model.objects.get(quiz_id=self.id, lang=lang).translation
                cache.set('%s_%s_%s' % (cache_id, self.id, lang), cached, 60*60)
                return cached
            except model.DoesNotExist:
                return default
        else:
            return cached


class MyModel(Translation):
    # lots of things
    pass


class MyModelTrans(models.Model):
    mymodel = models.ForeignKey(MyModel)
    lang = models.CharField(max_length=5, choices=settings.LANGUAGES, db_index=True)

    class Meta:
        unique_together = ('mymodel', 'lang')
        abstract = True


class MyModelTitleTrans(MyModelTrans):
    translation = models.TextField()


mymodel_attr_mapping = {
    'title': [MyModelTitleTrans, 'qtitle', ''],
}

for key in mymodel_attr_mapping.keys():
    def attr_general(self):
        return self.get_prototype(
            mymodel_attr_mapping[key][0],
            mymodel_attr_mapping[key][1],
            mymodel_attr_mapping[key][2])

    setattr(MyModel, key, property(attr_general))

    for l in settings.LANGUAGES:
        def attr_lang(self):
            return self.get_prototype(
                mymodel_attr_mapping[key][0],
                mymodel_attr_mapping[key][1],
                mymodel_attr_mapping[key][2], l[0])
        setattr(MyModel, '%s_%s' % (key, l[0]), property(attr_lang))

预期结果

m = MyModel()
m.save()

mtt = MyModelTitleTrans.objects.create(mymodel=m, lang='de', translation='hallo')
mtt.save()

m.title_de
>>> 'hallo'

当前结果

空字符串

加分问题

还有其他方法可以达到这个目的吗?不幸的是我无法改变查询翻译的方式( m.title_de

提前感谢任何想法!

1 个答案:

答案 0 :(得分:0)

财产应该有效,而且确实有效。让我们把它归结为:

setattr(MyModel, 'title_de', property(lambda self: 'hallo'))

这样你就能得到预期的结果。

请在MyModel逻辑中查找问题或尝试清理缓存。