Compare commits
	
		
			29 Commits
		
	
	
		
			fix-querys
			...
			test-repli
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 2c247869f0 | ||
|  | 627cf90de0 | ||
|  | 2bedb36d7f | ||
|  | e93a95d0cb | ||
|  | 3f31666796 | ||
|  | 3fe8031cf3 | ||
|  | b27c7ce11b | ||
|  | ed34c2ca68 | ||
|  | 3ca2e953fb | ||
|  | d8a7328365 | ||
|  | f33cd625bf | ||
|  | 80530bb13c | ||
|  | affc12df4b | ||
|  | 4eedf00025 | ||
|  | e5acbcc0dd | ||
|  | 1b6743ee53 | ||
|  | b5fb82d95d | ||
|  | 193aa4e1f2 | ||
|  | ebd34427c7 | ||
|  | 3d75573889 | ||
|  | c6240ca415 | ||
|  | 2ee8984b44 | ||
|  | b7ec587e5b | ||
|  | 47c58bce2b | ||
|  | 96e95ac533 | ||
|  | b013a065f7 | ||
|  | 74b37d11cf | ||
|  | c6cc013617 | ||
|  | e07cb82c15 | 
							
								
								
									
										23
									
								
								.install_mongodb_on_travis.sh
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								.install_mongodb_on_travis.sh
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | #!/bin/bash | ||||||
|  |  | ||||||
|  | sudo apt-key adv --keyserver hkp://keyserver.ubuntu.com:80 --recv 7F0CEB10 | ||||||
|  |  | ||||||
|  | if [ "$MONGODB" = "2.4" ]; then | ||||||
|  |     echo "deb http://downloads-distro.mongodb.org/repo/ubuntu-upstart dist 10gen" | sudo tee /etc/apt/sources.list.d/mongodb.list | ||||||
|  |     sudo apt-get update | ||||||
|  |     sudo apt-get install mongodb-10gen=2.4.14 | ||||||
|  |     sudo service mongodb start | ||||||
|  | elif [ "$MONGODB" = "2.6" ]; then | ||||||
|  |     echo "deb http://downloads-distro.mongodb.org/repo/ubuntu-upstart dist 10gen" | sudo tee /etc/apt/sources.list.d/mongodb.list | ||||||
|  |     sudo apt-get update | ||||||
|  |     sudo apt-get install mongodb-org-server=2.6.12 | ||||||
|  |     # service should be started automatically | ||||||
|  | elif [ "$MONGODB" = "3.0" ]; then | ||||||
|  |     echo "deb http://repo.mongodb.org/apt/ubuntu precise/mongodb-org/3.0 multiverse" | sudo tee /etc/apt/sources.list.d/mongodb.list | ||||||
|  |     sudo apt-get update | ||||||
|  |     sudo apt-get install mongodb-org-server=3.0.14 | ||||||
|  |     # service should be started automatically | ||||||
|  | else | ||||||
|  |     echo "Invalid MongoDB version, expected 2.4, 2.6, or 3.0." | ||||||
|  |     exit 1 | ||||||
|  | fi; | ||||||
							
								
								
									
										67
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										67
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -1,28 +1,48 @@ | |||||||
|  | # For full coverage, we'd have to test all supported Python, MongoDB, and | ||||||
|  | # PyMongo combinations. However, that would result in an overly long build | ||||||
|  | # with a very large number of jobs, hence we only test a subset of all the | ||||||
|  | # combinations: | ||||||
|  | # * MongoDB v2.4 & v3.0 are only tested against Python v2.7 & v3.5. | ||||||
|  | # * MongoDB v2.4 is tested against PyMongo v2.7 & v3.x. | ||||||
|  | # * MongoDB v3.0 is tested against PyMongo v3.x. | ||||||
|  | # * MongoDB v2.6 is currently the "main" version tested against Python v2.7, | ||||||
|  | #   v3.5, PyPy & PyPy3, and PyMongo v2.7, v2.8 & v3.x. | ||||||
|  | # | ||||||
|  | # Reminder: Update README.rst if you change MongoDB versions we test. | ||||||
|  |  | ||||||
| language: python | language: python | ||||||
|  |  | ||||||
| python: | python: | ||||||
| - '2.7' | - 2.7 | ||||||
| - '3.3' | - 3.5 | ||||||
| - '3.4' |  | ||||||
| - '3.5' |  | ||||||
| - pypy | - pypy | ||||||
| - pypy3 | - pypy3 | ||||||
|  |  | ||||||
| env: | env: | ||||||
| - PYMONGO=2.7 | - MONGODB=2.6 PYMONGO=2.7 | ||||||
| - PYMONGO=2.8 | - MONGODB=2.6 PYMONGO=2.8 | ||||||
| - PYMONGO=3.0 | - MONGODB=2.6 PYMONGO=3.0 | ||||||
| - PYMONGO=dev |  | ||||||
|  |  | ||||||
| matrix: | matrix: | ||||||
|  |   # Finish the build as soon as one job fails | ||||||
|   fast_finish: true |   fast_finish: true | ||||||
|  |  | ||||||
|  |   include: | ||||||
|  |   - python: 2.7 | ||||||
|  |     env: MONGODB=2.4 PYMONGO=2.7 | ||||||
|  |   - python: 2.7 | ||||||
|  |     env: MONGODB=2.4 PYMONGO=3.0 | ||||||
|  |   - python: 2.7 | ||||||
|  |     env: MONGODB=3.0 PYMONGO=3.0 | ||||||
|  |   - python: 3.5 | ||||||
|  |     env: MONGODB=2.4 PYMONGO=2.7 | ||||||
|  |   - python: 3.5 | ||||||
|  |     env: MONGODB=2.4 PYMONGO=3.0 | ||||||
|  |   - python: 3.5 | ||||||
|  |     env: MONGODB=3.0 PYMONGO=3.0 | ||||||
|  |  | ||||||
| before_install: | before_install: | ||||||
| - travis_retry sudo apt-key adv --keyserver hkp://keyserver.ubuntu.com:80 --recv 7F0CEB10 | - bash .install_mongodb_on_travis.sh | ||||||
| - echo 'deb http://downloads-distro.mongodb.org/repo/ubuntu-upstart dist 10gen' | |  | ||||||
|   sudo tee /etc/apt/sources.list.d/mongodb.list |  | ||||||
| - travis_retry sudo apt-get update |  | ||||||
| - travis_retry sudo apt-get install mongodb-org-server |  | ||||||
|  |  | ||||||
| install: | install: | ||||||
| - sudo apt-get install python-dev python3-dev libopenjpeg-dev zlib1g-dev libjpeg-turbo8-dev | - sudo apt-get install python-dev python3-dev libopenjpeg-dev zlib1g-dev libjpeg-turbo8-dev | ||||||
| @@ -30,14 +50,17 @@ install: | |||||||
|   python-tk |   python-tk | ||||||
| - travis_retry pip install --upgrade pip | - travis_retry pip install --upgrade pip | ||||||
| - travis_retry pip install coveralls | - travis_retry pip install coveralls | ||||||
| - travis_retry pip install flake8 | - travis_retry pip install flake8 flake8-import-order | ||||||
| - travis_retry pip install tox>=1.9 | - travis_retry pip install tox>=1.9 | ||||||
| - travis_retry pip install "virtualenv<14.0.0"  # virtualenv>=14.0.0 has dropped Python 3.2 support (and pypy3 is based on py32) | - travis_retry pip install "virtualenv<14.0.0"  # virtualenv>=14.0.0 has dropped Python 3.2 support (and pypy3 is based on py32) | ||||||
| - travis_retry tox -e $(echo py$TRAVIS_PYTHON_VERSION-mg$PYMONGO | tr -d . | sed -e 's/pypypy/pypy/') -- -e test | - travis_retry tox -e $(echo py$TRAVIS_PYTHON_VERSION-mg$PYMONGO | tr -d . | sed -e 's/pypypy/pypy/') -- -e test | ||||||
|  |  | ||||||
|  | # Cache dependencies installed via pip | ||||||
|  | cache: pip | ||||||
|  |  | ||||||
| # Run flake8 for py27 | # Run flake8 for py27 | ||||||
| before_script: | before_script: | ||||||
| - if [[ $TRAVIS_PYTHON_VERSION == '2.7' ]]; then tox -e flake8; fi | - if [[ $TRAVIS_PYTHON_VERSION == '2.7' ]]; then flake8 .; else echo "flake8 only runs on py27"; fi | ||||||
|  |  | ||||||
| script: | script: | ||||||
| - tox -e $(echo py$TRAVIS_PYTHON_VERSION-mg$PYMONGO | tr -d . | sed -e 's/pypypy/pypy/') -- --with-coverage | - tox -e $(echo py$TRAVIS_PYTHON_VERSION-mg$PYMONGO | tr -d . | sed -e 's/pypypy/pypy/') -- --with-coverage | ||||||
| @@ -45,22 +68,34 @@ script: | |||||||
| # For now only submit coveralls for Python v2.7. Python v3.x currently shows | # For now only submit coveralls for Python v2.7. Python v3.x currently shows | ||||||
| # 0% coverage. That's caused by 'use_2to3', which builds the py3-compatible | # 0% coverage. That's caused by 'use_2to3', which builds the py3-compatible | ||||||
| # code in a separate dir and runs tests on that. | # code in a separate dir and runs tests on that. | ||||||
| after_script: | after_success: | ||||||
| - if [[ $TRAVIS_PYTHON_VERSION == '2.7' ]]; then coveralls --verbose; fi | - if [[ $TRAVIS_PYTHON_VERSION == '2.7' ]]; then coveralls --verbose; fi | ||||||
|  |  | ||||||
| notifications: | notifications: | ||||||
|   irc: irc.freenode.org#mongoengine |   irc: irc.freenode.org#mongoengine | ||||||
|  |  | ||||||
|  | # Only run builds on the master branch and GitHub releases (tagged as vX.Y.Z) | ||||||
| branches: | branches: | ||||||
|   only: |   only: | ||||||
|   - master |   - master | ||||||
|   - /^v.*$/ |   - /^v.*$/ | ||||||
|  |  | ||||||
|  | # Whenever a new release is created via GitHub, publish it on PyPI. | ||||||
| deploy: | deploy: | ||||||
|   provider: pypi |   provider: pypi | ||||||
|   user: the_drow |   user: the_drow | ||||||
|   password: |   password: | ||||||
|     secure: QMyatmWBnC6ZN3XLW2+fTBDU4LQcp1m/LjR2/0uamyeUzWKdlOoh/Wx5elOgLwt/8N9ppdPeG83ose1jOz69l5G0MUMjv8n/RIcMFSpCT59tGYqn3kh55b0cIZXFT9ar+5cxlif6a5rS72IHm5li7QQyxexJIII6Uxp0kpvUmek= |     secure: QMyatmWBnC6ZN3XLW2+fTBDU4LQcp1m/LjR2/0uamyeUzWKdlOoh/Wx5elOgLwt/8N9ppdPeG83ose1jOz69l5G0MUMjv8n/RIcMFSpCT59tGYqn3kh55b0cIZXFT9ar+5cxlif6a5rS72IHm5li7QQyxexJIII6Uxp0kpvUmek= | ||||||
|  |  | ||||||
|  |   # create a source distribution and a pure python wheel for faster installs | ||||||
|  |   distributions: "sdist bdist_wheel" | ||||||
|  |  | ||||||
|  |   # only deploy on tagged commits (aka GitHub releases) and only for the | ||||||
|  |   # parent repo's builds running Python 2.7 along with dev PyMongo (we run | ||||||
|  |   # Travis against many different Python and PyMongo versions and we don't | ||||||
|  |   # want the deploy to occur multiple times). | ||||||
|   on: |   on: | ||||||
|     tags: true |     tags: true | ||||||
|     repo: MongoEngine/mongoengine |     repo: MongoEngine/mongoengine | ||||||
|  |     condition: "$PYMONGO = 3.0" | ||||||
|  |     python: 2.7 | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ 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?state=open>`_ or `create one | ||||||
| <https://github.com/MongoEngine/mongoengine/issues>`_ for your specific | <https://github.com/MongoEngine/mongoengine/issues>`_ for your specific | ||||||
| issue or feature request. That way you avoid working on something | 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>` | post to the `user group <http://groups.google.com/group/mongoengine-users>` | ||||||
|  |  | ||||||
| Supported Interpreters | Supported Interpreters | ||||||
| @@ -29,19 +29,20 @@ Style Guide | |||||||
| ----------- | ----------- | ||||||
|  |  | ||||||
| MongoEngine aims to follow `PEP8 <http://www.python.org/dev/peps/pep-0008/>`_ | MongoEngine aims to follow `PEP8 <http://www.python.org/dev/peps/pep-0008/>`_ | ||||||
| including 4 space indents. When possible we try to stick to 79 character line limits. | including 4 space indents. When possible we try to stick to 79 character line | ||||||
| However, screens got bigger and an ORM has a strong focus on readability and | limits. However, screens got bigger and an ORM has a strong focus on | ||||||
| if it can help, we accept 119 as maximum line length, in a similar way as | readability and if it can help, we accept 119 as maximum line length, in a | ||||||
| `django does <https://docs.djangoproject.com/en/dev/internals/contributing/writing-code/coding-style/#python-style>`_ | similar way as `django does | ||||||
|  | <https://docs.djangoproject.com/en/dev/internals/contributing/writing-code/coding-style/#python-style>`_ | ||||||
|  |  | ||||||
| Testing | Testing | ||||||
| ------- | ------- | ||||||
|  |  | ||||||
| All tests are run on `Travis <http://travis-ci.org/MongoEngine/mongoengine>`_ | All tests are run on `Travis <http://travis-ci.org/MongoEngine/mongoengine>`_ | ||||||
| and any pull requests are automatically tested by Travis. Any pull requests | and any pull requests are automatically tested. Any pull requests without | ||||||
| without tests will take longer to be integrated and might be refused. | tests will take longer to be integrated and might be refused. | ||||||
|  |  | ||||||
| You may also submit a simple failing test as a PullRequest if you don't know | You may also submit a simple failing test as a pull request if you don't know | ||||||
| how to fix it, it will be easier for other people to work on it and it may get | how to fix it, it will be easier for other people to work on it and it may get | ||||||
| fixed faster. | fixed faster. | ||||||
|  |  | ||||||
| @@ -49,13 +50,18 @@ General Guidelines | |||||||
| ------------------ | ------------------ | ||||||
|  |  | ||||||
| - Avoid backward breaking changes if at all possible. | - Avoid backward breaking changes if at all possible. | ||||||
|  | - If you *have* to introduce a breaking change, make it very clear in your | ||||||
|  |   pull request's description. Also, describe how users of this package | ||||||
|  |   should adapt to the breaking change in docs/upgrade.rst. | ||||||
| - Write inline documentation for new classes and methods. | - Write inline documentation for new classes and methods. | ||||||
| - Write tests and make sure they pass (make sure you have a mongod | - Write tests and make sure they pass (make sure you have a mongod | ||||||
|   running on the default port, then execute ``python setup.py nosetests`` |   running on the default port, then execute ``python setup.py nosetests`` | ||||||
|   from the cmd line to run the test suite). |   from the cmd line to run the test suite). | ||||||
| - Ensure tests pass on every Python and PyMongo versions. | - Ensure tests pass on all supported Python, PyMongo, and MongoDB versions. | ||||||
|   You can test on these versions locally by executing ``tox`` |   You can test various Python and PyMongo versions locally by executing | ||||||
| - Add enhancements or problematic bug fixes to docs/changelog.rst |   ``tox``. For different MongoDB versions, you can rely on our automated | ||||||
|  |   Travis tests. | ||||||
|  | - Add enhancements or problematic bug fixes to docs/changelog.rst. | ||||||
| - Add yourself to AUTHORS :) | - Add yourself to AUTHORS :) | ||||||
|  |  | ||||||
| Documentation | Documentation | ||||||
| @@ -69,3 +75,6 @@ just make your changes to the inline documentation of the appropriate | |||||||
| branch and submit a `pull request <https://help.github.com/articles/using-pull-requests>`_. | branch and submit a `pull request <https://help.github.com/articles/using-pull-requests>`_. | ||||||
| You might also use the github `Edit <https://github.com/blog/844-forking-with-the-edit-button>`_ | You might also use the github `Edit <https://github.com/blog/844-forking-with-the-edit-button>`_ | ||||||
| button. | button. | ||||||
|  |  | ||||||
|  | If you want to test your documentation changes locally, you need to install | ||||||
|  | the ``sphinx`` package. | ||||||
|   | |||||||
							
								
								
									
										73
									
								
								README.rst
									
									
									
									
									
								
							
							
						
						
									
										73
									
								
								README.rst
									
									
									
									
									
								
							| @@ -19,32 +19,42 @@ MongoEngine | |||||||
