123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207 |
- from django.contrib.gis.db.models import GeometryField
- from django.contrib.gis.db.models.functions import Distance
- from django.contrib.gis.measure import Area as AreaMeasure
- from django.contrib.gis.measure import Distance as DistanceMeasure
- from django.db import NotSupportedError
- from django.utils.functional import cached_property
- class BaseSpatialOperations:
-
-
- postgis = False
- spatialite = False
- mariadb = False
- mysql = False
- oracle = False
- spatial_version = None
-
- select = "%s"
- @cached_property
- def select_extent(self):
- return self.select
-
- disallowed_aggregates = ()
- geom_func_prefix = ""
-
-
- function_names = {}
-
- unsupported_functions = {
- "Area",
- "AsGeoJSON",
- "AsGML",
- "AsKML",
- "AsSVG",
- "Azimuth",
- "BoundingCircle",
- "Centroid",
- "Difference",
- "Distance",
- "Envelope",
- "GeoHash",
- "GeometryDistance",
- "Intersection",
- "IsEmpty",
- "IsValid",
- "Length",
- "LineLocatePoint",
- "MakeValid",
- "MemSize",
- "NumGeometries",
- "NumPoints",
- "Perimeter",
- "PointOnSurface",
- "Reverse",
- "Scale",
- "SnapToGrid",
- "SymDifference",
- "Transform",
- "Translate",
- "Union",
- }
-
- from_text = False
-
-
- def convert_extent(self, box, srid):
- raise NotImplementedError(
- "Aggregate extent not implemented for this spatial backend."
- )
- def convert_extent3d(self, box, srid):
- raise NotImplementedError(
- "Aggregate 3D extent not implemented for this spatial backend."
- )
-
- def geo_quote_name(self, name):
- return "'%s'" % name
-
- def geo_db_type(self, f):
- """
- Return the database column type for the geometry field on
- the spatial backend.
- """
- raise NotImplementedError(
- "subclasses of BaseSpatialOperations must provide a geo_db_type() method"
- )
- def get_distance(self, f, value, lookup_type):
- """
- Return the distance parameters for the given geometry field,
- lookup value, and lookup type.
- """
- raise NotImplementedError(
- "Distance operations not available on this spatial backend."
- )
- def get_geom_placeholder(self, f, value, compiler):
- """
- Return the placeholder for the given geometry field with the given
- value. Depending on the spatial backend, the placeholder may contain a
- stored procedure call to the transformation function of the spatial
- backend.
- """
- def transform_value(value, field):
- return value is not None and value.srid != field.srid
- if hasattr(value, "as_sql"):
- return (
- "%s(%%s, %s)" % (self.spatial_function_name("Transform"), f.srid)
- if transform_value(value.output_field, f)
- else "%s"
- )
- if transform_value(value, f):
-
- return "%s(%s(%%s,%s), %s)" % (
- self.spatial_function_name("Transform"),
- self.from_text,
- value.srid,
- f.srid,
- )
- elif self.connection.features.has_spatialrefsys_table:
- return "%s(%%s,%s)" % (self.from_text, f.srid)
- else:
-
- return "%s(%%s)" % self.from_text
- def check_expression_support(self, expression):
- if isinstance(expression, self.disallowed_aggregates):
- raise NotSupportedError(
- "%s spatial aggregation is not supported by this database backend."
- % expression.name
- )
- super().check_expression_support(expression)
- def spatial_aggregate_name(self, agg_name):
- raise NotImplementedError(
- "Aggregate support not implemented for this spatial backend."
- )
- def spatial_function_name(self, func_name):
- if func_name in self.unsupported_functions:
- raise NotSupportedError(
- "This backend doesn't support the %s function." % func_name
- )
- return self.function_names.get(func_name, self.geom_func_prefix + func_name)
-
- def geometry_columns(self):
- raise NotImplementedError(
- "Subclasses of BaseSpatialOperations must provide a geometry_columns() "
- "method."
- )
- def spatial_ref_sys(self):
- raise NotImplementedError(
- "subclasses of BaseSpatialOperations must a provide spatial_ref_sys() "
- "method"
- )
- distance_expr_for_lookup = staticmethod(Distance)
- def get_db_converters(self, expression):
- converters = super().get_db_converters(expression)
- if isinstance(expression.output_field, GeometryField):
- converters.append(self.get_geometry_converter(expression))
- return converters
- def get_geometry_converter(self, expression):
- raise NotImplementedError(
- "Subclasses of BaseSpatialOperations must provide a "
- "get_geometry_converter() method."
- )
- def get_area_att_for_field(self, field):
- if field.geodetic(self.connection):
- if self.connection.features.supports_area_geodetic:
- return "sq_m"
- raise NotImplementedError(
- "Area on geodetic coordinate systems not supported."
- )
- else:
- units_name = field.units_name(self.connection)
- if units_name:
- return AreaMeasure.unit_attname(units_name)
- def get_distance_att_for_field(self, field):
- dist_att = None
- if field.geodetic(self.connection):
- if self.connection.features.supports_distance_geodetic:
- dist_att = "m"
- else:
- units = field.units_name(self.connection)
- if units:
- dist_att = DistanceMeasure.unit_attname(units)
- return dist_att
|