Compare commits
	
		
			245 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 6affbbe865 | ||
|  | e3600ef4de | ||
|  | 6dcd7006d0 | ||
|  | 3f49923298 | ||
|  | c277be8b6b | ||
|  | 03bfd01862 | ||
|  | dcf3c86dce | ||
|  | c2d77f51bb | ||
|  | b4d87d9128 | ||
|  | 4401a309ee | ||
|  | b562e209d1 | ||
|  | 3a85422e8f | ||
|  | e45397c975 | ||
|  | 1f9ec0c888 | ||
|  | f8ee470e70 | ||
|  | d02de0798f | ||
|  | 6fe074fb13 | ||
|  | 4db339c5f4 | ||
|  | a525764359 | ||
|  | f970d5878a | ||
|  | cc0a2cbc6f | ||
|  | add0b463f5 | ||
|  | d80b1a7749 | ||
|  | 6186691259 | ||
|  | b451cc567d | ||
|  | 757ff31661 | ||
|  | 97a98f0045 | ||
|  | 8f05896bc9 | ||
|  | da7a8939df | ||
|  | b6977a88ea | ||
|  | eafbc7f20d | ||
|  | c9a5710554 | ||
|  | f10e946896 | ||
|  | 2f19b22bb2 | ||
|  | d134e11c6d | ||
|  | 63edd16a92 | ||
|  | 37740dc010 | ||
|  | 836dc96f67 | ||
|  | 49a7542b14 | ||
|  | a84ffce5a0 | ||
|  | 210b3e5192 | ||
|  | 5f1d5ea056 | ||
|  | 19a7372ff9 | ||
|  | cc5b60b004 | ||
|  | b06f9dbf8d | ||
|  | d9b8ee7895 | ||
|  | e9ff655b0e | ||
|  | 669d21a114 | ||
|  | 7e980a16d0 | ||
|  | 47df8deb58 | ||
|  | dd006a502e | ||
|  | 07d3e52e6a | ||
|  | fc1ce6d39b | ||
|  | 32d5c0c946 | ||
|  | dfabfce01b | ||
|  | 74f3f4eb15 | ||
|  | 20cb0285f0 | ||
|  | faf840f924 | ||
|  | 165bea5bb9 | ||
|  | f7515cfca8 | ||
|  | a762a10dec | ||
|  | a192029901 | ||
|  | 67182713d9 | ||
|  | e9464e32db | ||
|  | 2d6ae16912 | ||
|  | f9cd8b1841 | ||
|  | 41a698b442 | ||
|  | 9f58bc9207 | ||
|  | d36f6e7f24 | ||
|  | eeb672feb9 | ||
|  | 063a162ce0 | ||
|  | 3e4a900279 | ||
|  | 43327ea4e1 | ||
|  | 0d2e84b16b | ||
|  | 3c78757778 | ||
|  | d0245bb5ba | ||
|  | 3477b0107a | ||
|  | 8df9ff90cb | ||
|  | d6b4ca7a98 | ||
|  | 2e18199eb2 | ||
|  | e508625935 | ||
|  | 87c965edd3 | ||
|  | 06681a453f | ||
|  | 8e038dd563 | ||
|  | e537369d98 | ||
|  | 25cdf16cc0 | ||
|  | 74343841e4 | ||
|  | 3b3738b36b | ||
|  | b15c3f6a3f | ||
|  | 2459f9b0aa | ||
|  | 6ff1bd9b3c | ||
|  | 1bc2d2ec37 | ||
|  | d7fd6a4628 | ||
|  | 9236f365fa | ||
|  | 90d22c2a28 | ||
|  | c9f6e6b62a | ||
|  | 260d9377f5 | ||
|  | 22d1ce6319 | ||
|  | 6997e02476 | ||
|  | 155d79ff4d | ||
|  | 452cd125fa | ||
|  | e62c35b040 | ||
|  | d5ec3c6a31 | ||
|  | ad983dc279 | ||
|  | bb15bf8d13 | ||
|  | 94adc207ad | ||
|  | 376d1c97ab | ||
|  | 4fe87b40da | ||
|  | b10d76cf4b | ||
|  | 3bdc9a2f09 | ||
|  | 9d52e18659 | ||
|  | 653c4259ee | ||
|  | 9f5ab8149f | ||
|  | 66c6d14f7a | ||
|  | 2c0fc142a3 | ||
|  | 0da2dfd191 | ||
|  | 787fc1cd8b | ||
|  | c31488add9 | ||
|  | 31ec7907b5 | ||
|  | 12f3f8c694 | ||
|  | 79098e997e | ||
|  | dc1849bad5 | ||
|  | e2d826c412 | ||
|  | e6d796832e | ||
|  | 6f0a6df4f6 | ||
|  | 7a877a00d5 | ||
|  | e8604d100e | ||
|  | 1647441ce8 | ||
|  | 9f8d6b3a00 | ||
|  | 4b2ad25405 | ||
|  | 3ce163b1a0 | ||
|  | 7c1ee28f13 | ||
|  | 2645e43da1 | ||
|  | 59bfe551a3 | ||
|  | e2c78047b1 | ||
|  | 6a4351e44f | ||
|  | adb60ef1ac | ||
|  | 3090adac04 | ||
|  | b9253d86cc | ||
|  | ab4d4e6230 | ||
|  | 7cd38c56c6 | ||
|  | 864053615b | ||
|  | db2366f112 | ||
|  | 4defc82192 | ||
|  | 5949970a95 | ||
|  | 0ea4abda81 | ||
|  | 5c6035d636 | ||
|  | a2183e3dcc | ||
|  | 99637151b5 | ||
|  | a8e787c120 | ||
|  | 53339c7c72 | ||
|  | 3534bf7d70 | ||
|  | 1cf3989664 | ||
|  | fd296918da | ||
|  | 8ad1f03dc5 | ||
|  | fe7e17dbd5 | ||
|  | d582394a42 | ||
|  | 02ef0df019 | ||
|  | 0dfd6aa518 | ||
|  | 0b23bc9cf2 | ||
|  | f108c4288e | ||
|  | 9b9696aefd | ||
|  | 576e198ece | ||
|  | 52f85aab18 | ||
|  | ab60fd0490 | ||
|  | d79ae30f31 | ||
|  | f27debe7f9 | ||
|  | 735e043ff6 | ||
|  | 6e7f2b73cf | ||
|  | d645ce9745 | ||
|  | 7c08c140da | ||
|  | 81d402dc17 | ||
|  | 966fa12358 | ||
|  | 87792e1921 | ||
|  | 4c8296acc6 | ||
|  | 9989da07ed | ||
|  | 1c5e6a3425 | ||
|  | eedf908770 | ||
|  | 5c9ef41403 | ||
|  | 0bf2ad5b67 | ||
|  | a0e3f382cd | ||
|  | f09c39b5d7 | ||
|  | 89c67bf259 | ||
|  | ea666d4607 | ||
|  | b8af154439 | ||
|  | f594ece32a | ||
|  | 03beb6852a | ||
|  | ab9e9a3329 | ||
|  | a4b09344af | ||
|  | 8cb8aa392c | ||
|  | 3255519792 | ||
|  | 7e64bb2503 | ||
|  | 86a78402c3 | ||
|  | ba276452fb | ||
|  | 4ffa8d0124 | ||
|  | 4bc5082681 | ||
|  | 0e3c34e1da | ||
|  | 658b3784ae | ||
|  | 0526f577ff | ||
|  | bb1b9bc1d3 | ||
|  | b1eeb77ddc | ||
|  | 999d4a7676 | ||
|  | 1b80193aac | ||
|  | be8d39a48c | ||
|  | a2f3d70f28 | ||
|  | 676a7bf712 | ||
|  | e990a6c70c | ||
|  | 90fa0f6c4a | ||
|  | 22010d7d95 | ||
|  | 66279bd90f | ||
|  | 19da228855 | ||
|  | 9e67941bad | ||
|  | 0454fc74e9 | ||
|  | 2f6b1c7611 | ||
|  | f00bed6058 | ||
|  | 529c522594 | ||
|  | 2bb9493fcf | ||
|  | 839ed8a64a | ||
|  | 017a31ffd0 | ||
|  | 83b961c84d | ||
|  | fa07423ca5 | ||
|  | dd4af2df81 | ||
|  | 44bd8cb85b | ||
|  | 52d80ac23c | ||
|  | 43a5d73e14 | ||
|  | abc764951d | ||
|  | 0575abab23 | ||
|  | 9eebcf7beb | ||
|  | ed74477150 | ||
|  | dd023edc0f | ||
|  | f45d4d781d | ||
|  | c95652d6a8 | ||
|  | 97b37f75d3 | ||
|  | 95dae48778 | ||
|  | 73635033bd | ||
|  | c1619d2a62 | ||
|  | b87ef982f6 | ||
|  | 91aa90ad4a | ||
|  | d1add62a06 | ||
|  | c419f3379a | ||
|  | 69d57209f7 | ||
|  | 7ca81d6fb8 | ||
|  | 8a046bfa5d | ||
|  | 3628a7653c | ||
|  | 1a4533a9cf | 
							
								
								
									
										25
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -1,12 +1,27 @@ | |||||||
| # http://travis-ci.org/#!/MongoEngine/mongoengine | # http://travis-ci.org/#!/MongoEngine/mongoengine | ||||||
| language: python | language: python | ||||||
|  | services: mongodb | ||||||
| python: | python: | ||||||
|     - 2.6 |     - "2.5" | ||||||
|     - 2.7 |     - "2.6" | ||||||
|  |     - "2.7" | ||||||
|  |     - "3.2" | ||||||
|  |     - "3.3" | ||||||
|  | env: | ||||||
|  |   - PYMONGO=dev | ||||||
|  |   - PYMONGO=2.5 | ||||||
|  |   - PYMONGO=2.4.2 | ||||||
| install: | install: | ||||||
|     - sudo apt-get install zlib1g zlib1g-dev |     - if [[ $TRAVIS_PYTHON_VERSION == '2.'* ]]; then cp /usr/lib/*/libz.so $VIRTUAL_ENV/lib/; fi | ||||||
|     - sudo ln -s /usr/lib/i386-linux-gnu/libz.so /usr/lib/ |     - if [[ $TRAVIS_PYTHON_VERSION == '2.'* ]]; then pip install pil --use-mirrors ; true; fi | ||||||
|     - pip install PIL --use-mirrors ; true |     - if [[ $PYMONGO == 'dev' ]]; then pip install https://github.com/mongodb/mongo-python-driver/tarball/master; true; fi | ||||||
|  |     - if [[ $PYMONGO != 'dev' ]]; then pip install pymongo==$PYMONGO --use-mirrors; true; fi | ||||||
|     - python setup.py install |     - python setup.py install | ||||||
| script: | script: | ||||||
|     - python setup.py test |     - python setup.py test | ||||||
|  | notifications: | ||||||
|  |   irc: "irc.freenode.org#mongoengine" | ||||||
|  | branches: | ||||||
|  |   only: | ||||||
|  |     - master | ||||||
|  |     - "0.8" | ||||||
|   | |||||||
							
								
								
									
										25
									
								
								AUTHORS
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								AUTHORS
									
									
									
									
									
								
							| @@ -8,6 +8,7 @@ Florian Schlachter <flori@n-schlachter.de> | |||||||
| Steve Challis <steve@stevechallis.com> | Steve Challis <steve@stevechallis.com> | ||||||
| Wilson Júnior <wilsonpjunior@gmail.com> | Wilson Júnior <wilsonpjunior@gmail.com> | ||||||
| Dan Crosta https://github.com/dcrosta | Dan Crosta https://github.com/dcrosta | ||||||
|  | Laine Herron https://github.com/LaineHerron | ||||||
|  |  | ||||||
| CONTRIBUTORS | CONTRIBUTORS | ||||||
|  |  | ||||||
| @@ -105,7 +106,7 @@ that much better: | |||||||
|  * Adam Reeve |  * Adam Reeve | ||||||
|  * Anthony Nemitz |  * Anthony Nemitz | ||||||
|  * deignacio |  * deignacio | ||||||
|  * shaunduncan |  * Shaun Duncan | ||||||
|  * Meir Kriheli |  * Meir Kriheli | ||||||
|  * Andrey Fedoseev |  * Andrey Fedoseev | ||||||
|  * aparajita |  * aparajita | ||||||
| @@ -116,3 +117,25 @@ that much better: | |||||||
|  * Thomas Steinacher |  * Thomas Steinacher | ||||||
|  * Tommi Komulainen |  * Tommi Komulainen | ||||||
|  * Peter Landry |  * Peter Landry | ||||||
|  |  * biszkoptwielki | ||||||
|  |  * Anton Kolechkin | ||||||
|  |  * Sergey Nikitin | ||||||
|  |  * psychogenic | ||||||
|  |  * Stefan Wójcik | ||||||
|  |  * dimonb | ||||||
|  |  * Garry Polley | ||||||
|  |  * Adrian Scott | ||||||
|  |  * Peter Teichman | ||||||
|  |  * Jakub Kot | ||||||
|  |  * Jorge Bastida | ||||||
|  |  * Aleksandr Sorokoumov | ||||||
|  |  * Yohan Graterol | ||||||
|  |  * bool-dev | ||||||
|  |  * Russ Weeks | ||||||
|  |  * Paul Swartz | ||||||
|  |  * Sundar Raman | ||||||
|  |  * Benoit Louy | ||||||
|  |  * lraucy | ||||||
|  |  * hellysmile | ||||||
|  |  * Jaepil Jeong | ||||||
|  |  * Daniil Sharou | ||||||
|   | |||||||
							
								
								
									
										61
									
								
								CONTRIBUTING.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								CONTRIBUTING.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | |||||||
|  | Contributing to MongoEngine | ||||||
|  | =========================== | ||||||
|  |  | ||||||
|  | MongoEngine has a large `community | ||||||
|  | <https://raw.github.com/MongoEngine/mongoengine/master/AUTHORS>`_ and | ||||||
|  | contributions are always encouraged. Contributions can be as simple as | ||||||
|  | minor tweaks to the documentation. Please read these guidelines before | ||||||
|  | sending a pull request. | ||||||
|  |  | ||||||
|  | Bugfixes and New Features | ||||||
|  | ------------------------- | ||||||
|  |  | ||||||
|  | Before starting to write code, look for existing `tickets | ||||||
|  | <https://github.com/MongoEngine/mongoengine/issues?state=open>`_ or `create one | ||||||
|  | <https://github.com/MongoEngine/mongoengine/issues>`_ for your specific | ||||||
|  | issue or feature request. That way you avoid working on something | ||||||
|  | that might not be of interest or that has already been addressed.  If in doubt | ||||||
|  | post to the `user group <http://groups.google.com/group/mongoengine-users>` | ||||||
|  |  | ||||||
|  | Supported Interpreters | ||||||
|  | ---------------------- | ||||||
|  |  | ||||||
|  | PyMongo supports CPython 2.5 and newer. Language | ||||||
|  | features not supported by all interpreters can not be used. | ||||||
|  | Please also ensure that your code is properly converted by | ||||||
|  | `2to3 <http://docs.python.org/library/2to3.html>`_ for Python 3 support. | ||||||
|  |  | ||||||
|  | Style Guide | ||||||
|  | ----------- | ||||||
|  |  | ||||||
|  | MongoEngine aims to follow `PEP8 <http://www.python.org/dev/peps/pep-0008/>`_ | ||||||
|  | including 4 space indents and 79 character line limits. | ||||||
|  |  | ||||||
|  | Testing | ||||||
|  | ------- | ||||||
|  |  | ||||||
|  | All tests are run on `Travis <http://travis-ci.org/MongoEngine/mongoengine>`_ | ||||||
|  | and any pull requests are automatically tested by Travis. Any pull requests | ||||||
|  | without tests will take longer to be integrated and might be refused. | ||||||
|  |  | ||||||
|  | General Guidelines | ||||||
|  | ------------------ | ||||||
|  |  | ||||||
|  | - Avoid backward breaking changes if at all possible. | ||||||
|  | - Write inline documentation for new classes and methods. | ||||||
|  | - Write tests and make sure they pass (make sure you have a mongod | ||||||
|  |   running on the default port, then execute ``python setup.py test`` | ||||||
|  |   from the cmd line to run the test suite). | ||||||
|  | - Add yourself to AUTHORS.rst :) | ||||||
|  |  | ||||||
|  | Documentation | ||||||
|  | ------------- | ||||||
|  |  | ||||||
|  | To contribute to the `API documentation | ||||||
|  | <http://docs.mongoengine.org/en/latest/apireference.html>`_ | ||||||
|  | just make your changes to the inline documentation of the appropriate | ||||||
|  | `source code <https://github.com/MongoEngine/mongoengine>`_ or `rst file | ||||||
|  | <https://github.com/MongoEngine/mongoengine/tree/master/docs>`_ in a | ||||||
|  | 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>`_ | ||||||
|  | button. | ||||||
							
								
								
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						
									
										2
									
								
								LICENSE
									
									
									
									
									
								
							| @@ -1,4 +1,4 @@ | |||||||
| Copyright (c) 2009-2012 See AUTHORS | Copyright (c) 2009 See AUTHORS | ||||||
|  |  | ||||||
| Permission is hereby granted, free of charge, to any person | Permission is hereby granted, free of charge, to any person | ||||||
| obtaining a copy of this software and associated documentation | obtaining a copy of this software and associated documentation | ||||||
|   | |||||||
							
								
								
									
										16
									
								
								README.rst
									
									
									
									
									
								
							
							
						
						
									
										16
									
								
								README.rst
									
									
									
									
									
								
							| @@ -14,7 +14,7 @@ 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 http://mongoengine-odm.rtfd.org - there is currently | Documentation available at http://mongoengine-odm.rtfd.org - there is currently | ||||||
| a `tutorial <http://readthedocs.org/docs/mongoengine-odm/en/latest/tutorial.html>`_, a `user guide | a `tutorial <http://readthedocs.org/docs/mongoengine-odm/en/latest/tutorial.html>`_, a `user guide | ||||||
| <http://readthedocs.org/docs/mongoengine-odm/en/latest/userguide.html>`_ and an `API reference | <https://mongoengine-odm.readthedocs.org/en/latest/guide/index.html>`_ and an `API reference | ||||||
| <http://readthedocs.org/docs/mongoengine-odm/en/latest/apireference.html>`_. | <http://readthedocs.org/docs/mongoengine-odm/en/latest/apireference.html>`_. | ||||||
|  |  | ||||||
| Installation | Installation | ||||||
| @@ -63,11 +63,6 @@ Some simple examples of what MongoEngine code looks like:: | |||||||
|     ...         print 'Link:', post.url |     ...         print 'Link:', post.url | ||||||
|     ...     print |     ...     print | ||||||
|     ... |     ... | ||||||
|     === Using MongoEngine === |  | ||||||
|     See the tutorial |  | ||||||
|  |  | ||||||
|     === MongoEngine Docs === |  | ||||||
|     Link: hmarr.com/mongoengine |  | ||||||
|  |  | ||||||
|     >>> len(BlogPost.objects) |     >>> len(BlogPost.objects) | ||||||
|     2 |     2 | ||||||
| @@ -85,7 +80,7 @@ Some simple examples of what MongoEngine code looks like:: | |||||||
| 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 run ``python setup.py test``. | the standard port, and run: ``python setup.py test``. | ||||||
|  |  | ||||||
| Community | Community | ||||||
| ========= | ========= | ||||||
| @@ -93,11 +88,8 @@ 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 <irc://irc.freenode.net/mongoengine>`_ | - `#mongoengine IRC channel <http://webchat.freenode.net/?channels=mongoengine>`_ | ||||||
|  |  | ||||||
| Contributing | Contributing | ||||||
| ============ | ============ | ||||||
| The source is available on `GitHub <http://github.com/MongoEngine/mongoengine>`_ - to | We welcome contributions! see  the`Contribution guidelines <https://github.com/MongoEngine/mongoengine/blob/master/CONTRIBUTING.rst>`_ | ||||||
| contribute to the project, fork it on GitHub and send a pull request, all |  | ||||||
| contributions and suggestions are welcome! |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										49
									
								
								benchmark.py
									
									
									
									
									
								
							
							
						
						
									
										49
									
								
								benchmark.py
									
									
									
									
									
								
							| @@ -28,47 +28,64 @@ def main(): | |||||||
|  |  | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     Creating 10000 dictionaries - Pymongo |     Creating 10000 dictionaries - Pymongo | ||||||
|     1.1141769886 |     3.86744189262 | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     Creating 10000 dictionaries - MongoEngine |     Creating 10000 dictionaries - MongoEngine | ||||||
|     2.37724113464 |     6.23374891281 | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False | ||||||
|     1.92479610443 |     5.33027005196 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False, cascade=False | ||||||
|  |     pass - No Cascade | ||||||
|  |  | ||||||
|     0.5.X |     0.5.X | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     Creating 10000 dictionaries - Pymongo |     Creating 10000 dictionaries - Pymongo | ||||||
|     1.10552310944 |     3.89597702026 | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     Creating 10000 dictionaries - MongoEngine |     Creating 10000 dictionaries - MongoEngine | ||||||
|     16.5169169903 |     21.7735359669 | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False | ||||||
|     14.9446101189 |     19.8670389652 | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False, cascade=False |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False, cascade=False | ||||||
|     14.912801981 |     pass - No Cascade | ||||||
|     ---------------------------------------------------------------------------------------------------- |  | ||||||
|     Creating 10000 dictionaries - MongoEngine, force=True |  | ||||||
|     14.9617750645 |  | ||||||
|  |  | ||||||
|     Performance |     0.6.X | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     Creating 10000 dictionaries - Pymongo |     Creating 10000 dictionaries - Pymongo | ||||||
|     1.10072994232 |     3.81559205055 | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     Creating 10000 dictionaries - MongoEngine |     Creating 10000 dictionaries - MongoEngine | ||||||
|     5.27341103554 |     10.0446798801 | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False | ||||||
|     4.49365401268 |     9.51354718208 | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False, cascade=False |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False, cascade=False | ||||||
|     4.43459296227 |     9.02567505836 | ||||||
|     ---------------------------------------------------------------------------------------------------- |     ---------------------------------------------------------------------------------------------------- | ||||||
|     Creating 10000 dictionaries - MongoEngine, force=True |     Creating 10000 dictionaries - MongoEngine, force=True | ||||||
|     4.40114378929 |     8.44933390617 | ||||||
|  |  | ||||||
|  |     0.7.X | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - Pymongo | ||||||
|  |     3.78801012039 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine | ||||||
|  |     9.73050498962 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False | ||||||
|  |     8.33456707001 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False, cascade=False | ||||||
|  |     8.37778115273 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine, force=True | ||||||
|  |     8.36906409264 | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     setup = """ |     setup = """ | ||||||
|   | |||||||
| @@ -2,6 +2,96 @@ | |||||||
| Changelog | Changelog | ||||||
| ========= | ========= | ||||||
|  |  | ||||||
|  | Changes in 0.7.10 | ||||||
|  | ================= | ||||||
|  | - Fix UnicodeEncodeError for dbref (#278) | ||||||
|  | - Allow construction using positional parameters (#268) | ||||||
|  | - Updated EmailField length to support long domains (#243) | ||||||
|  | - Added 64-bit integer support (#251) | ||||||
|  | - Added Django sessions TTL support (#224) | ||||||
|  | - Fixed issue with numerical keys in MapField(EmbeddedDocumentField()) (#240) | ||||||
|  | - Fixed clearing _changed_fields for complex nested embedded documents (#237, #239, #242) | ||||||
|  | - Added "id" back to _data dictionary (#255) | ||||||
|  | - Only mark a field as changed if the value has changed (#258) | ||||||
|  | - Explicitly check for Document instances when dereferencing (#261) | ||||||
|  | - Fixed order_by chaining issue (#265) | ||||||
|  | - Added dereference support for tuples (#250) | ||||||
|  | - Resolve field name to db field name when using distinct(#260, #264, #269) | ||||||
|  | - Added kwargs to doc.save to help interop with django (#223, #270) | ||||||
|  | - Fixed cloning querysets in PY3 | ||||||
|  | - Int fields no longer unset in save when changed to 0 (#272) | ||||||
|  | - Fixed ReferenceField query chaining bug fixed (#254) | ||||||
|  |  | ||||||
|  | Changes in 0.7.9 | ||||||
|  | ================ | ||||||
|  | - Better fix handling for old style _types | ||||||
|  | - Embedded SequenceFields follow collection naming convention | ||||||
|  |  | ||||||
|  | Changes in 0.7.8 | ||||||
|  | ================ | ||||||
|  | - Fix sequence fields in embedded documents (MongoEngine/mongoengine#166) | ||||||
|  | - Fix query chaining with .order_by() (MongoEngine/mongoengine#176) | ||||||
|  | - Added optional encoding and collection config for Django sessions (MongoEngine/mongoengine#180, MongoEngine/mongoengine#181, MongoEngine/mongoengine#183) | ||||||
|  | - Fixed EmailField so can add extra validation (MongoEngine/mongoengine#173, MongoEngine/mongoengine#174, MongoEngine/mongoengine#187) | ||||||
|  | - Fixed bulk inserts can now handle custom pk's (MongoEngine/mongoengine#192) | ||||||
|  | - Added as_pymongo method to return raw or cast results from pymongo (MongoEngine/mongoengine#193) | ||||||
|  |  | ||||||
|  | Changes in 0.7.7 | ||||||
|  | ================ | ||||||
|  | - Fix handling for old style _types | ||||||
|  |  | ||||||
|  | Changes in 0.7.6 | ||||||
|  | ================ | ||||||
|  | - Unicode fix for repr (MongoEngine/mongoengine#133) | ||||||
|  | - Allow updates with match operators (MongoEngine/mongoengine#144) | ||||||
|  | - Updated URLField - now can have a override the regex (MongoEngine/mongoengine#136) | ||||||
|  | - Allow Django AuthenticationBackends to work with Django user (hmarr/mongoengine#573) | ||||||
|  | - Fixed reload issue with ReferenceField where dbref=False (MongoEngine/mongoengine#138) | ||||||
|  |  | ||||||
|  | Changes in 0.7.5 | ||||||
|  | ================ | ||||||
|  | - ReferenceFields with dbref=False use ObjectId instead of strings (MongoEngine/mongoengine#134) | ||||||
|  |   See ticket for upgrade notes (https://github.com/MongoEngine/mongoengine/issues/134) | ||||||
|  |  | ||||||
|  | Changes in 0.7.4 | ||||||
|  | ================ | ||||||
|  | - Fixed index inheritance issues - firmed up testcases (MongoEngine/mongoengine#123) (MongoEngine/mongoengine#125) | ||||||
|  |  | ||||||
|  | Changes in 0.7.3 | ||||||
|  | ================ | ||||||
|  | - Reverted EmbeddedDocuments meta handling - now can turn off inheritance (MongoEngine/mongoengine#119) | ||||||
|  |  | ||||||
|  | Changes in 0.7.2 | ||||||
|  | ================ | ||||||
|  | - Update index spec generation so its not destructive (MongoEngine/mongoengine#113) | ||||||
|  |  | ||||||
|  | Changes in 0.7.1 | ||||||
|  | ================= | ||||||
|  | - Fixed index spec inheritance (MongoEngine/mongoengine#111) | ||||||
|  |  | ||||||
|  | Changes in 0.7.0 | ||||||
|  | ================= | ||||||
|  | - Updated queryset.delete so you can use with skip / limit (MongoEngine/mongoengine#107) | ||||||
|  | - Updated index creation allows kwargs to be passed through refs (MongoEngine/mongoengine#104) | ||||||
|  | - Fixed Q object merge edge case (MongoEngine/mongoengine#109) | ||||||
|  | - Fixed reloading on sharded documents (hmarr/mongoengine#569) | ||||||
|  | - Added NotUniqueError for duplicate keys (MongoEngine/mongoengine#62) | ||||||
|  | - Added custom collection / sequence naming for SequenceFields (MongoEngine/mongoengine#92) | ||||||
|  | - Fixed UnboundLocalError in composite index with pk field (MongoEngine/mongoengine#88) | ||||||
|  | - Updated ReferenceField's to optionally store ObjectId strings | ||||||
|  |   this will become the default in 0.8 (MongoEngine/mongoengine#89) | ||||||
|  | - Added FutureWarning - save will default to `cascade=False` in 0.8 | ||||||
|  | - Added example of indexing embedded document fields (MongoEngine/mongoengine#75) | ||||||
|  | - Fixed ImageField resizing when forcing size (MongoEngine/mongoengine#80) | ||||||
|  | - Add flexibility for fields handling bad data (MongoEngine/mongoengine#78) | ||||||
|  | - Embedded Documents no longer handle meta definitions | ||||||
|  | - Use weakref proxies in base lists / dicts (MongoEngine/mongoengine#74) | ||||||
|  | - Improved queryset filtering (hmarr/mongoengine#554) | ||||||
|  | - Fixed Dynamic Documents and Embedded Documents (hmarr/mongoengine#561) | ||||||
|  | - Fixed abstract classes and shard keys (MongoEngine/mongoengine#64) | ||||||
|  | - Fixed Python 2.5 support | ||||||
|  | - Added Python 3 support (thanks to Laine Heron) | ||||||
|  |  | ||||||
| Changes in 0.6.20 | Changes in 0.6.20 | ||||||
| ================= | ================= | ||||||
| - Added support for distinct and db_alias (MongoEngine/mongoengine#59) | - Added support for distinct and db_alias (MongoEngine/mongoengine#59) | ||||||
| @@ -86,7 +176,7 @@ Changes in 0.6.8 | |||||||
| ================ | ================ | ||||||
| - Fixed FileField losing reference when no default set | - Fixed FileField losing reference when no default set | ||||||
| - Removed possible race condition from FileField (grid_file) | - Removed possible race condition from FileField (grid_file) | ||||||
| - Added assignment to save, can now do: b = MyDoc(**kwargs).save() | - Added assignment to save, can now do: `b = MyDoc(**kwargs).save()` | ||||||
| - Added support for pull operations on nested EmbeddedDocuments | - Added support for pull operations on nested EmbeddedDocuments | ||||||
| - Added support for choices with GenericReferenceFields | - Added support for choices with GenericReferenceFields | ||||||
| - Added support for choices with GenericEmbeddedDocumentFields | - Added support for choices with GenericEmbeddedDocumentFields | ||||||
|   | |||||||
| @@ -10,6 +10,16 @@ In your **settings.py** file, ignore the standard database settings (unless you | |||||||
| also plan to use the ORM in your project), and instead call | also plan to use the ORM in your project), and instead call | ||||||
| :func:`~mongoengine.connect` somewhere in the settings module. | :func:`~mongoengine.connect` somewhere in the settings module. | ||||||
|  |  | ||||||
|  | .. note :: | ||||||
|  |    If you are not using another Database backend make sure you add  a dummy | ||||||
|  |    backend, by adding the following to ``settings.py``:: | ||||||
|  |  | ||||||
|  |         DATABASES = { | ||||||
|  |             'default': { | ||||||
|  |                 'ENGINE': 'django.db.backends.dummy' | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
| Authentication | Authentication | ||||||
| ============== | ============== | ||||||
| MongoEngine includes a Django authentication backend, which uses MongoDB. The | MongoEngine includes a Django authentication backend, which uses MongoDB. The | ||||||
| @@ -45,6 +55,9 @@ into you settings module:: | |||||||
|  |  | ||||||
|     SESSION_ENGINE = 'mongoengine.django.sessions' |     SESSION_ENGINE = 'mongoengine.django.sessions' | ||||||
|  |  | ||||||
|  | Django provides session cookie, which expires after ```SESSION_COOKIE_AGE``` seconds, but doesnt delete cookie at sessions backend, so ``'mongoengine.django.sessions'`` supports  `mongodb TTL | ||||||
|  | <http://docs.mongodb.org/manual/tutorial/expire-data/>`_. | ||||||
|  |  | ||||||
| .. versionadded:: 0.2.1 | .. versionadded:: 0.2.1 | ||||||
|  |  | ||||||
| Storage | Storage | ||||||
|   | |||||||
| @@ -344,6 +344,10 @@ Its value can take any of the following constants: | |||||||
|    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, | ||||||
| @@ -465,13 +469,18 @@ If a dictionary is passed then the following options are available: | |||||||
|     Whether the index should be sparse. |     Whether the index should be sparse. | ||||||
|  |  | ||||||
| :attr:`unique` (Default: False) | :attr:`unique` (Default: False) | ||||||
|     Whether the index should be sparse. |     Whether the index should be unique. | ||||||
|  |  | ||||||
|  | .. note :: | ||||||
|  |  | ||||||
|  |     To index embedded files / dictionary fields use 'dot' notation eg: | ||||||
|  |     `rank.title` | ||||||
|  |  | ||||||
| .. warning:: | .. warning:: | ||||||
|  |  | ||||||
|  |     Inheritance adds extra indices. | ||||||
|    Inheritance adds extra indices. |     If don't need inheritance for a document turn inheritance off - | ||||||
|    If don't need inheritance for a document turn inheritance off - see :ref:`document-inheritance`. |     see :ref:`document-inheritance`. | ||||||
|  |  | ||||||
|  |  | ||||||
| Geospatial indexes | Geospatial indexes | ||||||
|   | |||||||
| @@ -50,4 +50,11 @@ Example usage:: | |||||||
|     signals.post_save.connect(Author.post_save, sender=Author) |     signals.post_save.connect(Author.post_save, sender=Author) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ReferenceFields and signals | ||||||
|  | --------------------------- | ||||||
|  |  | ||||||
|  | Currently `reverse_delete_rules` do not trigger signals on the other part of | ||||||
|  | the relationship.  If this is required you must manually handled the | ||||||
|  | reverse deletion. | ||||||
|  |  | ||||||
| .. _blinker: http://pypi.python.org/pypi/blinker | .. _blinker: http://pypi.python.org/pypi/blinker | ||||||
|   | |||||||
| @@ -34,10 +34,10 @@ To get help with using MongoEngine, use the `MongoEngine Users mailing list | |||||||
| Contributing | Contributing | ||||||
| ------------ | ------------ | ||||||
|  |  | ||||||
| The source is available on `GitHub <http://github.com/hmarr/mongoengine>`_ and | The source is available on `GitHub <http://github.com/MongoEngine/mongoengine>`_ and | ||||||
| contributions are always encouraged. Contributions can be as simple as | contributions are always encouraged. Contributions can be as simple as | ||||||
| minor tweaks to this documentation. To contribute, fork the project on | minor tweaks to this documentation. To contribute, fork the project on | ||||||
| `GitHub <http://github.com/hmarr/mongoengine>`_ and send a | `GitHub <http://github.com/MongoEngine/mongoengine>`_ and send a | ||||||
| pull request. | pull request. | ||||||
|  |  | ||||||
| Also, you can join the developers' `mailing list | Also, you can join the developers' `mailing list | ||||||
|   | |||||||
							
								
								
									
										100
									
								
								docs/upgrade.rst
									
									
									
									
									
								
							
							
						
						
									
										100
									
								
								docs/upgrade.rst
									
									
									
									
									
								
							| @@ -2,18 +2,86 @@ | |||||||
| Upgrading | Upgrading | ||||||
| ========= | ========= | ||||||
|  |  | ||||||
|  | 0.6 to 0.7 | ||||||
|  | ========== | ||||||
|  |  | ||||||
|  | Cascade saves | ||||||
|  | ------------- | ||||||
|  |  | ||||||
|  | Saves will raise a `FutureWarning` if they cascade and cascade hasn't been set | ||||||
|  | to True.  This is because in 0.8 it will default to False.  If you require | ||||||
|  | cascading saves then either set it in the `meta` or pass | ||||||
|  | via `save` eg :: | ||||||
|  |  | ||||||
|  |     # At the class level: | ||||||
|  |     class Person(Document): | ||||||
|  |         meta = {'cascade': True} | ||||||
|  |  | ||||||
|  |     # Or in code: | ||||||
|  |     my_document.save(cascade=True) | ||||||
|  |  | ||||||
|  | .. note :: | ||||||
|  |     Remember: cascading saves **do not** cascade through lists. | ||||||
|  |  | ||||||
|  | ReferenceFields | ||||||
|  | --------------- | ||||||
|  |  | ||||||
|  | ReferenceFields now can store references as ObjectId strings instead of DBRefs. | ||||||
|  | This will become the default in 0.8 and if `dbref` is not set a `FutureWarning` | ||||||
|  | will be raised. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | To explicitly continue to use DBRefs change the `dbref` flag | ||||||
|  | to True :: | ||||||
|  |  | ||||||
|  |    class Person(Document): | ||||||
|  |        groups = ListField(ReferenceField(Group, dbref=True)) | ||||||
|  |  | ||||||
|  | To migrate to using strings instead of DBRefs you will have to manually | ||||||
|  | migrate :: | ||||||
|  |  | ||||||
|  |         # Step 1 - Migrate the model definition | ||||||
|  |         class Group(Document): | ||||||
|  |             author = ReferenceField(User, dbref=False) | ||||||
|  |             members = ListField(ReferenceField(User, dbref=False)) | ||||||
|  |  | ||||||
|  |         # Step 2 - Migrate the data | ||||||
|  |         for g in Group.objects(): | ||||||
|  |             g.author = g.author | ||||||
|  |             g.members = g.members | ||||||
|  |             g.save() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | item_frequencies | ||||||
|  | ---------------- | ||||||
|  |  | ||||||
|  | In the 0.6 series we added support for null / zero / false values in | ||||||
|  | item_frequencies.  A side effect was to return keys in the value they are | ||||||
|  | stored in rather than as string representations.  Your code may need to be | ||||||
|  | updated to handle native types rather than strings keys for the results of | ||||||
|  | item frequency queries. | ||||||
|  |  | ||||||
|  | BinaryFields | ||||||
|  | ------------ | ||||||
|  |  | ||||||
|  | Binary fields have been updated so that they are native binary types.  If you | ||||||
|  | previously were doing `str` comparisons with binary field values you will have | ||||||
|  | to update and wrap the value in a `str`. | ||||||
|  |  | ||||||
| 0.5 to 0.6 | 0.5 to 0.6 | ||||||
| ========== | ========== | ||||||
|  |  | ||||||
| Embedded Documents - if you had a `pk` field you will have to rename it from `_id` | Embedded Documents - if you had a `pk` field you will have to rename it from | ||||||
| to `pk` as pk is no longer a property of Embedded Documents. | `_id` to `pk` as pk is no longer a property of Embedded Documents. | ||||||
|  |  | ||||||
| Reverse Delete Rules in Embedded Documents, MapFields and DictFields now throw | Reverse Delete Rules in Embedded Documents, MapFields and DictFields now throw | ||||||
| an InvalidDocument error as they aren't currently supported. | an InvalidDocument error as they aren't currently supported. | ||||||
|  |  | ||||||
| Document._get_subclasses - Is no longer used and the class method has been removed. | Document._get_subclasses - Is no longer used and the class method has been | ||||||
|  | removed. | ||||||
|  |  | ||||||
| Document.objects.with_id - now raises an InvalidQueryError if used with a filter. | Document.objects.with_id - now raises an InvalidQueryError if used with a | ||||||
|  | filter. | ||||||
|  |  | ||||||
| FutureWarning - A future warning has been added to all inherited classes that | FutureWarning - A future warning has been added to all inherited classes that | ||||||
| don't define `allow_inheritance` in their meta. | don't define `allow_inheritance` in their meta. | ||||||
| @@ -37,11 +105,11 @@ human-readable name for the option. | |||||||
| PyMongo / MongoDB | PyMongo / MongoDB | ||||||
| ----------------- | ----------------- | ||||||
|  |  | ||||||
| map reduce now requires pymongo 1.11+- The pymongo merge_output and reduce_output | map reduce now requires pymongo 1.11+- The pymongo `merge_output` and | ||||||
| parameters, have been depreciated. | `reduce_output` parameters, have been depreciated. | ||||||
|  |  | ||||||
| More methods now use map_reduce as db.eval is not supported for sharding as such | More methods now use map_reduce as db.eval is not supported for sharding as | ||||||
| the following have been changed: | such the following have been changed: | ||||||
|  |  | ||||||
|     * :meth:`~mongoengine.queryset.QuerySet.sum` |     * :meth:`~mongoengine.queryset.QuerySet.sum` | ||||||
|     * :meth:`~mongoengine.queryset.QuerySet.average` |     * :meth:`~mongoengine.queryset.QuerySet.average` | ||||||
| @@ -51,8 +119,8 @@ the following have been changed: | |||||||
| Default collection naming | Default collection naming | ||||||
| ------------------------- | ------------------------- | ||||||
|  |  | ||||||
| Previously it was just lowercase, its now much more pythonic and readable as its | Previously it was just lowercase, its now much more pythonic and readable as | ||||||
| lowercase and underscores, previously :: | its lowercase and underscores, previously :: | ||||||
|  |  | ||||||
|     class MyAceDocument(Document): |     class MyAceDocument(Document): | ||||||
|         pass |         pass | ||||||
| @@ -88,7 +156,8 @@ Alternatively, you can rename your collections eg :: | |||||||
|  |  | ||||||
|         failure = False |         failure = False | ||||||
|  |  | ||||||
|         collection_names = [d._get_collection_name() for d in _document_registry.values()] |         collection_names = [d._get_collection_name() | ||||||
|  |                             for d in _document_registry.values()] | ||||||
|  |  | ||||||
|         for new_style_name in collection_names: |         for new_style_name in collection_names: | ||||||
|             if not new_style_name:  # embedded documents don't have collections |             if not new_style_name:  # embedded documents don't have collections | ||||||
| @@ -106,10 +175,17 @@ Alternatively, you can rename your collections eg :: | |||||||
|                         old_style_name, new_style_name) |                         old_style_name, new_style_name) | ||||||
|                 else: |                 else: | ||||||
|                     db[old_style_name].rename(new_style_name) |                     db[old_style_name].rename(new_style_name) | ||||||
|                     print "Renamed:  %s to %s" % (old_style_name, new_style_name) |                     print "Renamed:  %s to %s" % (old_style_name, | ||||||
|  |                                                   new_style_name) | ||||||
|  |  | ||||||
|         if failure: |         if failure: | ||||||
|             print "Upgrading  collection names failed" |             print "Upgrading  collection names failed" | ||||||
|         else: |         else: | ||||||
|             print "Upgraded collection names" |             print "Upgraded collection names" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | mongodb 1.8 > 2.0 + | ||||||
|  | =================== | ||||||
|  |  | ||||||
|  | Its been reported that indexes may need to be recreated to the newer version of indexes.   | ||||||
|  | To do this drop indexes and call ``ensure_indexes`` on each model. | ||||||
|   | |||||||
| @@ -12,13 +12,12 @@ from signals import * | |||||||
| __all__ = (document.__all__ + fields.__all__ + connection.__all__ + | __all__ = (document.__all__ + fields.__all__ + connection.__all__ + | ||||||
|            queryset.__all__ + signals.__all__) |            queryset.__all__ + signals.__all__) | ||||||
|  |  | ||||||
| VERSION = (0, 6, 20) | VERSION = (0, 7, 10) | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_version(): | def get_version(): | ||||||
|     version = '%s.%s' % (VERSION[0], VERSION[1]) |     if isinstance(VERSION[-1], basestring): | ||||||
|     if VERSION[2]: |         return '.'.join(map(str, VERSION[:-1])) + VERSION[-1] | ||||||
|         version = '%s.%s' % (version, VERSION[2]) |     return '.'.join(map(str, VERSION)) | ||||||
|     return version |  | ||||||
|  |  | ||||||
| __version__ = get_version() | __version__ = get_version() | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -31,17 +31,34 @@ class DeReference(object): | |||||||
|             items = [i for i in items] |             items = [i for i in items] | ||||||
|  |  | ||||||
|         self.max_depth = max_depth |         self.max_depth = max_depth | ||||||
|  |  | ||||||
|         doc_type = None |         doc_type = None | ||||||
|         if instance and instance._fields: |  | ||||||
|             doc_type = instance._fields[name] |         if instance and isinstance(instance, (Document, TopLevelDocumentMetaclass)): | ||||||
|  |             doc_type = instance._fields.get(name) | ||||||
|             if hasattr(doc_type, 'field'): |             if hasattr(doc_type, 'field'): | ||||||
|                 doc_type = doc_type.field |                 doc_type = doc_type.field | ||||||
|  |  | ||||||
|             if isinstance(doc_type, ReferenceField): |             if isinstance(doc_type, ReferenceField): | ||||||
|  |                 field = doc_type | ||||||
|                 doc_type = doc_type.document_type |                 doc_type = doc_type.document_type | ||||||
|                 if all([i.__class__ == doc_type for i in items]): |                 is_list = not hasattr(items, 'items') | ||||||
|  |  | ||||||
|  |                 if is_list and all([i.__class__ == doc_type for i in items]): | ||||||
|                     return items |                     return items | ||||||
|  |                 elif not is_list and all([i.__class__ == doc_type | ||||||
|  |                                          for i in items.values()]): | ||||||
|  |                     return items | ||||||
|  |                 elif not field.dbref: | ||||||
|  |                     if not hasattr(items, 'items'): | ||||||
|  |                         items = [field.to_python(v) | ||||||
|  |                              if not isinstance(v, (DBRef, Document)) else v | ||||||
|  |                              for v in items] | ||||||
|  |                     else: | ||||||
|  |                         items = dict([ | ||||||
|  |                             (k, field.to_python(v)) | ||||||
|  |                             if not isinstance(v, (DBRef, Document)) else (k, v) | ||||||
|  |                             for k, v in items.iteritems()] | ||||||
|  |                         ) | ||||||
|  |  | ||||||
|         self.reference_map = self._find_references(items) |         self.reference_map = self._find_references(items) | ||||||
|         self.object_map = self._fetch_objects(doc_type=doc_type) |         self.object_map = self._fetch_objects(doc_type=doc_type) | ||||||
| @@ -67,7 +84,7 @@ class DeReference(object): | |||||||
|         # Recursively find dbreferences |         # Recursively find dbreferences | ||||||
|         depth += 1 |         depth += 1 | ||||||
|         for k, item in iterator: |         for k, item in iterator: | ||||||
|             if hasattr(item, '_fields'): |             if isinstance(item, Document): | ||||||
|                 for field_name, field in item._fields.iteritems(): |                 for field_name, field in item._fields.iteritems(): | ||||||
|                     v = item._data.get(field_name, None) |                     v = item._data.get(field_name, None) | ||||||
|                     if isinstance(v, (DBRef)): |                     if isinstance(v, (DBRef)): | ||||||
| @@ -98,7 +115,7 @@ class DeReference(object): | |||||||
|         object_map = {} |         object_map = {} | ||||||
|         for col, dbrefs in self.reference_map.iteritems(): |         for col, dbrefs in self.reference_map.iteritems(): | ||||||
|             keys = object_map.keys() |             keys = object_map.keys() | ||||||
|             refs = list(set([dbref for dbref in dbrefs if str(dbref) not in keys])) |             refs = list(set([dbref for dbref in dbrefs if unicode(dbref).encode('utf-8') not in keys])) | ||||||
|             if hasattr(col, 'objects'):  # We have a document class for the refs |             if hasattr(col, 'objects'):  # We have a document class for the refs | ||||||
|                 references = col.objects.in_bulk(refs) |                 references = col.objects.in_bulk(refs) | ||||||
|                 for key, doc in references.iteritems(): |                 for key, doc in references.iteritems(): | ||||||
| @@ -117,7 +134,7 @@ class DeReference(object): | |||||||
|                         elif doc_type is None: |                         elif doc_type is None: | ||||||
|                             doc = get_document( |                             doc = get_document( | ||||||
|                                 ''.join(x.capitalize() |                                 ''.join(x.capitalize() | ||||||
|                                         for x in col.split('_')))._from_son(ref) |                                     for x in col.split('_')))._from_son(ref) | ||||||
|                         else: |                         else: | ||||||
|                             doc = doc_type._from_son(ref) |                             doc = doc_type._from_son(ref) | ||||||
|                         object_map[doc.id] = doc |                         object_map[doc.id] = doc | ||||||
| @@ -149,11 +166,12 @@ class DeReference(object): | |||||||
|                 return self.object_map.get(items['_ref'].id, items) |                 return self.object_map.get(items['_ref'].id, items) | ||||||
|             elif '_types' in items and '_cls' in items: |             elif '_types' in items and '_cls' in items: | ||||||
|                 doc = get_document(items['_cls'])._from_son(items) |                 doc = get_document(items['_cls'])._from_son(items) | ||||||
|                 doc._data = self._attach_objects(doc._data, depth, doc, name) |                 doc._data = self._attach_objects(doc._data, depth, doc, None) | ||||||
|                 return doc |                 return doc | ||||||
|  |  | ||||||
|         if not hasattr(items, 'items'): |         if not hasattr(items, 'items'): | ||||||
|             is_list = True |             is_list = True | ||||||
|  |             as_tuple = isinstance(items, tuple) | ||||||
|             iterator = enumerate(items) |             iterator = enumerate(items) | ||||||
|             data = [] |             data = [] | ||||||
|         else: |         else: | ||||||
| @@ -170,7 +188,7 @@ class DeReference(object): | |||||||
|  |  | ||||||
|             if k in self.object_map and not is_list: |             if k in self.object_map and not is_list: | ||||||
|                 data[k] = self.object_map[k] |                 data[k] = self.object_map[k] | ||||||
|             elif hasattr(v, '_fields'): |             elif isinstance(v, Document): | ||||||
|                 for field_name, field in v._fields.iteritems(): |                 for field_name, field in v._fields.iteritems(): | ||||||
|                     v = data[k]._data.get(field_name, None) |                     v = data[k]._data.get(field_name, None) | ||||||
|                     if isinstance(v, (DBRef)): |                     if isinstance(v, (DBRef)): | ||||||
| @@ -188,7 +206,7 @@ class DeReference(object): | |||||||
|  |  | ||||||
|         if instance and name: |         if instance and name: | ||||||
|             if is_list: |             if is_list: | ||||||
|                 return BaseList(data, instance, name) |                 return tuple(data) if as_tuple else BaseList(data, instance, name) | ||||||
|             return BaseDict(data, instance, name) |             return BaseDict(data, instance, name) | ||||||
|         depth += 1 |         depth += 1 | ||||||
|         return data |         return data | ||||||
|   | |||||||
| @@ -3,6 +3,8 @@ import datetime | |||||||
| from mongoengine import * | from mongoengine import * | ||||||
|  |  | ||||||
| from django.utils.encoding import smart_str | from django.utils.encoding import smart_str | ||||||
|  | from django.contrib.auth.models import _user_get_all_permissions | ||||||
|  | from django.contrib.auth.models import _user_has_perm | ||||||
| from django.contrib.auth.models import AnonymousUser | from django.contrib.auth.models import AnonymousUser | ||||||
| from django.utils.translation import ugettext_lazy as _ | from django.utils.translation import ugettext_lazy as _ | ||||||
|  |  | ||||||
| @@ -104,6 +106,25 @@ class User(Document): | |||||||
|         """ |         """ | ||||||
|         return check_password(raw_password, self.password) |         return check_password(raw_password, self.password) | ||||||
|  |  | ||||||
|  |     def get_all_permissions(self, obj=None): | ||||||
|  |         return _user_get_all_permissions(self, obj) | ||||||
|  |  | ||||||
|  |     def has_perm(self, perm, obj=None): | ||||||
|  |         """ | ||||||
|  |         Returns True if the user has the specified permission. This method | ||||||
|  |         queries all available auth backends, but returns immediately if any | ||||||
|  |         backend returns True. Thus, a user who has permission from a single | ||||||
|  |         auth backend is assumed to have permission in general. If an object is | ||||||
|  |         provided, permissions for this specific object are checked. | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         # Active superusers have all permissions. | ||||||
|  |         if self.is_active and self.is_superuser: | ||||||
|  |             return True | ||||||
|  |  | ||||||
|  |         # Otherwise we need to check the backends. | ||||||
|  |         return _user_has_perm(self, perm, obj) | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def create_user(cls, username, password, email=None): |     def create_user(cls, username, password, email=None): | ||||||
|         """Create (and save) a new user with the given username, password and |         """Create (and save) a new user with the given username, password and | ||||||
|   | |||||||
| @@ -15,15 +15,33 @@ MONGOENGINE_SESSION_DB_ALIAS = getattr( | |||||||
|     settings, 'MONGOENGINE_SESSION_DB_ALIAS', |     settings, 'MONGOENGINE_SESSION_DB_ALIAS', | ||||||
|     DEFAULT_CONNECTION_NAME) |     DEFAULT_CONNECTION_NAME) | ||||||
|  |  | ||||||
|  | # a setting for the name of the collection used to store sessions | ||||||
|  | MONGOENGINE_SESSION_COLLECTION = getattr( | ||||||
|  |     settings, 'MONGOENGINE_SESSION_COLLECTION', | ||||||
|  |     'django_session') | ||||||
|  |  | ||||||
|  | # a setting for whether session data is stored encoded or not | ||||||
|  | MONGOENGINE_SESSION_DATA_ENCODE = getattr( | ||||||
|  |     settings, 'MONGOENGINE_SESSION_DATA_ENCODE', | ||||||
|  |     True) | ||||||
|  |  | ||||||
| class MongoSession(Document): | class MongoSession(Document): | ||||||
|     session_key = fields.StringField(primary_key=True, max_length=40) |     session_key = fields.StringField(primary_key=True, max_length=40) | ||||||
|     session_data = fields.StringField() |     session_data = fields.StringField() if MONGOENGINE_SESSION_DATA_ENCODE \ | ||||||
|  |                                         else fields.DictField() | ||||||
|     expire_date = fields.DateTimeField() |     expire_date = fields.DateTimeField() | ||||||
|  |  | ||||||
|     meta = {'collection': 'django_session', |     meta = { | ||||||
|             'db_alias': MONGOENGINE_SESSION_DB_ALIAS, |         'collection': MONGOENGINE_SESSION_COLLECTION, | ||||||
|             'allow_inheritance': False} |         'db_alias': MONGOENGINE_SESSION_DB_ALIAS, | ||||||
|  |         'allow_inheritance': False, | ||||||
|  |         'indexes': [ | ||||||
|  |             { | ||||||
|  |                 'fields': ['expire_date'], | ||||||
|  |                 'expireAfterSeconds': settings.SESSION_COOKIE_AGE | ||||||
|  |             } | ||||||
|  |         ] | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| class SessionStore(SessionBase): | class SessionStore(SessionBase): | ||||||
| @@ -34,7 +52,10 @@ class SessionStore(SessionBase): | |||||||
|         try: |         try: | ||||||
|             s = MongoSession.objects(session_key=self.session_key, |             s = MongoSession.objects(session_key=self.session_key, | ||||||
|                                      expire_date__gt=datetime.now())[0] |                                      expire_date__gt=datetime.now())[0] | ||||||
|             return self.decode(force_unicode(s.session_data)) |             if MONGOENGINE_SESSION_DATA_ENCODE: | ||||||
|  |                 return self.decode(force_unicode(s.session_data)) | ||||||
|  |             else: | ||||||
|  |                 return s.session_data | ||||||
|         except (IndexError, SuspiciousOperation): |         except (IndexError, SuspiciousOperation): | ||||||
|             self.create() |             self.create() | ||||||
|             return {} |             return {} | ||||||
| @@ -57,7 +78,10 @@ class SessionStore(SessionBase): | |||||||
|         if self.session_key is None: |         if self.session_key is None: | ||||||
|             self._session_key = self._get_new_session_key() |             self._session_key = self._get_new_session_key() | ||||||
|         s = MongoSession(session_key=self.session_key) |         s = MongoSession(session_key=self.session_key) | ||||||
|         s.session_data = self.encode(self._get_session(no_load=must_create)) |         if MONGOENGINE_SESSION_DATA_ENCODE: | ||||||
|  |             s.session_data = self.encode(self._get_session(no_load=must_create)) | ||||||
|  |         else: | ||||||
|  |             s.session_data = self._get_session(no_load=must_create) | ||||||
|         s.expire_date = self.get_expiry_date() |         s.expire_date = self.get_expiry_date() | ||||||
|         try: |         try: | ||||||
|             s.save(force_insert=must_create, safe=True) |             s.save(force_insert=must_create, safe=True) | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| from django.http import Http404 |  | ||||||
| from mongoengine.queryset import QuerySet | from mongoengine.queryset import QuerySet | ||||||
| from mongoengine.base import BaseDocument | from mongoengine.base import BaseDocument | ||||||
| from mongoengine.base import ValidationError | from mongoengine.base import ValidationError | ||||||
| @@ -27,6 +26,7 @@ def get_document_or_404(cls, *args, **kwargs): | |||||||
|     try: |     try: | ||||||
|         return queryset.get(*args, **kwargs) |         return queryset.get(*args, **kwargs) | ||||||
|     except (queryset._document.DoesNotExist, ValidationError): |     except (queryset._document.DoesNotExist, ValidationError): | ||||||
|  |         from django.http import Http404 | ||||||
|         raise Http404('No %s matches the given query.' % queryset._document._class_name) |         raise Http404('No %s matches the given query.' % queryset._document._class_name) | ||||||
|  |  | ||||||
| def get_list_or_404(cls, *args, **kwargs): | def get_list_or_404(cls, *args, **kwargs): | ||||||
| @@ -42,5 +42,6 @@ def get_list_or_404(cls, *args, **kwargs): | |||||||
|     queryset = _get_queryset(cls) |     queryset = _get_queryset(cls) | ||||||
|     obj_list = list(queryset.filter(*args, **kwargs)) |     obj_list = list(queryset.filter(*args, **kwargs)) | ||||||
|     if not obj_list: |     if not obj_list: | ||||||
|  |         from django.http import Http404 | ||||||
|         raise Http404('No %s matches the given query.' % queryset._document._class_name) |         raise Http404('No %s matches the given query.' % queryset._document._class_name) | ||||||
|     return obj_list |     return obj_list | ||||||
|   | |||||||
| @@ -1,10 +1,28 @@ | |||||||
| #coding: utf-8 | #coding: utf-8 | ||||||
| from django.test import TestCase | from nose.plugins.skip import SkipTest | ||||||
| from django.conf import settings |  | ||||||
|  |  | ||||||
|  | from mongoengine.python_support import PY3 | ||||||
| from mongoengine import connect | from mongoengine import connect | ||||||
|  |  | ||||||
|  | try: | ||||||
|  |     from django.test import TestCase | ||||||
|  |     from django.conf import settings | ||||||
|  | except Exception as err: | ||||||
|  |     if PY3: | ||||||
|  |         from unittest import TestCase | ||||||
|  |         # Dummy value so no error | ||||||
|  |         class settings: | ||||||
|  |             MONGO_DATABASE_NAME = 'dummy' | ||||||
|  |     else: | ||||||
|  |         raise err | ||||||
|  |  | ||||||
|  |  | ||||||
| class MongoTestCase(TestCase): | class MongoTestCase(TestCase): | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         if PY3: | ||||||
|  |             raise SkipTest('django does not have Python 3 support') | ||||||
|  |  | ||||||
|     """ |     """ | ||||||
|     TestCase class that clear the collection between the tests |     TestCase class that clear the collection between the tests | ||||||
|     """ |     """ | ||||||
|   | |||||||
| @@ -1,15 +1,19 @@ | |||||||
|  | import warnings | ||||||
|  |  | ||||||
| import pymongo | import pymongo | ||||||
|  | import re | ||||||
|  |  | ||||||
| from bson.dbref import DBRef | from bson.dbref import DBRef | ||||||
|  | from mongoengine import signals, queryset | ||||||
|  |  | ||||||
| from mongoengine import signals |  | ||||||
| from base import (DocumentMetaclass, TopLevelDocumentMetaclass, BaseDocument, | from base import (DocumentMetaclass, TopLevelDocumentMetaclass, BaseDocument, | ||||||
|                   BaseDict, BaseList) |                   BaseDict, BaseList) | ||||||
| from queryset import OperationError | from queryset import OperationError, NotUniqueError | ||||||
| from connection import get_db, DEFAULT_CONNECTION_NAME | from connection import get_db, DEFAULT_CONNECTION_NAME | ||||||
|  |  | ||||||
| __all__ = ['Document', 'EmbeddedDocument', 'DynamicDocument', | __all__ = ['Document', 'EmbeddedDocument', 'DynamicDocument', | ||||||
|            'DynamicEmbeddedDocument', 'OperationError', 'InvalidCollectionError'] |            'DynamicEmbeddedDocument', 'OperationError', | ||||||
|  |            'InvalidCollectionError', 'NotUniqueError'] | ||||||
|  |  | ||||||
|  |  | ||||||
| class InvalidCollectionError(Exception): | class InvalidCollectionError(Exception): | ||||||
| @@ -21,8 +25,19 @@ class EmbeddedDocument(BaseDocument): | |||||||
|     collection.  :class:`~mongoengine.EmbeddedDocument`\ s should be used as |     collection.  :class:`~mongoengine.EmbeddedDocument`\ s should be used as | ||||||
|     fields on :class:`~mongoengine.Document`\ s through the |     fields on :class:`~mongoengine.Document`\ s through the | ||||||
|     :class:`~mongoengine.EmbeddedDocumentField` field type. |     :class:`~mongoengine.EmbeddedDocumentField` field type. | ||||||
|  |  | ||||||
|  |     A :class:`~mongoengine.EmbeddedDocument` subclass may be itself subclassed, | ||||||
|  |     to create a specialised version of the embedded document that will be | ||||||
|  |     stored in the same collection. To facilitate this behaviour, `_cls` and | ||||||
|  |     `_types` fields are added to documents (hidden though the MongoEngine | ||||||
|  |     interface though). To disable this behaviour and remove the dependence on | ||||||
|  |     the presence of `_cls` and `_types`, set :attr:`allow_inheritance` to | ||||||
|  |     ``False`` in the :attr:`meta` dictionary. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|  |     # The __metaclass__ attribute is removed by 2to3 when running with Python3 | ||||||
|  |     # my_metaclass is defined so that metaclass can be queried in Python 2 & 3 | ||||||
|  |     my_metaclass  = DocumentMetaclass | ||||||
|     __metaclass__ = DocumentMetaclass |     __metaclass__ = DocumentMetaclass | ||||||
|  |  | ||||||
|     def __init__(self, *args, **kwargs): |     def __init__(self, *args, **kwargs): | ||||||
| @@ -91,9 +106,12 @@ class Document(BaseDocument): | |||||||
|     disabled by either setting types to False on the specific index or |     disabled by either setting types to False on the specific index or | ||||||
|     by setting index_types to False on the meta dictionary for the document. |     by setting index_types to False on the meta dictionary for the document. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|  |     # The __metaclass__ attribute is removed by 2to3 when running with Python3 | ||||||
|  |     # my_metaclass is defined so that metaclass can be queried in Python 2 & 3 | ||||||
|  |     my_metaclass  = TopLevelDocumentMetaclass | ||||||
|     __metaclass__ = TopLevelDocumentMetaclass |     __metaclass__ = TopLevelDocumentMetaclass | ||||||
|  |  | ||||||
|     @apply |  | ||||||
|     def pk(): |     def pk(): | ||||||
|         """Primary key alias |         """Primary key alias | ||||||
|         """ |         """ | ||||||
| @@ -102,6 +120,7 @@ class Document(BaseDocument): | |||||||
|         def fset(self, value): |         def fset(self, value): | ||||||
|             return setattr(self, self._meta['id_field'], value) |             return setattr(self, self._meta['id_field'], value) | ||||||
|         return property(fget, fset) |         return property(fget, fset) | ||||||
|  |     pk = pk() | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def _get_db(cls): |     def _get_db(cls): | ||||||
| @@ -127,8 +146,9 @@ class Document(BaseDocument): | |||||||
|                     options = cls._collection.options() |                     options = cls._collection.options() | ||||||
|                     if options.get('max') != max_documents or \ |                     if options.get('max') != max_documents or \ | ||||||
|                        options.get('size') != max_size: |                        options.get('size') != max_size: | ||||||
|                         msg = ('Cannot create collection "%s" as a capped ' |                         msg = (('Cannot create collection "%s" as a capped ' | ||||||
|                                'collection as it already exists') % cls._collection |                                'collection as it already exists') | ||||||
|  |                                 % cls._collection) | ||||||
|                         raise InvalidCollectionError(msg) |                         raise InvalidCollectionError(msg) | ||||||
|                 else: |                 else: | ||||||
|                     # Create the collection as a capped collection |                     # Create the collection as a capped collection | ||||||
| @@ -142,8 +162,9 @@ class Document(BaseDocument): | |||||||
|                 cls._collection = db[collection_name] |                 cls._collection = db[collection_name] | ||||||
|         return cls._collection |         return cls._collection | ||||||
|  |  | ||||||
|     def save(self, safe=True, force_insert=False, validate=True, write_options=None, |     def save(self, safe=True, force_insert=False, validate=True, | ||||||
|             cascade=None, cascade_kwargs=None, _refs=None): |              write_options=None,  cascade=None, cascade_kwargs=None, | ||||||
|  |              _refs=None, **kwargs): | ||||||
|         """Save the :class:`~mongoengine.Document` to the database. If the |         """Save the :class:`~mongoengine.Document` to the database. If the | ||||||
|         document already exists, it will be updated, otherwise it will be |         document already exists, it will be updated, otherwise it will be | ||||||
|         created. |         created. | ||||||
| @@ -156,27 +177,30 @@ class Document(BaseDocument): | |||||||
|             updates of existing documents |             updates of existing documents | ||||||
|         :param validate: validates the document; set to ``False`` to skip. |         :param validate: validates the document; set to ``False`` to skip. | ||||||
|         :param write_options: Extra keyword arguments are passed down to |         :param write_options: Extra keyword arguments are passed down to | ||||||
|                 :meth:`~pymongo.collection.Collection.save` OR |             :meth:`~pymongo.collection.Collection.save` OR | ||||||
|                 :meth:`~pymongo.collection.Collection.insert` |             :meth:`~pymongo.collection.Collection.insert` | ||||||
|                 which will be used as options for the resultant ``getLastError`` command. |             which will be used as options for the resultant | ||||||
|                 For example, ``save(..., write_options={w: 2, fsync: True}, ...)`` will |             ``getLastError`` command.  For example, | ||||||
|                 wait until at least two servers have recorded the write and will force an |             ``save(..., write_options={w: 2, fsync: True}, ...)`` will | ||||||
|                 fsync on each server being written to. |             wait until at least two servers have recorded the write and | ||||||
|         :param cascade: Sets the flag for cascading saves.  You can set a default by setting |             will force an fsync on the primary server. | ||||||
|             "cascade" in the document __meta__ |         :param cascade: Sets the flag for cascading saves.  You can set a | ||||||
|         :param cascade_kwargs: optional kwargs dictionary to be passed throw to cascading saves |             default by setting "cascade" in the document __meta__ | ||||||
|  |         :param cascade_kwargs: optional kwargs dictionary to be passed throw | ||||||
|  |             to cascading saves | ||||||
|         :param _refs: A list of processed references used in cascading saves |         :param _refs: A list of processed references used in cascading saves | ||||||
|  |  | ||||||
|         .. versionchanged:: 0.5 |         .. versionchanged:: 0.5 | ||||||
|             In existing documents it only saves changed fields using set / unset |             In existing documents it only saves changed fields using | ||||||
|             Saves are cascaded and any :class:`~bson.dbref.DBRef` objects |             set / unset.  Saves are cascaded and any | ||||||
|             that have changes are saved as well. |             :class:`~bson.dbref.DBRef` objects that have changes are | ||||||
|  |             saved as well. | ||||||
|         .. versionchanged:: 0.6 |         .. versionchanged:: 0.6 | ||||||
|             Cascade saves are optional = defaults to True, if you want fine grain |             Cascade saves are optional = defaults to True, if you want | ||||||
|             control then you can turn off using document meta['cascade'] = False |             fine grain control then you can turn off using document | ||||||
|             Also you can pass different kwargs to the cascade save using cascade_kwargs |             meta['cascade'] = False  Also you can pass different kwargs to | ||||||
|             which overwrites the existing kwargs with custom values |             the cascade save using cascade_kwargs which overwrites the | ||||||
|  |             existing kwargs with custom values | ||||||
|         """ |         """ | ||||||
|         signals.pre_save.send(self.__class__, document=self) |         signals.pre_save.send(self.__class__, document=self) | ||||||
|  |  | ||||||
| @@ -194,13 +218,14 @@ class Document(BaseDocument): | |||||||
|             collection = self.__class__.objects._collection |             collection = self.__class__.objects._collection | ||||||
|             if created: |             if created: | ||||||
|                 if force_insert: |                 if force_insert: | ||||||
|                     object_id = collection.insert(doc, safe=safe, **write_options) |                     object_id = collection.insert(doc, safe=safe, | ||||||
|  |                                                   **write_options) | ||||||
|                 else: |                 else: | ||||||
|                     object_id = collection.save(doc, safe=safe, **write_options) |                     object_id = collection.save(doc, safe=safe, | ||||||
|  |                                                 **write_options) | ||||||
|             else: |             else: | ||||||
|                 object_id = doc['_id'] |                 object_id = doc['_id'] | ||||||
|                 updates, removals = self._delta() |                 updates, removals = self._delta() | ||||||
|  |  | ||||||
|                 # Need to add shard key to query, or you get an error |                 # Need to add shard key to query, or you get an error | ||||||
|                 select_dict = {'_id': object_id} |                 select_dict = {'_id': object_id} | ||||||
|                 shard_key = self.__class__._meta.get('shard_key', tuple()) |                 shard_key = self.__class__._meta.get('shard_key', tuple()) | ||||||
| @@ -210,11 +235,15 @@ class Document(BaseDocument): | |||||||
|  |  | ||||||
|                 upsert = self._created |                 upsert = self._created | ||||||
|                 if updates: |                 if updates: | ||||||
|                     collection.update(select_dict, {"$set": updates}, upsert=upsert, safe=safe, **write_options) |                     collection.update(select_dict, {"$set": updates}, | ||||||
|  |                         upsert=upsert, safe=safe, **write_options) | ||||||
|                 if removals: |                 if removals: | ||||||
|                     collection.update(select_dict, {"$unset": removals}, upsert=upsert, safe=safe, **write_options) |                     collection.update(select_dict, {"$unset": removals}, | ||||||
|  |                         upsert=upsert, safe=safe, **write_options) | ||||||
|  |  | ||||||
|             cascade = self._meta.get('cascade', True) if cascade is None else cascade |             warn_cascade = not cascade and 'cascade' not in self._meta | ||||||
|  |             cascade = (self._meta.get('cascade', True) | ||||||
|  |                        if cascade is None else cascade) | ||||||
|             if cascade: |             if cascade: | ||||||
|                 kwargs = { |                 kwargs = { | ||||||
|                     "safe": safe, |                     "safe": safe, | ||||||
| @@ -226,45 +255,64 @@ class Document(BaseDocument): | |||||||
|                 if cascade_kwargs:  # Allow granular control over cascades |                 if cascade_kwargs:  # Allow granular control over cascades | ||||||
|                     kwargs.update(cascade_kwargs) |                     kwargs.update(cascade_kwargs) | ||||||
|                 kwargs['_refs'] = _refs |                 kwargs['_refs'] = _refs | ||||||
|                 #self._changed_fields = [] |                 self.cascade_save(warn_cascade=warn_cascade, **kwargs) | ||||||
|                 self.cascade_save(**kwargs) |  | ||||||
|  |  | ||||||
|         except pymongo.errors.OperationFailure, err: |         except pymongo.errors.OperationFailure, err: | ||||||
|             message = 'Could not save document (%s)' |             message = 'Could not save document (%s)' | ||||||
|             if u'duplicate key' in unicode(err): |             if re.match('^E1100[01] duplicate key', unicode(err)): | ||||||
|  |                 # E11000 - duplicate key error index | ||||||
|  |                 # E11001 - duplicate key on update | ||||||
|                 message = u'Tried to save duplicate unique keys (%s)' |                 message = u'Tried to save duplicate unique keys (%s)' | ||||||
|  |                 raise NotUniqueError(message % unicode(err)) | ||||||
|             raise OperationError(message % unicode(err)) |             raise OperationError(message % unicode(err)) | ||||||
|         id_field = self._meta['id_field'] |         id_field = self._meta['id_field'] | ||||||
|         self[id_field] = self._fields[id_field].to_python(object_id) |         if id_field not in self._meta.get('shard_key', []): | ||||||
|  |             self[id_field] = self._fields[id_field].to_python(object_id) | ||||||
|  |  | ||||||
|         self._changed_fields = [] |         self._clear_changed_fields() | ||||||
|         self._created = False |         self._created = False | ||||||
|         signals.post_save.send(self.__class__, document=self, created=created) |         signals.post_save.send(self.__class__, document=self, created=created) | ||||||
|         return self |         return self | ||||||
|  |  | ||||||
|     def cascade_save(self, *args, **kwargs): |     def cascade_save(self, warn_cascade=None, *args, **kwargs): | ||||||
|         """Recursively saves any references / generic references on an object""" |         """Recursively saves any references / | ||||||
|         from fields import ReferenceField, GenericReferenceField |            generic references on an objects""" | ||||||
|  |         import fields | ||||||
|         _refs = kwargs.get('_refs', []) or [] |         _refs = kwargs.get('_refs', []) or [] | ||||||
|  |  | ||||||
|         for name, cls in self._fields.items(): |         for name, cls in self._fields.items(): | ||||||
|  |             if not isinstance(cls, (fields.ReferenceField, | ||||||
|             if not isinstance(cls, (ReferenceField, GenericReferenceField)): |                                     fields.GenericReferenceField)): | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             ref = getattr(self, name) |             ref = getattr(self, name) | ||||||
|             if not ref: |             if not ref or isinstance(ref, DBRef): | ||||||
|                 continue |                 continue | ||||||
|             if isinstance(ref, DBRef): |  | ||||||
|  |             if not getattr(ref, '_changed_fields', True): | ||||||
|                 continue |                 continue | ||||||
|  |  | ||||||
|             ref_id = "%s,%s" % (ref.__class__.__name__, str(ref._data)) |             ref_id = "%s,%s" % (ref.__class__.__name__, str(ref._data)) | ||||||
|             if ref and ref_id not in _refs: |             if ref and ref_id not in _refs: | ||||||
|  |                 if warn_cascade: | ||||||
|  |                     msg = ("Cascading saves will default to off in 0.8, " | ||||||
|  |                           "please  explicitly set `.save(cascade=True)`") | ||||||
|  |                     warnings.warn(msg, FutureWarning) | ||||||
|                 _refs.append(ref_id) |                 _refs.append(ref_id) | ||||||
|                 kwargs["_refs"] = _refs |                 kwargs["_refs"] = _refs | ||||||
|                 ref.save(**kwargs) |                 ref.save(**kwargs) | ||||||
|                 ref._changed_fields = [] |                 ref._changed_fields = [] | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def _object_key(self): | ||||||
|  |         """Dict to identify object in collection | ||||||
|  |         """ | ||||||
|  |         select_dict = {'pk': self.pk} | ||||||
|  |         shard_key = self.__class__._meta.get('shard_key', tuple()) | ||||||
|  |         for k in shard_key: | ||||||
|  |             select_dict[k] = getattr(self, k) | ||||||
|  |         return select_dict | ||||||
|  |  | ||||||
|     def update(self, **kwargs): |     def update(self, **kwargs): | ||||||
|         """Performs an update on the :class:`~mongoengine.Document` |         """Performs an update on the :class:`~mongoengine.Document` | ||||||
|         A convenience wrapper to :meth:`~mongoengine.QuerySet.update`. |         A convenience wrapper to :meth:`~mongoengine.QuerySet.update`. | ||||||
| @@ -276,11 +324,7 @@ class Document(BaseDocument): | |||||||
|             raise OperationError('attempt to update a document not yet saved') |             raise OperationError('attempt to update a document not yet saved') | ||||||
|  |  | ||||||
|         # Need to add shard key to query, or you get an error |         # Need to add shard key to query, or you get an error | ||||||
|         select_dict = {'pk': self.pk} |         return self.__class__.objects(**self._object_key).update_one(**kwargs) | ||||||
|         shard_key = self.__class__._meta.get('shard_key', tuple()) |  | ||||||
|         for k in shard_key: |  | ||||||
|             select_dict[k] = getattr(self, k) |  | ||||||
|         return self.__class__.objects(**select_dict).update_one(**kwargs) |  | ||||||
|  |  | ||||||
|     def delete(self, safe=False): |     def delete(self, safe=False): | ||||||
|         """Delete the :class:`~mongoengine.Document` from the database. This |         """Delete the :class:`~mongoengine.Document` from the database. This | ||||||
| @@ -291,7 +335,7 @@ class Document(BaseDocument): | |||||||
|         signals.pre_delete.send(self.__class__, document=self) |         signals.pre_delete.send(self.__class__, document=self) | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             self.__class__.objects(pk=self.pk).delete(safe=safe) |             self.__class__.objects(**self._object_key).delete(safe=safe) | ||||||
|         except pymongo.errors.OperationFailure, err: |         except pymongo.errors.OperationFailure, err: | ||||||
|             message = u'Could not delete document (%s)' % err.message |             message = u'Could not delete document (%s)' % err.message | ||||||
|             raise OperationError(message) |             raise OperationError(message) | ||||||
| @@ -304,8 +348,8 @@ class Document(BaseDocument): | |||||||
|  |  | ||||||
|         .. versionadded:: 0.5 |         .. versionadded:: 0.5 | ||||||
|         """ |         """ | ||||||
|         from dereference import DeReference |         import dereference | ||||||
|         self._data = DeReference()(self._data, max_depth) |         self._data = dereference.DeReference()(self._data, max_depth) | ||||||
|         return self |         return self | ||||||
|  |  | ||||||
|     def reload(self, max_depth=1): |     def reload(self, max_depth=1): | ||||||
| @@ -317,7 +361,12 @@ class Document(BaseDocument): | |||||||
|         id_field = self._meta['id_field'] |         id_field = self._meta['id_field'] | ||||||
|         obj = self.__class__.objects( |         obj = self.__class__.objects( | ||||||
|                 **{id_field: self[id_field]} |                 **{id_field: self[id_field]} | ||||||
|               ).first().select_related(max_depth=max_depth) |               ).limit(1).select_related(max_depth=max_depth) | ||||||
|  |         if obj: | ||||||
|  |             obj = obj[0] | ||||||
|  |         else: | ||||||
|  |             msg = "Reloaded document has been deleted" | ||||||
|  |             raise OperationError(msg) | ||||||
|         for field in self._fields: |         for field in self._fields: | ||||||
|             setattr(self, field, self._reload(field, obj[field])) |             setattr(self, field, self._reload(field, obj[field])) | ||||||
|         if self._dynamic: |         if self._dynamic: | ||||||
| @@ -353,17 +402,18 @@ class Document(BaseDocument): | |||||||
|         """This method registers the delete rules to apply when removing this |         """This method registers the delete rules to apply when removing this | ||||||
|         object. |         object. | ||||||
|         """ |         """ | ||||||
|         cls._meta['delete_rules'][(document_cls, field_name)] = rule |         delete_rules = cls._meta.get('delete_rules') or {} | ||||||
|  |         delete_rules[(document_cls, field_name)] = rule | ||||||
|  |         cls._meta['delete_rules'] = delete_rules | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def drop_collection(cls): |     def drop_collection(cls): | ||||||
|         """Drops the entire collection associated with this |         """Drops the entire collection associated with this | ||||||
|         :class:`~mongoengine.Document` type from the database. |         :class:`~mongoengine.Document` type from the database. | ||||||
|         """ |         """ | ||||||
|         from mongoengine.queryset import QuerySet |  | ||||||
|         db = cls._get_db() |         db = cls._get_db() | ||||||
|         db.drop_collection(cls._get_collection_name()) |         db.drop_collection(cls._get_collection_name()) | ||||||
|         QuerySet._reset_already_indexed(cls) |         queryset.QuerySet._reset_already_indexed(cls) | ||||||
|  |  | ||||||
|  |  | ||||||
| class DynamicDocument(Document): | class DynamicDocument(Document): | ||||||
| @@ -379,7 +429,12 @@ class DynamicDocument(Document): | |||||||
|  |  | ||||||
|         There is one caveat on Dynamic Documents: fields cannot start with `_` |         There is one caveat on Dynamic Documents: fields cannot start with `_` | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|  |     # The __metaclass__ attribute is removed by 2to3 when running with Python3 | ||||||
|  |     # my_metaclass is defined so that metaclass can be queried in Python 2 & 3 | ||||||
|  |     my_metaclass  = TopLevelDocumentMetaclass | ||||||
|     __metaclass__ = TopLevelDocumentMetaclass |     __metaclass__ = TopLevelDocumentMetaclass | ||||||
|  |  | ||||||
|     _dynamic = True |     _dynamic = True | ||||||
|  |  | ||||||
|     def __delattr__(self, *args, **kwargs): |     def __delattr__(self, *args, **kwargs): | ||||||
| @@ -398,7 +453,11 @@ class DynamicEmbeddedDocument(EmbeddedDocument): | |||||||
|     information about dynamic documents. |     information about dynamic documents. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|  |     # The __metaclass__ attribute is removed by 2to3 when running with Python3 | ||||||
|  |     # my_metaclass is defined so that metaclass can be queried in Python 2 & 3 | ||||||
|  |     my_metaclass  = DocumentMetaclass | ||||||
|     __metaclass__ = DocumentMetaclass |     __metaclass__ = DocumentMetaclass | ||||||
|  |  | ||||||
|     _dynamic = True |     _dynamic = True | ||||||
|  |  | ||||||
|     def __delattr__(self, *args, **kwargs): |     def __delattr__(self, *args, **kwargs): | ||||||
|   | |||||||
| @@ -1,18 +1,24 @@ | |||||||
| import datetime | import datetime | ||||||
| import time |  | ||||||
| import decimal | import decimal | ||||||
| import gridfs | import itertools | ||||||
| import re | import re | ||||||
|  | import time | ||||||
|  | import urllib2 | ||||||
|  | import urlparse | ||||||
| import uuid | import uuid | ||||||
| import warnings | import warnings | ||||||
|  | from operator import itemgetter | ||||||
|  |  | ||||||
|  | import gridfs | ||||||
| from bson import Binary, DBRef, SON, ObjectId | from bson import Binary, DBRef, SON, ObjectId | ||||||
|  |  | ||||||
|  | from mongoengine.python_support import (PY3, bin_type, txt_type, | ||||||
|  |                                         str_types, StringIO) | ||||||
| from base import (BaseField, ComplexBaseField, ObjectIdField, | from base import (BaseField, ComplexBaseField, ObjectIdField, | ||||||
|                   ValidationError, get_document, BaseDocument) |                   ValidationError, get_document, BaseDocument) | ||||||
| from queryset import DO_NOTHING, QuerySet | from queryset import DO_NOTHING, QuerySet | ||||||
| from document import Document, EmbeddedDocument | from document import Document, EmbeddedDocument | ||||||
| from connection import get_db, DEFAULT_CONNECTION_NAME | from connection import get_db, DEFAULT_CONNECTION_NAME | ||||||
| from operator import itemgetter |  | ||||||
|  |  | ||||||
|  |  | ||||||
| try: | try: | ||||||
| @@ -21,13 +27,7 @@ except ImportError: | |||||||
|     Image = None |     Image = None | ||||||
|     ImageOps = None |     ImageOps = None | ||||||
|  |  | ||||||
| try: | __all__ = ['StringField', 'IntField', 'LongField', 'FloatField', 'BooleanField', | ||||||
|     from cStringIO import StringIO |  | ||||||
| except ImportError: |  | ||||||
|     from StringIO import StringIO |  | ||||||
|  |  | ||||||
|  |  | ||||||
| __all__ = ['StringField', 'IntField', 'FloatField', 'BooleanField', |  | ||||||
|            'DateTimeField', 'EmbeddedDocumentField', 'ListField', 'DictField', |            'DateTimeField', 'EmbeddedDocumentField', 'ListField', 'DictField', | ||||||
|            'ObjectIdField', 'ReferenceField', 'ValidationError', 'MapField', |            'ObjectIdField', 'ReferenceField', 'ValidationError', 'MapField', | ||||||
|            'DecimalField', 'ComplexDateTimeField', 'URLField', 'DynamicField', |            'DecimalField', 'ComplexDateTimeField', 'URLField', 'DynamicField', | ||||||
| @@ -51,8 +51,11 @@ class StringField(BaseField): | |||||||
|     def to_python(self, value): |     def to_python(self, value): | ||||||
|         if isinstance(value, unicode): |         if isinstance(value, unicode): | ||||||
|             return value |             return value | ||||||
|         else: |         try: | ||||||
|             return value.decode('utf-8') |             value = value.decode('utf-8') | ||||||
|  |         except: | ||||||
|  |             pass | ||||||
|  |         return value | ||||||
|  |  | ||||||
|     def validate(self, value): |     def validate(self, value): | ||||||
|         if not isinstance(value, basestring): |         if not isinstance(value, basestring): | ||||||
| @@ -100,25 +103,30 @@ class URLField(StringField): | |||||||
|     .. versionadded:: 0.3 |     .. versionadded:: 0.3 | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     URL_REGEX = re.compile( |     _URL_REGEX = re.compile( | ||||||
|         r'^https?://' |         r'^(?:http|ftp)s?://' # http:// or https:// | ||||||
|         r'(?:(?:[A-Z0-9](?:[A-Z0-9-]{0,61}[A-Z0-9])?\.)+[A-Z]{2,6}\.?|' |         r'(?:(?:[A-Z0-9](?:[A-Z0-9-]{0,61}[A-Z0-9])?\.)+(?:[A-Z]{2,6}\.?|[A-Z0-9-]{2,}\.?)|' #domain... | ||||||
|         r'localhost|' |         r'localhost|' #localhost... | ||||||
|         r'\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})' |         r'\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3})' # ...or ip | ||||||
|         r'(?::\d+)?' |         r'(?::\d+)?' # optional port | ||||||
|         r'(?:/?|[/?]\S+)$', re.IGNORECASE |         r'(?:/?|[/?]\S+)$', re.IGNORECASE) | ||||||
|     ) |  | ||||||
|  |  | ||||||
|     def __init__(self, verify_exists=False, **kwargs): |     def __init__(self, verify_exists=False, url_regex=None, **kwargs): | ||||||
|         self.verify_exists = verify_exists |         self.verify_exists = verify_exists | ||||||
|  |         self.url_regex = url_regex or self._URL_REGEX | ||||||
|         super(URLField, self).__init__(**kwargs) |         super(URLField, self).__init__(**kwargs) | ||||||
|  |  | ||||||
|     def validate(self, value): |     def validate(self, value): | ||||||
|         if not URLField.URL_REGEX.match(value): |         if not self.url_regex.match(value): | ||||||
|             self.error('Invalid URL: %s' % value) |             self.error('Invalid URL: %s' % value) | ||||||
|  |             return | ||||||
|  |  | ||||||
|         if self.verify_exists: |         if self.verify_exists: | ||||||
|             import urllib2 |             warnings.warn( | ||||||
|  |                 "The URLField verify_exists argument has intractable security " | ||||||
|  |                 "and performance issues. Accordingly, it has been deprecated.", | ||||||
|  |             DeprecationWarning | ||||||
|  |             ) | ||||||
|             try: |             try: | ||||||
|                 request = urllib2.Request(value) |                 request = urllib2.Request(value) | ||||||
|                 urllib2.urlopen(request) |                 urllib2.urlopen(request) | ||||||
| @@ -135,16 +143,17 @@ class EmailField(StringField): | |||||||
|     EMAIL_REGEX = re.compile( |     EMAIL_REGEX = re.compile( | ||||||
|         r"(^[-!#$%&'*+/=?^_`{}|~0-9A-Z]+(\.[-!#$%&'*+/=?^_`{}|~0-9A-Z]+)*"  # dot-atom |         r"(^[-!#$%&'*+/=?^_`{}|~0-9A-Z]+(\.[-!#$%&'*+/=?^_`{}|~0-9A-Z]+)*"  # dot-atom | ||||||
|         r'|^"([\001-\010\013\014\016-\037!#-\[\]-\177]|\\[\001-011\013\014\016-\177])*"'  # quoted-string |         r'|^"([\001-\010\013\014\016-\037!#-\[\]-\177]|\\[\001-011\013\014\016-\177])*"'  # quoted-string | ||||||
|         r')@(?:[A-Z0-9](?:[A-Z0-9-]{0,61}[A-Z0-9])?\.)+[A-Z]{2,6}\.?$', re.IGNORECASE  # domain |         r')@(?:[A-Z0-9](?:[A-Z0-9-]{0,253}[A-Z0-9])?\.)+[A-Z]{2,6}\.?$', re.IGNORECASE  # domain | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|     def validate(self, value): |     def validate(self, value): | ||||||
|         if not EmailField.EMAIL_REGEX.match(value): |         if not EmailField.EMAIL_REGEX.match(value): | ||||||
|             self.error('Invalid Mail-address: %s' % value) |             self.error('Invalid Mail-address: %s' % value) | ||||||
|  |         super(EmailField, self).validate(value) | ||||||
|  |  | ||||||
|  |  | ||||||
| class IntField(BaseField): | class IntField(BaseField): | ||||||
|     """An integer field. |     """An 32-bit integer field. | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def __init__(self, min_value=None, max_value=None, **kwargs): |     def __init__(self, min_value=None, max_value=None, **kwargs): | ||||||
| @@ -152,7 +161,11 @@ class IntField(BaseField): | |||||||
|         super(IntField, self).__init__(**kwargs) |         super(IntField, self).__init__(**kwargs) | ||||||
|  |  | ||||||
|     def to_python(self, value): |     def to_python(self, value): | ||||||
|         return int(value) |         try: | ||||||
|  |             value = int(value) | ||||||
|  |         except ValueError: | ||||||
|  |             pass | ||||||
|  |         return value | ||||||
|  |  | ||||||
|     def validate(self, value): |     def validate(self, value): | ||||||
|         try: |         try: | ||||||
| @@ -173,6 +186,40 @@ class IntField(BaseField): | |||||||
|         return int(value) |         return int(value) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class LongField(BaseField): | ||||||
|  |     """An 64-bit integer field. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self, min_value=None, max_value=None, **kwargs): | ||||||
|  |         self.min_value, self.max_value = min_value, max_value | ||||||
|  |         super(LongField, self).__init__(**kwargs) | ||||||
|  |  | ||||||
|  |     def to_python(self, value): | ||||||
|  |         try: | ||||||
|  |             value = long(value) | ||||||
|  |         except ValueError: | ||||||
|  |             pass | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def validate(self, value): | ||||||
|  |         try: | ||||||
|  |             value = long(value) | ||||||
|  |         except: | ||||||
|  |             self.error('%s could not be converted to long' % value) | ||||||
|  |  | ||||||
|  |         if self.min_value is not None and value < self.min_value: | ||||||
|  |             self.error('Long value is too small') | ||||||
|  |  | ||||||
|  |         if self.max_value is not None and value > self.max_value: | ||||||
|  |             self.error('Long value is too large') | ||||||
|  |  | ||||||
|  |     def prepare_query_value(self, op, value): | ||||||
|  |         if value is None: | ||||||
|  |             return value | ||||||
|  |  | ||||||
|  |         return long(value) | ||||||
|  |  | ||||||
|  |  | ||||||
| class FloatField(BaseField): | class FloatField(BaseField): | ||||||
|     """An floating point number field. |     """An floating point number field. | ||||||
|     """ |     """ | ||||||
| @@ -182,7 +229,11 @@ class FloatField(BaseField): | |||||||
|         super(FloatField, self).__init__(**kwargs) |         super(FloatField, self).__init__(**kwargs) | ||||||
|  |  | ||||||
|     def to_python(self, value): |     def to_python(self, value): | ||||||
|         return float(value) |         try: | ||||||
|  |             value = float(value) | ||||||
|  |         except ValueError: | ||||||
|  |             pass | ||||||
|  |         return value | ||||||
|  |  | ||||||
|     def validate(self, value): |     def validate(self, value): | ||||||
|         if isinstance(value, int): |         if isinstance(value, int): | ||||||
| @@ -214,9 +265,14 @@ class DecimalField(BaseField): | |||||||
|         super(DecimalField, self).__init__(**kwargs) |         super(DecimalField, self).__init__(**kwargs) | ||||||
|  |  | ||||||
|     def to_python(self, value): |     def to_python(self, value): | ||||||
|  |         original_value = value | ||||||
|         if not isinstance(value, basestring): |         if not isinstance(value, basestring): | ||||||
|             value = unicode(value) |             value = unicode(value) | ||||||
|         return decimal.Decimal(value) |         try: | ||||||
|  |             value = decimal.Decimal(value) | ||||||
|  |         except ValueError: | ||||||
|  |             return original_value | ||||||
|  |         return value | ||||||
|  |  | ||||||
|     def to_mongo(self, value): |     def to_mongo(self, value): | ||||||
|         return unicode(value) |         return unicode(value) | ||||||
| @@ -244,7 +300,11 @@ class BooleanField(BaseField): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def to_python(self, value): |     def to_python(self, value): | ||||||
|         return bool(value) |         try: | ||||||
|  |             value = bool(value) | ||||||
|  |         except ValueError: | ||||||
|  |             pass | ||||||
|  |         return value | ||||||
|  |  | ||||||
|     def validate(self, value): |     def validate(self, value): | ||||||
|         if not isinstance(value, bool): |         if not isinstance(value, bool): | ||||||
| @@ -375,6 +435,8 @@ class ComplexDateTimeField(StringField): | |||||||
|         data = super(ComplexDateTimeField, self).__get__(instance, owner) |         data = super(ComplexDateTimeField, self).__get__(instance, owner) | ||||||
|         if data == None: |         if data == None: | ||||||
|             return datetime.datetime.now() |             return datetime.datetime.now() | ||||||
|  |         if isinstance(data, datetime.datetime): | ||||||
|  |             return data | ||||||
|         return self._convert_from_string(data) |         return self._convert_from_string(data) | ||||||
|  |  | ||||||
|     def __set__(self, instance, value): |     def __set__(self, instance, value): | ||||||
| @@ -387,7 +449,11 @@ class ComplexDateTimeField(StringField): | |||||||
|                        'ComplexDateTimeField') |                        'ComplexDateTimeField') | ||||||
|  |  | ||||||
|     def to_python(self, value): |     def to_python(self, value): | ||||||
|         return self._convert_from_string(value) |         original_value = value | ||||||
|  |         try: | ||||||
|  |             return self._convert_from_string(value) | ||||||
|  |         except: | ||||||
|  |             return original_value | ||||||
|  |  | ||||||
|     def to_mongo(self, value): |     def to_mongo(self, value): | ||||||
|         return self._convert_from_datetime(value) |         return self._convert_from_datetime(value) | ||||||
| @@ -451,8 +517,9 @@ class GenericEmbeddedDocumentField(BaseField): | |||||||
|  |  | ||||||
|     Only valid values are subclasses of :class:`~mongoengine.EmbeddedDocument`. |     Only valid values are subclasses of :class:`~mongoengine.EmbeddedDocument`. | ||||||
|  |  | ||||||
|     .. note:: You can use the choices param to limit the acceptable |     .. note :: | ||||||
|     EmbeddedDocument types |         You can use the choices param to limit the acceptable | ||||||
|  |         EmbeddedDocument types | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def prepare_query_value(self, op, value): |     def prepare_query_value(self, op, value): | ||||||
| @@ -670,7 +737,8 @@ class ReferenceField(BaseField): | |||||||
|       * NULLIFY     - Updates the reference to null. |       * NULLIFY     - Updates the reference to null. | ||||||
|       * CASCADE     - Deletes the documents associated with the reference. |       * CASCADE     - Deletes the documents associated with the reference. | ||||||
|       * DENY        - Prevent the deletion of the reference object. |       * DENY        - Prevent the deletion of the reference object. | ||||||
|       * PULL        - Pull the reference from a :class:`~mongoengine.ListField` of references |       * PULL        - Pull the reference from a :class:`~mongoengine.ListField` | ||||||
|  |                       of references | ||||||
|  |  | ||||||
|     Alternative syntax for registering delete rules (useful when implementing |     Alternative syntax for registering delete rules (useful when implementing | ||||||
|     bi-directional delete rules) |     bi-directional delete rules) | ||||||
| @@ -683,12 +751,19 @@ class ReferenceField(BaseField): | |||||||
|  |  | ||||||
|         Bar.register_delete_rule(Foo, 'bar', NULLIFY) |         Bar.register_delete_rule(Foo, 'bar', NULLIFY) | ||||||
|  |  | ||||||
|  |     .. note :: | ||||||
|  |         `reverse_delete_rules` do not trigger pre / post delete signals to be | ||||||
|  |         triggered. | ||||||
|  |  | ||||||
|     .. versionchanged:: 0.5 added `reverse_delete_rule` |     .. versionchanged:: 0.5 added `reverse_delete_rule` | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     def __init__(self, document_type, reverse_delete_rule=DO_NOTHING, **kwargs): |     def __init__(self, document_type, dbref=None, | ||||||
|  |                  reverse_delete_rule=DO_NOTHING, **kwargs): | ||||||
|         """Initialises the Reference Field. |         """Initialises the Reference Field. | ||||||
|  |  | ||||||
|  |         :param dbref:  Store the reference as :class:`~pymongo.dbref.DBRef` | ||||||
|  |           or as the :class:`~pymongo.objectid.ObjectId`.id . | ||||||
|         :param reverse_delete_rule: Determines what to do when the referring |         :param reverse_delete_rule: Determines what to do when the referring | ||||||
|           object is deleted |           object is deleted | ||||||
|         """ |         """ | ||||||
| @@ -696,6 +771,13 @@ class ReferenceField(BaseField): | |||||||
|             if not issubclass(document_type, (Document, basestring)): |             if not issubclass(document_type, (Document, basestring)): | ||||||
|                 self.error('Argument to ReferenceField constructor must be a ' |                 self.error('Argument to ReferenceField constructor must be a ' | ||||||
|                            'document class or a string') |                            'document class or a string') | ||||||
|  |  | ||||||
|  |         if dbref is None: | ||||||
|  |             msg = ("ReferenceFields will default to using ObjectId " | ||||||
|  |                    " strings in 0.8, set DBRef=True if this isn't desired") | ||||||
|  |             warnings.warn(msg, FutureWarning) | ||||||
|  |  | ||||||
|  |         self.dbref = dbref if dbref is not None else True  # To change in 0.8 | ||||||
|         self.document_type_obj = document_type |         self.document_type_obj = document_type | ||||||
|         self.reverse_delete_rule = reverse_delete_rule |         self.reverse_delete_rule = reverse_delete_rule | ||||||
|         super(ReferenceField, self).__init__(**kwargs) |         super(ReferenceField, self).__init__(**kwargs) | ||||||
| @@ -718,8 +800,9 @@ class ReferenceField(BaseField): | |||||||
|  |  | ||||||
|         # Get value from document instance if available |         # Get value from document instance if available | ||||||
|         value = instance._data.get(self.name) |         value = instance._data.get(self.name) | ||||||
|  |  | ||||||
|         # Dereference DBRefs |         # Dereference DBRefs | ||||||
|         if isinstance(value, (DBRef)): |         if isinstance(value, DBRef): | ||||||
|             value = self.document_type._get_db().dereference(value) |             value = self.document_type._get_db().dereference(value) | ||||||
|             if value is not None: |             if value is not None: | ||||||
|                 instance._data[self.name] = self.document_type._from_son(value) |                 instance._data[self.name] = self.document_type._from_son(value) | ||||||
| @@ -728,6 +811,10 @@ class ReferenceField(BaseField): | |||||||
|  |  | ||||||
|     def to_mongo(self, document): |     def to_mongo(self, document): | ||||||
|         if isinstance(document, DBRef): |         if isinstance(document, DBRef): | ||||||
|  |             if not self.dbref: | ||||||
|  |                 return document.id | ||||||
|  |             return document | ||||||
|  |         elif not self.dbref and isinstance(document, basestring): | ||||||
|             return document |             return document | ||||||
|  |  | ||||||
|         id_field_name = self.document_type._meta['id_field'] |         id_field_name = self.document_type._meta['id_field'] | ||||||
| @@ -735,7 +822,7 @@ class ReferenceField(BaseField): | |||||||
|  |  | ||||||
|         if isinstance(document, Document): |         if isinstance(document, Document): | ||||||
|             # We need the id from the saved object to create the DBRef |             # We need the id from the saved object to create the DBRef | ||||||
|             id_ = document.id |             id_ = document.pk | ||||||
|             if id_ is None: |             if id_ is None: | ||||||
|                 self.error('You can only reference documents once they have' |                 self.error('You can only reference documents once they have' | ||||||
|                            ' been saved to the database') |                            ' been saved to the database') | ||||||
| @@ -743,18 +830,30 @@ class ReferenceField(BaseField): | |||||||
|             id_ = document |             id_ = document | ||||||
|  |  | ||||||
|         id_ = id_field.to_mongo(id_) |         id_ = id_field.to_mongo(id_) | ||||||
|         collection = self.document_type._get_collection_name() |         if self.dbref: | ||||||
|         return DBRef(collection, id_) |             collection = self.document_type._get_collection_name() | ||||||
|  |             return DBRef(collection, id_) | ||||||
|  |  | ||||||
|  |         return id_ | ||||||
|  |  | ||||||
|  |     def to_python(self, value): | ||||||
|  |         """Convert a MongoDB-compatible type to a Python type. | ||||||
|  |         """ | ||||||
|  |         if (not self.dbref and | ||||||
|  |             not isinstance(value, (DBRef, Document, EmbeddedDocument))): | ||||||
|  |             collection = self.document_type._get_collection_name() | ||||||
|  |             value = DBRef(collection, self.document_type.id.to_python(value)) | ||||||
|  |         return value | ||||||
|  |  | ||||||
|     def prepare_query_value(self, op, value): |     def prepare_query_value(self, op, value): | ||||||
|         if value is None: |         if value is None: | ||||||
|             return None |             return None | ||||||
|  |  | ||||||
|         return self.to_mongo(value) |         return self.to_mongo(value) | ||||||
|  |  | ||||||
|     def validate(self, value): |     def validate(self, value): | ||||||
|  |  | ||||||
|         if not isinstance(value, (self.document_type, DBRef)): |         if not isinstance(value, (self.document_type, DBRef)): | ||||||
|             self.error('A ReferenceField only accepts DBRef') |             self.error("A ReferenceField only accepts DBRef or documents") | ||||||
|  |  | ||||||
|         if isinstance(value, Document) and value.id is None: |         if isinstance(value, Document) and value.id is None: | ||||||
|             self.error('You can only reference documents once they have been ' |             self.error('You can only reference documents once they have been ' | ||||||
| @@ -768,10 +867,12 @@ class GenericReferenceField(BaseField): | |||||||
|     """A reference to *any* :class:`~mongoengine.document.Document` subclass |     """A reference to *any* :class:`~mongoengine.document.Document` subclass | ||||||
|     that will be automatically dereferenced on access (lazily). |     that will be automatically dereferenced on access (lazily). | ||||||
|  |  | ||||||
|     .. note:: Any documents used as a generic reference must be registered in the |     .. note :: | ||||||
|     document registry.  Importing the model will automatically register it. |         * Any documents used as a generic reference must be registered in the | ||||||
|  |           document registry.  Importing the model will automatically register | ||||||
|  |           it. | ||||||
|  |  | ||||||
|     .. note:: You can use the choices param to limit the acceptable Document types |         * You can use the choices param to limit the acceptable Document types | ||||||
|  |  | ||||||
|     .. versionadded:: 0.3 |     .. versionadded:: 0.3 | ||||||
|     """ |     """ | ||||||
| @@ -842,12 +943,20 @@ class BinaryField(BaseField): | |||||||
|         self.max_bytes = max_bytes |         self.max_bytes = max_bytes | ||||||
|         super(BinaryField, self).__init__(**kwargs) |         super(BinaryField, self).__init__(**kwargs) | ||||||
|  |  | ||||||
|  |     def __set__(self, instance, value): | ||||||
|  |         """Handle bytearrays in python 3.1""" | ||||||
|  |         if PY3 and isinstance(value, bytearray): | ||||||
|  |             value = bin_type(value) | ||||||
|  |         return super(BinaryField, self).__set__(instance, value) | ||||||
|  |  | ||||||
|     def to_mongo(self, value): |     def to_mongo(self, value): | ||||||
|         return Binary(value) |         return Binary(value) | ||||||
|  |  | ||||||
|     def validate(self, value): |     def validate(self, value): | ||||||
|         if not isinstance(value, (basestring, Binary)): |         if not isinstance(value, (bin_type, txt_type, Binary)): | ||||||
|             self.error('BinaryField only accepts string or bson Binary values') |             self.error("BinaryField only accepts instances of " | ||||||
|  |                        "(%s, %s, Binary)" % ( | ||||||
|  |                         bin_type.__name__, txt_type.__name__)) | ||||||
|  |  | ||||||
|         if self.max_bytes is not None and len(value) > self.max_bytes: |         if self.max_bytes is not None and len(value) > self.max_bytes: | ||||||
|             self.error('Binary value is too long') |             self.error('Binary value is too long') | ||||||
| @@ -903,11 +1012,13 @@ class GridFSProxy(object): | |||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return '<%s: %s>' % (self.__class__.__name__, self.grid_id) |         return '<%s: %s>' % (self.__class__.__name__, self.grid_id) | ||||||
|  |  | ||||||
|     def __cmp__(self, other): |     def __eq__(self, other): | ||||||
|         if not isinstance(other, GridFSProxy): |         if isinstance(other, GridFSProxy): | ||||||
|             return -1 |             return  ((self.grid_id == other.grid_id) and | ||||||
|         return cmp((self.grid_id, self.collection_name, self.db_alias), |                      (self.collection_name == other.collection_name) and | ||||||
|                    (other.grid_id, other.collection_name, other.db_alias)) |                      (self.db_alias == other.db_alias)) | ||||||
|  |         else: | ||||||
|  |             return False | ||||||
|  |  | ||||||
|     @property |     @property | ||||||
|     def fs(self): |     def fs(self): | ||||||
| @@ -1020,7 +1131,8 @@ class FileField(BaseField): | |||||||
|  |  | ||||||
|     def __set__(self, instance, value): |     def __set__(self, instance, value): | ||||||
|         key = self.name |         key = self.name | ||||||
|         if (hasattr(value, 'read') and not isinstance(value, GridFSProxy)) or isinstance(value, basestring): |         if ((hasattr(value, 'read') and not | ||||||
|  |              isinstance(value, GridFSProxy)) or isinstance(value, str_types)): | ||||||
|             # using "FileField() = file/string" notation |             # using "FileField() = file/string" notation | ||||||
|             grid_file = instance._data.get(self.name) |             grid_file = instance._data.get(self.name) | ||||||
|             # If a file already exists, delete it |             # If a file already exists, delete it | ||||||
| @@ -1076,6 +1188,7 @@ class ImageGridFsProxy(GridFSProxy): | |||||||
|  |  | ||||||
|         try: |         try: | ||||||
|             img = Image.open(file_obj) |             img = Image.open(file_obj) | ||||||
|  |             img_format = img.format | ||||||
|         except: |         except: | ||||||
|             raise ValidationError('Invalid image') |             raise ValidationError('Invalid image') | ||||||
|  |  | ||||||
| @@ -1110,20 +1223,20 @@ class ImageGridFsProxy(GridFSProxy): | |||||||
|  |  | ||||||
|         if thumbnail: |         if thumbnail: | ||||||
|             thumb_id = self._put_thumbnail(thumbnail, |             thumb_id = self._put_thumbnail(thumbnail, | ||||||
|                                           img.format) |                                           img_format) | ||||||
|         else: |         else: | ||||||
|             thumb_id = None |             thumb_id = None | ||||||
|  |  | ||||||
|         w, h = img.size |         w, h = img.size | ||||||
|  |  | ||||||
|         io = StringIO() |         io = StringIO() | ||||||
|         img.save(io, img.format) |         img.save(io, img_format) | ||||||
|         io.seek(0) |         io.seek(0) | ||||||
|  |  | ||||||
|         return super(ImageGridFsProxy, self).put(io, |         return super(ImageGridFsProxy, self).put(io, | ||||||
|                                                  width=w, |                                                  width=w, | ||||||
|                                                  height=h, |                                                  height=h, | ||||||
|                                                  format=img.format, |                                                  format=img_format, | ||||||
|                                                  thumbnail_id=thumb_id, |                                                  thumbnail_id=thumb_id, | ||||||
|                                                  **kwargs) |                                                  **kwargs) | ||||||
|  |  | ||||||
| @@ -1209,11 +1322,15 @@ class ImageField(FileField): | |||||||
|         params_size = ('width', 'height', 'force') |         params_size = ('width', 'height', 'force') | ||||||
|         extra_args = dict(size=size, thumbnail_size=thumbnail_size) |         extra_args = dict(size=size, thumbnail_size=thumbnail_size) | ||||||
|         for att_name, att in extra_args.items(): |         for att_name, att in extra_args.items(): | ||||||
|             if att and (isinstance(att, tuple) or isinstance(att, list)): |             value = None | ||||||
|                 setattr(self, att_name, dict( |             if isinstance(att, (tuple, list)): | ||||||
|                         map(None, params_size, att))) |                 if PY3: | ||||||
|             else: |                     value = dict(itertools.zip_longest(params_size, att, | ||||||
|                 setattr(self, att_name, None) |                                                         fillvalue=None)) | ||||||
|  |                 else: | ||||||
|  |                     value = dict(map(None, params_size, att)) | ||||||
|  |  | ||||||
|  |             setattr(self, att_name, value) | ||||||
|  |  | ||||||
|         super(ImageField, self).__init__( |         super(ImageField, self).__init__( | ||||||
|             collection_name=collection_name, |             collection_name=collection_name, | ||||||
| @@ -1255,24 +1372,35 @@ class SequenceField(IntField): | |||||||
|  |  | ||||||
|     .. versionadded:: 0.5 |     .. versionadded:: 0.5 | ||||||
|     """ |     """ | ||||||
|     def __init__(self, collection_name=None, db_alias = None, *args, **kwargs): |     def __init__(self, collection_name=None, db_alias=None, sequence_name=None, *args, **kwargs): | ||||||
|         self.collection_name = collection_name or 'mongoengine.counters' |         self.collection_name = collection_name or 'mongoengine.counters' | ||||||
|         self.db_alias = db_alias or DEFAULT_CONNECTION_NAME |         self.db_alias = db_alias or DEFAULT_CONNECTION_NAME | ||||||
|  |         self.sequence_name = sequence_name | ||||||
|         return super(SequenceField, self).__init__(*args, **kwargs) |         return super(SequenceField, self).__init__(*args, **kwargs) | ||||||
|  |  | ||||||
|     def generate_new_value(self): |     def generate_new_value(self): | ||||||
|         """ |         """ | ||||||
|         Generate and Increment the counter |         Generate and Increment the counter | ||||||
|         """ |         """ | ||||||
|         sequence_id = "{0}.{1}".format(self.owner_document._get_collection_name(), |         sequence_name = self.get_sequence_name() | ||||||
|                                        self.name) |         sequence_id = "%s.%s" % (sequence_name, self.name) | ||||||
|         collection = get_db(alias = self.db_alias )[self.collection_name] |         collection = get_db(alias=self.db_alias)[self.collection_name] | ||||||
|         counter = collection.find_and_modify(query={"_id": sequence_id}, |         counter = collection.find_and_modify(query={"_id": sequence_id}, | ||||||
|                                              update={"$inc": {"next": 1}}, |                                              update={"$inc": {"next": 1}}, | ||||||
|                                              new=True, |                                              new=True, | ||||||
|                                              upsert=True) |                                              upsert=True) | ||||||
|         return counter['next'] |         return counter['next'] | ||||||
|  |  | ||||||
|  |     def get_sequence_name(self): | ||||||
|  |         if self.sequence_name: | ||||||
|  |             return self.sequence_name | ||||||
|  |         owner = self.owner_document | ||||||
|  |         if issubclass(owner, Document): | ||||||
|  |             return owner._get_collection_name() | ||||||
|  |         else: | ||||||
|  |             return ''.join('_%s' % c if c.isupper() else c | ||||||
|  |                             for c in owner._class_name).strip('_').lower() | ||||||
|  |  | ||||||
|     def __get__(self, instance, owner): |     def __get__(self, instance, owner): | ||||||
|  |  | ||||||
|         if instance is None: |         if instance is None: | ||||||
| @@ -1328,9 +1456,13 @@ class UUIDField(BaseField): | |||||||
|  |  | ||||||
|     def to_python(self, value): |     def to_python(self, value): | ||||||
|         if not self._binary: |         if not self._binary: | ||||||
|             if not isinstance(value, basestring): |             original_value = value | ||||||
|                 value = unicode(value) |             try: | ||||||
|             return uuid.UUID(value) |                 if not isinstance(value, basestring): | ||||||
|  |                     value = unicode(value) | ||||||
|  |                 return uuid.UUID(value) | ||||||
|  |             except: | ||||||
|  |                 return original_value | ||||||
|         return value |         return value | ||||||
|  |  | ||||||
|     def to_mongo(self, value): |     def to_mongo(self, value): | ||||||
|   | |||||||
							
								
								
									
										61
									
								
								mongoengine/python_support.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								mongoengine/python_support.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | |||||||
|  | """Helper functions and types to aid with Python 2.5 - 3 support.""" | ||||||
|  |  | ||||||
|  | import sys | ||||||
|  |  | ||||||
|  | PY3 = sys.version_info[0] == 3 | ||||||
|  | PY25 = sys.version_info[:2] == (2, 5) | ||||||
|  | UNICODE_KWARGS = int(''.join([str(x) for x in sys.version_info[:3]])) > 264 | ||||||
|  |  | ||||||
|  | if PY3: | ||||||
|  |     import codecs | ||||||
|  |     from io import BytesIO as StringIO | ||||||
|  |     # return s converted to binary.  b('test') should be equivalent to b'test' | ||||||
|  |     def b(s): | ||||||
|  |         return codecs.latin_1_encode(s)[0] | ||||||
|  |  | ||||||
|  |     bin_type = bytes | ||||||
|  |     txt_type   = str | ||||||
|  | else: | ||||||
|  |     try: | ||||||
|  |         from cStringIO import StringIO | ||||||
|  |     except ImportError: | ||||||
|  |         from StringIO import StringIO | ||||||
|  |  | ||||||
|  |     # Conversion to binary only necessary in Python 3 | ||||||
|  |     def b(s): | ||||||
|  |         return s | ||||||
|  |  | ||||||
|  |     bin_type = str | ||||||
|  |     txt_type = unicode | ||||||
|  |  | ||||||
|  | str_types = (bin_type, txt_type) | ||||||
|  |  | ||||||
|  | if PY25: | ||||||
|  |     def product(*args, **kwds): | ||||||
|  |         pools = map(tuple, args) * kwds.get('repeat', 1) | ||||||
|  |         result = [[]] | ||||||
|  |         for pool in pools: | ||||||
|  |             result = [x + [y] for x in result for y in pool] | ||||||
|  |         for prod in result: | ||||||
|  |             yield tuple(prod) | ||||||
|  |     reduce = reduce | ||||||
|  | else: | ||||||
|  |     from itertools import product | ||||||
|  |     from functools import reduce | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # For use with Python 2.5 | ||||||
|  | # converts all keys from unicode to str for d and all nested dictionaries | ||||||
|  | def to_str_keys_recursive(d): | ||||||
|  |     if isinstance(d, list): | ||||||
|  |         for val in d: | ||||||
|  |             if isinstance(val, (dict, list)): | ||||||
|  |                 to_str_keys_recursive(val) | ||||||
|  |     elif isinstance(d, dict): | ||||||
|  |         for key, val in d.items(): | ||||||
|  |             if isinstance(val, (dict, list)): | ||||||
|  |                 to_str_keys_recursive(val) | ||||||
|  |             if isinstance(key, unicode): | ||||||
|  |                 d[str(key)] = d.pop(key) | ||||||
|  |     else: | ||||||
|  |         raise ValueError("non list/dict parameter not allowed") | ||||||
| @@ -4,10 +4,14 @@ import copy | |||||||
| import itertools | import itertools | ||||||
| import operator | import operator | ||||||
|  |  | ||||||
|  | from collections import defaultdict | ||||||
| from functools import partial | from functools import partial | ||||||
|  |  | ||||||
|  | from mongoengine.python_support import product, reduce, PY3 | ||||||
|  |  | ||||||
| import pymongo | import pymongo | ||||||
| from bson.code import Code | from bson.code import Code | ||||||
|  | from bson.son import SON | ||||||
|  |  | ||||||
| from mongoengine import signals | from mongoengine import signals | ||||||
|  |  | ||||||
| @@ -42,6 +46,10 @@ class OperationError(Exception): | |||||||
|     pass |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class NotUniqueError(OperationError): | ||||||
|  |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
| RE_TYPE = type(re.compile('')) | RE_TYPE = type(re.compile('')) | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -120,7 +128,7 @@ class QueryTreeTransformerVisitor(QNodeVisitor): | |||||||
|             # the necessary parts. Then for each $or part, create a new query |             # the necessary parts. Then for each $or part, create a new query | ||||||
|             # that ANDs the necessary part with the $or part. |             # that ANDs the necessary part with the $or part. | ||||||
|             clauses = [] |             clauses = [] | ||||||
|             for or_group in itertools.product(*or_groups): |             for or_group in product(*or_groups): | ||||||
|                 q_object = reduce(lambda a, b: a & b, and_parts, Q()) |                 q_object = reduce(lambda a, b: a & b, and_parts, Q()) | ||||||
|                 q_object = reduce(lambda a, b: a & b, or_group, q_object) |                 q_object = reduce(lambda a, b: a & b, or_group, q_object) | ||||||
|                 clauses.append(q_object) |                 clauses.append(q_object) | ||||||
| @@ -211,7 +219,7 @@ class QNode(object): | |||||||
|     def _combine(self, other, operation): |     def _combine(self, other, operation): | ||||||
|         """Combine this node with another node into a QCombination object. |         """Combine this node with another node into a QCombination object. | ||||||
|         """ |         """ | ||||||
|         if other.empty: |         if getattr(other, 'empty', True): | ||||||
|             return self |             return self | ||||||
|  |  | ||||||
|         if self.empty: |         if self.empty: | ||||||
| @@ -329,6 +337,7 @@ class QuerySet(object): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     __already_indexed = set() |     __already_indexed = set() | ||||||
|  |     __dereference = False | ||||||
|  |  | ||||||
|     def __init__(self, document, collection): |     def __init__(self, document, collection): | ||||||
|         self._document = document |         self._document = document | ||||||
| @@ -345,10 +354,12 @@ class QuerySet(object): | |||||||
|         self._slave_okay = False |         self._slave_okay = False | ||||||
|         self._iter = False |         self._iter = False | ||||||
|         self._scalar = [] |         self._scalar = [] | ||||||
|  |         self._as_pymongo = False | ||||||
|  |         self._as_pymongo_coerce = False | ||||||
|  |  | ||||||
|         # If inheritance is allowed, only return instances and instances of |         # If inheritance is allowed, only return instances and instances of | ||||||
|         # subclasses of the class being used |         # subclasses of the class being used | ||||||
|         if document._meta.get('allow_inheritance'): |         if document._meta.get('allow_inheritance') != False: | ||||||
|             self._initial_query = {'_types': self._document._class_name} |             self._initial_query = {'_types': self._document._class_name} | ||||||
|             self._loaded_fields = QueryFieldList(always_include=['_cls']) |             self._loaded_fields = QueryFieldList(always_include=['_cls']) | ||||||
|         self._cursor_obj = None |         self._cursor_obj = None | ||||||
| @@ -356,6 +367,10 @@ class QuerySet(object): | |||||||
|         self._skip = None |         self._skip = None | ||||||
|         self._hint = -1  # Using -1 as None is a valid value for hint |         self._hint = -1  # Using -1 as None is a valid value for hint | ||||||
|  |  | ||||||
|  |     def __deepcopy__(self, memo): | ||||||
|  |         """Essential for chained queries with ReferenceFields involved""" | ||||||
|  |         return self.clone() | ||||||
|  |  | ||||||
|     def clone(self): |     def clone(self): | ||||||
|         """Creates a copy of the current :class:`~mongoengine.queryset.QuerySet` |         """Creates a copy of the current :class:`~mongoengine.queryset.QuerySet` | ||||||
|  |  | ||||||
| @@ -364,8 +379,8 @@ class QuerySet(object): | |||||||
|         c = self.__class__(self._document, self._collection_obj) |         c = self.__class__(self._document, self._collection_obj) | ||||||
|  |  | ||||||
|         copy_props = ('_initial_query', '_query_obj', '_where_clause', |         copy_props = ('_initial_query', '_query_obj', '_where_clause', | ||||||
|                     '_loaded_fields', '_ordering', '_snapshot', |                       '_loaded_fields', '_ordering', '_snapshot', '_timeout', | ||||||
|                     '_timeout', '_limit', '_skip', '_slave_okay', '_hint') |                       '_limit', '_skip', '_slave_okay', '_hint') | ||||||
|  |  | ||||||
|         for prop in copy_props: |         for prop in copy_props: | ||||||
|             val = getattr(self, prop) |             val = getattr(self, prop) | ||||||
| @@ -382,7 +397,7 @@ class QuerySet(object): | |||||||
|         return self._mongo_query |         return self._mongo_query | ||||||
|  |  | ||||||
|     def ensure_index(self, key_or_list, drop_dups=False, background=False, |     def ensure_index(self, key_or_list, drop_dups=False, background=False, | ||||||
|         **kwargs): |                      **kwargs): | ||||||
|         """Ensure that the given indexes are in place. |         """Ensure that the given indexes are in place. | ||||||
|  |  | ||||||
|         :param key_or_list: a single index key or a list of index keys (to |         :param key_or_list: a single index key or a list of index keys (to | ||||||
| @@ -390,12 +405,13 @@ class QuerySet(object): | |||||||
|             or a **-** to determine the index ordering |             or a **-** to determine the index ordering | ||||||
|         """ |         """ | ||||||
|         index_spec = QuerySet._build_index_spec(self._document, key_or_list) |         index_spec = QuerySet._build_index_spec(self._document, key_or_list) | ||||||
|         self._collection.ensure_index( |         index_spec = index_spec.copy() | ||||||
|             index_spec['fields'], |         fields = index_spec.pop('fields') | ||||||
|             drop_dups=drop_dups, |         index_spec['drop_dups'] = drop_dups | ||||||
|             background=background, |         index_spec['background'] = background | ||||||
|             sparse=index_spec.get('sparse', False), |         index_spec.update(kwargs) | ||||||
|             unique=index_spec.get('unique', False)) |  | ||||||
|  |         self._collection.ensure_index(fields, **index_spec) | ||||||
|         return self |         return self | ||||||
|  |  | ||||||
|     def __call__(self, q_obj=None, class_check=True, slave_okay=False, **query): |     def __call__(self, q_obj=None, class_check=True, slave_okay=False, **query): | ||||||
| @@ -438,7 +454,7 @@ class QuerySet(object): | |||||||
|         """ |         """ | ||||||
|         background = self._document._meta.get('index_background', False) |         background = self._document._meta.get('index_background', False) | ||||||
|         drop_dups = self._document._meta.get('index_drop_dups', False) |         drop_dups = self._document._meta.get('index_drop_dups', False) | ||||||
|         index_opts = self._document._meta.get('index_opts', {}) |         index_opts = self._document._meta.get('index_opts') or {} | ||||||
|         index_types = self._document._meta.get('index_types', True) |         index_types = self._document._meta.get('index_types', True) | ||||||
|  |  | ||||||
|         # determine if an index which we are creating includes |         # determine if an index which we are creating includes | ||||||
| @@ -446,6 +462,7 @@ class QuerySet(object): | |||||||
|         # an extra index on _type, as mongodb will use the existing |         # an extra index on _type, as mongodb will use the existing | ||||||
|         # index to service queries against _type |         # index to service queries against _type | ||||||
|         types_indexed = False |         types_indexed = False | ||||||
|  |  | ||||||
|         def includes_types(fields): |         def includes_types(fields): | ||||||
|             first_field = None |             first_field = None | ||||||
|             if len(fields): |             if len(fields): | ||||||
| @@ -462,13 +479,15 @@ class QuerySet(object): | |||||||
|                 background=background, drop_dups=drop_dups, **index_opts) |                 background=background, drop_dups=drop_dups, **index_opts) | ||||||
|  |  | ||||||
|         # Ensure document-defined indexes are created |         # Ensure document-defined indexes are created | ||||||
|         if self._document._meta['indexes']: |         if self._document._meta['index_specs']: | ||||||
|             for spec in self._document._meta['indexes']: |             index_spec = self._document._meta['index_specs'] | ||||||
|                 types_indexed = types_indexed or includes_types(spec['fields']) |             for spec in index_spec: | ||||||
|  |                 spec = spec.copy() | ||||||
|  |                 fields = spec.pop('fields') | ||||||
|  |                 types_indexed = types_indexed or includes_types(fields) | ||||||
|                 opts = index_opts.copy() |                 opts = index_opts.copy() | ||||||
|                 opts['unique'] = spec.get('unique', False) |                 opts.update(spec) | ||||||
|                 opts['sparse'] = spec.get('sparse', False) |                 self._collection.ensure_index(fields, | ||||||
|                 self._collection.ensure_index(spec['fields'], |  | ||||||
|                     background=background, **opts) |                     background=background, **opts) | ||||||
|  |  | ||||||
|         # If _types is being used (for polymorphism), it needs an index, |         # If _types is being used (for polymorphism), it needs an index, | ||||||
| @@ -489,13 +508,24 @@ class QuerySet(object): | |||||||
|         """ |         """ | ||||||
|         if isinstance(spec, basestring): |         if isinstance(spec, basestring): | ||||||
|             spec = {'fields': [spec]} |             spec = {'fields': [spec]} | ||||||
|         if isinstance(spec, (list, tuple)): |         elif isinstance(spec, (list, tuple)): | ||||||
|             spec = {'fields': spec} |             spec = {'fields': list(spec)} | ||||||
|  |         elif isinstance(spec, dict): | ||||||
|  |             spec = dict(spec) | ||||||
|  |  | ||||||
|         index_list = [] |         index_list = [] | ||||||
|         direction = None |         direction = None | ||||||
|         use_types = doc_cls._meta.get('allow_inheritance', True) |  | ||||||
|  |         allow_inheritance = doc_cls._meta.get('allow_inheritance') != False | ||||||
|  |  | ||||||
|  |         # If sparse - dont include types | ||||||
|  |         use_types = allow_inheritance and not spec.get('sparse', False) | ||||||
|  |  | ||||||
|         for key in spec['fields']: |         for key in spec['fields']: | ||||||
|  |             # If inherited spec continue | ||||||
|  |             if isinstance(key, (list, tuple)): | ||||||
|  |                 continue | ||||||
|  |  | ||||||
|             # Get ASCENDING direction from +, DESCENDING from -, and GEO2D from * |             # Get ASCENDING direction from +, DESCENDING from -, and GEO2D from * | ||||||
|             direction = pymongo.ASCENDING |             direction = pymongo.ASCENDING | ||||||
|             if key.startswith("-"): |             if key.startswith("-"): | ||||||
| @@ -510,24 +540,23 @@ class QuerySet(object): | |||||||
|             parts = key.split('.') |             parts = key.split('.') | ||||||
|             if parts in (['pk'], ['id'], ['_id']): |             if parts in (['pk'], ['id'], ['_id']): | ||||||
|                 key = '_id' |                 key = '_id' | ||||||
|  |                 fields = [] | ||||||
|             else: |             else: | ||||||
|                 fields = QuerySet._lookup_field(doc_cls, parts) |                 fields = QuerySet._lookup_field(doc_cls, parts) | ||||||
|                 parts = [field if field == '_id' else field.db_field for field in fields] |                 parts = [field if field == '_id' else field.db_field | ||||||
|  |                          for field in fields] | ||||||
|                 key = '.'.join(parts) |                 key = '.'.join(parts) | ||||||
|             index_list.append((key, direction)) |             index_list.append((key, direction)) | ||||||
|  |  | ||||||
|             # If sparse - dont include types |  | ||||||
|             if spec.get('sparse', False): |  | ||||||
|                 use_types = False |  | ||||||
|  |  | ||||||
|             # Check if a list field is being used, don't use _types if it is |             # Check if a list field is being used, don't use _types if it is | ||||||
|             if use_types and not all(f._index_with_types for f in fields): |             if use_types and not all(f._index_with_types for f in fields): | ||||||
|                 use_types = False |                 use_types = False | ||||||
|  |  | ||||||
|         # If _types is being used, prepend it to every specified index |         # If _types is being used, prepend it to every specified index | ||||||
|         index_types = doc_cls._meta.get('index_types', True) |         index_types = doc_cls._meta.get('index_types', True) | ||||||
|         allow_inheritance = doc_cls._meta.get('allow_inheritance') |  | ||||||
|         if spec.get('types', index_types) and allow_inheritance and use_types and direction is not pymongo.GEO2D: |         if (spec.get('types', index_types) and use_types | ||||||
|  |             and direction is not pymongo.GEO2D): | ||||||
|             index_list.insert(0, ('_types', 1)) |             index_list.insert(0, ('_types', 1)) | ||||||
|  |  | ||||||
|         spec['fields'] = index_list |         spec['fields'] = index_list | ||||||
| @@ -586,11 +615,13 @@ class QuerySet(object): | |||||||
|             if self._where_clause: |             if self._where_clause: | ||||||
|                 self._cursor_obj.where(self._where_clause) |                 self._cursor_obj.where(self._where_clause) | ||||||
|  |  | ||||||
|             # apply default ordering |  | ||||||
|             if self._ordering: |             if self._ordering: | ||||||
|  |                 # Apply query ordering | ||||||
|                 self._cursor_obj.sort(self._ordering) |                 self._cursor_obj.sort(self._ordering) | ||||||
|             elif self._document._meta['ordering']: |             elif self._document._meta['ordering']: | ||||||
|  |                 # Otherwise, apply the ordering from the document model | ||||||
|                 self.order_by(*self._document._meta['ordering']) |                 self.order_by(*self._document._meta['ordering']) | ||||||
|  |                 self._cursor_obj.sort(self._ordering) | ||||||
|  |  | ||||||
|             if self._limit is not None: |             if self._limit is not None: | ||||||
|                 self._cursor_obj.limit(self._limit - (self._skip or 0)) |                 self._cursor_obj.limit(self._limit - (self._skip or 0)) | ||||||
| @@ -600,7 +631,6 @@ class QuerySet(object): | |||||||
|  |  | ||||||
|             if self._hint != -1: |             if self._hint != -1: | ||||||
|                 self._cursor_obj.hint(self._hint) |                 self._cursor_obj.hint(self._hint) | ||||||
|  |  | ||||||
|         return self._cursor_obj |         return self._cursor_obj | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
| @@ -678,7 +708,8 @@ class QuerySet(object): | |||||||
|         custom_operators = ['match'] |         custom_operators = ['match'] | ||||||
|  |  | ||||||
|         mongo_query = {} |         mongo_query = {} | ||||||
|         for key, value in query.items(): |         merge_query = defaultdict(list) | ||||||
|  |         for key, value in sorted(query.items()): | ||||||
|             if key == "__raw__": |             if key == "__raw__": | ||||||
|                 mongo_query.update(value) |                 mongo_query.update(value) | ||||||
|                 continue |                 continue | ||||||
| @@ -766,22 +797,22 @@ class QuerySet(object): | |||||||
|             if op is None or key not in mongo_query: |             if op is None or key not in mongo_query: | ||||||
|                 mongo_query[key] = value |                 mongo_query[key] = value | ||||||
|             elif key in mongo_query: |             elif key in mongo_query: | ||||||
|                 if isinstance(mongo_query[key], dict) and isinstance(value, dict): |                 if key in mongo_query and isinstance(mongo_query[key], dict): | ||||||
|                     mongo_query[key].update(value) |                     mongo_query[key].update(value) | ||||||
|                 elif isinstance(mongo_query[key], list): |  | ||||||
|                     mongo_query[key].append(value) |  | ||||||
|                 else: |                 else: | ||||||
|                     mongo_query[key] = [mongo_query[key], value] |                     # Store for manually merging later | ||||||
|  |                     merge_query[key].append(value) | ||||||
|  |  | ||||||
|         for k, v in mongo_query.items(): |         # The queryset has been filter in such a way we must manually merge | ||||||
|  |         for k, v in merge_query.items(): | ||||||
|  |             merge_query[k].append(mongo_query[k]) | ||||||
|  |             del mongo_query[k] | ||||||
|             if isinstance(v, list): |             if isinstance(v, list): | ||||||
|                 value = [{k:val} for val in v] |                 value = [{k:val} for val in v] | ||||||
|                 if '$and' in mongo_query.keys(): |                 if '$and' in mongo_query.keys(): | ||||||
|                     mongo_query['$and'].append(value) |                     mongo_query['$and'].append(value) | ||||||
|                 else: |                 else: | ||||||
|                     mongo_query['$and'] = value |                     mongo_query['$and'] = value | ||||||
|                 del mongo_query[k] |  | ||||||
|  |  | ||||||
|         return mongo_query |         return mongo_query | ||||||
|  |  | ||||||
|     def get(self, *q_objs, **query): |     def get(self, *q_objs, **query): | ||||||
| @@ -904,7 +935,7 @@ class QuerySet(object): | |||||||
|             if not isinstance(doc, self._document): |             if not isinstance(doc, self._document): | ||||||
|                 msg = "Some documents inserted aren't instances of %s" % str(self._document) |                 msg = "Some documents inserted aren't instances of %s" % str(self._document) | ||||||
|                 raise OperationError(msg) |                 raise OperationError(msg) | ||||||
|             if doc.pk: |             if doc.pk and not doc._created: | ||||||
|                 msg = "Some documents have ObjectIds use doc.update() instead" |                 msg = "Some documents have ObjectIds use doc.update() instead" | ||||||
|                 raise OperationError(msg) |                 raise OperationError(msg) | ||||||
|             raw.append(doc.to_mongo()) |             raw.append(doc.to_mongo()) | ||||||
| @@ -914,8 +945,11 @@ class QuerySet(object): | |||||||
|             ids = self._collection.insert(raw, **write_options) |             ids = self._collection.insert(raw, **write_options) | ||||||
|         except pymongo.errors.OperationFailure, err: |         except pymongo.errors.OperationFailure, err: | ||||||
|             message = 'Could not save document (%s)' |             message = 'Could not save document (%s)' | ||||||
|             if u'duplicate key' in unicode(err): |             if re.match('^E1100[01] duplicate key', unicode(err)): | ||||||
|  |                 # E11000 - duplicate key error index | ||||||
|  |                 # E11001 - duplicate key on update | ||||||
|                 message = u'Tried to save duplicate unique keys (%s)' |                 message = u'Tried to save duplicate unique keys (%s)' | ||||||
|  |                 raise NotUniqueError(message % unicode(err)) | ||||||
|             raise OperationError(message % unicode(err)) |             raise OperationError(message % unicode(err)) | ||||||
|  |  | ||||||
|         if not load_bulk: |         if not load_bulk: | ||||||
| @@ -960,6 +994,9 @@ class QuerySet(object): | |||||||
|             for doc in docs: |             for doc in docs: | ||||||
|                 doc_map[doc['_id']] = self._get_scalar( |                 doc_map[doc['_id']] = self._get_scalar( | ||||||
|                         self._document._from_son(doc)) |                         self._document._from_son(doc)) | ||||||
|  |         elif self._as_pymongo: | ||||||
|  |             for doc in docs: | ||||||
|  |                 doc_map[doc['_id']] = self._get_as_pymongo(doc) | ||||||
|         else: |         else: | ||||||
|             for doc in docs: |             for doc in docs: | ||||||
|                 doc_map[doc['_id']] = self._document._from_son(doc) |                 doc_map[doc['_id']] = self._document._from_son(doc) | ||||||
| @@ -976,6 +1013,9 @@ class QuerySet(object): | |||||||
|             if self._scalar: |             if self._scalar: | ||||||
|                 return self._get_scalar(self._document._from_son( |                 return self._get_scalar(self._document._from_son( | ||||||
|                         self._cursor.next())) |                         self._cursor.next())) | ||||||
|  |             if self._as_pymongo: | ||||||
|  |                 return self._get_as_pymongo(self._cursor.next()) | ||||||
|  |  | ||||||
|             return self._document._from_son(self._cursor.next()) |             return self._document._from_son(self._cursor.next()) | ||||||
|         except StopIteration, e: |         except StopIteration, e: | ||||||
|             self.rewind() |             self.rewind() | ||||||
| @@ -1015,6 +1055,8 @@ class QuerySet(object): | |||||||
|                          :class:`~bson.code.Code` or string |                          :class:`~bson.code.Code` or string | ||||||
|         :param output: output collection name, if set to 'inline' will try to |         :param output: output collection name, if set to 'inline' will try to | ||||||
|                        use :class:`~pymongo.collection.Collection.inline_map_reduce` |                        use :class:`~pymongo.collection.Collection.inline_map_reduce` | ||||||
|  |                        This can also be a dictionary containing output options | ||||||
|  |                        see: http://docs.mongodb.org/manual/reference/commands/#mapReduce | ||||||
|         :param finalize_f: finalize function, an optional function that |         :param finalize_f: finalize function, an optional function that | ||||||
|                            performs any post-reduction processing. |                            performs any post-reduction processing. | ||||||
|         :param scope: values to insert into map/reduce global scope. Optional. |         :param scope: values to insert into map/reduce global scope. Optional. | ||||||
| @@ -1156,6 +1198,8 @@ class QuerySet(object): | |||||||
|             if self._scalar: |             if self._scalar: | ||||||
|                 return self._get_scalar(self._document._from_son( |                 return self._get_scalar(self._document._from_son( | ||||||
|                         self._cursor[key])) |                         self._cursor[key])) | ||||||
|  |             if self._as_pymongo: | ||||||
|  |                 return self._get_as_pymongo(self._cursor.next()) | ||||||
|             return self._document._from_son(self._cursor[key]) |             return self._document._from_son(self._cursor[key]) | ||||||
|         raise AttributeError |         raise AttributeError | ||||||
|  |  | ||||||
| @@ -1166,10 +1210,13 @@ class QuerySet(object): | |||||||
|  |  | ||||||
|         .. versionadded:: 0.4 |         .. versionadded:: 0.4 | ||||||
|         .. versionchanged:: 0.5 - Fixed handling references |         .. versionchanged:: 0.5 - Fixed handling references | ||||||
|  |         .. versionchanged:: 0.6 - Improved db_field refrence handling | ||||||
|         """ |         """ | ||||||
|         from dereference import DeReference |         try: | ||||||
|         return DeReference()(self._cursor.distinct(field), 1, |             field = self._fields_to_dbfields([field]).pop() | ||||||
|                              name=field, instance=self._document) |         finally: | ||||||
|  |             return self._dereference(self._cursor.distinct(field), 1, | ||||||
|  |                                      name=field, instance=self._document) | ||||||
|  |  | ||||||
|     def only(self, *fields): |     def only(self, *fields): | ||||||
|         """Load only a subset of this document's fields. :: |         """Load only a subset of this document's fields. :: | ||||||
| @@ -1274,7 +1321,8 @@ class QuerySet(object): | |||||||
|             key_list.append((key, direction)) |             key_list.append((key, direction)) | ||||||
|  |  | ||||||
|         self._ordering = key_list |         self._ordering = key_list | ||||||
|         self._cursor.sort(key_list) |         if self._cursor_obj: | ||||||
|  |             self._cursor_obj.sort(key_list) | ||||||
|         return self |         return self | ||||||
|  |  | ||||||
|     def explain(self, format=False): |     def explain(self, format=False): | ||||||
| @@ -1324,9 +1372,16 @@ class QuerySet(object): | |||||||
|         """ |         """ | ||||||
|         doc = self._document |         doc = self._document | ||||||
|  |  | ||||||
|  |         # Handle deletes where skips or limits have been applied | ||||||
|  |         if self._skip or self._limit: | ||||||
|  |             for doc in self: | ||||||
|  |                 doc.delete() | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         delete_rules = doc._meta.get('delete_rules') or {} | ||||||
|         # Check for DENY rules before actually deleting/nullifying any other |         # Check for DENY rules before actually deleting/nullifying any other | ||||||
|         # references |         # references | ||||||
|         for rule_entry in doc._meta['delete_rules']: |         for rule_entry in delete_rules: | ||||||
|             document_cls, field_name = rule_entry |             document_cls, field_name = rule_entry | ||||||
|             rule = doc._meta['delete_rules'][rule_entry] |             rule = doc._meta['delete_rules'][rule_entry] | ||||||
|             if rule == DENY and document_cls.objects(**{field_name + '__in': self}).count() > 0: |             if rule == DENY and document_cls.objects(**{field_name + '__in': self}).count() > 0: | ||||||
| @@ -1334,12 +1389,14 @@ class QuerySet(object): | |||||||
|                         (document_cls.__name__, field_name) |                         (document_cls.__name__, field_name) | ||||||
|                 raise OperationError(msg) |                 raise OperationError(msg) | ||||||
|  |  | ||||||
|         for rule_entry in doc._meta['delete_rules']: |         for rule_entry in delete_rules: | ||||||
|             document_cls, field_name = rule_entry |             document_cls, field_name = rule_entry | ||||||
|             rule = doc._meta['delete_rules'][rule_entry] |             rule = doc._meta['delete_rules'][rule_entry] | ||||||
|             if rule == CASCADE: |             if rule == CASCADE: | ||||||
|                 ref_q = document_cls.objects(**{field_name + '__in': self}) |                 ref_q = document_cls.objects(**{field_name + '__in': self}) | ||||||
|                 if doc != document_cls or (doc == document_cls and ref_q.count() > 0): |                 ref_q_count = ref_q.count() | ||||||
|  |                 if (doc != document_cls and ref_q_count > 0 | ||||||
|  |                     or (doc == document_cls and ref_q_count > 0)): | ||||||
|                     ref_q.delete(safe=safe) |                     ref_q.delete(safe=safe) | ||||||
|             elif rule == NULLIFY: |             elif rule == NULLIFY: | ||||||
|                 document_cls.objects(**{field_name + '__in': self}).update( |                 document_cls.objects(**{field_name + '__in': self}).update( | ||||||
| @@ -1358,6 +1415,8 @@ class QuerySet(object): | |||||||
|         """ |         """ | ||||||
|         operators = ['set', 'unset', 'inc', 'dec', 'pop', 'push', 'push_all', |         operators = ['set', 'unset', 'inc', 'dec', 'pop', 'push', 'push_all', | ||||||
|                      'pull', 'pull_all', 'add_to_set'] |                      'pull', 'pull_all', 'add_to_set'] | ||||||
|  |         match_operators = ['ne', 'gt', 'gte', 'lt', 'lte', 'in', 'nin', 'mod', | ||||||
|  |                            'all', 'size', 'exists', 'not'] | ||||||
|  |  | ||||||
|         mongo_update = {} |         mongo_update = {} | ||||||
|         for key, value in update.items(): |         for key, value in update.items(): | ||||||
| @@ -1381,6 +1440,10 @@ class QuerySet(object): | |||||||
|                 elif op == 'add_to_set': |                 elif op == 'add_to_set': | ||||||
|                     op = op.replace('_to_set', 'ToSet') |                     op = op.replace('_to_set', 'ToSet') | ||||||
|  |  | ||||||
|  |             match = None | ||||||
|  |             if parts[-1] in match_operators: | ||||||
|  |                 match = parts.pop() | ||||||
|  |  | ||||||
|             if _doc_cls: |             if _doc_cls: | ||||||
|                 # Switch field names to proper names [set in Field(name='foo')] |                 # Switch field names to proper names [set in Field(name='foo')] | ||||||
|                 fields = QuerySet._lookup_field(_doc_cls, parts) |                 fields = QuerySet._lookup_field(_doc_cls, parts) | ||||||
| @@ -1414,16 +1477,22 @@ class QuerySet(object): | |||||||
|                     elif field.required or value is not None: |                     elif field.required or value is not None: | ||||||
|                         value = field.prepare_query_value(op, value) |                         value = field.prepare_query_value(op, value) | ||||||
|  |  | ||||||
|  |             if match: | ||||||
|  |                 match = '$' + match | ||||||
|  |                 value = {match: value} | ||||||
|  |  | ||||||
|             key = '.'.join(parts) |             key = '.'.join(parts) | ||||||
|  |  | ||||||
|             if not op: |             if not op: | ||||||
|                 raise InvalidQueryError("Updates must supply an operation eg: set__FIELD=value") |                 raise InvalidQueryError("Updates must supply an operation " | ||||||
|  |                                         "eg: set__FIELD=value") | ||||||
|  |  | ||||||
|             if 'pull' in op and '.' in key: |             if 'pull' in op and '.' in key: | ||||||
|                 # Dot operators don't work on pull operations |                 # Dot operators don't work on pull operations | ||||||
|                 # it uses nested dict syntax |                 # it uses nested dict syntax | ||||||
|                 if op == 'pullAll': |                 if op == 'pullAll': | ||||||
|                     raise InvalidQueryError("pullAll operations only support a single field depth") |                     raise InvalidQueryError("pullAll operations only support " | ||||||
|  |                                             "a single field depth") | ||||||
|  |  | ||||||
|                 parts.reverse() |                 parts.reverse() | ||||||
|                 for key in parts: |                 for key in parts: | ||||||
| @@ -1531,6 +1600,48 @@ class QuerySet(object): | |||||||
|  |  | ||||||
|         return tuple(data) |         return tuple(data) | ||||||
|  |  | ||||||
|  |     def _get_as_pymongo(self, row): | ||||||
|  |         # Extract which fields paths we should follow if .fields(...) was | ||||||
|  |         # used. If not, handle all fields. | ||||||
|  |         if not getattr(self, '__as_pymongo_fields', None): | ||||||
|  |             self.__as_pymongo_fields = [] | ||||||
|  |             for field in self._loaded_fields.fields - set(['_cls', '_id', '_types']): | ||||||
|  |                 self.__as_pymongo_fields.append(field) | ||||||
|  |                 while '.' in field: | ||||||
|  |                     field, _ = field.rsplit('.', 1) | ||||||
|  |                     self.__as_pymongo_fields.append(field) | ||||||
|  |  | ||||||
|  |         all_fields = not self.__as_pymongo_fields | ||||||
|  |  | ||||||
|  |         def clean(data, path=None): | ||||||
|  |             path = path or '' | ||||||
|  |  | ||||||
|  |             if isinstance(data, dict): | ||||||
|  |                 new_data = {} | ||||||
|  |                 for key, value in data.iteritems(): | ||||||
|  |                     new_path = '%s.%s' % (path, key) if path else key | ||||||
|  |                     if all_fields or new_path in self.__as_pymongo_fields: | ||||||
|  |                         new_data[key] = clean(value, path=new_path) | ||||||
|  |                 data = new_data | ||||||
|  |             elif isinstance(data, list): | ||||||
|  |                 data = [clean(d, path=path) for d in data] | ||||||
|  |             else: | ||||||
|  |                 if self._as_pymongo_coerce: | ||||||
|  |                     # If we need to coerce types, we need to determine the | ||||||
|  |                     # type of this field and use the corresponding .to_python(...) | ||||||
|  |                     from mongoengine.fields import EmbeddedDocumentField | ||||||
|  |                     obj = self._document | ||||||
|  |                     for chunk in path.split('.'): | ||||||
|  |                         obj = getattr(obj, chunk, None) | ||||||
|  |                         if obj is None: | ||||||
|  |                             break | ||||||
|  |                         elif isinstance(obj, EmbeddedDocumentField): | ||||||
|  |                             obj = obj.document_type | ||||||
|  |                     if obj and data is not None: | ||||||
|  |                         data = obj.to_python(data) | ||||||
|  |             return data | ||||||
|  |         return clean(row) | ||||||
|  |  | ||||||
|     def scalar(self, *fields): |     def scalar(self, *fields): | ||||||
|         """Instead of returning Document instances, return either a specific |         """Instead of returning Document instances, return either a specific | ||||||
|         value or a tuple of values in order. |         value or a tuple of values in order. | ||||||
| @@ -1553,6 +1664,16 @@ class QuerySet(object): | |||||||
|         """An alias for scalar""" |         """An alias for scalar""" | ||||||
|         return self.scalar(*fields) |         return self.scalar(*fields) | ||||||
|  |  | ||||||
|  |     def as_pymongo(self, coerce_types=False): | ||||||
|  |         """Instead of returning Document instances, return raw values from | ||||||
|  |         pymongo. | ||||||
|  |  | ||||||
|  |         :param coerce_type: Field types (if applicable) would be use to coerce types. | ||||||
|  |         """ | ||||||
|  |         self._as_pymongo = True | ||||||
|  |         self._as_pymongo_coerce = coerce_types | ||||||
|  |         return self | ||||||
|  |  | ||||||
|     def _sub_js_fields(self, code): |     def _sub_js_fields(self, code): | ||||||
|         """When fields are specified with [~fieldname] syntax, where |         """When fields are specified with [~fieldname] syntax, where | ||||||
|         *fieldname* is the Python name of a field, *fieldname* will be |         *fieldname* is the Python name of a field, *fieldname* will be | ||||||
| @@ -1869,10 +1990,16 @@ class QuerySet(object): | |||||||
|  |  | ||||||
|         .. versionadded:: 0.5 |         .. versionadded:: 0.5 | ||||||
|         """ |         """ | ||||||
|         from dereference import DeReference |  | ||||||
|         # Make select related work the same for querysets |         # Make select related work the same for querysets | ||||||
|         max_depth += 1 |         max_depth += 1 | ||||||
|         return DeReference()(self, max_depth=max_depth) |         return self._dereference(self, max_depth=max_depth) | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def _dereference(self): | ||||||
|  |         if not self.__dereference: | ||||||
|  |             from dereference import DeReference | ||||||
|  |             self.__dereference = DeReference()  # Cached | ||||||
|  |         return self.__dereference | ||||||
|  |  | ||||||
|  |  | ||||||
| class QuerySetManager(object): | class QuerySetManager(object): | ||||||
| @@ -1904,7 +2031,7 @@ class QuerySetManager(object): | |||||||
|             return self |             return self | ||||||
|  |  | ||||||
|         # owner is the document that contains the QuerySetManager |         # owner is the document that contains the QuerySetManager | ||||||
|         queryset_class = owner._meta['queryset_class'] or QuerySet |         queryset_class = owner._meta.get('queryset_class') or QuerySet | ||||||
|         queryset = queryset_class(owner, owner._get_collection()) |         queryset = queryset_class(owner, owner._get_collection()) | ||||||
|         if self.get_queryset: |         if self.get_queryset: | ||||||
|             arg_count = self.get_queryset.func_code.co_argcount |             arg_count = self.get_queryset.func_code.co_argcount | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ | |||||||
| %define srcname mongoengine | %define srcname mongoengine | ||||||
|  |  | ||||||
| Name:           python-%{srcname} | Name:           python-%{srcname} | ||||||
| Version:        0.6.20 | Version:        0.7.10 | ||||||
| Release:        1%{?dist} | Release:        1%{?dist} | ||||||
| Summary:        A Python Document-Object Mapper for working with MongoDB | Summary:        A Python Document-Object Mapper for working with MongoDB | ||||||
|  |  | ||||||
| @@ -51,4 +51,4 @@ rm -rf $RPM_BUILD_ROOT | |||||||
| # %{python_sitearch}/* | # %{python_sitearch}/* | ||||||
|  |  | ||||||
| %changelog | %changelog | ||||||
| * See: http://readthedocs.org/docs/mongoengine-odm/en/latest/changelog.html | * See: http://docs.mongoengine.org/en/latest/changelog.html | ||||||
| @@ -1,13 +1,11 @@ | |||||||
| [aliases] |  | ||||||
| test = nosetests |  | ||||||
|  |  | ||||||
| [nosetests] | [nosetests] | ||||||
| verbosity = 2 | verbosity = 3 | ||||||
| detailed-errors = 1 | detailed-errors = 1 | ||||||
| #with-coverage = 1 | #with-coverage = 1 | ||||||
| #cover-erase = 1 | #cover-erase = 1 | ||||||
| #cover-html = 1 | #cover-html = 1 | ||||||
| #cover-html-dir = ../htmlcov | #cover-html-dir = ../htmlcov | ||||||
| #cover-package = mongoengine | #cover-package = mongoengine | ||||||
|  | py3where = build | ||||||
| where = tests | where = tests | ||||||
| #tests = test_bugfix.py | #tests =  test_bugfix.py | ||||||
							
								
								
									
										49
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										49
									
								
								setup.py
									
									
									
									
									
								
							| @@ -1,27 +1,35 @@ | |||||||
| from setuptools import setup, find_packages |  | ||||||
| import os | import os | ||||||
|  | import sys | ||||||
|  | from setuptools import setup, find_packages | ||||||
|  |  | ||||||
| DESCRIPTION = "A Python Document-Object Mapper for working with MongoDB" | # Hack to silence atexit traceback in newer python versions | ||||||
|  | try: | ||||||
|  |     import multiprocessing | ||||||
|  | except ImportError: | ||||||
|  |     pass | ||||||
|  |  | ||||||
|  | DESCRIPTION = """MongoEngine is a Python Object-Document | ||||||
|  | Mapper for working with MongoDB.""" | ||||||
| LONG_DESCRIPTION = None | LONG_DESCRIPTION = None | ||||||
| try: | try: | ||||||
|     LONG_DESCRIPTION = open('README.rst').read() |     LONG_DESCRIPTION = open('README.rst').read() | ||||||
| except: | except: | ||||||
|     pass |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
| def get_version(version_tuple): | def get_version(version_tuple): | ||||||
|     version = '%s.%s' % (version_tuple[0], version_tuple[1]) |     if not isinstance(version_tuple[-1], int): | ||||||
|     if version_tuple[2]: |         return '.'.join(map(str, version_tuple[:-1])) + version_tuple[-1] | ||||||
|         version = '%s.%s' % (version, version_tuple[2]) |     return '.'.join(map(str, version_tuple)) | ||||||
|     return version |  | ||||||
|  |  | ||||||
| # Dirty hack to get version number from monogengine/__init__.py - we can't | # Dirty hack to get version number from monogengine/__init__.py - we can't | ||||||
| # import it as it depends on PyMongo and PyMongo isn't installed until this | # import it as it depends on PyMongo and PyMongo isn't installed until this | ||||||
| # file is read | # file is read | ||||||
| init = os.path.join(os.path.dirname(__file__), 'mongoengine', '__init__.py') | init = os.path.join(os.path.dirname(__file__), 'mongoengine', '__init__.py') | ||||||
| version_line = filter(lambda l: l.startswith('VERSION'), open(init))[0] | version_line = list(filter(lambda l: l.startswith('VERSION'), open(init)))[0] | ||||||
|  |  | ||||||
| VERSION = get_version(eval(version_line.split('=')[-1])) | VERSION = get_version(eval(version_line.split('=')[-1])) | ||||||
| print VERSION | print(VERSION) | ||||||
|  |  | ||||||
| CLASSIFIERS = [ | CLASSIFIERS = [ | ||||||
|     'Development Status :: 4 - Beta', |     'Development Status :: 4 - Beta', | ||||||
| @@ -29,18 +37,38 @@ CLASSIFIERS = [ | |||||||
|     'License :: OSI Approved :: MIT License', |     'License :: OSI Approved :: MIT License', | ||||||
|     'Operating System :: OS Independent', |     'Operating System :: OS Independent', | ||||||
|     'Programming Language :: Python', |     'Programming Language :: Python', | ||||||
|  |     "Programming Language :: Python :: 2", | ||||||
|  |     "Programming Language :: Python :: 2.5", | ||||||
|  |     "Programming Language :: Python :: 2.6", | ||||||
|  |     "Programming Language :: Python :: 2.7", | ||||||
|  |     "Programming Language :: Python :: 3", | ||||||
|  |     "Programming Language :: Python :: 3.1", | ||||||
|  |     "Programming Language :: Python :: 3.2", | ||||||
|  |     "Programming Language :: Python :: Implementation :: CPython", | ||||||
|     'Topic :: Database', |     'Topic :: Database', | ||||||
|     'Topic :: Software Development :: Libraries :: Python Modules', |     'Topic :: Software Development :: Libraries :: Python Modules', | ||||||
| ] | ] | ||||||
|  |  | ||||||
|  | extra_opts = {} | ||||||
|  | if sys.version_info[0] == 3: | ||||||
|  |     extra_opts['use_2to3'] = True | ||||||
|  |     extra_opts['tests_require'] = ['nose', 'coverage', 'blinker'] | ||||||
|  |     extra_opts['packages'] = find_packages(exclude=('tests',)) | ||||||
|  |     if "test" in sys.argv or "nosetests" in sys.argv: | ||||||
|  |         extra_opts['packages'].append("tests") | ||||||
|  |         extra_opts['package_data'] = {"tests": ["mongoengine.png"]} | ||||||
|  | else: | ||||||
|  |     extra_opts['tests_require'] = ['nose', 'coverage', 'blinker', 'django==1.4.2', 'PIL'] | ||||||
|  |     extra_opts['packages'] = find_packages(exclude=('tests',)) | ||||||
|  |  | ||||||
| setup(name='mongoengine', | setup(name='mongoengine', | ||||||
|       version=VERSION, |       version=VERSION, | ||||||
|       packages=find_packages(exclude=('tests',)), |  | ||||||
|       author='Harry Marr', |       author='Harry Marr', | ||||||
|       author_email='harry.marr@{nospam}gmail.com', |       author_email='harry.marr@{nospam}gmail.com', | ||||||
|       maintainer="Ross Lawley", |       maintainer="Ross Lawley", | ||||||
|       maintainer_email="ross.lawley@{nospam}gmail.com", |       maintainer_email="ross.lawley@{nospam}gmail.com", | ||||||
|       url='http://mongoengine.org/', |       url='http://mongoengine.org/', | ||||||
|  |       download_url='https://github.com/MongoEngine/mongoengine/tarball/master', | ||||||
|       license='MIT', |       license='MIT', | ||||||
|       include_package_data=True, |       include_package_data=True, | ||||||
|       description=DESCRIPTION, |       description=DESCRIPTION, | ||||||
| @@ -48,5 +76,6 @@ setup(name='mongoengine', | |||||||
|       platforms=['any'], |       platforms=['any'], | ||||||
|       classifiers=CLASSIFIERS, |       classifiers=CLASSIFIERS, | ||||||
|       install_requires=['pymongo'], |       install_requires=['pymongo'], | ||||||
|       tests_require=['nose', 'coverage', 'blinker', 'django>=1.3', 'PIL'] |       test_suite='nose.collector', | ||||||
|  |       **extra_opts | ||||||
| ) | ) | ||||||
|   | |||||||
							
								
								
									
										98
									
								
								tests/test_all_warnings.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								tests/test_all_warnings.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,98 @@ | |||||||
|  | import unittest | ||||||
|  | import warnings | ||||||
|  |  | ||||||
|  | from mongoengine import * | ||||||
|  | from mongoengine.tests import query_counter | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TestWarnings(unittest.TestCase): | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         conn = connect(db='mongoenginetest') | ||||||
|  |         self.warning_list = [] | ||||||
|  |         self.showwarning_default = warnings.showwarning | ||||||
|  |         warnings.showwarning = self.append_to_warning_list | ||||||
|  |  | ||||||
|  |     def append_to_warning_list(self, message, category, *args): | ||||||
|  |         self.warning_list.append({"message": message, | ||||||
|  |                                   "category": category}) | ||||||
|  |  | ||||||
|  |     def tearDown(self): | ||||||
|  |         # restore default handling of warnings | ||||||
|  |         warnings.showwarning = self.showwarning_default | ||||||
|  |  | ||||||
|  |     def test_allow_inheritance_future_warning(self): | ||||||
|  |         """Add FutureWarning for future allow_inhertiance default change. | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         class SimpleBase(Document): | ||||||
|  |             a = IntField() | ||||||
|  |  | ||||||
|  |         class InheritedClass(SimpleBase): | ||||||
|  |             b = IntField() | ||||||
|  |  | ||||||
|  |         InheritedClass() | ||||||
|  |         self.assertEqual(len(self.warning_list), 1) | ||||||
|  |         warning = self.warning_list[0] | ||||||
|  |         self.assertEqual(FutureWarning, warning["category"]) | ||||||
|  |         self.assertTrue("InheritedClass" in str(warning["message"])) | ||||||
|  |  | ||||||
|  |     def test_dbref_reference_field_future_warning(self): | ||||||
|  |  | ||||||
|  |         class Person(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             parent = ReferenceField('self') | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         p1 = Person() | ||||||
|  |         p1.parent = None | ||||||
|  |         p1.save() | ||||||
|  |  | ||||||
|  |         p2 = Person(name="Wilson Jr") | ||||||
|  |         p2.parent = p1 | ||||||
|  |         p2.save(cascade=False) | ||||||
|  |  | ||||||
|  |         self.assertTrue(len(self.warning_list) > 0) | ||||||
|  |         warning = self.warning_list[0] | ||||||
|  |         self.assertEqual(FutureWarning, warning["category"]) | ||||||
|  |         self.assertTrue("ReferenceFields will default to using ObjectId" | ||||||
|  |                         in str(warning["message"])) | ||||||
|  |  | ||||||
|  |     def test_document_save_cascade_future_warning(self): | ||||||
|  |  | ||||||
|  |         class Person(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             parent = ReferenceField('self') | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         p1 = Person(name="Wilson Snr") | ||||||
|  |         p1.parent = None | ||||||
|  |         p1.save() | ||||||
|  |  | ||||||
|  |         p2 = Person(name="Wilson Jr") | ||||||
|  |         p2.parent = p1 | ||||||
|  |         p2.parent.name = "Poppa Wilson" | ||||||
|  |         p2.save() | ||||||
|  |  | ||||||
|  |         self.assertTrue(len(self.warning_list) > 0) | ||||||
|  |         if len(self.warning_list) > 1: | ||||||
|  |             print self.warning_list | ||||||
|  |         warning = self.warning_list[0] | ||||||
|  |         self.assertEqual(FutureWarning, warning["category"]) | ||||||
|  |         self.assertTrue("Cascading saves will default to off in 0.8" | ||||||
|  |                         in str(warning["message"])) | ||||||
|  |  | ||||||
|  |     def test_document_collection_syntax_warning(self): | ||||||
|  |  | ||||||
|  |         class NonAbstractBase(Document): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         class InheritedDocumentFailTest(NonAbstractBase): | ||||||
|  |             meta = {'collection': 'fail'} | ||||||
|  |  | ||||||
|  |         warning = self.warning_list[0] | ||||||
|  |         self.assertEqual(SyntaxWarning, warning["category"]) | ||||||
|  |         self.assertEqual('non_abstract_base', | ||||||
|  |                          InheritedDocumentFailTest._get_collection_name()) | ||||||
| @@ -1,5 +1,9 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | from __future__ import with_statement | ||||||
| import unittest | import unittest | ||||||
|  |  | ||||||
|  | from bson import DBRef, ObjectId | ||||||
|  |  | ||||||
| from mongoengine import * | from mongoengine import * | ||||||
| from mongoengine.connection import get_db | from mongoengine.connection import get_db | ||||||
| from mongoengine.tests import query_counter | from mongoengine.tests import query_counter | ||||||
| @@ -39,6 +43,12 @@ class FieldTest(unittest.TestCase): | |||||||
|             group_obj = Group.objects.first() |             group_obj = Group.objects.first() | ||||||
|             self.assertEqual(q, 1) |             self.assertEqual(q, 1) | ||||||
|  |  | ||||||
|  |             len(group_obj._data['members']) | ||||||
|  |             self.assertEqual(q, 1) | ||||||
|  |  | ||||||
|  |             len(group_obj.members) | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|             [m for m in group_obj.members] |             [m for m in group_obj.members] | ||||||
|             self.assertEqual(q, 2) |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
| @@ -63,6 +73,132 @@ class FieldTest(unittest.TestCase): | |||||||
|         User.drop_collection() |         User.drop_collection() | ||||||
|         Group.drop_collection() |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_list_item_dereference_dref_false(self): | ||||||
|  |         """Ensure that DBRef items in ListFields are dereferenced. | ||||||
|  |         """ | ||||||
|  |         class User(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Group(Document): | ||||||
|  |             members = ListField(ReferenceField(User, dbref=False)) | ||||||
|  |  | ||||||
|  |         User.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |         for i in xrange(1, 51): | ||||||
|  |             user = User(name='user %s' % i) | ||||||
|  |             user.save() | ||||||
|  |  | ||||||
|  |         group = Group(members=User.objects) | ||||||
|  |         group.save() | ||||||
|  |         group.reload()  # Confirm reload works | ||||||
|  |  | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             group_obj = Group.objects.first() | ||||||
|  |             self.assertEqual(q, 1) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |         # Document select_related | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             group_obj = Group.objects.first().select_related() | ||||||
|  |  | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |         # Queryset select_related | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |             group_objs = Group.objects.select_related() | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |             for group_obj in group_objs: | ||||||
|  |                 [m for m in group_obj.members] | ||||||
|  |                 self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |         User.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_handle_old_style_references(self): | ||||||
|  |         """Ensure that DBRef items in ListFields are dereferenced. | ||||||
|  |         """ | ||||||
|  |         class User(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Group(Document): | ||||||
|  |             members = ListField(ReferenceField(User, dbref=True)) | ||||||
|  |  | ||||||
|  |         User.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |         for i in xrange(1, 26): | ||||||
|  |             user = User(name='user %s' % i) | ||||||
|  |             user.save() | ||||||
|  |  | ||||||
|  |         group = Group(members=User.objects) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         group = Group._get_collection().find_one() | ||||||
|  |  | ||||||
|  |         # Update the model to change the reference | ||||||
|  |         class Group(Document): | ||||||
|  |             members = ListField(ReferenceField(User, dbref=False)) | ||||||
|  |  | ||||||
|  |         group = Group.objects.first() | ||||||
|  |         group.members.append(User(name="String!").save()) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         group = Group.objects.first() | ||||||
|  |         self.assertEqual(group.members[0].name, 'user 1') | ||||||
|  |         self.assertEqual(group.members[-1].name, 'String!') | ||||||
|  |  | ||||||
|  |     def test_migrate_references(self): | ||||||
|  |         """Example of migrating ReferenceField storage | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         # Create some sample data | ||||||
|  |         class User(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Group(Document): | ||||||
|  |             author = ReferenceField(User, dbref=True) | ||||||
|  |             members = ListField(ReferenceField(User, dbref=True)) | ||||||
|  |  | ||||||
|  |         User.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |         user = User(name="Ross").save() | ||||||
|  |         group = Group(author=user, members=[user]).save() | ||||||
|  |  | ||||||
|  |         raw_data = Group._get_collection().find_one() | ||||||
|  |         self.assertTrue(isinstance(raw_data['author'], DBRef)) | ||||||
|  |         self.assertTrue(isinstance(raw_data['members'][0], DBRef)) | ||||||
|  |  | ||||||
|  |         # Migrate the model definition | ||||||
|  |         class Group(Document): | ||||||
|  |             author = ReferenceField(User, dbref=False) | ||||||
|  |             members = ListField(ReferenceField(User, dbref=False)) | ||||||
|  |  | ||||||
|  |         # Migrate the data | ||||||
|  |         for g in Group.objects(): | ||||||
|  |             # Explicitly mark as changed so resets | ||||||
|  |             g._mark_as_changed('author') | ||||||
|  |             g._mark_as_changed('members') | ||||||
|  |             g.save() | ||||||
|  |  | ||||||
|  |         group = Group.objects.first() | ||||||
|  |         self.assertEqual(group.author, user) | ||||||
|  |         self.assertEqual(group.members, [user]) | ||||||
|  |  | ||||||
|  |         raw_data = Group._get_collection().find_one() | ||||||
|  |         self.assertTrue(isinstance(raw_data['author'], ObjectId)) | ||||||
|  |         self.assertTrue(isinstance(raw_data['members'][0], ObjectId)) | ||||||
|  |  | ||||||
|     def test_recursive_reference(self): |     def test_recursive_reference(self): | ||||||
|         """Ensure that ReferenceFields can reference their own documents. |         """Ensure that ReferenceFields can reference their own documents. | ||||||
|         """ |         """ | ||||||
| @@ -109,10 +245,10 @@ class FieldTest(unittest.TestCase): | |||||||
|             peter = Employee.objects.with_id(peter.id).select_related() |             peter = Employee.objects.with_id(peter.id).select_related() | ||||||
|             self.assertEqual(q, 2) |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|             self.assertEquals(peter.boss, bill) |             self.assertEqual(peter.boss, bill) | ||||||
|             self.assertEqual(q, 2) |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|             self.assertEquals(peter.friends, friends) |             self.assertEqual(peter.friends, friends) | ||||||
|             self.assertEqual(q, 2) |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|         # Queryset select_related |         # Queryset select_related | ||||||
| @@ -123,10 +259,10 @@ class FieldTest(unittest.TestCase): | |||||||
|             self.assertEqual(q, 2) |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|             for employee in employees: |             for employee in employees: | ||||||
|                 self.assertEquals(employee.boss, bill) |                 self.assertEqual(employee.boss, bill) | ||||||
|                 self.assertEqual(q, 2) |                 self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|                 self.assertEquals(employee.friends, friends) |                 self.assertEqual(employee.friends, friends) | ||||||
|                 self.assertEqual(q, 2) |                 self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|     def test_circular_reference(self): |     def test_circular_reference(self): | ||||||
| @@ -160,7 +296,7 @@ class FieldTest(unittest.TestCase): | |||||||
|         daughter.relations.append(self_rel) |         daughter.relations.append(self_rel) | ||||||
|         daughter.save() |         daughter.save() | ||||||
|  |  | ||||||
|         self.assertEquals("[<Person: Mother>, <Person: Daughter>]", "%s" % Person.objects()) |         self.assertEqual("[<Person: Mother>, <Person: Daughter>]", "%s" % Person.objects()) | ||||||
|  |  | ||||||
|     def test_circular_reference_on_self(self): |     def test_circular_reference_on_self(self): | ||||||
|         """Ensure you can handle circular references |         """Ensure you can handle circular references | ||||||
| @@ -186,7 +322,7 @@ class FieldTest(unittest.TestCase): | |||||||
|         daughter.relations.append(daughter) |         daughter.relations.append(daughter) | ||||||
|         daughter.save() |         daughter.save() | ||||||
|  |  | ||||||
|         self.assertEquals("[<Person: Mother>, <Person: Daughter>]", "%s" % Person.objects()) |         self.assertEqual("[<Person: Mother>, <Person: Daughter>]", "%s" % Person.objects()) | ||||||
|  |  | ||||||
|     def test_circular_tree_reference(self): |     def test_circular_tree_reference(self): | ||||||
|         """Ensure you can handle circular references with more than one level |         """Ensure you can handle circular references with more than one level | ||||||
| @@ -228,7 +364,7 @@ class FieldTest(unittest.TestCase): | |||||||
|         anna.other.name = "Anna's friends" |         anna.other.name = "Anna's friends" | ||||||
|         anna.save() |         anna.save() | ||||||
|  |  | ||||||
|         self.assertEquals( |         self.assertEqual( | ||||||
|             "[<Person: Paul>, <Person: Maria>, <Person: Julia>, <Person: Anna>]", |             "[<Person: Paul>, <Person: Maria>, <Person: Julia>, <Person: Anna>]", | ||||||
|             "%s" % Person.objects() |             "%s" % Person.objects() | ||||||
|         ) |         ) | ||||||
| @@ -781,8 +917,8 @@ class FieldTest(unittest.TestCase): | |||||||
|         root.save() |         root.save() | ||||||
|  |  | ||||||
|         root = root.reload() |         root = root.reload() | ||||||
|         self.assertEquals(root.children, [company]) |         self.assertEqual(root.children, [company]) | ||||||
|         self.assertEquals(company.parents, [root]) |         self.assertEqual(company.parents, [root]) | ||||||
|  |  | ||||||
|     def test_dict_in_dbref_instance(self): |     def test_dict_in_dbref_instance(self): | ||||||
|  |  | ||||||
| @@ -808,8 +944,8 @@ class FieldTest(unittest.TestCase): | |||||||
|         room_101.save() |         room_101.save() | ||||||
|  |  | ||||||
|         room = Room.objects.first().select_related() |         room = Room.objects.first().select_related() | ||||||
|         self.assertEquals(room.staffs_with_position[0]['staff'], sarah) |         self.assertEqual(room.staffs_with_position[0]['staff'], sarah) | ||||||
|         self.assertEquals(room.staffs_with_position[1]['staff'], bob) |         self.assertEqual(room.staffs_with_position[1]['staff'], bob) | ||||||
|  |  | ||||||
|     def test_document_reload_no_inheritance(self): |     def test_document_reload_no_inheritance(self): | ||||||
|         class Foo(Document): |         class Foo(Document): | ||||||
| @@ -839,8 +975,8 @@ class FieldTest(unittest.TestCase): | |||||||
|         foo.save() |         foo.save() | ||||||
|         foo.reload() |         foo.reload() | ||||||
|  |  | ||||||
|         self.assertEquals(type(foo.bar), Bar) |         self.assertEqual(type(foo.bar), Bar) | ||||||
|         self.assertEquals(type(foo.baz), Baz) |         self.assertEqual(type(foo.baz), Baz) | ||||||
|  |  | ||||||
|     def test_list_lookup_not_checked_in_map(self): |     def test_list_lookup_not_checked_in_map(self): | ||||||
|         """Ensure we dereference list data correctly |         """Ensure we dereference list data correctly | ||||||
| @@ -863,3 +999,57 @@ class FieldTest(unittest.TestCase): | |||||||
|         msg = Message.objects.get(id=1) |         msg = Message.objects.get(id=1) | ||||||
|         self.assertEqual(0, msg.comments[0].id) |         self.assertEqual(0, msg.comments[0].id) | ||||||
|         self.assertEqual(1, msg.comments[1].id) |         self.assertEqual(1, msg.comments[1].id) | ||||||
|  |  | ||||||
|  |     def test_tuples_as_tuples(self): | ||||||
|  |         """ | ||||||
|  |         Ensure that tuples remain tuples when they are | ||||||
|  |         inside a ComplexBaseField | ||||||
|  |         """ | ||||||
|  |         from mongoengine.base import BaseField | ||||||
|  |         class EnumField(BaseField): | ||||||
|  |             def __init__(self, **kwargs): | ||||||
|  |                 super(EnumField,self).__init__(**kwargs) | ||||||
|  |  | ||||||
|  |             def to_mongo(self, value): | ||||||
|  |                 return value | ||||||
|  |  | ||||||
|  |             def to_python(self, value): | ||||||
|  |                 return tuple(value) | ||||||
|  |  | ||||||
|  |         class TestDoc(Document): | ||||||
|  |             items = ListField(EnumField()) | ||||||
|  |  | ||||||
|  |         TestDoc.drop_collection() | ||||||
|  |         tuples = [(100, 'Testing')] | ||||||
|  |         doc = TestDoc() | ||||||
|  |         doc.items = tuples | ||||||
|  |         doc.save() | ||||||
|  |         x = TestDoc.objects().get() | ||||||
|  |         self.assertTrue(x is not None) | ||||||
|  |         self.assertTrue(len(x.items) == 1) | ||||||
|  |         self.assertTrue(tuple(x.items[0]) in tuples) | ||||||
|  |         self.assertTrue(x.items[0] in tuples) | ||||||
|  |  | ||||||
|  |     def test_non_ascii_pk(self): | ||||||
|  |         """ | ||||||
|  |         Ensure that dbref conversion to string does not fail when | ||||||
|  |         non-ascii characters are used in primary key | ||||||
|  |         """ | ||||||
|  |         class Brand(Document): | ||||||
|  |             title = StringField(max_length=255, primary_key=True) | ||||||
|  |  | ||||||
|  |         class BrandGroup(Document): | ||||||
|  |             title = StringField(max_length=255, primary_key=True) | ||||||
|  |             brands = ListField(ReferenceField("Brand", dbref=True)) | ||||||
|  |  | ||||||
|  |         Brand.drop_collection() | ||||||
|  |         BrandGroup.drop_collection() | ||||||
|  |  | ||||||
|  |         brand1 = Brand(title="Moschino").save() | ||||||
|  |         brand2 = Brand(title=u"Денис Симачёв").save() | ||||||
|  |  | ||||||
|  |         BrandGroup(title="top_brands", brands=[brand1, brand2]).save() | ||||||
|  |         brand_groups = BrandGroup.objects().all() | ||||||
|  |  | ||||||
|  |         self.assertEqual(2, len([brand for bg in brand_groups for brand in bg.brands])) | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,24 +1,34 @@ | |||||||
| # -*- coding: utf-8 -*- | from __future__ import with_statement | ||||||
|  |  | ||||||
| import unittest | import unittest | ||||||
|  | from nose.plugins.skip import SkipTest | ||||||
|  | from mongoengine.python_support import PY3 | ||||||
| from mongoengine import * | from mongoengine import * | ||||||
| from mongoengine.django.shortcuts import get_document_or_404 |  | ||||||
|  |  | ||||||
| from django.http import Http404 | try: | ||||||
| from django.template import Context, Template |     from mongoengine.django.shortcuts import get_document_or_404 | ||||||
| from django.conf import settings |  | ||||||
| from django.core.paginator import Paginator |  | ||||||
|  |  | ||||||
| settings.configure() |     from django.http import Http404 | ||||||
|  |     from django.template import Context, Template | ||||||
|  |     from django.conf import settings | ||||||
|  |     from django.core.paginator import Paginator | ||||||
|  |  | ||||||
| from django.contrib.sessions.tests import SessionTestsMixin |     settings.configure() | ||||||
| from mongoengine.django.sessions import SessionStore, MongoSession |  | ||||||
|  |     from django.contrib.sessions.tests import SessionTestsMixin | ||||||
|  |     from mongoengine.django.sessions import SessionStore, MongoSession | ||||||
|  | except Exception, err: | ||||||
|  |     if PY3: | ||||||
|  |         SessionTestsMixin = type  # dummy value so no error | ||||||
|  |         SessionStore = None  # dummy value so no error | ||||||
|  |     else: | ||||||
|  |         raise err | ||||||
|  |  | ||||||
|  |  | ||||||
| class QuerySetTest(unittest.TestCase): | class QuerySetTest(unittest.TestCase): | ||||||
|  |  | ||||||
|     def setUp(self): |     def setUp(self): | ||||||
|  |         if PY3: | ||||||
|  |             raise SkipTest('django does not have Python 3 support') | ||||||
|         connect(db='mongoenginetest') |         connect(db='mongoenginetest') | ||||||
|  |  | ||||||
|         class Person(Document): |         class Person(Document): | ||||||
| @@ -99,6 +109,8 @@ class MongoDBSessionTest(SessionTestsMixin, unittest.TestCase): | |||||||
|     backend = SessionStore |     backend = SessionStore | ||||||
|  |  | ||||||
|     def setUp(self): |     def setUp(self): | ||||||
|  |         if PY3: | ||||||
|  |             raise SkipTest('django does not have Python 3 support') | ||||||
|         connect(db='mongoenginetest') |         connect(db='mongoenginetest') | ||||||
|         MongoSession.drop_collection() |         MongoSession.drop_collection() | ||||||
|         super(MongoDBSessionTest, self).setUp() |         super(MongoDBSessionTest, self).setUp() | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -25,14 +25,14 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         p.name = "James" |         p.name = "James" | ||||||
|         p.age = 34 |         p.age = 34 | ||||||
|  |  | ||||||
|         self.assertEquals(p.to_mongo(), |         self.assertEqual(p.to_mongo(), | ||||||
|             {"_types": ["Person"], "_cls": "Person", |             {"_types": ["Person"], "_cls": "Person", | ||||||
|              "name": "James", "age": 34} |              "name": "James", "age": 34} | ||||||
|         ) |         ) | ||||||
|  |  | ||||||
|         p.save() |         p.save() | ||||||
|  |  | ||||||
|         self.assertEquals(self.Person.objects.first().age, 34) |         self.assertEqual(self.Person.objects.first().age, 34) | ||||||
|  |  | ||||||
|         # Confirm no changes to self.Person |         # Confirm no changes to self.Person | ||||||
|         self.assertFalse(hasattr(self.Person, 'age')) |         self.assertFalse(hasattr(self.Person, 'age')) | ||||||
| @@ -40,11 +40,11 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|     def test_dynamic_document_delta(self): |     def test_dynamic_document_delta(self): | ||||||
|         """Ensures simple dynamic documents can delta correctly""" |         """Ensures simple dynamic documents can delta correctly""" | ||||||
|         p = self.Person(name="James", age=34) |         p = self.Person(name="James", age=34) | ||||||
|         self.assertEquals(p._delta(), ({'_types': ['Person'], 'age': 34, 'name': 'James', '_cls': 'Person'}, {})) |         self.assertEqual(p._delta(), ({'_types': ['Person'], 'age': 34, 'name': 'James', '_cls': 'Person'}, {})) | ||||||
|  |  | ||||||
|         p.doc = 123 |         p.doc = 123 | ||||||
|         del(p.doc) |         del(p.doc) | ||||||
|         self.assertEquals(p._delta(), ({'_types': ['Person'], 'age': 34, 'name': 'James', '_cls': 'Person'}, {'doc': 1})) |         self.assertEqual(p._delta(), ({'_types': ['Person'], 'age': 34, 'name': 'James', '_cls': 'Person'}, {'doc': 1})) | ||||||
|  |  | ||||||
|     def test_change_scope_of_variable(self): |     def test_change_scope_of_variable(self): | ||||||
|         """Test changing the scope of a dynamic field has no adverse effects""" |         """Test changing the scope of a dynamic field has no adverse effects""" | ||||||
| @@ -58,7 +58,7 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         p.save() |         p.save() | ||||||
|  |  | ||||||
|         p = self.Person.objects.get() |         p = self.Person.objects.get() | ||||||
|         self.assertEquals(p.misc, {'hello': 'world'}) |         self.assertEqual(p.misc, {'hello': 'world'}) | ||||||
|  |  | ||||||
|     def test_delete_dynamic_field(self): |     def test_delete_dynamic_field(self): | ||||||
|         """Test deleting a dynamic field works""" |         """Test deleting a dynamic field works""" | ||||||
| @@ -73,10 +73,10 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         p.save() |         p.save() | ||||||
|  |  | ||||||
|         p = self.Person.objects.get() |         p = self.Person.objects.get() | ||||||
|         self.assertEquals(p.misc, {'hello': 'world'}) |         self.assertEqual(p.misc, {'hello': 'world'}) | ||||||
|         collection = self.db[self.Person._get_collection_name()] |         collection = self.db[self.Person._get_collection_name()] | ||||||
|         obj = collection.find_one() |         obj = collection.find_one() | ||||||
|         self.assertEquals(sorted(obj.keys()), ['_cls', '_id', '_types', 'misc', 'name']) |         self.assertEqual(sorted(obj.keys()), ['_cls', '_id', '_types', 'misc', 'name']) | ||||||
|  |  | ||||||
|         del(p.misc) |         del(p.misc) | ||||||
|         p.save() |         p.save() | ||||||
| @@ -85,7 +85,7 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         self.assertFalse(hasattr(p, 'misc')) |         self.assertFalse(hasattr(p, 'misc')) | ||||||
|  |  | ||||||
|         obj = collection.find_one() |         obj = collection.find_one() | ||||||
|         self.assertEquals(sorted(obj.keys()), ['_cls', '_id', '_types', 'name']) |         self.assertEqual(sorted(obj.keys()), ['_cls', '_id', '_types', 'name']) | ||||||
|  |  | ||||||
|     def test_dynamic_document_queries(self): |     def test_dynamic_document_queries(self): | ||||||
|         """Ensure we can query dynamic fields""" |         """Ensure we can query dynamic fields""" | ||||||
| @@ -94,10 +94,10 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         p.age = 22 |         p.age = 22 | ||||||
|         p.save() |         p.save() | ||||||
|  |  | ||||||
|         self.assertEquals(1, self.Person.objects(age=22).count()) |         self.assertEqual(1, self.Person.objects(age=22).count()) | ||||||
|         p = self.Person.objects(age=22) |         p = self.Person.objects(age=22) | ||||||
|         p = p.get() |         p = p.get() | ||||||
|         self.assertEquals(22, p.age) |         self.assertEqual(22, p.age) | ||||||
|  |  | ||||||
|     def test_complex_dynamic_document_queries(self): |     def test_complex_dynamic_document_queries(self): | ||||||
|         class Person(DynamicDocument): |         class Person(DynamicDocument): | ||||||
| @@ -117,8 +117,8 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         p2.age = 10 |         p2.age = 10 | ||||||
|         p2.save() |         p2.save() | ||||||
|  |  | ||||||
|         self.assertEquals(Person.objects(age__icontains='ten').count(), 2) |         self.assertEqual(Person.objects(age__icontains='ten').count(), 2) | ||||||
|         self.assertEquals(Person.objects(age__gte=10).count(), 1) |         self.assertEqual(Person.objects(age__gte=10).count(), 1) | ||||||
|  |  | ||||||
|     def test_complex_data_lookups(self): |     def test_complex_data_lookups(self): | ||||||
|         """Ensure you can query dynamic document dynamic fields""" |         """Ensure you can query dynamic document dynamic fields""" | ||||||
| @@ -126,7 +126,7 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         p.misc = {'hello': 'world'} |         p.misc = {'hello': 'world'} | ||||||
|         p.save() |         p.save() | ||||||
|  |  | ||||||
|         self.assertEquals(1, self.Person.objects(misc__hello='world').count()) |         self.assertEqual(1, self.Person.objects(misc__hello='world').count()) | ||||||
|  |  | ||||||
|     def test_inheritance(self): |     def test_inheritance(self): | ||||||
|         """Ensure that dynamic document plays nice with inheritance""" |         """Ensure that dynamic document plays nice with inheritance""" | ||||||
| @@ -146,8 +146,8 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         joe_bloggs.age = 20 |         joe_bloggs.age = 20 | ||||||
|         joe_bloggs.save() |         joe_bloggs.save() | ||||||
|  |  | ||||||
|         self.assertEquals(1, self.Person.objects(age=20).count()) |         self.assertEqual(1, self.Person.objects(age=20).count()) | ||||||
|         self.assertEquals(1, Employee.objects(age=20).count()) |         self.assertEqual(1, Employee.objects(age=20).count()) | ||||||
|  |  | ||||||
|         joe_bloggs = self.Person.objects.first() |         joe_bloggs = self.Person.objects.first() | ||||||
|         self.assertTrue(isinstance(joe_bloggs, Employee)) |         self.assertTrue(isinstance(joe_bloggs, Employee)) | ||||||
| @@ -170,7 +170,7 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         embedded_1.list_field = ['1', 2, {'hello': 'world'}] |         embedded_1.list_field = ['1', 2, {'hello': 'world'}] | ||||||
|         doc.embedded_field = embedded_1 |         doc.embedded_field = embedded_1 | ||||||
|  |  | ||||||
|         self.assertEquals(doc.to_mongo(), {"_types": ['Doc'], "_cls": "Doc", |         self.assertEqual(doc.to_mongo(), {"_types": ['Doc'], "_cls": "Doc", | ||||||
|             "embedded_field": { |             "embedded_field": { | ||||||
|                 "_types": ['Embedded'], "_cls": "Embedded", |                 "_types": ['Embedded'], "_cls": "Embedded", | ||||||
|                 "string_field": "hello", |                 "string_field": "hello", | ||||||
| @@ -182,11 +182,11 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         doc.save() |         doc.save() | ||||||
|  |  | ||||||
|         doc = Doc.objects.first() |         doc = Doc.objects.first() | ||||||
|         self.assertEquals(doc.embedded_field.__class__, Embedded) |         self.assertEqual(doc.embedded_field.__class__, Embedded) | ||||||
|         self.assertEquals(doc.embedded_field.string_field, "hello") |         self.assertEqual(doc.embedded_field.string_field, "hello") | ||||||
|         self.assertEquals(doc.embedded_field.int_field, 1) |         self.assertEqual(doc.embedded_field.int_field, 1) | ||||||
|         self.assertEquals(doc.embedded_field.dict_field, {'hello': 'world'}) |         self.assertEqual(doc.embedded_field.dict_field, {'hello': 'world'}) | ||||||
|         self.assertEquals(doc.embedded_field.list_field, ['1', 2, {'hello': 'world'}]) |         self.assertEqual(doc.embedded_field.list_field, ['1', 2, {'hello': 'world'}]) | ||||||
|  |  | ||||||
|     def test_complex_embedded_documents(self): |     def test_complex_embedded_documents(self): | ||||||
|         """Test complex dynamic embedded documents setups""" |         """Test complex dynamic embedded documents setups""" | ||||||
| @@ -213,7 +213,7 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         embedded_1.list_field = ['1', 2, embedded_2] |         embedded_1.list_field = ['1', 2, embedded_2] | ||||||
|         doc.embedded_field = embedded_1 |         doc.embedded_field = embedded_1 | ||||||
|  |  | ||||||
|         self.assertEquals(doc.to_mongo(), {"_types": ['Doc'], "_cls": "Doc", |         self.assertEqual(doc.to_mongo(), {"_types": ['Doc'], "_cls": "Doc", | ||||||
|             "embedded_field": { |             "embedded_field": { | ||||||
|                 "_types": ['Embedded'], "_cls": "Embedded", |                 "_types": ['Embedded'], "_cls": "Embedded", | ||||||
|                 "string_field": "hello", |                 "string_field": "hello", | ||||||
| @@ -230,20 +230,20 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         }) |         }) | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc = Doc.objects.first() |         doc = Doc.objects.first() | ||||||
|         self.assertEquals(doc.embedded_field.__class__, Embedded) |         self.assertEqual(doc.embedded_field.__class__, Embedded) | ||||||
|         self.assertEquals(doc.embedded_field.string_field, "hello") |         self.assertEqual(doc.embedded_field.string_field, "hello") | ||||||
|         self.assertEquals(doc.embedded_field.int_field, 1) |         self.assertEqual(doc.embedded_field.int_field, 1) | ||||||
|         self.assertEquals(doc.embedded_field.dict_field, {'hello': 'world'}) |         self.assertEqual(doc.embedded_field.dict_field, {'hello': 'world'}) | ||||||
|         self.assertEquals(doc.embedded_field.list_field[0], '1') |         self.assertEqual(doc.embedded_field.list_field[0], '1') | ||||||
|         self.assertEquals(doc.embedded_field.list_field[1], 2) |         self.assertEqual(doc.embedded_field.list_field[1], 2) | ||||||
|  |  | ||||||
|         embedded_field = doc.embedded_field.list_field[2] |         embedded_field = doc.embedded_field.list_field[2] | ||||||
|  |  | ||||||
|         self.assertEquals(embedded_field.__class__, Embedded) |         self.assertEqual(embedded_field.__class__, Embedded) | ||||||
|         self.assertEquals(embedded_field.string_field, "hello") |         self.assertEqual(embedded_field.string_field, "hello") | ||||||
|         self.assertEquals(embedded_field.int_field, 1) |         self.assertEqual(embedded_field.int_field, 1) | ||||||
|         self.assertEquals(embedded_field.dict_field, {'hello': 'world'}) |         self.assertEqual(embedded_field.dict_field, {'hello': 'world'}) | ||||||
|         self.assertEquals(embedded_field.list_field, ['1', 2, {'hello': 'world'}]) |         self.assertEqual(embedded_field.list_field, ['1', 2, {'hello': 'world'}]) | ||||||
|  |  | ||||||
|     def test_delta_for_dynamic_documents(self): |     def test_delta_for_dynamic_documents(self): | ||||||
|         p = self.Person() |         p = self.Person() | ||||||
| @@ -252,18 +252,18 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         p.save() |         p.save() | ||||||
|  |  | ||||||
|         p.age = 24 |         p.age = 24 | ||||||
|         self.assertEquals(p.age, 24) |         self.assertEqual(p.age, 24) | ||||||
|         self.assertEquals(p._get_changed_fields(), ['age']) |         self.assertEqual(p._get_changed_fields(), ['age']) | ||||||
|         self.assertEquals(p._delta(), ({'age': 24}, {})) |         self.assertEqual(p._delta(), ({'age': 24}, {})) | ||||||
|  |  | ||||||
|         p = self.Person.objects(age=22).get() |         p = self.Person.objects(age=22).get() | ||||||
|         p.age = 24 |         p.age = 24 | ||||||
|         self.assertEquals(p.age, 24) |         self.assertEqual(p.age, 24) | ||||||
|         self.assertEquals(p._get_changed_fields(), ['age']) |         self.assertEqual(p._get_changed_fields(), ['age']) | ||||||
|         self.assertEquals(p._delta(), ({'age': 24}, {})) |         self.assertEqual(p._delta(), ({'age': 24}, {})) | ||||||
|  |  | ||||||
|         p.save() |         p.save() | ||||||
|         self.assertEquals(1, self.Person.objects(age=24).count()) |         self.assertEqual(1, self.Person.objects(age=24).count()) | ||||||
|  |  | ||||||
|     def test_delta(self): |     def test_delta(self): | ||||||
|  |  | ||||||
| @@ -275,40 +275,40 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         doc.save() |         doc.save() | ||||||
|  |  | ||||||
|         doc = Doc.objects.first() |         doc = Doc.objects.first() | ||||||
|         self.assertEquals(doc._get_changed_fields(), []) |         self.assertEqual(doc._get_changed_fields(), []) | ||||||
|         self.assertEquals(doc._delta(), ({}, {})) |         self.assertEqual(doc._delta(), ({}, {})) | ||||||
|  |  | ||||||
|         doc.string_field = 'hello' |         doc.string_field = 'hello' | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['string_field']) |         self.assertEqual(doc._get_changed_fields(), ['string_field']) | ||||||
|         self.assertEquals(doc._delta(), ({'string_field': 'hello'}, {})) |         self.assertEqual(doc._delta(), ({'string_field': 'hello'}, {})) | ||||||
|  |  | ||||||
|         doc._changed_fields = [] |         doc._changed_fields = [] | ||||||
|         doc.int_field = 1 |         doc.int_field = 1 | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['int_field']) |         self.assertEqual(doc._get_changed_fields(), ['int_field']) | ||||||
|         self.assertEquals(doc._delta(), ({'int_field': 1}, {})) |         self.assertEqual(doc._delta(), ({'int_field': 1}, {})) | ||||||
|  |  | ||||||
|         doc._changed_fields = [] |         doc._changed_fields = [] | ||||||
|         dict_value = {'hello': 'world', 'ping': 'pong'} |         dict_value = {'hello': 'world', 'ping': 'pong'} | ||||||
|         doc.dict_field = dict_value |         doc.dict_field = dict_value | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['dict_field']) |         self.assertEqual(doc._get_changed_fields(), ['dict_field']) | ||||||
|         self.assertEquals(doc._delta(), ({'dict_field': dict_value}, {})) |         self.assertEqual(doc._delta(), ({'dict_field': dict_value}, {})) | ||||||
|  |  | ||||||
|         doc._changed_fields = [] |         doc._changed_fields = [] | ||||||
|         list_value = ['1', 2, {'hello': 'world'}] |         list_value = ['1', 2, {'hello': 'world'}] | ||||||
|         doc.list_field = list_value |         doc.list_field = list_value | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['list_field']) |         self.assertEqual(doc._get_changed_fields(), ['list_field']) | ||||||
|         self.assertEquals(doc._delta(), ({'list_field': list_value}, {})) |         self.assertEqual(doc._delta(), ({'list_field': list_value}, {})) | ||||||
|  |  | ||||||
|         # Test unsetting |         # Test unsetting | ||||||
|         doc._changed_fields = [] |         doc._changed_fields = [] | ||||||
|         doc.dict_field = {} |         doc.dict_field = {} | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['dict_field']) |         self.assertEqual(doc._get_changed_fields(), ['dict_field']) | ||||||
|         self.assertEquals(doc._delta(), ({}, {'dict_field': 1})) |         self.assertEqual(doc._delta(), ({}, {'dict_field': 1})) | ||||||
|  |  | ||||||
|         doc._changed_fields = [] |         doc._changed_fields = [] | ||||||
|         doc.list_field = [] |         doc.list_field = [] | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['list_field']) |         self.assertEqual(doc._get_changed_fields(), ['list_field']) | ||||||
|         self.assertEquals(doc._delta(), ({}, {'list_field': 1})) |         self.assertEqual(doc._delta(), ({}, {'list_field': 1})) | ||||||
|  |  | ||||||
|     def test_delta_recursive(self): |     def test_delta_recursive(self): | ||||||
|         """Testing deltaing works with dynamic documents""" |         """Testing deltaing works with dynamic documents""" | ||||||
| @@ -323,8 +323,8 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         doc.save() |         doc.save() | ||||||
|  |  | ||||||
|         doc = Doc.objects.first() |         doc = Doc.objects.first() | ||||||
|         self.assertEquals(doc._get_changed_fields(), []) |         self.assertEqual(doc._get_changed_fields(), []) | ||||||
|         self.assertEquals(doc._delta(), ({}, {})) |         self.assertEqual(doc._delta(), ({}, {})) | ||||||
|  |  | ||||||
|         embedded_1 = Embedded() |         embedded_1 = Embedded() | ||||||
|         embedded_1.string_field = 'hello' |         embedded_1.string_field = 'hello' | ||||||
| @@ -333,7 +333,7 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         embedded_1.list_field = ['1', 2, {'hello': 'world'}] |         embedded_1.list_field = ['1', 2, {'hello': 'world'}] | ||||||
|         doc.embedded_field = embedded_1 |         doc.embedded_field = embedded_1 | ||||||
|  |  | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['embedded_field']) |         self.assertEqual(doc._get_changed_fields(), ['embedded_field']) | ||||||
|  |  | ||||||
|         embedded_delta = { |         embedded_delta = { | ||||||
|             'string_field': 'hello', |             'string_field': 'hello', | ||||||
| @@ -341,28 +341,28 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|             'dict_field': {'hello': 'world'}, |             'dict_field': {'hello': 'world'}, | ||||||
|             'list_field': ['1', 2, {'hello': 'world'}] |             'list_field': ['1', 2, {'hello': 'world'}] | ||||||
|         } |         } | ||||||
|         self.assertEquals(doc.embedded_field._delta(), (embedded_delta, {})) |         self.assertEqual(doc.embedded_field._delta(), (embedded_delta, {})) | ||||||
|         embedded_delta.update({ |         embedded_delta.update({ | ||||||
|             '_types': ['Embedded'], |             '_types': ['Embedded'], | ||||||
|             '_cls': 'Embedded', |             '_cls': 'Embedded', | ||||||
|         }) |         }) | ||||||
|         self.assertEquals(doc._delta(), ({'embedded_field': embedded_delta}, {})) |         self.assertEqual(doc._delta(), ({'embedded_field': embedded_delta}, {})) | ||||||
|  |  | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc.reload() |         doc.reload() | ||||||
|  |  | ||||||
|         doc.embedded_field.dict_field = {} |         doc.embedded_field.dict_field = {} | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['embedded_field.dict_field']) |         self.assertEqual(doc._get_changed_fields(), ['embedded_field.dict_field']) | ||||||
|         self.assertEquals(doc.embedded_field._delta(), ({}, {'dict_field': 1})) |         self.assertEqual(doc.embedded_field._delta(), ({}, {'dict_field': 1})) | ||||||
|  |  | ||||||
|         self.assertEquals(doc._delta(), ({}, {'embedded_field.dict_field': 1})) |         self.assertEqual(doc._delta(), ({}, {'embedded_field.dict_field': 1})) | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc.reload() |         doc.reload() | ||||||
|  |  | ||||||
|         doc.embedded_field.list_field = [] |         doc.embedded_field.list_field = [] | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['embedded_field.list_field']) |         self.assertEqual(doc._get_changed_fields(), ['embedded_field.list_field']) | ||||||
|         self.assertEquals(doc.embedded_field._delta(), ({}, {'list_field': 1})) |         self.assertEqual(doc.embedded_field._delta(), ({}, {'list_field': 1})) | ||||||
|         self.assertEquals(doc._delta(), ({}, {'embedded_field.list_field': 1})) |         self.assertEqual(doc._delta(), ({}, {'embedded_field.list_field': 1})) | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc.reload() |         doc.reload() | ||||||
|  |  | ||||||
| @@ -373,8 +373,8 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         embedded_2.list_field = ['1', 2, {'hello': 'world'}] |         embedded_2.list_field = ['1', 2, {'hello': 'world'}] | ||||||
|  |  | ||||||
|         doc.embedded_field.list_field = ['1', 2, embedded_2] |         doc.embedded_field.list_field = ['1', 2, embedded_2] | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['embedded_field.list_field']) |         self.assertEqual(doc._get_changed_fields(), ['embedded_field.list_field']) | ||||||
|         self.assertEquals(doc.embedded_field._delta(), ({ |         self.assertEqual(doc.embedded_field._delta(), ({ | ||||||
|             'list_field': ['1', 2, { |             'list_field': ['1', 2, { | ||||||
|                 '_cls': 'Embedded', |                 '_cls': 'Embedded', | ||||||
|                 '_types': ['Embedded'], |                 '_types': ['Embedded'], | ||||||
| @@ -385,7 +385,7 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|             }] |             }] | ||||||
|         }, {})) |         }, {})) | ||||||
|  |  | ||||||
|         self.assertEquals(doc._delta(), ({ |         self.assertEqual(doc._delta(), ({ | ||||||
|             'embedded_field.list_field': ['1', 2, { |             'embedded_field.list_field': ['1', 2, { | ||||||
|                 '_cls': 'Embedded', |                 '_cls': 'Embedded', | ||||||
|                  '_types': ['Embedded'], |                  '_types': ['Embedded'], | ||||||
| @@ -398,25 +398,25 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         doc.save() |         doc.save() | ||||||
|         doc.reload() |         doc.reload() | ||||||
|  |  | ||||||
|         self.assertEquals(doc.embedded_field.list_field[2]._changed_fields, []) |         self.assertEqual(doc.embedded_field.list_field[2]._changed_fields, []) | ||||||
|         self.assertEquals(doc.embedded_field.list_field[0], '1') |         self.assertEqual(doc.embedded_field.list_field[0], '1') | ||||||
|         self.assertEquals(doc.embedded_field.list_field[1], 2) |         self.assertEqual(doc.embedded_field.list_field[1], 2) | ||||||
|         for k in doc.embedded_field.list_field[2]._fields: |         for k in doc.embedded_field.list_field[2]._fields: | ||||||
|             self.assertEquals(doc.embedded_field.list_field[2][k], embedded_2[k]) |             self.assertEqual(doc.embedded_field.list_field[2][k], embedded_2[k]) | ||||||
|  |  | ||||||
|         doc.embedded_field.list_field[2].string_field = 'world' |         doc.embedded_field.list_field[2].string_field = 'world' | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['embedded_field.list_field.2.string_field']) |         self.assertEqual(doc._get_changed_fields(), ['embedded_field.list_field.2.string_field']) | ||||||
|         self.assertEquals(doc.embedded_field._delta(), ({'list_field.2.string_field': 'world'}, {})) |         self.assertEqual(doc.embedded_field._delta(), ({'list_field.2.string_field': 'world'}, {})) | ||||||
|         self.assertEquals(doc._delta(), ({'embedded_field.list_field.2.string_field': 'world'}, {})) |         self.assertEqual(doc._delta(), ({'embedded_field.list_field.2.string_field': 'world'}, {})) | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc.reload() |         doc.reload() | ||||||
|         self.assertEquals(doc.embedded_field.list_field[2].string_field, 'world') |         self.assertEqual(doc.embedded_field.list_field[2].string_field, 'world') | ||||||
|  |  | ||||||
|         # Test multiple assignments |         # Test multiple assignments | ||||||
|         doc.embedded_field.list_field[2].string_field = 'hello world' |         doc.embedded_field.list_field[2].string_field = 'hello world' | ||||||
|         doc.embedded_field.list_field[2] = doc.embedded_field.list_field[2] |         doc.embedded_field.list_field[2] = doc.embedded_field.list_field[2] | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['embedded_field.list_field']) |         self.assertEqual(doc._get_changed_fields(), ['embedded_field.list_field']) | ||||||
|         self.assertEquals(doc.embedded_field._delta(), ({ |         self.assertEqual(doc.embedded_field._delta(), ({ | ||||||
|             'list_field': ['1', 2, { |             'list_field': ['1', 2, { | ||||||
|             '_types': ['Embedded'], |             '_types': ['Embedded'], | ||||||
|             '_cls': 'Embedded', |             '_cls': 'Embedded', | ||||||
| @@ -424,7 +424,7 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|             'int_field': 1, |             'int_field': 1, | ||||||
|             'list_field': ['1', 2, {'hello': 'world'}], |             'list_field': ['1', 2, {'hello': 'world'}], | ||||||
|             'dict_field': {'hello': 'world'}}]}, {})) |             'dict_field': {'hello': 'world'}}]}, {})) | ||||||
|         self.assertEquals(doc._delta(), ({ |         self.assertEqual(doc._delta(), ({ | ||||||
|             'embedded_field.list_field': ['1', 2, { |             'embedded_field.list_field': ['1', 2, { | ||||||
|                 '_types': ['Embedded'], |                 '_types': ['Embedded'], | ||||||
|                 '_cls': 'Embedded', |                 '_cls': 'Embedded', | ||||||
| @@ -435,32 +435,32 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|             ]}, {})) |             ]}, {})) | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc.reload() |         doc.reload() | ||||||
|         self.assertEquals(doc.embedded_field.list_field[2].string_field, 'hello world') |         self.assertEqual(doc.embedded_field.list_field[2].string_field, 'hello world') | ||||||
|  |  | ||||||
|         # Test list native methods |         # Test list native methods | ||||||
|         doc.embedded_field.list_field[2].list_field.pop(0) |         doc.embedded_field.list_field[2].list_field.pop(0) | ||||||
|         self.assertEquals(doc._delta(), ({'embedded_field.list_field.2.list_field': [2, {'hello': 'world'}]}, {})) |         self.assertEqual(doc._delta(), ({'embedded_field.list_field.2.list_field': [2, {'hello': 'world'}]}, {})) | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc.reload() |         doc.reload() | ||||||
|  |  | ||||||
|         doc.embedded_field.list_field[2].list_field.append(1) |         doc.embedded_field.list_field[2].list_field.append(1) | ||||||
|         self.assertEquals(doc._delta(), ({'embedded_field.list_field.2.list_field': [2, {'hello': 'world'}, 1]}, {})) |         self.assertEqual(doc._delta(), ({'embedded_field.list_field.2.list_field': [2, {'hello': 'world'}, 1]}, {})) | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc.reload() |         doc.reload() | ||||||
|         self.assertEquals(doc.embedded_field.list_field[2].list_field, [2, {'hello': 'world'}, 1]) |         self.assertEqual(doc.embedded_field.list_field[2].list_field, [2, {'hello': 'world'}, 1]) | ||||||
|  |  | ||||||
|         doc.embedded_field.list_field[2].list_field.sort() |         doc.embedded_field.list_field[2].list_field.sort(key=str)# use str as a key to allow comparing uncomperable types | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc.reload() |         doc.reload() | ||||||
|         self.assertEquals(doc.embedded_field.list_field[2].list_field, [1, 2, {'hello': 'world'}]) |         self.assertEqual(doc.embedded_field.list_field[2].list_field, [1, 2, {'hello': 'world'}]) | ||||||
|  |  | ||||||
|         del(doc.embedded_field.list_field[2].list_field[2]['hello']) |         del(doc.embedded_field.list_field[2].list_field[2]['hello']) | ||||||
|         self.assertEquals(doc._delta(), ({'embedded_field.list_field.2.list_field': [1, 2, {}]}, {})) |         self.assertEqual(doc._delta(), ({'embedded_field.list_field.2.list_field': [1, 2, {}]}, {})) | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc.reload() |         doc.reload() | ||||||
|  |  | ||||||
|         del(doc.embedded_field.list_field[2].list_field) |         del(doc.embedded_field.list_field[2].list_field) | ||||||
|         self.assertEquals(doc._delta(), ({}, {'embedded_field.list_field.2.list_field': 1})) |         self.assertEqual(doc._delta(), ({}, {'embedded_field.list_field.2.list_field': 1})) | ||||||
|  |  | ||||||
|         doc.save() |         doc.save() | ||||||
|         doc.reload() |         doc.reload() | ||||||
| @@ -470,8 +470,8 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         doc.reload() |         doc.reload() | ||||||
|  |  | ||||||
|         doc.dict_field['embedded'].string_field = 'Hello World' |         doc.dict_field['embedded'].string_field = 'Hello World' | ||||||
|         self.assertEquals(doc._get_changed_fields(), ['dict_field.embedded.string_field']) |         self.assertEqual(doc._get_changed_fields(), ['dict_field.embedded.string_field']) | ||||||
|         self.assertEquals(doc._delta(), ({'dict_field.embedded.string_field': 'Hello World'}, {})) |         self.assertEqual(doc._delta(), ({'dict_field.embedded.string_field': 'Hello World'}, {})) | ||||||
|  |  | ||||||
|     def test_indexes(self): |     def test_indexes(self): | ||||||
|         """Ensure that indexes are used when meta[indexes] is specified. |         """Ensure that indexes are used when meta[indexes] is specified. | ||||||
| @@ -500,3 +500,34 @@ class DynamicDocTest(unittest.TestCase): | |||||||
|         self.assertTrue([('_types', 1), ('category', 1), ('date', -1)] |         self.assertTrue([('_types', 1), ('category', 1), ('date', -1)] | ||||||
|                         in info) |                         in info) | ||||||
|         self.assertTrue([('_types', 1), ('date', -1)] in info) |         self.assertTrue([('_types', 1), ('date', -1)] in info) | ||||||
|  |  | ||||||
|  |     def test_dynamic_and_embedded(self): | ||||||
|  |         """Ensure embedded documents play nicely""" | ||||||
|  |  | ||||||
|  |         class Address(EmbeddedDocument): | ||||||
|  |             city = StringField() | ||||||
|  |  | ||||||
|  |         class Person(DynamicDocument): | ||||||
|  |             name = StringField() | ||||||
|  |             meta = {'allow_inheritance': True} | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         Person(name="Ross", address=Address(city="London")).save() | ||||||
|  |  | ||||||
|  |         person = Person.objects.first() | ||||||
|  |         person.address.city = "Lundenne" | ||||||
|  |         person.save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(Person.objects.first().address.city, "Lundenne") | ||||||
|  |  | ||||||
|  |         person = Person.objects.first() | ||||||
|  |         person.address = Address(city="Londinium") | ||||||
|  |         person.save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(Person.objects.first().address.city, "Londinium") | ||||||
|  |  | ||||||
|  |         person = Person.objects.first() | ||||||
|  |         person.age = 35 | ||||||
|  |         person.save() | ||||||
|  |         self.assertEqual(Person.objects.first().age, 35) | ||||||
|   | |||||||
| @@ -1,21 +1,24 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | from __future__ import with_statement | ||||||
| import datetime | import datetime | ||||||
| import os | import os | ||||||
| import unittest | import unittest | ||||||
| import uuid | import uuid | ||||||
| import StringIO |  | ||||||
| import tempfile | import tempfile | ||||||
| import gridfs |  | ||||||
|  |  | ||||||
| from bson import Binary |  | ||||||
| from decimal import Decimal | from decimal import Decimal | ||||||
|  |  | ||||||
|  | from bson import Binary, DBRef, ObjectId | ||||||
|  | import gridfs | ||||||
|  |  | ||||||
|  | from nose.plugins.skip import SkipTest | ||||||
| from mongoengine import * | from mongoengine import * | ||||||
| from mongoengine.connection import get_db | from mongoengine.connection import get_db | ||||||
| from mongoengine.base import _document_registry, NotRegistered | from mongoengine.base import _document_registry, NotRegistered | ||||||
|  | from mongoengine.python_support import PY3, b, StringIO, bin_type | ||||||
|  |  | ||||||
| TEST_IMAGE_PATH = os.path.join(os.path.dirname(__file__), 'mongoengine.png') | TEST_IMAGE_PATH = os.path.join(os.path.dirname(__file__), 'mongoengine.png') | ||||||
|  |  | ||||||
|  |  | ||||||
| class FieldTest(unittest.TestCase): | class FieldTest(unittest.TestCase): | ||||||
|  |  | ||||||
|     def setUp(self): |     def setUp(self): | ||||||
| @@ -124,7 +127,7 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertEqual(ret.int_fld, None) |         self.assertEqual(ret.int_fld, None) | ||||||
|         self.assertEqual(ret.flt_fld, None) |         self.assertEqual(ret.flt_fld, None) | ||||||
|         # Return current time if retrived value is None. |         # Return current time if retrived value is None. | ||||||
|         self.assert_(isinstance(ret.comp_dt_fld, datetime.datetime)) |         self.assertTrue(isinstance(ret.comp_dt_fld, datetime.datetime)) | ||||||
|  |  | ||||||
|         self.assertRaises(ValidationError, ret.validate) |         self.assertRaises(ValidationError, ret.validate) | ||||||
|  |  | ||||||
| @@ -141,6 +144,17 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertEqual(1, TestDocument.objects(int_fld__ne=None).count()) |         self.assertEqual(1, TestDocument.objects(int_fld__ne=None).count()) | ||||||
|         self.assertEqual(1, TestDocument.objects(float_fld__ne=None).count()) |         self.assertEqual(1, TestDocument.objects(float_fld__ne=None).count()) | ||||||
|  |  | ||||||
|  |     def test_long_ne_operator(self): | ||||||
|  |         class TestDocument(Document): | ||||||
|  |             long_fld = LongField() | ||||||
|  |  | ||||||
|  |         TestDocument.drop_collection() | ||||||
|  |  | ||||||
|  |         TestDocument(long_fld=None).save() | ||||||
|  |         TestDocument(long_fld=1).save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(1, TestDocument.objects(long_fld__ne=None).count()) | ||||||
|  |  | ||||||
|     def test_object_id_validation(self): |     def test_object_id_validation(self): | ||||||
|         """Ensure that invalid values cannot be assigned to string fields. |         """Ensure that invalid values cannot be assigned to string fields. | ||||||
|         """ |         """ | ||||||
| @@ -214,6 +228,23 @@ class FieldTest(unittest.TestCase): | |||||||
|         person.age = 'ten' |         person.age = 'ten' | ||||||
|         self.assertRaises(ValidationError, person.validate) |         self.assertRaises(ValidationError, person.validate) | ||||||
|  |  | ||||||
|  |     def test_long_validation(self): | ||||||
|  |         """Ensure that invalid values cannot be assigned to long fields. | ||||||
|  |         """ | ||||||
|  |         class TestDocument(Document): | ||||||
|  |             value = LongField(min_value=0, max_value=110) | ||||||
|  |  | ||||||
|  |         doc = TestDocument() | ||||||
|  |         doc.value = 50 | ||||||
|  |         doc.validate() | ||||||
|  |  | ||||||
|  |         doc.value = -1 | ||||||
|  |         self.assertRaises(ValidationError, doc.validate) | ||||||
|  |         doc.age = 120 | ||||||
|  |         self.assertRaises(ValidationError, doc.validate) | ||||||
|  |         doc.age = 'ten' | ||||||
|  |         self.assertRaises(ValidationError, doc.validate) | ||||||
|  |  | ||||||
|     def test_float_validation(self): |     def test_float_validation(self): | ||||||
|         """Ensure that invalid values cannot be assigned to float fields. |         """Ensure that invalid values cannot be assigned to float fields. | ||||||
|         """ |         """ | ||||||
| @@ -358,7 +389,7 @@ class FieldTest(unittest.TestCase): | |||||||
|         log.date = datetime.date.today() |         log.date = datetime.date.today() | ||||||
|         log.save() |         log.save() | ||||||
|         log.reload() |         log.reload() | ||||||
|         self.assertEquals(log.date.date(), datetime.date.today()) |         self.assertEqual(log.date.date(), datetime.date.today()) | ||||||
|  |  | ||||||
|         LogEntry.drop_collection() |         LogEntry.drop_collection() | ||||||
|  |  | ||||||
| @@ -369,8 +400,8 @@ class FieldTest(unittest.TestCase): | |||||||
|         log.date = d1 |         log.date = d1 | ||||||
|         log.save() |         log.save() | ||||||
|         log.reload() |         log.reload() | ||||||
|         self.assertNotEquals(log.date, d1) |         self.assertNotEqual(log.date, d1) | ||||||
|         self.assertEquals(log.date, d2) |         self.assertEqual(log.date, d2) | ||||||
|  |  | ||||||
|         # Post UTC - microseconds are rounded (down) nearest millisecond |         # Post UTC - microseconds are rounded (down) nearest millisecond | ||||||
|         d1 = datetime.datetime(1970, 01, 01, 00, 00, 01, 9999) |         d1 = datetime.datetime(1970, 01, 01, 00, 00, 01, 9999) | ||||||
| @@ -378,17 +409,19 @@ class FieldTest(unittest.TestCase): | |||||||
|         log.date = d1 |         log.date = d1 | ||||||
|         log.save() |         log.save() | ||||||
|         log.reload() |         log.reload() | ||||||
|         self.assertNotEquals(log.date, d1) |         self.assertNotEqual(log.date, d1) | ||||||
|         self.assertEquals(log.date, d2) |         self.assertEqual(log.date, d2) | ||||||
|  |  | ||||||
|         # Pre UTC dates microseconds below 1000 are dropped |         if not PY3: | ||||||
|         d1 = datetime.datetime(1969, 12, 31, 23, 59, 59, 999) |             # Pre UTC dates microseconds below 1000 are dropped | ||||||
|         d2 = datetime.datetime(1969, 12, 31, 23, 59, 59) |             # This does not seem to be true in PY3 | ||||||
|         log.date = d1 |             d1 = datetime.datetime(1969, 12, 31, 23, 59, 59, 999) | ||||||
|         log.save() |             d2 = datetime.datetime(1969, 12, 31, 23, 59, 59) | ||||||
|         log.reload() |             log.date = d1 | ||||||
|         self.assertNotEquals(log.date, d1) |             log.save() | ||||||
|         self.assertEquals(log.date, d2) |             log.reload() | ||||||
|  |             self.assertNotEqual(log.date, d1) | ||||||
|  |             self.assertEqual(log.date, d2) | ||||||
|  |  | ||||||
|         LogEntry.drop_collection() |         LogEntry.drop_collection() | ||||||
|  |  | ||||||
| @@ -407,21 +440,21 @@ class FieldTest(unittest.TestCase): | |||||||
|         log.date = d1 |         log.date = d1 | ||||||
|         log.save() |         log.save() | ||||||
|         log.reload() |         log.reload() | ||||||
|         self.assertEquals(log.date, d1) |         self.assertEqual(log.date, d1) | ||||||
|  |  | ||||||
|         # Post UTC - microseconds are rounded (down) nearest millisecond - with default datetimefields |         # Post UTC - microseconds are rounded (down) nearest millisecond - with default datetimefields | ||||||
|         d1 = datetime.datetime(1970, 01, 01, 00, 00, 01, 9999) |         d1 = datetime.datetime(1970, 01, 01, 00, 00, 01, 9999) | ||||||
|         log.date = d1 |         log.date = d1 | ||||||
|         log.save() |         log.save() | ||||||
|         log.reload() |         log.reload() | ||||||
|         self.assertEquals(log.date, d1) |         self.assertEqual(log.date, d1) | ||||||
|  |  | ||||||
|         # Pre UTC dates microseconds below 1000 are dropped - with default datetimefields |         # Pre UTC dates microseconds below 1000 are dropped - with default datetimefields | ||||||
|         d1 = datetime.datetime(1969, 12, 31, 23, 59, 59, 999) |         d1 = datetime.datetime(1969, 12, 31, 23, 59, 59, 999) | ||||||
|         log.date = d1 |         log.date = d1 | ||||||
|         log.save() |         log.save() | ||||||
|         log.reload() |         log.reload() | ||||||
|         self.assertEquals(log.date, d1) |         self.assertEqual(log.date, d1) | ||||||
|  |  | ||||||
|         # Pre UTC microseconds above 1000 is wonky - with default datetimefields |         # Pre UTC microseconds above 1000 is wonky - with default datetimefields | ||||||
|         # log.date has an invalid microsecond value so I can't construct |         # log.date has an invalid microsecond value so I can't construct | ||||||
| @@ -431,7 +464,7 @@ class FieldTest(unittest.TestCase): | |||||||
|             log.date = d1 |             log.date = d1 | ||||||
|             log.save() |             log.save() | ||||||
|             log.reload() |             log.reload() | ||||||
|             self.assertEquals(log.date, d1) |             self.assertEqual(log.date, d1) | ||||||
|             log1 = LogEntry.objects.get(date=d1) |             log1 = LogEntry.objects.get(date=d1) | ||||||
|             self.assertEqual(log, log1) |             self.assertEqual(log, log1) | ||||||
|  |  | ||||||
| @@ -452,7 +485,7 @@ class FieldTest(unittest.TestCase): | |||||||
|         log.save() |         log.save() | ||||||
|  |  | ||||||
|         log1 = LogEntry.objects.get(date=d1) |         log1 = LogEntry.objects.get(date=d1) | ||||||
|         self.assertEquals(log, log1) |         self.assertEqual(log, log1) | ||||||
|  |  | ||||||
|         LogEntry.drop_collection() |         LogEntry.drop_collection() | ||||||
|  |  | ||||||
| @@ -640,13 +673,13 @@ class FieldTest(unittest.TestCase): | |||||||
|         post.info = [{'test': 3}] |         post.info = [{'test': 3}] | ||||||
|         post.save() |         post.save() | ||||||
|  |  | ||||||
|         self.assertEquals(BlogPost.objects.count(), 3) |         self.assertEqual(BlogPost.objects.count(), 3) | ||||||
|         self.assertEquals(BlogPost.objects.filter(info__exact='test').count(), 1) |         self.assertEqual(BlogPost.objects.filter(info__exact='test').count(), 1) | ||||||
|         self.assertEquals(BlogPost.objects.filter(info__0__test='test').count(), 1) |         self.assertEqual(BlogPost.objects.filter(info__0__test='test').count(), 1) | ||||||
|  |  | ||||||
|         # Confirm handles non strings or non existing keys |         # Confirm handles non strings or non existing keys | ||||||
|         self.assertEquals(BlogPost.objects.filter(info__0__test__exact='5').count(), 0) |         self.assertEqual(BlogPost.objects.filter(info__0__test__exact='5').count(), 0) | ||||||
|         self.assertEquals(BlogPost.objects.filter(info__100__test__exact='test').count(), 0) |         self.assertEqual(BlogPost.objects.filter(info__100__test__exact='test').count(), 0) | ||||||
|         BlogPost.drop_collection() |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
|     def test_list_field_passed_in_value(self): |     def test_list_field_passed_in_value(self): | ||||||
| @@ -661,7 +694,7 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         foo = Foo(bars=[]) |         foo = Foo(bars=[]) | ||||||
|         foo.bars.append(bar) |         foo.bars.append(bar) | ||||||
|         self.assertEquals(repr(foo.bars), '[<Bar: Bar object>]') |         self.assertEqual(repr(foo.bars), '[<Bar: Bar object>]') | ||||||
|  |  | ||||||
|  |  | ||||||
|     def test_list_field_strict(self): |     def test_list_field_strict(self): | ||||||
| @@ -746,20 +779,20 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertTrue(isinstance(e2.mapping[1], IntegerSetting)) |         self.assertTrue(isinstance(e2.mapping[1], IntegerSetting)) | ||||||
|  |  | ||||||
|         # Test querying |         # Test querying | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__1__value=42).count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__1__value=42).count(), 1) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__2__number=1).count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__2__number=1).count(), 1) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__2__complex__value=42).count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__2__complex__value=42).count(), 1) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__2__list__0__value=42).count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__2__list__0__value=42).count(), 1) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__2__list__1__value='foo').count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__2__list__1__value='foo').count(), 1) | ||||||
|  |  | ||||||
|         # Confirm can update |         # Confirm can update | ||||||
|         Simple.objects().update(set__mapping__1=IntegerSetting(value=10)) |         Simple.objects().update(set__mapping__1=IntegerSetting(value=10)) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__1__value=10).count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__1__value=10).count(), 1) | ||||||
|  |  | ||||||
|         Simple.objects().update( |         Simple.objects().update( | ||||||
|             set__mapping__2__list__1=StringSetting(value='Boo')) |             set__mapping__2__list__1=StringSetting(value='Boo')) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__2__list__1__value='foo').count(), 0) |         self.assertEqual(Simple.objects.filter(mapping__2__list__1__value='foo').count(), 0) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__2__list__1__value='Boo').count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__2__list__1__value='Boo').count(), 1) | ||||||
|  |  | ||||||
|         Simple.drop_collection() |         Simple.drop_collection() | ||||||
|  |  | ||||||
| @@ -798,19 +831,19 @@ class FieldTest(unittest.TestCase): | |||||||
|         post.info = {'details': {'test': 3}} |         post.info = {'details': {'test': 3}} | ||||||
|         post.save() |         post.save() | ||||||
|  |  | ||||||
|         self.assertEquals(BlogPost.objects.count(), 3) |         self.assertEqual(BlogPost.objects.count(), 3) | ||||||
|         self.assertEquals(BlogPost.objects.filter(info__title__exact='test').count(), 1) |         self.assertEqual(BlogPost.objects.filter(info__title__exact='test').count(), 1) | ||||||
|         self.assertEquals(BlogPost.objects.filter(info__details__test__exact='test').count(), 1) |         self.assertEqual(BlogPost.objects.filter(info__details__test__exact='test').count(), 1) | ||||||
|  |  | ||||||
|         # Confirm handles non strings or non existing keys |         # Confirm handles non strings or non existing keys | ||||||
|         self.assertEquals(BlogPost.objects.filter(info__details__test__exact=5).count(), 0) |         self.assertEqual(BlogPost.objects.filter(info__details__test__exact=5).count(), 0) | ||||||
|         self.assertEquals(BlogPost.objects.filter(info__made_up__test__exact='test').count(), 0) |         self.assertEqual(BlogPost.objects.filter(info__made_up__test__exact='test').count(), 0) | ||||||
|  |  | ||||||
|         post = BlogPost.objects.create(info={'title': 'original'}) |         post = BlogPost.objects.create(info={'title': 'original'}) | ||||||
|         post.info.update({'title': 'updated'}) |         post.info.update({'title': 'updated'}) | ||||||
|         post.save() |         post.save() | ||||||
|         post.reload() |         post.reload() | ||||||
|         self.assertEquals('updated', post.info['title']) |         self.assertEqual('updated', post.info['title']) | ||||||
|  |  | ||||||
|         BlogPost.drop_collection() |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
| @@ -863,19 +896,19 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertTrue(isinstance(e2.mapping['someint'], IntegerSetting)) |         self.assertTrue(isinstance(e2.mapping['someint'], IntegerSetting)) | ||||||
|  |  | ||||||
|         # Test querying |         # Test querying | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__someint__value=42).count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__someint__value=42).count(), 1) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__nested_dict__number=1).count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__nested_dict__number=1).count(), 1) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__nested_dict__complex__value=42).count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__nested_dict__complex__value=42).count(), 1) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__nested_dict__list__0__value=42).count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__nested_dict__list__0__value=42).count(), 1) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__nested_dict__list__1__value='foo').count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__nested_dict__list__1__value='foo').count(), 1) | ||||||
|  |  | ||||||
|         # Confirm can update |         # Confirm can update | ||||||
|         Simple.objects().update( |         Simple.objects().update( | ||||||
|             set__mapping={"someint": IntegerSetting(value=10)}) |             set__mapping={"someint": IntegerSetting(value=10)}) | ||||||
|         Simple.objects().update( |         Simple.objects().update( | ||||||
|             set__mapping__nested_dict__list__1=StringSetting(value='Boo')) |             set__mapping__nested_dict__list__1=StringSetting(value='Boo')) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__nested_dict__list__1__value='foo').count(), 0) |         self.assertEqual(Simple.objects.filter(mapping__nested_dict__list__1__value='foo').count(), 0) | ||||||
|         self.assertEquals(Simple.objects.filter(mapping__nested_dict__list__1__value='Boo').count(), 1) |         self.assertEqual(Simple.objects.filter(mapping__nested_dict__list__1__value='Boo').count(), 1) | ||||||
|  |  | ||||||
|         Simple.drop_collection() |         Simple.drop_collection() | ||||||
|  |  | ||||||
| @@ -960,6 +993,24 @@ class FieldTest(unittest.TestCase): | |||||||
|         doc = self.db.test.find_one() |         doc = self.db.test.find_one() | ||||||
|         self.assertEqual(doc['x']['DICTIONARY_KEY']['i'], 2) |         self.assertEqual(doc['x']['DICTIONARY_KEY']['i'], 2) | ||||||
|  |  | ||||||
|  |     def test_mapfield_numerical_index(self): | ||||||
|  |         """Ensure that MapField accept numeric strings as indexes.""" | ||||||
|  |         class Embedded(EmbeddedDocument): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Test(Document): | ||||||
|  |             my_map = MapField(EmbeddedDocumentField(Embedded)) | ||||||
|  |  | ||||||
|  |         Test.drop_collection() | ||||||
|  |  | ||||||
|  |         test = Test() | ||||||
|  |         test.my_map['1'] = Embedded(name='test') | ||||||
|  |         test.save() | ||||||
|  |         test.my_map['1'].name = 'test updated' | ||||||
|  |         test.save() | ||||||
|  |  | ||||||
|  |         Test.drop_collection() | ||||||
|  |  | ||||||
|     def test_map_field_lookup(self): |     def test_map_field_lookup(self): | ||||||
|         """Ensure MapField lookups succeed on Fields without a lookup method""" |         """Ensure MapField lookups succeed on Fields without a lookup method""" | ||||||
|  |  | ||||||
| @@ -1082,6 +1133,52 @@ class FieldTest(unittest.TestCase): | |||||||
|         User.drop_collection() |         User.drop_collection() | ||||||
|         BlogPost.drop_collection() |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_dbref_reference_fields(self): | ||||||
|  |  | ||||||
|  |         class Person(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             parent = ReferenceField('self', dbref=True) | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         p1 = Person(name="John").save() | ||||||
|  |         Person(name="Ross", parent=p1).save() | ||||||
|  |  | ||||||
|  |         col = Person._get_collection() | ||||||
|  |         data = col.find_one({'name': 'Ross'}) | ||||||
|  |         self.assertEqual(data['parent'], DBRef('person', p1.pk)) | ||||||
|  |  | ||||||
|  |         p = Person.objects.get(name="Ross") | ||||||
|  |         self.assertEqual(p.parent, p1) | ||||||
|  |  | ||||||
|  |     def test_dbref_to_mongo(self): | ||||||
|  |         class Person(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             parent = ReferenceField('self', dbref=False) | ||||||
|  |  | ||||||
|  |         p1 = Person._from_son({'name': "Yakxxx", | ||||||
|  |                                'parent': "50a234ea469ac1eda42d347d"}) | ||||||
|  |         mongoed = p1.to_mongo() | ||||||
|  |         self.assertTrue(isinstance(mongoed['parent'], ObjectId)) | ||||||
|  |  | ||||||
|  |     def test_objectid_reference_fields(self): | ||||||
|  |  | ||||||
|  |         class Person(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             parent = ReferenceField('self', dbref=False) | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         p1 = Person(name="John").save() | ||||||
|  |         Person(name="Ross", parent=p1).save() | ||||||
|  |  | ||||||
|  |         col = Person._get_collection() | ||||||
|  |         data = col.find_one({'name': 'Ross'}) | ||||||
|  |         self.assertEqual(data['parent'], p1.pk) | ||||||
|  |  | ||||||
|  |         p = Person.objects.get(name="Ross") | ||||||
|  |         self.assertEqual(p.parent, p1) | ||||||
|  |  | ||||||
|     def test_list_item_dereference(self): |     def test_list_item_dereference(self): | ||||||
|         """Ensure that DBRef items in ListFields are dereferenced. |         """Ensure that DBRef items in ListFields are dereferenced. | ||||||
|         """ |         """ | ||||||
| @@ -1118,6 +1215,7 @@ class FieldTest(unittest.TestCase): | |||||||
|             boss = ReferenceField('self') |             boss = ReferenceField('self') | ||||||
|             friends = ListField(ReferenceField('self')) |             friends = ListField(ReferenceField('self')) | ||||||
|  |  | ||||||
|  |         Employee.drop_collection() | ||||||
|         bill = Employee(name='Bill Lumbergh') |         bill = Employee(name='Bill Lumbergh') | ||||||
|         bill.save() |         bill.save() | ||||||
|  |  | ||||||
| @@ -1241,7 +1339,41 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         class BlogPost(Document): |         class BlogPost(Document): | ||||||
|             title = StringField() |             title = StringField() | ||||||
|             author = ReferenceField(Member) |             author = ReferenceField(Member, dbref=False) | ||||||
|  |  | ||||||
|  |         Member.drop_collection() | ||||||
|  |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
|  |         m1 = Member(user_num=1) | ||||||
|  |         m1.save() | ||||||
|  |         m2 = Member(user_num=2) | ||||||
|  |         m2.save() | ||||||
|  |  | ||||||
|  |         post1 = BlogPost(title='post 1', author=m1) | ||||||
|  |         post1.save() | ||||||
|  |  | ||||||
|  |         post2 = BlogPost(title='post 2', author=m2) | ||||||
|  |         post2.save() | ||||||
|  |  | ||||||
|  |         post = BlogPost.objects(author=m1).first() | ||||||
|  |         self.assertEqual(post.id, post1.id) | ||||||
|  |  | ||||||
|  |         post = BlogPost.objects(author=m2).first() | ||||||
|  |         self.assertEqual(post.id, post2.id) | ||||||
|  |  | ||||||
|  |         Member.drop_collection() | ||||||
|  |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_reference_query_conversion_dbref(self): | ||||||
|  |         """Ensure that ReferenceFields can be queried using objects and values | ||||||
|  |         of the type of the primary key of the referenced object. | ||||||
|  |         """ | ||||||
|  |         class Member(Document): | ||||||
|  |             user_num = IntField(primary_key=True) | ||||||
|  |  | ||||||
|  |         class BlogPost(Document): | ||||||
|  |             title = StringField() | ||||||
|  |             author = ReferenceField(Member, dbref=True) | ||||||
|  |  | ||||||
|         Member.drop_collection() |         Member.drop_collection() | ||||||
|         BlogPost.drop_collection() |         BlogPost.drop_collection() | ||||||
| @@ -1385,7 +1517,7 @@ class FieldTest(unittest.TestCase): | |||||||
|         Person.drop_collection() |         Person.drop_collection() | ||||||
|         Person(name="Wilson Jr").save() |         Person(name="Wilson Jr").save() | ||||||
|  |  | ||||||
|         self.assertEquals(repr(Person.objects(city=None)), |         self.assertEqual(repr(Person.objects(city=None)), | ||||||
|                             "[<Person: Person object>]") |                             "[<Person: Person object>]") | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -1463,7 +1595,7 @@ class FieldTest(unittest.TestCase): | |||||||
|             content_type = StringField() |             content_type = StringField() | ||||||
|             blob = BinaryField() |             blob = BinaryField() | ||||||
|  |  | ||||||
|         BLOB = '\xe6\x00\xc4\xff\x07' |         BLOB = b('\xe6\x00\xc4\xff\x07') | ||||||
|         MIME_TYPE = 'application/octet-stream' |         MIME_TYPE = 'application/octet-stream' | ||||||
|  |  | ||||||
|         Attachment.drop_collection() |         Attachment.drop_collection() | ||||||
| @@ -1473,7 +1605,7 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         attachment_1 = Attachment.objects().first() |         attachment_1 = Attachment.objects().first() | ||||||
|         self.assertEqual(MIME_TYPE, attachment_1.content_type) |         self.assertEqual(MIME_TYPE, attachment_1.content_type) | ||||||
|         self.assertEqual(BLOB, str(attachment_1.blob)) |         self.assertEqual(BLOB, bin_type(attachment_1.blob)) | ||||||
|  |  | ||||||
|         Attachment.drop_collection() |         Attachment.drop_collection() | ||||||
|  |  | ||||||
| @@ -1500,12 +1632,12 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         attachment_required = AttachmentRequired() |         attachment_required = AttachmentRequired() | ||||||
|         self.assertRaises(ValidationError, attachment_required.validate) |         self.assertRaises(ValidationError, attachment_required.validate) | ||||||
|         attachment_required.blob = Binary('\xe6\x00\xc4\xff\x07') |         attachment_required.blob = Binary(b('\xe6\x00\xc4\xff\x07')) | ||||||
|         attachment_required.validate() |         attachment_required.validate() | ||||||
|  |  | ||||||
|         attachment_size_limit = AttachmentSizeLimit(blob='\xe6\x00\xc4\xff\x07') |         attachment_size_limit = AttachmentSizeLimit(blob=b('\xe6\x00\xc4\xff\x07')) | ||||||
|         self.assertRaises(ValidationError, attachment_size_limit.validate) |         self.assertRaises(ValidationError, attachment_size_limit.validate) | ||||||
|         attachment_size_limit.blob = '\xe6\x00\xc4\xff' |         attachment_size_limit.blob = b('\xe6\x00\xc4\xff') | ||||||
|         attachment_size_limit.validate() |         attachment_size_limit.validate() | ||||||
|  |  | ||||||
|         Attachment.drop_collection() |         Attachment.drop_collection() | ||||||
| @@ -1632,8 +1764,8 @@ class FieldTest(unittest.TestCase): | |||||||
|         class SetFile(Document): |         class SetFile(Document): | ||||||
|             the_file = FileField() |             the_file = FileField() | ||||||
|  |  | ||||||
|         text = 'Hello, World!' |         text = b('Hello, World!') | ||||||
|         more_text = 'Foo Bar' |         more_text = b('Foo Bar') | ||||||
|         content_type = 'text/plain' |         content_type = 'text/plain' | ||||||
|  |  | ||||||
|         PutFile.drop_collection() |         PutFile.drop_collection() | ||||||
| @@ -1646,14 +1778,14 @@ class FieldTest(unittest.TestCase): | |||||||
|         putfile.validate() |         putfile.validate() | ||||||
|         result = PutFile.objects.first() |         result = PutFile.objects.first() | ||||||
|         self.assertTrue(putfile == result) |         self.assertTrue(putfile == result) | ||||||
|         self.assertEquals(result.the_file.read(), text) |         self.assertEqual(result.the_file.read(), text) | ||||||
|         self.assertEquals(result.the_file.content_type, content_type) |         self.assertEqual(result.the_file.content_type, content_type) | ||||||
|         result.the_file.delete() # Remove file from GridFS |         result.the_file.delete() # Remove file from GridFS | ||||||
|         PutFile.objects.delete() |         PutFile.objects.delete() | ||||||
|  |  | ||||||
|         # Ensure file-like objects are stored |         # Ensure file-like objects are stored | ||||||
|         putfile = PutFile() |         putfile = PutFile() | ||||||
|         putstring = StringIO.StringIO() |         putstring = StringIO() | ||||||
|         putstring.write(text) |         putstring.write(text) | ||||||
|         putstring.seek(0) |         putstring.seek(0) | ||||||
|         putfile.the_file.put(putstring, content_type=content_type) |         putfile.the_file.put(putstring, content_type=content_type) | ||||||
| @@ -1661,8 +1793,8 @@ class FieldTest(unittest.TestCase): | |||||||
|         putfile.validate() |         putfile.validate() | ||||||
|         result = PutFile.objects.first() |         result = PutFile.objects.first() | ||||||
|         self.assertTrue(putfile == result) |         self.assertTrue(putfile == result) | ||||||
|         self.assertEquals(result.the_file.read(), text) |         self.assertEqual(result.the_file.read(), text) | ||||||
|         self.assertEquals(result.the_file.content_type, content_type) |         self.assertEqual(result.the_file.content_type, content_type) | ||||||
|         result.the_file.delete() |         result.the_file.delete() | ||||||
|  |  | ||||||
|         streamfile = StreamFile() |         streamfile = StreamFile() | ||||||
| @@ -1674,14 +1806,14 @@ class FieldTest(unittest.TestCase): | |||||||
|         streamfile.validate() |         streamfile.validate() | ||||||
|         result = StreamFile.objects.first() |         result = StreamFile.objects.first() | ||||||
|         self.assertTrue(streamfile == result) |         self.assertTrue(streamfile == result) | ||||||
|         self.assertEquals(result.the_file.read(), text + more_text) |         self.assertEqual(result.the_file.read(), text + more_text) | ||||||
|         self.assertEquals(result.the_file.content_type, content_type) |         self.assertEqual(result.the_file.content_type, content_type) | ||||||
|         result.the_file.seek(0) |         result.the_file.seek(0) | ||||||
|         self.assertEquals(result.the_file.tell(), 0) |         self.assertEqual(result.the_file.tell(), 0) | ||||||
|         self.assertEquals(result.the_file.read(len(text)), text) |         self.assertEqual(result.the_file.read(len(text)), text) | ||||||
|         self.assertEquals(result.the_file.tell(), len(text)) |         self.assertEqual(result.the_file.tell(), len(text)) | ||||||
|         self.assertEquals(result.the_file.read(len(more_text)), more_text) |         self.assertEqual(result.the_file.read(len(more_text)), more_text) | ||||||
|         self.assertEquals(result.the_file.tell(), len(text + more_text)) |         self.assertEqual(result.the_file.tell(), len(text + more_text)) | ||||||
|         result.the_file.delete() |         result.the_file.delete() | ||||||
|  |  | ||||||
|         # Ensure deleted file returns None |         # Ensure deleted file returns None | ||||||
| @@ -1693,7 +1825,7 @@ class FieldTest(unittest.TestCase): | |||||||
|         setfile.validate() |         setfile.validate() | ||||||
|         result = SetFile.objects.first() |         result = SetFile.objects.first() | ||||||
|         self.assertTrue(setfile == result) |         self.assertTrue(setfile == result) | ||||||
|         self.assertEquals(result.the_file.read(), text) |         self.assertEqual(result.the_file.read(), text) | ||||||
|  |  | ||||||
|         # Try replacing file with new one |         # Try replacing file with new one | ||||||
|         result.the_file.replace(more_text) |         result.the_file.replace(more_text) | ||||||
| @@ -1701,7 +1833,7 @@ class FieldTest(unittest.TestCase): | |||||||
|         result.validate() |         result.validate() | ||||||
|         result = SetFile.objects.first() |         result = SetFile.objects.first() | ||||||
|         self.assertTrue(setfile == result) |         self.assertTrue(setfile == result) | ||||||
|         self.assertEquals(result.the_file.read(), more_text) |         self.assertEqual(result.the_file.read(), more_text) | ||||||
|         result.the_file.delete() |         result.the_file.delete() | ||||||
|  |  | ||||||
|         PutFile.drop_collection() |         PutFile.drop_collection() | ||||||
| @@ -1722,7 +1854,7 @@ class FieldTest(unittest.TestCase): | |||||||
|         GridDocument.drop_collection() |         GridDocument.drop_collection() | ||||||
|  |  | ||||||
|         with tempfile.TemporaryFile() as f: |         with tempfile.TemporaryFile() as f: | ||||||
|             f.write("Hello World!") |             f.write(b("Hello World!")) | ||||||
|             f.flush() |             f.flush() | ||||||
|  |  | ||||||
|             # Test without default |             # Test without default | ||||||
| @@ -1733,28 +1865,28 @@ class FieldTest(unittest.TestCase): | |||||||
|             doc_b = GridDocument.objects.with_id(doc_a.id) |             doc_b = GridDocument.objects.with_id(doc_a.id) | ||||||
|             doc_b.the_file.replace(f, filename='doc_b') |             doc_b.the_file.replace(f, filename='doc_b') | ||||||
|             doc_b.save() |             doc_b.save() | ||||||
|             self.assertNotEquals(doc_b.the_file.grid_id, None) |             self.assertNotEqual(doc_b.the_file.grid_id, None) | ||||||
|  |  | ||||||
|             # Test it matches |             # Test it matches | ||||||
|             doc_c = GridDocument.objects.with_id(doc_b.id) |             doc_c = GridDocument.objects.with_id(doc_b.id) | ||||||
|             self.assertEquals(doc_b.the_file.grid_id, doc_c.the_file.grid_id) |             self.assertEqual(doc_b.the_file.grid_id, doc_c.the_file.grid_id) | ||||||
|  |  | ||||||
|             # Test with default |             # Test with default | ||||||
|             doc_d = GridDocument(the_file='') |             doc_d = GridDocument(the_file=b('')) | ||||||
|             doc_d.save() |             doc_d.save() | ||||||
|  |  | ||||||
|             doc_e = GridDocument.objects.with_id(doc_d.id) |             doc_e = GridDocument.objects.with_id(doc_d.id) | ||||||
|             self.assertEquals(doc_d.the_file.grid_id, doc_e.the_file.grid_id) |             self.assertEqual(doc_d.the_file.grid_id, doc_e.the_file.grid_id) | ||||||
|  |  | ||||||
|             doc_e.the_file.replace(f, filename='doc_e') |             doc_e.the_file.replace(f, filename='doc_e') | ||||||
|             doc_e.save() |             doc_e.save() | ||||||
|  |  | ||||||
|             doc_f = GridDocument.objects.with_id(doc_e.id) |             doc_f = GridDocument.objects.with_id(doc_e.id) | ||||||
|             self.assertEquals(doc_e.the_file.grid_id, doc_f.the_file.grid_id) |             self.assertEqual(doc_e.the_file.grid_id, doc_f.the_file.grid_id) | ||||||
|  |  | ||||||
|         db = GridDocument._get_db() |         db = GridDocument._get_db() | ||||||
|         grid_fs = gridfs.GridFS(db) |         grid_fs = gridfs.GridFS(db) | ||||||
|         self.assertEquals(['doc_b', 'doc_e'], grid_fs.list()) |         self.assertEqual(['doc_b', 'doc_e'], grid_fs.list()) | ||||||
|  |  | ||||||
|     def test_file_uniqueness(self): |     def test_file_uniqueness(self): | ||||||
|         """Ensure that each instance of a FileField is unique |         """Ensure that each instance of a FileField is unique | ||||||
| @@ -1766,7 +1898,7 @@ class FieldTest(unittest.TestCase): | |||||||
|         # First instance |         # First instance | ||||||
|         test_file = TestFile() |         test_file = TestFile() | ||||||
|         test_file.name = "Hello, World!" |         test_file.name = "Hello, World!" | ||||||
|         test_file.the_file.put('Hello, World!') |         test_file.the_file.put(b('Hello, World!')) | ||||||
|         test_file.save() |         test_file.save() | ||||||
|  |  | ||||||
|         # Second instance |         # Second instance | ||||||
| @@ -1786,7 +1918,7 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         test_file = TestFile() |         test_file = TestFile() | ||||||
|         self.assertFalse(bool(test_file.the_file)) |         self.assertFalse(bool(test_file.the_file)) | ||||||
|         test_file.the_file = 'Hello, World!' |         test_file.the_file = b('Hello, World!') | ||||||
|         test_file.the_file.content_type = 'text/plain' |         test_file.the_file.content_type = 'text/plain' | ||||||
|         test_file.save() |         test_file.save() | ||||||
|         self.assertTrue(bool(test_file.the_file)) |         self.assertTrue(bool(test_file.the_file)) | ||||||
| @@ -1802,6 +1934,8 @@ class FieldTest(unittest.TestCase): | |||||||
|         self.assertFalse(test_file.the_file in [{"test": 1}]) |         self.assertFalse(test_file.the_file in [{"test": 1}]) | ||||||
|  |  | ||||||
|     def test_image_field(self): |     def test_image_field(self): | ||||||
|  |         if PY3: | ||||||
|  |             raise SkipTest('PIL does not have Python 3 support') | ||||||
|  |  | ||||||
|         class TestImage(Document): |         class TestImage(Document): | ||||||
|             image = ImageField() |             image = ImageField() | ||||||
| @@ -1814,15 +1948,17 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         t = TestImage.objects.first() |         t = TestImage.objects.first() | ||||||
|  |  | ||||||
|         self.assertEquals(t.image.format, 'PNG') |         self.assertEqual(t.image.format, 'PNG') | ||||||
|  |  | ||||||
|         w, h = t.image.size |         w, h = t.image.size | ||||||
|         self.assertEquals(w, 371) |         self.assertEqual(w, 371) | ||||||
|         self.assertEquals(h, 76) |         self.assertEqual(h, 76) | ||||||
|  |  | ||||||
|         t.image.delete() |         t.image.delete() | ||||||
|  |  | ||||||
|     def test_image_field_resize(self): |     def test_image_field_resize(self): | ||||||
|  |         if PY3: | ||||||
|  |             raise SkipTest('PIL does not have Python 3 support') | ||||||
|  |  | ||||||
|         class TestImage(Document): |         class TestImage(Document): | ||||||
|             image = ImageField(size=(185, 37)) |             image = ImageField(size=(185, 37)) | ||||||
| @@ -1835,15 +1971,40 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         t = TestImage.objects.first() |         t = TestImage.objects.first() | ||||||
|  |  | ||||||
|         self.assertEquals(t.image.format, 'PNG') |         self.assertEqual(t.image.format, 'PNG') | ||||||
|         w, h = t.image.size |         w, h = t.image.size | ||||||
|  |  | ||||||
|         self.assertEquals(w, 185) |         self.assertEqual(w, 185) | ||||||
|         self.assertEquals(h, 37) |         self.assertEqual(h, 37) | ||||||
|  |  | ||||||
|  |         t.image.delete() | ||||||
|  |  | ||||||
|  |     def test_image_field_resize_force(self): | ||||||
|  |         if PY3: | ||||||
|  |             raise SkipTest('PIL does not have Python 3 support') | ||||||
|  |  | ||||||
|  |         class TestImage(Document): | ||||||
|  |             image = ImageField(size=(185, 37, True)) | ||||||
|  |  | ||||||
|  |         TestImage.drop_collection() | ||||||
|  |  | ||||||
|  |         t = TestImage() | ||||||
|  |         t.image.put(open(TEST_IMAGE_PATH, 'r')) | ||||||
|  |         t.save() | ||||||
|  |  | ||||||
|  |         t = TestImage.objects.first() | ||||||
|  |  | ||||||
|  |         self.assertEqual(t.image.format, 'PNG') | ||||||
|  |         w, h = t.image.size | ||||||
|  |  | ||||||
|  |         self.assertEqual(w, 185) | ||||||
|  |         self.assertEqual(h, 37) | ||||||
|  |  | ||||||
|         t.image.delete() |         t.image.delete() | ||||||
|  |  | ||||||
|     def test_image_field_thumbnail(self): |     def test_image_field_thumbnail(self): | ||||||
|  |         if PY3: | ||||||
|  |             raise SkipTest('PIL does not have Python 3 support') | ||||||
|  |  | ||||||
|         class TestImage(Document): |         class TestImage(Document): | ||||||
|             image = ImageField(thumbnail_size=(92, 18)) |             image = ImageField(thumbnail_size=(92, 18)) | ||||||
| @@ -1856,13 +2017,12 @@ class FieldTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         t = TestImage.objects.first() |         t = TestImage.objects.first() | ||||||
|  |  | ||||||
|         self.assertEquals(t.image.thumbnail.format, 'PNG') |         self.assertEqual(t.image.thumbnail.format, 'PNG') | ||||||
|         self.assertEquals(t.image.thumbnail.width, 92) |         self.assertEqual(t.image.thumbnail.width, 92) | ||||||
|         self.assertEquals(t.image.thumbnail.height, 18) |         self.assertEqual(t.image.thumbnail.height, 18) | ||||||
|  |  | ||||||
|         t.image.delete() |         t.image.delete() | ||||||
|  |  | ||||||
|  |  | ||||||
|     def test_file_multidb(self): |     def test_file_multidb(self): | ||||||
|         register_connection('test_files', 'test_files') |         register_connection('test_files', 'test_files') | ||||||
|         class TestFile(Document): |         class TestFile(Document): | ||||||
| @@ -1879,16 +2039,16 @@ class FieldTest(unittest.TestCase): | |||||||
|         # First instance |         # First instance | ||||||
|         test_file = TestFile() |         test_file = TestFile() | ||||||
|         test_file.name = "Hello, World!" |         test_file.name = "Hello, World!" | ||||||
|         test_file.the_file.put('Hello, World!', |         test_file.the_file.put(b('Hello, World!'), | ||||||
|                           name="hello.txt") |                           name="hello.txt") | ||||||
|         test_file.save() |         test_file.save() | ||||||
|  |  | ||||||
|         data = get_db("test_files").macumba.files.find_one() |         data = get_db("test_files").macumba.files.find_one() | ||||||
|         self.assertEquals(data.get('name'), 'hello.txt') |         self.assertEqual(data.get('name'), 'hello.txt') | ||||||
|  |  | ||||||
|         test_file = TestFile.objects.first() |         test_file = TestFile.objects.first() | ||||||
|         self.assertEquals(test_file.the_file.read(), |         self.assertEqual(test_file.the_file.read(), | ||||||
|                           'Hello, World!') |                           b('Hello, World!')) | ||||||
|  |  | ||||||
|     def test_geo_indexes(self): |     def test_geo_indexes(self): | ||||||
|         """Ensure that indexes are created automatically for GeoPointFields. |         """Ensure that indexes are created automatically for GeoPointFields. | ||||||
| @@ -1963,6 +2123,27 @@ class FieldTest(unittest.TestCase): | |||||||
|         c = self.db['mongoengine.counters'].find_one({'_id': 'person.id'}) |         c = self.db['mongoengine.counters'].find_one({'_id': 'person.id'}) | ||||||
|         self.assertEqual(c['next'], 10) |         self.assertEqual(c['next'], 10) | ||||||
|  |  | ||||||
|  |     def test_sequence_field_sequence_name(self): | ||||||
|  |         class Person(Document): | ||||||
|  |             id = SequenceField(primary_key=True, sequence_name='jelly') | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         self.db['mongoengine.counters'].drop() | ||||||
|  |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         for x in xrange(10): | ||||||
|  |             p = Person(name="Person %s" % x) | ||||||
|  |             p.save() | ||||||
|  |  | ||||||
|  |         c = self.db['mongoengine.counters'].find_one({'_id': 'jelly.id'}) | ||||||
|  |         self.assertEqual(c['next'], 10) | ||||||
|  |  | ||||||
|  |         ids = [i.id for i in Person.objects] | ||||||
|  |         self.assertEqual(ids, range(1, 11)) | ||||||
|  |  | ||||||
|  |         c = self.db['mongoengine.counters'].find_one({'_id': 'jelly.id'}) | ||||||
|  |         self.assertEqual(c['next'], 10) | ||||||
|  |  | ||||||
|     def test_multiple_sequence_fields(self): |     def test_multiple_sequence_fields(self): | ||||||
|         class Person(Document): |         class Person(Document): | ||||||
|             id = SequenceField(primary_key=True) |             id = SequenceField(primary_key=True) | ||||||
| @@ -2050,6 +2231,28 @@ class FieldTest(unittest.TestCase): | |||||||
|         c = self.db['mongoengine.counters'].find_one({'_id': 'animal.id'}) |         c = self.db['mongoengine.counters'].find_one({'_id': 'animal.id'}) | ||||||
|         self.assertEqual(c['next'], 10) |         self.assertEqual(c['next'], 10) | ||||||
|  |  | ||||||
|  |     def test_embedded_sequence_field(self): | ||||||
|  |         class Comment(EmbeddedDocument): | ||||||
|  |             id = SequenceField() | ||||||
|  |             content = StringField(required=True) | ||||||
|  |  | ||||||
|  |         class Post(Document): | ||||||
|  |             title = StringField(required=True) | ||||||
|  |             comments = ListField(EmbeddedDocumentField(Comment)) | ||||||
|  |  | ||||||
|  |         self.db['mongoengine.counters'].drop() | ||||||
|  |         Post.drop_collection() | ||||||
|  |  | ||||||
|  |         Post(title="MongoEngine", | ||||||
|  |              comments=[Comment(content="NoSQL Rocks"), | ||||||
|  |                        Comment(content="MongoEngine Rocks")]).save() | ||||||
|  |  | ||||||
|  |         c = self.db['mongoengine.counters'].find_one({'_id': 'comment.id'}) | ||||||
|  |         self.assertEqual(c['next'], 2) | ||||||
|  |         post = Post.objects.first() | ||||||
|  |         self.assertEqual(1, post.comments[0].id) | ||||||
|  |         self.assertEqual(2, post.comments[1].id) | ||||||
|  |  | ||||||
|     def test_generic_embedded_document(self): |     def test_generic_embedded_document(self): | ||||||
|         class Car(EmbeddedDocument): |         class Car(EmbeddedDocument): | ||||||
|             name = StringField() |             name = StringField() | ||||||
| @@ -2149,31 +2352,57 @@ class FieldTest(unittest.TestCase): | |||||||
|         post.comments.append(Comment(content='hello', author=bob)) |         post.comments.append(Comment(content='hello', author=bob)) | ||||||
|         post.comments.append(Comment(author=bob)) |         post.comments.append(Comment(author=bob)) | ||||||
|  |  | ||||||
|  |         self.assertRaises(ValidationError, post.validate) | ||||||
|         try: |         try: | ||||||
|             post.validate() |             post.validate() | ||||||
|         except ValidationError, error: |         except ValidationError, error: | ||||||
|             pass |             # ValidationError.errors property | ||||||
|  |             self.assertTrue(hasattr(error, 'errors')) | ||||||
|  |             self.assertTrue(isinstance(error.errors, dict)) | ||||||
|  |             self.assertTrue('comments' in error.errors) | ||||||
|  |             self.assertTrue(1 in error.errors['comments']) | ||||||
|  |             self.assertTrue(isinstance(error.errors['comments'][1]['content'], | ||||||
|  |                             ValidationError)) | ||||||
|  |  | ||||||
|         # ValidationError.errors property |             # ValidationError.schema property | ||||||
|         self.assertTrue(hasattr(error, 'errors')) |             error_dict = error.to_dict() | ||||||
|         self.assertTrue(isinstance(error.errors, dict)) |             self.assertTrue(isinstance(error_dict, dict)) | ||||||
|         self.assertTrue('comments' in error.errors) |             self.assertTrue('comments' in error_dict) | ||||||
|         self.assertTrue(1 in error.errors['comments']) |             self.assertTrue(1 in error_dict['comments']) | ||||||
|         self.assertTrue(isinstance(error.errors['comments'][1]['content'], |             self.assertTrue('content' in error_dict['comments'][1]) | ||||||
|                         ValidationError)) |             self.assertEqual(error_dict['comments'][1]['content'], | ||||||
|  |                              u'Field is required') | ||||||
|         # ValidationError.schema property |  | ||||||
|         error_dict = error.to_dict() |  | ||||||
|         self.assertTrue(isinstance(error_dict, dict)) |  | ||||||
|         self.assertTrue('comments' in error_dict) |  | ||||||
|         self.assertTrue(1 in error_dict['comments']) |  | ||||||
|         self.assertTrue('content' in error_dict['comments'][1]) |  | ||||||
|         self.assertEquals(error_dict['comments'][1]['content'], |  | ||||||
|                           'Field is required') |  | ||||||
|  |  | ||||||
|         post.comments[1].content = 'here we go' |         post.comments[1].content = 'here we go' | ||||||
|         post.validate() |         post.validate() | ||||||
|  |  | ||||||
|  |     def test_email_field(self): | ||||||
|  |         class User(Document): | ||||||
|  |             email = EmailField() | ||||||
|  |  | ||||||
|  |         user = User(email="ross@example.com") | ||||||
|  |         self.assertTrue(user.validate() is None) | ||||||
|  |  | ||||||
|  |         user = User(email=("Kofq@rhom0e4klgauOhpbpNdogawnyIKvQS0wk2mjqrgGQ5S" | ||||||
|  |                            "ucictfqpdkK9iS1zeFw8sg7s7cwAF7suIfUfeyueLpfosjn3" | ||||||
|  |                            "aJIazqqWkm7.net")) | ||||||
|  |         self.assertTrue(user.validate() is None) | ||||||
|  |  | ||||||
|  |         user = User(email='me@localhost') | ||||||
|  |         self.assertRaises(ValidationError, user.validate) | ||||||
|  |  | ||||||
|  |     def test_email_field_honors_regex(self): | ||||||
|  |         class User(Document): | ||||||
|  |             email = EmailField(regex=r'\w+@example.com') | ||||||
|  |  | ||||||
|  |         # Fails regex validation | ||||||
|  |         user = User(email='me@foo.com') | ||||||
|  |         self.assertRaises(ValidationError, user.validate) | ||||||
|  |  | ||||||
|  |         # Passes regex validation | ||||||
|  |         user = User(email='me@example.com') | ||||||
|  |         self.assertTrue(user.validate() is None) | ||||||
|  |  | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -1,16 +1,19 @@ | |||||||
| # -*- coding: utf-8 -*- | from __future__ import with_statement | ||||||
| import unittest | import unittest | ||||||
| import pymongo |  | ||||||
| from bson import ObjectId |  | ||||||
| from datetime import datetime, timedelta | from datetime import datetime, timedelta | ||||||
|  |  | ||||||
|  | import pymongo | ||||||
|  |  | ||||||
|  | from bson import ObjectId | ||||||
|  |  | ||||||
|  | from mongoengine import * | ||||||
|  | from mongoengine.connection import get_connection | ||||||
|  | from mongoengine.python_support import PY3 | ||||||
|  | from mongoengine.tests import query_counter | ||||||
| from mongoengine.queryset import (QuerySet, QuerySetManager, | from mongoengine.queryset import (QuerySet, QuerySetManager, | ||||||
|                                   MultipleObjectsReturned, DoesNotExist, |                                   MultipleObjectsReturned, DoesNotExist, | ||||||
|                                   QueryFieldList) |                                   QueryFieldList) | ||||||
| from mongoengine import * |  | ||||||
| from mongoengine.connection import get_connection |  | ||||||
| from mongoengine.tests import query_counter |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class QuerySetTest(unittest.TestCase): | class QuerySetTest(unittest.TestCase): | ||||||
|  |  | ||||||
| @@ -21,6 +24,8 @@ class QuerySetTest(unittest.TestCase): | |||||||
|             name = StringField() |             name = StringField() | ||||||
|             age = IntField() |             age = IntField() | ||||||
|             meta = {'allow_inheritance': True} |             meta = {'allow_inheritance': True} | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|         self.Person = Person |         self.Person = Person | ||||||
|  |  | ||||||
|     def test_initialisation(self): |     def test_initialisation(self): | ||||||
| @@ -42,7 +47,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         self.assertEqual(QuerySet._transform_query(age__gt=20, age__lt=50), |         self.assertEqual(QuerySet._transform_query(age__gt=20, age__lt=50), | ||||||
|                          {'age': {'$gt': 20, '$lt': 50}}) |                          {'age': {'$gt': 20, '$lt': 50}}) | ||||||
|         self.assertEqual(QuerySet._transform_query(age=20, age__gt=50), |         self.assertEqual(QuerySet._transform_query(age=20, age__gt=50), | ||||||
|                          {'age': 20}) |                          {'$and': [{'age': {'$gt': 50}}, {'age': 20}]}) | ||||||
|         self.assertEqual(QuerySet._transform_query(friend__age__gte=30), |         self.assertEqual(QuerySet._transform_query(friend__age__gte=30), | ||||||
|                          {'friend.age': {'$gte': 30}}) |                          {'friend.age': {'$gte': 30}}) | ||||||
|         self.assertEqual(QuerySet._transform_query(name__exists=True), |         self.assertEqual(QuerySet._transform_query(name__exists=True), | ||||||
| @@ -225,6 +230,35 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         Blog.drop_collection() |         Blog.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_chaining(self): | ||||||
|  |         class A(Document): | ||||||
|  |             s = StringField() | ||||||
|  |  | ||||||
|  |         class B(Document): | ||||||
|  |             ref = ReferenceField(A) | ||||||
|  |             boolfield = BooleanField(default=False) | ||||||
|  |  | ||||||
|  |         A.drop_collection() | ||||||
|  |         B.drop_collection() | ||||||
|  |  | ||||||
|  |         a1 = A(s="test1").save() | ||||||
|  |         a2 = A(s="test2").save() | ||||||
|  |  | ||||||
|  |         B(ref=a1, boolfield=True).save() | ||||||
|  |  | ||||||
|  |         # Works | ||||||
|  |         q1 = B.objects.filter(ref__in=[a1, a2], ref=a1)._query | ||||||
|  |  | ||||||
|  |         # Doesn't work | ||||||
|  |         q2 = B.objects.filter(ref__in=[a1, a2]) | ||||||
|  |         q2 = q2.filter(ref=a1)._query | ||||||
|  |         self.assertEqual(q1, q2) | ||||||
|  |  | ||||||
|  |         a_objects = A.objects(s='test1') | ||||||
|  |         query = B.objects(ref__in=a_objects) | ||||||
|  |         query = query.filter(boolfield=True) | ||||||
|  |         self.assertEquals(query.count(), 1) | ||||||
|  |  | ||||||
|     def test_update_write_options(self): |     def test_update_write_options(self): | ||||||
|         """Test that passing write_options works""" |         """Test that passing write_options works""" | ||||||
|  |  | ||||||
| @@ -239,11 +273,11 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         self.Person.objects.update(set__name='Ross', write_options=write_options) |         self.Person.objects.update(set__name='Ross', write_options=write_options) | ||||||
|  |  | ||||||
|         author = self.Person.objects.first() |         author = self.Person.objects.first() | ||||||
|         self.assertEquals(author.name, 'Ross') |         self.assertEqual(author.name, 'Ross') | ||||||
|  |  | ||||||
|         self.Person.objects.update_one(set__name='Test User', write_options=write_options) |         self.Person.objects.update_one(set__name='Test User', write_options=write_options) | ||||||
|         author = self.Person.objects.first() |         author = self.Person.objects.first() | ||||||
|         self.assertEquals(author.name, 'Test User') |         self.assertEqual(author.name, 'Test User') | ||||||
|  |  | ||||||
|     def test_update_update_has_a_value(self): |     def test_update_update_has_a_value(self): | ||||||
|         """Test to ensure that update is passed a value to update to""" |         """Test to ensure that update is passed a value to update to""" | ||||||
| @@ -332,8 +366,8 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         BlogPost.objects(comments__by="jane").update(inc__comments__S__votes=1) |         BlogPost.objects(comments__by="jane").update(inc__comments__S__votes=1) | ||||||
|  |  | ||||||
|         post = BlogPost.objects.first() |         post = BlogPost.objects.first() | ||||||
|         self.assertEquals(post.comments[1].by, 'jane') |         self.assertEqual(post.comments[1].by, 'jane') | ||||||
|         self.assertEquals(post.comments[1].votes, 8) |         self.assertEqual(post.comments[1].votes, 8) | ||||||
|  |  | ||||||
|         # Currently the $ operator only applies to the first matched item in |         # Currently the $ operator only applies to the first matched item in | ||||||
|         # the query |         # the query | ||||||
| @@ -346,7 +380,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         Simple.objects(x=2).update(inc__x__S=1) |         Simple.objects(x=2).update(inc__x__S=1) | ||||||
|  |  | ||||||
|         simple = Simple.objects.first() |         simple = Simple.objects.first() | ||||||
|         self.assertEquals(simple.x, [1, 3, 3, 2]) |         self.assertEqual(simple.x, [1, 3, 3, 2]) | ||||||
|         Simple.drop_collection() |         Simple.drop_collection() | ||||||
|  |  | ||||||
|         # You can set multiples |         # You can set multiples | ||||||
| @@ -358,10 +392,10 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         Simple.objects(x=3).update(set__x__S=0) |         Simple.objects(x=3).update(set__x__S=0) | ||||||
|  |  | ||||||
|         s = Simple.objects() |         s = Simple.objects() | ||||||
|         self.assertEquals(s[0].x, [1, 2, 0, 4]) |         self.assertEqual(s[0].x, [1, 2, 0, 4]) | ||||||
|         self.assertEquals(s[1].x, [2, 0, 4, 5]) |         self.assertEqual(s[1].x, [2, 0, 4, 5]) | ||||||
|         self.assertEquals(s[2].x, [0, 4, 5, 6]) |         self.assertEqual(s[2].x, [0, 4, 5, 6]) | ||||||
|         self.assertEquals(s[3].x, [4, 5, 6, 7]) |         self.assertEqual(s[3].x, [4, 5, 6, 7]) | ||||||
|  |  | ||||||
|         # Using "$unset" with an expression like this "array.$" will result in |         # Using "$unset" with an expression like this "array.$" will result in | ||||||
|         # the array item becoming None, not being removed. |         # the array item becoming None, not being removed. | ||||||
| @@ -369,14 +403,14 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         Simple(x=[1, 2, 3, 4, 3, 2, 3, 4]).save() |         Simple(x=[1, 2, 3, 4, 3, 2, 3, 4]).save() | ||||||
|         Simple.objects(x=3).update(unset__x__S=1) |         Simple.objects(x=3).update(unset__x__S=1) | ||||||
|         simple = Simple.objects.first() |         simple = Simple.objects.first() | ||||||
|         self.assertEquals(simple.x, [1, 2, None, 4, 3, 2, 3, 4]) |         self.assertEqual(simple.x, [1, 2, None, 4, 3, 2, 3, 4]) | ||||||
|  |  | ||||||
|         # Nested updates arent supported yet.. |         # Nested updates arent supported yet.. | ||||||
|         def update_nested(): |         def update_nested(): | ||||||
|             Simple.drop_collection() |             Simple.drop_collection() | ||||||
|             Simple(x=[{'test': [1, 2, 3, 4]}]).save() |             Simple(x=[{'test': [1, 2, 3, 4]}]).save() | ||||||
|             Simple.objects(x__test=2).update(set__x__S__test__S=3) |             Simple.objects(x__test=2).update(set__x__S__test__S=3) | ||||||
|             self.assertEquals(simple.x, [1, 2, 3, 4]) |             self.assertEqual(simple.x, [1, 2, 3, 4]) | ||||||
|  |  | ||||||
|         self.assertRaises(OperationError, update_nested) |         self.assertRaises(OperationError, update_nested) | ||||||
|         Simple.drop_collection() |         Simple.drop_collection() | ||||||
| @@ -406,8 +440,32 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         BlogPost.objects(comments__by="joe").update(set__comments__S__votes=Vote(score=4)) |         BlogPost.objects(comments__by="joe").update(set__comments__S__votes=Vote(score=4)) | ||||||
|  |  | ||||||
|         post = BlogPost.objects.first() |         post = BlogPost.objects.first() | ||||||
|         self.assertEquals(post.comments[0].by, 'joe') |         self.assertEqual(post.comments[0].by, 'joe') | ||||||
|         self.assertEquals(post.comments[0].votes.score, 4) |         self.assertEqual(post.comments[0].votes.score, 4) | ||||||
|  |  | ||||||
|  |     def test_updates_can_have_match_operators(self): | ||||||
|  |  | ||||||
|  |         class Post(Document): | ||||||
|  |             title = StringField(required=True) | ||||||
|  |             tags = ListField(StringField()) | ||||||
|  |             comments = ListField(EmbeddedDocumentField("Comment")) | ||||||
|  |  | ||||||
|  |         class Comment(EmbeddedDocument): | ||||||
|  |             content = StringField() | ||||||
|  |             name = StringField(max_length=120) | ||||||
|  |             vote = IntField() | ||||||
|  |  | ||||||
|  |         Post.drop_collection() | ||||||
|  |  | ||||||
|  |         comm1 = Comment(content="very funny indeed", name="John S", vote=1) | ||||||
|  |         comm2 = Comment(content="kind of funny", name="Mark P", vote=0) | ||||||
|  |  | ||||||
|  |         Post(title='Fun with MongoEngine', tags=['mongodb', 'mongoengine'], | ||||||
|  |              comments=[comm1, comm2]).save() | ||||||
|  |  | ||||||
|  |         Post.objects().update_one(pull__comments__vote__lt=1) | ||||||
|  |  | ||||||
|  |         self.assertEqual(1, len(Post.objects.first().comments)) | ||||||
|  |  | ||||||
|     def test_mapfield_update(self): |     def test_mapfield_update(self): | ||||||
|         """Ensure that the MapField can be updated.""" |         """Ensure that the MapField can be updated.""" | ||||||
| @@ -496,7 +554,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         Blog.drop_collection() |         Blog.drop_collection() | ||||||
|  |  | ||||||
|         # Recreates the collection |         # Recreates the collection | ||||||
|         self.assertEqual(0, Blog.objects.count()) |         self.assertEqual(0, Blog.objects.count()) | ||||||
|  |  | ||||||
|         with query_counter() as q: |         with query_counter() as q: | ||||||
| @@ -538,6 +596,10 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         self.assertRaises(OperationError, throw_operation_error) |         self.assertRaises(OperationError, throw_operation_error) | ||||||
|  |  | ||||||
|  |         # Test can insert new doc | ||||||
|  |         new_post = Blog(title="code", id=ObjectId()) | ||||||
|  |         Blog.objects.insert(new_post) | ||||||
|  |  | ||||||
|         # test handles other classes being inserted |         # test handles other classes being inserted | ||||||
|         def throw_operation_error_wrong_doc(): |         def throw_operation_error_wrong_doc(): | ||||||
|             class Author(Document): |             class Author(Document): | ||||||
| @@ -561,7 +623,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         Blog.drop_collection() |         Blog.drop_collection() | ||||||
|         blog1 = Blog(title="code", posts=[post1, post2]) |         blog1 = Blog(title="code", posts=[post1, post2]) | ||||||
|         obj_id = Blog.objects.insert(blog1, load_bulk=False) |         obj_id = Blog.objects.insert(blog1, load_bulk=False) | ||||||
|         self.assertEquals(obj_id.__class__.__name__, 'ObjectId') |         self.assertEqual(obj_id.__class__.__name__, 'ObjectId') | ||||||
|  |  | ||||||
|         Blog.drop_collection() |         Blog.drop_collection() | ||||||
|         post3 = Post(comments=[comment1, comment1]) |         post3 = Post(comments=[comment1, comment1]) | ||||||
| @@ -573,7 +635,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         def throw_operation_error_not_unique(): |         def throw_operation_error_not_unique(): | ||||||
|             Blog.objects.insert([blog2, blog3], safe=True) |             Blog.objects.insert([blog2, blog3], safe=True) | ||||||
|  |  | ||||||
|         self.assertRaises(OperationError, throw_operation_error_not_unique) |         self.assertRaises(NotUniqueError, throw_operation_error_not_unique) | ||||||
|         self.assertEqual(Blog.objects.count(), 2) |         self.assertEqual(Blog.objects.count(), 2) | ||||||
|  |  | ||||||
|         Blog.objects.insert([blog2, blog3], write_options={'continue_on_error': True}) |         Blog.objects.insert([blog2, blog3], write_options={'continue_on_error': True}) | ||||||
| @@ -619,7 +681,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|             fresh_o1 = Organization.objects.get(id=o1.id) |             fresh_o1 = Organization.objects.get(id=o1.id) | ||||||
|             fresh_o1.save() |             fresh_o1.save() | ||||||
|  |  | ||||||
|             self.assertEquals(q, 2) |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|         with query_counter() as q: |         with query_counter() as q: | ||||||
|             self.assertEqual(q, 0) |             self.assertEqual(q, 0) | ||||||
| @@ -627,7 +689,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|             fresh_o1 = Organization.objects.get(id=o1.id) |             fresh_o1 = Organization.objects.get(id=o1.id) | ||||||
|             fresh_o1.save(cascade=False) |             fresh_o1.save(cascade=False) | ||||||
|  |  | ||||||
|             self.assertEquals(q, 2) |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|         with query_counter() as q: |         with query_counter() as q: | ||||||
|             self.assertEqual(q, 0) |             self.assertEqual(q, 0) | ||||||
| @@ -636,7 +698,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|             fresh_o1.employees.append(p2) |             fresh_o1.employees.append(p2) | ||||||
|             fresh_o1.save(cascade=False) |             fresh_o1.save(cascade=False) | ||||||
|  |  | ||||||
|             self.assertEquals(q, 3) |             self.assertEqual(q, 3) | ||||||
|  |  | ||||||
|     def test_slave_okay(self): |     def test_slave_okay(self): | ||||||
|         """Ensures that a query can take slave_okay syntax |         """Ensures that a query can take slave_okay syntax | ||||||
| @@ -710,20 +772,20 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         docs = Doc.objects.order_by('number') |         docs = Doc.objects.order_by('number') | ||||||
|  |  | ||||||
|         self.assertEquals(docs.count(), 1000) |         self.assertEqual(docs.count(), 1000) | ||||||
|         self.assertEquals(len(docs), 1000) |         self.assertEqual(len(docs), 1000) | ||||||
|  |  | ||||||
|         docs_string = "%s" % docs |         docs_string = "%s" % docs | ||||||
|         self.assertTrue("Doc: 0" in docs_string) |         self.assertTrue("Doc: 0" in docs_string) | ||||||
|  |  | ||||||
|         self.assertEquals(docs.count(), 1000) |         self.assertEqual(docs.count(), 1000) | ||||||
|         self.assertEquals(len(docs), 1000) |         self.assertEqual(len(docs), 1000) | ||||||
|  |  | ||||||
|         # Limit and skip |         # Limit and skip | ||||||
|         self.assertEquals('[<Doc: 1>, <Doc: 2>, <Doc: 3>]', "%s" % docs[1:4]) |         self.assertEqual('[<Doc: 1>, <Doc: 2>, <Doc: 3>]', "%s" % docs[1:4]) | ||||||
|  |  | ||||||
|         self.assertEquals(docs.count(), 3) |         self.assertEqual(docs.count(), 3) | ||||||
|         self.assertEquals(len(docs), 3) |         self.assertEqual(len(docs), 3) | ||||||
|         for doc in docs: |         for doc in docs: | ||||||
|             self.assertEqual('.. queryset mid-iteration ..', repr(docs)) |             self.assertEqual('.. queryset mid-iteration ..', repr(docs)) | ||||||
|  |  | ||||||
| @@ -880,6 +942,26 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         BlogPost.drop_collection() |         BlogPost.drop_collection() | ||||||
|         Blog.drop_collection() |         Blog.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_raw_and_merging(self): | ||||||
|  |         class Doc(Document): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         raw_query = Doc.objects(__raw__={'deleted': False, | ||||||
|  |                                 'scraped': 'yes', | ||||||
|  |                                 '$nor': [{'views.extracted': 'no'}, | ||||||
|  |                                          {'attachments.views.extracted':'no'}] | ||||||
|  |                                 })._query | ||||||
|  |  | ||||||
|  |         expected = {'deleted': False, '_types': 'Doc', 'scraped': 'yes', | ||||||
|  |                     '$nor': [{'views.extracted': 'no'}, | ||||||
|  |                              {'attachments.views.extracted': 'no'}]} | ||||||
|  |         self.assertEqual(expected, raw_query) | ||||||
|  |  | ||||||
|  |     def assertSequence(self, qs, expected): | ||||||
|  |         self.assertEqual(len(qs), len(expected)) | ||||||
|  |         for i in range(len(qs)): | ||||||
|  |             self.assertEqual(qs[i], expected[i]) | ||||||
|  |  | ||||||
|     def test_ordering(self): |     def test_ordering(self): | ||||||
|         """Ensure default ordering is applied and can be overridden. |         """Ensure default ordering is applied and can be overridden. | ||||||
|         """ |         """ | ||||||
| @@ -893,10 +975,10 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         BlogPost.drop_collection() |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
|         blog_post_1 = BlogPost(title="Blog Post #1", |  | ||||||
|                                published_date=datetime(2010, 1, 5, 0, 0 ,0)) |  | ||||||
|         blog_post_2 = BlogPost(title="Blog Post #2", |         blog_post_2 = BlogPost(title="Blog Post #2", | ||||||
|                                published_date=datetime(2010, 1, 6, 0, 0 ,0)) |                                published_date=datetime(2010, 1, 6, 0, 0 ,0)) | ||||||
|  |         blog_post_1 = BlogPost(title="Blog Post #1", | ||||||
|  |                                published_date=datetime(2010, 1, 5, 0, 0 ,0)) | ||||||
|         blog_post_3 = BlogPost(title="Blog Post #3", |         blog_post_3 = BlogPost(title="Blog Post #3", | ||||||
|                                published_date=datetime(2010, 1, 7, 0, 0 ,0)) |                                published_date=datetime(2010, 1, 7, 0, 0 ,0)) | ||||||
|  |  | ||||||
| @@ -906,14 +988,13 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         # get the "first" BlogPost using default ordering |         # get the "first" BlogPost using default ordering | ||||||
|         # from BlogPost.meta.ordering |         # from BlogPost.meta.ordering | ||||||
|         latest_post = BlogPost.objects.first() |         expected = [blog_post_3, blog_post_2, blog_post_1] | ||||||
|         self.assertEqual(latest_post.title, "Blog Post #3") |         self.assertSequence(BlogPost.objects.all(), expected) | ||||||
|  |  | ||||||
|         # override default ordering, order BlogPosts by "published_date" |         # override default ordering, order BlogPosts by "published_date" | ||||||
|         first_post = BlogPost.objects.order_by("+published_date").first() |         qs = BlogPost.objects.order_by("+published_date") | ||||||
|         self.assertEqual(first_post.title, "Blog Post #1") |         expected = [blog_post_1, blog_post_2, blog_post_3] | ||||||
|  |         self.assertSequence(qs, expected) | ||||||
|         BlogPost.drop_collection() |  | ||||||
|  |  | ||||||
|     def test_only(self): |     def test_only(self): | ||||||
|         """Ensure that QuerySet.only only returns the requested fields. |         """Ensure that QuerySet.only only returns the requested fields. | ||||||
| @@ -1104,27 +1185,27 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         # first three |         # first three | ||||||
|         numbers = Numbers.objects.fields(slice__n=3).get() |         numbers = Numbers.objects.fields(slice__n=3).get() | ||||||
|         self.assertEquals(numbers.n, [0, 1, 2]) |         self.assertEqual(numbers.n, [0, 1, 2]) | ||||||
|  |  | ||||||
|         # last three |         # last three | ||||||
|         numbers = Numbers.objects.fields(slice__n=-3).get() |         numbers = Numbers.objects.fields(slice__n=-3).get() | ||||||
|         self.assertEquals(numbers.n, [-3, -2, -1]) |         self.assertEqual(numbers.n, [-3, -2, -1]) | ||||||
|  |  | ||||||
|         # skip 2, limit 3 |         # skip 2, limit 3 | ||||||
|         numbers = Numbers.objects.fields(slice__n=[2, 3]).get() |         numbers = Numbers.objects.fields(slice__n=[2, 3]).get() | ||||||
|         self.assertEquals(numbers.n, [2, 3, 4]) |         self.assertEqual(numbers.n, [2, 3, 4]) | ||||||
|  |  | ||||||
|         # skip to fifth from last, limit 4 |         # skip to fifth from last, limit 4 | ||||||
|         numbers = Numbers.objects.fields(slice__n=[-5, 4]).get() |         numbers = Numbers.objects.fields(slice__n=[-5, 4]).get() | ||||||
|         self.assertEquals(numbers.n, [-5, -4, -3, -2]) |         self.assertEqual(numbers.n, [-5, -4, -3, -2]) | ||||||
|  |  | ||||||
|         # skip to fifth from last, limit 10 |         # skip to fifth from last, limit 10 | ||||||
|         numbers = Numbers.objects.fields(slice__n=[-5, 10]).get() |         numbers = Numbers.objects.fields(slice__n=[-5, 10]).get() | ||||||
|         self.assertEquals(numbers.n, [-5, -4, -3, -2, -1]) |         self.assertEqual(numbers.n, [-5, -4, -3, -2, -1]) | ||||||
|  |  | ||||||
|         # skip to fifth from last, limit 10 dict method |         # skip to fifth from last, limit 10 dict method | ||||||
|         numbers = Numbers.objects.fields(n={"$slice": [-5, 10]}).get() |         numbers = Numbers.objects.fields(n={"$slice": [-5, 10]}).get() | ||||||
|         self.assertEquals(numbers.n, [-5, -4, -3, -2, -1]) |         self.assertEqual(numbers.n, [-5, -4, -3, -2, -1]) | ||||||
|  |  | ||||||
|     def test_slicing_nested_fields(self): |     def test_slicing_nested_fields(self): | ||||||
|         """Ensure that query slicing an embedded array works. |         """Ensure that query slicing an embedded array works. | ||||||
| @@ -1144,27 +1225,27 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         # first three |         # first three | ||||||
|         numbers = Numbers.objects.fields(slice__embedded__n=3).get() |         numbers = Numbers.objects.fields(slice__embedded__n=3).get() | ||||||
|         self.assertEquals(numbers.embedded.n, [0, 1, 2]) |         self.assertEqual(numbers.embedded.n, [0, 1, 2]) | ||||||
|  |  | ||||||
|         # last three |         # last three | ||||||
|         numbers = Numbers.objects.fields(slice__embedded__n=-3).get() |         numbers = Numbers.objects.fields(slice__embedded__n=-3).get() | ||||||
|         self.assertEquals(numbers.embedded.n, [-3, -2, -1]) |         self.assertEqual(numbers.embedded.n, [-3, -2, -1]) | ||||||
|  |  | ||||||
|         # skip 2, limit 3 |         # skip 2, limit 3 | ||||||
|         numbers = Numbers.objects.fields(slice__embedded__n=[2, 3]).get() |         numbers = Numbers.objects.fields(slice__embedded__n=[2, 3]).get() | ||||||
|         self.assertEquals(numbers.embedded.n, [2, 3, 4]) |         self.assertEqual(numbers.embedded.n, [2, 3, 4]) | ||||||
|  |  | ||||||
|         # skip to fifth from last, limit 4 |         # skip to fifth from last, limit 4 | ||||||
|         numbers = Numbers.objects.fields(slice__embedded__n=[-5, 4]).get() |         numbers = Numbers.objects.fields(slice__embedded__n=[-5, 4]).get() | ||||||
|         self.assertEquals(numbers.embedded.n, [-5, -4, -3, -2]) |         self.assertEqual(numbers.embedded.n, [-5, -4, -3, -2]) | ||||||
|  |  | ||||||
|         # skip to fifth from last, limit 10 |         # skip to fifth from last, limit 10 | ||||||
|         numbers = Numbers.objects.fields(slice__embedded__n=[-5, 10]).get() |         numbers = Numbers.objects.fields(slice__embedded__n=[-5, 10]).get() | ||||||
|         self.assertEquals(numbers.embedded.n, [-5, -4, -3, -2, -1]) |         self.assertEqual(numbers.embedded.n, [-5, -4, -3, -2, -1]) | ||||||
|  |  | ||||||
|         # skip to fifth from last, limit 10 dict method |         # skip to fifth from last, limit 10 dict method | ||||||
|         numbers = Numbers.objects.fields(embedded__n={"$slice": [-5, 10]}).get() |         numbers = Numbers.objects.fields(embedded__n={"$slice": [-5, 10]}).get() | ||||||
|         self.assertEquals(numbers.embedded.n, [-5, -4, -3, -2, -1]) |         self.assertEqual(numbers.embedded.n, [-5, -4, -3, -2, -1]) | ||||||
|  |  | ||||||
|     def test_find_embedded(self): |     def test_find_embedded(self): | ||||||
|         """Ensure that an embedded document is properly returned from a query. |         """Ensure that an embedded document is properly returned from a query. | ||||||
| @@ -1248,7 +1329,6 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         published_posts = (post1, post2, post3, post5, post6) |         published_posts = (post1, post2, post3, post5, post6) | ||||||
|         self.assertTrue(all(obj.id in posts for obj in published_posts)) |         self.assertTrue(all(obj.id in posts for obj in published_posts)) | ||||||
|  |  | ||||||
|  |  | ||||||
|         # Check Q object combination |         # Check Q object combination | ||||||
|         date = datetime(2010, 1, 10) |         date = datetime(2010, 1, 10) | ||||||
|         q = BlogPost.objects(Q(publish_date__lte=date) | Q(published=True)) |         q = BlogPost.objects(Q(publish_date__lte=date) | Q(published=True)) | ||||||
| @@ -1307,6 +1387,42 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         BlogPost.drop_collection() |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_raw_query_and_Q_objects(self): | ||||||
|  |         """ | ||||||
|  |         Test raw plays nicely | ||||||
|  |         """ | ||||||
|  |         class Foo(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             a = StringField() | ||||||
|  |             b = StringField() | ||||||
|  |             c = StringField() | ||||||
|  |  | ||||||
|  |             meta = { | ||||||
|  |                 'allow_inheritance': False | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         query = Foo.objects(__raw__={'$nor': [{'name': 'bar'}]})._query | ||||||
|  |         self.assertEqual(query, {'$nor': [{'name': 'bar'}]}) | ||||||
|  |  | ||||||
|  |         q1 = {'$or': [{'a': 1}, {'b': 1}]} | ||||||
|  |         query = Foo.objects(Q(__raw__=q1) & Q(c=1))._query | ||||||
|  |         self.assertEqual(query, {'$or': [{'a': 1}, {'b': 1}], 'c': 1}) | ||||||
|  |  | ||||||
|  |     def test_q_merge_queries_edge_case(self): | ||||||
|  |  | ||||||
|  |         class User(Document): | ||||||
|  |             email = EmailField(required=False) | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         User.drop_collection() | ||||||
|  |         pk = ObjectId() | ||||||
|  |         User(email='example@example.com', pk=pk).save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(1, User.objects.filter( | ||||||
|  |                                 Q(email='example@example.com') | | ||||||
|  |                                 Q(name='John Doe') | ||||||
|  |                                 ).limit(2).filter(pk=pk).count()) | ||||||
|  |  | ||||||
|     def test_exec_js_query(self): |     def test_exec_js_query(self): | ||||||
|         """Ensure that queries are properly formed for use in exec_js. |         """Ensure that queries are properly formed for use in exec_js. | ||||||
|         """ |         """ | ||||||
| @@ -1405,7 +1521,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         # Test template style |         # Test template style | ||||||
|         code = "{{~comments.content}}" |         code = "{{~comments.content}}" | ||||||
|         sub_code = BlogPost.objects._sub_js_fields(code) |         sub_code = BlogPost.objects._sub_js_fields(code) | ||||||
|         self.assertEquals("cmnts.body", sub_code) |         self.assertEqual("cmnts.body", sub_code) | ||||||
|  |  | ||||||
|         BlogPost.drop_collection() |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
| @@ -1446,12 +1562,15 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         self.assertEqual(1, BlogPost.objects.count()) |         self.assertEqual(1, BlogPost.objects.count()) | ||||||
|  |  | ||||||
|     def test_reverse_delete_rule_cascade_self_referencing(self): |     def test_reverse_delete_rule_cascade_self_referencing(self): | ||||||
|         """Ensure self-referencing CASCADE deletes do not result in infinite loop |         """Ensure self-referencing CASCADE deletes do not result in infinite | ||||||
|  |         loop | ||||||
|         """ |         """ | ||||||
|         class Category(Document): |         class Category(Document): | ||||||
|             name = StringField() |             name = StringField() | ||||||
|             parent = ReferenceField('self', reverse_delete_rule=CASCADE) |             parent = ReferenceField('self', reverse_delete_rule=CASCADE) | ||||||
|  |  | ||||||
|  |         Category.drop_collection() | ||||||
|  |  | ||||||
|         num_children = 3 |         num_children = 3 | ||||||
|         base = Category(name='Root') |         base = Category(name='Root') | ||||||
|         base.save() |         base.save() | ||||||
| @@ -1468,13 +1587,13 @@ class QuerySetTest(unittest.TestCase): | |||||||
|                 child_child.save() |                 child_child.save() | ||||||
|  |  | ||||||
|         tree_size = 1 + num_children + (num_children * num_children) |         tree_size = 1 + num_children + (num_children * num_children) | ||||||
|         self.assertEquals(tree_size, Category.objects.count()) |         self.assertEqual(tree_size, Category.objects.count()) | ||||||
|         self.assertEquals(num_children, Category.objects(parent=base).count()) |         self.assertEqual(num_children, Category.objects(parent=base).count()) | ||||||
|  |  | ||||||
|         # The delete should effectively wipe out the Category collection |         # The delete should effectively wipe out the Category collection | ||||||
|         # without resulting in infinite parent-child cascade recursion |         # without resulting in infinite parent-child cascade recursion | ||||||
|         base.delete() |         base.delete() | ||||||
|         self.assertEquals(0, Category.objects.count()) |         self.assertEqual(0, Category.objects.count()) | ||||||
|  |  | ||||||
|     def test_reverse_delete_rule_nullify(self): |     def test_reverse_delete_rule_nullify(self): | ||||||
|         """Ensure nullification of references to deleted documents. |         """Ensure nullification of references to deleted documents. | ||||||
| @@ -1550,6 +1669,40 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         self.assertEqual(post.authors, [me]) |         self.assertEqual(post.authors, [me]) | ||||||
|         self.assertEqual(another.authors, []) |         self.assertEqual(another.authors, []) | ||||||
|  |  | ||||||
|  |     def test_delete_with_limits(self): | ||||||
|  |  | ||||||
|  |         class Log(Document): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         Log.drop_collection() | ||||||
|  |  | ||||||
|  |         for i in xrange(10): | ||||||
|  |             Log().save() | ||||||
|  |  | ||||||
|  |         Log.objects()[3:5].delete() | ||||||
|  |         self.assertEqual(8, Log.objects.count()) | ||||||
|  |  | ||||||
|  |     def test_delete_with_limit_handles_delete_rules(self): | ||||||
|  |         """Ensure cascading deletion of referring documents from the database. | ||||||
|  |         """ | ||||||
|  |         class BlogPost(Document): | ||||||
|  |             content = StringField() | ||||||
|  |             author = ReferenceField(self.Person, reverse_delete_rule=CASCADE) | ||||||
|  |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
|  |         me = self.Person(name='Test User') | ||||||
|  |         me.save() | ||||||
|  |         someoneelse = self.Person(name='Some-one Else') | ||||||
|  |         someoneelse.save() | ||||||
|  |  | ||||||
|  |         BlogPost(content='Watching TV', author=me).save() | ||||||
|  |         BlogPost(content='Chilling out', author=me).save() | ||||||
|  |         BlogPost(content='Pro Testing', author=someoneelse).save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(3, BlogPost.objects.count()) | ||||||
|  |         self.Person.objects()[:1].delete() | ||||||
|  |         self.assertEqual(1, BlogPost.objects.count()) | ||||||
|  |  | ||||||
|     def test_update(self): |     def test_update(self): | ||||||
|         """Ensure that atomic updates work properly. |         """Ensure that atomic updates work properly. | ||||||
|         """ |         """ | ||||||
| @@ -1735,7 +1888,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         BlogPost.objects(slug="test-2").update_one(set__tags__0__name="python") |         BlogPost.objects(slug="test-2").update_one(set__tags__0__name="python") | ||||||
|         post.reload() |         post.reload() | ||||||
|         self.assertEquals(post.tags[0].name, 'python') |         self.assertEqual(post.tags[0].name, 'python') | ||||||
|  |  | ||||||
|         BlogPost.objects(slug="test-2").update_one(pop__tags=-1) |         BlogPost.objects(slug="test-2").update_one(pop__tags=-1) | ||||||
|         post.reload() |         post.reload() | ||||||
| @@ -1762,7 +1915,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|             set__authors__S=Author(name="Ross")) |             set__authors__S=Author(name="Ross")) | ||||||
|  |  | ||||||
|         message = message.reload() |         message = message.reload() | ||||||
|         self.assertEquals(message.authors[0].name, "Ross") |         self.assertEqual(message.authors[0].name, "Ross") | ||||||
|  |  | ||||||
|         Message.objects(authors__name="Ross").update_one( |         Message.objects(authors__name="Ross").update_one( | ||||||
|             set__authors=[Author(name="Harry"), |             set__authors=[Author(name="Harry"), | ||||||
| @@ -1770,15 +1923,15 @@ class QuerySetTest(unittest.TestCase): | |||||||
|                           Author(name="Adam")]) |                           Author(name="Adam")]) | ||||||
|  |  | ||||||
|         message = message.reload() |         message = message.reload() | ||||||
|         self.assertEquals(message.authors[0].name, "Harry") |         self.assertEqual(message.authors[0].name, "Harry") | ||||||
|         self.assertEquals(message.authors[1].name, "Ross") |         self.assertEqual(message.authors[1].name, "Ross") | ||||||
|         self.assertEquals(message.authors[2].name, "Adam") |         self.assertEqual(message.authors[2].name, "Adam") | ||||||
|  |  | ||||||
|     def test_order_by(self): |     def test_order_by(self): | ||||||
|         """Ensure that QuerySets may be ordered. |         """Ensure that QuerySets may be ordered. | ||||||
|         """ |         """ | ||||||
|         self.Person(name="User A", age=20).save() |  | ||||||
|         self.Person(name="User B", age=40).save() |         self.Person(name="User B", age=40).save() | ||||||
|  |         self.Person(name="User A", age=20).save() | ||||||
|         self.Person(name="User C", age=30).save() |         self.Person(name="User C", age=30).save() | ||||||
|  |  | ||||||
|         names = [p.name for p in self.Person.objects.order_by('-age')] |         names = [p.name for p in self.Person.objects.order_by('-age')] | ||||||
| @@ -1793,6 +1946,93 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         ages = [p.age for p in self.Person.objects.order_by('-name')] |         ages = [p.age for p in self.Person.objects.order_by('-name')] | ||||||
|         self.assertEqual(ages, [30, 40, 20]) |         self.assertEqual(ages, [30, 40, 20]) | ||||||
|  |  | ||||||
|  |     def test_order_by_optional(self): | ||||||
|  |         class BlogPost(Document): | ||||||
|  |             title = StringField() | ||||||
|  |             published_date = DateTimeField(required=False) | ||||||
|  |  | ||||||
|  |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
|  |         blog_post_3 = BlogPost(title="Blog Post #3", | ||||||
|  |                                published_date=datetime(2010, 1, 6, 0, 0 ,0)) | ||||||
|  |         blog_post_2 = BlogPost(title="Blog Post #2", | ||||||
|  |                                published_date=datetime(2010, 1, 5, 0, 0 ,0)) | ||||||
|  |         blog_post_4 = BlogPost(title="Blog Post #4", | ||||||
|  |                                published_date=datetime(2010, 1, 7, 0, 0 ,0)) | ||||||
|  |         blog_post_1 = BlogPost(title="Blog Post #1", published_date=None) | ||||||
|  |  | ||||||
|  |         blog_post_3.save() | ||||||
|  |         blog_post_1.save() | ||||||
|  |         blog_post_4.save() | ||||||
|  |         blog_post_2.save() | ||||||
|  |  | ||||||
|  |         expected = [blog_post_1, blog_post_2, blog_post_3, blog_post_4] | ||||||
|  |         self.assertSequence(BlogPost.objects.order_by('published_date'), | ||||||
|  |                             expected) | ||||||
|  |         self.assertSequence(BlogPost.objects.order_by('+published_date'), | ||||||
|  |                             expected) | ||||||
|  |  | ||||||
|  |         expected.reverse() | ||||||
|  |         self.assertSequence(BlogPost.objects.order_by('-published_date'), | ||||||
|  |                             expected) | ||||||
|  |  | ||||||
|  |     def test_order_by_list(self): | ||||||
|  |         class BlogPost(Document): | ||||||
|  |             title = StringField() | ||||||
|  |             published_date = DateTimeField(required=False) | ||||||
|  |  | ||||||
|  |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
|  |         blog_post_1 = BlogPost(title="A", | ||||||
|  |                                published_date=datetime(2010, 1, 6, 0, 0 ,0)) | ||||||
|  |         blog_post_2 = BlogPost(title="B", | ||||||
|  |                                published_date=datetime(2010, 1, 6, 0, 0 ,0)) | ||||||
|  |         blog_post_3 = BlogPost(title="C", | ||||||
|  |                                published_date=datetime(2010, 1, 7, 0, 0 ,0)) | ||||||
|  |  | ||||||
|  |         blog_post_2.save() | ||||||
|  |         blog_post_3.save() | ||||||
|  |         blog_post_1.save() | ||||||
|  |  | ||||||
|  |         qs = BlogPost.objects.order_by('published_date', 'title') | ||||||
|  |         expected = [blog_post_1, blog_post_2, blog_post_3] | ||||||
|  |         self.assertSequence(qs, expected) | ||||||
|  |  | ||||||
|  |         qs = BlogPost.objects.order_by('-published_date', '-title') | ||||||
|  |         expected.reverse() | ||||||
|  |         self.assertSequence(qs, expected) | ||||||
|  |  | ||||||
|  |     def test_order_by_chaining(self): | ||||||
|  |         """Ensure that an order_by query chains properly and allows .only() | ||||||
|  |         """ | ||||||
|  |         self.Person(name="User B", age=40).save() | ||||||
|  |         self.Person(name="User A", age=20).save() | ||||||
|  |         self.Person(name="User C", age=30).save() | ||||||
|  |  | ||||||
|  |         only_age = self.Person.objects.order_by('-age').only('age') | ||||||
|  |  | ||||||
|  |         names = [p.name for p in only_age] | ||||||
|  |         ages = [p.age for p in only_age] | ||||||
|  |  | ||||||
|  |         # The .only('age') clause should mean that all names are None | ||||||
|  |         self.assertEqual(names, [None, None, None]) | ||||||
|  |         self.assertEqual(ages, [40, 30, 20]) | ||||||
|  |  | ||||||
|  |         qs = self.Person.objects.all().order_by('-age') | ||||||
|  |         qs = qs.limit(10) | ||||||
|  |         ages = [p.age for p in qs] | ||||||
|  |         self.assertEqual(ages, [40, 30, 20]) | ||||||
|  |  | ||||||
|  |         qs = self.Person.objects.all().limit(10) | ||||||
|  |         qs = qs.order_by('-age') | ||||||
|  |         ages = [p.age for p in qs] | ||||||
|  |         self.assertEqual(ages, [40, 30, 20]) | ||||||
|  |  | ||||||
|  |         qs = self.Person.objects.all().skip(0) | ||||||
|  |         qs = qs.order_by('-age') | ||||||
|  |         ages = [p.age for p in qs] | ||||||
|  |         self.assertEqual(ages, [40, 30, 20]) | ||||||
|  |  | ||||||
|     def test_confirm_order_by_reference_wont_work(self): |     def test_confirm_order_by_reference_wont_work(self): | ||||||
|         """Ordering by reference is not possible.  Use map / reduce.. or |         """Ordering by reference is not possible.  Use map / reduce.. or | ||||||
|         denormalise""" |         denormalise""" | ||||||
| @@ -1852,10 +2092,10 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         results = list(results) |         results = list(results) | ||||||
|         self.assertEqual(len(results), 4) |         self.assertEqual(len(results), 4) | ||||||
|  |  | ||||||
|         music = filter(lambda r: r.key == "music", results)[0] |         music = list(filter(lambda r: r.key == "music", results))[0] | ||||||
|         self.assertEqual(music.value, 2) |         self.assertEqual(music.value, 2) | ||||||
|  |  | ||||||
|         film = filter(lambda r: r.key == "film", results)[0] |         film = list(filter(lambda r: r.key == "film", results))[0] | ||||||
|         self.assertEqual(film.value, 3) |         self.assertEqual(film.value, 3) | ||||||
|  |  | ||||||
|         BlogPost.drop_collection() |         BlogPost.drop_collection() | ||||||
| @@ -2155,15 +2395,15 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         Person(name="Wilson Jr").save() |         Person(name="Wilson Jr").save() | ||||||
|  |  | ||||||
|         freq = Person.objects.item_frequencies('city') |         freq = Person.objects.item_frequencies('city') | ||||||
|         self.assertEquals(freq, {'CRB': 1.0, None: 1.0}) |         self.assertEqual(freq, {'CRB': 1.0, None: 1.0}) | ||||||
|         freq = Person.objects.item_frequencies('city', normalize=True) |         freq = Person.objects.item_frequencies('city', normalize=True) | ||||||
|         self.assertEquals(freq, {'CRB': 0.5, None: 0.5}) |         self.assertEqual(freq, {'CRB': 0.5, None: 0.5}) | ||||||
|  |  | ||||||
|  |  | ||||||
|         freq = Person.objects.item_frequencies('city', map_reduce=True) |         freq = Person.objects.item_frequencies('city', map_reduce=True) | ||||||
|         self.assertEquals(freq, {'CRB': 1.0, None: 1.0}) |         self.assertEqual(freq, {'CRB': 1.0, None: 1.0}) | ||||||
|         freq = Person.objects.item_frequencies('city', normalize=True, map_reduce=True) |         freq = Person.objects.item_frequencies('city', normalize=True, map_reduce=True) | ||||||
|         self.assertEquals(freq, {'CRB': 0.5, None: 0.5}) |         self.assertEqual(freq, {'CRB': 0.5, None: 0.5}) | ||||||
|  |  | ||||||
|     def test_item_frequencies_with_null_embedded(self): |     def test_item_frequencies_with_null_embedded(self): | ||||||
|         class Data(EmbeddedDocument): |         class Data(EmbeddedDocument): | ||||||
| @@ -2188,10 +2428,10 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         p.save() |         p.save() | ||||||
|  |  | ||||||
|         ot = Person.objects.item_frequencies('extra.tag', map_reduce=False) |         ot = Person.objects.item_frequencies('extra.tag', map_reduce=False) | ||||||
|         self.assertEquals(ot, {None: 1.0, u'friend': 1.0}) |         self.assertEqual(ot, {None: 1.0, u'friend': 1.0}) | ||||||
|  |  | ||||||
|         ot = Person.objects.item_frequencies('extra.tag', map_reduce=True) |         ot = Person.objects.item_frequencies('extra.tag', map_reduce=True) | ||||||
|         self.assertEquals(ot, {None: 1.0, u'friend': 1.0}) |         self.assertEqual(ot, {None: 1.0, u'friend': 1.0}) | ||||||
|  |  | ||||||
|     def test_item_frequencies_with_0_values(self): |     def test_item_frequencies_with_0_values(self): | ||||||
|         class Test(Document): |         class Test(Document): | ||||||
| @@ -2203,9 +2443,9 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         t.save() |         t.save() | ||||||
|  |  | ||||||
|         ot = Test.objects.item_frequencies('val', map_reduce=True) |         ot = Test.objects.item_frequencies('val', map_reduce=True) | ||||||
|         self.assertEquals(ot, {0: 1}) |         self.assertEqual(ot, {0: 1}) | ||||||
|         ot = Test.objects.item_frequencies('val', map_reduce=False) |         ot = Test.objects.item_frequencies('val', map_reduce=False) | ||||||
|         self.assertEquals(ot, {0: 1}) |         self.assertEqual(ot, {0: 1}) | ||||||
|  |  | ||||||
|     def test_item_frequencies_with_False_values(self): |     def test_item_frequencies_with_False_values(self): | ||||||
|         class Test(Document): |         class Test(Document): | ||||||
| @@ -2217,9 +2457,9 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         t.save() |         t.save() | ||||||
|  |  | ||||||
|         ot = Test.objects.item_frequencies('val', map_reduce=True) |         ot = Test.objects.item_frequencies('val', map_reduce=True) | ||||||
|         self.assertEquals(ot, {False: 1}) |         self.assertEqual(ot, {False: 1}) | ||||||
|         ot = Test.objects.item_frequencies('val', map_reduce=False) |         ot = Test.objects.item_frequencies('val', map_reduce=False) | ||||||
|         self.assertEquals(ot, {False: 1}) |         self.assertEqual(ot, {False: 1}) | ||||||
|  |  | ||||||
|     def test_item_frequencies_normalize(self): |     def test_item_frequencies_normalize(self): | ||||||
|         class Test(Document): |         class Test(Document): | ||||||
| @@ -2234,10 +2474,10 @@ class QuerySetTest(unittest.TestCase): | |||||||
|             Test(val=2).save() |             Test(val=2).save() | ||||||
|  |  | ||||||
|         freqs = Test.objects.item_frequencies('val', map_reduce=False, normalize=True) |         freqs = Test.objects.item_frequencies('val', map_reduce=False, normalize=True) | ||||||
|         self.assertEquals(freqs, {1: 50.0/70, 2: 20.0/70}) |         self.assertEqual(freqs, {1: 50.0/70, 2: 20.0/70}) | ||||||
|  |  | ||||||
|         freqs = Test.objects.item_frequencies('val', map_reduce=True, normalize=True) |         freqs = Test.objects.item_frequencies('val', map_reduce=True, normalize=True) | ||||||
|         self.assertEquals(freqs, {1: 50.0/70, 2: 20.0/70}) |         self.assertEqual(freqs, {1: 50.0/70, 2: 20.0/70}) | ||||||
|  |  | ||||||
|     def test_average(self): |     def test_average(self): | ||||||
|         """Ensure that field can be averaged correctly. |         """Ensure that field can be averaged correctly. | ||||||
| @@ -2297,7 +2537,7 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         foo = Foo(bar=bar) |         foo = Foo(bar=bar) | ||||||
|         foo.save() |         foo.save() | ||||||
|  |  | ||||||
|         self.assertEquals(Foo.objects.distinct("bar"), [bar]) |         self.assertEqual(Foo.objects.distinct("bar"), [bar]) | ||||||
|  |  | ||||||
|     def test_distinct_handles_references_to_alias(self): |     def test_distinct_handles_references_to_alias(self): | ||||||
|         register_connection('testdb', 'mongoenginetest2') |         register_connection('testdb', 'mongoenginetest2') | ||||||
| @@ -2319,7 +2559,26 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         foo = Foo(bar=bar) |         foo = Foo(bar=bar) | ||||||
|         foo.save() |         foo.save() | ||||||
|  |  | ||||||
|         self.assertEquals(Foo.objects.distinct("bar"), [bar]) |         self.assertEqual(Foo.objects.distinct("bar"), [bar]) | ||||||
|  |  | ||||||
|  |     def test_distinct_handles_db_field(self): | ||||||
|  |         """Ensure that distinct resolves field name to db_field as expected. | ||||||
|  |         """ | ||||||
|  |         class Product(Document): | ||||||
|  |             product_id = IntField(db_field='pid') | ||||||
|  |  | ||||||
|  |         Product.drop_collection() | ||||||
|  |  | ||||||
|  |         Product(product_id=1).save() | ||||||
|  |         Product(product_id=2).save() | ||||||
|  |         Product(product_id=1).save() | ||||||
|  |  | ||||||
|  |         self.assertEqual(set(Product.objects.distinct('product_id')), | ||||||
|  |                          set([1, 2])) | ||||||
|  |         self.assertEqual(set(Product.objects.distinct('pid')), | ||||||
|  |                          set([1, 2])) | ||||||
|  |  | ||||||
|  |         Product.drop_collection() | ||||||
|  |  | ||||||
|     def test_custom_manager(self): |     def test_custom_manager(self): | ||||||
|         """Ensure that custom QuerySetManager instances work as expected. |         """Ensure that custom QuerySetManager instances work as expected. | ||||||
| @@ -2477,30 +2736,48 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         """Ensure that index_types will, when disabled, prevent _types |         """Ensure that index_types will, when disabled, prevent _types | ||||||
|         being added to all indices. |         being added to all indices. | ||||||
|         """ |         """ | ||||||
|         class BlogPost(Document): |         class BloggPost(Document): | ||||||
|             date = DateTimeField() |             date = DateTimeField() | ||||||
|             meta = {'index_types': False, |             meta = {'index_types': False, | ||||||
|                     'indexes': ['-date']} |                     'indexes': ['-date']} | ||||||
|  |  | ||||||
|         # Indexes are lazy so use list() to perform query |         # Indexes are lazy so use list() to perform query | ||||||
|         list(BlogPost.objects) |         list(BloggPost.objects) | ||||||
|         info = BlogPost.objects._collection.index_information() |         info = BloggPost.objects._collection.index_information() | ||||||
|         info = [value['key'] for key, value in info.iteritems()] |         info = [value['key'] for key, value in info.iteritems()] | ||||||
|         self.assertTrue([('_types', 1)] not in info) |         self.assertTrue([('_types', 1)] not in info) | ||||||
|         self.assertTrue([('date', -1)] in info) |         self.assertTrue([('date', -1)] in info) | ||||||
|  |  | ||||||
|         BlogPost.drop_collection() |         BloggPost.drop_collection() | ||||||
|  |  | ||||||
|         class BlogPost(Document): |         class BloggPost(Document): | ||||||
|             title = StringField() |             title = StringField() | ||||||
|             meta = {'allow_inheritance': False} |             meta = {'allow_inheritance': False} | ||||||
|  |  | ||||||
|         # _types is not used on objects where allow_inheritance is False |         # _types is not used on objects where allow_inheritance is False | ||||||
|         list(BlogPost.objects) |         list(BloggPost.objects) | ||||||
|         info = BlogPost.objects._collection.index_information() |         info = BloggPost.objects._collection.index_information() | ||||||
|         self.assertFalse([('_types', 1)] in info.values()) |         self.assertFalse([('_types', 1)] in info.values()) | ||||||
|  |  | ||||||
|         BlogPost.drop_collection() |         BloggPost.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_types_index_with_pk(self): | ||||||
|  |  | ||||||
|  |         class Comment(EmbeddedDocument): | ||||||
|  |             comment_id = IntField(required=True) | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             class BlogPost(Document): | ||||||
|  |                 comments = EmbeddedDocumentField(Comment) | ||||||
|  |                 meta = {'indexes': [{'fields': ['pk', 'comments.comment_id'], | ||||||
|  |                     'unique': True}]} | ||||||
|  |         except UnboundLocalError: | ||||||
|  |             self.fail('Unbound local error at types index + pk definition') | ||||||
|  |  | ||||||
|  |         info = BlogPost.objects._collection.index_information() | ||||||
|  |         info = [value['key'] for key, value in info.iteritems()] | ||||||
|  |         index_item = [(u'_types', 1), (u'_id', 1), (u'comments.comment_id', 1)] | ||||||
|  |         self.assertTrue(index_item in info) | ||||||
|  |  | ||||||
|     def test_dict_with_custom_baseclass(self): |     def test_dict_with_custom_baseclass(self): | ||||||
|         """Ensure DictField working with custom base clases. |         """Ensure DictField working with custom base clases. | ||||||
| @@ -2779,8 +3056,8 @@ class QuerySetTest(unittest.TestCase): | |||||||
|  |  | ||||||
|         Post().save() |         Post().save() | ||||||
|         Post(is_published=True).save() |         Post(is_published=True).save() | ||||||
|         self.assertEquals(Post.objects.count(), 2) |         self.assertEqual(Post.objects.count(), 2) | ||||||
|         self.assertEquals(Post.published.count(), 1) |         self.assertEqual(Post.published.count(), 1) | ||||||
|  |  | ||||||
|         Post.drop_collection() |         Post.drop_collection() | ||||||
|  |  | ||||||
| @@ -2946,10 +3223,10 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         Number(n=3).save() |         Number(n=3).save() | ||||||
|  |  | ||||||
|         numbers = [n.n for n in Number.objects.order_by('-n')] |         numbers = [n.n for n in Number.objects.order_by('-n')] | ||||||
|         self.assertEquals([3, 2, 1], numbers) |         self.assertEqual([3, 2, 1], numbers) | ||||||
|  |  | ||||||
|         numbers = [n.n for n in Number.objects.order_by('+n')] |         numbers = [n.n for n in Number.objects.order_by('+n')] | ||||||
|         self.assertEquals([1, 2, 3], numbers) |         self.assertEqual([1, 2, 3], numbers) | ||||||
|         Number.drop_collection() |         Number.drop_collection() | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -3271,15 +3548,22 @@ class QuerySetTest(unittest.TestCase): | |||||||
|         self.assertEqual(len(self.Person.objects.scalar('name')), 55) |         self.assertEqual(len(self.Person.objects.scalar('name')), 55) | ||||||
|         self.assertEqual("A0", "%s" % self.Person.objects.order_by('name').scalar('name').first()) |         self.assertEqual("A0", "%s" % self.Person.objects.order_by('name').scalar('name').first()) | ||||||
|         self.assertEqual("A0", "%s" % self.Person.objects.scalar('name').order_by('name')[0]) |         self.assertEqual("A0", "%s" % self.Person.objects.scalar('name').order_by('name')[0]) | ||||||
|         self.assertEqual("[u'A1', u'A2']",  "%s" % self.Person.objects.order_by('age').scalar('name')[1:3]) |         if PY3: | ||||||
|         self.assertEqual("[u'A51', u'A52']",  "%s" % self.Person.objects.order_by('age').scalar('name')[51:53]) |             self.assertEqual("['A1', 'A2']",  "%s" % self.Person.objects.order_by('age').scalar('name')[1:3]) | ||||||
|  |             self.assertEqual("['A51', 'A52']",  "%s" % self.Person.objects.order_by('age').scalar('name')[51:53]) | ||||||
|  |         else: | ||||||
|  |             self.assertEqual("[u'A1', u'A2']",  "%s" % self.Person.objects.order_by('age').scalar('name')[1:3]) | ||||||
|  |             self.assertEqual("[u'A51', u'A52']",  "%s" % self.Person.objects.order_by('age').scalar('name')[51:53]) | ||||||
|  |  | ||||||
|         # with_id and in_bulk |         # with_id and in_bulk | ||||||
|         person = self.Person.objects.order_by('name').first() |         person = self.Person.objects.order_by('name').first() | ||||||
|         self.assertEqual("A0", "%s" % self.Person.objects.scalar('name').with_id(person.id)) |         self.assertEqual("A0", "%s" % self.Person.objects.scalar('name').with_id(person.id)) | ||||||
|  |  | ||||||
|         pks = self.Person.objects.order_by('age').scalar('pk')[1:3] |         pks = self.Person.objects.order_by('age').scalar('pk')[1:3] | ||||||
|         self.assertEqual("[u'A1', u'A2']",  "%s" % sorted(self.Person.objects.scalar('name').in_bulk(list(pks)).values())) |         if PY3: | ||||||
|  |             self.assertEqual("['A1', 'A2']",  "%s" % sorted(self.Person.objects.scalar('name').in_bulk(list(pks)).values())) | ||||||
|  |         else: | ||||||
|  |             self.assertEqual("[u'A1', u'A2']",  "%s" % sorted(self.Person.objects.scalar('name').in_bulk(list(pks)).values())) | ||||||
|  |  | ||||||
|  |  | ||||||
| class QTest(unittest.TestCase): | class QTest(unittest.TestCase): | ||||||
| @@ -3526,6 +3810,38 @@ class QueryFieldListTest(unittest.TestCase): | |||||||
|         ak = list(Bar.objects(foo__match={'shape': "square", "color": "purple"})) |         ak = list(Bar.objects(foo__match={'shape': "square", "color": "purple"})) | ||||||
|         self.assertEqual([b1], ak) |         self.assertEqual([b1], ak) | ||||||
|  |  | ||||||
|  |     def test_as_pymongo(self): | ||||||
|  |  | ||||||
|  |         from decimal import Decimal | ||||||
|  |  | ||||||
|  |         class User(Document): | ||||||
|  |             id = ObjectIdField('_id') | ||||||
|  |             name = StringField() | ||||||
|  |             age = IntField() | ||||||
|  |             price = DecimalField() | ||||||
|  |  | ||||||
|  |         User.drop_collection() | ||||||
|  |         User(name="Bob Dole", age=89, price=Decimal('1.11')).save() | ||||||
|  |         User(name="Barack Obama", age=51, price=Decimal('2.22')).save() | ||||||
|  |  | ||||||
|  |         users = User.objects.only('name', 'price').as_pymongo() | ||||||
|  |         results = list(users) | ||||||
|  |         self.assertTrue(isinstance(results[0], dict)) | ||||||
|  |         self.assertTrue(isinstance(results[1], dict)) | ||||||
|  |         self.assertEqual(results[0]['name'], 'Bob Dole') | ||||||
|  |         self.assertEqual(results[0]['price'], '1.11') | ||||||
|  |         self.assertEqual(results[1]['name'], 'Barack Obama') | ||||||
|  |         self.assertEqual(results[1]['price'], '2.22') | ||||||
|  |  | ||||||
|  |         # Test coerce_types | ||||||
|  |         users = User.objects.only('name', 'price').as_pymongo(coerce_types=True) | ||||||
|  |         results = list(users) | ||||||
|  |         self.assertTrue(isinstance(results[0], dict)) | ||||||
|  |         self.assertTrue(isinstance(results[1], dict)) | ||||||
|  |         self.assertEqual(results[0]['name'], 'Bob Dole') | ||||||
|  |         self.assertEqual(results[0]['price'], Decimal('1.11')) | ||||||
|  |         self.assertEqual(results[1]['name'], 'Barack Obama') | ||||||
|  |         self.assertEqual(results[1]['price'], Decimal('2.22')) | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -1,4 +1,5 @@ | |||||||
| import unittest | import unittest | ||||||
|  |  | ||||||
| import pymongo | import pymongo | ||||||
| from pymongo import ReadPreference, ReplicaSetConnection | from pymongo import ReadPreference, ReplicaSetConnection | ||||||
|  |  | ||||||
| @@ -26,7 +27,7 @@ class ConnectionTest(unittest.TestCase): | |||||||
|         if not isinstance(conn, ReplicaSetConnection): |         if not isinstance(conn, ReplicaSetConnection): | ||||||
|             return |             return | ||||||
|  |  | ||||||
|         self.assertEquals(conn.read_preference, ReadPreference.SECONDARY_ONLY) |         self.assertEqual(conn.read_preference, ReadPreference.SECONDARY_ONLY) | ||||||
|  |  | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|   | |||||||
| @@ -212,9 +212,9 @@ class SignalTests(unittest.TestCase): | |||||||
|  |  | ||||||
|         # The output of this signal is not entirely deterministic. The reloaded |         # The output of this signal is not entirely deterministic. The reloaded | ||||||
|         # object will have an object ID. Hence, we only check part of the output |         # object will have an object ID. Hence, we only check part of the output | ||||||
|         self.assertEquals(signal_output[3], |         self.assertEqual(signal_output[3], | ||||||
|             "pre_bulk_insert signal, [<Author: Bill Shakespeare>]") |             "pre_bulk_insert signal, [<Author: Bill Shakespeare>]") | ||||||
|         self.assertEquals(signal_output[-2:], |         self.assertEqual(signal_output[-2:], | ||||||
|             ["post_bulk_insert signal, [<Author: Bill Shakespeare>]", |             ["post_bulk_insert signal, [<Author: Bill Shakespeare>]", | ||||||
|              "Is loaded",]) |              "Is loaded",]) | ||||||
|  |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user