Django Rest Framwork为什么不显示发布的纬度?

时间:2019-11-05 16:06:34

标签: python django serialization django-rest-framework

我有一个Django代码库,它使用Django Rest Framework (DRF)作为API,该API将数据库的内容提供给前端。

我现在还希望用户能够通过此API发布新内容。我有可以接受POST的API,但是遇到gis Pointfield问题。

在我的一个模型中,我具有以下字段:

from django.contrib.gis.db import models as gis_models

class Device(models.Model):
    geometry = gis_models.PointField(name='geometry', null=True, blank=True)
    # And some other fields

我让人们在两个单独的字段(latitudelongitude)中发布几何图形,然后将其写入geometry字段中。我有一个如下所示的DeviceSerializer:

类DeviceSerializer(HALSerializer):     类型= TypeSerializer(很多=真)     经度= serializers.SerializerMethodField()     纬度= serializers.SerializerMethodField()     组织= serializers.SerializerMethodField()

class Meta:
    model = Device
    fields = (
        '_links',
        'id',
        'types',
        'longitude',
        'latitude',
        'organisation'
    )

def get_organisation(self, obj):
    if obj.owner:
        return obj.owner.organisation
    return 'Unknown'

def get_longitude(self, obj):
    if obj.geometry:
        return obj.geometry.x

def get_latitude(self, obj):
    if obj.geometry:
        return obj.geometry.y

def create(self, validated_data):
    print(validated_data)  # IN HERE I DON'T RECEIVE THE POSTED LATITUDE AND LONGITUDE

    types_data = validated_data.pop('types')
    device = Device.objects.create(**validated_data)
    for type_data in types_data:
        Type.objects.create(device=device, **type_data)

    if 'longitude' in validated_data and 'latitude' in validated_data:
        # WE NEVER REACH THIS POINT BECAUSE THE LAT AND LONG ARE NEVER POSTED
        device.geometry = Point(validated_data.pop('longitude'), validated_data.pop('latitude'))

    return device

然后在测试中,我这样做:

response = self.client.post(
    url,
    data={'types': [], 'latitude': 1, 'longitude': 1, 'organisation': 'abc'},
    format='json'
)

但是latitudelongitude从未出现在print(validated_data)中。有人知道他们为什么在那里失踪吗?

1 个答案:

答案 0 :(得分:1)

编辑

下面的解决方案并不像其他许多解决方案那么干净,例如此答案How I can create a PointField using an api?中给出的解决方案。我建议您使用此工具,但如有需要,将其留在下面供其他人使用


问题是您要在数据中发送latitudelongitude,但是模型只有一个称为几何的字段。我看到您正在尝试使用经过验证的数据进行设置,但是经过验证的数据是针对模型字段进行检查的数据,因此您将无法在此处获得它。解决此问题的一种方法是在视图中添加上下文。

如果您的观点是这样的:

class MyView():
    def create(self, request, *args, **kwargs):
        serializer = self.serializer_class(
            data=request.data,
            context={
                "latitude": request.data.get('latitude')
                "longitude": request.data.get('longitude')
            },
        )
        if serializer.is_valid:
            serilazer.save()

然后在序列化程序中,您可以使用{p>覆盖validate函数

def validate(self, attrs):
    latitude = self.context.get("latitude")
    longitude = self.context.get("longitude")
    attrs.update(
        {
            "geometry": Point(latitude, longitude)
        }
    )
    return super().validate(attrs)