Compare commits
	
		
			1 Commits
		
	
	
		
			fix-cursor
			...
			fix-old-py
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | cedb5bda22 | 
| @@ -14,13 +14,13 @@ Before starting to write code, look for existing `tickets | ||||
| <https://github.com/MongoEngine/mongoengine/issues?state=open>`_ or `create one | ||||
| <https://github.com/MongoEngine/mongoengine/issues>`_ for your specific | ||||
| issue or feature request. That way you avoid working on something | ||||
| that might not be of interest or that has already been addressed. If in doubt | ||||
| that might not be of interest or that has already been addressed.  If in doubt | ||||
| post to the `user group <http://groups.google.com/group/mongoengine-users>` | ||||
|  | ||||
| Supported Interpreters | ||||
| ---------------------- | ||||
|  | ||||
| MongoEngine supports CPython 2.7 and newer. Language | ||||
| MongoEngine supports CPython 2.6 and newer. Language | ||||
| features not supported by all interpreters can not be used. | ||||
| Please also ensure that your code is properly converted by | ||||
| `2to3 <http://docs.python.org/library/2to3.html>`_ for Python 3 support. | ||||
|   | ||||
							
								
								
									
										36
									
								
								README.rst
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								README.rst
									
									
									
									
									
								
							| @@ -35,22 +35,16 @@ setup.py install``. | ||||
