项目作者: bukowa

项目描述 :
postgres + django: filter_in_range that allows quick filtering for locations in N range from X/Y (lat/lng) (earth_box, ll_to_earth, cube, earthdistance)
高级语言: HTML
项目地址: git://github.com/bukowa/django-georangefilter.git
创建时间: 2018-06-03T11:24:26Z
项目社区:https://github.com/bukowa/django-georangefilter

开源协议:The Unlicense

下载


django-georangefilter

Python function that will annonate your queryset using Postgres earth_box and ll_to_earth functions that allow filtering in range N kilometers from given latitude/longitude point.

IMPORTANT:
https://www.postgresql.org/docs/devel/static/earthdistance.html

  1. In this module, the Earth is assumed to be perfectly spherical. (If that's too inaccurate for you, you might want to look at the PostGIS project).
  1. pip install git+https://github.com/bukowa/django-georangefilter

Results for filtering in 15km radius, with second circle 30km radius, as you can see its pretty inaccurate.

  1. If you want to test in your custom radius, take a look at tests.py file.


For example, given model:

  1. class City(models.Model):
  2. longitude = models.FloatField(db_index=True)
  3. latitude = models.FloatField(db_index=True)
  4. def in_range(self, range_in_meters, **kwargs):
  5. return filter_in_range(
  6. queryset=self.__class__.objects,
  7. latitude=self.latitude,
  8. longitude=self.longitude,
  9. range_in_meters=range_in_meters,
  10. latitude_column_name="latitude",
  11. longitude_column_name="longitude",
  12. **kwargs,
  13. )

Get all cities in range 10km from city:

  1. city = City.objects.all()[0]
  2. cities_in_range = city.in_range(10000)

How the sql query really looks like:

  1. {'sql': 'SELECT "app_city"."longitude", "app_city"."latitude", earth_box(ll_to_earth(51.03923, 16.97184), 10000) AS "earthbox" FROM "X" WHERE earth_box(ll_to_earth(51.03923, 16.97184), 10000) @> (ll_to_earth("X"."latitude", "X"."longitude")) LIMIT 21', 'time': '0.004'}

Requires cube and earthdistance extensions. To enable them you can:

  1. operations = [
  2. CreateExtension("cube"),
  3. CreateExtension("earthdistance"),
  4. # you can also create an index like that:
  5. migrations.RunSQL(
  6. "CREATE INDEX indexname ON yourapp_yourmodel USING gist(ll_to_earth(latitude_column_name, longitude_column_name));"
  7. ),
  8. ]

or

  1. from django.db import connection
  2. cursor = connection.cursor()
  3. cursor.execute("CREATE EXTENSION cube;")
  4. cursor.execute("CREATE EXTENSION earthdistance;")

How the functions looks like:

  1. def filter_in_range(
  2. queryset: QuerySet,
  3. latitude: float,
  4. longitude: float,
  5. range_in_meters: int,
  6. latitude_column_name: str = "latitude",
  7. longitude_column_name: str = "longitude",
  8. field_name: str = "earthbox",
  9. lookup_exp: str = "in_georange",
  10. ):
  11. earthbox = {field_name: EarthBox(LLToEarth(latitude, longitude), range_in_meters)}
  12. lookup = "%s__%s" % (field_name, lookup_exp)
  13. in_range = {lookup: LLToEarth(latitude_column_name, longitude_column_name)}
  14. return queryset.annotate(**earthbox).filter(**in_range)

data for map generation used from geonames.org