如何枚举实体的反向引用

时间:2011-04-04 05:48:23

标签: python google-app-engine google-cloud-datastore bigtable referenceproperty

在这种情况下,company.py包含:

from django.utils import simplejson

class Company(db.Model):
    name=db.StringProperty()
    address=db.StringProperty()

    def to_JSON():
        d = dict([(p, unicode(getattr(self, p))) for p in self.properties()])
        return simplejson.dumps(d)

office.py包含:

from company import Company

class Office(db.Model):
    name = db.StringProperty()
    company = db.ReferenceProperty(Company, collection_name='offices')

模块3包含:

from company import Company

class Region(db.Model):
    name = db.StringProperty()
    company = db.ReferenceProperty(Company, collection_name='regions')

这是我需要做的简化示例,但应该很好地捕获它...我需要公司能够将自己表达为JSON字符串,并且它不知道其他类(Office和区域)引用它。这适用于普通属性:

c = Company(name='A Company',address='123 Main St, New York, NY 12345')
c.put()

dallas = Office(name='Dallas',company=c)
dallas.put()

ny = Office(name='New York',company=c)
ny.put()

northeast = Region(name='NorthEast',company=c)
northeast.put()

southwest = Region(name='SouthWest',company=c)
southwest.put()

logging.info('Company as JSON: %s' % c.to_JSON())

我得到的输出是:

{"name": "A Company", "address": "123 Main St, New York, NY 12345"}

因此,properties()包含了预期的实体属性,但我还希望包含来自关联对象的ReferenceProperty属性的反向引用。基本上我正在寻找的只是一个反向引用(collection_name)名称的列表,因为我知道它们代表了查询,但我找不到任何方法来枚举反向引用名称本身。

请记住,此代码实际上并不代表真实代码,而是大大简化了。现在这个有效:

offices = getattr(company, 'offices')

所以我期待看到的是:

{"name": "A Company", "address": "123 Main St, New York, NY 12345", 
 "offices": "<db.Query object at 0x110e992d0>", 
 "regions": "<db.Query object at 0x110e99300>"
}

我实际上需要后面的引用“offices”和“regions”要包含在properties()枚举中,但它们会以某种方式检查它们的唯一性(我不能例如,在公司中也有一个名为“offices”的StringProperty,因此必须有一种方法来枚举它们。有人知道吗?

2 个答案:

答案 0 :(得分:1)

检查唯一性只是针对类dict(即,提供类对象的名称和值之间的映射的内部对象)。您可以通过检查Company.__dict__来迭代这些。这包括所有内容 - 内置Python object的所有属性,属性和方法,同样适用于db.Model和任何其他父类。

您最好的选择是遍历类dict,检查每个项目以查看它是否是_ReverseReferenceProperty的实例:

for propname, prop in Company.__dict__.iteritems():
  if isinstance(prop, db._ReverseReferenceProperty):
    # Do something with prop, such as getting a query with getattr(a_company, propname).

答案 1 :(得分:0)

假设您没有为collection_name分配值,只需将其保留为Modelname_set即可。然后您可以dir(model),选择以_set结尾的属性,这些是后向引用。或者你可以制定一个约定来命名它们。 groups_refs,然后搜索以_refs结尾的属性。