|  | ||||
| Dependencies | ||||
| ============ | ||||
| All of the dependencies can easily be installed via `pip <https://pip.pypa.io/>`_. At the very least, you'll need these two packages to use MongoEngine: | ||||
|  | ||||
| - pymongo>=2.7.1 | ||||
| - six>=1.10.0 | ||||
|  | ||||
| If you utilize a ``DateTimeField``, you might also use a more flexible date parser: | ||||
| - sphinx (optional - for documentation generation) | ||||
|  | ||||
| Optional Dependencies | ||||
| --------------------- | ||||
| - **Image Fields**: Pillow>=2.0.0 | ||||
| - dateutil>=2.1.0 | ||||
|  | ||||
| If you need to use an ``ImageField`` or ``ImageGridFsProxy``: | ||||
|  | ||||
| - Pillow>=2.0.0 | ||||
|  | ||||
| If you want to generate the documentation (e.g. to contribute to it): | ||||
|  | ||||
| - sphinx | ||||
| .. note | ||||
|    MongoEngine always runs it's test suite against the latest patch version of each dependecy. e.g.: PyMongo 3.0.1 | ||||
|  | ||||
| Examples | ||||
| ======== | ||||
| @@ -63,7 +57,7 @@ Some simple examples of what MongoEngine code looks like: | ||||
|  | ||||
|     class BlogPost(Document): | ||||
|         title = StringField(required=True, max_length=200) | ||||
|         posted = DateTimeField(default=datetime.datetime.utcnow) | ||||
|         posted = DateTimeField(default=datetime.datetime.now) | ||||
|         tags = ListField(StringField(max_length=50)) | ||||
|         meta = {'allow_inheritance': True} | ||||
|  | ||||
| @@ -93,18 +87,17 @@ Some simple examples of what MongoEngine code looks like: | ||||
|     ...     print | ||||
|     ... | ||||
|  | ||||
|     # Count all blog posts and its subtypes | ||||
|     >>> BlogPost.objects.count() | ||||
|     >>> len(BlogPost.objects) | ||||
|     2 | ||||
|     >>> TextPost.objects.count() | ||||
|     >>> len(TextPost.objects) | ||||
|     1 | ||||
|     >>> LinkPost.objects.count() | ||||
|     >>> len(LinkPost.objects) | ||||
|     1 | ||||
|  | ||||
|     # Count tagged posts | ||||
|     >>> BlogPost.objects(tags='mongoengine').count() | ||||
|     # Find tagged posts | ||||
|     >>> len(BlogPost.objects(tags='mongoengine')) | ||||
|     2 | ||||
|     >>> BlogPost.objects(tags='mongodb').count() | ||||
|     >>> len(BlogPost.objects(tags='mongodb')) | ||||
|     1 | ||||
|  | ||||
| Tests | ||||
| @@ -137,7 +130,8 @@ Community | ||||
|   <http://groups.google.com/group/mongoengine-users>`_ | ||||
| - `MongoEngine Developers mailing list | ||||
|   <http://groups.google.com/group/mongoengine-dev>`_ | ||||
| - `#mongoengine IRC channel <http://webchat.freenode.net/?channels=mongoengine>`_ | ||||
|  | ||||
| Contributing | ||||
| ============ | ||||
| We welcome contributions! See the `Contribution guidelines <https://github.com/MongoEngine/mongoengine/blob/master/CONTRIBUTING.rst>`_ | ||||
| We welcome contributions! see  the `Contribution guidelines <https://github.com/MongoEngine/mongoengine/blob/master/CONTRIBUTING.rst>`_ | ||||
|   | ||||
| @@ -5,15 +5,12 @@ Changelog | ||||
| Development | ||||
| =========== | ||||
| - (Fill this out as you fix issues and develop you features). | ||||
| - Fixed connecting to a replica set with PyMongo 2.x #1436 | ||||
| - Fixed an obscure error message when filtering by `field__in=non_iterable`. #1237 | ||||
|  | ||||
| Changes in 0.11.0 | ||||
| ================= | ||||
| - BREAKING CHANGE: Renamed `ConnectionError` to `MongoEngineConnectionError` since the former is a built-in exception name in Python v3.x. #1428 | ||||
| - BREAKING CHANGE: Dropped Python 2.6 support. #1428 | ||||
| - BREAKING CHANGE: `from mongoengine.base import ErrorClass` won't work anymore for any error from `mongoengine.errors` (e.g. `ValidationError`). Use `from mongoengine.errors import ErrorClass instead`. #1428 | ||||
| - BREAKING CHANGE: Accessing a broken reference will raise a `DoesNotExist` error. In the past it used to return `None`. #1334 | ||||
| - Fixed absent rounding for DecimalField when `force_string` is set. #1103 | ||||
|  | ||||
| Changes in 0.10.8 | ||||
|   | ||||
| @@ -33,7 +33,7 @@ the :attr:`host` to | ||||
|     corresponding parameters in :func:`~mongoengine.connect`: :: | ||||
|  | ||||
|         connect( | ||||
|             db='test', | ||||
|             name='test', | ||||
|             username='user', | ||||
|             password='12345', | ||||
|             host='mongodb://admin:qwerty@localhost/production' | ||||
|   | ||||
| @@ -361,6 +361,11 @@ Its value can take any of the following constants: | ||||
|    In Django, be sure to put all apps that have such delete rule declarations in | ||||
|    their :file:`models.py` in the :const:`INSTALLED_APPS` tuple. | ||||
|  | ||||
|  | ||||
| .. warning:: | ||||
|    Signals are not triggered when doing cascading updates / deletes - if this | ||||
|    is required you must manually handle the update / delete. | ||||
|  | ||||
| Generic reference fields | ||||
| '''''''''''''''''''''''' | ||||
| A second kind of reference field also exists, | ||||
|   | ||||
| @@ -479,8 +479,6 @@ operators. To use a :class:`~mongoengine.queryset.Q` object, pass it in as the | ||||
| first positional argument to :attr:`Document.objects` when you filter it by | ||||
| calling it with keyword arguments:: | ||||
|  | ||||
|     from mongoengine.queryset.visitor import Q | ||||
|  | ||||
|     # Get published posts | ||||
|     Post.objects(Q(published=True) | Q(publish_date__lte=datetime.now())) | ||||
|  | ||||
|   | ||||
| @@ -142,4 +142,11 @@ cleaner looking while still allowing manual execution of the callback:: | ||||
|         modified = DateTimeField() | ||||
|  | ||||
|  | ||||
| ReferenceFields and Signals | ||||
| --------------------------- | ||||
|  | ||||
| Currently `reverse_delete_rule` does not trigger signals on the other part of | ||||
| the relationship.  If this is required you must manually handle the | ||||
| reverse deletion. | ||||
|  | ||||
| .. _blinker: http://pypi.python.org/pypi/blinker | ||||
|   | ||||
| @@ -5,7 +5,7 @@ __all__ = ('UPDATE_OPERATORS', 'get_document', '_document_registry') | ||||
|  | ||||
| UPDATE_OPERATORS = set(['set', 'unset', 'inc', 'dec', 'pop', 'push', | ||||
|                         'push_all', 'pull', 'pull_all', 'add_to_set', | ||||
|                         'set_on_insert', 'min', 'max', 'rename']) | ||||
|                         'set_on_insert', 'min', 'max']) | ||||
|  | ||||
|  | ||||
| _document_registry = {} | ||||
|   | ||||
| @@ -138,7 +138,10 @@ class BaseList(list): | ||||
|         return super(BaseList, self).__setitem__(key, value) | ||||
|  | ||||
|     def __delitem__(self, key, *args, **kwargs): | ||||
|         self._mark_as_changed() | ||||
|         if isinstance(key, slice): | ||||
|             self._mark_as_changed() | ||||
|         else: | ||||
|             self._mark_as_changed(key) | ||||
|         return super(BaseList, self).__delitem__(key) | ||||
|  | ||||
|     def __setslice__(self, *args, **kwargs): | ||||
| @@ -429,7 +432,7 @@ class StrictDict(object): | ||||
|     def __eq__(self, other): | ||||
|         return self.items() == other.items() | ||||
|  | ||||
|     def __ne__(self, other): | ||||
|     def __neq__(self, other): | ||||
|         return self.items() != other.items() | ||||
|  | ||||
|     @classmethod | ||||
|   | ||||
| @@ -402,11 +402,9 @@ class BaseDocument(object): | ||||
|             raise ValidationError(message, errors=errors) | ||||
|  | ||||
|     def to_json(self, *args, **kwargs): | ||||
|         """Convert this document to JSON. | ||||
|  | ||||
|         :param use_db_field: Serialize field names as they appear in | ||||
|             MongoDB (as opposed to attribute names on this document). | ||||
|             Defaults to True. | ||||
|         """Converts a document to JSON. | ||||
|         :param use_db_field: Set to True by default but enables the output of the json structure with the field names | ||||
|             and not the mongodb store db_names in case of set to False | ||||
|         """ | ||||
|         use_db_field = kwargs.pop('use_db_field', True) | ||||
|         return json_util.dumps(self.to_mongo(use_db_field), *args, **kwargs) | ||||
| @@ -677,9 +675,6 @@ class BaseDocument(object): | ||||
|         if not only_fields: | ||||
|             only_fields = [] | ||||
|  | ||||
|         if son and not isinstance(son, dict): | ||||
|             raise ValueError("The source SON object needs to be of type 'dict'") | ||||
|  | ||||
|         # Get the class name from the document, falling back to the given | ||||
|         # class if unavailable | ||||
|         class_name = son.get('_cls', cls._class_name) | ||||
|   | ||||
| @@ -41,7 +41,7 @@ class BaseField(object): | ||||
|         """ | ||||
|         :param db_field: The database field to store this field in | ||||
|             (defaults to the name of the field) | ||||
|         :param name: Deprecated - use db_field | ||||
|         :param name: Depreciated - use db_field | ||||
|         :param required: If the field is required. Whether it has to have a | ||||
|             value or not. Defaults to False. | ||||
|         :param default: (optional) The default value for this field if no value | ||||
| @@ -81,17 +81,6 @@ class BaseField(object): | ||||
|         self.sparse = sparse | ||||
|         self._owner_document = None | ||||
|  | ||||
|         # Validate the db_field | ||||
|         if isinstance(self.db_field, six.string_types) and ( | ||||
|             '.' in self.db_field or | ||||
|             '\0' in self.db_field or | ||||
|             self.db_field.startswith('$') | ||||
|         ): | ||||
|             raise ValueError( | ||||
|                 'field names cannot contain dots (".") or null characters ' | ||||
|                 '("\\0"), and they must not start with a dollar sign ("$").' | ||||
|             ) | ||||
|  | ||||
|         # Detect and report conflicts between metadata and base properties. | ||||
|         conflicts = set(dir(self)) & set(kwargs) | ||||
|         if conflicts: | ||||
|   | ||||
| @@ -34,10 +34,7 @@ def _import_class(cls_name): | ||||
|     queryset_classes = ('OperationError',) | ||||
|     deref_classes = ('DeReference',) | ||||
|  | ||||
|     if cls_name == 'BaseDocument': | ||||
|         from mongoengine.base import document as module | ||||
|         import_classes = ['BaseDocument'] | ||||
|     elif cls_name in doc_classes: | ||||
|     if cls_name in doc_classes: | ||||
|         from mongoengine import document as module | ||||
|         import_classes = doc_classes | ||||
|     elif cls_name in field_classes: | ||||
|   | ||||
| @@ -96,7 +96,7 @@ def register_connection(alias, name=None, host=None, port=None, | ||||
|  | ||||
|             uri_options = uri_dict['options'] | ||||
|             if 'replicaset' in uri_options: | ||||
|                 conn_settings['replicaSet'] = uri_options['replicaset'] | ||||
|                 conn_settings['replicaSet'] = True | ||||
|             if 'authsource' in uri_options: | ||||
|                 conn_settings['authentication_source'] = uri_options['authsource'] | ||||
|             if 'authmechanism' in uri_options: | ||||
|   | ||||
| @@ -313,9 +313,6 @@ class Document(BaseDocument): | ||||
|         .. versionchanged:: 0.10.7 | ||||
|             Add signal_kwargs argument | ||||
|         """ | ||||
|         if self._meta.get('abstract'): | ||||
|             raise InvalidDocumentError('Cannot save an abstract document.') | ||||
|  | ||||
|         signal_kwargs = signal_kwargs or {} | ||||
|         signals.pre_save.send(self.__class__, document=self, **signal_kwargs) | ||||
|  | ||||
| @@ -332,20 +329,68 @@ class Document(BaseDocument): | ||||
|         signals.pre_save_post_validation.send(self.__class__, document=self, | ||||
|                                               created=created, **signal_kwargs) | ||||
|  | ||||
|         if self._meta.get('auto_create_index', True): | ||||
|             self.ensure_indexes() | ||||
|  | ||||
|         try: | ||||
|             # Save a new document or update an existing one | ||||
|             collection = self._get_collection() | ||||
|             if self._meta.get('auto_create_index', True): | ||||
|                 self.ensure_indexes() | ||||
|             if created: | ||||
|                 object_id = self._save_create(doc, force_insert, write_concern) | ||||
|                 if force_insert: | ||||
|                     object_id = collection.insert(doc, **write_concern) | ||||
|                 else: | ||||
|                     object_id = collection.save(doc, **write_concern) | ||||
|                     # In PyMongo 3.0, the save() call calls internally the _update() call | ||||
|                     # but they forget to return the _id value passed back, therefore getting it back here | ||||
|                     # Correct behaviour in 2.X and in 3.0.1+ versions | ||||
|                     if not object_id and pymongo.version_tuple == (3, 0): | ||||
|                         pk_as_mongo_obj = self._fields.get(self._meta['id_field']).to_mongo(self.pk) | ||||
|                         object_id = ( | ||||
|                             self._qs.filter(pk=pk_as_mongo_obj).first() and | ||||
|                             self._qs.filter(pk=pk_as_mongo_obj).first().pk | ||||
|                         )  # TODO doesn't this make 2 queries? | ||||
|             else: | ||||
|                 object_id, created = self._save_update(doc, save_condition, | ||||
|                                                        write_concern) | ||||
|                 object_id = doc['_id'] | ||||
|                 updates, removals = self._delta() | ||||
|                 # Need to add shard key to query, or you get an error | ||||
|                 if save_condition is not None: | ||||
|                     select_dict = transform.query(self.__class__, | ||||
|                                                   **save_condition) | ||||
|                 else: | ||||
|                     select_dict = {} | ||||
|                 select_dict['_id'] = object_id | ||||
|                 shard_key = self._meta.get('shard_key', tuple()) | ||||
|                 for k in shard_key: | ||||
|                     path = self._lookup_field(k.split('.')) | ||||
|                     actual_key = [p.db_field for p in path] | ||||
|                     val = doc | ||||
|                     for ak in actual_key: | ||||
|                         val = val[ak] | ||||
|                     select_dict['.'.join(actual_key)] = val | ||||
|  | ||||
|                 def is_new_object(last_error): | ||||
|                     if last_error is not None: | ||||
|                         updated = last_error.get('updatedExisting') | ||||
|                         if updated is not None: | ||||
|                             return not updated | ||||
|                     return created | ||||
|  | ||||
|                 update_query = {} | ||||
|  | ||||
|                 if updates: | ||||
|                     update_query['$set'] = updates | ||||
|                 if removals: | ||||
|                     update_query['$unset'] = removals | ||||
|                 if updates or removals: | ||||
|                     upsert = save_condition is None | ||||
|                     last_error = collection.update(select_dict, update_query, | ||||
|                                                    upsert=upsert, **write_concern) | ||||
|                     if not upsert and last_error['n'] == 0: | ||||
|                         raise SaveConditionError('Race condition preventing' | ||||
|                                                  ' document update detected') | ||||
|                     created = is_new_object(last_error) | ||||
|  | ||||
|             if cascade is None: | ||||
|                 cascade = (self._meta.get('cascade', False) or | ||||
|                            cascade_kwargs is not None) | ||||
|                 cascade = self._meta.get( | ||||
|                     'cascade', False) or cascade_kwargs is not None | ||||
|  | ||||
|             if cascade: | ||||
|                 kwargs = { | ||||
| @@ -358,7 +403,6 @@ class Document(BaseDocument): | ||||
|                     kwargs.update(cascade_kwargs) | ||||
|                 kwargs['_refs'] = _refs | ||||
|                 self.cascade_save(**kwargs) | ||||
|  | ||||
|         except pymongo.errors.DuplicateKeyError as err: | ||||
|             message = u'Tried to save duplicate unique keys (%s)' | ||||
|             raise NotUniqueError(message % six.text_type(err)) | ||||
| @@ -371,91 +415,16 @@ class Document(BaseDocument): | ||||
|                 raise NotUniqueError(message % six.text_type(err)) | ||||
|             raise OperationError(message % six.text_type(err)) | ||||
|  | ||||
|         # Make sure we store the PK on this document now that it's saved | ||||
|         id_field = self._meta['id_field'] | ||||
|         if created or id_field not in self._meta.get('shard_key', []): | ||||
|             self[id_field] = self._fields[id_field].to_python(object_id) | ||||
|  | ||||
|         signals.post_save.send(self.__class__, document=self, | ||||
|                                created=created, **signal_kwargs) | ||||
|  | ||||
|         self._clear_changed_fields() | ||||
|         self._created = False | ||||
|  | ||||
|         return self | ||||
|  | ||||
|     def _save_create(self, doc, force_insert, write_concern): | ||||
|         """Save a new document. | ||||
|  | ||||
|         Helper method, should only be used inside save(). | ||||
|         """ | ||||
|         collection = self._get_collection() | ||||
|  | ||||
|         if force_insert: | ||||
|             return collection.insert(doc, **write_concern) | ||||
|  | ||||
|         object_id = collection.save(doc, **write_concern) | ||||
|  | ||||
|         # In PyMongo 3.0, the save() call calls internally the _update() call | ||||
|         # but they forget to return the _id value passed back, therefore getting it back here | ||||
|         # Correct behaviour in 2.X and in 3.0.1+ versions | ||||
|         if not object_id and pymongo.version_tuple == (3, 0): | ||||
|             pk_as_mongo_obj = self._fields.get(self._meta['id_field']).to_mongo(self.pk) | ||||
|             object_id = ( | ||||
|                 self._qs.filter(pk=pk_as_mongo_obj).first() and | ||||
|                 self._qs.filter(pk=pk_as_mongo_obj).first().pk | ||||
|             )  # TODO doesn't this make 2 queries? | ||||
|  | ||||
|         return object_id | ||||
|  | ||||
|     def _save_update(self, doc, save_condition, write_concern): | ||||
|         """Update an existing document. | ||||
|  | ||||
|         Helper method, should only be used inside save(). | ||||
|         """ | ||||
|         collection = self._get_collection() | ||||
|         object_id = doc['_id'] | ||||
|         created = False | ||||
|  | ||||
|         select_dict = {} | ||||
|         if save_condition is not None: | ||||
|             select_dict = transform.query(self.__class__, **save_condition) | ||||
|  | ||||
|         select_dict['_id'] = object_id | ||||
|  | ||||
|         # Need to add shard key to query, or you get an error | ||||
|         shard_key = self._meta.get('shard_key', tuple()) | ||||
|         for k in shard_key: | ||||
|             path = self._lookup_field(k.split('.')) | ||||
|             actual_key = [p.db_field for p in path] | ||||
|             val = doc | ||||
|             for ak in actual_key: | ||||
|                 val = val[ak] | ||||
|             select_dict['.'.join(actual_key)] = val | ||||
|  | ||||
|         updates, removals = self._delta() | ||||
|         update_query = {} | ||||
|         if updates: | ||||
|             update_query['$set'] = updates | ||||
|         if removals: | ||||
|             update_query['$unset'] = removals | ||||
|         if updates or removals: | ||||
|             upsert = save_condition is None | ||||
|             last_error = collection.update(select_dict, update_query, | ||||
|                                            upsert=upsert, **write_concern) | ||||
|             if not upsert and last_error['n'] == 0: | ||||
|                 raise SaveConditionError('Race condition preventing' | ||||
|                                          ' document update detected') | ||||
|             if last_error is not None: | ||||
|                 updated_existing = last_error.get('updatedExisting') | ||||
|                 if updated_existing is False: | ||||
|                     created = True | ||||
|                     # !!! This is bad, means we accidentally created a new, | ||||
|                     # potentially corrupted document. See | ||||
|                     # https://github.com/MongoEngine/mongoengine/issues/564 | ||||
|  | ||||
|         return object_id, created | ||||
|  | ||||
|     def cascade_save(self, **kwargs): | ||||
|         """Recursively save any references and generic references on the | ||||
|         document. | ||||
| @@ -859,6 +828,7 @@ class Document(BaseDocument): | ||||
|         """ Lists all of the indexes that should be created for given | ||||
|         collection. It includes all the indexes from super- and sub-classes. | ||||
|         """ | ||||
|  | ||||
|         if cls._meta.get('abstract'): | ||||
|             return [] | ||||
|  | ||||
|   | ||||
| @@ -50,8 +50,8 @@ class FieldDoesNotExist(Exception): | ||||
|     or an :class:`~mongoengine.EmbeddedDocument`. | ||||
|  | ||||
|     To avoid this behavior on data loading, | ||||
|     you should set the :attr:`strict` to ``False`` | ||||
|     in the :attr:`meta` dictionary. | ||||
|     you should the :attr:`strict` to ``False`` | ||||
|     in the :attr:`meta` dictionnary. | ||||
|     """ | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -28,7 +28,7 @@ from mongoengine.base import (BaseDocument, BaseField, ComplexBaseField, | ||||
|                               GeoJsonBaseField, ObjectIdField, get_document) | ||||
| from mongoengine.connection import DEFAULT_CONNECTION_NAME, get_db | ||||
| from mongoengine.document import Document, EmbeddedDocument | ||||
| from mongoengine.errors import DoesNotExist, InvalidQueryError, ValidationError | ||||
| from mongoengine.errors import DoesNotExist, ValidationError | ||||
| from mongoengine.python_support import StringIO | ||||
| from mongoengine.queryset import DO_NOTHING, QuerySet | ||||
|  | ||||
| @@ -566,11 +566,7 @@ class EmbeddedDocumentField(BaseField): | ||||
|  | ||||
|     def prepare_query_value(self, op, value): | ||||
|         if value is not None and not isinstance(value, self.document_type): | ||||
|             try: | ||||
|                 value = self.document_type._from_son(value) | ||||
|             except ValueError: | ||||
|                 raise InvalidQueryError("Querying the embedded document '%s' failed, due to an invalid query value" % | ||||
|                                         (self.document_type._class_name,)) | ||||
|             value = self.document_type._from_son(value) | ||||
|         super(EmbeddedDocumentField, self).prepare_query_value(op, value) | ||||
|         return self.to_mongo(value) | ||||
|  | ||||
| @@ -888,6 +884,10 @@ class ReferenceField(BaseField): | ||||
|  | ||||
|         Foo.register_delete_rule(Bar, 'foo', NULLIFY) | ||||
|  | ||||
|     .. note :: | ||||
|         `reverse_delete_rule` does not trigger pre / post delete signals to be | ||||
|         triggered. | ||||
|  | ||||
|     .. versionchanged:: 0.5 added `reverse_delete_rule` | ||||
|     """ | ||||
|  | ||||
|   | ||||
| @@ -86,7 +86,6 @@ class BaseQuerySet(object): | ||||
|         self._batch_size = None | ||||
|         self.only_fields = [] | ||||
|         self._max_time_ms = None | ||||
|         self._comment = None | ||||
|  | ||||
|     def __call__(self, q_obj=None, class_check=True, read_preference=None, | ||||
|                  **query): | ||||
| @@ -707,36 +706,39 @@ class BaseQuerySet(object): | ||||
|         with switch_db(self._document, alias) as cls: | ||||
|             collection = cls._get_collection() | ||||
|  | ||||
|         return self._clone_into(self.__class__(self._document, collection)) | ||||
|         return self.clone_into(self.__class__(self._document, collection)) | ||||
|  | ||||
|     def clone(self): | ||||
|         """Create a copy of the current queryset.""" | ||||
|         return self._clone_into(self.__class__(self._document, self._collection_obj)) | ||||
|         """Creates a copy of the current | ||||
|           :class:`~mongoengine.queryset.QuerySet` | ||||
|  | ||||
|     def _clone_into(self, new_qs): | ||||
|         """Copy all of the relevant properties of this queryset to | ||||
|         a new queryset (which has to be an instance of | ||||
|         :class:`~mongoengine.queryset.base.BaseQuerySet`). | ||||
|         .. versionadded:: 0.5 | ||||
|         """ | ||||
|         if not isinstance(new_qs, BaseQuerySet): | ||||
|         return self.clone_into(self.__class__(self._document, self._collection_obj)) | ||||
|  | ||||
|     def clone_into(self, cls): | ||||
|         """Creates a copy of the current | ||||
|           :class:`~mongoengine.queryset.base.BaseQuerySet` into another child class | ||||
|         """ | ||||
|         if not isinstance(cls, BaseQuerySet): | ||||
|             raise OperationError( | ||||
|                 '%s is not a subclass of BaseQuerySet' % new_qs.__name__) | ||||
|                 '%s is not a subclass of BaseQuerySet' % cls.__name__) | ||||
|  | ||||
|         copy_props = ('_mongo_query', '_initial_query', '_none', '_query_obj', | ||||
|                       '_where_clause', '_loaded_fields', '_ordering', '_snapshot', | ||||
|                       '_timeout', '_class_check', '_slave_okay', '_read_preference', | ||||
|                       '_iter', '_scalar', '_as_pymongo', '_as_pymongo_coerce', | ||||
|                       '_limit', '_skip', '_hint', '_auto_dereference', | ||||
|                       '_search_text', 'only_fields', '_max_time_ms', '_comment') | ||||
|                       '_search_text', 'only_fields', '_max_time_ms') | ||||
|  | ||||
|         for prop in copy_props: | ||||
|             val = getattr(self, prop) | ||||
|             setattr(new_qs, prop, copy.copy(val)) | ||||
|             setattr(cls, prop, copy.copy(val)) | ||||
|  | ||||
|         if self._cursor_obj: | ||||
|             new_qs._cursor_obj = self._cursor_obj.clone() | ||||
|             cls._cursor_obj = self._cursor_obj.clone() | ||||
|  | ||||
|         return new_qs | ||||
|         return cls | ||||
|  | ||||
|     def select_related(self, max_depth=1): | ||||
|         """Handles dereferencing of :class:`~bson.dbref.DBRef` objects or | ||||
| @@ -758,11 +760,7 @@ class BaseQuerySet(object): | ||||
|         """ | ||||
|         queryset = self.clone() | ||||
|         queryset._limit = n if n != 0 else 1 | ||||
|  | ||||
|         # If a cursor object has already been created, apply the limit to it. | ||||
|         if queryset._cursor_obj: | ||||
|             queryset._cursor_obj.limit(queryset._limit) | ||||
|  | ||||
|         # Return self to allow chaining | ||||
|         return queryset | ||||
|  | ||||
|     def skip(self, n): | ||||
| @@ -773,11 +771,6 @@ class BaseQuerySet(object): | ||||
|         """ | ||||
|         queryset = self.clone() | ||||
|         queryset._skip = n | ||||
|  | ||||
|         # If a cursor object has already been created, apply the skip to it. | ||||
|         if queryset._cursor_obj: | ||||
|             queryset._cursor_obj.skip(queryset._skip) | ||||
|  | ||||
|         return queryset | ||||
|  | ||||
|     def hint(self, index=None): | ||||
| @@ -795,11 +788,6 @@ class BaseQuerySet(object): | ||||
|         """ | ||||
|         queryset = self.clone() | ||||
|         queryset._hint = index | ||||
|  | ||||
|         # If a cursor object has already been created, apply the hint to it. | ||||
|         if queryset._cursor_obj: | ||||
|             queryset._cursor_obj.hint(queryset._hint) | ||||
|  | ||||
|         return queryset | ||||
|  | ||||
|     def batch_size(self, size): | ||||
| @@ -813,11 +801,6 @@ class BaseQuerySet(object): | ||||
|         """ | ||||
|         queryset = self.clone() | ||||
|         queryset._batch_size = size | ||||
|  | ||||
|         # If a cursor object has already been created, apply the batch size to it. | ||||
|         if queryset._cursor_obj: | ||||
|             queryset._cursor_obj.batch_size(queryset._batch_size) | ||||
|  | ||||
|         return queryset | ||||
|  | ||||
|     def distinct(self, field): | ||||
| @@ -917,24 +900,18 @@ class BaseQuerySet(object): | ||||
|         return self.fields(**fields) | ||||
|  | ||||
|     def fields(self, _only_called=False, **kwargs): | ||||
|         """Manipulate how you load this document's fields. Used by `.only()` | ||||
|         and `.exclude()` to manipulate which fields to retrieve. If called | ||||
|         directly, use a set of kwargs similar to the MongoDB projection | ||||
|         document. For example: | ||||
|         """Manipulate how you load this document's fields.  Used by `.only()` | ||||
|         and `.exclude()` to manipulate which fields to retrieve.  Fields also | ||||
|         allows for a greater level of control for example: | ||||
|  | ||||
|         Include only a subset of fields: | ||||
|         Retrieving a Subrange of Array Elements: | ||||
|  | ||||
|             posts = BlogPost.objects(...).fields(author=1, title=1) | ||||
|         You can use the $slice operator to retrieve a subrange of elements in | ||||
|         an array. For example to get the first 5 comments:: | ||||
|  | ||||
|         Exclude a specific field: | ||||
|             post = BlogPost.objects(...).fields(slice__comments=5) | ||||
|  | ||||
|             posts = BlogPost.objects(...).fields(comments=0) | ||||
|  | ||||
|         To retrieve a subrange of array elements: | ||||
|  | ||||
|             posts = BlogPost.objects(...).fields(slice__comments=5) | ||||
|  | ||||
|         :param kwargs: A set keywors arguments identifying what to include. | ||||
|         :param kwargs: A dictionary identifying what to include | ||||
|  | ||||
|         .. versionadded:: 0.5 | ||||
|         """ | ||||
| @@ -950,20 +927,7 @@ class BaseQuerySet(object): | ||||
|             key = '.'.join(parts) | ||||
|             cleaned_fields.append((key, value)) | ||||
|  | ||||
|         # Sort fields by their values, explicitly excluded fields first, then | ||||
|         # explicitly included, and then more complicated operators such as | ||||
|         # $slice. | ||||
|         def _sort_key(field_tuple): | ||||
|             key, value = field_tuple | ||||
|             if isinstance(value, (int)): | ||||
|                 return value  # 0 for exclusion, 1 for inclusion | ||||
|             else: | ||||
|                 return 2  # so that complex values appear last | ||||
|  | ||||
|         fields = sorted(cleaned_fields, key=_sort_key) | ||||
|  | ||||
|         # Clone the queryset, group all fields by their value, convert | ||||
|         # each of them to db_fields, and set the queryset's _loaded_fields | ||||
|         fields = sorted(cleaned_fields, key=operator.itemgetter(1)) | ||||
|         queryset = self.clone() | ||||
|         for value, group in itertools.groupby(fields, lambda x: x[1]): | ||||
|             fields = [field for field, value in group] | ||||
| @@ -989,31 +953,13 @@ class BaseQuerySet(object): | ||||
|     def order_by(self, *keys): | ||||
|         """Order the :class:`~mongoengine.queryset.QuerySet` by the keys. The | ||||
|         order may be specified by prepending each of the keys by a + or a -. | ||||
|         Ascending order is assumed. If no keys are passed, existing ordering | ||||
|         is cleared instead. | ||||
|         Ascending order is assumed. | ||||
|  | ||||
|         :param keys: fields to order the query results by; keys may be | ||||
|             prefixed with **+** or **-** to determine the ordering direction | ||||
|         """ | ||||
|         queryset = self.clone() | ||||
|  | ||||
|         old_ordering = queryset._ordering | ||||
|         new_ordering = queryset._get_order_by(keys) | ||||
|  | ||||
|         if queryset._cursor_obj: | ||||
|  | ||||
|             # If a cursor object has already been created, apply the sort to it | ||||
|             if new_ordering: | ||||
|                 queryset._cursor_obj.sort(new_ordering) | ||||
|  | ||||
|             # If we're trying to clear a previous explicit ordering, we need | ||||
|             # to clear the cursor entirely (because PyMongo doesn't allow | ||||
|             # clearing an existing sort on a cursor). | ||||
|             elif old_ordering: | ||||
|                 queryset._cursor_obj = None | ||||
|  | ||||
|         queryset._ordering = new_ordering | ||||
|  | ||||
|         queryset._ordering = queryset._get_order_by(keys) | ||||
|         return queryset | ||||
|  | ||||
|     def comment(self, text): | ||||
| @@ -1459,13 +1405,10 @@ class BaseQuerySet(object): | ||||
|             raise StopIteration | ||||
|  | ||||
|         raw_doc = self._cursor.next() | ||||
|  | ||||
|         if self._as_pymongo: | ||||
|             return self._get_as_pymongo(raw_doc) | ||||
|  | ||||
|         doc = self._document._from_son( | ||||
|             raw_doc, _auto_dereference=self._auto_dereference, | ||||
|             only_fields=self.only_fields) | ||||
|         doc = self._document._from_son(raw_doc, | ||||
|                                        _auto_dereference=self._auto_dereference, only_fields=self.only_fields) | ||||
|  | ||||
|         if self._scalar: | ||||
|             return self._get_scalar(doc) | ||||
| @@ -1475,6 +1418,7 @@ class BaseQuerySet(object): | ||||
|     def rewind(self): | ||||
|         """Rewind the cursor to its unevaluated state. | ||||
|  | ||||
|  | ||||
|         .. versionadded:: 0.3 | ||||
|         """ | ||||
|         self._iter = False | ||||
| @@ -1524,54 +1468,43 @@ class BaseQuerySet(object): | ||||
|  | ||||
|     @property | ||||
|     def _cursor(self): | ||||
|         """Return a PyMongo cursor object corresponding to this queryset.""" | ||||
|         if self._cursor_obj is None: | ||||
|  | ||||
|         # If _cursor_obj already exists, return it immediately. | ||||
|         if self._cursor_obj is not None: | ||||
|             return self._cursor_obj | ||||
|             # In PyMongo 3+, we define the read preference on a collection | ||||
|             # level, not a cursor level. Thus, we need to get a cloned | ||||
|             # collection object using `with_options` first. | ||||
|             if IS_PYMONGO_3 and self._read_preference is not None: | ||||
|                 self._cursor_obj = self._collection\ | ||||
|                     .with_options(read_preference=self._read_preference)\ | ||||
|                     .find(self._query, **self._cursor_args) | ||||
|             else: | ||||
|                 self._cursor_obj = self._collection.find(self._query, | ||||
|                                                          **self._cursor_args) | ||||
|             # Apply where clauses to cursor | ||||
|             if self._where_clause: | ||||
|                 where_clause = self._sub_js_fields(self._where_clause) | ||||
|                 self._cursor_obj.where(where_clause) | ||||
|  | ||||
|         # Create a new PyMongo cursor. | ||||
|         # XXX In PyMongo 3+, we define the read preference on a collection | ||||
|         # level, not a cursor level. Thus, we need to get a cloned collection | ||||
|         # object using `with_options` first. | ||||
|         if IS_PYMONGO_3 and self._read_preference is not None: | ||||
|             self._cursor_obj = self._collection\ | ||||
|                 .with_options(read_preference=self._read_preference)\ | ||||
|                 .find(self._query, **self._cursor_args) | ||||
|         else: | ||||
|             self._cursor_obj = self._collection.find(self._query, | ||||
|                                                      **self._cursor_args) | ||||
|         # Apply "where" clauses to cursor | ||||
|         if self._where_clause: | ||||
|             where_clause = self._sub_js_fields(self._where_clause) | ||||
|             self._cursor_obj.where(where_clause) | ||||
|             if self._ordering: | ||||
|                 # Apply query ordering | ||||
|                 self._cursor_obj.sort(self._ordering) | ||||
|             elif self._ordering is None and self._document._meta['ordering']: | ||||
|                 # Otherwise, apply the ordering from the document model, unless | ||||
|                 # it's been explicitly cleared via order_by with no arguments | ||||
|                 order = self._get_order_by(self._document._meta['ordering']) | ||||
|                 self._cursor_obj.sort(order) | ||||
|  | ||||
|         # Apply ordering to the cursor. | ||||
|         # XXX self._ordering can be equal to: | ||||
|         # * None if we didn't explicitly call order_by on this queryset. | ||||
|         # * A list of PyMongo-style sorting tuples. | ||||
|         # * An empty list if we explicitly called order_by() without any | ||||
|         #   arguments. This indicates that we want to clear the default | ||||
|         #   ordering. | ||||
|         if self._ordering: | ||||
|             # explicit ordering | ||||
|             self._cursor_obj.sort(self._ordering) | ||||
|         elif self._ordering is None and self._document._meta['ordering']: | ||||
|             # default ordering | ||||
|             order = self._get_order_by(self._document._meta['ordering']) | ||||
|             self._cursor_obj.sort(order) | ||||
|             if self._limit is not None: | ||||
|                 self._cursor_obj.limit(self._limit) | ||||
|  | ||||
|         if self._limit is not None: | ||||
|             self._cursor_obj.limit(self._limit) | ||||
|             if self._skip is not None: | ||||
|                 self._cursor_obj.skip(self._skip) | ||||
|  | ||||
|         if self._skip is not None: | ||||
|             self._cursor_obj.skip(self._skip) | ||||
|             if self._hint != -1: | ||||
|                 self._cursor_obj.hint(self._hint) | ||||
|  | ||||
|         if self._hint != -1: | ||||
|             self._cursor_obj.hint(self._hint) | ||||
|  | ||||
|         if self._batch_size is not None: | ||||
|             self._cursor_obj.batch_size(self._batch_size) | ||||
|             if self._batch_size is not None: | ||||
|                 self._cursor_obj.batch_size(self._batch_size) | ||||
|  | ||||
|         return self._cursor_obj | ||||
|  | ||||
| @@ -1746,13 +1679,7 @@ class BaseQuerySet(object): | ||||
|         return ret | ||||
|  | ||||
|     def _get_order_by(self, keys): | ||||
|         """Given a list of MongoEngine-style sort keys, return a list | ||||
|         of sorting tuples that can be applied to a PyMongo cursor. For | ||||
|         example: | ||||
|  | ||||
|         >>> qs._get_order_by(['-last_name', 'first_name']) | ||||
|         [('last_name', -1), ('first_name', 1)] | ||||
|         """ | ||||
|         """Creates a list of order by fields""" | ||||
|         key_list = [] | ||||
|         for key in keys: | ||||
|             if not key: | ||||
| @@ -1765,19 +1692,17 @@ class BaseQuerySet(object): | ||||
|             direction = pymongo.ASCENDING | ||||
|             if key[0] == '-': | ||||
|                 direction = pymongo.DESCENDING | ||||
|  | ||||
|             if key[0] in ('-', '+'): | ||||
|                 key = key[1:] | ||||
|  | ||||
|             key = key.replace('__', '.') | ||||
|             try: | ||||
|                 key = self._document._translate_field_name(key) | ||||
|             except Exception: | ||||
|                 # TODO this exception should be more specific | ||||
|                 pass | ||||
|  | ||||
|             key_list.append((key, direction)) | ||||
|  | ||||
|         if self._cursor_obj and key_list: | ||||
|             self._cursor_obj.sort(key_list) | ||||
|         return key_list | ||||
|  | ||||
|     def _get_scalar(self, doc): | ||||
| @@ -1875,21 +1800,10 @@ class BaseQuerySet(object): | ||||
|         return code | ||||
|  | ||||
|     def _chainable_method(self, method_name, val): | ||||
|         """Call a particular method on the PyMongo cursor call a particular chainable method | ||||
|         with the provided value. | ||||
|         """ | ||||
|         queryset = self.clone() | ||||
|  | ||||
|         # Get an existing cursor object or create a new one | ||||
|         cursor = queryset._cursor | ||||
|  | ||||
|         # Find the requested method on the cursor and call it with the | ||||
|         # provided value | ||||
|         getattr(cursor, method_name)(val) | ||||
|  | ||||
|         # Cache the value on the queryset._{method_name} | ||||
|         method = getattr(queryset._cursor, method_name) | ||||
|         method(val) | ||||
|         setattr(queryset, '_' + method_name, val) | ||||
|  | ||||
|         return queryset | ||||
|  | ||||
|     # Deprecated | ||||
|   | ||||
| @@ -136,15 +136,13 @@ class QuerySet(BaseQuerySet): | ||||
|         return self._len | ||||
|  | ||||
|     def no_cache(self): | ||||
|         """Convert to a non-caching queryset | ||||
|         """Convert to a non_caching queryset | ||||
|  | ||||
|         .. versionadded:: 0.8.3 Convert to non caching queryset | ||||
|         """ | ||||
|         if self._result_cache is not None: | ||||
|             raise OperationError('QuerySet already cached') | ||||
|  | ||||
|         return self._clone_into(QuerySetNoCache(self._document, | ||||
|                                                 self._collection)) | ||||
|         return self.clone_into(QuerySetNoCache(self._document, self._collection)) | ||||
|  | ||||
|  | ||||
| class QuerySetNoCache(BaseQuerySet): | ||||
| @@ -155,7 +153,7 @@ class QuerySetNoCache(BaseQuerySet): | ||||
|  | ||||
|         .. versionadded:: 0.8.3 Convert to caching queryset | ||||
|         """ | ||||
|         return self._clone_into(QuerySet(self._document, self._collection)) | ||||
|         return self.clone_into(QuerySet(self._document, self._collection)) | ||||
|  | ||||
|     def __repr__(self): | ||||
|         """Provides the string representation of the QuerySet | ||||
|   | ||||
| @@ -101,21 +101,8 @@ def query(_doc_cls=None, **kwargs): | ||||
|                         value = value['_id'] | ||||
|  | ||||
|             elif op in ('in', 'nin', 'all', 'near') and not isinstance(value, dict): | ||||
|                 # Raise an error if the in/nin/all/near param is not iterable. We need a | ||||
|                 # special check for BaseDocument, because - although it's iterable - using | ||||
|                 # it as such in the context of this method is most definitely a mistake. | ||||
|                 BaseDocument = _import_class('BaseDocument') | ||||
|                 if isinstance(value, BaseDocument): | ||||
|                     raise TypeError("When using the `in`, `nin`, `all`, or " | ||||
|                                     "`near`-operators you can\'t use a " | ||||
|                                     "`Document`, you must wrap your object " | ||||
|                                     "in a list (object -> [object]).") | ||||
|                 elif not hasattr(value, '__iter__'): | ||||
|                     raise TypeError("The `in`, `nin`, `all`, or " | ||||
|                                     "`near`-operators must be applied to an " | ||||
|                                     "iterable (e.g. a list).") | ||||
|                 else: | ||||
|                     value = [field.prepare_query_value(op, v) for v in value] | ||||
|                 # 'in', 'nin' and 'all' require a list of values | ||||
|                 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: | ||||
| @@ -233,7 +220,8 @@ def update(_doc_cls=None, **update): | ||||
|                 # Support decrement by flipping a positive value's sign | ||||
|                 # and using 'inc' | ||||
|                 op = 'inc' | ||||
|                 value = -value | ||||
|                 if value > 0: | ||||
|                     value = -value | ||||
|             elif op == 'add_to_set': | ||||
|                 op = 'addToSet' | ||||
|             elif op == 'set_on_insert': | ||||
|   | ||||
| @@ -7,5 +7,5 @@ cover-package=mongoengine | ||||
| [flake8] | ||||
| ignore=E501,F401,F403,F405,I201 | ||||
| exclude=build,dist,docs,venv,venv3,.tox,.eggs,tests | ||||
| max-complexity=47 | ||||
| max-complexity=45 | ||||
| application-import-names=mongoengine,tests | ||||
|   | ||||
| @@ -435,15 +435,6 @@ class InstanceTest(unittest.TestCase): | ||||
|  | ||||
|         person.to_dbref() | ||||
|  | ||||
|     def test_save_abstract_document(self): | ||||
|         """Saving an abstract document should fail.""" | ||||
|         class Doc(Document): | ||||
|             name = StringField() | ||||
|             meta = {'abstract': True} | ||||
|  | ||||
|         with self.assertRaises(InvalidDocumentError): | ||||
|             Doc(name='aaa').save() | ||||
|  | ||||
|     def test_reload(self): | ||||
|         """Ensure that attributes may be reloaded. | ||||
|         """ | ||||
| @@ -1232,19 +1223,6 @@ class InstanceTest(unittest.TestCase): | ||||
|         self.assertEqual(person.name, None) | ||||
|         self.assertEqual(person.age, None) | ||||
|  | ||||
|     def test_update_rename_operator(self): | ||||
|         """Test the $rename operator.""" | ||||
|         coll = self.Person._get_collection() | ||||
|         doc = self.Person(name='John').save() | ||||
|         raw_doc = coll.find_one({'_id': doc.pk}) | ||||
|         self.assertEqual(set(raw_doc.keys()), set(['_id', '_cls', 'name'])) | ||||
|  | ||||
|         doc.update(rename__name='first_name') | ||||
|         raw_doc = coll.find_one({'_id': doc.pk}) | ||||
|         self.assertEqual(set(raw_doc.keys()), | ||||
|                          set(['_id', '_cls', 'first_name'])) | ||||
|         self.assertEqual(raw_doc['first_name'], 'John') | ||||
|  | ||||
|     def test_inserts_if_you_set_the_pk(self): | ||||
|         p1 = self.Person(name='p1', id=bson.ObjectId()).save() | ||||
|         p2 = self.Person(name='p2') | ||||
| @@ -1882,10 +1860,6 @@ class InstanceTest(unittest.TestCase): | ||||
|                 'occurs': {"hello": None} | ||||
|             }) | ||||
|  | ||||
|         # Tests for issue #1438: https://github.com/MongoEngine/mongoengine/issues/1438 | ||||
|         with self.assertRaises(ValueError): | ||||
|             Word._from_son('this is not a valid SON dict') | ||||
|  | ||||
|     def test_reverse_delete_rule_cascade_and_nullify(self): | ||||
|         """Ensure that a referenced document is also deleted upon deletion. | ||||
|         """ | ||||
|   | ||||
| @@ -306,24 +306,6 @@ class FieldTest(unittest.TestCase): | ||||
|         person.id = '497ce96f395f2f052a494fd4' | ||||
|         person.validate() | ||||
|  | ||||
|     def test_db_field_validation(self): | ||||
|         """Ensure that db_field doesn't accept invalid values.""" | ||||
|  | ||||
|         # dot in the name | ||||
|         with self.assertRaises(ValueError): | ||||
|             class User(Document): | ||||
|                 name = StringField(db_field='user.name') | ||||
|  | ||||
|         # name starting with $ | ||||
|         with self.assertRaises(ValueError): | ||||
|             class User(Document): | ||||
|                 name = StringField(db_field='$name') | ||||
|  | ||||
|         # name containing a null character | ||||
|         with self.assertRaises(ValueError): | ||||
|             class User(Document): | ||||
|                 name = StringField(db_field='name\0') | ||||
|  | ||||
|     def test_string_validation(self): | ||||
|         """Ensure that invalid values cannot be assigned to string fields. | ||||
|         """ | ||||
| @@ -1060,7 +1042,6 @@ class FieldTest(unittest.TestCase): | ||||
|         self.assertEqual( | ||||
|             BlogPost.objects.filter(info__100__test__exact='test').count(), 0) | ||||
|  | ||||
|         # test queries by list | ||||
|         post = BlogPost() | ||||
|         post.info = ['1', '2'] | ||||
|         post.save() | ||||
| @@ -1072,248 +1053,6 @@ class FieldTest(unittest.TestCase): | ||||
|         post.info *= 2 | ||||
|         post.save() | ||||
|         self.assertEqual(BlogPost.objects(info=['1', '2', '3', '4', '1', '2', '3', '4']).count(), 1) | ||||
|  | ||||
|         BlogPost.drop_collection() | ||||
|  | ||||
|     def test_list_field_manipulative_operators(self): | ||||
|         """Ensure that ListField works with standard list operators that manipulate the list. | ||||
|         """ | ||||
|         class BlogPost(Document): | ||||
|             ref = StringField() | ||||
|             info = ListField(StringField()) | ||||
|  | ||||
|         BlogPost.drop_collection() | ||||
|  | ||||
|         post = BlogPost() | ||||
|         post.ref = "1234" | ||||
|         post.info = ['0', '1', '2', '3', '4', '5'] | ||||
|         post.save() | ||||
|  | ||||
|         def reset_post(): | ||||
|             post.info = ['0', '1', '2', '3', '4', '5'] | ||||
|             post.save() | ||||
|  | ||||
|         # '__add__(listB)' | ||||
|         # listA+listB | ||||
|         # operator.add(listA, listB) | ||||
|         reset_post() | ||||
|         temp = ['a', 'b'] | ||||
|         post.info = post.info + temp | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', 'a', 'b']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', 'a', 'b']) | ||||
|  | ||||
|         # '__delitem__(index)' | ||||
|         # aka 'del list[index]' | ||||
|         # aka 'operator.delitem(list, index)' | ||||
|         reset_post() | ||||
|         del post.info[2] # del from middle ('2') | ||||
|         self.assertEqual(post.info, ['0', '1', '3', '4', '5']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', '1', '3', '4', '5']) | ||||
|  | ||||
|         # '__delitem__(slice(i, j))' | ||||
|         # aka 'del list[i:j]' | ||||
|         # aka 'operator.delitem(list, slice(i,j))' | ||||
|         reset_post() | ||||
|         del post.info[1:3] # removes '1', '2' | ||||
|         self.assertEqual(post.info, ['0', '3', '4', '5']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', '3', '4', '5']) | ||||
|  | ||||
|         # '__iadd__' | ||||
|         # aka 'list += list' | ||||
|         reset_post() | ||||
|         temp = ['a', 'b'] | ||||
|         post.info += temp | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', 'a', 'b']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', 'a', 'b']) | ||||
|  | ||||
|         # '__imul__' | ||||
|         # aka 'list *= number' | ||||
|         reset_post() | ||||
|         post.info *= 2 | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', '0', '1', '2', '3', '4', '5']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', '0', '1', '2', '3', '4', '5']) | ||||
|  | ||||
|         # '__mul__' | ||||
|         # aka 'listA*listB' | ||||
|         reset_post() | ||||
|         post.info = post.info * 2 | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', '0', '1', '2', '3', '4', '5']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', '0', '1', '2', '3', '4', '5']) | ||||
|  | ||||
|         # '__rmul__' | ||||
|         # aka 'listB*listA' | ||||
|         reset_post() | ||||
|         post.info = 2 * post.info | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', '0', '1', '2', '3', '4', '5']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', '0', '1', '2', '3', '4', '5']) | ||||
|  | ||||
|         # '__setitem__(index, value)' | ||||
|         # aka 'list[index]=value' | ||||
|         # aka 'setitem(list, value)' | ||||
|         reset_post() | ||||
|         post.info[4] = 'a' | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', 'a', '5']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', 'a', '5']) | ||||
|  | ||||
|         # '__setitem__(slice(i, j), listB)' | ||||
|         # aka 'listA[i:j] = listB' | ||||
|         # aka 'setitem(listA, slice(i, j), listB)' | ||||
|         reset_post() | ||||
|         post.info[1:3] = ['h', 'e', 'l', 'l', 'o'] | ||||
|         self.assertEqual(post.info, ['0', 'h', 'e', 'l', 'l', 'o', '3', '4', '5']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', 'h', 'e', 'l', 'l', 'o', '3', '4', '5']) | ||||
|  | ||||
|         # 'append' | ||||
|         reset_post() | ||||
|         post.info.append('h') | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', 'h']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', 'h']) | ||||
|  | ||||
|         # 'extend' | ||||
|         reset_post() | ||||
|         post.info.extend(['h', 'e', 'l', 'l', 'o']) | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', 'h', 'e', 'l', 'l', 'o']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', '1', '2', '3', '4', '5', 'h', 'e', 'l', 'l', 'o']) | ||||
|         # 'insert' | ||||
|  | ||||
|         # 'pop' | ||||
|         reset_post() | ||||
|         x = post.info.pop(2) | ||||
|         y = post.info.pop() | ||||
|         self.assertEqual(post.info, ['0', '1', '3', '4']) | ||||
|         self.assertEqual(x, '2') | ||||
|         self.assertEqual(y, '5') | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', '1', '3', '4']) | ||||
|  | ||||
|         # 'remove' | ||||
|         reset_post() | ||||
|         post.info.remove('2') | ||||
|         self.assertEqual(post.info, ['0', '1', '3', '4', '5']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['0', '1', '3', '4', '5']) | ||||
|  | ||||
|         # 'reverse' | ||||
|         reset_post() | ||||
|         post.info.reverse() | ||||
|         self.assertEqual(post.info, ['5', '4', '3', '2', '1', '0']) | ||||
|         post.save() | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, ['5', '4', '3', '2', '1', '0']) | ||||
|  | ||||
|         # 'sort': though this operator method does manipulate the list, it is tested in | ||||
|         #     the 'test_list_field_lexicograpic_operators' function | ||||
|         BlogPost.drop_collection() | ||||
|  | ||||
|     def test_list_field_invalid_operators(self): | ||||
|         class BlogPost(Document): | ||||
|             ref = StringField() | ||||
|             info = ListField(StringField()) | ||||
|         post = BlogPost() | ||||
|         post.ref = "1234" | ||||
|         post.info = ['0', '1', '2', '3', '4', '5'] | ||||
|         # '__hash__' | ||||
|         # aka 'hash(list)' | ||||
|         # # assert TypeError | ||||
|         self.assertRaises(TypeError, lambda: hash(post.info)) | ||||
|  | ||||
|     def test_list_field_lexicographic_operators(self): | ||||
|         """Ensure that ListField works with standard list operators that do lexigraphic ordering. | ||||
|         """ | ||||
|         class BlogPost(Document): | ||||
|             ref = StringField() | ||||
|             text_info = ListField(StringField()) | ||||
|             oid_info = ListField(ObjectIdField()) | ||||
|             bool_info = ListField(BooleanField()) | ||||
|         BlogPost.drop_collection() | ||||
|  | ||||
|         blogSmall = BlogPost(ref="small") | ||||
|         blogSmall.text_info = ["a", "a", "a"] | ||||
|         blogSmall.bool_info = [False, False] | ||||
|         blogSmall.save() | ||||
|         blogSmall.reload() | ||||
|  | ||||
|         blogLargeA = BlogPost(ref="big") | ||||
|         blogLargeA.text_info = ["a", "z", "j"] | ||||
|         blogLargeA.bool_info = [False, True] | ||||
|         blogLargeA.save() | ||||
|         blogLargeA.reload() | ||||
|  | ||||
|         blogLargeB = BlogPost(ref="big2") | ||||
|         blogLargeB.text_info = ["a", "z", "j"] | ||||
|         blogLargeB.oid_info = [ | ||||
|             "54495ad94c934721ede76f90", | ||||
|             "54495ad94c934721ede76d23", | ||||
|             "54495ad94c934721ede76d00" | ||||
|         ] | ||||
|         blogLargeB.bool_info = [False, True] | ||||
|         blogLargeB.save() | ||||
|         blogLargeB.reload() | ||||
|         # '__eq__' aka '==' | ||||
|         self.assertEqual(blogLargeA.text_info, blogLargeB.text_info) | ||||
|         self.assertEqual(blogLargeA.bool_info, blogLargeB.bool_info) | ||||
|         # '__ge__' aka '>=' | ||||
|         self.assertGreaterEqual(blogLargeA.text_info, blogSmall.text_info) | ||||
|         self.assertGreaterEqual(blogLargeA.text_info, blogLargeB.text_info) | ||||
|         self.assertGreaterEqual(blogLargeA.bool_info, blogSmall.bool_info) | ||||
|         self.assertGreaterEqual(blogLargeA.bool_info, blogLargeB.bool_info) | ||||
|         # '__gt__' aka '>' | ||||
|         self.assertGreaterEqual(blogLargeA.text_info, blogSmall.text_info) | ||||
|         self.assertGreaterEqual(blogLargeA.bool_info, blogSmall.bool_info) | ||||
|         # '__le__' aka '<=' | ||||
|         self.assertLessEqual(blogSmall.text_info, blogLargeB.text_info) | ||||
|         self.assertLessEqual(blogLargeA.text_info, blogLargeB.text_info) | ||||
|         self.assertLessEqual(blogSmall.bool_info, blogLargeB.bool_info) | ||||
|         self.assertLessEqual(blogLargeA.bool_info, blogLargeB.bool_info) | ||||
|         # '__lt__' aka '<' | ||||
|         self.assertLess(blogSmall.text_info, blogLargeB.text_info) | ||||
|         self.assertLess(blogSmall.bool_info, blogLargeB.bool_info) | ||||
|         # '__ne__' aka '!=' | ||||
|         self.assertNotEqual(blogSmall.text_info, blogLargeB.text_info) | ||||
|         self.assertNotEqual(blogSmall.bool_info, blogLargeB.bool_info) | ||||
|         # 'sort' | ||||
|         blogLargeB.bool_info = [True, False, True, False] | ||||
|         blogLargeB.text_info.sort() | ||||
|         blogLargeB.oid_info.sort() | ||||
|         blogLargeB.bool_info.sort() | ||||
|         sorted_target_list = [ | ||||
|             ObjectId("54495ad94c934721ede76d00"), | ||||
|             ObjectId("54495ad94c934721ede76d23"), | ||||
|             ObjectId("54495ad94c934721ede76f90") | ||||
|         ] | ||||
|         self.assertEqual(blogLargeB.text_info, ["a", "j", "z"]) | ||||
|         self.assertEqual(blogLargeB.oid_info, sorted_target_list) | ||||
|         self.assertEqual(blogLargeB.bool_info, [False, False, True, True]) | ||||
|         blogLargeB.save() | ||||
|         blogLargeB.reload() | ||||
|         self.assertEqual(blogLargeB.text_info, ["a", "j", "z"]) | ||||
|         self.assertEqual(blogLargeB.oid_info, sorted_target_list) | ||||
|         self.assertEqual(blogLargeB.bool_info, [False, False, True, True]) | ||||
|  | ||||
|         BlogPost.drop_collection() | ||||
|  | ||||
|     def test_list_assignment(self): | ||||
| @@ -1363,6 +1102,7 @@ class FieldTest(unittest.TestCase): | ||||
|         post.reload() | ||||
|         self.assertEqual(post.info, [1, 2, 3, 4, 'n5']) | ||||
|  | ||||
|  | ||||
|     def test_list_field_passed_in_value(self): | ||||
|         class Foo(Document): | ||||
|             bars = ListField(ReferenceField("Bar")) | ||||
| @@ -1985,7 +1725,7 @@ class FieldTest(unittest.TestCase): | ||||
|         self.assertEqual(content, User.objects.first().groups[0].content) | ||||
|  | ||||
|     def test_reference_miss(self): | ||||
|         """Ensure an exception is raised when dereferencing unknown document | ||||
|         """Ensure an exception is raised when dereferencing unknow document | ||||
|         """ | ||||
|  | ||||
|         class Foo(Document): | ||||
| @@ -3991,25 +3731,30 @@ class FieldTest(unittest.TestCase): | ||||
|         """Tests if a `FieldDoesNotExist` exception is raised when trying to | ||||
|         instanciate a document with a field that's not defined. | ||||
|         """ | ||||
|         class Doc(Document): | ||||
|             foo = StringField() | ||||
|  | ||||
|         with self.assertRaises(FieldDoesNotExist): | ||||
|         class Doc(Document): | ||||
|             foo = StringField(db_field='f') | ||||
|  | ||||
|         def test(): | ||||
|             Doc(bar='test') | ||||
|  | ||||
|         self.assertRaises(FieldDoesNotExist, test) | ||||
|  | ||||
|     def test_undefined_field_exception_with_strict(self): | ||||
|         """Tests if a `FieldDoesNotExist` exception is raised when trying to | ||||
|         instanciate a document with a field that's not defined, | ||||
|         even when strict is set to False. | ||||
|         """ | ||||
|  | ||||
|         class Doc(Document): | ||||
|             foo = StringField() | ||||
|             foo = StringField(db_field='f') | ||||
|             meta = {'strict': False} | ||||
|  | ||||
|         with self.assertRaises(FieldDoesNotExist): | ||||
|         def test(): | ||||
|             Doc(bar='test') | ||||
|  | ||||
|         self.assertRaises(FieldDoesNotExist, test) | ||||
|  | ||||
|     def test_long_field_is_considered_as_int64(self): | ||||
|         """ | ||||
|         Tests that long fields are stored as long in mongo, even if long value | ||||
|   | ||||
| @@ -141,16 +141,6 @@ class OnlyExcludeAllTest(unittest.TestCase): | ||||
|         self.assertEqual(qs._loaded_fields.as_dict(), | ||||
|                          {'b': {'$slice': 5}}) | ||||
|  | ||||
|     def test_mix_slice_with_other_fields(self): | ||||
|         class MyDoc(Document): | ||||
|             a = ListField() | ||||
|             b = ListField() | ||||
|             c = ListField() | ||||
|  | ||||
|         qs = MyDoc.objects.fields(a=1, b=0, slice__c=2) | ||||
|         self.assertEqual(qs._loaded_fields.as_dict(), | ||||
|                          {'c': {'$slice': 2}, 'a': 1}) | ||||
|  | ||||
|     def test_only(self): | ||||
|         """Ensure that QuerySet.only only returns the requested fields. | ||||
|         """ | ||||
|   | ||||
| @@ -106,111 +106,58 @@ class QuerySetTest(unittest.TestCase): | ||||
|             list(BlogPost.objects(author2__name="test")) | ||||
|  | ||||
|     def test_find(self): | ||||
|         """Ensure that a query returns a valid set of results.""" | ||||
|         user_a = self.Person.objects.create(name='User A', age=20) | ||||
|         user_b = self.Person.objects.create(name='User B', age=30) | ||||
|         """Ensure that a query returns a valid set of results. | ||||
|         """ | ||||
|         self.Person(name="User A", age=20).save() | ||||
|         self.Person(name="User B", age=30).save() | ||||
|  | ||||
|         # Find all people in the collection | ||||
|         people = self.Person.objects | ||||
|         self.assertEqual(people.count(), 2) | ||||
|         results = list(people) | ||||
|  | ||||
|         self.assertTrue(isinstance(results[0], self.Person)) | ||||
|         self.assertTrue(isinstance(results[0].id, (ObjectId, str, unicode))) | ||||
|  | ||||
|         self.assertEqual(results[0], user_a) | ||||
|         self.assertEqual(results[0].name, 'User A') | ||||
|         self.assertEqual(results[0].name, "User A") | ||||
|         self.assertEqual(results[0].age, 20) | ||||
|  | ||||
|         self.assertEqual(results[1], user_b) | ||||
|         self.assertEqual(results[1].name, 'User B') | ||||
|         self.assertEqual(results[1].name, "User B") | ||||
|         self.assertEqual(results[1].age, 30) | ||||
|  | ||||
|         # Filter people by age | ||||
|         # Use a query to filter the people found to just person1 | ||||
|         people = self.Person.objects(age=20) | ||||
|         self.assertEqual(people.count(), 1) | ||||
|         person = people.next() | ||||
|         self.assertEqual(person, user_a) | ||||
|         self.assertEqual(person.name, "User A") | ||||
|         self.assertEqual(person.age, 20) | ||||
|  | ||||
|     def test_limit(self): | ||||
|         """Ensure that QuerySet.limit works as expected.""" | ||||
|         user_a = self.Person.objects.create(name='User A', age=20) | ||||
|         user_b = self.Person.objects.create(name='User B', age=30) | ||||
|  | ||||
|         # Test limit on a new queryset | ||||
|         # Test limit | ||||
|         people = list(self.Person.objects.limit(1)) | ||||
|         self.assertEqual(len(people), 1) | ||||
|         self.assertEqual(people[0], user_a) | ||||
|         self.assertEqual(people[0].name, 'User A') | ||||
|  | ||||
|         # Test limit on an existing queryset | ||||
|         people = self.Person.objects | ||||
|         self.assertEqual(len(people), 2) | ||||
|         people2 = people.limit(1) | ||||
|         self.assertEqual(len(people), 2) | ||||
|         self.assertEqual(len(people2), 1) | ||||
|         self.assertEqual(people2[0], user_a) | ||||
|  | ||||
|         # Test chaining of only after limit | ||||
|         person = self.Person.objects().limit(1).only('name').first() | ||||
|         self.assertEqual(person, user_a) | ||||
|         self.assertEqual(person.name, 'User A') | ||||
|         self.assertEqual(person.age, None) | ||||
|  | ||||
|     def test_skip(self): | ||||
|         """Ensure that QuerySet.skip works as expected.""" | ||||
|         user_a = self.Person.objects.create(name='User A', age=20) | ||||
|         user_b = self.Person.objects.create(name='User B', age=30) | ||||
|  | ||||
|         # Test skip on a new queryset | ||||
|         # Test skip | ||||
|         people = list(self.Person.objects.skip(1)) | ||||
|         self.assertEqual(len(people), 1) | ||||
|         self.assertEqual(people[0], user_b) | ||||
|         self.assertEqual(people[0].name, 'User B') | ||||
|  | ||||
|         # Test skip on an existing queryset | ||||
|         people = self.Person.objects | ||||
|         self.assertEqual(len(people), 2) | ||||
|         people2 = people.skip(1) | ||||
|         self.assertEqual(len(people), 2) | ||||
|         self.assertEqual(len(people2), 1) | ||||
|         self.assertEqual(people2[0], user_b) | ||||
|  | ||||
|         # Test chaining of only after skip | ||||
|         person = self.Person.objects().skip(1).only('name').first() | ||||
|         self.assertEqual(person, user_b) | ||||
|         self.assertEqual(person.name, 'User B') | ||||
|         self.assertEqual(person.age, None) | ||||
|  | ||||
|     def test_slice(self): | ||||
|         """Ensure slicing a queryset works as expected.""" | ||||
|         user_a = self.Person.objects.create(name='User A', age=20) | ||||
|         user_b = self.Person.objects.create(name='User B', age=30) | ||||
|         user_c = self.Person.objects.create(name="User C", age=40) | ||||
|         person3 = self.Person(name="User C", age=40) | ||||
|         person3.save() | ||||
|  | ||||
|         # Test slice limit | ||||
|         people = list(self.Person.objects[:2]) | ||||
|         self.assertEqual(len(people), 2) | ||||
|         self.assertEqual(people[0], user_a) | ||||
|         self.assertEqual(people[1], user_b) | ||||
|         self.assertEqual(people[0].name, 'User A') | ||||
|         self.assertEqual(people[1].name, 'User B') | ||||
|  | ||||
|         # Test slice skip | ||||
|         people = list(self.Person.objects[1:]) | ||||
|         self.assertEqual(len(people), 2) | ||||
|         self.assertEqual(people[0], user_b) | ||||
|         self.assertEqual(people[1], user_c) | ||||
|         self.assertEqual(people[0].name, 'User B') | ||||
|         self.assertEqual(people[1].name, 'User C') | ||||
|  | ||||
|         # Test slice limit and skip | ||||
|         people = list(self.Person.objects[1:2]) | ||||
|         self.assertEqual(len(people), 1) | ||||
|         self.assertEqual(people[0], user_b) | ||||
|  | ||||
|         # Test slice limit and skip on an existing queryset | ||||
|         people = self.Person.objects | ||||
|         self.assertEqual(len(people), 3) | ||||
|         people2 = people[1:2] | ||||
|         self.assertEqual(len(people2), 1) | ||||
|         self.assertEqual(people2[0], user_b) | ||||
|         self.assertEqual(people[0].name, 'User B') | ||||
|  | ||||
|         # Test slice limit and skip cursor reset | ||||
|         qs = self.Person.objects[1:2] | ||||
| @@ -221,7 +168,6 @@ class QuerySetTest(unittest.TestCase): | ||||
|         self.assertEqual(len(people), 1) | ||||
|         self.assertEqual(people[0].name, 'User B') | ||||
|  | ||||
|         # Test empty slice | ||||
|         people = list(self.Person.objects[1:1]) | ||||
|         self.assertEqual(len(people), 0) | ||||
|  | ||||
| @@ -241,6 +187,12 @@ class QuerySetTest(unittest.TestCase): | ||||
|         self.assertEqual("[<Person: Person object>, <Person: Person object>]", | ||||
|                          "%s" % self.Person.objects[51:53]) | ||||
|  | ||||
|         # Test only after limit | ||||
|         self.assertEqual(self.Person.objects().limit(2).only('name')[0].age, None) | ||||
|  | ||||
|         # Test only after skip | ||||
|         self.assertEqual(self.Person.objects().skip(2).only('name')[0].age, None) | ||||
|  | ||||
|     def test_find_one(self): | ||||
|         """Ensure that a query using find_one returns a valid result. | ||||
|         """ | ||||
| @@ -1274,7 +1226,6 @@ class QuerySetTest(unittest.TestCase): | ||||
|  | ||||
|         BlogPost.drop_collection() | ||||
|  | ||||
|         # default ordering should be used by default | ||||
|         with db_ops_tracker() as q: | ||||
|             BlogPost.objects.filter(title='whatever').first() | ||||
|             self.assertEqual(len(q.get_ops()), 1) | ||||
| @@ -1283,28 +1234,11 @@ class QuerySetTest(unittest.TestCase): | ||||
|                 {'published_date': -1} | ||||
|             ) | ||||
|  | ||||
|         # calling order_by() should clear the default ordering | ||||
|         with db_ops_tracker() as q: | ||||
|             BlogPost.objects.filter(title='whatever').order_by().first() | ||||
|             self.assertEqual(len(q.get_ops()), 1) | ||||
|             self.assertFalse('$orderby' in q.get_ops()[0]['query']) | ||||
|  | ||||
|         # calling an explicit order_by should use a specified sort | ||||
|         with db_ops_tracker() as q: | ||||
|             BlogPost.objects.filter(title='whatever').order_by('published_date').first() | ||||
|             self.assertEqual(len(q.get_ops()), 1) | ||||
|             self.assertEqual( | ||||
|                 q.get_ops()[0]['query']['$orderby'], | ||||
|                 {'published_date': 1} | ||||
|             ) | ||||
|  | ||||
|         # calling order_by() after an explicit sort should clear it | ||||
|         with db_ops_tracker() as q: | ||||
|             qs = BlogPost.objects.filter(title='whatever').order_by('published_date') | ||||
|             qs.order_by().first() | ||||
|             self.assertEqual(len(q.get_ops()), 1) | ||||
|             self.assertFalse('$orderby' in q.get_ops()[0]['query']) | ||||
|  | ||||
|     def test_no_ordering_for_get(self): | ||||
|         """ Ensure that Doc.objects.get doesn't use any ordering. | ||||
|         """ | ||||
| @@ -1332,7 +1266,7 @@ class QuerySetTest(unittest.TestCase): | ||||
|  | ||||
|     def test_find_embedded(self): | ||||
|         """Ensure that an embedded document is properly returned from | ||||
|         different manners of querying. | ||||
|         a query. | ||||
|         """ | ||||
|         class User(EmbeddedDocument): | ||||
|             name = StringField() | ||||
| @@ -1343,9 +1277,8 @@ class QuerySetTest(unittest.TestCase): | ||||
|  | ||||
|         BlogPost.drop_collection() | ||||
|  | ||||
|         user = User(name='Test User') | ||||
|         BlogPost.objects.create( | ||||
|             author=user, | ||||
|             author=User(name='Test User'), | ||||
|             content='Had a good coffee today...' | ||||
|         ) | ||||
|  | ||||
| @@ -1353,19 +1286,6 @@ class QuerySetTest(unittest.TestCase): | ||||
|         self.assertTrue(isinstance(result.author, User)) | ||||
|         self.assertEqual(result.author.name, 'Test User') | ||||
|  | ||||
|         result = BlogPost.objects.get(author__name=user.name) | ||||
|         self.assertTrue(isinstance(result.author, User)) | ||||
|         self.assertEqual(result.author.name, 'Test User') | ||||
|  | ||||
|         result = BlogPost.objects.get(author={'name': user.name}) | ||||
|         self.assertTrue(isinstance(result.author, User)) | ||||
|         self.assertEqual(result.author.name, 'Test User') | ||||
|  | ||||
|         # Fails, since the string is not a type that is able to represent the | ||||
|         # author's document structure (should be dict) | ||||
|         with self.assertRaises(InvalidQueryError): | ||||
|             BlogPost.objects.get(author=user.name) | ||||
|  | ||||
|     def test_find_empty_embedded(self): | ||||
|         """Ensure that you can save and find an empty embedded document.""" | ||||
|         class User(EmbeddedDocument): | ||||
| @@ -1892,11 +1812,6 @@ class QuerySetTest(unittest.TestCase): | ||||
|         post.reload() | ||||
|         self.assertEqual(post.hits, 10) | ||||
|  | ||||
|         # Negative dec operator is equal to a positive inc operator | ||||
|         BlogPost.objects.update_one(dec__hits=-1) | ||||
|         post.reload() | ||||
|         self.assertEqual(post.hits, 11) | ||||
|  | ||||
|         BlogPost.objects.update(push__tags='mongo') | ||||
|         post.reload() | ||||
|         self.assertTrue('mongo' in post.tags) | ||||
| @@ -5048,35 +4963,6 @@ class QuerySetTest(unittest.TestCase): | ||||
|         self.assertEqual(i, 249) | ||||
|         self.assertEqual(j, 249) | ||||
|  | ||||
|     def test_in_operator_on_non_iterable(self): | ||||
|         """Ensure that using the `__in` operator on a non-iterable raises an | ||||
|         error. | ||||
|         """ | ||||
|         class User(Document): | ||||
|             name = StringField() | ||||
|  | ||||
|         class BlogPost(Document): | ||||
|             content = StringField() | ||||
|             authors = ListField(ReferenceField(User)) | ||||
|  | ||||
|         User.drop_collection() | ||||
|         BlogPost.drop_collection() | ||||
|  | ||||
|         author = User.objects.create(name='Test User') | ||||
|         post = BlogPost.objects.create(content='Had a good coffee today...', | ||||
|                                        authors=[author]) | ||||
|  | ||||
|         # Make sure using `__in` with a list works | ||||
|         blog_posts = BlogPost.objects(authors__in=[author]) | ||||
|         self.assertEqual(list(blog_posts), [post]) | ||||
|  | ||||
|         # Using `__in` with a non-iterable should raise a TypeError | ||||
|         self.assertRaises(TypeError, BlogPost.objects(authors__in=author.pk).count) | ||||
|  | ||||
|         # Using `__in` with a `Document` (which is seemingly iterable but not | ||||
|         # in a way we'd expect) should raise a TypeError, too | ||||
|         self.assertRaises(TypeError, BlogPost.objects(authors__in=author).count) | ||||
|  | ||||
|  | ||||
| if __name__ == '__main__': | ||||
|     unittest.main() | ||||
|   | ||||
| @@ -200,19 +200,6 @@ class ConnectionTest(unittest.TestCase): | ||||
|         self.assertTrue(isinstance(db, pymongo.database.Database)) | ||||
|         self.assertEqual(db.name, 'test') | ||||
|  | ||||
|     def test_connect_uri_with_replicaset(self): | ||||
|         """Ensure connect() works when specifying a replicaSet.""" | ||||
|         if IS_PYMONGO_3: | ||||
|             c = connect(host='mongodb://localhost/test?replicaSet=local-rs') | ||||
|             db = get_db() | ||||
|             self.assertTrue(isinstance(db, pymongo.database.Database)) | ||||
|             self.assertEqual(db.name, 'test') | ||||
|         else: | ||||
|             # PyMongo < v3.x raises an exception: | ||||
|             # "localhost:27017 is not a member of replica set local-rs" | ||||
|             with self.assertRaises(MongoEngineConnectionError): | ||||
|                 c = connect(host='mongodb://localhost/test?replicaSet=local-rs') | ||||
|  | ||||
|     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. | ||||
| @@ -296,19 +283,6 @@ class ConnectionTest(unittest.TestCase): | ||||
|         conn = get_connection('t2') | ||||
|         self.assertFalse(get_tz_awareness(conn)) | ||||
|  | ||||
|     def test_write_concern(self): | ||||
|         """Ensure write concern can be specified in connect() via | ||||
|         a kwarg or as part of the connection URI. | ||||
|         """ | ||||
|         conn1 = connect(alias='conn1', host='mongodb://localhost/testing?w=1&j=true') | ||||
|         conn2 = connect('testing', alias='conn2', w=1, j=True) | ||||
|         if IS_PYMONGO_3: | ||||
|             self.assertEqual(conn1.write_concern.document, {'w': 1, 'j': True}) | ||||
|             self.assertEqual(conn2.write_concern.document, {'w': 1, 'j': True}) | ||||
|         else: | ||||
|             self.assertEqual(dict(conn1.write_concern), {'w': 1, 'j': True}) | ||||
|             self.assertEqual(dict(conn2.write_concern), {'w': 1, 'j': True}) | ||||
|  | ||||
|     def test_datetime(self): | ||||
|         connect('mongoenginetest', tz_aware=True) | ||||
|         d = datetime.datetime(2010, 5, 5, tzinfo=utc) | ||||
|   | ||||
		Reference in New Issue
	
	Block a user