Django虚拟字段:从虚拟字段设置属性的值

时间:2014-06-20 17:45:59

标签: python django django-models

我试图理解django ModelFields的内部实现django-hstore上的新功能。

基本上我想从具有预定义模式的HStore Dictionary创建虚拟字段。

作为第一步,我想隐藏实际的HStore Dictionary字段,而是使用虚拟字段中的值来组成最终的HStore字典。

除了保存操作之外,我已经能够使管理员正常运行所有操作,而保存操作不会将虚拟字段的值保存到HStore字典中。

这是我的实际VirtualField代码:

# virtual.py
from django.db.models.fields import Field


class VirtualField(Field):
    """ Virtual Field """

    def __init__(self, *args, **kwargs):
        try:
            self.hstore_field_name = kwargs.pop('hstore_field_name')
        except KeyError:
            raise ValueError('missing hstore_field_name keyword argument')
        super(VirtualField, self).__init__(*args, **kwargs)

    def contribute_to_class(self, cls, name, virtual_only=True):
        super(VirtualField, self).contribute_to_class(cls, name, virtual_only)

    def value_from_object(self, obj):
        """
        Returns the value of this field in the given model instance.
        """
        hstore_field = getattr(obj, self.hstore_field_name)
        return hstore_field[self.attname]

    def save_form_data(self, instance, data):
        hstore_field = getattr(instance, self.hstore_field_name)
        hstore_field[self.attname] = data
        setattr(instance, self.hstore_field_name, hstore_field)

models.py(只是原型设计)

class ModeledDataBag(models.Model):
    name = models.CharField(max_length=32)
    data = hstore.ModeledDictionaryField(schema={
        'number': {
            'type': int,
            'default': 0
        }
    })

    number = VirtualField(hstore_field_name='data')

    objects = hstore.HStoreManager()

我认为 save_form_data 可以解决问题,但事实并非如此。

在django docs中,我在"自定义模型字段"中找到了"The SubfieldBase metaclass"部分。看起来像我需要的页面。

这是正确的道路吗?

有研究的例子吗?

任何人都可以提供一个如何设置密钥值的示例" number" hstore字段"数据"并将其存储在数据库中? 从那里我想我知道如何继续前进。

...谢谢

1 个答案:

答案 0 :(得分:3)

正确的答案是使用python描述符,实现的解决方案在这里:

https://github.com/djangonauts/django-hstore/blob/8229e850e1631d8fd038436d2aa1fccb26b9a699/django_hstore/virtual.py#L29

此功能最初是为名为 nodeshot 的开源项目开发的,这是一个示例实现,其架构取自 settings.py

https://github.com/ninuxorg/nodeshot/blob/028bb28009bf73b2a4fc087f811e1b66c958907e/nodeshot/core/nodes/models/node.py#L49