| About | About | ||||||
| ===== | ===== | ||||||
| MongoEngine is a Python Object-Document Mapper for working with MongoDB. | MongoEngine is a Python Object-Document Mapper for working with MongoDB. | ||||||
| Documentation available at https://mongoengine-odm.readthedocs.io - there is currently | Documentation is available at https://mongoengine-odm.readthedocs.io - there | ||||||
| a `tutorial <https://mongoengine-odm.readthedocs.io/tutorial.html>`_, a `user guide | is currently a `tutorial <https://mongoengine-odm.readthedocs.io/tutorial.html>`_, | ||||||
| <https://mongoengine-odm.readthedocs.io/guide/index.html>`_ and an `API reference | a `user guide <https://mongoengine-odm.readthedocs.io/guide/index.html>`_, and | ||||||
| <https://mongoengine-odm.readthedocs.io/apireference.html>`_. | an `API reference <https://mongoengine-odm.readthedocs.io/apireference.html>`_. | ||||||
|  |  | ||||||
|  | Supported MongoDB Versions | ||||||
|  | ========================== | ||||||
|  | MongoEngine is currently tested against MongoDB v2.4, v2.6, and v3.0. Future | ||||||
|  | versions should be supported as well, but aren't actively tested at the moment. | ||||||
|  | Make sure to open an issue or submit a pull request if you experience any | ||||||
|  | problems with MongoDB v3.2+. | ||||||
|  |  | ||||||
| Installation | Installation | ||||||
| ============ | ============ | ||||||
| We recommend the use of `virtualenv <https://virtualenv.pypa.io/>`_ and of | We recommend the use of `virtualenv <https://virtualenv.pypa.io/>`_ and of | ||||||
| `pip <https://pip.pypa.io/>`_. You can then use ``pip install -U mongoengine``. | `pip <https://pip.pypa.io/>`_. You can then use ``pip install -U mongoengine``. | ||||||
| You may also have `setuptools <http://peak.telecommunity.com/DevCenter/setuptools>`_ and thus | You may also have `setuptools <http://peak.telecommunity.com/DevCenter/setuptools>`_ | ||||||
| you can use ``easy_install -U mongoengine``. Otherwise, you can download the | and thus you can use ``easy_install -U mongoengine``. Otherwise, you can download the | ||||||
| source from `GitHub <http://github.com/MongoEngine/mongoengine>`_ and run ``python | source from `GitHub <http://github.com/MongoEngine/mongoengine>`_ and run ``python | ||||||
| setup.py install``. | setup.py install``. | ||||||
|  |  | ||||||
| Dependencies | Dependencies | ||||||
| ============ | ============ | ||||||
| - pymongo>=2.7.1 | All of the dependencies can easily be installed via `pip <https://pip.pypa.io/>`_. | ||||||
| - sphinx (optional - for documentation generation) | 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: | ||||||
|  |  | ||||||
| Optional Dependencies |  | ||||||
| --------------------- |  | ||||||
| - **Image Fields**: Pillow>=2.0.0 |  | ||||||
| - dateutil>=2.1.0 | - dateutil>=2.1.0 | ||||||
|  |  | ||||||
| .. note | If you need to use an ``ImageField`` or ``ImageGridFsProxy``: | ||||||
|    MongoEngine always runs it's test suite against the latest patch version of each dependecy. e.g.: PyMongo 3.0.1 |  | ||||||
|  | - Pillow>=2.0.0 | ||||||
|  |  | ||||||
| Examples | Examples | ||||||
| ======== | ======== | ||||||
| @@ -57,7 +67,7 @@ Some simple examples of what MongoEngine code looks like: | |||||||
|  |  | ||||||
|     class BlogPost(Document): |     class BlogPost(Document): | ||||||
|         title = StringField(required=True, max_length=200) |         title = StringField(required=True, max_length=200) | ||||||
|         posted = DateTimeField(default=datetime.datetime.now) |         posted = DateTimeField(default=datetime.datetime.utcnow) | ||||||
|         tags = ListField(StringField(max_length=50)) |         tags = ListField(StringField(max_length=50)) | ||||||
|         meta = {'allow_inheritance': True} |         meta = {'allow_inheritance': True} | ||||||
|  |  | ||||||
| @@ -87,27 +97,28 @@ Some simple examples of what MongoEngine code looks like: | |||||||
|     ...     print |     ...     print | ||||||
|     ... |     ... | ||||||
|  |  | ||||||
|     >>> len(BlogPost.objects) |     # Count all blog posts and its subtypes | ||||||
|  |     >>> BlogPost.objects.count() | ||||||
|     2 |     2 | ||||||
|     >>> len(TextPost.objects) |     >>> TextPost.objects.count() | ||||||
|     1 |     1 | ||||||
|     >>> len(LinkPost.objects) |     >>> LinkPost.objects.count() | ||||||
|     1 |     1 | ||||||
|  |  | ||||||
|     # Find tagged posts |     # Count tagged posts | ||||||
|     >>> len(BlogPost.objects(tags='mongoengine')) |     >>> BlogPost.objects(tags='mongoengine').count() | ||||||
|     2 |     2 | ||||||
|     >>> len(BlogPost.objects(tags='mongodb')) |     >>> BlogPost.objects(tags='mongodb').count() | ||||||
|     1 |     1 | ||||||
|  |  | ||||||
| Tests | Tests | ||||||
| ===== | ===== | ||||||
| To run the test suite, ensure you are running a local instance of MongoDB on | To run the test suite, ensure you are running a local instance of MongoDB on | ||||||
| the standard port and have ``nose`` installed. Then, run: ``python setup.py nosetests``. | the standard port and have ``nose`` installed. Then, run ``python setup.py nosetests``. | ||||||
|  |  | ||||||
| To run the test suite on every supported Python version and every supported PyMongo version, | To run the test suite on every supported Python and PyMongo version, you can | ||||||
| you can use ``tox``. | use ``tox``. You'll need to make sure you have each supported Python version | ||||||
| tox and each supported Python version should be installed in your environment: | installed in your environment and then: | ||||||
|  |  | ||||||
| .. code-block:: shell | .. code-block:: shell | ||||||
|  |  | ||||||
| @@ -116,13 +127,16 @@ tox and each supported Python version should be installed in your environment: | |||||||
|     # Run the test suites |     # Run the test suites | ||||||
|     $ tox |     $ tox | ||||||
|  |  | ||||||
| If you wish to run one single or selected tests, use the nosetest convention. It will find the folder, | If you wish to run a subset of tests, use the nosetests convention: | ||||||
| eventually the file, go to the TestClass specified after the colon and eventually right to the single test. |  | ||||||
| Also use the -s argument if you want to print out whatever or access pdb while testing. |  | ||||||
|  |  | ||||||
| .. code-block:: shell | .. code-block:: shell | ||||||
|  |  | ||||||
|     $ python setup.py nosetests --tests tests/fields/fields.py:FieldTest.test_cls_field -s |     # Run all the tests in a particular test file | ||||||
|  |     $ python setup.py nosetests --tests tests/fields/fields.py | ||||||
|  |     # Run only particular test class in that file | ||||||
|  |     $ python setup.py nosetests --tests tests/fields/fields.py:FieldTest | ||||||
|  |     # Use the -s option if you want to print some debug statements or use pdb | ||||||
|  |     $ python setup.py nosetests --tests tests/fields/fields.py:FieldTest -s | ||||||
|  |  | ||||||
| Community | Community | ||||||
| ========= | ========= | ||||||
| @@ -130,8 +144,7 @@ Community | |||||||
|   <http://groups.google.com/group/mongoengine-users>`_ |   <http://groups.google.com/group/mongoengine-users>`_ | ||||||
| - `MongoEngine Developers mailing list | - `MongoEngine Developers mailing list | ||||||
|   <http://groups.google.com/group/mongoengine-dev>`_ |   <http://groups.google.com/group/mongoengine-dev>`_ | ||||||
| - `#mongoengine IRC channel <http://webchat.freenode.net/?channels=mongoengine>`_ |  | ||||||
|  |  | ||||||
| Contributing | 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>`_ | ||||||
|   | |||||||
| @@ -4,7 +4,10 @@ Changelog | |||||||
|  |  | ||||||
| Development | Development | ||||||
| =========== | =========== | ||||||
| - (Fill this out as you fix issues and develop you features). | - (Fill this out as you fix issues and develop your features). | ||||||
|  | - Fixed using sets in field choices #1481 | ||||||
|  | - POTENTIAL BREAKING CHANGE: Fixed limit/skip/hint/batch_size chaining #1476 | ||||||
|  | - POTENTIAL BREAKING CHANGE: Changed a public `QuerySet.clone_into` method to a private `QuerySet._clone_into` #1476 | ||||||
| - Fixed connecting to a replica set with PyMongo 2.x #1436 | - Fixed connecting to a replica set with PyMongo 2.x #1436 | ||||||
| - Fixed an obscure error message when filtering by `field__in=non_iterable`. #1237 | - Fixed an obscure error message when filtering by `field__in=non_iterable`. #1237 | ||||||
|  |  | ||||||
| @@ -13,6 +16,7 @@ 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: 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: 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: `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 | - Fixed absent rounding for DecimalField when `force_string` is set. #1103 | ||||||
|  |  | ||||||
| Changes in 0.10.8 | Changes in 0.10.8 | ||||||
|   | |||||||
| @@ -33,7 +33,7 @@ the :attr:`host` to | |||||||
|     corresponding parameters in :func:`~mongoengine.connect`: :: |     corresponding parameters in :func:`~mongoengine.connect`: :: | ||||||
|  |  | ||||||
|         connect( |         connect( | ||||||
|             name='test', |             db='test', | ||||||
|             username='user', |             username='user', | ||||||
|             password='12345', |             password='12345', | ||||||
|             host='mongodb://admin:qwerty@localhost/production' |             host='mongodb://admin:qwerty@localhost/production' | ||||||
| @@ -42,13 +42,18 @@ the :attr:`host` to | |||||||
|     will establish connection to ``production`` database using |     will establish connection to ``production`` database using | ||||||
|     ``admin`` username and ``qwerty`` password. |     ``admin`` username and ``qwerty`` password. | ||||||
|  |  | ||||||
| ReplicaSets | Replica Sets | ||||||
| =========== | ============ | ||||||
|  |  | ||||||
| MongoEngine supports | MongoEngine supports connecting to replica sets:: | ||||||
| :class:`~pymongo.mongo_replica_set_client.MongoReplicaSetClient`. To use them, |  | ||||||
| please use an URI style connection and provide the ``replicaSet`` name |     from mongoengine import connect | ||||||
| in the connection kwargs. |  | ||||||
|  |     # Regular connect | ||||||
|  |     connect('dbname', replicaset='rs-name') | ||||||
|  |  | ||||||
|  |     # MongoDB URI-style connect | ||||||
|  |     connect(host='mongodb://localhost/dbname?replicaSet=rs-name') | ||||||
|  |  | ||||||
| Read preferences are supported through the connection or via individual | Read preferences are supported through the connection or via individual | ||||||
| queries by passing the read_preference :: | queries by passing the read_preference :: | ||||||
| @@ -59,76 +64,74 @@ queries by passing the read_preference :: | |||||||
| Multiple Databases | Multiple Databases | ||||||
| ================== | ================== | ||||||
|  |  | ||||||
| Multiple database support was added in MongoEngine 0.6. To use multiple | To use multiple databases you can use :func:`~mongoengine.connect` and provide | ||||||
| databases you can use :func:`~mongoengine.connect` and provide an `alias` name | an `alias` name for the connection - if no `alias` is provided then "default" | ||||||
| for the connection - if no `alias` is provided then "default" is used. | is used. | ||||||
|  |  | ||||||
| In the background this uses :func:`~mongoengine.register_connection` to | In the background this uses :func:`~mongoengine.register_connection` to | ||||||
| store the data and you can register all aliases up front if required. | store the data and you can register all aliases up front if required. | ||||||
|  |  | ||||||
| Individual documents can also support multiple databases by providing a | Individual documents can also support multiple databases by providing a | ||||||
| `db_alias` in their meta data.  This allows :class:`~pymongo.dbref.DBRef` objects | `db_alias` in their meta data. This allows :class:`~pymongo.dbref.DBRef` | ||||||
| to point across databases and collections.  Below is an example schema, using | objects to point across databases and collections. Below is an example schema, | ||||||
| 3 different databases to store data:: | using 3 different databases to store data:: | ||||||
|  |  | ||||||
|         class User(Document): |         class User(Document): | ||||||
|             name = StringField() |             name = StringField() | ||||||
|  |  | ||||||
|             meta = {"db_alias": "user-db"} |             meta = {'db_alias': 'user-db'} | ||||||
|  |  | ||||||
|         class Book(Document): |         class Book(Document): | ||||||
|             name = StringField() |             name = StringField() | ||||||
|  |  | ||||||
|             meta = {"db_alias": "book-db"} |             meta = {'db_alias': 'book-db'} | ||||||
|  |  | ||||||
|         class AuthorBooks(Document): |         class AuthorBooks(Document): | ||||||
|             author = ReferenceField(User) |             author = ReferenceField(User) | ||||||
|             book = ReferenceField(Book) |             book = ReferenceField(Book) | ||||||
|  |  | ||||||
|             meta = {"db_alias": "users-books-db"} |             meta = {'db_alias': 'users-books-db'} | ||||||
|  |  | ||||||
|  |  | ||||||
| Context Managers | Context Managers | ||||||
| ================ | ================ | ||||||
| Sometimes you may want to switch the database or collection to query against | Sometimes you may want to switch the database or collection to query against. | ||||||
| for a class. |  | ||||||
| For example, archiving older data into a separate database for performance | For example, archiving older data into a separate database for performance | ||||||
| reasons or writing functions that dynamically choose collections to write | reasons or writing functions that dynamically choose collections to write | ||||||
| document to. | a document to. | ||||||
|  |  | ||||||
| Switch Database | Switch Database | ||||||
| --------------- | --------------- | ||||||
| The :class:`~mongoengine.context_managers.switch_db` context manager allows | The :class:`~mongoengine.context_managers.switch_db` context manager allows | ||||||
| you to change the database alias for a given class allowing quick and easy | you to change the database alias for a given class allowing quick and easy | ||||||
| access the same User document across databases:: | access to the same User document across databases:: | ||||||
|  |  | ||||||
|     from mongoengine.context_managers import switch_db |     from mongoengine.context_managers import switch_db | ||||||
|  |  | ||||||
|     class User(Document): |     class User(Document): | ||||||
|         name = StringField() |         name = StringField() | ||||||
|  |  | ||||||
|         meta = {"db_alias": "user-db"} |         meta = {'db_alias': 'user-db'} | ||||||
|  |  | ||||||
|     with switch_db(User, 'archive-user-db') as User: |     with switch_db(User, 'archive-user-db') as User: | ||||||
|         User(name="Ross").save()  # Saves the 'archive-user-db' |         User(name='Ross').save()  # Saves the 'archive-user-db' | ||||||
|  |  | ||||||
|  |  | ||||||
| Switch Collection | Switch Collection | ||||||
| ----------------- | ----------------- | ||||||
| The :class:`~mongoengine.context_managers.switch_collection` context manager | The :class:`~mongoengine.context_managers.switch_collection` context manager | ||||||
| allows you to change the collection for a given class allowing quick and easy | allows you to change the collection for a given class allowing quick and easy | ||||||
| access the same Group document across collection:: | access to the same Group document across collection:: | ||||||
|  |  | ||||||
|         from mongoengine.context_managers import switch_collection |         from mongoengine.context_managers import switch_collection | ||||||
|  |  | ||||||
|         class Group(Document): |         class Group(Document): | ||||||
|             name = StringField() |             name = StringField() | ||||||
|  |  | ||||||
|         Group(name="test").save()  # Saves in the default db |         Group(name='test').save()  # Saves in the default db | ||||||
|  |  | ||||||
|         with switch_collection(Group, 'group2000') as Group: |         with switch_collection(Group, 'group2000') as Group: | ||||||
|             Group(name="hello Group 2000 collection!").save()  # Saves in group2000 collection |             Group(name='hello Group 2000 collection!').save()  # Saves in group2000 collection | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| .. note:: Make sure any aliases have been registered with | .. note:: Make sure any aliases have been registered with | ||||||
|   | |||||||
| @@ -150,7 +150,7 @@ arguments can be set on all fields: | |||||||
|     .. note:: If set, this field is also accessible through the `pk` field. |     .. note:: If set, this field is also accessible through the `pk` field. | ||||||
|  |  | ||||||
| :attr:`choices` (Default: None) | :attr:`choices` (Default: None) | ||||||
|     An iterable (e.g. a list or tuple) of choices to which the value of this |     An iterable (e.g. list, tuple or set) of choices to which the value of this | ||||||
|     field should be limited. |     field should be limited. | ||||||
|  |  | ||||||
|     Can be either be a nested tuples of value (stored in mongo) and a |     Can be either be a nested tuples of value (stored in mongo) and a | ||||||
| @@ -361,11 +361,6 @@ Its value can take any of the following constants: | |||||||
|    In Django, be sure to put all apps that have such delete rule declarations in |    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. |    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 | Generic reference fields | ||||||
| '''''''''''''''''''''''' | '''''''''''''''''''''''' | ||||||
| A second kind of reference field also exists, | A second kind of reference field also exists, | ||||||
|   | |||||||
| @@ -2,13 +2,13 @@ | |||||||
| Installing MongoEngine | Installing MongoEngine | ||||||
| ====================== | ====================== | ||||||
|  |  | ||||||
| To use MongoEngine, you will need to download `MongoDB <http://mongodb.org/>`_ | To use MongoEngine, you will need to download `MongoDB <http://mongodb.com/>`_ | ||||||
| and ensure it is running in an accessible location. You will also need | and ensure it is running in an accessible location. You will also need | ||||||
| `PyMongo <http://api.mongodb.org/python>`_ to use MongoEngine, but if you | `PyMongo <http://api.mongodb.org/python>`_ to use MongoEngine, but if you | ||||||
| install MongoEngine using setuptools, then the dependencies will be handled for | install MongoEngine using setuptools, then the dependencies will be handled for | ||||||
| you. | you. | ||||||
|  |  | ||||||
| MongoEngine is available on PyPI, so to use it you can use :program:`pip`: | MongoEngine is available on PyPI, so you can use :program:`pip`: | ||||||
|  |  | ||||||
| .. code-block:: console | .. code-block:: console | ||||||
|  |  | ||||||
|   | |||||||
| @@ -142,11 +142,4 @@ cleaner looking while still allowing manual execution of the callback:: | |||||||
|         modified = DateTimeField() |         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 | .. _blinker: http://pypi.python.org/pypi/blinker | ||||||
|   | |||||||
| @@ -3,11 +3,10 @@ Tutorial | |||||||
| ======== | ======== | ||||||
|  |  | ||||||
| This tutorial introduces **MongoEngine** by means of example --- we will walk | This tutorial introduces **MongoEngine** by means of example --- we will walk | ||||||
| through how to create a simple **Tumblelog** application. A Tumblelog is a type | through how to create a simple **Tumblelog** application. A tumblelog is a | ||||||
| of blog where posts are not constrained to being conventional text-based posts. | blog that supports mixed media content, including text, images, links, video, | ||||||
| As well as text-based entries, users may post images, links, videos, etc. For | audio, etc. For simplicity's sake, we'll stick to text, image, and link | ||||||
| simplicity's sake, we'll stick to text, image and link entries in our | entries. As the purpose of this tutorial is to introduce MongoEngine, we'll | ||||||
| application. As the purpose of this tutorial is to introduce MongoEngine, we'll |  | ||||||
| focus on the data-modelling side of the application, leaving out a user | focus on the data-modelling side of the application, leaving out a user | ||||||
| interface. | interface. | ||||||
|  |  | ||||||
| @@ -16,14 +15,14 @@ Getting started | |||||||
|  |  | ||||||
| Before we start, make sure that a copy of MongoDB is running in an accessible | Before we start, make sure that a copy of MongoDB is running in an accessible | ||||||
| location --- running it locally will be easier, but if that is not an option | location --- running it locally will be easier, but if that is not an option | ||||||
| then it may be run on a remote server. If you haven't installed mongoengine, | then it may be run on a remote server. If you haven't installed MongoEngine, | ||||||
| simply use pip to install it like so:: | simply use pip to install it like so:: | ||||||
|  |  | ||||||
|     $ pip install mongoengine |     $ pip install mongoengine | ||||||
|  |  | ||||||
| Before we can start using MongoEngine, we need to tell it how to connect to our | Before we can start using MongoEngine, we need to tell it how to connect to our | ||||||
| instance of :program:`mongod`. For this we use the :func:`~mongoengine.connect` | instance of :program:`mongod`. For this we use the :func:`~mongoengine.connect` | ||||||
| function. If running locally the only argument we need to provide is the name | function. If running locally, the only argument we need to provide is the name | ||||||
| of the MongoDB database to use:: | of the MongoDB database to use:: | ||||||
|  |  | ||||||
|     from mongoengine import * |     from mongoengine import * | ||||||
| @@ -39,18 +38,18 @@ Defining our documents | |||||||
| MongoDB is *schemaless*, which means that no schema is enforced by the database | MongoDB is *schemaless*, which means that no schema is enforced by the database | ||||||
| --- we may add and remove fields however we want and MongoDB won't complain. | --- we may add and remove fields however we want and MongoDB won't complain. | ||||||
| This makes life a lot easier in many regards, especially when there is a change | This makes life a lot easier in many regards, especially when there is a change | ||||||
| to the data model. However, defining schemata for our documents can help to | to the data model. However, defining schemas for our documents can help to iron | ||||||
| iron out bugs involving incorrect types or missing fields, and also allow us to | out bugs involving incorrect types or missing fields, and also allow us to | ||||||
| define utility methods on our documents in the same way that traditional | define utility methods on our documents in the same way that traditional | ||||||
| :abbr:`ORMs (Object-Relational Mappers)` do. | :abbr:`ORMs (Object-Relational Mappers)` do. | ||||||
|  |  | ||||||
| In our Tumblelog application we need to store several different types of | In our Tumblelog application we need to store several different types of | ||||||
| information.  We will need to have a collection of **users**, so that we may | information. We will need to have a collection of **users**, so that we may | ||||||
| link posts to an individual. We also need to store our different types of | link posts to an individual. We also need to store our different types of | ||||||
| **posts** (eg: text, image and link) in the database. To aid navigation of our | **posts** (eg: text, image and link) in the database. To aid navigation of our | ||||||
| Tumblelog, posts may have **tags** associated with them, so that the list of | Tumblelog, posts may have **tags** associated with them, so that the list of | ||||||
| posts shown to the user may be limited to posts that have been assigned a | posts shown to the user may be limited to posts that have been assigned a | ||||||
| specific tag.  Finally, it would be nice if **comments** could be added to | specific tag. Finally, it would be nice if **comments** could be added to | ||||||
| posts. We'll start with **users**, as the other document models are slightly | posts. We'll start with **users**, as the other document models are slightly | ||||||
| more involved. | more involved. | ||||||
|  |  | ||||||
| @@ -78,7 +77,7 @@ Now we'll think about how to store the rest of the information. If we were | |||||||
| using a relational database, we would most likely have a table of **posts**, a | using a relational database, we would most likely have a table of **posts**, a | ||||||
| table of **comments** and a table of **tags**.  To associate the comments with | table of **comments** and a table of **tags**.  To associate the comments with | ||||||
| individual posts, we would put a column in the comments table that contained a | individual posts, we would put a column in the comments table that contained a | ||||||
| foreign key to the posts table.  We'd also need a link table to provide the | foreign key to the posts table. We'd also need a link table to provide the | ||||||
| many-to-many relationship between posts and tags. Then we'd need to address the | many-to-many relationship between posts and tags. Then we'd need to address the | ||||||
| problem of storing the specialised post-types (text, image and link). There are | problem of storing the specialised post-types (text, image and link). There are | ||||||
| several ways we can achieve this, but each of them have their problems --- none | several ways we can achieve this, but each of them have their problems --- none | ||||||
| @@ -96,7 +95,7 @@ using* the new fields we need to support video posts. This fits with the | |||||||
| Object-Oriented principle of *inheritance* nicely. We can think of | Object-Oriented principle of *inheritance* nicely. We can think of | ||||||
| :class:`Post` as a base class, and :class:`TextPost`, :class:`ImagePost` and | :class:`Post` as a base class, and :class:`TextPost`, :class:`ImagePost` and | ||||||
| :class:`LinkPost` as subclasses of :class:`Post`. In fact, MongoEngine supports | :class:`LinkPost` as subclasses of :class:`Post`. In fact, MongoEngine supports | ||||||
| this kind of modelling out of the box --- all you need do is turn on inheritance | this kind of modeling out of the box --- all you need do is turn on inheritance | ||||||
| by setting :attr:`allow_inheritance` to True in the :attr:`meta`:: | by setting :attr:`allow_inheritance` to True in the :attr:`meta`:: | ||||||
|  |  | ||||||
|     class Post(Document): |     class Post(Document): | ||||||
| @@ -128,8 +127,8 @@ link table, we can just store a list of tags in each post. So, for both | |||||||
| efficiency and simplicity's sake, we'll store the tags as strings directly | efficiency and simplicity's sake, we'll store the tags as strings directly | ||||||
| within the post, rather than storing references to tags in a separate | within the post, rather than storing references to tags in a separate | ||||||
| collection. Especially as tags are generally very short (often even shorter | collection. Especially as tags are generally very short (often even shorter | ||||||
| than a document's id), this denormalisation won't impact very strongly on the | than a document's id), this denormalization won't impact the size of the | ||||||
| size of our database. So let's take a look that the code our modified | database very strongly. Let's take a look at the code of our modified | ||||||
| :class:`Post` class:: | :class:`Post` class:: | ||||||
|  |  | ||||||
|     class Post(Document): |     class Post(Document): | ||||||
| @@ -141,7 +140,7 @@ The :class:`~mongoengine.fields.ListField` object that is used to define a Post' | |||||||
| takes a field object as its first argument --- this means that you can have | takes a field object as its first argument --- this means that you can have | ||||||
| lists of any type of field (including lists). | lists of any type of field (including lists). | ||||||
|  |  | ||||||
| .. note:: We don't need to modify the specialised post types as they all | .. note:: We don't need to modify the specialized post types as they all | ||||||
|     inherit from :class:`Post`. |     inherit from :class:`Post`. | ||||||
|  |  | ||||||
| Comments | Comments | ||||||
| @@ -149,7 +148,7 @@ Comments | |||||||
|  |  | ||||||
| A comment is typically associated with *one* post. In a relational database, to | A comment is typically associated with *one* post. In a relational database, to | ||||||
| display a post with its comments, we would have to retrieve the post from the | display a post with its comments, we would have to retrieve the post from the | ||||||
| database, then query the database again for the comments associated with the | database and then query the database again for the comments associated with the | ||||||
| post. This works, but there is no real reason to be storing the comments | post. This works, but there is no real reason to be storing the comments | ||||||
| separately from their associated posts, other than to work around the | separately from their associated posts, other than to work around the | ||||||
| relational model. Using MongoDB we can store the comments as a list of | relational model. Using MongoDB we can store the comments as a list of | ||||||
| @@ -219,8 +218,8 @@ Now that we've got our user in the database, let's add a couple of posts:: | |||||||
|     post2.tags = ['mongoengine'] |     post2.tags = ['mongoengine'] | ||||||
|     post2.save() |     post2.save() | ||||||
|  |  | ||||||
| .. note:: If you change a field on a object that has already been saved, then | .. note:: If you change a field on an object that has already been saved and | ||||||
|     call :meth:`save` again, the document will be updated. |     then call :meth:`save` again, the document will be updated. | ||||||
|  |  | ||||||
| Accessing our data | Accessing our data | ||||||
| ================== | ================== | ||||||
| @@ -232,17 +231,17 @@ used to access the documents in the database collection associated with that | |||||||
| class. So let's see how we can get our posts' titles:: | class. So let's see how we can get our posts' titles:: | ||||||
|  |  | ||||||
|     for post in Post.objects: |     for post in Post.objects: | ||||||
|         print post.title |         print(post.title) | ||||||
|  |  | ||||||
| Retrieving type-specific information | Retrieving type-specific information | ||||||
| ------------------------------------ | ------------------------------------ | ||||||
|  |  | ||||||
| This will print the titles of our posts, one on each line. But What if we want | This will print the titles of our posts, one on each line. But what if we want | ||||||
| to access the type-specific data (link_url, content, etc.)? One way is simply | to access the type-specific data (link_url, content, etc.)? One way is simply | ||||||
| to use the :attr:`objects` attribute of a subclass of :class:`Post`:: | to use the :attr:`objects` attribute of a subclass of :class:`Post`:: | ||||||
|  |  | ||||||
|     for post in TextPost.objects: |     for post in TextPost.objects: | ||||||
|         print post.content |         print(post.content) | ||||||
|  |  | ||||||
| Using TextPost's :attr:`objects` attribute only returns documents that were | Using TextPost's :attr:`objects` attribute only returns documents that were | ||||||
| created using :class:`TextPost`. Actually, there is a more general rule here: | created using :class:`TextPost`. Actually, there is a more general rule here: | ||||||
| @@ -259,16 +258,14 @@ instances of :class:`Post` --- they were instances of the subclass of | |||||||
| practice:: | practice:: | ||||||
|  |  | ||||||
|     for post in Post.objects: |     for post in Post.objects: | ||||||
|         print post.title |         print(post.title) | ||||||
|         print '=' * len(post.title) |         print('=' * len(post.title)) | ||||||
|  |  | ||||||
|         if isinstance(post, TextPost): |         if isinstance(post, TextPost): | ||||||
|             print post.content |             print(post.content) | ||||||
|  |  | ||||||
|         if isinstance(post, LinkPost): |         if isinstance(post, LinkPost): | ||||||
|             print 'Link:', post.link_url |             print('Link: {}'.format(post.link_url)) | ||||||
|  |  | ||||||
|         print |  | ||||||
|  |  | ||||||
| This would print the title of each post, followed by the content if it was a | This would print the title of each post, followed by the content if it was a | ||||||
| text post, and "Link: <url>" if it was a link post. | text post, and "Link: <url>" if it was a link post. | ||||||
| @@ -283,7 +280,7 @@ your query.  Let's adjust our query so that only posts with the tag "mongodb" | |||||||
| are returned:: | are returned:: | ||||||
|  |  | ||||||
|     for post in Post.objects(tags='mongodb'): |     for post in Post.objects(tags='mongodb'): | ||||||
|         print post.title |         print(post.title) | ||||||
|  |  | ||||||
| There are also methods available on :class:`~mongoengine.queryset.QuerySet` | There are also methods available on :class:`~mongoengine.queryset.QuerySet` | ||||||
| objects that allow different results to be returned, for example, calling | objects that allow different results to be returned, for example, calling | ||||||
| @@ -292,11 +289,11 @@ the first matched by the query you provide. Aggregation functions may also be | |||||||
| used on :class:`~mongoengine.queryset.QuerySet` objects:: | used on :class:`~mongoengine.queryset.QuerySet` objects:: | ||||||
|  |  | ||||||
|     num_posts = Post.objects(tags='mongodb').count() |     num_posts = Post.objects(tags='mongodb').count() | ||||||
|     print 'Found %d posts with tag "mongodb"' % num_posts |     print('Found {} posts with tag "mongodb"'.format(num_posts)) | ||||||
|  |  | ||||||
| Learning more about mongoengine | Learning more about MongoEngine | ||||||
| ------------------------------- | ------------------------------- | ||||||
|  |  | ||||||
| If you got this far you've made a great start, so well done!  The next step on | If you got this far you've made a great start, so well done! The next step on | ||||||
| your mongoengine journey is the `full user guide <guide/index.html>`_, where you | your MongoEngine journey is the `full user guide <guide/index.html>`_, where | ||||||
| can learn indepth about how to use mongoengine and mongodb. | you can learn in-depth about how to use MongoEngine and MongoDB. | ||||||
|   | |||||||
| @@ -2,6 +2,20 @@ | |||||||
| Upgrading | Upgrading | ||||||
| ######### | ######### | ||||||
|  |  | ||||||
|  | Development | ||||||
|  | *********** | ||||||
|  | (Fill this out whenever you introduce breaking changes to MongoEngine) | ||||||
|  |  | ||||||
|  | This release includes various fixes for the `BaseQuerySet` methods and how they | ||||||
|  | are chained together. Since version 0.10.1 applying limit/skip/hint/batch_size | ||||||
|  | to an already-existing queryset wouldn't modify the underlying PyMongo cursor. | ||||||
|  | This has been fixed now, so you'll need to make sure that your code didn't rely | ||||||
|  | on the broken implementation. | ||||||
|  |  | ||||||
|  | Additionally, a public `BaseQuerySet.clone_into` has been renamed to a private | ||||||
|  | `_clone_into`. If you directly used that method in your code, you'll need to | ||||||
|  | rename its occurrences. | ||||||
|  |  | ||||||
| 0.11.0 | 0.11.0 | ||||||
| ****** | ****** | ||||||
| This release includes a major rehaul of MongoEngine's code quality and | This release includes a major rehaul of MongoEngine's code quality and | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ __all__ = ('UPDATE_OPERATORS', 'get_document', '_document_registry') | |||||||
|  |  | ||||||
| UPDATE_OPERATORS = set(['set', 'unset', 'inc', 'dec', 'pop', 'push', | UPDATE_OPERATORS = set(['set', 'unset', 'inc', 'dec', 'pop', 'push', | ||||||
|                         'push_all', 'pull', 'pull_all', 'add_to_set', |                         'push_all', 'pull', 'pull_all', 'add_to_set', | ||||||
|                         'set_on_insert', 'min', 'max']) |                         'set_on_insert', 'min', 'max', 'rename']) | ||||||
|  |  | ||||||
|  |  | ||||||
| _document_registry = {} | _document_registry = {} | ||||||
|   | |||||||
| @@ -429,7 +429,7 @@ class StrictDict(object): | |||||||
|     def __eq__(self, other): |     def __eq__(self, other): | ||||||
|         return self.items() == other.items() |         return self.items() == other.items() | ||||||
|  |  | ||||||
|     def __neq__(self, other): |     def __ne__(self, other): | ||||||
|         return self.items() != other.items() |         return self.items() != other.items() | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|   | |||||||
| @@ -402,9 +402,11 @@ class BaseDocument(object): | |||||||
|             raise ValidationError(message, errors=errors) |             raise ValidationError(message, errors=errors) | ||||||
|  |  | ||||||
|     def to_json(self, *args, **kwargs): |     def to_json(self, *args, **kwargs): | ||||||
|         """Converts a document to JSON. |         """Convert this 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 |         :param use_db_field: Serialize field names as they appear in | ||||||
|  |             MongoDB (as opposed to attribute names on this document). | ||||||
|  |             Defaults to True. | ||||||
|         """ |         """ | ||||||
|         use_db_field = kwargs.pop('use_db_field', True) |         use_db_field = kwargs.pop('use_db_field', True) | ||||||
|         return json_util.dumps(self.to_mongo(use_db_field), *args, **kwargs) |         return json_util.dumps(self.to_mongo(use_db_field), *args, **kwargs) | ||||||
|   | |||||||
| @@ -41,7 +41,7 @@ class BaseField(object): | |||||||
|         """ |         """ | ||||||
|         :param db_field: The database field to store this field in |         :param db_field: The database field to store this field in | ||||||
|             (defaults to the name of the field) |             (defaults to the name of the field) | ||||||
|         :param name: Depreciated - use db_field |         :param name: Deprecated - use db_field | ||||||
|         :param required: If the field is required. Whether it has to have a |         :param required: If the field is required. Whether it has to have a | ||||||
|             value or not. Defaults to False. |             value or not. Defaults to False. | ||||||
|         :param default: (optional) The default value for this field if no value |         :param default: (optional) The default value for this field if no value | ||||||
| @@ -81,6 +81,17 @@ class BaseField(object): | |||||||
|         self.sparse = sparse |         self.sparse = sparse | ||||||
|         self._owner_document = None |         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. |         # Detect and report conflicts between metadata and base properties. | ||||||
|         conflicts = set(dir(self)) & set(kwargs) |         conflicts = set(dir(self)) & set(kwargs) | ||||||
|         if conflicts: |         if conflicts: | ||||||
| @@ -182,7 +193,8 @@ class BaseField(object): | |||||||
|         EmbeddedDocument = _import_class('EmbeddedDocument') |         EmbeddedDocument = _import_class('EmbeddedDocument') | ||||||
|  |  | ||||||
|         choice_list = self.choices |         choice_list = self.choices | ||||||
|         if isinstance(choice_list[0], (list, tuple)): |         if isinstance(next(iter(choice_list)), (list, tuple)): | ||||||
|  |             # next(iter) is useful for sets | ||||||
|             choice_list = [k for k, _ in choice_list] |             choice_list = [k for k, _ in choice_list] | ||||||
|  |  | ||||||
|         # Choices which are other types of Documents |         # Choices which are other types of Documents | ||||||
|   | |||||||
| @@ -51,7 +51,9 @@ def register_connection(alias, name=None, host=None, port=None, | |||||||
|         MONGODB-CR (MongoDB Challenge Response protocol) for older servers. |         MONGODB-CR (MongoDB Challenge Response protocol) for older servers. | ||||||
|     :param is_mock: explicitly use mongomock for this connection |     :param is_mock: explicitly use mongomock for this connection | ||||||
|         (can also be done by using `mongomock://` as db host prefix) |         (can also be done by using `mongomock://` as db host prefix) | ||||||
|     :param kwargs: allow ad-hoc parameters to be passed into the pymongo driver |     :param kwargs: ad-hoc parameters to be passed into the pymongo driver, | ||||||
|  |         for example maxpoolsize, tz_aware, etc. See the documentation | ||||||
|  |         for pymongo's `MongoClient` for a full list. | ||||||
|  |  | ||||||
|     .. versionchanged:: 0.10.6 - added mongomock support |     .. versionchanged:: 0.10.6 - added mongomock support | ||||||
|     """ |     """ | ||||||
| @@ -241,9 +243,12 @@ def connect(db=None, alias=DEFAULT_CONNECTION_NAME, **kwargs): | |||||||
|     running on the default port on localhost. If authentication is needed, |     running on the default port on localhost. If authentication is needed, | ||||||
|     provide username and password arguments as well. |     provide username and password arguments as well. | ||||||
|  |  | ||||||
|     Multiple databases are supported by using aliases.  Provide a separate |     Multiple databases are supported by using aliases. Provide a separate | ||||||
|     `alias` to connect to a different instance of :program:`mongod`. |     `alias` to connect to a different instance of :program:`mongod`. | ||||||
|  |  | ||||||
|  |     See the docstring for `register_connection` for more details about all | ||||||
|  |     supported kwargs. | ||||||
|  |  | ||||||
|     .. versionchanged:: 0.6 - added multiple database support. |     .. versionchanged:: 0.6 - added multiple database support. | ||||||
|     """ |     """ | ||||||
|     if alias not in _connections: |     if alias not in _connections: | ||||||
|   | |||||||
| @@ -332,68 +332,20 @@ class Document(BaseDocument): | |||||||
|         signals.pre_save_post_validation.send(self.__class__, document=self, |         signals.pre_save_post_validation.send(self.__class__, document=self, | ||||||
|                                               created=created, **signal_kwargs) |                                               created=created, **signal_kwargs) | ||||||
|  |  | ||||||
|  |         if self._meta.get('auto_create_index', True): | ||||||
|  |             self.ensure_indexes() | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             collection = self._get_collection() |             # Save a new document or update an existing one | ||||||
|             if self._meta.get('auto_create_index', True): |  | ||||||
|                 self.ensure_indexes() |  | ||||||
|             if created: |             if created: | ||||||
|                 if force_insert: |                 object_id = self._save_create(doc, force_insert, write_concern) | ||||||
|                     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: |             else: | ||||||
|                 object_id = doc['_id'] |                 object_id, created = self._save_update(doc, save_condition, | ||||||
|                 updates, removals = self._delta() |                                                        write_concern) | ||||||
|                 # 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: |             if cascade is None: | ||||||
|                 cascade = self._meta.get( |                 cascade = (self._meta.get('cascade', False) or | ||||||
|                     'cascade', False) or cascade_kwargs is not None |                            cascade_kwargs is not None) | ||||||
|  |  | ||||||
|             if cascade: |             if cascade: | ||||||
|                 kwargs = { |                 kwargs = { | ||||||
| @@ -406,6 +358,7 @@ class Document(BaseDocument): | |||||||
|                     kwargs.update(cascade_kwargs) |                     kwargs.update(cascade_kwargs) | ||||||
|                 kwargs['_refs'] = _refs |                 kwargs['_refs'] = _refs | ||||||
|                 self.cascade_save(**kwargs) |                 self.cascade_save(**kwargs) | ||||||
|  |  | ||||||
|         except pymongo.errors.DuplicateKeyError as err: |         except pymongo.errors.DuplicateKeyError as err: | ||||||
|             message = u'Tried to save duplicate unique keys (%s)' |             message = u'Tried to save duplicate unique keys (%s)' | ||||||
|             raise NotUniqueError(message % six.text_type(err)) |             raise NotUniqueError(message % six.text_type(err)) | ||||||
| @@ -418,16 +371,91 @@ class Document(BaseDocument): | |||||||
|                 raise NotUniqueError(message % six.text_type(err)) |                 raise NotUniqueError(message % six.text_type(err)) | ||||||
|             raise OperationError(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'] |         id_field = self._meta['id_field'] | ||||||
|         if created or id_field not in self._meta.get('shard_key', []): |         if created or id_field not in self._meta.get('shard_key', []): | ||||||
|             self[id_field] = self._fields[id_field].to_python(object_id) |             self[id_field] = self._fields[id_field].to_python(object_id) | ||||||
|  |  | ||||||
|         signals.post_save.send(self.__class__, document=self, |         signals.post_save.send(self.__class__, document=self, | ||||||
|                                created=created, **signal_kwargs) |                                created=created, **signal_kwargs) | ||||||
|  |  | ||||||
|         self._clear_changed_fields() |         self._clear_changed_fields() | ||||||
|         self._created = False |         self._created = False | ||||||
|  |  | ||||||
|         return self |         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): |     def cascade_save(self, **kwargs): | ||||||
|         """Recursively save any references and generic references on the |         """Recursively save any references and generic references on the | ||||||
|         document. |         document. | ||||||
|   | |||||||
| @@ -50,8 +50,8 @@ class FieldDoesNotExist(Exception): | |||||||
|     or an :class:`~mongoengine.EmbeddedDocument`. |     or an :class:`~mongoengine.EmbeddedDocument`. | ||||||
|  |  | ||||||
|     To avoid this behavior on data loading, |     To avoid this behavior on data loading, | ||||||
|     you should the :attr:`strict` to ``False`` |     you should set the :attr:`strict` to ``False`` | ||||||
|     in the :attr:`meta` dictionnary. |     in the :attr:`meta` dictionary. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|  |  | ||||||
|   | |||||||
| @@ -139,12 +139,12 @@ class URLField(StringField): | |||||||
|         # Check first if the scheme is valid |         # Check first if the scheme is valid | ||||||
|         scheme = value.split('://')[0].lower() |         scheme = value.split('://')[0].lower() | ||||||
|         if scheme not in self.schemes: |         if scheme not in self.schemes: | ||||||
|             self.error('Invalid scheme {} in URL: {}'.format(scheme, value)) |             self.error(u'Invalid scheme {} in URL: {}'.format(scheme, value)) | ||||||
|             return |             return | ||||||
|  |  | ||||||
|         # Then check full URL |         # Then check full URL | ||||||
|         if not self.url_regex.match(value): |         if not self.url_regex.match(value): | ||||||
|             self.error('Invalid URL: {}'.format(value)) |             self.error(u'Invalid URL: {}'.format(value)) | ||||||
|             return |             return | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -888,10 +888,6 @@ class ReferenceField(BaseField): | |||||||
|  |  | ||||||
|         Foo.register_delete_rule(Bar, 'foo', NULLIFY) |         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` |     .. versionchanged:: 0.5 added `reverse_delete_rule` | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|   | |||||||
| @@ -86,6 +86,7 @@ class BaseQuerySet(object): | |||||||
|         self._batch_size = None |         self._batch_size = None | ||||||
|         self.only_fields = [] |         self.only_fields = [] | ||||||
|         self._max_time_ms = None |         self._max_time_ms = None | ||||||
|  |         self._comment = None | ||||||
|  |  | ||||||
|     def __call__(self, q_obj=None, class_check=True, read_preference=None, |     def __call__(self, q_obj=None, class_check=True, read_preference=None, | ||||||
|                  **query): |                  **query): | ||||||
| @@ -706,39 +707,36 @@ class BaseQuerySet(object): | |||||||
|         with switch_db(self._document, alias) as cls: |         with switch_db(self._document, alias) as cls: | ||||||
|             collection = cls._get_collection() |             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): |     def clone(self): | ||||||
|         """Creates a copy of the current |         """Create a copy of the current queryset.""" | ||||||
|           :class:`~mongoengine.queryset.QuerySet` |         return self._clone_into(self.__class__(self._document, self._collection_obj)) | ||||||
|  |  | ||||||
|         .. versionadded:: 0.5 |     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`). | ||||||
|         """ |         """ | ||||||
|         return self.clone_into(self.__class__(self._document, self._collection_obj)) |         if not isinstance(new_qs, BaseQuerySet): | ||||||
|  |  | ||||||
|     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( |             raise OperationError( | ||||||
|                 '%s is not a subclass of BaseQuerySet' % cls.__name__) |                 '%s is not a subclass of BaseQuerySet' % new_qs.__name__) | ||||||
|  |  | ||||||
|         copy_props = ('_mongo_query', '_initial_query', '_none', '_query_obj', |         copy_props = ('_mongo_query', '_initial_query', '_none', '_query_obj', | ||||||
|                       '_where_clause', '_loaded_fields', '_ordering', '_snapshot', |                       '_where_clause', '_loaded_fields', '_ordering', '_snapshot', | ||||||
|                       '_timeout', '_class_check', '_slave_okay', '_read_preference', |                       '_timeout', '_class_check', '_slave_okay', '_read_preference', | ||||||
|                       '_iter', '_scalar', '_as_pymongo', '_as_pymongo_coerce', |                       '_iter', '_scalar', '_as_pymongo', '_as_pymongo_coerce', | ||||||
|                       '_limit', '_skip', '_hint', '_auto_dereference', |                       '_limit', '_skip', '_hint', '_auto_dereference', | ||||||
|                       '_search_text', 'only_fields', '_max_time_ms') |                       '_search_text', 'only_fields', '_max_time_ms', '_comment') | ||||||
|  |  | ||||||
|         for prop in copy_props: |         for prop in copy_props: | ||||||
|             val = getattr(self, prop) |             val = getattr(self, prop) | ||||||
|             setattr(cls, prop, copy.copy(val)) |             setattr(new_qs, prop, copy.copy(val)) | ||||||
|  |  | ||||||
|         if self._cursor_obj: |         if self._cursor_obj: | ||||||
|             cls._cursor_obj = self._cursor_obj.clone() |             new_qs._cursor_obj = self._cursor_obj.clone() | ||||||
|  |  | ||||||
|         return cls |         return new_qs | ||||||
|  |  | ||||||
|     def select_related(self, max_depth=1): |     def select_related(self, max_depth=1): | ||||||
|         """Handles dereferencing of :class:`~bson.dbref.DBRef` objects or |         """Handles dereferencing of :class:`~bson.dbref.DBRef` objects or | ||||||
| @@ -760,7 +758,11 @@ class BaseQuerySet(object): | |||||||
|         """ |         """ | ||||||
|         queryset = self.clone() |         queryset = self.clone() | ||||||
|         queryset._limit = n if n != 0 else 1 |         queryset._limit = n if n != 0 else 1 | ||||||
|         # Return self to allow chaining |  | ||||||
|  |         # If a cursor object has already been created, apply the limit to it. | ||||||
|  |         if queryset._cursor_obj: | ||||||
|  |             queryset._cursor_obj.limit(queryset._limit) | ||||||
|  |  | ||||||
|         return queryset |         return queryset | ||||||
|  |  | ||||||
|     def skip(self, n): |     def skip(self, n): | ||||||
| @@ -771,6 +773,11 @@ class BaseQuerySet(object): | |||||||
|         """ |         """ | ||||||
|         queryset = self.clone() |         queryset = self.clone() | ||||||
|         queryset._skip = n |         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 |         return queryset | ||||||
|  |  | ||||||
|     def hint(self, index=None): |     def hint(self, index=None): | ||||||
| @@ -788,6 +795,11 @@ class BaseQuerySet(object): | |||||||
|         """ |         """ | ||||||
|         queryset = self.clone() |         queryset = self.clone() | ||||||
|         queryset._hint = index |         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 |         return queryset | ||||||
|  |  | ||||||
|     def batch_size(self, size): |     def batch_size(self, size): | ||||||
| @@ -801,6 +813,11 @@ class BaseQuerySet(object): | |||||||
|         """ |         """ | ||||||
|         queryset = self.clone() |         queryset = self.clone() | ||||||
|         queryset._batch_size = size |         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 |         return queryset | ||||||
|  |  | ||||||
|     def distinct(self, field): |     def distinct(self, field): | ||||||
| @@ -972,13 +989,31 @@ class BaseQuerySet(object): | |||||||
|     def order_by(self, *keys): |     def order_by(self, *keys): | ||||||
|         """Order the :class:`~mongoengine.queryset.QuerySet` by the keys. The |         """Order the :class:`~mongoengine.queryset.QuerySet` by the keys. The | ||||||
|         order may be specified by prepending each of the keys by a + or a -. |         order may be specified by prepending each of the keys by a + or a -. | ||||||
|         Ascending order is assumed. |         Ascending order is assumed. If no keys are passed, existing ordering | ||||||
|  |         is cleared instead. | ||||||
|  |  | ||||||
|         :param keys: fields to order the query results by; keys may be |         :param keys: fields to order the query results by; keys may be | ||||||
|             prefixed with **+** or **-** to determine the ordering direction |             prefixed with **+** or **-** to determine the ordering direction | ||||||
|         """ |         """ | ||||||
|         queryset = self.clone() |         queryset = self.clone() | ||||||
|         queryset._ordering = queryset._get_order_by(keys) |  | ||||||
|  |         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 | ||||||
|  |  | ||||||
|         return queryset |         return queryset | ||||||
|  |  | ||||||
|     def comment(self, text): |     def comment(self, text): | ||||||
| @@ -1424,10 +1459,13 @@ class BaseQuerySet(object): | |||||||
|             raise StopIteration |             raise StopIteration | ||||||
|  |  | ||||||
|         raw_doc = self._cursor.next() |         raw_doc = self._cursor.next() | ||||||
|  |  | ||||||
|         if self._as_pymongo: |         if self._as_pymongo: | ||||||
|             return self._get_as_pymongo(raw_doc) |             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: |         if self._scalar: | ||||||
|             return self._get_scalar(doc) |             return self._get_scalar(doc) | ||||||
| @@ -1437,7 +1475,6 @@ class BaseQuerySet(object): | |||||||
|     def rewind(self): |     def rewind(self): | ||||||
|         """Rewind the cursor to its unevaluated state. |         """Rewind the cursor to its unevaluated state. | ||||||
|  |  | ||||||
|  |  | ||||||
|         .. versionadded:: 0.3 |         .. versionadded:: 0.3 | ||||||
|         """ |         """ | ||||||
|         self._iter = False |         self._iter = False | ||||||
| @@ -1487,43 +1524,54 @@ class BaseQuerySet(object): | |||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def _cursor(self): |     def _cursor(self): | ||||||
|         if self._cursor_obj is None: |         """Return a PyMongo cursor object corresponding to this queryset.""" | ||||||
|  |  | ||||||
|             # In PyMongo 3+, we define the read preference on a collection |         # If _cursor_obj already exists, return it immediately. | ||||||
|             # level, not a cursor level. Thus, we need to get a cloned |         if self._cursor_obj is not None: | ||||||
|             # collection object using `with_options` first. |             return self._cursor_obj | ||||||
|             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: |         # Create a new PyMongo cursor. | ||||||
|                 # Apply query ordering |         # XXX In PyMongo 3+, we define the read preference on a collection | ||||||
|                 self._cursor_obj.sort(self._ordering) |         # level, not a cursor level. Thus, we need to get a cloned collection | ||||||
|             elif self._ordering is None and self._document._meta['ordering']: |         # object using `with_options` first. | ||||||
|                 # Otherwise, apply the ordering from the document model, unless |         if IS_PYMONGO_3 and self._read_preference is not None: | ||||||
|                 # it's been explicitly cleared via order_by with no arguments |             self._cursor_obj = self._collection\ | ||||||
|                 order = self._get_order_by(self._document._meta['ordering']) |                 .with_options(read_preference=self._read_preference)\ | ||||||
|                 self._cursor_obj.sort(order) |                 .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._limit is not None: |         # Apply ordering to the cursor. | ||||||
|                 self._cursor_obj.limit(self._limit) |         # 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._skip is not None: |         if self._limit is not None: | ||||||
|                 self._cursor_obj.skip(self._skip) |             self._cursor_obj.limit(self._limit) | ||||||
|  |  | ||||||
|             if self._hint != -1: |         if self._skip is not None: | ||||||
|                 self._cursor_obj.hint(self._hint) |             self._cursor_obj.skip(self._skip) | ||||||
|  |  | ||||||
|             if self._batch_size is not None: |         if self._hint != -1: | ||||||
|                 self._cursor_obj.batch_size(self._batch_size) |             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 | ||||||
|  |  | ||||||
| @@ -1698,7 +1746,13 @@ class BaseQuerySet(object): | |||||||
|         return ret |         return ret | ||||||
|  |  | ||||||
|     def _get_order_by(self, keys): |     def _get_order_by(self, keys): | ||||||
|         """Creates a list of order by fields""" |         """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)] | ||||||
|  |         """ | ||||||
|         key_list = [] |         key_list = [] | ||||||
|         for key in keys: |         for key in keys: | ||||||
|             if not key: |             if not key: | ||||||
| @@ -1711,17 +1765,19 @@ class BaseQuerySet(object): | |||||||
|             direction = pymongo.ASCENDING |             direction = pymongo.ASCENDING | ||||||
|             if key[0] == '-': |             if key[0] == '-': | ||||||
|                 direction = pymongo.DESCENDING |                 direction = pymongo.DESCENDING | ||||||
|  |  | ||||||
|             if key[0] in ('-', '+'): |             if key[0] in ('-', '+'): | ||||||
|                 key = key[1:] |                 key = key[1:] | ||||||
|  |  | ||||||
|             key = key.replace('__', '.') |             key = key.replace('__', '.') | ||||||
|             try: |             try: | ||||||
|                 key = self._document._translate_field_name(key) |                 key = self._document._translate_field_name(key) | ||||||
|             except Exception: |             except Exception: | ||||||
|  |                 # TODO this exception should be more specific | ||||||
|                 pass |                 pass | ||||||
|  |  | ||||||
|             key_list.append((key, direction)) |             key_list.append((key, direction)) | ||||||
|  |  | ||||||
|         if self._cursor_obj and key_list: |  | ||||||
|             self._cursor_obj.sort(key_list) |  | ||||||
|         return key_list |         return key_list | ||||||
|  |  | ||||||
|     def _get_scalar(self, doc): |     def _get_scalar(self, doc): | ||||||
| @@ -1819,10 +1875,21 @@ class BaseQuerySet(object): | |||||||
|         return code |         return code | ||||||
|  |  | ||||||
|     def _chainable_method(self, method_name, val): |     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() |         queryset = self.clone() | ||||||
|         method = getattr(queryset._cursor, method_name) |  | ||||||
|         method(val) |         # 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} | ||||||
|         setattr(queryset, '_' + method_name, val) |         setattr(queryset, '_' + method_name, val) | ||||||
|  |  | ||||||
|         return queryset |         return queryset | ||||||
|  |  | ||||||
|     # Deprecated |     # Deprecated | ||||||
|   | |||||||
| @@ -136,13 +136,15 @@ class QuerySet(BaseQuerySet): | |||||||
|         return self._len |         return self._len | ||||||
|  |  | ||||||
|     def no_cache(self): |     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 |         .. versionadded:: 0.8.3 Convert to non caching queryset | ||||||
|         """ |         """ | ||||||
|         if self._result_cache is not None: |         if self._result_cache is not None: | ||||||
|             raise OperationError('QuerySet already cached') |             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): | class QuerySetNoCache(BaseQuerySet): | ||||||
| @@ -153,7 +155,7 @@ class QuerySetNoCache(BaseQuerySet): | |||||||
|  |  | ||||||
|         .. versionadded:: 0.8.3 Convert to caching queryset |         .. 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): |     def __repr__(self): | ||||||
|         """Provides the string representation of the QuerySet |         """Provides the string representation of the QuerySet | ||||||
|   | |||||||
| @@ -2,14 +2,14 @@ | |||||||
| import unittest | import unittest | ||||||
| import sys | import sys | ||||||
|  |  | ||||||
|  |  | ||||||
| import pymongo |  | ||||||
|  |  | ||||||
| from nose.plugins.skip import SkipTest | from nose.plugins.skip import SkipTest | ||||||
| from datetime import datetime | from datetime import datetime | ||||||
|  | import pymongo | ||||||
|  |  | ||||||
| from mongoengine import * | from mongoengine import * | ||||||
| from mongoengine.connection import get_db, get_connection | from mongoengine.connection import get_db | ||||||
|  |  | ||||||
|  | from tests.utils import get_mongodb_version, needs_mongodb_v26 | ||||||
|  |  | ||||||
| __all__ = ("IndexesTest", ) | __all__ = ("IndexesTest", ) | ||||||
|  |  | ||||||
| @@ -494,8 +494,7 @@ class IndexesTest(unittest.TestCase): | |||||||
|         obj = Test(a=1) |         obj = Test(a=1) | ||||||
|         obj.save() |         obj.save() | ||||||
|  |  | ||||||
|         connection = get_connection() |         IS_MONGODB_3 = get_mongodb_version()[0] >= 3 | ||||||
|         IS_MONGODB_3 = connection.server_info()['versionArray'][0] >= 3 |  | ||||||
|  |  | ||||||
|         # Need to be explicit about covered indexes as mongoDB doesn't know if |         # Need to be explicit about covered indexes as mongoDB doesn't know if | ||||||
|         # the documents returned might have more keys in that here. |         # the documents returned might have more keys in that here. | ||||||
| @@ -733,14 +732,6 @@ class IndexesTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         Log.drop_collection() |         Log.drop_collection() | ||||||
|  |  | ||||||
|         if pymongo.version_tuple[0] < 2 and pymongo.version_tuple[1] < 3: |  | ||||||
|             raise SkipTest('pymongo needs to be 2.3 or higher for this test') |  | ||||||
|  |  | ||||||
|         connection = get_connection() |  | ||||||
|         version_array = connection.server_info()['versionArray'] |  | ||||||
|         if version_array[0] < 2 and version_array[1] < 2: |  | ||||||
|             raise SkipTest('MongoDB needs to be 2.2 or higher for this test') |  | ||||||
|  |  | ||||||
|         # Indexes are lazy so use list() to perform query |         # Indexes are lazy so use list() to perform query | ||||||
|         list(Log.objects) |         list(Log.objects) | ||||||
|         info = Log.objects._collection.index_information() |         info = Log.objects._collection.index_information() | ||||||
| @@ -874,8 +865,8 @@ class IndexesTest(unittest.TestCase): | |||||||
|                          info['provider_ids.foo_1_provider_ids.bar_1']['key']) |                          info['provider_ids.foo_1_provider_ids.bar_1']['key']) | ||||||
|         self.assertTrue(info['provider_ids.foo_1_provider_ids.bar_1']['sparse']) |         self.assertTrue(info['provider_ids.foo_1_provider_ids.bar_1']['sparse']) | ||||||
|  |  | ||||||
|  |     @needs_mongodb_v26 | ||||||
|     def test_text_indexes(self): |     def test_text_indexes(self): | ||||||
|  |  | ||||||
|         class Book(Document): |         class Book(Document): | ||||||
|             title = DictField() |             title = DictField() | ||||||
|             meta = { |             meta = { | ||||||
|   | |||||||
| @@ -1232,6 +1232,19 @@ class InstanceTest(unittest.TestCase): | |||||||
|         self.assertEqual(person.name, None) |         self.assertEqual(person.name, None) | ||||||
|         self.assertEqual(person.age, 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): |     def test_inserts_if_you_set_the_pk(self): | ||||||
|         p1 = self.Person(name='p1', id=bson.ObjectId()).save() |         p1 = self.Person(name='p1', id=bson.ObjectId()).save() | ||||||
|         p2 = self.Person(name='p2') |         p2 = self.Person(name='p2') | ||||||
|   | |||||||
| @@ -1,13 +1,12 @@ | |||||||
| # -*- coding: utf-8 -*- | # -*- coding: utf-8 -*- | ||||||
| import six |  | ||||||
| from nose.plugins.skip import SkipTest |  | ||||||
|  |  | ||||||
| import datetime | import datetime | ||||||
| import unittest | import unittest | ||||||
| import uuid | import uuid | ||||||
| import math | import math | ||||||
| import itertools | import itertools | ||||||
| import re | import re | ||||||
|  |  | ||||||
|  | from nose.plugins.skip import SkipTest | ||||||
| import six | import six | ||||||
|  |  | ||||||
| try: | try: | ||||||
| @@ -27,21 +26,13 @@ from mongoengine import * | |||||||
| from mongoengine.connection import get_db | from mongoengine.connection import get_db | ||||||
| from mongoengine.base import (BaseDict, BaseField, EmbeddedDocumentList, | from mongoengine.base import (BaseDict, BaseField, EmbeddedDocumentList, | ||||||
|                               _document_registry) |                               _document_registry) | ||||||
| from mongoengine.errors import NotRegistered, DoesNotExist |  | ||||||
|  | from tests.utils import MongoDBTestCase | ||||||
|  |  | ||||||
| __all__ = ("FieldTest", "EmbeddedDocumentListFieldTestCase") | __all__ = ("FieldTest", "EmbeddedDocumentListFieldTestCase") | ||||||
|  |  | ||||||
|  |  | ||||||
| class FieldTest(unittest.TestCase): | class FieldTest(MongoDBTestCase): | ||||||
|  |  | ||||||
|     def setUp(self): |  | ||||||
|         connect(db='mongoenginetest') |  | ||||||
|         self.db = get_db() |  | ||||||
|  |  | ||||||
|     def tearDown(self): |  | ||||||
|         self.db.drop_collection('fs.files') |  | ||||||
|         self.db.drop_collection('fs.chunks') |  | ||||||
|         self.db.drop_collection('mongoengine.counters') |  | ||||||
|  |  | ||||||
|     def test_default_values_nothing_set(self): |     def test_default_values_nothing_set(self): | ||||||
|         """Ensure that default field values are used when creating a document. |         """Ensure that default field values are used when creating a document. | ||||||
| @@ -227,9 +218,9 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertTrue(isinstance(ret.comp_dt_fld, datetime.datetime)) |         self.assertTrue(isinstance(ret.comp_dt_fld, datetime.datetime)) | ||||||
|  |  | ||||||
|     def test_not_required_handles_none_from_database(self): |     def test_not_required_handles_none_from_database(self): | ||||||
|         """Ensure that every fields can handle null values from the database. |         """Ensure that every field can handle null values from the | ||||||
|  |         database. | ||||||
|         """ |         """ | ||||||
|  |  | ||||||
|         class HandleNoneFields(Document): |         class HandleNoneFields(Document): | ||||||
|             str_fld = StringField(required=True) |             str_fld = StringField(required=True) | ||||||
|             int_fld = IntField(required=True) |             int_fld = IntField(required=True) | ||||||
| @@ -306,6 +297,24 @@ class FieldTest(unittest.TestCase): | |||||||
|         person.id = '497ce96f395f2f052a494fd4' |         person.id = '497ce96f395f2f052a494fd4' | ||||||
|         person.validate() |         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): |     def test_string_validation(self): | ||||||
|         """Ensure that invalid values cannot be assigned to string fields. |         """Ensure that invalid values cannot be assigned to string fields. | ||||||
|         """ |         """ | ||||||
| @@ -332,11 +341,12 @@ class FieldTest(unittest.TestCase): | |||||||
|         person.validate() |         person.validate() | ||||||
|  |  | ||||||
|     def test_url_validation(self): |     def test_url_validation(self): | ||||||
|         """Ensure that URLFields validate urls properly. |         """Ensure that URLFields validate urls properly.""" | ||||||
|         """ |  | ||||||
|         class Link(Document): |         class Link(Document): | ||||||
|             url = URLField() |             url = URLField() | ||||||
|  |  | ||||||
|  |         Link.drop_collection() | ||||||
|  |  | ||||||
|         link = Link() |         link = Link() | ||||||
|         link.url = 'google' |         link.url = 'google' | ||||||
|         self.assertRaises(ValidationError, link.validate) |         self.assertRaises(ValidationError, link.validate) | ||||||
| @@ -344,6 +354,27 @@ class FieldTest(unittest.TestCase): | |||||||
|         link.url = 'http://www.google.com:8080' |         link.url = 'http://www.google.com:8080' | ||||||
|         link.validate() |         link.validate() | ||||||
|  |  | ||||||
|  |     def test_unicode_url_validation(self): | ||||||
|  |         """Ensure unicode URLs are validated properly.""" | ||||||
|  |         class Link(Document): | ||||||
|  |             url = URLField() | ||||||
|  |  | ||||||
|  |         Link.drop_collection() | ||||||
|  |  | ||||||
|  |         link = Link() | ||||||
|  |         link.url = u'http://привет.com' | ||||||
|  |  | ||||||
|  |         # TODO fix URL validation - this *IS* a valid URL | ||||||
|  |         # For now we just want to make sure that the error message is correct | ||||||
|  |         try: | ||||||
|  |             link.validate() | ||||||
|  |             self.assertTrue(False) | ||||||
|  |         except ValidationError as e: | ||||||
|  |             self.assertEqual( | ||||||
|  |                 unicode(e), | ||||||
|  |                 u"ValidationError (Link:None) (Invalid URL: http://\u043f\u0440\u0438\u0432\u0435\u0442.com: ['url'])" | ||||||
|  |             ) | ||||||
|  |  | ||||||
|     def test_url_scheme_validation(self): |     def test_url_scheme_validation(self): | ||||||
|         """Ensure that URLFields validate urls with specific schemes properly. |         """Ensure that URLFields validate urls with specific schemes properly. | ||||||
|         """ |         """ | ||||||
| @@ -1967,7 +1998,7 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertEqual(content, User.objects.first().groups[0].content) |         self.assertEqual(content, User.objects.first().groups[0].content) | ||||||
|  |  | ||||||
|     def test_reference_miss(self): |     def test_reference_miss(self): | ||||||
|         """Ensure an exception is raised when dereferencing unknow document |         """Ensure an exception is raised when dereferencing unknown document | ||||||
|         """ |         """ | ||||||
|  |  | ||||||
|         class Foo(Document): |         class Foo(Document): | ||||||
| @@ -3168,26 +3199,42 @@ class FieldTest(unittest.TestCase): | |||||||
|         att.delete() |         att.delete() | ||||||
|         self.assertEqual(0, Attachment.objects.count()) |         self.assertEqual(0, Attachment.objects.count()) | ||||||
|  |  | ||||||
|     def test_choices_validation(self): |     def test_choices_allow_using_sets_as_choices(self): | ||||||
|         """Ensure that value is in a container of allowed values. |         """Ensure that sets can be used when setting choices | ||||||
|         """ |         """ | ||||||
|         class Shirt(Document): |         class Shirt(Document): | ||||||
|             size = StringField(max_length=3, choices=( |             size = StringField(choices={'M', 'L'}) | ||||||
|                 ('S', 'Small'), ('M', 'Medium'), ('L', 'Large'), |  | ||||||
|                 ('XL', 'Extra Large'), ('XXL', 'Extra Extra Large'))) |  | ||||||
|  |  | ||||||
|         Shirt.drop_collection() |         Shirt(size='M').validate() | ||||||
|  |  | ||||||
|  |     def test_choices_validation_allow_no_value(self): | ||||||
|  |         """Ensure that .validate passes and no value was provided | ||||||
|  |         for a field setup with choices | ||||||
|  |         """ | ||||||
|  |         class Shirt(Document): | ||||||
|  |             size = StringField(choices=('S', 'M')) | ||||||
|  |  | ||||||
|         shirt = Shirt() |         shirt = Shirt() | ||||||
|         shirt.validate() |         shirt.validate() | ||||||
|  |  | ||||||
|         shirt.size = "S" |     def test_choices_validation_accept_possible_value(self): | ||||||
|  |         """Ensure that value is in a container of allowed values. | ||||||
|  |         """ | ||||||
|  |         class Shirt(Document): | ||||||
|  |             size = StringField(choices=('S', 'M')) | ||||||
|  |  | ||||||
|  |         shirt = Shirt(size='S') | ||||||
|         shirt.validate() |         shirt.validate() | ||||||
|  |  | ||||||
|         shirt.size = "XS" |     def test_choices_validation_reject_unknown_value(self): | ||||||
|         self.assertRaises(ValidationError, shirt.validate) |         """Ensure that unallowed value are rejected upon validation | ||||||
|  |         """ | ||||||
|  |         class Shirt(Document): | ||||||
|  |             size = StringField(choices=('S', 'M')) | ||||||
|  |  | ||||||
|         Shirt.drop_collection() |         shirt = Shirt(size="XS") | ||||||
|  |         with self.assertRaises(ValidationError): | ||||||
|  |             shirt.validate() | ||||||
|  |  | ||||||
|     def test_choices_validation_documents(self): |     def test_choices_validation_documents(self): | ||||||
|         """ |         """ | ||||||
| @@ -3973,30 +4020,25 @@ class FieldTest(unittest.TestCase): | |||||||
|         """Tests if a `FieldDoesNotExist` exception is raised when trying to |         """Tests if a `FieldDoesNotExist` exception is raised when trying to | ||||||
|         instanciate a document with a field that's not defined. |         instanciate a document with a field that's not defined. | ||||||
|         """ |         """ | ||||||
|  |  | ||||||
|         class Doc(Document): |         class Doc(Document): | ||||||
|             foo = StringField(db_field='f') |             foo = StringField() | ||||||
|  |  | ||||||
|         def test(): |         with self.assertRaises(FieldDoesNotExist): | ||||||
|             Doc(bar='test') |             Doc(bar='test') | ||||||
|  |  | ||||||
|         self.assertRaises(FieldDoesNotExist, test) |  | ||||||
|  |  | ||||||
|     def test_undefined_field_exception_with_strict(self): |     def test_undefined_field_exception_with_strict(self): | ||||||
|         """Tests if a `FieldDoesNotExist` exception is raised when trying to |         """Tests if a `FieldDoesNotExist` exception is raised when trying to | ||||||
|         instanciate a document with a field that's not defined, |         instanciate a document with a field that's not defined, | ||||||
|         even when strict is set to False. |         even when strict is set to False. | ||||||
|         """ |         """ | ||||||
|  |  | ||||||
|         class Doc(Document): |         class Doc(Document): | ||||||
|             foo = StringField(db_field='f') |             foo = StringField() | ||||||
|             meta = {'strict': False} |             meta = {'strict': False} | ||||||
|  |  | ||||||
|         def test(): |         with self.assertRaises(FieldDoesNotExist): | ||||||
|             Doc(bar='test') |             Doc(bar='test') | ||||||
|  |  | ||||||
|         self.assertRaises(FieldDoesNotExist, test) |  | ||||||
|  |  | ||||||
|     def test_long_field_is_considered_as_int64(self): |     def test_long_field_is_considered_as_int64(self): | ||||||
|         """ |         """ | ||||||
|         Tests that long fields are stored as long in mongo, even if long value |         Tests that long fields are stored as long in mongo, even if long value | ||||||
| @@ -4011,12 +4053,13 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertTrue(isinstance(doc.some_long, six.integer_types)) |         self.assertTrue(isinstance(doc.some_long, six.integer_types)) | ||||||
|  |  | ||||||
|  |  | ||||||
| class EmbeddedDocumentListFieldTestCase(unittest.TestCase): | class EmbeddedDocumentListFieldTestCase(MongoDBTestCase): | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def setUpClass(cls): |  | ||||||
|         cls.db = connect(db='EmbeddedDocumentListFieldTestCase') |  | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         """ | ||||||
|  |         Create two BlogPost entries in the database, each with | ||||||
|  |         several EmbeddedDocuments. | ||||||
|  |         """ | ||||||
|         class Comments(EmbeddedDocument): |         class Comments(EmbeddedDocument): | ||||||
|             author = StringField() |             author = StringField() | ||||||
|             message = StringField() |             message = StringField() | ||||||
| @@ -4024,14 +4067,11 @@ class EmbeddedDocumentListFieldTestCase(unittest.TestCase): | |||||||
|         class BlogPost(Document): |         class BlogPost(Document): | ||||||
|             comments = EmbeddedDocumentListField(Comments) |             comments = EmbeddedDocumentListField(Comments) | ||||||
|  |  | ||||||
|         cls.Comments = Comments |         BlogPost.drop_collection() | ||||||
|         cls.BlogPost = BlogPost |  | ||||||
|  |         self.Comments = Comments | ||||||
|  |         self.BlogPost = BlogPost | ||||||
|  |  | ||||||
|     def setUp(self): |  | ||||||
|         """ |  | ||||||
|         Create two BlogPost entries in the database, each with |  | ||||||
|         several EmbeddedDocuments. |  | ||||||
|         """ |  | ||||||
|         self.post1 = self.BlogPost(comments=[ |         self.post1 = self.BlogPost(comments=[ | ||||||
|             self.Comments(author='user1', message='message1'), |             self.Comments(author='user1', message='message1'), | ||||||
|             self.Comments(author='user2', message='message1') |             self.Comments(author='user2', message='message1') | ||||||
| @@ -4043,13 +4083,6 @@ class EmbeddedDocumentListFieldTestCase(unittest.TestCase): | |||||||
|             self.Comments(author='user3', message='message1') |             self.Comments(author='user3', message='message1') | ||||||
|         ]).save() |         ]).save() | ||||||
|  |  | ||||||
|     def tearDown(self): |  | ||||||
|         self.BlogPost.drop_collection() |  | ||||||
|  |  | ||||||
|     @classmethod |  | ||||||
|     def tearDownClass(cls): |  | ||||||
|         cls.db.drop_database('EmbeddedDocumentListFieldTestCase') |  | ||||||
|  |  | ||||||
|     def test_no_keyword_filter(self): |     def test_no_keyword_filter(self): | ||||||
|         """ |         """ | ||||||
|         Tests the filter method of a List of Embedded Documents |         Tests the filter method of a List of Embedded Documents | ||||||
| @@ -4407,7 +4440,8 @@ class EmbeddedDocumentListFieldTestCase(unittest.TestCase): | |||||||
|             my_list = ListField(EmbeddedDocumentField(EmbeddedWithUnique)) |             my_list = ListField(EmbeddedDocumentField(EmbeddedWithUnique)) | ||||||
|  |  | ||||||
|         A(my_list=[]).save() |         A(my_list=[]).save() | ||||||
|         self.assertRaises(NotUniqueError, lambda: A(my_list=[]).save()) |         with self.assertRaises(NotUniqueError): | ||||||
|  |             A(my_list=[]).save() | ||||||
|  |  | ||||||
|         class EmbeddedWithSparseUnique(EmbeddedDocument): |         class EmbeddedWithSparseUnique(EmbeddedDocument): | ||||||
|             number = IntField(unique=True, sparse=True) |             number = IntField(unique=True, sparse=True) | ||||||
| @@ -4415,6 +4449,9 @@ class EmbeddedDocumentListFieldTestCase(unittest.TestCase): | |||||||
|         class B(Document): |         class B(Document): | ||||||
|             my_list = ListField(EmbeddedDocumentField(EmbeddedWithSparseUnique)) |             my_list = ListField(EmbeddedDocumentField(EmbeddedWithSparseUnique)) | ||||||
|  |  | ||||||
|  |         A.drop_collection() | ||||||
|  |         B.drop_collection() | ||||||
|  |  | ||||||
|         B(my_list=[]).save() |         B(my_list=[]).save() | ||||||
|         B(my_list=[]).save() |         B(my_list=[]).save() | ||||||
|  |  | ||||||
| @@ -4454,6 +4491,8 @@ class EmbeddedDocumentListFieldTestCase(unittest.TestCase): | |||||||
|             a_field = IntField() |             a_field = IntField() | ||||||
|             c_field = IntField(custom_data=custom_data) |             c_field = IntField(custom_data=custom_data) | ||||||
|  |  | ||||||
|  |         CustomData.drop_collection() | ||||||
|  |  | ||||||
|         a1 = CustomData(a_field=1, c_field=2).save() |         a1 = CustomData(a_field=1, c_field=2).save() | ||||||
|         self.assertEqual(2, a1.c_field) |         self.assertEqual(2, a1.c_field) | ||||||
|         self.assertFalse(hasattr(a1.c_field, 'custom_data')) |         self.assertFalse(hasattr(a1.c_field, 'custom_data')) | ||||||
|   | |||||||
| @@ -18,15 +18,13 @@ try: | |||||||
| except ImportError: | except ImportError: | ||||||
|     HAS_PIL = False |     HAS_PIL = False | ||||||
|  |  | ||||||
|  | from tests.utils import MongoDBTestCase | ||||||
|  |  | ||||||
| TEST_IMAGE_PATH = os.path.join(os.path.dirname(__file__), 'mongoengine.png') | TEST_IMAGE_PATH = os.path.join(os.path.dirname(__file__), 'mongoengine.png') | ||||||
| TEST_IMAGE2_PATH = os.path.join(os.path.dirname(__file__), 'mongodb_leaf.png') | TEST_IMAGE2_PATH = os.path.join(os.path.dirname(__file__), 'mongodb_leaf.png') | ||||||
|  |  | ||||||
|  |  | ||||||
| class FileTest(unittest.TestCase): | class FileTest(MongoDBTestCase): | ||||||
|  |  | ||||||
|     def setUp(self): |  | ||||||
|         connect(db='mongoenginetest') |  | ||||||
|         self.db = get_db() |  | ||||||
|  |  | ||||||
|     def tearDown(self): |     def tearDown(self): | ||||||
|         self.db.drop_collection('fs.files') |         self.db.drop_collection('fs.files') | ||||||
|   | |||||||
| @@ -1,105 +1,139 @@ | |||||||
| from datetime import datetime, timedelta | import datetime | ||||||
| import unittest | import unittest | ||||||
|  |  | ||||||
| from pymongo.errors import OperationFailure |  | ||||||
| from mongoengine import * | from mongoengine import * | ||||||
| from mongoengine.connection import get_connection |  | ||||||
| from nose.plugins.skip import SkipTest | from tests.utils import MongoDBTestCase, needs_mongodb_v3 | ||||||
|  |  | ||||||
|  |  | ||||||
| __all__ = ("GeoQueriesTest",) | __all__ = ("GeoQueriesTest",) | ||||||
|  |  | ||||||
|  |  | ||||||
| class GeoQueriesTest(unittest.TestCase): | class GeoQueriesTest(MongoDBTestCase): | ||||||
|  |  | ||||||
|     def setUp(self): |     def _create_event_data(self, point_field_class=GeoPointField): | ||||||
|         connect(db='mongoenginetest') |         """Create some sample data re-used in many of the tests below.""" | ||||||
|  |  | ||||||
|     def test_geospatial_operators(self): |  | ||||||
|         """Ensure that geospatial queries are working. |  | ||||||
|         """ |  | ||||||
|         class Event(Document): |         class Event(Document): | ||||||
|             title = StringField() |             title = StringField() | ||||||
|             date = DateTimeField() |             date = DateTimeField() | ||||||
|             location = GeoPointField() |             location = point_field_class() | ||||||
|  |  | ||||||
|             def __unicode__(self): |             def __unicode__(self): | ||||||
|                 return self.title |                 return self.title | ||||||
|  |  | ||||||
|  |         self.Event = Event | ||||||
|  |  | ||||||
|         Event.drop_collection() |         Event.drop_collection() | ||||||
|  |  | ||||||
|         event1 = Event(title="Coltrane Motion @ Double Door", |         event1 = Event.objects.create( | ||||||
|                        date=datetime.now() - timedelta(days=1), |             title="Coltrane Motion @ Double Door", | ||||||
|                        location=[-87.677137, 41.909889]).save() |             date=datetime.datetime.now() - datetime.timedelta(days=1), | ||||||
|         event2 = Event(title="Coltrane Motion @ Bottom of the Hill", |             location=[-87.677137, 41.909889]) | ||||||
|                        date=datetime.now() - timedelta(days=10), |         event2 = Event.objects.create( | ||||||
|                        location=[-122.4194155, 37.7749295]).save() |             title="Coltrane Motion @ Bottom of the Hill", | ||||||
|         event3 = Event(title="Coltrane Motion @ Empty Bottle", |             date=datetime.datetime.now() - datetime.timedelta(days=10), | ||||||
|                        date=datetime.now(), |             location=[-122.4194155, 37.7749295]) | ||||||
|                        location=[-87.686638, 41.900474]).save() |         event3 = Event.objects.create( | ||||||
|  |             title="Coltrane Motion @ Empty Bottle", | ||||||
|  |             date=datetime.datetime.now(), | ||||||
|  |             location=[-87.686638, 41.900474]) | ||||||
|  |  | ||||||
|  |         return event1, event2, event3 | ||||||
|  |  | ||||||
|  |     def test_near(self): | ||||||
|  |         """Make sure the "near" operator works.""" | ||||||
|  |         event1, event2, event3 = self._create_event_data() | ||||||
|  |  | ||||||
|         # find all events "near" pitchfork office, chicago. |         # find all events "near" pitchfork office, chicago. | ||||||
|         # note that "near" will show the san francisco event, too, |         # note that "near" will show the san francisco event, too, | ||||||
|         # although it sorts to last. |         # although it sorts to last. | ||||||
|         events = Event.objects(location__near=[-87.67892, 41.9120459]) |         events = self.Event.objects(location__near=[-87.67892, 41.9120459]) | ||||||
|         self.assertEqual(events.count(), 3) |         self.assertEqual(events.count(), 3) | ||||||
|         self.assertEqual(list(events), [event1, event3, event2]) |         self.assertEqual(list(events), [event1, event3, event2]) | ||||||
|  |  | ||||||
|  |         # ensure ordering is respected by "near" | ||||||
|  |         events = self.Event.objects(location__near=[-87.67892, 41.9120459]) | ||||||
|  |         events = events.order_by("-date") | ||||||
|  |         self.assertEqual(events.count(), 3) | ||||||
|  |         self.assertEqual(list(events), [event3, event1, event2]) | ||||||
|  |  | ||||||
|  |     def test_near_and_max_distance(self): | ||||||
|  |         """Ensure the "max_distance" operator works alongside the "near" | ||||||
|  |         operator. | ||||||
|  |         """ | ||||||
|  |         event1, event2, event3 = self._create_event_data() | ||||||
|  |  | ||||||
|  |         # find events within 10 degrees of san francisco | ||||||
|  |         point = [-122.415579, 37.7566023] | ||||||
|  |         events = self.Event.objects(location__near=point, | ||||||
|  |                                     location__max_distance=10) | ||||||
|  |         self.assertEqual(events.count(), 1) | ||||||
|  |         self.assertEqual(events[0], event2) | ||||||
|  |  | ||||||
|  |     # $minDistance was added in MongoDB v2.6, but continued being buggy | ||||||
|  |     # until v3.0; skip for older versions | ||||||
|  |     @needs_mongodb_v3 | ||||||
|  |     def test_near_and_min_distance(self): | ||||||
|  |         """Ensure the "min_distance" operator works alongside the "near" | ||||||
|  |         operator. | ||||||
|  |         """ | ||||||
|  |         event1, event2, event3 = self._create_event_data() | ||||||
|  |  | ||||||
|  |         # find events at least 10 degrees away of san francisco | ||||||
|  |         point = [-122.415579, 37.7566023] | ||||||
|  |         events = self.Event.objects(location__near=point, | ||||||
|  |                                     location__min_distance=10) | ||||||
|  |         self.assertEqual(events.count(), 2) | ||||||
|  |  | ||||||
|  |     def test_within_distance(self): | ||||||
|  |         """Make sure the "within_distance" operator works.""" | ||||||
|  |         event1, event2, event3 = self._create_event_data() | ||||||
|  |  | ||||||
|         # find events within 5 degrees of pitchfork office, chicago |         # find events within 5 degrees of pitchfork office, chicago | ||||||
|         point_and_distance = [[-87.67892, 41.9120459], 5] |         point_and_distance = [[-87.67892, 41.9120459], 5] | ||||||
|         events = Event.objects(location__within_distance=point_and_distance) |         events = self.Event.objects( | ||||||
|  |             location__within_distance=point_and_distance) | ||||||
|         self.assertEqual(events.count(), 2) |         self.assertEqual(events.count(), 2) | ||||||
|         events = list(events) |         events = list(events) | ||||||
|         self.assertTrue(event2 not in events) |         self.assertTrue(event2 not in events) | ||||||
|         self.assertTrue(event1 in events) |         self.assertTrue(event1 in events) | ||||||
|         self.assertTrue(event3 in events) |         self.assertTrue(event3 in events) | ||||||
|  |  | ||||||
|         # ensure ordering is respected by "near" |  | ||||||
|         events = Event.objects(location__near=[-87.67892, 41.9120459]) |  | ||||||
|         events = events.order_by("-date") |  | ||||||
|         self.assertEqual(events.count(), 3) |  | ||||||
|         self.assertEqual(list(events), [event3, event1, event2]) |  | ||||||
|  |  | ||||||
|         # find events within 10 degrees of san francisco |  | ||||||
|         point = [-122.415579, 37.7566023] |  | ||||||
|         events = Event.objects(location__near=point, location__max_distance=10) |  | ||||||
|         self.assertEqual(events.count(), 1) |  | ||||||
|         self.assertEqual(events[0], event2) |  | ||||||
|  |  | ||||||
|         # find events at least 10 degrees away of san francisco |  | ||||||
|         point = [-122.415579, 37.7566023] |  | ||||||
|         events = Event.objects(location__near=point, location__min_distance=10) |  | ||||||
|         # The following real test passes on MongoDB 3 but minDistance seems |  | ||||||
|         # buggy on older MongoDB versions |  | ||||||
|         if get_connection().server_info()['versionArray'][0] > 2: |  | ||||||
|             self.assertEqual(events.count(), 2) |  | ||||||
|         else: |  | ||||||
|             self.assertTrue(events.count() >= 2) |  | ||||||
|  |  | ||||||
|         # find events within 10 degrees of san francisco |         # find events within 10 degrees of san francisco | ||||||
|         point_and_distance = [[-122.415579, 37.7566023], 10] |         point_and_distance = [[-122.415579, 37.7566023], 10] | ||||||
|         events = Event.objects(location__within_distance=point_and_distance) |         events = self.Event.objects( | ||||||
|  |             location__within_distance=point_and_distance) | ||||||
|         self.assertEqual(events.count(), 1) |         self.assertEqual(events.count(), 1) | ||||||
|         self.assertEqual(events[0], event2) |         self.assertEqual(events[0], event2) | ||||||
|  |  | ||||||
|         # find events within 1 degree of greenpoint, broolyn, nyc, ny |         # find events within 1 degree of greenpoint, broolyn, nyc, ny | ||||||
|         point_and_distance = [[-73.9509714, 40.7237134], 1] |         point_and_distance = [[-73.9509714, 40.7237134], 1] | ||||||
|         events = Event.objects(location__within_distance=point_and_distance) |         events = self.Event.objects( | ||||||
|  |             location__within_distance=point_and_distance) | ||||||
|         self.assertEqual(events.count(), 0) |         self.assertEqual(events.count(), 0) | ||||||
|  |  | ||||||
|         # ensure ordering is respected by "within_distance" |         # ensure ordering is respected by "within_distance" | ||||||
|         point_and_distance = [[-87.67892, 41.9120459], 10] |         point_and_distance = [[-87.67892, 41.9120459], 10] | ||||||
|         events = Event.objects(location__within_distance=point_and_distance) |         events = self.Event.objects( | ||||||
|  |             location__within_distance=point_and_distance) | ||||||
|         events = events.order_by("-date") |         events = events.order_by("-date") | ||||||
|         self.assertEqual(events.count(), 2) |         self.assertEqual(events.count(), 2) | ||||||
|         self.assertEqual(events[0], event3) |         self.assertEqual(events[0], event3) | ||||||
|  |  | ||||||
|  |     def test_within_box(self): | ||||||
|  |         """Ensure the "within_box" operator works.""" | ||||||
|  |         event1, event2, event3 = self._create_event_data() | ||||||
|  |  | ||||||
|         # check that within_box works |         # check that within_box works | ||||||
|         box = [(-125.0, 35.0), (-100.0, 40.0)] |         box = [(-125.0, 35.0), (-100.0, 40.0)] | ||||||
|         events = Event.objects(location__within_box=box) |         events = self.Event.objects(location__within_box=box) | ||||||
|         self.assertEqual(events.count(), 1) |         self.assertEqual(events.count(), 1) | ||||||
|         self.assertEqual(events[0].id, event2.id) |         self.assertEqual(events[0].id, event2.id) | ||||||
|  |  | ||||||
|  |     def test_within_polygon(self): | ||||||
|  |         """Ensure the "within_polygon" operator works.""" | ||||||
|  |         event1, event2, event3 = self._create_event_data() | ||||||
|  |  | ||||||
|         polygon = [ |         polygon = [ | ||||||
|             (-87.694445, 41.912114), |             (-87.694445, 41.912114), | ||||||
|             (-87.69084, 41.919395), |             (-87.69084, 41.919395), | ||||||
| @@ -107,7 +141,7 @@ class GeoQueriesTest(unittest.TestCase): | |||||||
|             (-87.654276, 41.911731), |             (-87.654276, 41.911731), | ||||||
|             (-87.656164, 41.898061), |             (-87.656164, 41.898061), | ||||||
|         ] |         ] | ||||||
|         events = Event.objects(location__within_polygon=polygon) |         events = self.Event.objects(location__within_polygon=polygon) | ||||||
|         self.assertEqual(events.count(), 1) |         self.assertEqual(events.count(), 1) | ||||||
|         self.assertEqual(events[0].id, event1.id) |         self.assertEqual(events[0].id, event1.id) | ||||||
|  |  | ||||||
| @@ -116,13 +150,151 @@ class GeoQueriesTest(unittest.TestCase): | |||||||
|             (-1.225891, 52.792797), |             (-1.225891, 52.792797), | ||||||
|             (-4.40094, 53.389881) |             (-4.40094, 53.389881) | ||||||
|         ] |         ] | ||||||
|         events = Event.objects(location__within_polygon=polygon2) |         events = self.Event.objects(location__within_polygon=polygon2) | ||||||
|         self.assertEqual(events.count(), 0) |         self.assertEqual(events.count(), 0) | ||||||
|  |  | ||||||
|     def test_geo_spatial_embedded(self): |     def test_2dsphere_near(self): | ||||||
|  |         """Make sure the "near" operator works with a PointField, which | ||||||
|  |         corresponds to a 2dsphere index. | ||||||
|  |         """ | ||||||
|  |         event1, event2, event3 = self._create_event_data( | ||||||
|  |             point_field_class=PointField | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         # find all events "near" pitchfork office, chicago. | ||||||
|  |         # note that "near" will show the san francisco event, too, | ||||||
|  |         # although it sorts to last. | ||||||
|  |         events = self.Event.objects(location__near=[-87.67892, 41.9120459]) | ||||||
|  |         self.assertEqual(events.count(), 3) | ||||||
|  |         self.assertEqual(list(events), [event1, event3, event2]) | ||||||
|  |  | ||||||
|  |         # ensure ordering is respected by "near" | ||||||
|  |         events = self.Event.objects(location__near=[-87.67892, 41.9120459]) | ||||||
|  |         events = events.order_by("-date") | ||||||
|  |         self.assertEqual(events.count(), 3) | ||||||
|  |         self.assertEqual(list(events), [event3, event1, event2]) | ||||||
|  |  | ||||||
|  |     def test_2dsphere_near_and_max_distance(self): | ||||||
|  |         """Ensure the "max_distance" operator works alongside the "near" | ||||||
|  |         operator with a 2dsphere index. | ||||||
|  |         """ | ||||||
|  |         event1, event2, event3 = self._create_event_data( | ||||||
|  |             point_field_class=PointField | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         # find events within 10km of san francisco | ||||||
|  |         point = [-122.415579, 37.7566023] | ||||||
|  |         events = self.Event.objects(location__near=point, | ||||||
|  |                                     location__max_distance=10000) | ||||||
|  |         self.assertEqual(events.count(), 1) | ||||||
|  |         self.assertEqual(events[0], event2) | ||||||
|  |  | ||||||
|  |         # find events within 1km of greenpoint, broolyn, nyc, ny | ||||||
|  |         events = self.Event.objects(location__near=[-73.9509714, 40.7237134], | ||||||
|  |                                     location__max_distance=1000) | ||||||
|  |         self.assertEqual(events.count(), 0) | ||||||
|  |  | ||||||
|  |         # ensure ordering is respected by "near" | ||||||
|  |         events = self.Event.objects( | ||||||
|  |             location__near=[-87.67892, 41.9120459], | ||||||
|  |             location__max_distance=10000 | ||||||
|  |         ).order_by("-date") | ||||||
|  |         self.assertEqual(events.count(), 2) | ||||||
|  |         self.assertEqual(events[0], event3) | ||||||
|  |  | ||||||
|  |     def test_2dsphere_geo_within_box(self): | ||||||
|  |         """Ensure the "geo_within_box" operator works with a 2dsphere | ||||||
|  |         index. | ||||||
|  |         """ | ||||||
|  |         event1, event2, event3 = self._create_event_data( | ||||||
|  |             point_field_class=PointField | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         # check that within_box works | ||||||
|  |         box = [(-125.0, 35.0), (-100.0, 40.0)] | ||||||
|  |         events = self.Event.objects(location__geo_within_box=box) | ||||||
|  |         self.assertEqual(events.count(), 1) | ||||||
|  |         self.assertEqual(events[0].id, event2.id) | ||||||
|  |  | ||||||
|  |     def test_2dsphere_geo_within_polygon(self): | ||||||
|  |         """Ensure the "geo_within_polygon" operator works with a | ||||||
|  |         2dsphere index. | ||||||
|  |         """ | ||||||
|  |         event1, event2, event3 = self._create_event_data( | ||||||
|  |             point_field_class=PointField | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         polygon = [ | ||||||
|  |             (-87.694445, 41.912114), | ||||||
|  |             (-87.69084, 41.919395), | ||||||
|  |             (-87.681742, 41.927186), | ||||||
|  |             (-87.654276, 41.911731), | ||||||
|  |             (-87.656164, 41.898061), | ||||||
|  |         ] | ||||||
|  |         events = self.Event.objects(location__geo_within_polygon=polygon) | ||||||
|  |         self.assertEqual(events.count(), 1) | ||||||
|  |         self.assertEqual(events[0].id, event1.id) | ||||||
|  |  | ||||||
|  |         polygon2 = [ | ||||||
|  |             (-1.742249, 54.033586), | ||||||
|  |             (-1.225891, 52.792797), | ||||||
|  |             (-4.40094, 53.389881) | ||||||
|  |         ] | ||||||
|  |         events = self.Event.objects(location__geo_within_polygon=polygon2) | ||||||
|  |         self.assertEqual(events.count(), 0) | ||||||
|  |  | ||||||
|  |     # $minDistance was added in MongoDB v2.6, but continued being buggy | ||||||
|  |     # until v3.0; skip for older versions | ||||||
|  |     @needs_mongodb_v3 | ||||||
|  |     def test_2dsphere_near_and_min_max_distance(self): | ||||||
|  |         """Ensure "min_distace" and "max_distance" operators work well | ||||||
|  |         together with the "near" operator in a 2dsphere index. | ||||||
|  |         """ | ||||||
|  |         event1, event2, event3 = self._create_event_data( | ||||||
|  |             point_field_class=PointField | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         # ensure min_distance and max_distance combine well | ||||||
|  |         events = self.Event.objects( | ||||||
|  |             location__near=[-87.67892, 41.9120459], | ||||||
|  |             location__min_distance=1000, | ||||||
|  |             location__max_distance=10000 | ||||||
|  |         ).order_by("-date") | ||||||
|  |         self.assertEqual(events.count(), 1) | ||||||
|  |         self.assertEqual(events[0], event3) | ||||||
|  |  | ||||||
|  |         # ensure ordering is respected by "near" with "min_distance" | ||||||
|  |         events = self.Event.objects( | ||||||
|  |             location__near=[-87.67892, 41.9120459], | ||||||
|  |             location__min_distance=10000 | ||||||
|  |         ).order_by("-date") | ||||||
|  |         self.assertEqual(events.count(), 1) | ||||||
|  |         self.assertEqual(events[0], event2) | ||||||
|  |  | ||||||
|  |     def test_2dsphere_geo_within_center(self): | ||||||
|  |         """Make sure the "geo_within_center" operator works with a | ||||||
|  |         2dsphere index. | ||||||
|  |         """ | ||||||
|  |         event1, event2, event3 = self._create_event_data( | ||||||
|  |             point_field_class=PointField | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         # find events within 5 degrees of pitchfork office, chicago | ||||||
|  |         point_and_distance = [[-87.67892, 41.9120459], 2] | ||||||
|  |         events = self.Event.objects( | ||||||
|  |             location__geo_within_center=point_and_distance) | ||||||
|  |         self.assertEqual(events.count(), 2) | ||||||
|  |         events = list(events) | ||||||
|  |         self.assertTrue(event2 not in events) | ||||||
|  |         self.assertTrue(event1 in events) | ||||||
|  |         self.assertTrue(event3 in events) | ||||||
|  |  | ||||||
|  |     def _test_embedded(self, point_field_class): | ||||||
|  |         """Helper test method ensuring given point field class works | ||||||
|  |         well in an embedded document. | ||||||
|  |         """ | ||||||
|         class Venue(EmbeddedDocument): |         class Venue(EmbeddedDocument): | ||||||
|             location = GeoPointField() |             location = point_field_class() | ||||||
|             name = StringField() |             name = StringField() | ||||||
|  |  | ||||||
|         class Event(Document): |         class Event(Document): | ||||||
| @@ -148,16 +320,18 @@ class GeoQueriesTest(unittest.TestCase): | |||||||
|         self.assertEqual(events.count(), 3) |         self.assertEqual(events.count(), 3) | ||||||
|         self.assertEqual(list(events), [event1, event3, event2]) |         self.assertEqual(list(events), [event1, event3, event2]) | ||||||
|  |  | ||||||
|     def test_spherical_geospatial_operators(self): |     def test_geo_spatial_embedded(self): | ||||||
|         """Ensure that spherical geospatial queries are working |         """Make sure GeoPointField works properly in an embedded document.""" | ||||||
|         """ |         self._test_embedded(point_field_class=GeoPointField) | ||||||
|         # Needs MongoDB > 2.6.4 https://jira.mongodb.org/browse/SERVER-14039 |  | ||||||
|         connection = get_connection() |  | ||||||
|         info = connection.test.command('buildInfo') |  | ||||||
|         mongodb_version = tuple([int(i) for i in info['version'].split('.')]) |  | ||||||
|         if mongodb_version < (2, 6, 4): |  | ||||||
|             raise SkipTest("Need MongoDB version 2.6.4+") |  | ||||||
|  |  | ||||||
|  |     def test_2dsphere_point_embedded(self): | ||||||
|  |         """Make sure PointField works properly in an embedded document.""" | ||||||
|  |         self._test_embedded(point_field_class=PointField) | ||||||
|  |  | ||||||
|  |     # Needs MongoDB > 2.6.4 https://jira.mongodb.org/browse/SERVER-14039 | ||||||
|  |     @needs_mongodb_v3 | ||||||
|  |     def test_spherical_geospatial_operators(self): | ||||||
|  |         """Ensure that spherical geospatial queries are working.""" | ||||||
|         class Point(Document): |         class Point(Document): | ||||||
|             location = GeoPointField() |             location = GeoPointField() | ||||||
|  |  | ||||||
| @@ -177,7 +351,10 @@ class GeoQueriesTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         # Same behavior for _within_spherical_distance |         # Same behavior for _within_spherical_distance | ||||||
|         points = Point.objects( |         points = Point.objects( | ||||||
|             location__within_spherical_distance=[[-122, 37.5], 60 / earth_radius] |             location__within_spherical_distance=[ | ||||||
|  |                 [-122, 37.5], | ||||||
|  |                 60 / earth_radius | ||||||
|  |             ] | ||||||
|         ) |         ) | ||||||
|         self.assertEqual(points.count(), 2) |         self.assertEqual(points.count(), 2) | ||||||
|  |  | ||||||
| @@ -194,14 +371,9 @@ class GeoQueriesTest(unittest.TestCase): | |||||||
|         # Test query works with min_distance, being farer from one point |         # Test query works with min_distance, being farer from one point | ||||||
|         points = Point.objects(location__near_sphere=[-122, 37.8], |         points = Point.objects(location__near_sphere=[-122, 37.8], | ||||||
|                                location__min_distance=60 / earth_radius) |                                location__min_distance=60 / earth_radius) | ||||||
|         # The following real test passes on MongoDB 3 but minDistance seems |         self.assertEqual(points.count(), 1) | ||||||
|         # buggy on older MongoDB versions |         far_point = points.first() | ||||||
|         if get_connection().server_info()['versionArray'][0] > 2: |         self.assertNotEqual(close_point, far_point) | ||||||
|             self.assertEqual(points.count(), 1) |  | ||||||
|             far_point = points.first() |  | ||||||
|             self.assertNotEqual(close_point, far_point) |  | ||||||
|         else: |  | ||||||
|             self.assertTrue(points.count() >= 1) |  | ||||||
|  |  | ||||||
|         # Finds both points, but orders the north point first because it's |         # Finds both points, but orders the north point first because it's | ||||||
|         # closer to the reference point to the north. |         # closer to the reference point to the north. | ||||||
| @@ -220,141 +392,15 @@ class GeoQueriesTest(unittest.TestCase): | |||||||
|         # Finds only one point because only the first point is within 60km of |         # Finds only one point because only the first point is within 60km of | ||||||
|         # the reference point to the south. |         # the reference point to the south. | ||||||
|         points = Point.objects( |         points = Point.objects( | ||||||
|             location__within_spherical_distance=[[-122, 36.5], 60/earth_radius]) |             location__within_spherical_distance=[ | ||||||
|  |                 [-122, 36.5], | ||||||
|  |                 60 / earth_radius | ||||||
|  |             ] | ||||||
|  |         ) | ||||||
|         self.assertEqual(points.count(), 1) |         self.assertEqual(points.count(), 1) | ||||||
|         self.assertEqual(points[0].id, south_point.id) |         self.assertEqual(points[0].id, south_point.id) | ||||||
|  |  | ||||||
|     def test_2dsphere_point(self): |  | ||||||
|  |  | ||||||
|         class Event(Document): |  | ||||||
|             title = StringField() |  | ||||||
|             date = DateTimeField() |  | ||||||
|             location = PointField() |  | ||||||
|  |  | ||||||
|             def __unicode__(self): |  | ||||||
|                 return self.title |  | ||||||
|  |  | ||||||
|         Event.drop_collection() |  | ||||||
|  |  | ||||||
|         event1 = Event(title="Coltrane Motion @ Double Door", |  | ||||||
|                        date=datetime.now() - timedelta(days=1), |  | ||||||
|                        location=[-87.677137, 41.909889]) |  | ||||||
|         event1.save() |  | ||||||
|         event2 = Event(title="Coltrane Motion @ Bottom of the Hill", |  | ||||||
|                        date=datetime.now() - timedelta(days=10), |  | ||||||
|                        location=[-122.4194155, 37.7749295]).save() |  | ||||||
|         event3 = Event(title="Coltrane Motion @ Empty Bottle", |  | ||||||
|                        date=datetime.now(), |  | ||||||
|                        location=[-87.686638, 41.900474]).save() |  | ||||||
|  |  | ||||||
|         # find all events "near" pitchfork office, chicago. |  | ||||||
|         # note that "near" will show the san francisco event, too, |  | ||||||
|         # although it sorts to last. |  | ||||||
|         events = Event.objects(location__near=[-87.67892, 41.9120459]) |  | ||||||
|         self.assertEqual(events.count(), 3) |  | ||||||
|         self.assertEqual(list(events), [event1, event3, event2]) |  | ||||||
|  |  | ||||||
|         # find events within 5 degrees of pitchfork office, chicago |  | ||||||
|         point_and_distance = [[-87.67892, 41.9120459], 2] |  | ||||||
|         events = Event.objects(location__geo_within_center=point_and_distance) |  | ||||||
|         self.assertEqual(events.count(), 2) |  | ||||||
|         events = list(events) |  | ||||||
|         self.assertTrue(event2 not in events) |  | ||||||
|         self.assertTrue(event1 in events) |  | ||||||
|         self.assertTrue(event3 in events) |  | ||||||
|  |  | ||||||
|         # ensure ordering is respected by "near" |  | ||||||
|         events = Event.objects(location__near=[-87.67892, 41.9120459]) |  | ||||||
|         events = events.order_by("-date") |  | ||||||
|         self.assertEqual(events.count(), 3) |  | ||||||
|         self.assertEqual(list(events), [event3, event1, event2]) |  | ||||||
|  |  | ||||||
|         # find events within 10km of san francisco |  | ||||||
|         point = [-122.415579, 37.7566023] |  | ||||||
|         events = Event.objects(location__near=point, location__max_distance=10000) |  | ||||||
|         self.assertEqual(events.count(), 1) |  | ||||||
|         self.assertEqual(events[0], event2) |  | ||||||
|  |  | ||||||
|         # find events within 1km of greenpoint, broolyn, nyc, ny |  | ||||||
|         events = Event.objects(location__near=[-73.9509714, 40.7237134], location__max_distance=1000) |  | ||||||
|         self.assertEqual(events.count(), 0) |  | ||||||
|  |  | ||||||
|         # ensure ordering is respected by "near" |  | ||||||
|         events = Event.objects(location__near=[-87.67892, 41.9120459], |  | ||||||
|                                location__max_distance=10000).order_by("-date") |  | ||||||
|         self.assertEqual(events.count(), 2) |  | ||||||
|         self.assertEqual(events[0], event3) |  | ||||||
|  |  | ||||||
|         # ensure min_distance and max_distance combine well |  | ||||||
|         events = Event.objects(location__near=[-87.67892, 41.9120459], |  | ||||||
|                                location__min_distance=1000, |  | ||||||
|                                location__max_distance=10000).order_by("-date") |  | ||||||
|         self.assertEqual(events.count(), 1) |  | ||||||
|         self.assertEqual(events[0], event3) |  | ||||||
|  |  | ||||||
|         # ensure ordering is respected by "near" |  | ||||||
|         events = Event.objects(location__near=[-87.67892, 41.9120459], |  | ||||||
|                                # location__min_distance=10000 |  | ||||||
|                                location__min_distance=10000).order_by("-date") |  | ||||||
|         self.assertEqual(events.count(), 1) |  | ||||||
|         self.assertEqual(events[0], event2) |  | ||||||
|  |  | ||||||
|         # check that within_box works |  | ||||||
|         box = [(-125.0, 35.0), (-100.0, 40.0)] |  | ||||||
|         events = Event.objects(location__geo_within_box=box) |  | ||||||
|         self.assertEqual(events.count(), 1) |  | ||||||
|         self.assertEqual(events[0].id, event2.id) |  | ||||||
|  |  | ||||||
|         polygon = [ |  | ||||||
|             (-87.694445, 41.912114), |  | ||||||
|             (-87.69084, 41.919395), |  | ||||||
|             (-87.681742, 41.927186), |  | ||||||
|             (-87.654276, 41.911731), |  | ||||||
|             (-87.656164, 41.898061), |  | ||||||
|         ] |  | ||||||
|         events = Event.objects(location__geo_within_polygon=polygon) |  | ||||||
|         self.assertEqual(events.count(), 1) |  | ||||||
|         self.assertEqual(events[0].id, event1.id) |  | ||||||
|  |  | ||||||
|         polygon2 = [ |  | ||||||
|             (-1.742249, 54.033586), |  | ||||||
|             (-1.225891, 52.792797), |  | ||||||
|             (-4.40094, 53.389881) |  | ||||||
|         ] |  | ||||||
|         events = Event.objects(location__geo_within_polygon=polygon2) |  | ||||||
|         self.assertEqual(events.count(), 0) |  | ||||||
|  |  | ||||||
|     def test_2dsphere_point_embedded(self): |  | ||||||
|  |  | ||||||
|         class Venue(EmbeddedDocument): |  | ||||||
|             location = GeoPointField() |  | ||||||
|             name = StringField() |  | ||||||
|  |  | ||||||
|         class Event(Document): |  | ||||||
|             title = StringField() |  | ||||||
|             venue = EmbeddedDocumentField(Venue) |  | ||||||
|  |  | ||||||
|         Event.drop_collection() |  | ||||||
|  |  | ||||||
|         venue1 = Venue(name="The Rock", location=[-87.677137, 41.909889]) |  | ||||||
|         venue2 = Venue(name="The Bridge", location=[-122.4194155, 37.7749295]) |  | ||||||
|  |  | ||||||
|         event1 = Event(title="Coltrane Motion @ Double Door", |  | ||||||
|                        venue=venue1).save() |  | ||||||
|         event2 = Event(title="Coltrane Motion @ Bottom of the Hill", |  | ||||||
|                        venue=venue2).save() |  | ||||||
|         event3 = Event(title="Coltrane Motion @ Empty Bottle", |  | ||||||
|                        venue=venue1).save() |  | ||||||
|  |  | ||||||
|         # find all events "near" pitchfork office, chicago. |  | ||||||
|         # note that "near" will show the san francisco event, too, |  | ||||||
|         # although it sorts to last. |  | ||||||
|         events = Event.objects(venue__location__near=[-87.67892, 41.9120459]) |  | ||||||
|         self.assertEqual(events.count(), 3) |  | ||||||
|         self.assertEqual(list(events), [event1, event3, event2]) |  | ||||||
|  |  | ||||||
|     def test_linestring(self): |     def test_linestring(self): | ||||||
|  |  | ||||||
|         class Road(Document): |         class Road(Document): | ||||||
|             name = StringField() |             name = StringField() | ||||||
|             line = LineStringField() |             line = LineStringField() | ||||||
| @@ -410,7 +456,6 @@ class GeoQueriesTest(unittest.TestCase): | |||||||
|         self.assertEqual(1, roads) |         self.assertEqual(1, roads) | ||||||
|  |  | ||||||
|     def test_polygon(self): |     def test_polygon(self): | ||||||
|  |  | ||||||
|         class Road(Document): |         class Road(Document): | ||||||
|             name = StringField() |             name = StringField() | ||||||
|             poly = PolygonField() |             poly = PolygonField() | ||||||
| @@ -507,5 +552,6 @@ class GeoQueriesTest(unittest.TestCase): | |||||||
|         loc = Location.objects.as_pymongo()[0] |         loc = Location.objects.as_pymongo()[0] | ||||||
|         self.assertEqual(loc["poly"], {"type": "Polygon", "coordinates": [[[40, 4], [40, 6], [41, 6], [40, 4]]]}) |         self.assertEqual(loc["poly"], {"type": "Polygon", "coordinates": [[[40, 4], [40, 6], [41, 6], [40, 4]]]}) | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -19,6 +19,9 @@ from mongoengine.python_support import IS_PYMONGO_3 | |||||||
| from mongoengine.queryset import (DoesNotExist, MultipleObjectsReturned, | from mongoengine.queryset import (DoesNotExist, MultipleObjectsReturned, | ||||||
|                                   QuerySet, QuerySetManager, queryset_manager) |                                   QuerySet, QuerySetManager, queryset_manager) | ||||||
|  |  | ||||||
|  | from tests.utils import needs_mongodb_v26, skip_pymongo3 | ||||||
|  |  | ||||||
|  |  | ||||||
| __all__ = ("QuerySetTest",) | __all__ = ("QuerySetTest",) | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -32,37 +35,6 @@ class db_ops_tracker(query_counter): | |||||||
|         return list(self.db.system.profile.find(ignore_query)) |         return list(self.db.system.profile.find(ignore_query)) | ||||||
|  |  | ||||||
|  |  | ||||||
| def skip_older_mongodb(f): |  | ||||||
|     def _inner(*args, **kwargs): |  | ||||||
|         connection = get_connection() |  | ||||||
|         info = connection.test.command('buildInfo') |  | ||||||
|         mongodb_version = tuple([int(i) for i in info['version'].split('.')]) |  | ||||||
|  |  | ||||||
|         if mongodb_version < (2, 6): |  | ||||||
|             raise SkipTest("Need MongoDB version 2.6+") |  | ||||||
|  |  | ||||||
|         return f(*args, **kwargs) |  | ||||||
|  |  | ||||||
|     _inner.__name__ = f.__name__ |  | ||||||
|     _inner.__doc__ = f.__doc__ |  | ||||||
|  |  | ||||||
|     return _inner |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def skip_pymongo3(f): |  | ||||||
|     def _inner(*args, **kwargs): |  | ||||||
|  |  | ||||||
|         if IS_PYMONGO_3: |  | ||||||
|             raise SkipTest("Useless with PyMongo 3+") |  | ||||||
|  |  | ||||||
|         return f(*args, **kwargs) |  | ||||||
|  |  | ||||||
|     _inner.__name__ = f.__name__ |  | ||||||
|     _inner.__doc__ = f.__doc__ |  | ||||||
|  |  | ||||||
|     return _inner |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class QuerySetTest(unittest.TestCase): | class QuerySetTest(unittest.TestCase): | ||||||
|  |  | ||||||
|     def setUp(self): |     def setUp(self): | ||||||
| @@ -106,58 +78,111 @@ class QuerySetTest(unittest.TestCase): | |||||||
|             list(BlogPost.objects(author2__name="test")) |             list(BlogPost.objects(author2__name="test")) | ||||||
|  |  | ||||||
|     def test_find(self): |     def test_find(self): | ||||||
|         """Ensure that a query returns a valid set of results. |         """Ensure that a query returns a valid set of results.""" | ||||||
|         """ |         user_a = self.Person.objects.create(name='User A', age=20) | ||||||
|         self.Person(name="User A", age=20).save() |         user_b = self.Person.objects.create(name='User B', age=30) | ||||||
|         self.Person(name="User B", age=30).save() |  | ||||||
|  |  | ||||||
|         # Find all people in the collection |         # Find all people in the collection | ||||||
|         people = self.Person.objects |         people = self.Person.objects | ||||||
|         self.assertEqual(people.count(), 2) |         self.assertEqual(people.count(), 2) | ||||||
|         results = list(people) |         results = list(people) | ||||||
|  |  | ||||||
|         self.assertTrue(isinstance(results[0], self.Person)) |         self.assertTrue(isinstance(results[0], self.Person)) | ||||||
|         self.assertTrue(isinstance(results[0].id, (ObjectId, str, unicode))) |         self.assertTrue(isinstance(results[0].id, (ObjectId, str, unicode))) | ||||||
|         self.assertEqual(results[0].name, "User A") |  | ||||||
|  |         self.assertEqual(results[0], user_a) | ||||||
|  |         self.assertEqual(results[0].name, 'User A') | ||||||
|         self.assertEqual(results[0].age, 20) |         self.assertEqual(results[0].age, 20) | ||||||
|         self.assertEqual(results[1].name, "User B") |  | ||||||
|  |         self.assertEqual(results[1], user_b) | ||||||
|  |         self.assertEqual(results[1].name, 'User B') | ||||||
|         self.assertEqual(results[1].age, 30) |         self.assertEqual(results[1].age, 30) | ||||||
|  |  | ||||||
|         # Use a query to filter the people found to just person1 |         # Filter people by age | ||||||
|         people = self.Person.objects(age=20) |         people = self.Person.objects(age=20) | ||||||
|         self.assertEqual(people.count(), 1) |         self.assertEqual(people.count(), 1) | ||||||
|         person = people.next() |         person = people.next() | ||||||
|  |         self.assertEqual(person, user_a) | ||||||
|         self.assertEqual(person.name, "User A") |         self.assertEqual(person.name, "User A") | ||||||
|         self.assertEqual(person.age, 20) |         self.assertEqual(person.age, 20) | ||||||
|  |  | ||||||
|         # Test limit |     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 | ||||||
|         people = list(self.Person.objects.limit(1)) |         people = list(self.Person.objects.limit(1)) | ||||||
|         self.assertEqual(len(people), 1) |         self.assertEqual(len(people), 1) | ||||||
|         self.assertEqual(people[0].name, 'User A') |         self.assertEqual(people[0], user_a) | ||||||
|  |  | ||||||
|         # Test skip |         # 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 | ||||||
|         people = list(self.Person.objects.skip(1)) |         people = list(self.Person.objects.skip(1)) | ||||||
|         self.assertEqual(len(people), 1) |         self.assertEqual(len(people), 1) | ||||||
|         self.assertEqual(people[0].name, 'User B') |         self.assertEqual(people[0], user_b) | ||||||
|  |  | ||||||
|         person3 = self.Person(name="User C", age=40) |         # Test skip on an existing queryset | ||||||
|         person3.save() |         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) | ||||||
|  |  | ||||||
|         # Test slice limit |         # Test slice limit | ||||||
|         people = list(self.Person.objects[:2]) |         people = list(self.Person.objects[:2]) | ||||||
|         self.assertEqual(len(people), 2) |         self.assertEqual(len(people), 2) | ||||||
|         self.assertEqual(people[0].name, 'User A') |         self.assertEqual(people[0], user_a) | ||||||
|         self.assertEqual(people[1].name, 'User B') |         self.assertEqual(people[1], user_b) | ||||||
|  |  | ||||||
|         # Test slice skip |         # Test slice skip | ||||||
|         people = list(self.Person.objects[1:]) |         people = list(self.Person.objects[1:]) | ||||||
|         self.assertEqual(len(people), 2) |         self.assertEqual(len(people), 2) | ||||||
|         self.assertEqual(people[0].name, 'User B') |         self.assertEqual(people[0], user_b) | ||||||
|         self.assertEqual(people[1].name, 'User C') |         self.assertEqual(people[1], user_c) | ||||||
|  |  | ||||||
|         # Test slice limit and skip |         # Test slice limit and skip | ||||||
|         people = list(self.Person.objects[1:2]) |         people = list(self.Person.objects[1:2]) | ||||||
|         self.assertEqual(len(people), 1) |         self.assertEqual(len(people), 1) | ||||||
|         self.assertEqual(people[0].name, 'User B') |         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) | ||||||
|  |  | ||||||
|         # Test slice limit and skip cursor reset |         # Test slice limit and skip cursor reset | ||||||
|         qs = self.Person.objects[1:2] |         qs = self.Person.objects[1:2] | ||||||
| @@ -168,6 +193,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         self.assertEqual(len(people), 1) |         self.assertEqual(len(people), 1) | ||||||
|         self.assertEqual(people[0].name, 'User B') |         self.assertEqual(people[0].name, 'User B') | ||||||
|  |  | ||||||
|  |         # Test empty slice | ||||||
|         people = list(self.Person.objects[1:1]) |         people = list(self.Person.objects[1:1]) | ||||||
|         self.assertEqual(len(people), 0) |         self.assertEqual(len(people), 0) | ||||||
|  |  | ||||||
| @@ -187,12 +213,6 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         self.assertEqual("[<Person: Person object>, <Person: Person object>]", |         self.assertEqual("[<Person: Person object>, <Person: Person object>]", | ||||||
|                          "%s" % self.Person.objects[51:53]) |                          "%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): |     def test_find_one(self): | ||||||
|         """Ensure that a query using find_one returns a valid result. |         """Ensure that a query using find_one returns a valid result. | ||||||
|         """ |         """ | ||||||
| @@ -551,16 +571,23 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         self.assertEqual(post.comments[0].by, 'joe') |         self.assertEqual(post.comments[0].by, 'joe') | ||||||
|         self.assertEqual(post.comments[0].votes.score, 4) |         self.assertEqual(post.comments[0].votes.score, 4) | ||||||
|  |  | ||||||
|  |     @needs_mongodb_v26 | ||||||
|     def test_update_min_max(self): |     def test_update_min_max(self): | ||||||
|         class Scores(Document): |         class Scores(Document): | ||||||
|             high_score = IntField() |             high_score = IntField() | ||||||
|             low_score = IntField() |             low_score = IntField() | ||||||
|         scores = Scores(high_score=800, low_score=200) |  | ||||||
|         scores.save() |         scores = Scores.objects.create(high_score=800, low_score=200) | ||||||
|  |  | ||||||
|         Scores.objects(id=scores.id).update(min__low_score=150) |         Scores.objects(id=scores.id).update(min__low_score=150) | ||||||
|         self.assertEqual(Scores.objects(id=scores.id).get().low_score, 150) |         self.assertEqual(Scores.objects.get(id=scores.id).low_score, 150) | ||||||
|         Scores.objects(id=scores.id).update(min__low_score=250) |         Scores.objects(id=scores.id).update(min__low_score=250) | ||||||
|         self.assertEqual(Scores.objects(id=scores.id).get().low_score, 150) |         self.assertEqual(Scores.objects.get(id=scores.id).low_score, 150) | ||||||
|  |  | ||||||
|  |         Scores.objects(id=scores.id).update(max__high_score=1000) | ||||||
|  |         self.assertEqual(Scores.objects.get(id=scores.id).high_score, 1000) | ||||||
|  |         Scores.objects(id=scores.id).update(max__high_score=500) | ||||||
|  |         self.assertEqual(Scores.objects.get(id=scores.id).high_score, 1000) | ||||||
|  |  | ||||||
|     def test_updates_can_have_match_operators(self): |     def test_updates_can_have_match_operators(self): | ||||||
|  |  | ||||||
| @@ -964,7 +991,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         self.assertEqual(person.name, "User A") |         self.assertEqual(person.name, "User A") | ||||||
|         self.assertEqual(person.age, 20) |         self.assertEqual(person.age, 20) | ||||||
|  |  | ||||||
|     @skip_older_mongodb |     @needs_mongodb_v26 | ||||||
|     @skip_pymongo3 |     @skip_pymongo3 | ||||||
|     def test_cursor_args(self): |     def test_cursor_args(self): | ||||||
|         """Ensures the cursor args can be set as expected |         """Ensures the cursor args can be set as expected | ||||||
| @@ -1226,6 +1253,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         BlogPost.drop_collection() |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
|  |         # default ordering should be used by default | ||||||
|         with db_ops_tracker() as q: |         with db_ops_tracker() as q: | ||||||
|             BlogPost.objects.filter(title='whatever').first() |             BlogPost.objects.filter(title='whatever').first() | ||||||
|             self.assertEqual(len(q.get_ops()), 1) |             self.assertEqual(len(q.get_ops()), 1) | ||||||
| @@ -1234,11 +1262,28 @@ class QuerySetTest(unittest.TestCase): | |||||||
|                 {'published_date': -1} |                 {'published_date': -1} | ||||||
|             ) |             ) | ||||||
|  |  | ||||||
|  |         # calling order_by() should clear the default ordering | ||||||
|         with db_ops_tracker() as q: |         with db_ops_tracker() as q: | ||||||
|             BlogPost.objects.filter(title='whatever').order_by().first() |             BlogPost.objects.filter(title='whatever').order_by().first() | ||||||
|             self.assertEqual(len(q.get_ops()), 1) |             self.assertEqual(len(q.get_ops()), 1) | ||||||
|             self.assertFalse('$orderby' in q.get_ops()[0]['query']) |             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): |     def test_no_ordering_for_get(self): | ||||||
|         """ Ensure that Doc.objects.get doesn't use any ordering. |         """ Ensure that Doc.objects.get doesn't use any ordering. | ||||||
|         """ |         """ | ||||||
| @@ -3063,7 +3108,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         self.assertEqual(Foo.objects.distinct("bar"), [bar]) |         self.assertEqual(Foo.objects.distinct("bar"), [bar]) | ||||||
|  |  | ||||||
|     @skip_older_mongodb |     @needs_mongodb_v26 | ||||||
|     def test_text_indexes(self): |     def test_text_indexes(self): | ||||||
|         class News(Document): |         class News(Document): | ||||||
|             title = StringField() |             title = StringField() | ||||||
| @@ -3150,7 +3195,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|             'brasil').order_by('$text_score').first() |             'brasil').order_by('$text_score').first() | ||||||
|         self.assertEqual(item.get_text_score(), max_text_score) |         self.assertEqual(item.get_text_score(), max_text_score) | ||||||
|  |  | ||||||
|     @skip_older_mongodb |     @needs_mongodb_v26 | ||||||
|     def test_distinct_handles_references_to_alias(self): |     def test_distinct_handles_references_to_alias(self): | ||||||
|         register_connection('testdb', 'mongoenginetest2') |         register_connection('testdb', 'mongoenginetest2') | ||||||
|  |  | ||||||
| @@ -4825,6 +4870,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|             self.assertTrue(Person.objects._has_data(), |             self.assertTrue(Person.objects._has_data(), | ||||||
|                             'Cursor has data and returned False') |                             'Cursor has data and returned False') | ||||||
|  |  | ||||||
|  |     @needs_mongodb_v26 | ||||||
|     def test_queryset_aggregation_framework(self): |     def test_queryset_aggregation_framework(self): | ||||||
|         class Person(Document): |         class Person(Document): | ||||||
|             name = StringField() |             name = StringField() | ||||||
| @@ -4859,17 +4905,13 @@ class QuerySetTest(unittest.TestCase): | |||||||
|             {'_id': p1.pk, 'name': "ISABELLA LUANNA"} |             {'_id': p1.pk, 'name': "ISABELLA LUANNA"} | ||||||
|         ]) |         ]) | ||||||
|  |  | ||||||
|         data = Person.objects( |         data = Person.objects(age__gte=17, age__lte=40).order_by('-age').aggregate({ | ||||||
|             age__gte=17, age__lte=40).order_by('-age').aggregate( |             '$group': { | ||||||
|                 {'$group': { |                 '_id': None, | ||||||
|                     '_id': None, |                 'total': {'$sum': 1}, | ||||||
|                     'total': {'$sum': 1}, |                 'avg': {'$avg': '$age'} | ||||||
|                     'avg': {'$avg': '$age'} |             } | ||||||
|                 } |         }) | ||||||
|                 } |  | ||||||
|  |  | ||||||
|         ) |  | ||||||
|  |  | ||||||
|         self.assertEqual(list(data), [ |         self.assertEqual(list(data), [ | ||||||
|             {'_id': None, 'avg': 29, 'total': 2} |             {'_id': None, 'avg': 29, 'total': 2} | ||||||
|         ]) |         ]) | ||||||
| @@ -4910,11 +4952,13 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         self.assertEquals(Animal.objects(folded_ears=True).count(), 1) |         self.assertEquals(Animal.objects(folded_ears=True).count(), 1) | ||||||
|         self.assertEquals(Animal.objects(whiskers_length=5.1).count(), 1) |         self.assertEquals(Animal.objects(whiskers_length=5.1).count(), 1) | ||||||
|  |  | ||||||
|     def test_loop_via_invalid_id_does_not_crash(self): |     def test_loop_over_invalid_id_does_not_crash(self): | ||||||
|         class Person(Document): |         class Person(Document): | ||||||
|             name = StringField() |             name = StringField() | ||||||
|         Person.objects.delete() |  | ||||||
|         Person._get_collection().update({"name": "a"}, {"$set": {"_id": ""}}, upsert=True) |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         Person._get_collection().insert({'name': 'a', 'id': ''}) | ||||||
|         for p in Person.objects(): |         for p in Person.objects(): | ||||||
|             self.assertEqual(p.name, 'a') |             self.assertEqual(p.name, 'a') | ||||||
|  |  | ||||||
|   | |||||||
| @@ -35,8 +35,7 @@ class ConnectionTest(unittest.TestCase): | |||||||
|         mongoengine.connection._dbs = {} |         mongoengine.connection._dbs = {} | ||||||
|  |  | ||||||
|     def test_connect(self): |     def test_connect(self): | ||||||
|         """Ensure that the connect() method works properly. |         """Ensure that the connect() method works properly.""" | ||||||
|         """ |  | ||||||
|         connect('mongoenginetest') |         connect('mongoenginetest') | ||||||
|  |  | ||||||
|         conn = get_connection() |         conn = get_connection() | ||||||
| @@ -146,8 +145,7 @@ class ConnectionTest(unittest.TestCase): | |||||||
|         self.assertEqual(expected_connection, actual_connection) |         self.assertEqual(expected_connection, actual_connection) | ||||||
|  |  | ||||||
|     def test_connect_uri(self): |     def test_connect_uri(self): | ||||||
|         """Ensure that the connect() method works properly with uri's |         """Ensure that the connect() method works properly with URIs.""" | ||||||
|         """ |  | ||||||
|         c = connect(db='mongoenginetest', alias='admin') |         c = connect(db='mongoenginetest', alias='admin') | ||||||
|         c.admin.system.users.remove({}) |         c.admin.system.users.remove({}) | ||||||
|         c.mongoenginetest.system.users.remove({}) |         c.mongoenginetest.system.users.remove({}) | ||||||
| @@ -200,19 +198,6 @@ class ConnectionTest(unittest.TestCase): | |||||||
|         self.assertTrue(isinstance(db, pymongo.database.Database)) |         self.assertTrue(isinstance(db, pymongo.database.Database)) | ||||||
|         self.assertEqual(db.name, 'test') |         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): |     def test_uri_without_credentials_doesnt_override_conn_settings(self): | ||||||
|         """Ensure connect() uses the username & password params if the URI |         """Ensure connect() uses the username & password params if the URI | ||||||
|         doesn't explicitly specify them. |         doesn't explicitly specify them. | ||||||
| @@ -227,9 +212,8 @@ class ConnectionTest(unittest.TestCase): | |||||||
|         self.assertRaises(OperationFailure, get_db) |         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 `authSource` | ||||||
|         the option `authSource` in URI. |         option in the URI. | ||||||
|         This feature was introduced in MongoDB 2.4 and removed in 2.6 |  | ||||||
|         """ |         """ | ||||||
|         # Create users |         # Create users | ||||||
|         c = connect('mongoenginetest') |         c = connect('mongoenginetest') | ||||||
| @@ -238,30 +222,31 @@ class ConnectionTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         # Authentication fails without "authSource" |         # Authentication fails without "authSource" | ||||||
|         if IS_PYMONGO_3: |         if IS_PYMONGO_3: | ||||||
|             test_conn = connect('mongoenginetest', alias='test1', |             test_conn = connect( | ||||||
|                                 host='mongodb://username2:password@localhost/mongoenginetest') |                 'mongoenginetest', alias='test1', | ||||||
|  |                 host='mongodb://username2:password@localhost/mongoenginetest' | ||||||
|  |             ) | ||||||
|             self.assertRaises(OperationFailure, test_conn.server_info) |             self.assertRaises(OperationFailure, test_conn.server_info) | ||||||
|         else: |         else: | ||||||
|             self.assertRaises( |             self.assertRaises( | ||||||
|                 MongoEngineConnectionError, connect, 'mongoenginetest', |                 MongoEngineConnectionError, | ||||||
|                 alias='test1', |                 connect, 'mongoenginetest', alias='test1', | ||||||
|                 host='mongodb://username2:password@localhost/mongoenginetest' |                 host='mongodb://username2:password@localhost/mongoenginetest' | ||||||
|             ) |             ) | ||||||
|             self.assertRaises(MongoEngineConnectionError, get_db, 'test1') |             self.assertRaises(MongoEngineConnectionError, get_db, 'test1') | ||||||
|  |  | ||||||
|         # Authentication succeeds with "authSource" |         # Authentication succeeds with "authSource" | ||||||
|         connect( |         authd_conn = connect( | ||||||
|             'mongoenginetest', alias='test2', |             'mongoenginetest', alias='test2', | ||||||
|             host=('mongodb://username2:password@localhost/' |             host=('mongodb://username2:password@localhost/' | ||||||
|                   'mongoenginetest?authSource=admin') |                   'mongoenginetest?authSource=admin') | ||||||
|         ) |         ) | ||||||
|         # This will fail starting from MongoDB 2.6+ |  | ||||||
|         db = get_db('test2') |         db = get_db('test2') | ||||||
|         self.assertTrue(isinstance(db, pymongo.database.Database)) |         self.assertTrue(isinstance(db, pymongo.database.Database)) | ||||||
|         self.assertEqual(db.name, 'mongoenginetest') |         self.assertEqual(db.name, 'mongoenginetest') | ||||||
|  |  | ||||||
|         # Clear all users |         # Clear all users | ||||||
|         c.admin.system.users.remove({}) |         authd_conn.admin.system.users.remove({}) | ||||||
|  |  | ||||||
|     def test_register_connection(self): |     def test_register_connection(self): | ||||||
|         """Ensure that connections with different aliases may be registered. |         """Ensure that connections with different aliases may be registered. | ||||||
| @@ -285,8 +270,7 @@ class ConnectionTest(unittest.TestCase): | |||||||
|         self.assertTrue(isinstance(conn, pymongo.mongo_client.MongoClient)) |         self.assertTrue(isinstance(conn, pymongo.mongo_client.MongoClient)) | ||||||
|  |  | ||||||
|     def test_connection_kwargs(self): |     def test_connection_kwargs(self): | ||||||
|         """Ensure that connection kwargs get passed to pymongo. |         """Ensure that connection kwargs get passed to pymongo.""" | ||||||
|         """ |  | ||||||
|         connect('mongoenginetest', alias='t1', tz_aware=True) |         connect('mongoenginetest', alias='t1', tz_aware=True) | ||||||
|         conn = get_connection('t1') |         conn = get_connection('t1') | ||||||
|  |  | ||||||
| @@ -296,6 +280,77 @@ class ConnectionTest(unittest.TestCase): | |||||||
|         conn = get_connection('t2') |         conn = get_connection('t2') | ||||||
|         self.assertFalse(get_tz_awareness(conn)) |         self.assertFalse(get_tz_awareness(conn)) | ||||||
|  |  | ||||||
|  |     def test_connection_pool_via_kwarg(self): | ||||||
|  |         """Ensure we can specify a max connection pool size using | ||||||
|  |         a connection kwarg. | ||||||
|  |         """ | ||||||
|  |         # Use "max_pool_size" or "maxpoolsize" depending on PyMongo version | ||||||
|  |         # (former was changed to the latter as described in | ||||||
|  |         # https://jira.mongodb.org/browse/PYTHON-854). | ||||||
|  |         # TODO remove once PyMongo < 3.0 support is dropped | ||||||
|  |         if pymongo.version_tuple[0] >= 3: | ||||||
|  |             pool_size_kwargs = {'maxpoolsize': 100} | ||||||
|  |         else: | ||||||
|  |             pool_size_kwargs = {'max_pool_size': 100} | ||||||
|  |  | ||||||
|  |         conn = connect('mongoenginetest', alias='max_pool_size_via_kwarg', **pool_size_kwargs) | ||||||
|  |         self.assertEqual(conn.max_pool_size, 100) | ||||||
|  |  | ||||||
|  |     def test_connection_pool_via_uri(self): | ||||||
|  |         """Ensure we can specify a max connection pool size using | ||||||
|  |         an option in a connection URI. | ||||||
|  |         """ | ||||||
|  |         if pymongo.version_tuple[0] == 2 and pymongo.version_tuple[1] < 9: | ||||||
|  |             raise SkipTest('maxpoolsize as a URI option is only supported in PyMongo v2.9+') | ||||||
|  |  | ||||||
|  |         conn = connect(host='mongodb://localhost/test?maxpoolsize=100', alias='max_pool_size_via_uri') | ||||||
|  |         self.assertEqual(conn.max_pool_size, 100) | ||||||
|  |  | ||||||
|  |     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_connect_with_replicaset_via_uri(self): | ||||||
|  |         """Ensure connect() works when specifying a replicaSet via the | ||||||
|  |         MongoDB URI. | ||||||
|  |         """ | ||||||
|  |         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_connect_with_replicaset_via_kwargs(self): | ||||||
|  |         """Ensure connect() works when specifying a replicaSet via the | ||||||
|  |         connection kwargs | ||||||
|  |         """ | ||||||
|  |         if IS_PYMONGO_3: | ||||||
|  |             c = connect(replicaset='local-rs') | ||||||
|  |             self.assertEqual(c._MongoClient__options.replica_set_name, | ||||||
|  |                              '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(replicaset='local-rs') | ||||||
|  |  | ||||||
|     def test_datetime(self): |     def test_datetime(self): | ||||||
|         connect('mongoenginetest', tz_aware=True) |         connect('mongoenginetest', tz_aware=True) | ||||||
|         d = datetime.datetime(2010, 5, 5, tzinfo=utc) |         d = datetime.datetime(2010, 5, 5, tzinfo=utc) | ||||||
|   | |||||||
							
								
								
									
										78
									
								
								tests/utils.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										78
									
								
								tests/utils.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,78 @@ | |||||||
|  | import unittest | ||||||
|  |  | ||||||
|  | from nose.plugins.skip import SkipTest | ||||||
|  |  | ||||||
|  | from mongoengine import connect | ||||||
|  | from mongoengine.connection import get_db, get_connection | ||||||
|  | from mongoengine.python_support import IS_PYMONGO_3 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | MONGO_TEST_DB = 'mongoenginetest' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MongoDBTestCase(unittest.TestCase): | ||||||
|  |     """Base class for tests that need a mongodb connection | ||||||
|  |     db is being dropped automatically | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def setUpClass(cls): | ||||||
|  |         cls._connection = connect(db=MONGO_TEST_DB) | ||||||
|  |         cls._connection.drop_database(MONGO_TEST_DB) | ||||||
|  |         cls.db = get_db() | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def tearDownClass(cls): | ||||||
|  |         cls._connection.drop_database(MONGO_TEST_DB) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_mongodb_version(): | ||||||
|  |     """Return the version tuple of the MongoDB server that the default | ||||||
|  |     connection is connected to. | ||||||
|  |     """ | ||||||
|  |     return tuple(get_connection().server_info()['versionArray']) | ||||||
|  |  | ||||||
|  | def _decorated_with_ver_requirement(func, ver_tuple): | ||||||
|  |     """Return a given function decorated with the version requirement | ||||||
|  |     for a particular MongoDB version tuple. | ||||||
|  |     """ | ||||||
|  |     def _inner(*args, **kwargs): | ||||||
|  |         mongodb_ver = get_mongodb_version() | ||||||
|  |         if mongodb_ver >= ver_tuple: | ||||||
|  |             return func(*args, **kwargs) | ||||||
|  |  | ||||||
|  |         raise SkipTest('Needs MongoDB v{}+'.format( | ||||||
|  |             '.'.join([str(v) for v in ver_tuple]) | ||||||
|  |         )) | ||||||
|  |  | ||||||
|  |     _inner.__name__ = func.__name__ | ||||||
|  |     _inner.__doc__ = func.__doc__ | ||||||
|  |  | ||||||
|  |     return _inner | ||||||
|  |  | ||||||
|  | def needs_mongodb_v26(func): | ||||||
|  |     """Raise a SkipTest exception if we're working with MongoDB version | ||||||
|  |     lower than v2.6. | ||||||
|  |     """ | ||||||
|  |     return _decorated_with_ver_requirement(func, (2, 6)) | ||||||
|  |  | ||||||
|  | def needs_mongodb_v3(func): | ||||||
|  |     """Raise a SkipTest exception if we're working with MongoDB version | ||||||
|  |     lower than v3.0. | ||||||
|  |     """ | ||||||
|  |     return _decorated_with_ver_requirement(func, (3, 0)) | ||||||
|  |  | ||||||
|  | def skip_pymongo3(f): | ||||||
|  |     """Raise a SkipTest exception if we're running a test against | ||||||
|  |     PyMongo v3.x. | ||||||
|  |     """ | ||||||
|  |     def _inner(*args, **kwargs): | ||||||
|  |         if IS_PYMONGO_3: | ||||||
|  |             raise SkipTest("Useless with PyMongo 3+") | ||||||
|  |         return f(*args, **kwargs) | ||||||
|  |  | ||||||
|  |     _inner.__name__ = f.__name__ | ||||||
|  |     _inner.__doc__ = f.__doc__ | ||||||
|  |  | ||||||
|  |     return _inner | ||||||
|  |  | ||||||
							
								
								
									
										13
									
								
								tox.ini
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								tox.ini
									
									
									
									
									
								
							| @@ -1,5 +1,5 @@ | |||||||
| [tox] | [tox] | ||||||
| envlist = {py26,py27,py33,py34,py35,pypy,pypy3}-{mg27,mg28},flake8 | envlist = {py27,py35,pypy,pypy3}-{mg27,mg28,mg30} | ||||||
|  |  | ||||||
| [testenv] | [testenv] | ||||||
| commands = | commands = | ||||||
| @@ -7,16 +7,7 @@ commands = | |||||||
| deps = | deps = | ||||||
|     nose |     nose | ||||||
|     mg27: PyMongo<2.8 |     mg27: PyMongo<2.8 | ||||||
|     mg28: PyMongo>=2.8,<3.0 |     mg28: PyMongo>=2.8,<2.9 | ||||||
|     mg30: PyMongo>=3.0 |     mg30: PyMongo>=3.0 | ||||||
|     mgdev: https://github.com/mongodb/mongo-python-driver/tarball/master |  | ||||||
| setenv = | setenv = | ||||||
|     PYTHON_EGG_CACHE = {envdir}/python-eggs |     PYTHON_EGG_CACHE = {envdir}/python-eggs | ||||||
| passenv = windir |  | ||||||
|  |  | ||||||
| [testenv:flake8] |  | ||||||
| deps = |  | ||||||
|     flake8 |  | ||||||
|     flake8-import-order |  | ||||||
| commands = |  | ||||||
|    flake8 |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user