Compare commits
	
		
			2 Commits
		
	
	
		
			v0.10.8
			...
			fix-iterat
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 894678da39 | ||
|  | 0a66a4b8a9 | 
							
								
								
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										4
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -14,6 +14,4 @@ env/ | |||||||
| .project | .project | ||||||
| .pydevproject | .pydevproject | ||||||
| tests/test_bugfix.py | tests/test_bugfix.py | ||||||
| htmlcov/ | htmlcov/ | ||||||
| venv |  | ||||||
| venv3 |  | ||||||
| @@ -1,7 +1,7 @@ | |||||||
| language: python | language: python | ||||||
|  |  | ||||||
| python: | python: | ||||||
| - '2.6'  # TODO remove in v0.11.0 | - '2.6' | ||||||
| - '2.7' | - '2.7' | ||||||
| - '3.3' | - '3.3' | ||||||
| - '3.4' | - '3.4' | ||||||
|   | |||||||
| @@ -4,19 +4,9 @@ Changelog | |||||||
|  |  | ||||||
| Changes in 0.10.8 | Changes in 0.10.8 | ||||||
| ================= | ================= | ||||||
| - Added support for QuerySet.batch_size (#1426) |  | ||||||
| - Fixed query set iteration within iteration #1427 |  | ||||||
| - Fixed an issue where specifying a MongoDB URI host would override more information than it should #1421 |  | ||||||
| - Added ability to filter the generic reference field by ObjectId and DBRef #1425 |  | ||||||
| - Fixed delete cascade for models with a custom primary key field #1247 |  | ||||||
| - Added ability to specify an authentication mechanism (e.g. X.509) #1333 | - Added ability to specify an authentication mechanism (e.g. X.509) #1333 | ||||||
| - Added support for falsey primary keys (e.g. doc.pk = 0) #1354 | - Added support for falsey primary keys (e.g. doc.pk = 0) #1354 | ||||||
| - Fixed QuerySet#sum/average for fields w/ explicit db_field #1417 | - Fixed BaseQuerySet#sum/average for fields w/ explicit db_field #1417 | ||||||
| - Fixed filtering by embedded_doc=None #1422 |  | ||||||
| - Added support for cursor.comment #1420 |  | ||||||
| - Fixed doc.get_<field>_display #1419 |  | ||||||
| - Fixed __repr__ method of the StrictDict #1424 |  | ||||||
| - Added a deprecation warning for Python 2.6 |  | ||||||
|  |  | ||||||
| Changes in 0.10.7 | Changes in 0.10.7 | ||||||
| ================= | ================= | ||||||
|   | |||||||
| @@ -25,8 +25,7 @@ _dbs = {} | |||||||
|  |  | ||||||
| def register_connection(alias, name=None, host=None, port=None, | def register_connection(alias, name=None, host=None, port=None, | ||||||
|                         read_preference=READ_PREFERENCE, |                         read_preference=READ_PREFERENCE, | ||||||
|                         username=None, password=None, |                         username=None, password=None, authentication_source=None, | ||||||
|                         authentication_source=None, |  | ||||||
|                         authentication_mechanism=None, |                         authentication_mechanism=None, | ||||||
|                         **kwargs): |                         **kwargs): | ||||||
|     """Add a connection. |     """Add a connection. | ||||||
| @@ -71,26 +70,20 @@ def register_connection(alias, name=None, host=None, port=None, | |||||||
|  |  | ||||||
|     resolved_hosts = [] |     resolved_hosts = [] | ||||||
|     for entity in conn_host: |     for entity in conn_host: | ||||||
|  |         # Handle uri style connections | ||||||
|         # Handle Mongomock |  | ||||||
|         if entity.startswith('mongomock://'): |         if entity.startswith('mongomock://'): | ||||||
|             conn_settings['is_mock'] = True |             conn_settings['is_mock'] = True | ||||||
|             # `mongomock://` is not a valid url prefix and must be replaced by `mongodb://` |             # `mongomock://` is not a valid url prefix and must be replaced by `mongodb://` | ||||||
|             resolved_hosts.append(entity.replace('mongomock://', 'mongodb://', 1)) |             resolved_hosts.append(entity.replace('mongomock://', 'mongodb://', 1)) | ||||||
|  |  | ||||||
|         # Handle URI style connections, only updating connection params which |  | ||||||
|         # were explicitly specified in the URI. |  | ||||||
|         elif '://' in entity: |         elif '://' in entity: | ||||||
|             uri_dict = uri_parser.parse_uri(entity) |             uri_dict = uri_parser.parse_uri(entity) | ||||||
|             resolved_hosts.append(entity) |             resolved_hosts.append(entity) | ||||||
|  |             conn_settings.update({ | ||||||
|             if uri_dict.get('database'): |                 'name': uri_dict.get('database') or name, | ||||||
|                 conn_settings['name'] = uri_dict.get('database') |                 'username': uri_dict.get('username'), | ||||||
|  |                 'password': uri_dict.get('password'), | ||||||
|             for param in ('read_preference', 'username', 'password'): |                 'read_preference': read_preference, | ||||||
|                 if uri_dict.get(param): |             }) | ||||||
|                     conn_settings[param] = uri_dict[param] |  | ||||||
|  |  | ||||||
|             uri_options = uri_dict['options'] |             uri_options = uri_dict['options'] | ||||||
|             if 'replicaset' in uri_options: |             if 'replicaset' in uri_options: | ||||||
|                 conn_settings['replicaSet'] = True |                 conn_settings['replicaSet'] = True | ||||||
|   | |||||||
| @@ -1249,7 +1249,7 @@ class GenericReferenceField(BaseField): | |||||||
|         if document is None: |         if document is None: | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
|         if isinstance(document, (dict, SON, ObjectId, DBRef)): |         if isinstance(document, (dict, SON)): | ||||||
|             return document |             return document | ||||||
|  |  | ||||||
|         id_field_name = document.__class__._meta['id_field'] |         id_field_name = document.__class__._meta['id_field'] | ||||||
|   | |||||||
| @@ -1,22 +1,9 @@ | |||||||
| """Helper functions and types to aid with Python 2.6 - 3 support.""" | """Helper functions and types to aid with Python 2.5 - 3 support.""" | ||||||
|  |  | ||||||
| import sys | import sys | ||||||
| import warnings |  | ||||||
|  |  | ||||||
| import pymongo | import pymongo | ||||||
|  |  | ||||||
|  |  | ||||||
| # Show a deprecation warning for people using Python v2.6 |  | ||||||
| # TODO remove in mongoengine v0.11.0 |  | ||||||
| if sys.version_info[0] == 2 and sys.version_info[1] == 6: |  | ||||||
|     warnings.warn( |  | ||||||
|         'Python v2.6 support is deprecated and is going to be dropped ' |  | ||||||
|         'entirely in the upcoming v0.11.0 release. Update your Python ' |  | ||||||
|         'version if you want to have access to the latest features and ' |  | ||||||
|         'bug fixes in MongoEngine.', |  | ||||||
|         DeprecationWarning |  | ||||||
|     ) |  | ||||||
|  |  | ||||||
| if pymongo.version_tuple[0] < 3: | if pymongo.version_tuple[0] < 3: | ||||||
|     IS_PYMONGO_3 = False |     IS_PYMONGO_3 = False | ||||||
| else: | else: | ||||||
|   | |||||||
| @@ -82,7 +82,6 @@ class BaseQuerySet(object): | |||||||
|         self._limit = None |         self._limit = None | ||||||
|         self._skip = None |         self._skip = None | ||||||
|         self._hint = -1  # Using -1 as None is a valid value for hint |         self._hint = -1  # Using -1 as None is a valid value for hint | ||||||
|         self._batch_size = None |  | ||||||
|         self.only_fields = [] |         self.only_fields = [] | ||||||
|         self._max_time_ms = None |         self._max_time_ms = None | ||||||
|  |  | ||||||
| @@ -447,7 +446,7 @@ class BaseQuerySet(object): | |||||||
|                 if doc._collection == document_cls._collection: |                 if doc._collection == document_cls._collection: | ||||||
|                     for ref in queryset: |                     for ref in queryset: | ||||||
|                         cascade_refs.add(ref.id) |                         cascade_refs.add(ref.id) | ||||||
|                 ref_q = document_cls.objects(**{field_name + '__in': self, 'pk__nin': cascade_refs}) |                 ref_q = document_cls.objects(**{field_name + '__in': self, 'id__nin': cascade_refs}) | ||||||
|                 ref_q_count = ref_q.count() |                 ref_q_count = ref_q.count() | ||||||
|                 if ref_q_count > 0: |                 if ref_q_count > 0: | ||||||
|                     ref_q.delete(write_concern=write_concern, cascade_refs=cascade_refs) |                     ref_q.delete(write_concern=write_concern, cascade_refs=cascade_refs) | ||||||
| @@ -784,19 +783,6 @@ class BaseQuerySet(object): | |||||||
|         queryset._hint = index |         queryset._hint = index | ||||||
|         return queryset |         return queryset | ||||||
|  |  | ||||||
|     def batch_size(self, size): |  | ||||||
|         """Limit the number of documents returned in a single batch (each |  | ||||||
|         batch requires a round trip to the server). |  | ||||||
|  |  | ||||||
|         See http://api.mongodb.com/python/current/api/pymongo/cursor.html#pymongo.cursor.Cursor.batch_size |  | ||||||
|         for details. |  | ||||||
|  |  | ||||||
|         :param size: desired size of each batch. |  | ||||||
|         """ |  | ||||||
|         queryset = self.clone() |  | ||||||
|         queryset._batch_size = size |  | ||||||
|         return queryset |  | ||||||
|  |  | ||||||
|     def distinct(self, field): |     def distinct(self, field): | ||||||
|         """Return a list of distinct values for a given field. |         """Return a list of distinct values for a given field. | ||||||
|  |  | ||||||
| @@ -1483,9 +1469,6 @@ class BaseQuerySet(object): | |||||||
|             if self._hint != -1: |             if self._hint != -1: | ||||||
|                 self._cursor_obj.hint(self._hint) |                 self._cursor_obj.hint(self._hint) | ||||||
|  |  | ||||||
|             if self._batch_size is not None: |  | ||||||
|                 self._cursor_obj.batch_size(self._batch_size) |  | ||||||
|  |  | ||||||
|         return self._cursor_obj |         return self._cursor_obj | ||||||
|  |  | ||||||
|     def __deepcopy__(self, memo): |     def __deepcopy__(self, memo): | ||||||
|   | |||||||
| @@ -1,7 +1,6 @@ | |||||||
| from collections import defaultdict | from collections import defaultdict | ||||||
|  |  | ||||||
| from bson import ObjectId, SON | from bson import SON | ||||||
| from bson.dbref import DBRef |  | ||||||
| import pymongo | import pymongo | ||||||
|  |  | ||||||
| from mongoengine.base.fields import UPDATE_OPERATORS | from mongoengine.base.fields import UPDATE_OPERATORS | ||||||
| @@ -27,7 +26,6 @@ MATCH_OPERATORS = (COMPARISON_OPERATORS + GEO_OPERATORS + | |||||||
|                    STRING_OPERATORS + CUSTOM_OPERATORS) |                    STRING_OPERATORS + CUSTOM_OPERATORS) | ||||||
|  |  | ||||||
|  |  | ||||||
| # TODO make this less complex |  | ||||||
| def query(_doc_cls=None, **kwargs): | def query(_doc_cls=None, **kwargs): | ||||||
|     """Transform a query from Django-style format to Mongo format. |     """Transform a query from Django-style format to Mongo format. | ||||||
|     """ |     """ | ||||||
| @@ -64,7 +62,6 @@ def query(_doc_cls=None, **kwargs): | |||||||
|             parts = [] |             parts = [] | ||||||
|  |  | ||||||
|             CachedReferenceField = _import_class('CachedReferenceField') |             CachedReferenceField = _import_class('CachedReferenceField') | ||||||
|             GenericReferenceField = _import_class('GenericReferenceField') |  | ||||||
|  |  | ||||||
|             cleaned_fields = [] |             cleaned_fields = [] | ||||||
|             for field in fields: |             for field in fields: | ||||||
| @@ -104,16 +101,6 @@ def query(_doc_cls=None, **kwargs): | |||||||
|                 # 'in', 'nin' and 'all' require a list of values |                 # 'in', 'nin' and 'all' require a list of values | ||||||
|                 value = [field.prepare_query_value(op, v) for v in value] |                 value = [field.prepare_query_value(op, v) for v in value] | ||||||
|  |  | ||||||
|             # If we're querying a GenericReferenceField, we need to alter the |  | ||||||
|             # key depending on the value: |  | ||||||
|             # * If the value is a DBRef, the key should be "field_name._ref". |  | ||||||
|             # * If the value is an ObjectId, the key should be "field_name._ref.$id". |  | ||||||
|             if isinstance(field, GenericReferenceField): |  | ||||||
|                 if isinstance(value, DBRef): |  | ||||||
|                     parts[-1] += '._ref' |  | ||||||
|                 elif isinstance(value, ObjectId): |  | ||||||
|                     parts[-1] += '._ref.$id' |  | ||||||
|  |  | ||||||
|         # if op and op not in COMPARISON_OPERATORS: |         # if op and op not in COMPARISON_OPERATORS: | ||||||
|         if op: |         if op: | ||||||
|             if op in GEO_OPERATORS: |             if op in GEO_OPERATORS: | ||||||
| @@ -141,13 +128,11 @@ def query(_doc_cls=None, **kwargs): | |||||||
|  |  | ||||||
|         for i, part in indices: |         for i, part in indices: | ||||||
|             parts.insert(i, part) |             parts.insert(i, part) | ||||||
|  |  | ||||||
|         key = '.'.join(parts) |         key = '.'.join(parts) | ||||||
|  |  | ||||||
|         if op is None or key not in mongo_query: |         if op is None or key not in mongo_query: | ||||||
|             mongo_query[key] = value |             mongo_query[key] = value | ||||||
|         elif key in mongo_query: |         elif key in mongo_query: | ||||||
|             if 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) | ||||||
|                 # $max/minDistance 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] | ||||||
|   | |||||||
| @@ -9,5 +9,5 @@ tests = tests | |||||||
| [flake8] | [flake8] | ||||||
| ignore=E501,F401,F403,F405,I201 | ignore=E501,F401,F403,F405,I201 | ||||||
| exclude=build,dist,docs,venv,.tox,.eggs,tests | exclude=build,dist,docs,venv,.tox,.eggs,tests | ||||||
| max-complexity=45 | max-complexity=42 | ||||||
| application-import-names=mongoengine,tests | application-import-names=mongoengine,tests | ||||||
|   | |||||||
| @@ -2810,38 +2810,6 @@ class FieldTest(unittest.TestCase): | |||||||
|         Post.drop_collection() |         Post.drop_collection() | ||||||
|         User.drop_collection() |         User.drop_collection() | ||||||
|  |  | ||||||
|     def test_generic_reference_filter_by_dbref(self): |  | ||||||
|         """Ensure we can search for a specific generic reference by |  | ||||||
|         providing its ObjectId. |  | ||||||
|         """ |  | ||||||
|         class Doc(Document): |  | ||||||
|             ref = GenericReferenceField() |  | ||||||
|  |  | ||||||
|         Doc.drop_collection() |  | ||||||
|  |  | ||||||
|         doc1 = Doc.objects.create() |  | ||||||
|         doc2 = Doc.objects.create(ref=doc1) |  | ||||||
|  |  | ||||||
|         doc = Doc.objects.get(ref=DBRef('doc', doc1.pk)) |  | ||||||
|         self.assertEqual(doc, doc2) |  | ||||||
|  |  | ||||||
|     def test_generic_reference_filter_by_objectid(self): |  | ||||||
|         """Ensure we can search for a specific generic reference by |  | ||||||
|         providing its DBRef. |  | ||||||
|         """ |  | ||||||
|         class Doc(Document): |  | ||||||
|             ref = GenericReferenceField() |  | ||||||
|  |  | ||||||
|         Doc.drop_collection() |  | ||||||
|  |  | ||||||
|         doc1 = Doc.objects.create() |  | ||||||
|         doc2 = Doc.objects.create(ref=doc1) |  | ||||||
|  |  | ||||||
|         self.assertTrue(isinstance(doc1.pk, ObjectId)) |  | ||||||
|  |  | ||||||
|         doc = Doc.objects.get(ref=doc1.pk) |  | ||||||
|         self.assertEqual(doc, doc2) |  | ||||||
|  |  | ||||||
|     def test_binary_fields(self): |     def test_binary_fields(self): | ||||||
|         """Ensure that binary fields can be stored and retrieved. |         """Ensure that binary fields can be stored and retrieved. | ||||||
|         """ |         """ | ||||||
|   | |||||||
| @@ -337,34 +337,6 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         query = query.filter(boolfield=True) |         query = query.filter(boolfield=True) | ||||||
|         self.assertEqual(query.count(), 1) |         self.assertEqual(query.count(), 1) | ||||||
|  |  | ||||||
|     def test_batch_size(self): |  | ||||||
|         """Ensure that batch_size works.""" |  | ||||||
|         class A(Document): |  | ||||||
|             s = StringField() |  | ||||||
|  |  | ||||||
|         A.drop_collection() |  | ||||||
|  |  | ||||||
|         for i in range(100): |  | ||||||
|             A.objects.create(s=str(i)) |  | ||||||
|  |  | ||||||
|         # test iterating over the result set |  | ||||||
|         cnt = 0 |  | ||||||
|         for a in A.objects.batch_size(10): |  | ||||||
|             cnt += 1 |  | ||||||
|         self.assertEqual(cnt, 100) |  | ||||||
|  |  | ||||||
|         # test chaining |  | ||||||
|         qs = A.objects.all() |  | ||||||
|         qs = qs.limit(10).batch_size(20).skip(91) |  | ||||||
|         cnt = 0 |  | ||||||
|         for a in qs: |  | ||||||
|             cnt += 1 |  | ||||||
|         self.assertEqual(cnt, 9) |  | ||||||
|  |  | ||||||
|         # test invalid batch size |  | ||||||
|         qs = A.objects.batch_size(-1) |  | ||||||
|         self.assertRaises(ValueError, lambda: list(qs)) |  | ||||||
|  |  | ||||||
|     def test_update_write_concern(self): |     def test_update_write_concern(self): | ||||||
|         """Test that passing write_concern works""" |         """Test that passing write_concern works""" | ||||||
|         self.Person.drop_collection() |         self.Person.drop_collection() | ||||||
|   | |||||||
| @@ -174,9 +174,19 @@ 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 connect() method works properly if the URI doesn't |         """Ensure connect() method works properly with uri's without database_name | ||||||
|         include a database name. |  | ||||||
|         """ |         """ | ||||||
|  |         c = connect(db='mongoenginetest', alias='admin') | ||||||
|  |         c.admin.system.users.remove({}) | ||||||
|  |         c.mongoenginetest.system.users.remove({}) | ||||||
|  |  | ||||||
|  |         c.admin.add_user("admin", "password") | ||||||
|  |         c.admin.authenticate("admin", "password") | ||||||
|  |         c.mongoenginetest.add_user("username", "password") | ||||||
|  |  | ||||||
|  |         if not IS_PYMONGO_3: | ||||||
|  |             self.assertRaises(ConnectionError, connect, "testdb_uri_bad", host='mongodb://test:password@localhost') | ||||||
|  |  | ||||||
|         connect("mongoenginetest", host='mongodb://localhost/') |         connect("mongoenginetest", host='mongodb://localhost/') | ||||||
|  |  | ||||||
|         conn = get_connection() |         conn = get_connection() | ||||||
| @@ -186,31 +196,8 @@ class ConnectionTest(unittest.TestCase): | |||||||
|         self.assertTrue(isinstance(db, pymongo.database.Database)) |         self.assertTrue(isinstance(db, pymongo.database.Database)) | ||||||
|         self.assertEqual(db.name, 'mongoenginetest') |         self.assertEqual(db.name, 'mongoenginetest') | ||||||
|  |  | ||||||
|     def test_connect_uri_default_db(self): |         c.admin.system.users.remove({}) | ||||||
|         """Ensure connect() defaults to the right database name if |         c.mongoenginetest.system.users.remove({}) | ||||||
|         the URI and the database_name don't explicitly specify it. |  | ||||||
|         """ |  | ||||||
|         connect(host='mongodb://localhost/') |  | ||||||
|  |  | ||||||
|         conn = get_connection() |  | ||||||
|         self.assertTrue(isinstance(conn, pymongo.mongo_client.MongoClient)) |  | ||||||
|  |  | ||||||
|         db = get_db() |  | ||||||
|         self.assertTrue(isinstance(db, pymongo.database.Database)) |  | ||||||
|         self.assertEqual(db.name, 'test') |  | ||||||
|  |  | ||||||
|     def test_uri_without_credentials_doesnt_override_conn_settings(self): |  | ||||||
|         """Ensure connect() uses the username & password params if the URI |  | ||||||
|         doesn't explicitly specify them. |  | ||||||
|         """ |  | ||||||
|         c = connect(host='mongodb://localhost/mongoenginetest', |  | ||||||
|                     username='user', |  | ||||||
|                     password='pass') |  | ||||||
|  |  | ||||||
|         # OperationFailure means that mongoengine attempted authentication |  | ||||||
|         # w/ the provided username/password and failed - that's the desired |  | ||||||
|         # behavior. If the MongoDB URI would override the credentials |  | ||||||
|         self.assertRaises(OperationFailure, get_db) |  | ||||||
|  |  | ||||||
|     def test_connect_uri_with_authsource(self): |     def test_connect_uri_with_authsource(self): | ||||||
|         """Ensure that the connect() method works well with |         """Ensure that the connect() method works well with | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user