3

我有一个 GeoDjango 项目,它有一个像这样的管理器模型;

class AdvertManager(models.GeoManager):

    def within_box(self, x0, y0, x1, y1):
        geometry = Polygon.from_bbox((x0, y0, x1, y1,))
        return self.filter(point__within=geometry)

我试图让我的资源模型(AdvertResource)通过 GET 参数公开 inside_box 函数,例如;

http://127.0.0.1:8000/api/v1/advert/?format=json&box=51.623349,-3.25362,51.514195,-3.4754133

我开始build_filters像这样在资源模型上写一个方法;

def build_filters(self, filters=None):
        if not filters:
            filters = {}
        orm_filters = super(AdvertResource, self).build_filters(filters)

        if 'box' in filters:
            points = [float(p.strip()) for p in filters['box'].split(',')]
            orm_filters = {'box': Advert.objects.within_box(*points).all()}

        return orm_filters

但这会引发错误“无法将关键字'box'解析为字段......”。

是否可以将自定义管理器中的方法公开给 api url?

编辑 - 我现在已经用以下解决方案解决了这个问题。

class AdvertResource(ModelResource):

    longitude = fields.FloatField(attribute='longitude', default=0.0)
    latitude = fields.FloatField(attribute='latitude', default=0.0)
    author = fields.ForeignKey(UserResource, 'author')

    def build_filters(self, filters=None):
        """
        Build additional filters
        """
        if not filters:
            filters = {}
        orm_filters = super(AdvertResource, self).build_filters(filters)

        if 'point__within_box' in filters:
            points = filters['point__within_box']
            points = [float(p.strip()) for p in points.split(',')]
            orm_filters['within_box'] = points

        return orm_filters

    def apply_filters(self, request, applicable_filters):
        """
        Apply the filters
        """
        if 'within_box' in applicable_filters:
            area = applicable_filters.pop('within_box')
            poly = Polygon.from_bbox(area)
            applicable_filters['point__within'] = poly
        return super(AdvertResource, self).apply_filters(request, 
                                                        applicable_filters)

现在这意味着请求http://127.0.0.1:8000/api/v1/advert/?format=json&point__within_box=51.623349,-3.25362,51.514195,-3.4754133现在过滤边界框中的所有结果。

4

1 回答 1

4

上面的代码有几个问题。

首先,是的,您可以将任何自定义管理器暴露给任何东西,无论您是否使用它。仅当您将其默认管理器替换为您自己的版本时,您上面定义的 AdvertManager 才能通过 Advert.objects 访问。

其次,您希望在 AdvertResource 上公开美味派过滤的方式与过滤的实际工作方式是正交的。

所有过滤器都以实际的形式应用为 ORM 过滤器<field_name>__<filter_name>=<value_or_values>。由于在您的示例中,您使用的是box=<number>,<number>,...,<number>sweetpie 将其爆炸box__exact=...并尝试box在 AdvertResource 中查找字段并按预期失败。

如果您的广告有一个名为的字段,location您可以添加withinbox作为该字段的过滤器并过滤:location__withinbox=<values>

如果您想保留原来的方法,您必须自己从 request.GET 字典中解析框过滤器,然后将它们传递给您自己的覆盖版本obj_getobj_get_listAdvertResource 中。

最后,在扩展时,build_filters您只是在 Tastypie 过滤器和 ORM 过滤器之间进行映射。在您的示例中,您将对象作为过滤器返回;而是简单地将其定义为:

{ 'withinbox' : 'point__within' }

并将值列表转换为 inside,Polygon然后apply_filters再将其传递给实际的过滤器方法。

于 2012-07-23T05:28:10.867 回答