2014-11-08 17:08:12 +01:00
|
|
|
import json
|
|
|
|
|
2015-04-24 17:24:07 +02:00
|
|
|
from django.contrib.gis.geos import LinearRing, Point, Polygon
|
2014-11-08 17:08:12 +01:00
|
|
|
from django.core import serializers
|
2017-05-04 10:14:35 -04:00
|
|
|
from django.test import TestCase
|
2014-11-08 17:08:12 +01:00
|
|
|
|
2015-04-24 17:24:07 +02:00
|
|
|
from .models import City, MultiFields, PennsylvaniaCity
|
2014-11-08 17:08:12 +01:00
|
|
|
|
|
|
|
|
|
|
|
class GeoJSONSerializerTests(TestCase):
|
2022-02-03 20:24:19 +01:00
|
|
|
fixtures = ["initial"]
|
2014-11-08 17:08:12 +01:00
|
|
|
|
|
|
|
def test_builtin_serializers(self):
|
|
|
|
"""
|
|
|
|
'geojson' should be listed in available serializers.
|
|
|
|
"""
|
|
|
|
all_formats = set(serializers.get_serializer_formats())
|
|
|
|
public_formats = set(serializers.get_public_serializer_formats())
|
|
|
|
|
2023-08-22 12:42:57 +02:00
|
|
|
self.assertIn("geojson", all_formats)
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertIn("geojson", public_formats)
|
2014-11-08 17:08:12 +01:00
|
|
|
|
|
|
|
def test_serialization_base(self):
|
2022-02-22 09:29:38 +00:00
|
|
|
geojson = serializers.serialize("geojson", City.objects.order_by("name"))
|
2016-06-28 11:21:26 -04:00
|
|
|
geodata = json.loads(geojson)
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(len(geodata["features"]), len(City.objects.all()))
|
|
|
|
self.assertEqual(geodata["features"][0]["geometry"]["type"], "Point")
|
|
|
|
self.assertEqual(geodata["features"][0]["properties"]["name"], "Chicago")
|
2022-02-22 09:29:38 +00:00
|
|
|
first_city = City.objects.order_by("name").first()
|
2022-05-28 12:33:15 +03:00
|
|
|
self.assertEqual(geodata["features"][0]["id"], first_city.pk)
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(geodata["features"][0]["properties"]["pk"], str(first_city.pk))
|
2014-11-08 17:08:12 +01:00
|
|
|
|
|
|
|
def test_geometry_field_option(self):
|
|
|
|
"""
|
|
|
|
When a model has several geometry fields, the 'geometry_field' option
|
|
|
|
can be used to specify the field to use as the 'geometry' key.
|
|
|
|
"""
|
|
|
|
MultiFields.objects.create(
|
2022-02-03 20:24:19 +01:00
|
|
|
city=City.objects.first(),
|
|
|
|
name="Name",
|
|
|
|
point=Point(5, 23),
|
|
|
|
poly=Polygon(LinearRing((0, 0), (0, 5), (5, 5), (5, 0), (0, 0))),
|
|
|
|
)
|
2014-11-08 17:08:12 +01:00
|
|
|
|
2022-02-03 20:24:19 +01:00
|
|
|
geojson = serializers.serialize("geojson", MultiFields.objects.all())
|
2014-11-08 17:08:12 +01:00
|
|
|
geodata = json.loads(geojson)
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(geodata["features"][0]["geometry"]["type"], "Point")
|
2014-11-08 17:08:12 +01:00
|
|
|
|
2016-01-26 11:06:29 +01:00
|
|
|
geojson = serializers.serialize(
|
2022-02-03 20:24:19 +01:00
|
|
|
"geojson", MultiFields.objects.all(), geometry_field="poly"
|
2016-01-26 11:06:29 +01:00
|
|
|
)
|
|
|
|
geodata = json.loads(geojson)
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(geodata["features"][0]["geometry"]["type"], "Polygon")
|
2016-01-26 11:06:29 +01:00
|
|
|
|
|
|
|
# geometry_field is considered even if not in fields (#26138).
|
|
|
|
geojson = serializers.serialize(
|
2022-02-03 20:24:19 +01:00
|
|
|
"geojson",
|
2016-01-26 11:06:29 +01:00
|
|
|
MultiFields.objects.all(),
|
2022-02-03 20:24:19 +01:00
|
|
|
geometry_field="poly",
|
|
|
|
fields=("city",),
|
2016-01-26 11:06:29 +01:00
|
|
|
)
|
2014-11-08 17:08:12 +01:00
|
|
|
geodata = json.loads(geojson)
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertEqual(geodata["features"][0]["geometry"]["type"], "Polygon")
|
2014-11-08 17:08:12 +01:00
|
|
|
|
2022-05-28 12:33:15 +03:00
|
|
|
def test_id_field_option(self):
|
|
|
|
"""
|
|
|
|
By default Django uses the pk of the object as the id for a feature.
|
|
|
|
The 'id_field' option can be used to specify a different field to use
|
|
|
|
as the id.
|
|
|
|
"""
|
|
|
|
cities = City.objects.order_by("name")
|
|
|
|
geojson = serializers.serialize("geojson", cities, id_field="name")
|
|
|
|
geodata = json.loads(geojson)
|
|
|
|
self.assertEqual(geodata["features"][0]["id"], cities[0].name)
|
|
|
|
|
2014-11-08 17:08:12 +01:00
|
|
|
def test_fields_option(self):
|
|
|
|
"""
|
|
|
|
The fields option allows to define a subset of fields to be present in
|
|
|
|
the 'properties' of the generated output.
|
|
|
|
"""
|
2022-02-03 20:24:19 +01:00
|
|
|
PennsylvaniaCity.objects.create(
|
|
|
|
name="Mansfield", county="Tioga", point="POINT(-77.071445 41.823881)"
|
|
|
|
)
|
2016-04-07 22:04:45 -04:00
|
|
|
geojson = serializers.serialize(
|
2022-02-03 20:24:19 +01:00
|
|
|
"geojson",
|
|
|
|
PennsylvaniaCity.objects.all(),
|
|
|
|
fields=("county", "point"),
|
2016-04-07 22:04:45 -04:00
|
|
|
)
|
2014-11-08 17:08:12 +01:00
|
|
|
geodata = json.loads(geojson)
|
2022-02-03 20:24:19 +01:00
|
|
|
self.assertIn("county", geodata["features"][0]["properties"])
|
|
|
|
self.assertNotIn("founded", geodata["features"][0]["properties"])
|
|
|
|
self.assertNotIn("pk", geodata["features"][0]["properties"])
|
2014-11-08 17:08:12 +01:00
|
|
|
|
|
|
|
def test_srid_option(self):
|
2022-02-03 20:24:19 +01:00
|
|
|
geojson = serializers.serialize(
|
2022-02-22 09:29:38 +00:00
|
|
|
"geojson", City.objects.order_by("name"), srid=2847
|
2022-02-03 20:24:19 +01:00
|
|
|
)
|
2014-11-08 17:08:12 +01:00
|
|
|
geodata = json.loads(geojson)
|
2022-02-03 20:24:19 +01:00
|
|
|
coordinates = geodata["features"][0]["geometry"]["coordinates"]
|
2021-03-23 09:16:33 +01:00
|
|
|
# Different PROJ versions use different transformations, all are
|
|
|
|
# correct as having a 1 meter accuracy.
|
|
|
|
self.assertAlmostEqual(coordinates[0], 1564802, -1)
|
|
|
|
self.assertAlmostEqual(coordinates[1], 5613214, -1)
|
2014-11-08 17:08:12 +01:00
|
|
|
|
|
|
|
def test_deserialization_exception(self):
|
|
|
|
"""
|
|
|
|
GeoJSON cannot be deserialized.
|
|
|
|
"""
|
|
|
|
with self.assertRaises(serializers.base.SerializerDoesNotExist):
|
2022-02-03 20:24:19 +01:00
|
|
|
serializers.deserialize("geojson", "{}")
|