Merge pull request #1036 from MRigal/snario-min-distance
Added test, doc to implementation of min_distance query
This commit is contained in:
commit
33ea2b4844
@ -29,6 +29,7 @@ Changes in 0.9.X - DEV
|
|||||||
- Added `BaseQuerySet.aggregate_sum` and `BaseQuerySet.aggregate_average` methods.
|
- Added `BaseQuerySet.aggregate_sum` and `BaseQuerySet.aggregate_average` methods.
|
||||||
- Fix for delete with write_concern {'w': 0}. #1008
|
- Fix for delete with write_concern {'w': 0}. #1008
|
||||||
- Allow dynamic lookup for more than two parts. #882
|
- Allow dynamic lookup for more than two parts. #882
|
||||||
|
- Added support for min_distance on geo queries. #831
|
||||||
|
|
||||||
Changes in 0.9.0
|
Changes in 0.9.0
|
||||||
================
|
================
|
||||||
|
@ -146,9 +146,10 @@ The following were added in MongoEngine 0.8 for
|
|||||||
loc.objects(point__near=[40, 5])
|
loc.objects(point__near=[40, 5])
|
||||||
loc.objects(point__near={"type": "Point", "coordinates": [40, 5]})
|
loc.objects(point__near={"type": "Point", "coordinates": [40, 5]})
|
||||||
|
|
||||||
You can also set the maximum distance in meters as well::
|
You can also set the maximum and/or the minimum distance in meters as well::
|
||||||
|
|
||||||
loc.objects(point__near=[40, 5], point__max_distance=1000)
|
loc.objects(point__near=[40, 5], point__max_distance=1000)
|
||||||
|
loc.objects(point__near=[40, 5], point__min_distance=100)
|
||||||
|
|
||||||
The older 2D indexes are still supported with the
|
The older 2D indexes are still supported with the
|
||||||
:class:`~mongoengine.fields.GeoPointField`:
|
:class:`~mongoengine.fields.GeoPointField`:
|
||||||
@ -168,7 +169,8 @@ The older 2D indexes are still supported with the
|
|||||||
|
|
||||||
* ``max_distance`` -- can be added to your location queries to set a maximum
|
* ``max_distance`` -- can be added to your location queries to set a maximum
|
||||||
distance.
|
distance.
|
||||||
|
* ``min_distance`` -- can be added to your location queries to set a minimum
|
||||||
|
distance.
|
||||||
|
|
||||||
Querying lists
|
Querying lists
|
||||||
--------------
|
--------------
|
||||||
|
@ -7,6 +7,7 @@ from mongoengine.base.fields import UPDATE_OPERATORS
|
|||||||
from mongoengine.connection import get_connection
|
from mongoengine.connection import get_connection
|
||||||
from mongoengine.common import _import_class
|
from mongoengine.common import _import_class
|
||||||
from mongoengine.errors import InvalidQueryError
|
from mongoengine.errors import InvalidQueryError
|
||||||
|
from mongoengine.python_support import IS_PYMONGO_3
|
||||||
|
|
||||||
__all__ = ('query', 'update')
|
__all__ = ('query', 'update')
|
||||||
|
|
||||||
@ -15,7 +16,7 @@ COMPARISON_OPERATORS = ('ne', 'gt', 'gte', 'lt', 'lte', 'in', 'nin', 'mod',
|
|||||||
'all', 'size', 'exists', 'not', 'elemMatch', 'type')
|
'all', 'size', 'exists', 'not', 'elemMatch', 'type')
|
||||||
GEO_OPERATORS = ('within_distance', 'within_spherical_distance',
|
GEO_OPERATORS = ('within_distance', 'within_spherical_distance',
|
||||||
'within_box', 'within_polygon', 'near', 'near_sphere',
|
'within_box', 'within_polygon', 'near', 'near_sphere',
|
||||||
'max_distance', 'geo_within', 'geo_within_box',
|
'max_distance', 'min_distance', 'geo_within', 'geo_within_box',
|
||||||
'geo_within_polygon', 'geo_within_center',
|
'geo_within_polygon', 'geo_within_center',
|
||||||
'geo_within_sphere', 'geo_intersects')
|
'geo_within_sphere', 'geo_intersects')
|
||||||
STRING_OPERATORS = ('contains', 'icontains', 'startswith',
|
STRING_OPERATORS = ('contains', 'icontains', 'startswith',
|
||||||
@ -44,8 +45,8 @@ def query(_doc_cls=None, _field_operation=False, **query):
|
|||||||
if len(parts) > 1 and parts[-1] in MATCH_OPERATORS:
|
if len(parts) > 1 and parts[-1] in MATCH_OPERATORS:
|
||||||
op = parts.pop()
|
op = parts.pop()
|
||||||
|
|
||||||
# if user escape field name by __
|
#if user escape field name by __
|
||||||
if len(parts) > 1 and parts[-1] == "":
|
if len(parts) > 1 and parts[-1]=="":
|
||||||
parts.pop()
|
parts.pop()
|
||||||
|
|
||||||
negate = False
|
negate = False
|
||||||
@ -126,24 +127,34 @@ def query(_doc_cls=None, _field_operation=False, **query):
|
|||||||
elif key in mongo_query:
|
elif key in mongo_query:
|
||||||
if key in mongo_query and isinstance(mongo_query[key], dict):
|
if key in mongo_query and isinstance(mongo_query[key], dict):
|
||||||
mongo_query[key].update(value)
|
mongo_query[key].update(value)
|
||||||
# $maxDistance needs to come last - convert to SON
|
# $max/minDistance needs to come last - convert to SON
|
||||||
value_dict = mongo_query[key]
|
value_dict = mongo_query[key]
|
||||||
if '$maxDistance' in value_dict and '$near' in value_dict:
|
if ('$maxDistance' in value_dict or '$minDistance' in value_dict) and \
|
||||||
|
('$near' in value_dict or '$nearSphere' in value_dict):
|
||||||
value_son = SON()
|
value_son = SON()
|
||||||
if isinstance(value_dict['$near'], dict):
|
for k, v in value_dict.iteritems():
|
||||||
for k, v in value_dict.iteritems():
|
if k == '$maxDistance' or k == '$minDistance':
|
||||||
if k == '$maxDistance':
|
continue
|
||||||
continue
|
value_son[k] = v
|
||||||
value_son[k] = v
|
# Required for MongoDB >= 2.6, may fail when combining
|
||||||
value_son['$near'] = SON(value_son['$near'])
|
# PyMongo 3+ and MongoDB < 2.6
|
||||||
value_son['$near']['$maxDistance'] = value_dict['$maxDistance']
|
near_embedded = False
|
||||||
else:
|
for near_op in ('$near', '$nearSphere'):
|
||||||
for k, v in value_dict.iteritems():
|
if isinstance(value_dict.get(near_op), dict) and (
|
||||||
if k == '$maxDistance':
|
IS_PYMONGO_3 or get_connection().max_wire_version > 1):
|
||||||
continue
|
value_son[near_op] = SON(value_son[near_op])
|
||||||
value_son[k] = v
|
if '$maxDistance' in value_dict:
|
||||||
value_son['$maxDistance'] = value_dict['$maxDistance']
|
value_son[near_op][
|
||||||
|
'$maxDistance'] = value_dict['$maxDistance']
|
||||||
|
if '$minDistance' in value_dict:
|
||||||
|
value_son[near_op][
|
||||||
|
'$minDistance'] = value_dict['$minDistance']
|
||||||
|
near_embedded = True
|
||||||
|
if not near_embedded:
|
||||||
|
if '$maxDistance' in value_dict:
|
||||||
|
value_son['$maxDistance'] = value_dict['$maxDistance']
|
||||||
|
if '$minDistance' in value_dict:
|
||||||
|
value_son['$minDistance'] = value_dict['$minDistance']
|
||||||
mongo_query[key] = value_son
|
mongo_query[key] = value_son
|
||||||
else:
|
else:
|
||||||
# Store for manually merging later
|
# Store for manually merging later
|
||||||
@ -297,7 +308,11 @@ def update(_doc_cls=None, **update):
|
|||||||
|
|
||||||
def _geo_operator(field, op, value):
|
def _geo_operator(field, op, value):
|
||||||
"""Helper to return the query for a given geo query"""
|
"""Helper to return the query for a given geo query"""
|
||||||
if field._geo_index == pymongo.GEO2D:
|
if op == "max_distance":
|
||||||
|
value = {'$maxDistance': value}
|
||||||
|
elif op == "min_distance":
|
||||||
|
value = {'$minDistance': value}
|
||||||
|
elif field._geo_index == pymongo.GEO2D:
|
||||||
if op == "within_distance":
|
if op == "within_distance":
|
||||||
value = {'$within': {'$center': value}}
|
value = {'$within': {'$center': value}}
|
||||||
elif op == "within_spherical_distance":
|
elif op == "within_spherical_distance":
|
||||||
@ -310,8 +325,6 @@ def _geo_operator(field, op, value):
|
|||||||
value = {'$nearSphere': value}
|
value = {'$nearSphere': value}
|
||||||
elif op == 'within_box':
|
elif op == 'within_box':
|
||||||
value = {'$within': {'$box': value}}
|
value = {'$within': {'$box': value}}
|
||||||
elif op == "max_distance":
|
|
||||||
value = {'$maxDistance': value}
|
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError("Geo method '%s' has not "
|
raise NotImplementedError("Geo method '%s' has not "
|
||||||
"been implemented for a GeoPointField" % op)
|
"been implemented for a GeoPointField" % op)
|
||||||
@ -330,8 +343,6 @@ def _geo_operator(field, op, value):
|
|||||||
value = {"$geoIntersects": _infer_geometry(value)}
|
value = {"$geoIntersects": _infer_geometry(value)}
|
||||||
elif op == "near":
|
elif op == "near":
|
||||||
value = {'$near': _infer_geometry(value)}
|
value = {'$near': _infer_geometry(value)}
|
||||||
elif op == "max_distance":
|
|
||||||
value = {'$maxDistance': value}
|
|
||||||
else:
|
else:
|
||||||
raise NotImplementedError("Geo method '%s' has not "
|
raise NotImplementedError("Geo method '%s' has not "
|
||||||
"been implemented for a %s " % (op, field._name))
|
"been implemented for a %s " % (op, field._name))
|
||||||
|
@ -70,6 +70,16 @@ class GeoQueriesTest(unittest.TestCase):
|
|||||||
self.assertEqual(events.count(), 1)
|
self.assertEqual(events.count(), 1)
|
||||||
self.assertEqual(events[0], event2)
|
self.assertEqual(events[0], event2)
|
||||||
|
|
||||||
|
# find events at least 10 degrees away of san francisco
|
||||||
|
point = [-122.415579, 37.7566023]
|
||||||
|
events = Event.objects(location__near=point, location__min_distance=10)
|
||||||
|
# The following real test passes on MongoDB 3 but minDistance seems
|
||||||
|
# buggy on older MongoDB versions
|
||||||
|
if get_connection().server_info()['versionArray'][0] > 2:
|
||||||
|
self.assertEqual(events.count(), 2)
|
||||||
|
else:
|
||||||
|
self.assertTrue(events.count() >= 2)
|
||||||
|
|
||||||
# find events within 10 degrees of san francisco
|
# find events within 10 degrees of san francisco
|
||||||
point_and_distance = [[-122.415579, 37.7566023], 10]
|
point_and_distance = [[-122.415579, 37.7566023], 10]
|
||||||
events = Event.objects(location__within_distance=point_and_distance)
|
events = Event.objects(location__within_distance=point_and_distance)
|
||||||
@ -171,21 +181,32 @@ class GeoQueriesTest(unittest.TestCase):
|
|||||||
|
|
||||||
# Same behavior for _within_spherical_distance
|
# Same behavior for _within_spherical_distance
|
||||||
points = Point.objects(
|
points = Point.objects(
|
||||||
location__within_spherical_distance=[[-122, 37.5], 60/earth_radius]
|
location__within_spherical_distance=[[-122, 37.5], 60 / earth_radius]
|
||||||
)
|
)
|
||||||
self.assertEqual(points.count(), 2)
|
self.assertEqual(points.count(), 2)
|
||||||
|
|
||||||
points = Point.objects(location__near_sphere=[-122, 37.5],
|
points = Point.objects(location__near_sphere=[-122, 37.5],
|
||||||
location__max_distance=60 / earth_radius)
|
location__max_distance=60 / earth_radius)
|
||||||
# This test is sometimes failing with Mongo internals non-sense.
|
|
||||||
# See https://travis-ci.org/MongoEngine/mongoengine/builds/58729101
|
|
||||||
try:
|
|
||||||
points.count()
|
|
||||||
except OperationFailure:
|
|
||||||
raise SkipTest("Sometimes MongoDB ignores its capacities on maxDistance")
|
|
||||||
|
|
||||||
self.assertEqual(points.count(), 2)
|
self.assertEqual(points.count(), 2)
|
||||||
|
|
||||||
|
# Test query works with max_distance, being farer from one point
|
||||||
|
points = Point.objects(location__near_sphere=[-122, 37.8],
|
||||||
|
location__max_distance=60 / earth_radius)
|
||||||
|
close_point = points.first()
|
||||||
|
self.assertEqual(points.count(), 1)
|
||||||
|
|
||||||
|
# Test query works with min_distance, being farer from one point
|
||||||
|
points = Point.objects(location__near_sphere=[-122, 37.8],
|
||||||
|
location__min_distance=60 / earth_radius)
|
||||||
|
# The following real test passes on MongoDB 3 but minDistance seems
|
||||||
|
# buggy on older MongoDB versions
|
||||||
|
if get_connection().server_info()['versionArray'][0] > 2:
|
||||||
|
self.assertEqual(points.count(), 1)
|
||||||
|
far_point = points.first()
|
||||||
|
self.assertNotEqual(close_point, far_point)
|
||||||
|
else:
|
||||||
|
self.assertTrue(points.count() >= 1)
|
||||||
|
|
||||||
# Finds both points, but orders the north point first because it's
|
# Finds both points, but orders the north point first because it's
|
||||||
# closer to the reference point to the north.
|
# closer to the reference point to the north.
|
||||||
points = Point.objects(location__near_sphere=[-122, 38.5])
|
points = Point.objects(location__near_sphere=[-122, 38.5])
|
||||||
@ -268,6 +289,20 @@ class GeoQueriesTest(unittest.TestCase):
|
|||||||
self.assertEqual(events.count(), 2)
|
self.assertEqual(events.count(), 2)
|
||||||
self.assertEqual(events[0], event3)
|
self.assertEqual(events[0], event3)
|
||||||
|
|
||||||
|
# ensure min_distance and max_distance combine well
|
||||||
|
events = Event.objects(location__near=[-87.67892, 41.9120459],
|
||||||
|
location__min_distance=1000,
|
||||||
|
location__max_distance=10000).order_by("-date")
|
||||||
|
self.assertEqual(events.count(), 1)
|
||||||
|
self.assertEqual(events[0], event3)
|
||||||
|
|
||||||
|
# ensure ordering is respected by "near"
|
||||||
|
events = Event.objects(location__near=[-87.67892, 41.9120459],
|
||||||
|
# location__min_distance=10000
|
||||||
|
location__min_distance=10000).order_by("-date")
|
||||||
|
self.assertEqual(events.count(), 1)
|
||||||
|
self.assertEqual(events[0], event2)
|
||||||
|
|
||||||
# check that within_box works
|
# check that within_box works
|
||||||
box = [(-125.0, 35.0), (-100.0, 40.0)]
|
box = [(-125.0, 35.0), (-100.0, 40.0)]
|
||||||
events = Event.objects(location__geo_within_box=box)
|
events = Event.objects(location__geo_within_box=box)
|
||||||
|
@ -4763,5 +4763,19 @@ class QuerySetTest(unittest.TestCase):
|
|||||||
for p in Person.objects():
|
for p in Person.objects():
|
||||||
self.assertEqual(p.name, 'a')
|
self.assertEqual(p.name, 'a')
|
||||||
|
|
||||||
|
def test_last_field_name_like_operator(self):
|
||||||
|
class EmbeddedItem(EmbeddedDocument):
|
||||||
|
type = StringField()
|
||||||
|
|
||||||
|
class Doc(Document):
|
||||||
|
item = EmbeddedDocumentField(EmbeddedItem)
|
||||||
|
|
||||||
|
Doc.drop_collection()
|
||||||
|
|
||||||
|
doc = Doc(item=EmbeddedItem(type="axe"))
|
||||||
|
doc.save()
|
||||||
|
|
||||||
|
self.assertEqual(1, Doc.objects(item__type__="axe").count())
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
unittest.main()
|
unittest.main()
|
||||||
|
@ -54,11 +54,10 @@ class ConnectionTest(unittest.TestCase):
|
|||||||
def test_sharing_connections(self):
|
def test_sharing_connections(self):
|
||||||
"""Ensure that connections are shared when the connection settings are exactly the same
|
"""Ensure that connections are shared when the connection settings are exactly the same
|
||||||
"""
|
"""
|
||||||
connect('mongoenginetest', alias='testdb1')
|
connect('mongoenginetests', alias='testdb1')
|
||||||
|
|
||||||
expected_connection = get_connection('testdb1')
|
expected_connection = get_connection('testdb1')
|
||||||
|
|
||||||
connect('mongoenginetest', alias='testdb2')
|
connect('mongoenginetests', alias='testdb2')
|
||||||
actual_connection = get_connection('testdb2')
|
actual_connection = get_connection('testdb2')
|
||||||
|
|
||||||
# Handle PyMongo 3+ Async Connection
|
# Handle PyMongo 3+ Async Connection
|
||||||
@ -96,8 +95,7 @@ class ConnectionTest(unittest.TestCase):
|
|||||||
c.mongoenginetest.system.users.remove({})
|
c.mongoenginetest.system.users.remove({})
|
||||||
|
|
||||||
def test_connect_uri_without_db(self):
|
def test_connect_uri_without_db(self):
|
||||||
"""Ensure that the connect() method works properly with uri's
|
"""Ensure connect() method works properly with uri's without database_name
|
||||||
without database_name
|
|
||||||
"""
|
"""
|
||||||
c = connect(db='mongoenginetest', alias='admin')
|
c = connect(db='mongoenginetest', alias='admin')
|
||||||
c.admin.system.users.remove({})
|
c.admin.system.users.remove({})
|
||||||
@ -130,28 +128,27 @@ class ConnectionTest(unittest.TestCase):
|
|||||||
# Create users
|
# Create users
|
||||||
c = connect('mongoenginetest')
|
c = connect('mongoenginetest')
|
||||||
c.admin.system.users.remove({})
|
c.admin.system.users.remove({})
|
||||||
c.admin.add_user('username', 'password')
|
c.admin.add_user('username2', 'password')
|
||||||
|
|
||||||
# Authentication fails without "authSource"
|
# Authentication fails without "authSource"
|
||||||
if IS_PYMONGO_3:
|
if IS_PYMONGO_3:
|
||||||
test_conn = connect('mongoenginetest', alias='test2',
|
test_conn = connect('mongoenginetest', alias='test1',
|
||||||
host='mongodb://username:password@localhost/mongoenginetest')
|
host='mongodb://username2:password@localhost/mongoenginetest')
|
||||||
self.assertRaises(OperationFailure, test_conn.server_info)
|
self.assertRaises(OperationFailure, test_conn.server_info)
|
||||||
else:
|
else:
|
||||||
self.assertRaises(
|
self.assertRaises(
|
||||||
ConnectionError, connect, 'mongoenginetest', alias='test1',
|
ConnectionError, connect, 'mongoenginetest', alias='test1',
|
||||||
host='mongodb://username:password@localhost/mongoenginetest'
|
host='mongodb://username2:password@localhost/mongoenginetest'
|
||||||
)
|
)
|
||||||
self.assertRaises(ConnectionError, get_db, 'test1')
|
self.assertRaises(ConnectionError, get_db, 'test1')
|
||||||
|
|
||||||
# Authentication succeeds with "authSource"
|
# Authentication succeeds with "authSource"
|
||||||
test_conn2 = connect(
|
test_conn2 = connect(
|
||||||
'mongoenginetest', alias='test2',
|
'mongoenginetest', alias='test2',
|
||||||
host=('mongodb://username:password@localhost/'
|
host=('mongodb://username2:password@localhost/'
|
||||||
'mongoenginetest?authSource=admin')
|
'mongoenginetest?authSource=admin')
|
||||||
)
|
)
|
||||||
# This will fail starting from MongoDB 2.6+
|
# This will fail starting from MongoDB 2.6+
|
||||||
# test_conn2.server_info()
|
|
||||||
db = get_db('test2')
|
db = get_db('test2')
|
||||||
self.assertTrue(isinstance(db, pymongo.database.Database))
|
self.assertTrue(isinstance(db, pymongo.database.Database))
|
||||||
self.assertEqual(db.name, 'mongoenginetest')
|
self.assertEqual(db.name, 'mongoenginetest')
|
||||||
|
Loading…
x
Reference in New Issue
Block a user