mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Replaced no_spatialite by connection features
Refs #22632. Thanks Tim Graham for the review.
This commit is contained in:
		| @@ -11,8 +11,56 @@ from django.utils.encoding import python_2_unicode_compatible | |||||||
|  |  | ||||||
| class BaseSpatialFeatures(object): | class BaseSpatialFeatures(object): | ||||||
|     gis_enabled = True |     gis_enabled = True | ||||||
|  |  | ||||||
|  |     # Does the database contain a SpatialRefSys model to store SRID information? | ||||||
|     has_spatialrefsys_table = True |     has_spatialrefsys_table = True | ||||||
|  |  | ||||||
|  |     # Can the `distance` GeoQuerySet method be applied on geodetic coordinate systems? | ||||||
|  |     supports_distance_geodetic = True | ||||||
|  |     # Does the database supports `left` and `right` lookups? | ||||||
|  |     supports_left_right_lookups = False | ||||||
|  |     # Is the database able to count vertices on polygons (with `num_points`)? | ||||||
|  |     supports_num_points_poly = True | ||||||
|  |  | ||||||
|  |     # The following properties indicate if the database GIS extensions support | ||||||
|  |     # certain methods (dwithin, force_rhr, geohash, ...) | ||||||
|  |     @property | ||||||
|  |     def has_dwithin_lookup(self): | ||||||
|  |         return 'dwithin' in self.connection.ops.distance_functions | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def has_force_rhr_method(self): | ||||||
|  |         return bool(self.connection.ops.force_rhr) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def has_geohash_method(self): | ||||||
|  |         return bool(self.connection.ops.geohash) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def has_make_line_method(self): | ||||||
|  |         return bool(self.connection.ops.make_line) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def has_perimeter_method(self): | ||||||
|  |         return bool(self.connection.ops.perimeter) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def has_reverse_method(self): | ||||||
|  |         return bool(self.connection.ops.reverse) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def has_snap_to_grid_method(self): | ||||||
|  |         return bool(self.connection.ops.snap_to_grid) | ||||||
|  |  | ||||||
|  |     # Specifies whether the Collect and Extent aggregates are supported by the database | ||||||
|  |     @property | ||||||
|  |     def supports_collect_aggr(self): | ||||||
|  |         return 'Collect' in self.connection.ops.valid_aggregates | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def supports_extent_aggr(self): | ||||||
|  |         return 'Extent' in self.connection.ops.valid_aggregates | ||||||
|  |  | ||||||
|  |  | ||||||
| class BaseSpatialOperations(object): | class BaseSpatialOperations(object): | ||||||
|     """ |     """ | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ from django.contrib.gis.db.backends.mysql.operations import MySQLOperations | |||||||
|  |  | ||||||
| class DatabaseFeatures(BaseSpatialFeatures, MySQLDatabaseFeatures): | class DatabaseFeatures(BaseSpatialFeatures, MySQLDatabaseFeatures): | ||||||
|     has_spatialrefsys_table = False |     has_spatialrefsys_table = False | ||||||
|  |     supports_num_points_poly = False | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseWrapper(MySQLDatabaseWrapper): | class DatabaseWrapper(MySQLDatabaseWrapper): | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ from django.contrib.gis.db.backends.postgis.schema import PostGISSchemaEditor | |||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseFeatures(BaseSpatialFeatures, Psycopg2DatabaseFeatures): | class DatabaseFeatures(BaseSpatialFeatures, Psycopg2DatabaseFeatures): | ||||||
|     pass |     supports_left_right_lookups = True | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseWrapper(Psycopg2DatabaseWrapper): | class DatabaseWrapper(Psycopg2DatabaseWrapper): | ||||||
|   | |||||||
| @@ -16,7 +16,9 @@ from django.utils import six | |||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseFeatures(BaseSpatialFeatures, SQLiteDatabaseFeatures): | class DatabaseFeatures(BaseSpatialFeatures, SQLiteDatabaseFeatures): | ||||||
|     pass |     supports_distance_geodetic = False | ||||||
|  |     # SpatiaLite can only count vertices in LineStrings | ||||||
|  |     supports_num_points_poly = False | ||||||
|  |  | ||||||
|  |  | ||||||
| class DatabaseWrapper(SQLiteDatabaseWrapper): | class DatabaseWrapper(SQLiteDatabaseWrapper): | ||||||
|   | |||||||
| @@ -65,17 +65,25 @@ class SpatiaLiteOperations(DatabaseOperations, BaseSpatialOperations): | |||||||
|     name = 'spatialite' |     name = 'spatialite' | ||||||
|     spatialite = True |     spatialite = True | ||||||
|     version_regex = re.compile(r'^(?P<major>\d)\.(?P<minor1>\d)\.(?P<minor2>\d+)') |     version_regex = re.compile(r'^(?P<major>\d)\.(?P<minor1>\d)\.(?P<minor2>\d+)') | ||||||
|     valid_aggregates = {'Extent', 'Union'} |  | ||||||
|  |     @property | ||||||
|  |     def valid_aggregates(self): | ||||||
|  |         if self.spatial_version >= 3: | ||||||
|  |             return {'Collect', 'Extent', 'Union'} | ||||||
|  |         else: | ||||||
|  |             return {'Union'} | ||||||
|  |  | ||||||
|     Adapter = SpatiaLiteAdapter |     Adapter = SpatiaLiteAdapter | ||||||
|     Adaptor = Adapter  # Backwards-compatibility alias. |     Adaptor = Adapter  # Backwards-compatibility alias. | ||||||
|  |  | ||||||
|     area = 'Area' |     area = 'Area' | ||||||
|     centroid = 'Centroid' |     centroid = 'Centroid' | ||||||
|  |     collect = 'Collect' | ||||||
|     contained = 'MbrWithin' |     contained = 'MbrWithin' | ||||||
|     difference = 'Difference' |     difference = 'Difference' | ||||||
|     distance = 'Distance' |     distance = 'Distance' | ||||||
|     envelope = 'Envelope' |     envelope = 'Envelope' | ||||||
|  |     extent = 'Extent' | ||||||
|     intersection = 'Intersection' |     intersection = 'Intersection' | ||||||
|     length = 'GLength'  # OpenGis defines Length, but this conflicts with an SQLite reserved keyword |     length = 'GLength'  # OpenGis defines Length, but this conflicts with an SQLite reserved keyword | ||||||
|     num_geom = 'NumGeometries' |     num_geom = 'NumGeometries' | ||||||
| @@ -180,6 +188,15 @@ class SpatiaLiteOperations(DatabaseOperations, BaseSpatialOperations): | |||||||
|         agg_name = aggregate.__class__.__name__ |         agg_name = aggregate.__class__.__name__ | ||||||
|         return agg_name in self.valid_aggregates |         return agg_name in self.valid_aggregates | ||||||
|  |  | ||||||
|  |     def convert_extent(self, box): | ||||||
|  |         """ | ||||||
|  |         Convert the polygon data received from Spatialite to min/max values. | ||||||
|  |         """ | ||||||
|  |         shell = Geometry(box).shell | ||||||
|  |         xmin, ymin = shell[0][:2] | ||||||
|  |         xmax, ymax = shell[2][:2] | ||||||
|  |         return (xmin, ymin, xmax, ymax) | ||||||
|  |  | ||||||
|     def convert_geom(self, wkt, geo_field): |     def convert_geom(self, wkt, geo_field): | ||||||
|         """ |         """ | ||||||
|         Converts geometry WKT returned from a SpatiaLite aggregate. |         Converts geometry WKT returned from a SpatiaLite aggregate. | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ from django.db.models import Q | |||||||
| from django.contrib.gis.geos import HAS_GEOS | from django.contrib.gis.geos import HAS_GEOS | ||||||
| from django.contrib.gis.measure import D  # alias for Distance | from django.contrib.gis.measure import D  # alias for Distance | ||||||
| from django.contrib.gis.tests.utils import ( | from django.contrib.gis.tests.utils import ( | ||||||
|     mysql, oracle, postgis, spatialite, no_oracle, no_spatialite |     mysql, oracle, postgis, spatialite, no_oracle | ||||||
| ) | ) | ||||||
| from django.test import TestCase, skipUnlessDBFeature | from django.test import TestCase, skipUnlessDBFeature | ||||||
|  |  | ||||||
| @@ -50,7 +50,7 @@ class DistanceTest(TestCase): | |||||||
|         self.assertEqual(1, Interstate.objects.count()) |         self.assertEqual(1, Interstate.objects.count()) | ||||||
|         self.assertEqual(1, SouthTexasInterstate.objects.count()) |         self.assertEqual(1, SouthTexasInterstate.objects.count()) | ||||||
|  |  | ||||||
|     @no_spatialite |     @skipUnlessDBFeature("has_dwithin_lookup") | ||||||
|     def test_dwithin(self): |     def test_dwithin(self): | ||||||
|         """ |         """ | ||||||
|         Test the `dwithin` lookup type. |         Test the `dwithin` lookup type. | ||||||
| @@ -139,7 +139,7 @@ class DistanceTest(TestCase): | |||||||
|                 self.assertAlmostEqual(m_distances[i], c.distance.m, tol) |                 self.assertAlmostEqual(m_distances[i], c.distance.m, tol) | ||||||
|                 self.assertAlmostEqual(ft_distances[i], c.distance.survey_ft, tol) |                 self.assertAlmostEqual(ft_distances[i], c.distance.survey_ft, tol) | ||||||
|  |  | ||||||
|     @no_spatialite |     @skipUnlessDBFeature("supports_distance_geodetic") | ||||||
|     def test_distance_geodetic(self): |     def test_distance_geodetic(self): | ||||||
|         """ |         """ | ||||||
|         Test the `distance` GeoQuerySet method on geodetic coordinate systems. |         Test the `distance` GeoQuerySet method on geodetic coordinate systems. | ||||||
| @@ -354,7 +354,7 @@ class DistanceTest(TestCase): | |||||||
|         i10 = SouthTexasInterstate.objects.length().get(name='I-10') |         i10 = SouthTexasInterstate.objects.length().get(name='I-10') | ||||||
|         self.assertAlmostEqual(len_m2, i10.length.m, 2) |         self.assertAlmostEqual(len_m2, i10.length.m, 2) | ||||||
|  |  | ||||||
|     @no_spatialite |     @skipUnlessDBFeature("has_perimeter_method") | ||||||
|     def test_perimeter(self): |     def test_perimeter(self): | ||||||
|         """ |         """ | ||||||
|         Test the `perimeter` GeoQuerySet method. |         Test the `perimeter` GeoQuerySet method. | ||||||
|   | |||||||
| @@ -1,5 +1,5 @@ | |||||||
| from django.contrib.gis.db import models | from django.contrib.gis.db import models | ||||||
| from django.contrib.gis.tests.utils import mysql, spatialite | from django.contrib.gis.tests.utils import mysql | ||||||
| from django.utils.encoding import python_2_unicode_compatible | from django.utils.encoding import python_2_unicode_compatible | ||||||
|  |  | ||||||
| # MySQL spatial indices can't handle NULL geometries. | # MySQL spatial indices can't handle NULL geometries. | ||||||
| @@ -58,15 +58,14 @@ class Truth(models.Model): | |||||||
|         app_label = 'geoapp' |         app_label = 'geoapp' | ||||||
|  |  | ||||||
|  |  | ||||||
| if not spatialite: | class Feature(NamedModel): | ||||||
|  |     geom = models.GeometryField() | ||||||
|  |  | ||||||
|     class Feature(NamedModel): |  | ||||||
|         geom = models.GeometryField() |  | ||||||
|  |  | ||||||
|     class MinusOneSRID(models.Model): | class MinusOneSRID(models.Model): | ||||||
|         geom = models.PointField(srid=-1)  # Minus one SRID. |     geom = models.PointField(srid=-1)  # Minus one SRID. | ||||||
|  |  | ||||||
|         objects = models.GeoManager() |     objects = models.GeoManager() | ||||||
|  |  | ||||||
|         class Meta: |     class Meta: | ||||||
|             app_label = 'geoapp' |         app_label = 'geoapp' | ||||||
|   | |||||||
| @@ -4,7 +4,6 @@ from __future__ import unicode_literals | |||||||
| from datetime import datetime | from datetime import datetime | ||||||
|  |  | ||||||
| from django.contrib.gis.geos import HAS_GEOS | from django.contrib.gis.geos import HAS_GEOS | ||||||
| from django.contrib.gis.tests.utils import no_mysql, no_spatialite |  | ||||||
| from django.contrib.gis.shortcuts import render_to_kmz | from django.contrib.gis.shortcuts import render_to_kmz | ||||||
| from django.db.models import Count, Min | from django.db.models import Count, Min | ||||||
| from django.test import TestCase, skipUnlessDBFeature | from django.test import TestCase, skipUnlessDBFeature | ||||||
| @@ -39,8 +38,7 @@ class GeoRegressionTests(TestCase): | |||||||
|         }] |         }] | ||||||
|         render_to_kmz('gis/kml/placemarks.kml', {'places': places}) |         render_to_kmz('gis/kml/placemarks.kml', {'places': places}) | ||||||
|  |  | ||||||
|     @no_spatialite |     @skipUnlessDBFeature("supports_extent_aggr") | ||||||
|     @no_mysql |  | ||||||
|     def test_extent(self): |     def test_extent(self): | ||||||
|         "Testing `extent` on a table with a single point. See #11827." |         "Testing `extent` on a table with a single point. See #11827." | ||||||
|         pnt = City.objects.get(name='Pueblo').point |         pnt = City.objects.get(name='Pueblo').point | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ from django.db import connection | |||||||
| from django.contrib.gis import gdal | from django.contrib.gis import gdal | ||||||
| from django.contrib.gis.geos import HAS_GEOS | from django.contrib.gis.geos import HAS_GEOS | ||||||
| from django.contrib.gis.tests.utils import ( | from django.contrib.gis.tests.utils import ( | ||||||
|     no_mysql, no_oracle, no_spatialite, mysql, oracle, postgis, spatialite) |     no_mysql, no_oracle, mysql, oracle, postgis, spatialite) | ||||||
| from django.test import TestCase, skipUnlessDBFeature | from django.test import TestCase, skipUnlessDBFeature | ||||||
| from django.utils import six | from django.utils import six | ||||||
|  |  | ||||||
| @@ -17,8 +17,6 @@ if HAS_GEOS: | |||||||
|         Point, LineString, LinearRing, Polygon, GeometryCollection) |         Point, LineString, LinearRing, Polygon, GeometryCollection) | ||||||
|  |  | ||||||
|     from .models import Country, City, PennsylvaniaCity, State, Track |     from .models import Country, City, PennsylvaniaCity, State, Track | ||||||
|  |  | ||||||
| if HAS_GEOS and not spatialite: |  | ||||||
|     from .models import Feature, MinusOneSRID |     from .models import Feature, MinusOneSRID | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -156,7 +154,6 @@ class GeoModelTest(TestCase): | |||||||
|         c = City() |         c = City() | ||||||
|         self.assertEqual(c.point, None) |         self.assertEqual(c.point, None) | ||||||
|  |  | ||||||
|     @no_spatialite  # SpatiaLite does not support abstract geometry columns |  | ||||||
|     def test_geometryfield(self): |     def test_geometryfield(self): | ||||||
|         "Testing the general GeometryField." |         "Testing the general GeometryField." | ||||||
|         Feature(name='Point', geom=Point(1, 1)).save() |         Feature(name='Point', geom=Point(1, 1)).save() | ||||||
| @@ -266,9 +263,7 @@ class GeoLookupTest(TestCase): | |||||||
|             self.assertEqual('Texas', qs[0].name) |             self.assertEqual('Texas', qs[0].name) | ||||||
|  |  | ||||||
|     # Only PostGIS has `left` and `right` lookup types. |     # Only PostGIS has `left` and `right` lookup types. | ||||||
|     @no_mysql |     @skipUnlessDBFeature("supports_left_right_lookups") | ||||||
|     @no_oracle |  | ||||||
|     @no_spatialite |  | ||||||
|     def test_left_right_lookups(self): |     def test_left_right_lookups(self): | ||||||
|         "Testing the 'left' and 'right' lookup types." |         "Testing the 'left' and 'right' lookup types." | ||||||
|         # Left: A << B => true if xmax(A) < xmin(B) |         # Left: A << B => true if xmax(A) < xmin(B) | ||||||
| @@ -451,8 +446,7 @@ class GeoQuerySetTest(TestCase): | |||||||
|         for country in countries: |         for country in countries: | ||||||
|             self.assertIsInstance(country.envelope, Polygon) |             self.assertIsInstance(country.envelope, Polygon) | ||||||
|  |  | ||||||
|     @no_mysql |     @skipUnlessDBFeature("supports_extent_aggr") | ||||||
|     @no_spatialite  # SpatiaLite does not have an Extent function |  | ||||||
|     def test_extent(self): |     def test_extent(self): | ||||||
|         "Testing the `extent` GeoQuerySet method." |         "Testing the `extent` GeoQuerySet method." | ||||||
|         # Reference query: |         # Reference query: | ||||||
| @@ -466,9 +460,7 @@ class GeoQuerySetTest(TestCase): | |||||||
|         for val, exp in zip(extent, expected): |         for val, exp in zip(extent, expected): | ||||||
|             self.assertAlmostEqual(exp, val, 4) |             self.assertAlmostEqual(exp, val, 4) | ||||||
|  |  | ||||||
|     @no_mysql |     @skipUnlessDBFeature("has_force_rhr_method") | ||||||
|     @no_oracle |  | ||||||
|     @no_spatialite |  | ||||||
|     def test_force_rhr(self): |     def test_force_rhr(self): | ||||||
|         "Testing GeoQuerySet.force_rhr()." |         "Testing GeoQuerySet.force_rhr()." | ||||||
|         rings = ( |         rings = ( | ||||||
| @@ -483,13 +475,9 @@ class GeoQuerySetTest(TestCase): | |||||||
|         s = State.objects.force_rhr().get(name='Foo') |         s = State.objects.force_rhr().get(name='Foo') | ||||||
|         self.assertEqual(rhr_rings, s.force_rhr.coords) |         self.assertEqual(rhr_rings, s.force_rhr.coords) | ||||||
|  |  | ||||||
|     @no_mysql |     @skipUnlessDBFeature("has_geohash_method") | ||||||
|     @no_oracle |  | ||||||
|     @no_spatialite |  | ||||||
|     def test_geohash(self): |     def test_geohash(self): | ||||||
|         "Testing GeoQuerySet.geohash()." |         "Testing GeoQuerySet.geohash()." | ||||||
|         if not connection.ops.geohash: |  | ||||||
|             return |  | ||||||
|         # Reference query: |         # Reference query: | ||||||
|         # SELECT ST_GeoHash(point) FROM geoapp_city WHERE name='Houston'; |         # SELECT ST_GeoHash(point) FROM geoapp_city WHERE name='Houston'; | ||||||
|         # SELECT ST_GeoHash(point, 5) FROM geoapp_city WHERE name='Houston'; |         # SELECT ST_GeoHash(point, 5) FROM geoapp_city WHERE name='Houston'; | ||||||
| @@ -501,7 +489,7 @@ class GeoQuerySetTest(TestCase): | |||||||
|  |  | ||||||
|     def test_geojson(self): |     def test_geojson(self): | ||||||
|         "Testing GeoJSON output from the database using GeoQuerySet.geojson()." |         "Testing GeoJSON output from the database using GeoQuerySet.geojson()." | ||||||
|         # Only PostGIS 1.3.4+ and SpatiaLite 3.0+ support GeoJSON. |         # Only PostGIS and SpatiaLite 3.0+ support GeoJSON. | ||||||
|         if not connection.ops.geojson: |         if not connection.ops.geojson: | ||||||
|             self.assertRaises(NotImplementedError, Country.objects.all().geojson, field_name='mpoly') |             self.assertRaises(NotImplementedError, Country.objects.all().geojson, field_name='mpoly') | ||||||
|             return |             return | ||||||
| @@ -520,17 +508,15 @@ class GeoQuerySetTest(TestCase): | |||||||
|         # SELECT ST_AsGeoJson("geoapp_city"."point", 8, 0) FROM "geoapp_city" WHERE "geoapp_city"."name" = 'Pueblo'; |         # SELECT ST_AsGeoJson("geoapp_city"."point", 8, 0) FROM "geoapp_city" WHERE "geoapp_city"."name" = 'Pueblo'; | ||||||
|         self.assertEqual(pueblo_json, City.objects.geojson().get(name='Pueblo').geojson) |         self.assertEqual(pueblo_json, City.objects.geojson().get(name='Pueblo').geojson) | ||||||
|  |  | ||||||
|         # 1.3.x: SELECT ST_AsGeoJson("geoapp_city"."point", 8, 1) FROM "geoapp_city" WHERE "geoapp_city"."name" = 'Houston'; |         # SELECT ST_AsGeoJson("geoapp_city"."point", 8, 2) FROM "geoapp_city" WHERE "geoapp_city"."name" = 'Houston'; | ||||||
|         # 1.4.x: SELECT ST_AsGeoJson("geoapp_city"."point", 8, 2) FROM "geoapp_city" WHERE "geoapp_city"."name" = 'Houston'; |  | ||||||
|         # This time we want to include the CRS by using the `crs` keyword. |         # This time we want to include the CRS by using the `crs` keyword. | ||||||
|         self.assertEqual(houston_json, City.objects.geojson(crs=True, model_att='json').get(name='Houston').json) |         self.assertEqual(houston_json, City.objects.geojson(crs=True, model_att='json').get(name='Houston').json) | ||||||
|  |  | ||||||
|         # 1.3.x: SELECT ST_AsGeoJson("geoapp_city"."point", 8, 2) FROM "geoapp_city" WHERE "geoapp_city"."name" = 'Victoria'; |         # SELECT ST_AsGeoJson("geoapp_city"."point", 8, 1) FROM "geoapp_city" WHERE "geoapp_city"."name" = 'Houston'; | ||||||
|         # 1.4.x: SELECT ST_AsGeoJson("geoapp_city"."point", 8, 1) FROM "geoapp_city" WHERE "geoapp_city"."name" = 'Houston'; |  | ||||||
|         # This time we include the bounding box by using the `bbox` keyword. |         # This time we include the bounding box by using the `bbox` keyword. | ||||||
|         self.assertEqual(victoria_json, City.objects.geojson(bbox=True).get(name='Victoria').geojson) |         self.assertEqual(victoria_json, City.objects.geojson(bbox=True).get(name='Victoria').geojson) | ||||||
|  |  | ||||||
|         # 1.(3|4).x: SELECT ST_AsGeoJson("geoapp_city"."point", 5, 3) FROM "geoapp_city" WHERE "geoapp_city"."name" = 'Chicago'; |         # SELECT ST_AsGeoJson("geoapp_city"."point", 5, 3) FROM "geoapp_city" WHERE "geoapp_city"."name" = 'Chicago'; | ||||||
|         # Finally, we set every available keyword. |         # Finally, we set every available keyword. | ||||||
|         self.assertEqual(chicago_json, City.objects.geojson(bbox=True, crs=True, precision=5).get(name='Chicago').geojson) |         self.assertEqual(chicago_json, City.objects.geojson(bbox=True, crs=True, precision=5).get(name='Chicago').geojson) | ||||||
|  |  | ||||||
| @@ -581,9 +567,7 @@ class GeoQuerySetTest(TestCase): | |||||||
|             self.assertEqual('<Point><coordinates>-104.609252,38.255001</coordinates></Point>', ptown.kml) |             self.assertEqual('<Point><coordinates>-104.609252,38.255001</coordinates></Point>', ptown.kml) | ||||||
|  |  | ||||||
|     # Only PostGIS has support for the MakeLine aggregate. |     # Only PostGIS has support for the MakeLine aggregate. | ||||||
|     @no_mysql |     @skipUnlessDBFeature("has_make_line_method") | ||||||
|     @no_oracle |  | ||||||
|     @no_spatialite |  | ||||||
|     def test_make_line(self): |     def test_make_line(self): | ||||||
|         "Testing the `make_line` GeoQuerySet method." |         "Testing the `make_line` GeoQuerySet method." | ||||||
|         # Ensuring that a `TypeError` is raised on models without PointFields. |         # Ensuring that a `TypeError` is raised on models without PointFields. | ||||||
| @@ -610,8 +594,7 @@ class GeoQuerySetTest(TestCase): | |||||||
|             else: |             else: | ||||||
|                 self.assertEqual(1, c.num_geom) |                 self.assertEqual(1, c.num_geom) | ||||||
|  |  | ||||||
|     @no_mysql |     @skipUnlessDBFeature("supports_num_points_poly") | ||||||
|     @no_spatialite  # SpatiaLite can only count vertices in LineStrings |  | ||||||
|     def test_num_points(self): |     def test_num_points(self): | ||||||
|         "Testing the `num_points` GeoQuerySet method." |         "Testing the `num_points` GeoQuerySet method." | ||||||
|         for c in Country.objects.num_points(): |         for c in Country.objects.num_points(): | ||||||
| @@ -647,8 +630,7 @@ class GeoQuerySetTest(TestCase): | |||||||
|                 tol = 0.000000001 |                 tol = 0.000000001 | ||||||
|             self.assertEqual(True, ref[c.name].equals_exact(c.point_on_surface, tol)) |             self.assertEqual(True, ref[c.name].equals_exact(c.point_on_surface, tol)) | ||||||
|  |  | ||||||
|     @no_mysql |     @skipUnlessDBFeature("has_reverse_method") | ||||||
|     @no_spatialite |  | ||||||
|     def test_reverse_geom(self): |     def test_reverse_geom(self): | ||||||
|         "Testing GeoQuerySet.reverse_geom()." |         "Testing GeoQuerySet.reverse_geom()." | ||||||
|         coords = [(-95.363151, 29.763374), (-95.448601, 29.713803)] |         coords = [(-95.363151, 29.763374), (-95.448601, 29.713803)] | ||||||
| @@ -673,9 +655,7 @@ class GeoQuerySetTest(TestCase): | |||||||
|                         self.assertAlmostEqual(c1[0] * xfac, c2[0], tol) |                         self.assertAlmostEqual(c1[0] * xfac, c2[0], tol) | ||||||
|                         self.assertAlmostEqual(c1[1] * yfac, c2[1], tol) |                         self.assertAlmostEqual(c1[1] * yfac, c2[1], tol) | ||||||
|  |  | ||||||
|     @no_mysql |     @skipUnlessDBFeature("has_snap_to_grid_method") | ||||||
|     @no_oracle |  | ||||||
|     @no_spatialite |  | ||||||
|     def test_snap_to_grid(self): |     def test_snap_to_grid(self): | ||||||
|         "Testing GeoQuerySet.snap_to_grid()." |         "Testing GeoQuerySet.snap_to_grid()." | ||||||
|         # Let's try and break snap_to_grid() with bad combinations of arguments. |         # Let's try and break snap_to_grid() with bad combinations of arguments. | ||||||
|   | |||||||
| @@ -1,7 +1,7 @@ | |||||||
| from __future__ import unicode_literals | from __future__ import unicode_literals | ||||||
|  |  | ||||||
| from django.contrib.gis.geos import HAS_GEOS | from django.contrib.gis.geos import HAS_GEOS | ||||||
| from django.contrib.gis.tests.utils import mysql, no_mysql, no_oracle, no_spatialite | from django.contrib.gis.tests.utils import mysql, no_mysql, no_oracle | ||||||
| from django.test import TestCase, skipUnlessDBFeature | from django.test import TestCase, skipUnlessDBFeature | ||||||
|  |  | ||||||
| if HAS_GEOS: | if HAS_GEOS: | ||||||
| @@ -60,8 +60,7 @@ class RelatedGeoModelTest(TestCase): | |||||||
|             qs = list(City.objects.filter(name=name).transform(srid, field_name='location__point')) |             qs = list(City.objects.filter(name=name).transform(srid, field_name='location__point')) | ||||||
|             check_pnt(GEOSGeometry(wkt, srid), qs[0].location.point) |             check_pnt(GEOSGeometry(wkt, srid), qs[0].location.point) | ||||||
|  |  | ||||||
|     @no_mysql |     @skipUnlessDBFeature("supports_extent_aggr") | ||||||
|     @no_spatialite |  | ||||||
|     def test04a_related_extent_aggregate(self): |     def test04a_related_extent_aggregate(self): | ||||||
|         "Testing the `extent` GeoQuerySet aggregates on related geographic models." |         "Testing the `extent` GeoQuerySet aggregates on related geographic models." | ||||||
|         # This combines the Extent and Union aggregates into one query |         # This combines the Extent and Union aggregates into one query | ||||||
| @@ -265,9 +264,7 @@ class RelatedGeoModelTest(TestCase): | |||||||
|         # Should be `None`, and not a 'dummy' model. |         # Should be `None`, and not a 'dummy' model. | ||||||
|         self.assertEqual(None, b.author) |         self.assertEqual(None, b.author) | ||||||
|  |  | ||||||
|     @no_mysql |     @skipUnlessDBFeature("supports_collect_aggr") | ||||||
|     @no_oracle |  | ||||||
|     @no_spatialite |  | ||||||
|     def test14_collect(self): |     def test14_collect(self): | ||||||
|         "Testing the `collect` GeoQuerySet method and `Collect` aggregate." |         "Testing the `collect` GeoQuerySet method and `Collect` aggregate." | ||||||
|         # Reference query: |         # Reference query: | ||||||
|   | |||||||
| @@ -29,9 +29,6 @@ def no_mysql(func): | |||||||
|     return no_backend(func, 'mysql') |     return no_backend(func, 'mysql') | ||||||
|  |  | ||||||
|  |  | ||||||
| def no_spatialite(func): |  | ||||||
|     return no_backend(func, 'spatialite') |  | ||||||
|  |  | ||||||
| # Shortcut booleans to omit only portions of tests. | # Shortcut booleans to omit only portions of tests. | ||||||
| _default_db = settings.DATABASES[DEFAULT_DB_ALIAS]['ENGINE'].rsplit('.')[-1] | _default_db = settings.DATABASES[DEFAULT_DB_ALIAS]['ENGINE'].rsplit('.')[-1] | ||||||
| oracle = _default_db == 'oracle' | oracle = _default_db == 'oracle' | ||||||
|   | |||||||
| @@ -270,11 +270,11 @@ Method                                PostGIS  Oracle  SpatiaLite | |||||||
| ====================================  =======  ======  ========== | ====================================  =======  ======  ========== | ||||||
| :meth:`GeoQuerySet.area`              X        X       X | :meth:`GeoQuerySet.area`              X        X       X | ||||||
| :meth:`GeoQuerySet.centroid`          X        X       X | :meth:`GeoQuerySet.centroid`          X        X       X | ||||||
| :meth:`GeoQuerySet.collect`           X | :meth:`GeoQuerySet.collect`           X                (from v3.0) | ||||||
| :meth:`GeoQuerySet.difference`        X        X       X | :meth:`GeoQuerySet.difference`        X        X       X | ||||||
| :meth:`GeoQuerySet.distance`          X        X       X | :meth:`GeoQuerySet.distance`          X        X       X | ||||||
| :meth:`GeoQuerySet.envelope`          X                X | :meth:`GeoQuerySet.envelope`          X                X | ||||||
| :meth:`GeoQuerySet.extent`            X        X | :meth:`GeoQuerySet.extent`            X        X       (from v3.0) | ||||||
| :meth:`GeoQuerySet.extent3d`          X | :meth:`GeoQuerySet.extent3d`          X | ||||||
| :meth:`GeoQuerySet.force_rhr`         X | :meth:`GeoQuerySet.force_rhr`         X | ||||||
| :meth:`GeoQuerySet.geohash`           X | :meth:`GeoQuerySet.geohash`           X | ||||||
|   | |||||||
| @@ -83,6 +83,9 @@ Minor features | |||||||
| * Compatibility shims for ``SpatialRefSys`` and ``GeometryColumns`` changed in | * Compatibility shims for ``SpatialRefSys`` and ``GeometryColumns`` changed in | ||||||
|   Django 1.2 have been removed. |   Django 1.2 have been removed. | ||||||
|  |  | ||||||
|  | * The Spatialite backend now supports ``Collect`` and ``Extent`` aggregates | ||||||
|  |   when the database version is 3.0 or later. | ||||||
|  |  | ||||||
| :mod:`django.contrib.messages` | :mod:`django.contrib.messages` | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user