Compare commits
	
		
			837 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 859e9b3cc4 | ||
|  | c34e79fad9 | ||
|  | 82446d641e | ||
|  | 9451c9f331 | ||
|  | 7564bbdee8 | ||
|  | 69251e5000 | ||
|  | 6ecdc7b59d | ||
|  | b7d0d8f0cc | ||
|  | df52ed1162 | ||
|  | aa6370dd5d | ||
|  | c272b7901f | ||
|  | c61de6540a | ||
|  | 3c7bf50089 | ||
|  | 32fc4152a7 | ||
|  | bdf7187d5c | ||
|  | 1639576203 | ||
|  | ae20c785ea | ||
|  | a2eb876f8c | ||
|  | 5a1eaa0a98 | ||
|  | 398fd4a548 | ||
|  | 44b9fb66e1 | ||
|  | 2afa2171f9 | ||
|  | 1d7ea71c0d | ||
|  | 2a391f0f16 | ||
|  | e9b8093dac | ||
|  | 6a229cfbc5 | ||
|  | 3300f409ba | ||
|  | 4466005363 | ||
|  | 296ef5bddf | ||
|  | 1f2a432e82 | ||
|  | 855933ab2a | ||
|  | ece8d25187 | ||
|  | 589a720162 | ||
|  | a59b518cf2 | ||
|  | a15352a4f8 | ||
|  | df65f3fc3f | ||
|  | 734986c1b5 | ||
|  | 4a9ed5f2f2 | ||
|  | 088f229865 | ||
|  | cb2cb851e2 | ||
|  | d3962c4f7d | ||
|  | 0301135f96 | ||
|  | f59aa922ea | ||
|  | f60a49d6f6 | ||
|  | 9a190eb00d | ||
|  | 6bad4bd415 | ||
|  | 50d9b0b796 | ||
|  | 12f884e3ac | ||
|  | 02b1aa7355 | ||
|  | 90bfa608dd | ||
|  | 13f38b1c1d | ||
|  | 1afe7240f4 | ||
|  | 7a41155178 | ||
|  | 39a20ea471 | ||
|  | d8855a4a0f | ||
|  | de8da78042 | ||
|  | 318b42dff2 | ||
|  | 0018674b62 | ||
|  | 82913e8d69 | ||
|  | 0d867a108d | ||
|  | 5ee4b4a5ac | ||
|  | 62219d9648 | ||
|  | 6d9bfff19c | ||
|  | 7614b92197 | ||
|  | 7c1afd0031 | ||
|  | ca7b2371fb | ||
|  | ed5fba6b0f | ||
|  | 2b3b3bf652 | ||
|  | 11daf706df | ||
|  | 4a269eb2c4 | ||
|  | 9b3899476c | ||
|  | febb3d7e3d | ||
|  | 83e3c5c7d8 | ||
|  | 3c271845c9 | ||
|  | 56c4292164 | ||
|  | 2531ade3bb | ||
|  | 3e2f035400 | ||
|  | e7bcb5e366 | ||
|  | 112e921ce2 | ||
|  | 216f15602b | ||
|  | fbe1901e65 | ||
|  | 8d2bc444bb | ||
|  | cf4a45da11 | ||
|  | be78209f94 | ||
|  | 45b5bf73fe | ||
|  | 84f9e44b6c | ||
|  | 700bc1b4bb | ||
|  | beef2ede25 | ||
|  | 9bfc838029 | ||
|  | e9d7353294 | ||
|  | a6948771d8 | ||
|  | 403977cd49 | ||
|  | 153538cef9 | ||
|  | 9f1196e982 | ||
|  | 6419a8d09a | ||
|  | 769cee3d64 | ||
|  | fc460b775e | ||
|  | ba59e498de | ||
|  | 939bd2bb1f | ||
|  | e231f71b4a | ||
|  | d06c5f036b | ||
|  | 071562d755 | ||
|  | 391f659af1 | ||
|  | 8a44232bfc | ||
|  | 9188f9bf62 | ||
|  | 0187a0e113 | ||
|  | beacfae400 | ||
|  | fdc385ea33 | ||
|  | 8b97808931 | ||
|  | 179c4a10c8 | ||
|  | 6cef571bfb | ||
|  | fbe8b28b2e | ||
|  | a8d91a56bf | ||
|  | 8d7291506e | ||
|  | d9005ac2fc | ||
|  | c775c0a80c | ||
|  | 700e2cd93d | ||
|  | 083f00be84 | ||
|  | d00859ecfd | ||
|  | 4e73566c11 | ||
|  | 208a467b24 | ||
|  | e1bb453f32 | ||
|  | 4607b08be5 | ||
|  | aa5c776f3d | ||
|  | 0075c0a1e8 | ||
|  | 83fff80b0f | ||
|  | 5e553ffaf7 | ||
|  | 6d185b7f7a | ||
|  | e80144e9f2 | ||
|  | fa4b820931 | ||
|  | 63c5a4dd65 | ||
|  | 34646a414c | ||
|  | 5aeee9deb2 | ||
|  | 4c1509a62a | ||
|  | bfdaae944d | ||
|  | 4e44198bbd | ||
|  | a4e8177b76 | ||
|  | 81bf5cb78b | ||
|  | a9fc476fb8 | ||
|  | 26f0c06624 | ||
|  | 59bd72a888 | ||
|  | 7d808b483e | ||
|  | 3ee60affa9 | ||
|  | 558b8123b5 | ||
|  | ecdf2ae5c7 | ||
|  | aa9ed614ad | ||
|  | 1acdb880fc | ||
|  | 7cd22aaf83 | ||
|  | 5eb63cfa30 | ||
|  | 5dc998ed52 | ||
|  | 8074094568 | ||
|  | 56d1139d71 | ||
|  | 165cdc8840 | ||
|  | c42aef74de | ||
|  | 634e1f661f | ||
|  | a1db437c42 | ||
|  | b8e2bdc99f | ||
|  | 52d4ea7d78 | ||
|  | 7db5335420 | ||
|  | 62480fe940 | ||
|  | 3d7b30da77 | ||
|  | 8e87648d53 | ||
|  | f842c90007 | ||
|  | 7f2b686ab5 | ||
|  | b09c52fc7e | ||
|  | 202d6e414f | ||
|  | 3d817f145c | ||
|  | 181e191fee | ||
|  | 79ecf027dd | ||
|  | 76d771d20f | ||
|  | 4d5f602ee7 | ||
|  | 452bbcc19b | ||
|  | 24b8650026 | ||
|  | 269e6e29d6 | ||
|  | c4b0002ddb | ||
|  | 53598781b8 | ||
|  | 0624cdd6e4 | ||
|  | 5fb9d61d28 | ||
|  | 7b1860d17b | ||
|  | 8797565606 | ||
|  | 3d97c41fe9 | ||
|  | 5edfeb2e29 | ||
|  | 268908b3b2 | ||
|  | fb70b47acb | ||
|  | 591149b1f0 | ||
|  | 9a0a0b1bd4 | ||
|  | 219d316b49 | ||
|  | 3aa2233b5d | ||
|  | d59862ae6e | ||
|  | 0a03f9a31a | ||
|  | dca135190a | ||
|  | aedcf3dc81 | ||
|  | 6961a9494f | ||
|  | 6d70ef1a08 | ||
|  | e1fc15875d | ||
|  | 94ae1388b1 | ||
|  | 17728d4e74 | ||
|  | 417aa743ca | ||
|  | 2f26f7a827 | ||
|  | 09f9c59b3d | ||
|  | bec6805296 | ||
|  | d99c7c20cc | ||
|  | 60b6ad3fcf | ||
|  | 9b4d0f6450 | ||
|  | 1a2c74391c | ||
|  | 08288e591c | ||
|  | 823cf421fa | ||
|  | 3799f27734 | ||
|  | a7edd8602c | ||
|  | c081aca794 | ||
|  | 2ca6648227 | ||
|  | 1af54f93f5 | ||
|  | a9cacd2e06 | ||
|  | f7fbb3d2f6 | ||
|  | adb7bbeea0 | ||
|  | 89c44cd14e | ||
|  | 8105bfd8b3 | ||
|  | de5b678da3 | ||
|  | 66c53f949b | ||
|  | fdc34869ca | ||
|  | 88b1a29719 | ||
|  | b91db87ae0 | ||
|  | 050542c29b | ||
|  | 60f0491f62 | ||
|  | b8a5791de6 | ||
|  | 2bc3948726 | ||
|  | ee7d370751 | ||
|  | a6449a7b2c | ||
|  | bc9a09f52e | ||
|  | 1631788ab6 | ||
|  | dd49d1d4bb | ||
|  | d6c54c7c2a | ||
|  | bda716ef9d | ||
|  | d83d226396 | ||
|  | 91a0e499d9 | ||
|  | f549d8c0bc | ||
|  | 10c30f2224 | ||
|  | 11621c6f5a | ||
|  | 97ac7e5476 | ||
|  | b037fb3e21 | ||
|  | 10bc93dfa6 | ||
|  | 88cb8f3963 | ||
|  | bd005575c4 | ||
|  | ca3b004921 | ||
|  | 8071b23bff | ||
|  | 4bfed7e719 | ||
|  | b76590dc01 | ||
|  | 4e462ffdb5 | ||
|  | 3c8cbcfee7 | ||
|  | 2a8543b3b7 | ||
|  | fd2e40d735 | ||
|  | 5f05843403 | ||
|  | 8bdb42827c | ||
|  | f6961ae9c1 | ||
|  | 3f301f6b0f | ||
|  | 89ad7ef1ab | ||
|  | 81b69648ef | ||
|  | 672a5f190b | ||
|  | 447dd62c03 | ||
|  | c4db3b6cf2 | ||
|  | 2b1eb620fc | ||
|  | 4abfcb0188 | ||
|  | 048826f6f0 | ||
|  | 5446476d99 | ||
|  | 331f8b8ae7 | ||
|  | 63ee4fef1a | ||
|  | 376ca717fa | ||
|  | 7913ed1841 | ||
|  | 3f3f93b0fa | ||
|  | 6471c6e133 | ||
|  | e3cbeb9df0 | ||
|  | 130fb9916d | ||
|  | ac72722e57 | ||
|  | 382b9a61a8 | ||
|  | 13afead9fb | ||
|  | 72aa191e70 | ||
|  | 0d1804461d | ||
|  | 273412fda1 | ||
|  | 49764b51dc | ||
|  | 5834fa840c | ||
|  | 5eb895b952 | ||
|  | d5fb3a9167 | ||
|  | cb324595ef | ||
|  | fa39789bac | ||
|  | bbd3a6961e | ||
|  | 6eb0387a78 | ||
|  | b3ef67a544 | ||
|  | 72995a4b3e | ||
|  | 7395ce5b22 | ||
|  | a4c197a83c | ||
|  | 7a3412dc13 | ||
|  | e079924632 | ||
|  | 7f0d3638ba | ||
|  | cace665858 | ||
|  | 2a8d001213 | ||
|  | a2b0266e01 | ||
|  | 1452d3fac5 | ||
|  | 031c507fde | ||
|  | 0fb629e24c | ||
|  | 0847687fd1 | ||
|  | 859de712b4 | ||
|  | 803164a993 | ||
|  | 147e33c3ca | ||
|  | dc5a613bc7 | ||
|  | 16390c1dec | ||
|  | 4e6f91ae77 | ||
|  | 556e620c7a | ||
|  | 8e1d701c27 | ||
|  | d51d95a28e | ||
|  | 3d15a3b3e2 | ||
|  | 84e611b91e | ||
|  | 4036e9fe34 | ||
|  | b039a2293f | ||
|  | 87f486c4f1 | ||
|  | 14be7ba2e2 | ||
|  | 09c32a63ce | ||
|  | 08ba51f714 | ||
|  | e3cd398f70 | ||
|  | f41c5217c6 | ||
|  | 1b0323bc22 | ||
|  | e04e5f42ef | ||
|  | c24bc77c17 | ||
|  | 99f923e27f | ||
|  | f3d265bbe0 | ||
|  | 5e7efcc8c2 | ||
|  | 62c8823e64 | ||
|  | 5cc9188c5b | ||
|  | 5e8604967c | ||
|  | cae3f3eeff | ||
|  | 22a7ee5885 | ||
|  | 658b85d327 | ||
|  | 967e72723b | ||
|  | 5411cc5573 | ||
|  | ffb3e8b7b9 | ||
|  | 94cad89e32 | ||
|  | 0338ac17b1 | ||
|  | cb1dfdfac6 | ||
|  | 576db9ca88 | ||
|  | 4c2b83d9ca | ||
|  | 7cb24446ec | ||
|  | 0ed79a839d | ||
|  | e518c51de3 | ||
|  | ea35fb1c54 | ||
|  | 7b29378319 | ||
|  | 82fbe7128f | ||
|  | c1fadcac85 | ||
|  | fd7f882011 | ||
|  | fb09fde209 | ||
|  | b2848b8519 | ||
|  | 417bb1b35d | ||
|  | 199b4eb860 | ||
|  | a66417e9d0 | ||
|  | b9255f73c3 | ||
|  | 4b9bacf731 | ||
|  | 602d7dad00 | ||
|  | d32dd9ff62 | ||
|  | 28b7ef2304 | ||
|  | 6dc2672dba | ||
|  | 9a949984ee | ||
|  | aa32d43014 | ||
|  | 4174918476 | ||
|  | 6081fc6faf | ||
|  | 7c62fdc0b8 | ||
|  | 3c88faa889 | ||
|  | d15f5ccbf4 | ||
|  | cfcd77b193 | ||
|  | 525c25b9f6 | ||
|  | c059ad47f2 | ||
|  | 48fd6c1344 | ||
|  | 1ee50922d9 | ||
|  | d63bf0abde | ||
|  | 711db45c02 | ||
|  | 55e20bda12 | ||
|  | 56f00a64d7 | ||
|  | 8553022b0e | ||
|  | 74b5043ef9 | ||
|  | 0e45078116 | ||
|  | 7e87ed79ab | ||
|  | 7312db5c25 | ||
|  | ec7effa0ef | ||
|  | 9a2cf206b2 | ||
|  | 40df08c74c | ||
|  | 5d778648e6 | ||
|  | 1fa47206aa | ||
|  | 6f5bd7b0b9 | ||
|  | c903af032f | ||
|  | 9dd3504765 | ||
|  | 97a1310344 | ||
|  | bf6f03a412 | ||
|  | 5ab13518db | ||
|  | eb892241ee | ||
|  | fac3f038a8 | ||
|  | b1cdd1eb26 | ||
|  | 60c8254f58 | ||
|  | 2ce70448b0 | ||
|  | 3861103585 | ||
|  | 0708d1bedc | ||
|  | c3a8840435 | ||
|  | 3246cf8bdd | ||
|  | 7ecf84395a | ||
|  | 32bab13a8a | ||
|  | 088c40f9f2 | ||
|  | 305fd4b232 | ||
|  | fe5111743d | ||
|  | 8427877bd2 | ||
|  | 118c0deb7a | ||
|  | 1126c85903 | ||
|  | 13935fc335 | ||
|  | 36034ee15f | ||
|  | 1b72ea9cc1 | ||
|  | 04953351f1 | ||
|  | 07e71d9ce9 | ||
|  | 5f53cda3ab | ||
|  | 9260ff9e83 | ||
|  | 08d1689268 | ||
|  | 40b69baa29 | ||
|  | b3251818cc | ||
|  | da8a057ede | ||
|  | efba9ef52a | ||
|  | fb61c9a765 | ||
|  | 95c2643f63 | ||
|  | fc2aff342b | ||
|  | 371dbf009f | ||
|  | 5d5a84dbcf | ||
|  | 7526272f84 | ||
|  | 5cbc76ea81 | ||
|  | 7ba40062d3 | ||
|  | 1781c4638b | ||
|  | 1a049ee49d | ||
|  | 31521ccff5 | ||
|  | e3b4563c2b | ||
|  | c3f5ed0e0e | ||
|  | 378b52321b | ||
|  | 98436f271e | ||
|  | a76008e440 | ||
|  | 6cf0cf9e7d | ||
|  | 608f08c267 | ||
|  | bd3340c73f | ||
|  | c379ff883a | ||
|  | 3b7a8ce449 | ||
|  | e9ad04f763 | ||
|  | f0277736e2 | ||
|  | 49c978ad9e | ||
|  | eeae1b4aea | ||
|  | 9432d1a194 | ||
|  | 3b2dbf1897 | ||
|  | 9c1ad5f631 | ||
|  | c2fef4e791 | ||
|  | 76cbb66843 | ||
|  | 96dbeea171 | ||
|  | 829df581f0 | ||
|  | bd84d08b95 | ||
|  | 2c7469c62a | ||
|  | 6f7d7537f2 | ||
|  | d7c9694be0 | ||
|  | 69171282e9 | ||
|  | 53d66b7267 | ||
|  | ba9813e5a3 | ||
|  | ce8b3ea0a1 | ||
|  | 559fc46037 | ||
|  | 10c0b035ae | ||
|  | 37818d2d72 | ||
|  | 357dd0e7cc | ||
|  | 34b923b7ac | ||
|  | 846f5a868f | ||
|  | 0acb2d904d | ||
|  | 03a757bc6e | ||
|  | 0f68df3b4a | ||
|  | 07ef58c1a7 | ||
|  | 52f5deb456 | ||
|  | e05e6b89f3 | ||
|  | ffc8b21f67 | ||
|  | 16e1f72e65 | ||
|  | 620f4a222e | ||
|  | f30fd71c5e | ||
|  | 3b55deb472 | ||
|  | 4d5164c580 | ||
|  | 5b118f64ec | ||
|  | 07dae64d66 | ||
|  | 501f033712 | ||
|  | a68cb20266 | ||
|  | 3c98a4bff5 | ||
|  | 20eb920cb4 | ||
|  | b06d794854 | ||
|  | f3da5bc092 | ||
|  | d21434dfd6 | ||
|  | ad1aa5bd3e | ||
|  | dd21ce9eac | ||
|  | bba3aeb4fa | ||
|  | 86233bcdf5 | ||
|  | 4f3eacd72c | ||
|  | 67fcdca6d4 | ||
|  | 62cc8d2ab3 | ||
|  | 3a0523dd79 | ||
|  | cec8b67b08 | ||
|  | ca8c3981c4 | ||
|  | ca56785cbc | ||
|  | b12c34334c | ||
|  | 9c8411b251 | ||
|  | 66baa4eb61 | ||
|  | 89646439e7 | ||
|  | bda4776a18 | ||
|  | c6058fafed | ||
|  | 11950eabea | ||
|  | e1282028a5 | ||
|  | 6b880aa8b3 | ||
|  | a3830be4c9 | ||
|  | ef15733efe | ||
|  | f0c5dd1bce | ||
|  | e868f37c60 | ||
|  | 18baa2dd7a | ||
|  | 2560145551 | ||
|  | 3b88a4f728 | ||
|  | 69989365c7 | ||
|  | 2b9c526b47 | ||
|  | d7c42861fb | ||
|  | e9d478ed9f | ||
|  | d6cb5b9abe | ||
|  | 5580b003b5 | ||
|  | 67736c849d | ||
|  | 39e27735cc | ||
|  | 0902b95764 | ||
|  | dc7181a3fd | ||
|  | e93c4c87d8 | ||
|  | dcec61e9b2 | ||
|  | 007f116bfa | ||
|  | 6817f3b7ba | ||
|  | 36993029ad | ||
|  | 012352cf24 | ||
|  | 26723992e3 | ||
|  | 3591593ac7 | ||
|  | d3c2dfbaee | ||
|  | b2b4456f74 | ||
|  | f666141981 | ||
|  | fb4c4e3e08 | ||
|  | 34fa5cd241 | ||
|  | 833fa3d94d | ||
|  | 92471445ec | ||
|  | 3acfd90720 | ||
|  | 4742328b90 | ||
|  | b4c54b1b62 | ||
|  | 76cb851c40 | ||
|  | 3fcc0e9789 | ||
|  | 8e65154201 | ||
|  | c0f7c4ca2d | ||
|  | db2f64c290 | ||
|  | a3c46fec07 | ||
|  | 62388cb740 | ||
|  | 9c9903664a | ||
|  | 556eed0151 | ||
|  | 4012722a8d | ||
|  | 4c68bc6c96 | ||
|  | 159923fae2 | ||
|  | 72c7a010ff | ||
|  | 2c8f004103 | ||
|  | 67a9b358a0 | ||
|  | b5eb3ea1cd | ||
|  | 98bc0a7c10 | ||
|  | bb24879149 | ||
|  | 3d6ee0ce00 | ||
|  | ee72845701 | ||
|  | d158727154 | ||
|  | 73092dcb33 | ||
|  | 91ddd310ba | ||
|  | 20dd7562e0 | ||
|  | b7e84031e3 | ||
|  | f11ee1f9cf | ||
|  | 449f5a00dc | ||
|  | bd1bf9ba24 | ||
|  | 2af5f3c56e | ||
|  | 1849f75ad0 | ||
|  | 32e66b29f4 | ||
|  | 69012e8ad1 | ||
|  | 17642c8a8c | ||
|  | f1aec68f23 | ||
|  | 3e30d71263 | ||
|  | 266f33adc4 | ||
|  | dcc8d22cec | ||
|  | 9540555b26 | ||
|  | 185e7a6a7e | ||
|  | c39f315ddc | ||
|  | 5b230b90b9 | ||
|  | 1ed9a36d0a | ||
|  | e0911a5fe0 | ||
|  | 3297578e8d | ||
|  | 954d5c16d8 | ||
|  | 95efa39b52 | ||
|  | c27ccc91d2 | ||
|  | 4fb6fcabef | ||
|  | 2635e41f69 | ||
|  | ba01817ee3 | ||
|  | 1e1d7073c8 | ||
|  | 40eb23a97a | ||
|  | f4711699e4 | ||
|  | 3b62cf80cd | ||
|  | d99c5973c3 | ||
|  | 7de9adc6b1 | ||
|  | 17addbefe2 | ||
|  | d274576b47 | ||
|  | 6373e20696 | ||
|  | 809fe44b43 | ||
|  | 198ccc028a | ||
|  | b96e27a7e4 | ||
|  | 1147ac4350 | ||
|  | 21d267cb11 | ||
|  | 6791f205af | ||
|  | 7ab2e21c10 | ||
|  | 2f991ac6f1 | ||
|  | 9411b38508 | ||
|  | 386c48b116 | ||
|  | 9d82911f63 | ||
|  | 51065e7a4d | ||
|  | 327452622e | ||
|  | 13316e5380 | ||
|  | 9f98025b8c | ||
|  | 564f950037 | ||
|  | be651caa68 | ||
|  | aa00feb6a5 | ||
|  | 03c0fd9ada | ||
|  | 6093e88eeb | ||
|  | ec519f20fa | ||
|  | d3495896fa | ||
|  | 323c86308a | ||
|  | f9057e1a28 | ||
|  | 9596a25bb9 | ||
|  | 47bfeec115 | ||
|  | 6bfd6c322b | ||
|  | 0512dd4c25 | ||
|  | acbc741037 | ||
|  | c2163ecee5 | ||
|  | 71689fcf23 | ||
|  | 1c334141ee | ||
|  | b89d71bfa5 | ||
|  | 3179c4e4ac | ||
|  | f5e39c0064 | ||
|  | 86e2797c57 | ||
|  | 39b749432a | ||
|  | 0ad343484f | ||
|  | 196606438c | ||
|  | 6896818bfd | ||
|  | eb4f0ad7fb | ||
|  | 467e61bcc1 | ||
|  | a2c78c9063 | ||
|  | b23353e376 | ||
|  | b8e9790de3 | ||
|  | e37e8d9e65 | ||
|  | f657432be3 | ||
|  | 80c2895e56 | ||
|  | 88da998532 | ||
|  | 225972e151 | ||
|  | 4972bdb383 | ||
|  | 11c7a15067 | ||
|  | 9df725165b | ||
|  | 682326c130 | ||
|  | 86575cb035 | ||
|  | eecc6188a7 | ||
|  | 3b4df4615a | ||
|  | edfda6ad5b | ||
|  | 3c7e8be2e7 | ||
|  | 416fcba846 | ||
|  | e196e229cd | ||
|  | da57572409 | ||
|  | ef172712da | ||
|  | 170c56bcb9 | ||
|  | f3ca9fa4c5 | ||
|  | 48facec524 | ||
|  | ee0c75a26d | ||
|  | e9c92f30ba | ||
|  | 0a074e52e0 | ||
|  | da3f4c30e2 | ||
|  | 2b08ca7c99 | ||
|  | c8e466a160 | ||
|  | a39685d98c | ||
|  | 90200dbe9c | ||
|  | 2304dac8e3 | ||
|  | 38b2919c0d | ||
|  | 207fd9fcb7 | ||
|  | fbcf58c48f | ||
|  | 8f4a579df9 | ||
|  | 600ca3bcf9 | ||
|  | a4d2f22fd2 | ||
|  | 00c8d7e6f5 | ||
|  | 0d89e967f2 | ||
|  | 447f8d0113 | ||
|  | 60802796cb | ||
|  | 5b42578cb1 | ||
|  | 25a0a5364a | ||
|  | 047cc218a6 | ||
|  | 39fc862676 | ||
|  | f47d926f29 | ||
|  | f4d0938e3d | ||
|  | f156da4ec2 | ||
|  | 0c1e5da9a8 | ||
|  | d6b317c552 | ||
|  | 01826c6876 | ||
|  | 0b62c9d2f6 | ||
|  | 72161a9b71 | ||
|  | df8f4e7251 | ||
|  | aa13ab37c4 | ||
|  | acda64a837 | ||
|  | 49a001a93a | ||
|  | 22a6ec7794 | ||
|  | 26c6e4997c | ||
|  | d7086fc4a3 | ||
|  | 92150e07d3 | ||
|  | ac3c857e1a | ||
|  | 48e313fb44 | ||
|  | 5390117275 | ||
|  | 0b3af2052f | ||
|  | bb19ba3eb6 | ||
|  | 879bf08d18 | ||
|  | b99421e7ee | ||
|  | 3b6d8fab47 | ||
|  | 53c0cdc0c1 | ||
|  | 58f877de1a | ||
|  | 95a7b33fb4 | ||
|  | 81dd5adccf | ||
|  | 94e86a0be1 | ||
|  | 5b2dbfe007 | ||
|  | 4451843a39 | ||
|  | 5e2c5fa97b | ||
|  | 018b206177 | ||
|  | 03d31b1890 | ||
|  | 265776566e | ||
|  | 6e77e32855 | ||
|  | 0b1c506626 | ||
|  | 719a653375 | ||
|  | 66520c77f8 | ||
|  | ab2d019349 | ||
|  | d0e0b291df | ||
|  | 200e9eca92 | ||
|  | 634f771547 | ||
|  | 2996f8919d | ||
|  | 1b68efe7c7 | ||
|  | a19a7b976c | ||
|  | 145b0c33fc | ||
|  | 8b1a39f2c1 | ||
|  | 6dbc051409 | ||
|  | c148a5bbfc | ||
|  | 90d9bd9723 | ||
|  | bc7e6ccf53 | ||
|  | 6cab002214 | ||
|  | 3762a69537 | ||
|  | 348f7b5dfc | ||
|  | 008a62e4e9 | ||
|  | a4c5fa57e0 | ||
|  | 9be6c41af7 | ||
|  | 5c311eefb1 | ||
|  | d0ceb74a2e | ||
|  | ea1fe6a538 | ||
|  | a93509c9b3 | ||
|  | 210e9e23af | ||
|  | c4513f0286 | ||
|  | 1114572b47 | ||
|  | b2588d1c4f | ||
|  | 69d3e0c4b6 | ||
|  | e2414d8fea | ||
|  | 24db0d1499 | ||
|  | 89f505bb13 | ||
|  | df5b1f3806 | ||
|  | 755deb3ffe | ||
|  | 59f8c9f38e | ||
|  | 69e9b5d55e | ||
|  | a2d8b0ffbe | ||
|  | 0bbf3a3d76 | ||
|  | 10de19d38b | ||
|  | 73aff806f3 | ||
|  | 963a223e7e | ||
|  | bbfc2f416e | ||
|  | e05d31eaaf | ||
|  | 431f006751 | ||
|  | ffc9d7b152 | ||
|  | 79604180db | ||
|  | 7d6e117f68 | ||
|  | b3cc2f990a | ||
|  | 8d953f0bcb | ||
|  | 5cac52720c | ||
|  | bca6119db8 | ||
|  | 568000805f | ||
|  | 3fb6307596 | ||
|  | 7aa0031dec | ||
|  | 2585f1b724 | ||
|  | 470e08f616 | ||
|  | f1e51f9708 | ||
|  | e0becc109d | ||
|  | 47e4dd40cd | ||
|  | c38faebc25 | ||
|  | 21b7d8f8ea | ||
|  | 3357b55fbf | ||
|  | f01add9ef5 | ||
|  | b0b8e11c60 | ||
|  | 7e0fcb9e65 | ||
|  | 972235cf06 | ||
|  | b3c9a76619 | ||
|  | 5f84d6f8f8 | ||
|  | 1cdeb8130d | ||
|  | ce69428cc6 | ||
|  | 1818cf7114 | ||
|  | b375c41586 | ||
|  | d85ee4e051 | ||
|  | cfc394963f | ||
|  | e7380e3676 | ||
|  | 597ef8b947 | ||
|  | 484bc1e6f0 | ||
|  | afd416c84e | ||
|  | 84d7987108 | ||
|  | ec927bdd63 | ||
|  | df7d4cbc47 | ||
|  | dc51362f0b | ||
|  | da2d282cf6 | ||
|  | 3b37bf4794 | ||
|  | 42a58dda57 | ||
|  | 4d695a3544 | ||
|  | 45080d3fd1 | ||
|  | 9195d96705 | ||
|  | 54d276f6a7 | ||
|  | 2a7fc03e79 | ||
|  | eb3e6963fa | ||
|  | 960aea2fd4 | ||
|  | ef5815e4a5 | ||
|  | b7e8108edd | ||
|  | d48296eacc | ||
|  | e0a546000d | ||
|  | 4c93e2945c | ||
|  | a6d64b2010 | ||
|  | 2e74c93878 | ||
|  | f86496b545 | ||
|  | 557fb19d13 | ||
|  | 196f4471be | ||
|  | ccb4827ec9 | ||
|  | 4ae21a671d | ||
|  | af1d7ef664 | ||
|  | bb4444f54d | ||
|  | 3bead80f96 | ||
|  | 8ad0df41a0 | ||
|  | aa9cba38c4 | ||
|  | 12a7fc1af1 | 
							
								
								
									
										16
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,16 @@ | |||||||
| *.pyc | .* | ||||||
| .*.swp | !.gitignore | ||||||
|  | *~ | ||||||
|  | *.py[co] | ||||||
|  | .*.sw[po] | ||||||
|  | *.egg | ||||||
| docs/.build | docs/.build | ||||||
| docs/_build | docs/_build | ||||||
|  | build/ | ||||||
|  | dist/ | ||||||
|  | mongoengine.egg-info/ | ||||||
|  | env/ | ||||||
|  | .settings | ||||||
|  | .project | ||||||
|  | .pydevproject | ||||||
|  | tests/bugfix.py | ||||||
|   | |||||||
							
								
								
									
										99
									
								
								AUTHORS
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										99
									
								
								AUTHORS
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,99 @@ | |||||||
|  | The PRIMARY AUTHORS are (and/or have been): | ||||||
|  |  | ||||||
|  | Ross Lawley <ross.lawley@gmail.com> | ||||||
|  | Harry Marr <harry@hmarr.com> | ||||||
|  | Matt Dennewitz <mattdennewitz@gmail.com> | ||||||
|  | Deepak Thukral <iapain@yahoo.com> | ||||||
|  | Florian Schlachter <flori@n-schlachter.de> | ||||||
|  | Steve Challis <steve@stevechallis.com> | ||||||
|  | Wilson Júnior <wilsonpjunior@gmail.com> | ||||||
|  | Dan Crosta https://github.com/dcrosta | ||||||
|  |  | ||||||
|  | CONTRIBUTORS | ||||||
|  |  | ||||||
|  | Dervived from the git logs, inevitably incomplete but all of whom and others | ||||||
|  | have submitted patches, reported bugs and generally helped make MongoEngine | ||||||
|  | that much better: | ||||||
|  |  | ||||||
|  |  * Harry Marr | ||||||
|  |  * Ross Lawley | ||||||
|  |  * blackbrrr | ||||||
|  |  * Florian Schlachter | ||||||
|  |  * Vincent Driessen | ||||||
|  |  * Steve Challis | ||||||
|  |  * flosch | ||||||
|  |  * Deepak Thukral | ||||||
|  |  * Colin Howe | ||||||
|  |  * Wilson Júnior | ||||||
|  |  * Alistair Roche | ||||||
|  |  * Dan Crosta | ||||||
|  |  * Viktor Kerkez | ||||||
|  |  * Stephan Jaekel | ||||||
|  |  * Rached Ben Mustapha | ||||||
|  |  * Greg Turner | ||||||
|  |  * Daniel Hasselrot | ||||||
|  |  * Mircea Pasoi | ||||||
|  |  * Matt Chisholm | ||||||
|  |  * James Punteney | ||||||
|  |  * TimothéePeignier | ||||||
|  |  * Stuart Rackham | ||||||
|  |  * Serge Matveenko | ||||||
|  |  * Matt Dennewitz | ||||||
|  |  * Don Spaulding | ||||||
|  |  * Ales Zoulek | ||||||
|  |  * sshwsfc | ||||||
|  |  * sib | ||||||
|  |  * Samuel Clay | ||||||
|  |  * Nick Vlku | ||||||
|  |  * martin | ||||||
|  |  * Flavio Amieiro | ||||||
|  |  * Анхбаяр Лхагвадорж | ||||||
|  |  * Zak Johnson | ||||||
|  |  * Victor Farazdagi | ||||||
|  |  * vandersonmota | ||||||
|  |  * Theo Julienne | ||||||
|  |  * sp | ||||||
|  |  * Slavi Pantaleev | ||||||
|  |  * Richard Henry | ||||||
|  |  * Nicolas Perriault | ||||||
|  |  * Nick Vlku Jr | ||||||
|  |  * Michael Henson | ||||||
|  |  * Leo Honkanen | ||||||
|  |  * kuno | ||||||
|  |  * Josh Ourisman | ||||||
|  |  * Jaime | ||||||
|  |  * Igor Ivanov | ||||||
|  |  * Gregg Lind | ||||||
|  |  * Gareth Lloyd | ||||||
|  |  * Albert Choi | ||||||
|  |  * John Arnfield | ||||||
|  |  * grubberr | ||||||
|  |  * Paul Aliagas | ||||||
|  |  * Paul Cunnane | ||||||
|  |  * Julien Rebetez | ||||||
|  |  * Marc Tamlyn | ||||||
|  |  * Karim Allah | ||||||
|  |  * Adam Parrish | ||||||
|  |  * jpfarias | ||||||
|  |  * jonrscott | ||||||
|  |  * Alice Zoë Bevan-McGregor | ||||||
|  |  * Stephen Young | ||||||
|  |  * tkloc | ||||||
|  |  * aid | ||||||
|  |  * yamaneko1212 | ||||||
|  |  * dave mankoff | ||||||
|  |  * Alexander G. Morano | ||||||
|  |  * jwilder | ||||||
|  |  * Joe Shaw | ||||||
|  |  * Adam Flynn | ||||||
|  |  * Ankhbayar | ||||||
|  |  * Jan Schrewe | ||||||
|  |  * David Koblas | ||||||
|  |  * Crittercism | ||||||
|  |  * Alvin Liang | ||||||
|  |  * andrewmlevy | ||||||
|  |  * Chris Faulkner | ||||||
|  |  * Ashwin Purohit | ||||||
|  |  * Shalabh Aggarwal | ||||||
|  |  * Chris Williams | ||||||
|  |  * Robert Kajic | ||||||
| @@ -1,6 +1,6 @@ | |||||||
|  | include MANIFEST.in | ||||||
| include README.rst | include README.rst | ||||||
| include LICENSE | include LICENSE | ||||||
|  | include AUTHORS | ||||||
| recursive-include docs * | recursive-include docs * | ||||||
| prune docs/_build/* | prune docs/_build | ||||||
| recursive-include tests * |  | ||||||
| recursive-exclude * *.pyc *.swp |  | ||||||
|   | |||||||
							
								
								
									
										23
									
								
								README.rst
									
									
									
									
									
								
							
							
						
						
									
										23
									
								
								README.rst
									
									
									
									
									
								
							| @@ -3,20 +3,21 @@ MongoEngine | |||||||
| =========== | =========== | ||||||
| :Info: MongoEngine is an ORM-like layer on top of PyMongo. | :Info: MongoEngine is an ORM-like layer on top of PyMongo. | ||||||
| :Author: Harry Marr (http://github.com/hmarr) | :Author: Harry Marr (http://github.com/hmarr) | ||||||
|  | :Maintainer: Ross Lawley (http://github.com/rozza) | ||||||
|  |  | ||||||
| About | About | ||||||
| ===== | ===== | ||||||
| MongoEngine is a Python Object-Document Mapper for working with MongoDB. | MongoEngine is a Python Object-Document Mapper for working with MongoDB. | ||||||
| Documentation available at http://hmarr.com/mongoengine/ - there is currently  | Documentation available at http://mongoengine-odm.rtfd.org - there is currently | ||||||
| a `tutorial <http://hmarr.com/mongoengine/tutorial.html>`_, a `user guide  | a `tutorial <http://readthedocs.org/docs/mongoengine-odm/en/latest/tutorial.html>`_, a `user guide | ||||||
| <http://hmarr.com/mongoengine/userguide.html>`_ and an `API reference | <http://readthedocs.org/docs/mongoengine-odm/en/latest/userguide.html>`_ and an `API reference | ||||||
| <http://hmarr.com/mongoengine/apireference.html>`_. | <http://readthedocs.org/docs/mongoengine-odm/en/latest/apireference.html>`_. | ||||||
|  |  | ||||||
| Installation | Installation | ||||||
| ============ | ============ | ||||||
| If you have `setuptools <http://peak.telecommunity.com/DevCenter/setuptools>`_ | If you have `setuptools <http://peak.telecommunity.com/DevCenter/setuptools>`_ | ||||||
| you can use ``easy_install mongoengine``. Otherwise, you can download the | you can use ``easy_install -U mongoengine``. Otherwise, you can download the | ||||||
| source from `GitHub <http://github.com/hmarr/mongoengine>`_ and run ``python | source from `GitHub <http://github.com/MongoEngine/mongoengine>`_ and run ``python | ||||||
| setup.py install``. | setup.py install``. | ||||||
|  |  | ||||||
| Dependencies | Dependencies | ||||||
| @@ -82,8 +83,16 @@ 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 | ||||||
|  | ========= | ||||||
|  | - `MongoEngine Users mailing list | ||||||
|  |   <http://groups.google.com/group/mongoengine-users>`_ | ||||||
|  | - `MongoEngine Developers mailing list | ||||||
|  |   <http://groups.google.com/group/mongoengine-dev>`_ | ||||||
|  | - `#mongoengine IRC channel <irc://irc.freenode.net/mongoengine>`_ | ||||||
|  |  | ||||||
| Contributing | Contributing | ||||||
| ============ | ============ | ||||||
| The source is available on `GitHub <http://github.com/hmarr/mongoengine>`_ - to | The source is available on `GitHub <http://github.com/MongoEngine/mongoengine>`_ - to | ||||||
| contribute to the project, fork it on GitHub and send a pull request, all | contribute to the project, fork it on GitHub and send a pull request, all | ||||||
| contributions and suggestions are welcome! | contributions and suggestions are welcome! | ||||||
|   | |||||||
							
								
								
									
										182
									
								
								benchmark.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										182
									
								
								benchmark.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,182 @@ | |||||||
|  | #!/usr/bin/env python | ||||||
|  |  | ||||||
|  | import timeit | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def cprofile_main(): | ||||||
|  |     from pymongo import Connection | ||||||
|  |     connection = Connection() | ||||||
|  |     connection.drop_database('timeit_test') | ||||||
|  |     connection.disconnect() | ||||||
|  |  | ||||||
|  |     from mongoengine import Document, DictField, connect | ||||||
|  |     connect("timeit_test") | ||||||
|  |  | ||||||
|  |     class Noddy(Document): | ||||||
|  |         fields = DictField() | ||||||
|  |  | ||||||
|  |     for i in xrange(1): | ||||||
|  |         noddy = Noddy() | ||||||
|  |         for j in range(20): | ||||||
|  |             noddy.fields["key" + str(j)] = "value " + str(j) | ||||||
|  |         noddy.save() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def main(): | ||||||
|  |     """ | ||||||
|  |     0.4 Performance Figures ... | ||||||
|  |  | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - Pymongo | ||||||
|  |     1.1141769886 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine | ||||||
|  |     2.37724113464 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False | ||||||
|  |     1.92479610443 | ||||||
|  |  | ||||||
|  |     0.5.X | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - Pymongo | ||||||
|  |     1.10552310944 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine | ||||||
|  |     16.5169169903 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False | ||||||
|  |     14.9446101189 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False, cascade=False | ||||||
|  |     14.912801981 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine, force=True | ||||||
|  |     14.9617750645 | ||||||
|  |  | ||||||
|  |     Performance | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - Pymongo | ||||||
|  |     1.10072994232 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine | ||||||
|  |     5.27341103554 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False | ||||||
|  |     4.49365401268 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine, safe=False, validate=False, cascade=False | ||||||
|  |     4.43459296227 | ||||||
|  |     ---------------------------------------------------------------------------------------------------- | ||||||
|  |     Creating 10000 dictionaries - MongoEngine, force=True | ||||||
|  |     4.40114378929 | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     setup = """ | ||||||
|  | from pymongo import Connection | ||||||
|  | connection = Connection() | ||||||
|  | connection.drop_database('timeit_test') | ||||||
|  | """ | ||||||
|  |  | ||||||
|  |     stmt = """ | ||||||
|  | from pymongo import Connection | ||||||
|  | connection = Connection() | ||||||
|  |  | ||||||
|  | db = connection.timeit_test | ||||||
|  | noddy = db.noddy | ||||||
|  |  | ||||||
|  | for i in xrange(10000): | ||||||
|  |     example = {'fields': {}} | ||||||
|  |     for j in range(20): | ||||||
|  |         example['fields']["key"+str(j)] = "value "+str(j) | ||||||
|  |  | ||||||
|  |     noddy.insert(example) | ||||||
|  |  | ||||||
|  | myNoddys = noddy.find() | ||||||
|  | [n for n in myNoddys] # iterate | ||||||
|  | """ | ||||||
|  |  | ||||||
|  |     print "-" * 100 | ||||||
|  |     print """Creating 10000 dictionaries - Pymongo""" | ||||||
|  |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|  |     print t.timeit(1) | ||||||
|  |  | ||||||
|  |     setup = """ | ||||||
|  | from pymongo import Connection | ||||||
|  | connection = Connection() | ||||||
|  | connection.drop_database('timeit_test') | ||||||
|  | connection.disconnect() | ||||||
|  |  | ||||||
|  | from mongoengine import Document, DictField, connect | ||||||
|  | connect("timeit_test") | ||||||
|  |  | ||||||
|  | class Noddy(Document): | ||||||
|  |     fields = DictField() | ||||||
|  | """ | ||||||
|  |  | ||||||
|  |     stmt = """ | ||||||
|  | for i in xrange(10000): | ||||||
|  |     noddy = Noddy() | ||||||
|  |     for j in range(20): | ||||||
|  |         noddy.fields["key"+str(j)] = "value "+str(j) | ||||||
|  |     noddy.save() | ||||||
|  |  | ||||||
|  | myNoddys = Noddy.objects() | ||||||
|  | [n for n in myNoddys] # iterate | ||||||
|  | """ | ||||||
|  |  | ||||||
|  |     print "-" * 100 | ||||||
|  |     print """Creating 10000 dictionaries - MongoEngine""" | ||||||
|  |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|  |     print t.timeit(1) | ||||||
|  |  | ||||||
|  |     stmt = """ | ||||||
|  | for i in xrange(10000): | ||||||
|  |     noddy = Noddy() | ||||||
|  |     for j in range(20): | ||||||
|  |         noddy.fields["key"+str(j)] = "value "+str(j) | ||||||
|  |     noddy.save(safe=False, validate=False) | ||||||
|  |  | ||||||
|  | myNoddys = Noddy.objects() | ||||||
|  | [n for n in myNoddys] # iterate | ||||||
|  | """ | ||||||
|  |  | ||||||
|  |     print "-" * 100 | ||||||
|  |     print """Creating 10000 dictionaries - MongoEngine, safe=False, validate=False""" | ||||||
|  |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|  |     print t.timeit(1) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     stmt = """ | ||||||
|  | for i in xrange(10000): | ||||||
|  |     noddy = Noddy() | ||||||
|  |     for j in range(20): | ||||||
|  |         noddy.fields["key"+str(j)] = "value "+str(j) | ||||||
|  |     noddy.save(safe=False, validate=False, cascade=False) | ||||||
|  |  | ||||||
|  | myNoddys = Noddy.objects() | ||||||
|  | [n for n in myNoddys] # iterate | ||||||
|  | """ | ||||||
|  |  | ||||||
|  |     print "-" * 100 | ||||||
|  |     print """Creating 10000 dictionaries - MongoEngine, safe=False, validate=False, cascade=False""" | ||||||
|  |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|  |     print t.timeit(1) | ||||||
|  |  | ||||||
|  |     stmt = """ | ||||||
|  | for i in xrange(10000): | ||||||
|  |     noddy = Noddy() | ||||||
|  |     for j in range(20): | ||||||
|  |         noddy.fields["key"+str(j)] = "value "+str(j) | ||||||
|  |     noddy.save(force_insert=True, safe=False, validate=False, cascade=False) | ||||||
|  |  | ||||||
|  | myNoddys = Noddy.objects() | ||||||
|  | [n for n in myNoddys] # iterate | ||||||
|  | """ | ||||||
|  |  | ||||||
|  |     print "-" * 100 | ||||||
|  |     print """Creating 10000 dictionaries - MongoEngine, force=True""" | ||||||
|  |     t = timeit.Timer(stmt=stmt, setup=setup) | ||||||
|  |     print t.timeit(1) | ||||||
|  |  | ||||||
|  | if __name__ == "__main__": | ||||||
|  |     main() | ||||||
| @@ -6,6 +6,7 @@ Connecting | |||||||
| ========== | ========== | ||||||
|  |  | ||||||
| .. autofunction:: mongoengine.connect | .. autofunction:: mongoengine.connect | ||||||
|  | .. autofunction:: mongoengine.register_connection | ||||||
|  |  | ||||||
| Documents | Documents | ||||||
| ========= | ========= | ||||||
| @@ -21,29 +22,47 @@ Documents | |||||||
| .. autoclass:: mongoengine.EmbeddedDocument | .. autoclass:: mongoengine.EmbeddedDocument | ||||||
|    :members: |    :members: | ||||||
|  |  | ||||||
|  | .. autoclass:: mongoengine.DynamicDocument | ||||||
|  |    :members: | ||||||
|  |  | ||||||
|  | .. autoclass:: mongoengine.DynamicEmbeddedDocument | ||||||
|  |    :members: | ||||||
|  |  | ||||||
|  | .. autoclass:: mongoengine.document.MapReduceDocument | ||||||
|  |   :members: | ||||||
|  |  | ||||||
| Querying | Querying | ||||||
| ======== | ======== | ||||||
|  |  | ||||||
| .. autoclass:: mongoengine.queryset.QuerySet | .. autoclass:: mongoengine.queryset.QuerySet | ||||||
|    :members: |    :members: | ||||||
|  |  | ||||||
|  |    .. automethod:: mongoengine.queryset.QuerySet.__call__ | ||||||
|  |  | ||||||
| .. autofunction:: mongoengine.queryset.queryset_manager | .. autofunction:: mongoengine.queryset.queryset_manager | ||||||
|  |  | ||||||
| Fields | Fields | ||||||
| ====== | ====== | ||||||
|  |  | ||||||
| .. autoclass:: mongoengine.StringField | .. autoclass:: mongoengine.StringField | ||||||
|  | .. autoclass:: mongoengine.URLField | ||||||
|  | .. autoclass:: mongoengine.EmailField | ||||||
| .. autoclass:: mongoengine.IntField | .. autoclass:: mongoengine.IntField | ||||||
|  |  | ||||||
| .. autoclass:: mongoengine.FloatField | .. autoclass:: mongoengine.FloatField | ||||||
|  | .. autoclass:: mongoengine.DecimalField | ||||||
| .. autoclass:: mongoengine.DateTimeField | .. autoclass:: mongoengine.DateTimeField | ||||||
|  | .. autoclass:: mongoengine.ComplexDateTimeField | ||||||
| .. autoclass:: mongoengine.EmbeddedDocumentField |  | ||||||
|  |  | ||||||
| .. autoclass:: mongoengine.ListField | .. autoclass:: mongoengine.ListField | ||||||
|  | .. autoclass:: mongoengine.SortedListField | ||||||
|  | .. autoclass:: mongoengine.DictField | ||||||
|  | .. autoclass:: mongoengine.MapField | ||||||
| .. autoclass:: mongoengine.ObjectIdField | .. autoclass:: mongoengine.ObjectIdField | ||||||
|  |  | ||||||
| .. autoclass:: mongoengine.ReferenceField | .. autoclass:: mongoengine.ReferenceField | ||||||
|  | .. autoclass:: mongoengine.GenericReferenceField | ||||||
|  | .. autoclass:: mongoengine.EmbeddedDocumentField | ||||||
|  | .. autoclass:: mongoengine.GenericEmbeddedDocumentField | ||||||
|  | .. autoclass:: mongoengine.BooleanField | ||||||
|  | .. autoclass:: mongoengine.FileField | ||||||
|  | .. autoclass:: mongoengine.BinaryField | ||||||
|  | .. autoclass:: mongoengine.GeoPointField | ||||||
|  | .. autoclass:: mongoengine.SequenceField | ||||||
|   | |||||||
| @@ -2,6 +2,243 @@ | |||||||
| Changelog | Changelog | ||||||
| ========= | ========= | ||||||
|  |  | ||||||
|  | Changes in 0.6.x | ||||||
|  | ================ | ||||||
|  |  | ||||||
|  | - Added FutureWarning to inherited classes not declaring 'allow_inheritance' as the default will change in 0.7 | ||||||
|  | - Added support for covered indexes when inheritance is off | ||||||
|  | - No longer always upsert on save for items with a '_id' | ||||||
|  | - Error raised if update doesn't have an operation | ||||||
|  | - DeReferencing is now thread safe | ||||||
|  | - Errors raised if trying to perform a join in a query | ||||||
|  | - Updates can now take __raw__ queries | ||||||
|  | - Added custom 2D index declarations | ||||||
|  | - Added replicaSet connection support | ||||||
|  | - Updated deprecated imports from pymongo (safe for pymongo 2.2) | ||||||
|  | - Added uri support for connections | ||||||
|  | - Added scalar for efficiently returning partial data values (aliased to values_list) | ||||||
|  | - Fixed limit skip bug | ||||||
|  | - Improved Inheritance / Mixin | ||||||
|  | - Added sharding support | ||||||
|  | - Added pymongo 2.1 support | ||||||
|  | - Fixed Abstract documents can now declare indexes | ||||||
|  | - Added db_alias support to individual documents | ||||||
|  | - Fixed GridFS documents can now be pickled | ||||||
|  | - Added Now raises an InvalidDocumentError when declaring multiple fields with the same db_field | ||||||
|  | - Added InvalidQueryError when calling with_id with a filter | ||||||
|  | - Added support for DBRefs in distinct() | ||||||
|  | - Fixed issue saving False booleans | ||||||
|  | - Fixed issue with dynamic documents deltas | ||||||
|  | - Added Reverse Delete Rule support to ListFields - MapFields aren't supported | ||||||
|  | - Added customisable cascade kwarg options | ||||||
|  | - Fixed Handle None values for non-required fields | ||||||
|  | - Removed Document._get_subclasses() - no longer required | ||||||
|  | - Fixed bug requiring subclasses when not actually needed | ||||||
|  | - Fixed deletion of dynamic data | ||||||
|  | - Added support for the $elementMatch operator | ||||||
|  | - Added reverse option to SortedListFields | ||||||
|  | - Fixed dereferencing - multi directional list dereferencing | ||||||
|  | - Fixed issue creating indexes with recursive embedded documents | ||||||
|  | - Fixed recursive lookup in _unique_with_indexes | ||||||
|  | - Fixed passing ComplexField defaults to constructor for ReferenceFields | ||||||
|  | - Fixed validation of DictField Int keys | ||||||
|  | - Added optional cascade saving | ||||||
|  | - Fixed dereferencing - max_depth now taken into account | ||||||
|  | - Fixed document mutation saving issue | ||||||
|  | - Fixed positional operator when replacing embedded documents | ||||||
|  | - Added Non-Django Style choices back (you can have either) | ||||||
|  | - Fixed __repr__ of a sliced queryset | ||||||
|  | - Added recursive validation error of documents / complex fields | ||||||
|  | - Fixed breaking during queryset iteration | ||||||
|  | - Added pre and post bulk-insert signals | ||||||
|  | - Added ImageField - requires PIL | ||||||
|  | - Fixed Reference Fields can be None in get_or_create / queries | ||||||
|  | - Fixed accessing pk on an embedded document | ||||||
|  | - Fixed calling a queryset after drop_collection now recreates the collection | ||||||
|  | - Add field name to validation exception messages | ||||||
|  | - Added UUID field | ||||||
|  | - Improved efficiency of .get() | ||||||
|  | - Updated ComplexFields so if required they won't accept empty lists / dicts | ||||||
|  | - Added spec file for rpm-based distributions | ||||||
|  | - Fixed ListField so it doesnt accept strings | ||||||
|  | - Added DynamicDocument and EmbeddedDynamicDocument classes for expando schemas | ||||||
|  |  | ||||||
|  | Changes in v0.5.2 | ||||||
|  | ================= | ||||||
|  |  | ||||||
|  | - A Robust Circular reference bugfix | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Changes in v0.5.1 | ||||||
|  | ================= | ||||||
|  |  | ||||||
|  | - Fixed simple circular reference bug | ||||||
|  |  | ||||||
|  | Changes in v0.5 | ||||||
|  | =============== | ||||||
|  |  | ||||||
|  | - Added InvalidDocumentError - so Document core methods can't be overwritten | ||||||
|  | - Added GenericEmbeddedDocument - so you can embed any type of embeddable document | ||||||
|  | - Added within_polygon support - for those with mongodb 1.9 | ||||||
|  | - Updated sum / average to use map_reduce as db.eval doesn't work in sharded environments | ||||||
|  | - Added where() - filter to allowing users to specify query expressions as Javascript | ||||||
|  | - Added SequenceField - for creating sequential counters | ||||||
|  | - Added update() convenience method to a document | ||||||
|  | - Added cascading saves - so changes to Referenced documents are saved on .save() | ||||||
|  | - Added select_related() support | ||||||
|  | - Added support for the positional operator | ||||||
|  | - Updated geo index checking to be recursive and check in embedded documents | ||||||
|  | - Updated default collection naming convention | ||||||
|  | - Added Document Mixin support | ||||||
|  | - Fixed queryet __repr__ mid iteration | ||||||
|  | - Added hint() support, so cantell Mongo the proper index to use for the query | ||||||
|  | - Fixed issue with inconsitent setting of _cls breaking inherited referencing | ||||||
|  | - Added help_text and verbose_name to fields to help with some form libs | ||||||
|  | - Updated item_frequencies to handle embedded document lookups | ||||||
|  | - Added delta tracking now only sets / unsets explicitly changed fields | ||||||
|  | - Fixed saving so sets updated values rather than overwrites | ||||||
|  | - Added ComplexDateTimeField - Handles datetimes correctly with microseconds | ||||||
|  | - Added ComplexBaseField - for improved flexibility and performance | ||||||
|  | - Added get_FIELD_display() method for easy choice field displaying | ||||||
|  | - Added queryset.slave_okay(enabled) method | ||||||
|  | - Updated queryset.timeout(enabled) and queryset.snapshot(enabled) to be chainable | ||||||
|  | - Added insert method for bulk inserts | ||||||
|  | - Added blinker signal support | ||||||
|  | - Added query_counter context manager for tests | ||||||
|  | - Added map_reduce method item_frequencies and set as default (as db.eval doesn't work in sharded environments) | ||||||
|  | - Added inline_map_reduce option to map_reduce | ||||||
|  | - Updated connection exception so it provides more info on the cause. | ||||||
|  | - Added searching multiple levels deep in ``DictField`` | ||||||
|  | - Added ``DictField`` entries containing strings to use matching operators | ||||||
|  | - Added ``MapField``, similar to ``DictField`` | ||||||
|  | - Added Abstract Base Classes | ||||||
|  | - Added Custom Objects Managers | ||||||
|  | - Added sliced subfields updating | ||||||
|  | - Added ``NotRegistered`` exception if dereferencing ``Document`` not in the registry | ||||||
|  | - Added a write concern for ``save``, ``update``, ``update_one`` and ``get_or_create`` | ||||||
|  | - Added slicing / subarray fetching controls | ||||||
|  | - Fixed various unique index and other index issues | ||||||
|  | - Fixed threaded connection issues | ||||||
|  | - Added spherical geospatial query operators | ||||||
|  | - Updated queryset to handle latest version of pymongo | ||||||
|  |   map_reduce now requires an output. | ||||||
|  | - Added ``Document`` __hash__, __ne__ for pickling | ||||||
|  | - Added ``FileField`` optional size arg for read method | ||||||
|  | - Fixed ``FileField`` seek and tell methods for reading files | ||||||
|  | - Added ``QuerySet.clone`` to support copying querysets | ||||||
|  | - Fixed item_frequencies when using name thats the same as a native js function | ||||||
|  | - Added reverse delete rules | ||||||
|  | - Fixed issue with unset operation | ||||||
|  | - Fixed Q-object bug | ||||||
|  | - Added ``QuerySet.all_fields`` resets previous .only() and .exclude() | ||||||
|  | - Added ``QuerySet.exclude`` | ||||||
|  | - Added django style choices | ||||||
|  | - Fixed order and filter issue | ||||||
|  | - Added ``QuerySet.only`` subfield support | ||||||
|  | - Added creation_counter to ``BaseField`` allowing fields to be sorted in the | ||||||
|  |   way the user has specified them | ||||||
|  | - Fixed various errors | ||||||
|  | - Added many tests | ||||||
|  |  | ||||||
|  | Changes in v0.4 | ||||||
|  | =============== | ||||||
|  | - Added ``GridFSStorage`` Django storage backend | ||||||
|  | - Added ``FileField`` for GridFS support | ||||||
|  | - New Q-object implementation, which is no longer based on Javascript | ||||||
|  | - Added ``SortedListField`` | ||||||
|  | - Added ``EmailField`` | ||||||
|  | - Added ``GeoPointField`` | ||||||
|  | - Added ``exact`` and ``iexact`` match operators to ``QuerySet`` | ||||||
|  | - Added ``get_document_or_404`` and ``get_list_or_404`` Django shortcuts | ||||||
|  | - Added new query operators for Geo queries | ||||||
|  | - Added ``not`` query operator | ||||||
|  | - Added new update operators: ``pop`` and ``add_to_set`` | ||||||
|  | - Added ``__raw__`` query parameter | ||||||
|  | - Added support for custom querysets | ||||||
|  | - Fixed document inheritance primary key issue | ||||||
|  | - Added support for querying by array element position | ||||||
|  | - Base class can now be defined for ``DictField`` | ||||||
|  | - Fixed MRO error that occured on document inheritance | ||||||
|  | - Added ``QuerySet.distinct``, ``QuerySet.create``, ``QuerySet.snapshot``, | ||||||
|  |   ``QuerySet.timeout`` and ``QuerySet.all`` | ||||||
|  | - Subsequent calls to ``connect()`` now work | ||||||
|  | - Introduced ``min_length`` for ``StringField`` | ||||||
|  | - Fixed multi-process connection issue | ||||||
|  | - Other minor fixes | ||||||
|  |  | ||||||
|  | Changes in v0.3 | ||||||
|  | =============== | ||||||
|  | - Added MapReduce support | ||||||
|  | - Added ``contains``, ``startswith`` and ``endswith`` query operators (and | ||||||
|  |   case-insensitive versions that are prefixed with 'i') | ||||||
|  | - Deprecated fields' ``name`` parameter, replaced with ``db_field`` | ||||||
|  | - Added ``QuerySet.only`` for only retrieving specific fields | ||||||
|  | - Added ``QuerySet.in_bulk()`` for bulk querying using ids | ||||||
|  | - ``QuerySet``\ s now have a ``rewind()`` method, which is called automatically | ||||||
|  |   when the iterator is exhausted, allowing ``QuerySet``\ s to be reused | ||||||
|  | - Added ``DictField`` | ||||||
|  | - Added ``URLField`` | ||||||
|  | - Added ``DecimalField`` | ||||||
|  | - Added ``BinaryField`` | ||||||
|  | - Added ``GenericReferenceField`` | ||||||
|  | - Added ``get()`` and ``get_or_create()`` methods to ``QuerySet`` | ||||||
|  | - ``ReferenceField``\ s may now reference the document they are defined on | ||||||
|  |   (recursive references) and documents that have not yet been defined | ||||||
|  | - ``Document`` objects may now be compared for equality (equal if _ids are | ||||||
|  |   equal and documents are of same type) | ||||||
|  | - ``QuerySet`` update methods now have an ``upsert`` parameter | ||||||
|  | - Added field name substitution for Javascript code (allows the user to use the | ||||||
|  |   Python names for fields in JS, which are later substituted for the real field | ||||||
|  |   names) | ||||||
|  | - ``Q`` objects now support regex querying | ||||||
|  | - Fixed bug where referenced documents within lists weren't properly | ||||||
|  |   dereferenced | ||||||
|  | - ``ReferenceField``\ s may now be queried using their _id | ||||||
|  | - Fixed bug where ``EmbeddedDocuments`` couldn't be non-polymorphic | ||||||
|  | - ``queryset_manager`` functions now accept two arguments -- the document class | ||||||
|  |   as the first and the queryset as the second | ||||||
|  | - Fixed bug where ``QuerySet.exec_js`` ignored ``Q`` objects | ||||||
|  | - Other minor fixes | ||||||
|  |  | ||||||
|  | Changes in v0.2.2 | ||||||
|  | ================= | ||||||
|  | - Fixed bug that prevented indexes from being used on ``ListField``\ s | ||||||
|  | - ``Document.filter()`` added as an alias to ``Document.__call__()`` | ||||||
|  | - ``validate()`` may now be used on ``EmbeddedDocument``\ s | ||||||
|  |  | ||||||
|  | Changes in v0.2.1 | ||||||
|  | ================= | ||||||
|  | - Added a MongoEngine backend for Django sessions | ||||||
|  | - Added ``force_insert`` to ``Document.save()`` | ||||||
|  | - Improved querying syntax for ``ListField`` and ``EmbeddedDocumentField`` | ||||||
|  | - Added support for user-defined primary keys (``_id`` in MongoDB) | ||||||
|  |  | ||||||
|  | Changes in v0.2 | ||||||
|  | =============== | ||||||
|  | - Added ``Q`` class for building advanced queries | ||||||
|  | - Added ``QuerySet`` methods for atomic updates to documents | ||||||
|  | - Fields may now specify ``unique=True`` to enforce uniqueness across a | ||||||
|  |   collection | ||||||
|  | - Added option for default document ordering | ||||||
|  | - Fixed bug in index definitions | ||||||
|  |  | ||||||
|  | Changes in v0.1.3 | ||||||
|  | ================= | ||||||
|  | - Added Django authentication backend | ||||||
|  | - Added ``Document.meta`` support for indexes, which are ensured just before | ||||||
|  |   querying takes place | ||||||
|  | - A few minor bugfixes | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Changes in v0.1.2 | ||||||
|  | ================= | ||||||
|  | - Query values may be processed before before being used in queries | ||||||
|  | - Made connections lazy | ||||||
|  | - Fixed bug in Document dictionary-style access | ||||||
|  | - Added ``BooleanField`` | ||||||
|  | - Added ``Document.reload()`` method | ||||||
|  |  | ||||||
|  |  | ||||||
| Changes in v0.1.1 | Changes in v0.1.1 | ||||||
| ================= | ================= | ||||||
| - Documents may now use capped collections | - Documents may now use capped collections | ||||||
|   | |||||||
| @@ -22,10 +22,10 @@ sys.path.append(os.path.abspath('..')) | |||||||
|  |  | ||||||
| # Add any Sphinx extension module names here, as strings. They can be extensions | # Add any Sphinx extension module names here, as strings. They can be extensions | ||||||
| # coming with Sphinx (named 'sphinx.ext.*') or your custom ones. | # coming with Sphinx (named 'sphinx.ext.*') or your custom ones. | ||||||
| extensions = ['sphinx.ext.autodoc'] | extensions = ['sphinx.ext.autodoc', 'sphinx.ext.todo'] | ||||||
|  |  | ||||||
| # Add any paths that contain templates here, relative to this directory. | # Add any paths that contain templates here, relative to this directory. | ||||||
| templates_path = ['.templates'] | templates_path = ['_templates'] | ||||||
|  |  | ||||||
| # The suffix of source filenames. | # The suffix of source filenames. | ||||||
| source_suffix = '.rst' | source_suffix = '.rst' | ||||||
| @@ -38,7 +38,7 @@ master_doc = 'index' | |||||||
|  |  | ||||||
| # General information about the project. | # General information about the project. | ||||||
| project = u'MongoEngine' | project = u'MongoEngine' | ||||||
| copyright = u'2009, Harry Marr' | copyright = u'2009-2012, MongoEngine Authors' | ||||||
|  |  | ||||||
| # The version info for the project you're documenting, acts as replacement for | # The version info for the project you're documenting, acts as replacement for | ||||||
| # |version| and |release|, also used in various other places throughout the | # |version| and |release|, also used in various other places throughout the | ||||||
| @@ -121,7 +121,7 @@ html_theme_path = ['_themes'] | |||||||
| # Add any paths that contain custom static files (such as style sheets) here, | # Add any paths that contain custom static files (such as style sheets) here, | ||||||
| # relative to this directory. They are copied after the builtin static files, | # relative to this directory. They are copied after the builtin static files, | ||||||
| # so a file named "default.css" will overwrite the builtin "default.css". | # so a file named "default.css" will overwrite the builtin "default.css". | ||||||
| html_static_path = ['_static'] | #html_static_path = ['_static'] | ||||||
|  |  | ||||||
| # If not '', a 'Last updated on:' timestamp is inserted at every page bottom, | # If not '', a 'Last updated on:' timestamp is inserted at every page bottom, | ||||||
| # using the given strftime format. | # using the given strftime format. | ||||||
|   | |||||||
							
								
								
									
										88
									
								
								docs/django.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										88
									
								
								docs/django.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,88 @@ | |||||||
|  | ============================= | ||||||
|  | Using MongoEngine with Django | ||||||
|  | ============================= | ||||||
|  |  | ||||||
|  | Connecting | ||||||
|  | ========== | ||||||
|  | In your **settings.py** file, ignore the standard database settings (unless you | ||||||
|  | also plan to use the ORM in your project), and instead call  | ||||||
|  | :func:`~mongoengine.connect` somewhere in the settings module. | ||||||
|  |  | ||||||
|  | Authentication | ||||||
|  | ============== | ||||||
|  | MongoEngine includes a Django authentication backend, which uses MongoDB. The | ||||||
|  | :class:`~mongoengine.django.auth.User` model is a MongoEngine  | ||||||
|  | :class:`~mongoengine.Document`, but implements most of the methods and  | ||||||
|  | attributes that the standard Django :class:`User` model does - so the two are | ||||||
|  | moderately compatible. Using this backend will allow you to store users in  | ||||||
|  | MongoDB but still use many of the Django authentication infrastucture (such as | ||||||
|  | the :func:`login_required` decorator and the :func:`authenticate` function). To | ||||||
|  | enable the MongoEngine auth backend, add the following to you **settings.py** | ||||||
|  | file:: | ||||||
|  |  | ||||||
|  |     AUTHENTICATION_BACKENDS = ( | ||||||
|  |         'mongoengine.django.auth.MongoEngineBackend', | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  | The :mod:`~mongoengine.django.auth` module also contains a  | ||||||
|  | :func:`~mongoengine.django.auth.get_user` helper function, that takes a user's | ||||||
|  | :attr:`id` and returns a :class:`~mongoengine.django.auth.User` object. | ||||||
|  |  | ||||||
|  | .. versionadded:: 0.1.3 | ||||||
|  |  | ||||||
|  | Sessions | ||||||
|  | ======== | ||||||
|  | Django allows the use of different backend stores for its sessions. MongoEngine | ||||||
|  | provides a MongoDB-based session backend for Django, which allows you to use | ||||||
|  | sessions in you Django application with just MongoDB. To enable the MongoEngine | ||||||
|  | session backend, ensure that your settings module has | ||||||
|  | ``'django.contrib.sessions.middleware.SessionMiddleware'`` in the | ||||||
|  | ``MIDDLEWARE_CLASSES`` field  and ``'django.contrib.sessions'`` in your | ||||||
|  | ``INSTALLED_APPS``. From there, all you need to do is add the following line | ||||||
|  | into you settings module:: | ||||||
|  |  | ||||||
|  |     SESSION_ENGINE = 'mongoengine.django.sessions' | ||||||
|  |  | ||||||
|  | .. versionadded:: 0.2.1 | ||||||
|  |  | ||||||
|  | Storage | ||||||
|  | ======= | ||||||
|  | With MongoEngine's support for GridFS via the :class:`~mongoengine.FileField`, | ||||||
|  | it is useful to have a Django file storage backend that wraps this. The new | ||||||
|  | storage module is called :class:`~mongoengine.django.storage.GridFSStorage`.  | ||||||
|  | Using it is very similar to using the default FileSystemStorage.:: | ||||||
|  |      | ||||||
|  |     from mongoengine.django.storage import GridFSStorage | ||||||
|  |     fs = GridFSStorage() | ||||||
|  |  | ||||||
|  |     filename = fs.save('hello.txt', 'Hello, World!') | ||||||
|  |  | ||||||
|  | All of the `Django Storage API methods | ||||||
|  | <http://docs.djangoproject.com/en/dev/ref/files/storage/>`_ have been | ||||||
|  | implemented except :func:`path`. If the filename provided already exists, an | ||||||
|  | underscore and a number (before # the file extension, if one exists) will be | ||||||
|  | appended to the filename until the generated filename doesn't exist. The | ||||||
|  | :func:`save` method will return the new filename.:: | ||||||
|  |  | ||||||
|  |     >>> fs.exists('hello.txt') | ||||||
|  |     True | ||||||
|  |     >>> fs.open('hello.txt').read() | ||||||
|  |     'Hello, World!' | ||||||
|  |     >>> fs.size('hello.txt') | ||||||
|  |     13 | ||||||
|  |     >>> fs.url('hello.txt') | ||||||
|  |     'http://your_media_url/hello.txt' | ||||||
|  |     >>> fs.open('hello.txt').name | ||||||
|  |     'hello.txt' | ||||||
|  |     >>> fs.listdir() | ||||||
|  |     ([], [u'hello.txt']) | ||||||
|  |  | ||||||
|  | All files will be saved and retrieved in GridFS via the :class::`FileDocument` | ||||||
|  | document, allowing easy access to the files without the GridFSStorage | ||||||
|  | backend.:: | ||||||
|  |  | ||||||
|  |     >>> from mongoengine.django.storage import FileDocument | ||||||
|  |     >>> FileDocument.objects() | ||||||
|  |     [<FileDocument: FileDocument object>] | ||||||
|  |  | ||||||
|  | .. versionadded:: 0.4 | ||||||
							
								
								
									
										60
									
								
								docs/guide/connecting.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										60
									
								
								docs/guide/connecting.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,60 @@ | |||||||
|  | .. _guide-connecting: | ||||||
|  |  | ||||||
|  | ===================== | ||||||
|  | Connecting to MongoDB | ||||||
|  | ===================== | ||||||
|  |  | ||||||
|  | To connect to a running instance of :program:`mongod`, use the | ||||||
|  | :func:`~mongoengine.connect` function. The first argument is the name of the | ||||||
|  | database to connect to. If the database does not exist, it will be created. If | ||||||
|  | the database requires authentication, :attr:`username` and :attr:`password` | ||||||
|  | arguments may be provided:: | ||||||
|  |  | ||||||
|  |     from mongoengine import connect | ||||||
|  |     connect('project1', username='webapp', password='pwd123') | ||||||
|  |  | ||||||
|  | By default, MongoEngine assumes that the :program:`mongod` instance is running | ||||||
|  | on **localhost** on port **27017**. If MongoDB is running elsewhere, you may | ||||||
|  | provide :attr:`host` and :attr:`port` arguments to | ||||||
|  | :func:`~mongoengine.connect`:: | ||||||
|  |  | ||||||
|  |     connect('project1', host='192.168.1.35', port=12345) | ||||||
|  |  | ||||||
|  | Uri style connections are also supported as long as you include the database | ||||||
|  | name - just supply the uri as the :attr:`host` to | ||||||
|  | :func:`~mongoengine.connect`:: | ||||||
|  |  | ||||||
|  |     connect('project1', host='mongodb://localhost/database_name') | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Multiple Databases | ||||||
|  | ================== | ||||||
|  |  | ||||||
|  | Multiple database support was added in MongoEngine 0.6. To use multiple | ||||||
|  | databases you can use :func:`~mongoengine.connect` and provide an `alias` name | ||||||
|  | for the connection - if no `alias` is provided then "default" is used. | ||||||
|  |  | ||||||
|  | In the background this uses :func:`~mongoengine.register_connection` to | ||||||
|  | store the data and you can register all aliases up front if required. | ||||||
|  |  | ||||||
|  | Individual documents can also support multiple databases by providing a | ||||||
|  | `db_alias` in their meta data.  This allows :class:`~pymongo.dbref.DBRef` objects | ||||||
|  | to point across databases and collections.  Below is an example schema, using | ||||||
|  | 3 different databases to store data:: | ||||||
|  |  | ||||||
|  |         class User(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |             meta = {"db_alias": "user-db"} | ||||||
|  |  | ||||||
|  |         class Book(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |             meta = {"db_alias": "book-db"} | ||||||
|  |  | ||||||
|  |         class AuthorBooks(Document): | ||||||
|  |             author = ReferenceField(User) | ||||||
|  |             book = ReferenceField(Book) | ||||||
|  |  | ||||||
|  |             meta = {"db_alias": "users-books-db"} | ||||||
							
								
								
									
										563
									
								
								docs/guide/defining-documents.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										563
									
								
								docs/guide/defining-documents.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,563 @@ | |||||||
|  | ================== | ||||||
|  | Defining documents | ||||||
|  | ================== | ||||||
|  | In MongoDB, a **document** is roughly equivalent to a **row** in an RDBMS. When | ||||||
|  | working with relational databases, rows are stored in **tables**, which have a | ||||||
|  | strict **schema** that the rows follow. MongoDB stores documents in | ||||||
|  | **collections** rather than tables - the principle difference is that no schema | ||||||
|  | is enforced at a database level. | ||||||
|  |  | ||||||
|  | Defining a document's schema | ||||||
|  | ============================ | ||||||
|  | MongoEngine allows you to define schemata for documents as this helps to reduce | ||||||
|  | coding errors, and allows for utility methods to be defined on fields which may | ||||||
|  | be present. | ||||||
|  |  | ||||||
|  | To define a schema for a document, create a class that inherits from | ||||||
|  | :class:`~mongoengine.Document`. Fields are specified by adding **field | ||||||
|  | objects** as class attributes to the document class:: | ||||||
|  |  | ||||||
|  |     from mongoengine import * | ||||||
|  |     import datetime | ||||||
|  |  | ||||||
|  |     class Page(Document): | ||||||
|  |         title = StringField(max_length=200, required=True) | ||||||
|  |         date_modified = DateTimeField(default=datetime.datetime.now) | ||||||
|  |  | ||||||
|  | Dynamic document schemas | ||||||
|  | ======================== | ||||||
|  | One of the benefits of MongoDb is dynamic schemas for a collection, whilst data | ||||||
|  | should be planned and organised (after all explicit is better than implicit!) | ||||||
|  | there are scenarios where having dynamic / expando style documents is desirable. | ||||||
|  |  | ||||||
|  | :class:`~mongoengine.DynamicDocument` documents work in the same way as | ||||||
|  | :class:`~mongoengine.Document` but any data / attributes set to them will also | ||||||
|  | be saved :: | ||||||
|  |  | ||||||
|  |     from mongoengine import * | ||||||
|  |  | ||||||
|  |     class Page(DynamicDocument): | ||||||
|  |         title = StringField(max_length=200, required=True) | ||||||
|  |  | ||||||
|  |     # Create a new page and add tags | ||||||
|  |     >>> page = Page(title='Using MongoEngine') | ||||||
|  |     >>> page.tags = ['mongodb', 'mongoengine'] | ||||||
|  |     >>> page.save() | ||||||
|  |  | ||||||
|  |     >>> Page.objects(tags='mongoengine').count() | ||||||
|  |     >>> 1 | ||||||
|  |  | ||||||
|  | ..note:: | ||||||
|  |  | ||||||
|  |    There is one caveat on Dynamic Documents: fields cannot start with `_` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Fields | ||||||
|  | ====== | ||||||
|  | By default, fields are not required. To make a field mandatory, set the | ||||||
|  | :attr:`required` keyword argument of a field to ``True``. Fields also may have | ||||||
|  | validation constraints available (such as :attr:`max_length` in the example | ||||||
|  | above). Fields may also take default values, which will be used if a value is | ||||||
|  | not provided. Default values may optionally be a callable, which will be called | ||||||
|  | to retrieve the value (such as in the above example). The field types available | ||||||
|  | are as follows: | ||||||
|  |  | ||||||
|  | * :class:`~mongoengine.StringField` | ||||||
|  | * :class:`~mongoengine.URLField` | ||||||
|  | * :class:`~mongoengine.EmailField` | ||||||
|  | * :class:`~mongoengine.IntField` | ||||||
|  | * :class:`~mongoengine.FloatField` | ||||||
|  | * :class:`~mongoengine.DecimalField` | ||||||
|  | * :class:`~mongoengine.DateTimeField` | ||||||
|  | * :class:`~mongoengine.ComplexDateTimeField` | ||||||
|  | * :class:`~mongoengine.ListField` | ||||||
|  | * :class:`~mongoengine.SortedListField` | ||||||
|  | * :class:`~mongoengine.DictField` | ||||||
|  | * :class:`~mongoengine.MapField` | ||||||
|  | * :class:`~mongoengine.ObjectIdField` | ||||||
|  | * :class:`~mongoengine.ReferenceField` | ||||||
|  | * :class:`~mongoengine.GenericReferenceField` | ||||||
|  | * :class:`~mongoengine.EmbeddedDocumentField` | ||||||
|  | * :class:`~mongoengine.GenericEmbeddedDocumentField` | ||||||
|  | * :class:`~mongoengine.BooleanField` | ||||||
|  | * :class:`~mongoengine.FileField` | ||||||
|  | * :class:`~mongoengine.BinaryField` | ||||||
|  | * :class:`~mongoengine.GeoPointField` | ||||||
|  | * :class:`~mongoengine.SequenceField` | ||||||
|  |  | ||||||
|  | Field arguments | ||||||
|  | --------------- | ||||||
|  | Each field type can be customized by keyword arguments.  The following keyword | ||||||
|  | arguments can be set on all fields: | ||||||
|  |  | ||||||
|  | :attr:`db_field` (Default: None) | ||||||
|  |     The MongoDB field name. | ||||||
|  |  | ||||||
|  | :attr:`name` (Default: None) | ||||||
|  |     The mongoengine field name. | ||||||
|  |  | ||||||
|  | :attr:`required` (Default: False) | ||||||
|  |     If set to True and the field is not set on the document instance, a | ||||||
|  |     :class:`~mongoengine.base.ValidationError` will be raised when the document is | ||||||
|  |     validated. | ||||||
|  |  | ||||||
|  | :attr:`default` (Default: None) | ||||||
|  |     A value to use when no value is set for this field. | ||||||
|  |  | ||||||
|  |     The definion of default parameters follow `the general rules on Python | ||||||
|  |     <http://docs.python.org/reference/compound_stmts.html#function-definitions>`__, | ||||||
|  |     which means that some care should be taken when dealing with default mutable objects | ||||||
|  |     (like in :class:`~mongoengine.ListField` or :class:`~mongoengine.DictField`):: | ||||||
|  |  | ||||||
|  |         class ExampleFirst(Document): | ||||||
|  |             # Default an empty list | ||||||
|  |             values = ListField(IntField(), default=list) | ||||||
|  |  | ||||||
|  |         class ExampleSecond(Document): | ||||||
|  |             # Default a set of values | ||||||
|  |             values = ListField(IntField(), default=lambda: [1,2,3]) | ||||||
|  |  | ||||||
|  |         class ExampleDangerous(Document): | ||||||
|  |             # This can make an .append call to  add values to the default (and all the following objects), | ||||||
|  |             # instead to just an object | ||||||
|  |             values = ListField(IntField(), default=[1,2,3]) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | :attr:`unique` (Default: False) | ||||||
|  |     When True, no documents in the collection will have the same value for this | ||||||
|  |     field. | ||||||
|  |  | ||||||
|  | :attr:`unique_with` (Default: None) | ||||||
|  |     A field name (or list of field names) that when taken together with this | ||||||
|  |     field, will not have two documents in the collection with the same value. | ||||||
|  |  | ||||||
|  | :attr:`primary_key` (Default: False) | ||||||
|  |     When True, use this field as a primary key for the collection. | ||||||
|  |  | ||||||
|  | :attr:`choices` (Default: None) | ||||||
|  |     An iterable (e.g. a list or tuple) of choices to which the value of this | ||||||
|  |     field should be limited. | ||||||
|  |  | ||||||
|  |     Can be either be a nested tuples of value (stored in mongo) and a | ||||||
|  |     human readable key :: | ||||||
|  |  | ||||||
|  |         SIZE = (('S', 'Small'), | ||||||
|  |                 ('M', 'Medium'), | ||||||
|  |                 ('L', 'Large'), | ||||||
|  |                 ('XL', 'Extra Large'), | ||||||
|  |                 ('XXL', 'Extra Extra Large')) | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         class Shirt(Document): | ||||||
|  |             size = StringField(max_length=3, choices=SIZE) | ||||||
|  |  | ||||||
|  |     Or a flat iterable just containing values :: | ||||||
|  |  | ||||||
|  |         SIZE = ('S', 'M', 'L', 'XL', 'XXL') | ||||||
|  |  | ||||||
|  |         class Shirt(Document): | ||||||
|  |             size = StringField(max_length=3, choices=SIZE) | ||||||
|  |  | ||||||
|  | :attr:`help_text` (Default: None) | ||||||
|  |     Optional help text to output with the field - used by form libraries | ||||||
|  |  | ||||||
|  | :attr:`verbose_name` (Default: None) | ||||||
|  |     Optional human-readable name for the field - used by form libraries | ||||||
|  |  | ||||||
|  |  | ||||||
|  | List fields | ||||||
|  | ----------- | ||||||
|  | MongoDB allows the storage of lists of items. To add a list of items to a | ||||||
|  | :class:`~mongoengine.Document`, use the :class:`~mongoengine.ListField` field | ||||||
|  | type. :class:`~mongoengine.ListField` takes another field object as its first | ||||||
|  | argument, which specifies which type elements may be stored within the list:: | ||||||
|  |  | ||||||
|  |     class Page(Document): | ||||||
|  |         tags = ListField(StringField(max_length=50)) | ||||||
|  |  | ||||||
|  | Embedded documents | ||||||
|  | ------------------ | ||||||
|  | MongoDB has the ability to embed documents within other documents. Schemata may | ||||||
|  | be defined for these embedded documents, just as they may be for regular | ||||||
|  | documents. To create an embedded document, just define a document as usual, but | ||||||
|  | inherit from :class:`~mongoengine.EmbeddedDocument` rather than | ||||||
|  | :class:`~mongoengine.Document`:: | ||||||
|  |  | ||||||
|  |     class Comment(EmbeddedDocument): | ||||||
|  |         content = StringField() | ||||||
|  |  | ||||||
|  | To embed the document within another document, use the | ||||||
|  | :class:`~mongoengine.EmbeddedDocumentField` field type, providing the embedded | ||||||
|  | document class as the first argument:: | ||||||
|  |  | ||||||
|  |     class Page(Document): | ||||||
|  |         comments = ListField(EmbeddedDocumentField(Comment)) | ||||||
|  |  | ||||||
|  |     comment1 = Comment(content='Good work!') | ||||||
|  |     comment2 = Comment(content='Nice article!') | ||||||
|  |     page = Page(comments=[comment1, comment2]) | ||||||
|  |  | ||||||
|  | Dictionary Fields | ||||||
|  | ----------------- | ||||||
|  | Often, an embedded document may be used instead of a dictionary -- generally | ||||||
|  | this is recommended as dictionaries don't support validation or custom field | ||||||
|  | types. However, sometimes you will not know the structure of what you want to | ||||||
|  | store; in this situation a :class:`~mongoengine.DictField` is appropriate:: | ||||||
|  |  | ||||||
|  |     class SurveyResponse(Document): | ||||||
|  |         date = DateTimeField() | ||||||
|  |         user = ReferenceField(User) | ||||||
|  |         answers = DictField() | ||||||
|  |  | ||||||
|  |     survey_response = SurveyResponse(date=datetime.now(), user=request.user) | ||||||
|  |     response_form = ResponseForm(request.POST) | ||||||
|  |     survey_response.answers = response_form.cleaned_data() | ||||||
|  |     survey_response.save() | ||||||
|  |  | ||||||
|  | Dictionaries can store complex data, other dictionaries, lists, references to | ||||||
|  | other objects, so are the most flexible field type available. | ||||||
|  |  | ||||||
|  | Reference fields | ||||||
|  | ---------------- | ||||||
|  | References may be stored to other documents in the database using the | ||||||
|  | :class:`~mongoengine.ReferenceField`. Pass in another document class as the | ||||||
|  | first argument to the constructor, then simply assign document objects to the | ||||||
|  | field:: | ||||||
|  |  | ||||||
|  |     class User(Document): | ||||||
|  |         name = StringField() | ||||||
|  |  | ||||||
|  |     class Page(Document): | ||||||
|  |         content = StringField() | ||||||
|  |         author = ReferenceField(User) | ||||||
|  |  | ||||||
|  |     john = User(name="John Smith") | ||||||
|  |     john.save() | ||||||
|  |  | ||||||
|  |     post = Page(content="Test Page") | ||||||
|  |     post.author = john | ||||||
|  |     post.save() | ||||||
|  |  | ||||||
|  | The :class:`User` object is automatically turned into a reference behind the | ||||||
|  | scenes, and dereferenced when the :class:`Page` object is retrieved. | ||||||
|  |  | ||||||
|  | To add a :class:`~mongoengine.ReferenceField` that references the document | ||||||
|  | being defined, use the string ``'self'`` in place of the document class as the | ||||||
|  | argument to :class:`~mongoengine.ReferenceField`'s constructor. To reference a | ||||||
|  | document that has not yet been defined, use the name of the undefined document | ||||||
|  | as the constructor's argument:: | ||||||
|  |  | ||||||
|  |     class Employee(Document): | ||||||
|  |         name = StringField() | ||||||
|  |         boss = ReferenceField('self') | ||||||
|  |         profile_page = ReferenceField('ProfilePage') | ||||||
|  |  | ||||||
|  |     class ProfilePage(Document): | ||||||
|  |         content = StringField() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Dealing with deletion of referred documents | ||||||
|  | ''''''''''''''''''''''''''''''''''''''''''' | ||||||
|  | By default, MongoDB doesn't check the integrity of your data, so deleting | ||||||
|  | documents that other documents still hold references to will lead to consistency | ||||||
|  | issues.  Mongoengine's :class:`ReferenceField` adds some functionality to | ||||||
|  | safeguard against these kinds of database integrity problems, providing each | ||||||
|  | reference with a delete rule specification.  A delete rule is specified by | ||||||
|  | supplying the :attr:`reverse_delete_rule` attributes on the | ||||||
|  | :class:`ReferenceField` definition, like this:: | ||||||
|  |  | ||||||
|  |     class Employee(Document): | ||||||
|  |         ... | ||||||
|  |         profile_page = ReferenceField('ProfilePage', reverse_delete_rule=mongoengine.NULLIFY) | ||||||
|  |  | ||||||
|  | The declaration in this example means that when an :class:`Employee` object is | ||||||
|  | removed, the :class:`ProfilePage` that belongs to that employee is removed as | ||||||
|  | well.  If a whole batch of employees is removed, all profile pages that are | ||||||
|  | linked are removed as well. | ||||||
|  |  | ||||||
|  | Its value can take any of the following constants: | ||||||
|  |  | ||||||
|  | :const:`mongoengine.DO_NOTHING` | ||||||
|  |   This is the default and won't do anything.  Deletes are fast, but may cause | ||||||
|  |   database inconsistency or dangling references. | ||||||
|  | :const:`mongoengine.DENY` | ||||||
|  |   Deletion is denied if there still exist references to the object being | ||||||
|  |   deleted. | ||||||
|  | :const:`mongoengine.NULLIFY` | ||||||
|  |   Any object's fields still referring to the object being deleted are removed | ||||||
|  |   (using MongoDB's "unset" operation), effectively nullifying the relationship. | ||||||
|  | :const:`mongoengine.CASCADE` | ||||||
|  |   Any object containing fields that are refererring to the object being deleted | ||||||
|  |   are deleted first. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. warning:: | ||||||
|  |    A safety note on setting up these delete rules!  Since the delete rules are | ||||||
|  |    not recorded on the database level by MongoDB itself, but instead at runtime, | ||||||
|  |    in-memory, by the MongoEngine module, it is of the upmost importance | ||||||
|  |    that the module that declares the relationship is loaded **BEFORE** the | ||||||
|  |    delete is invoked. | ||||||
|  |  | ||||||
|  |    If, for example, the :class:`Employee` object lives in the | ||||||
|  |    :mod:`payroll` app, and the :class:`ProfilePage` in the :mod:`people` | ||||||
|  |    app, it is extremely important that the :mod:`people` app is loaded | ||||||
|  |    before any employee is removed, because otherwise, MongoEngine could | ||||||
|  |    never know this relationship exists. | ||||||
|  |  | ||||||
|  |    In Django, be sure to put all apps that have such delete rule declarations in | ||||||
|  |    their :file:`models.py` in the :const:`INSTALLED_APPS` tuple. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Generic reference fields | ||||||
|  | '''''''''''''''''''''''' | ||||||
|  | A second kind of reference field also exists, | ||||||
|  | :class:`~mongoengine.GenericReferenceField`. This allows you to reference any | ||||||
|  | kind of :class:`~mongoengine.Document`, and hence doesn't take a | ||||||
|  | :class:`~mongoengine.Document` subclass as a constructor argument:: | ||||||
|  |  | ||||||
|  |     class Link(Document): | ||||||
|  |         url = StringField() | ||||||
|  |  | ||||||
|  |     class Post(Document): | ||||||
|  |         title = StringField() | ||||||
|  |  | ||||||
|  |     class Bookmark(Document): | ||||||
|  |         bookmark_object = GenericReferenceField() | ||||||
|  |  | ||||||
|  |     link = Link(url='http://hmarr.com/mongoengine/') | ||||||
|  |     link.save() | ||||||
|  |  | ||||||
|  |     post = Post(title='Using MongoEngine') | ||||||
|  |     post.save() | ||||||
|  |  | ||||||
|  |     Bookmark(bookmark_object=link).save() | ||||||
|  |     Bookmark(bookmark_object=post).save() | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |  | ||||||
|  |    Using :class:`~mongoengine.GenericReferenceField`\ s is slightly less | ||||||
|  |    efficient than the standard :class:`~mongoengine.ReferenceField`\ s, so if | ||||||
|  |    you will only be referencing one document type, prefer the standard | ||||||
|  |    :class:`~mongoengine.ReferenceField`. | ||||||
|  |  | ||||||
|  | Uniqueness constraints | ||||||
|  | ---------------------- | ||||||
|  | MongoEngine allows you to specify that a field should be unique across a | ||||||
|  | collection by providing ``unique=True`` to a :class:`~mongoengine.Field`\ 's | ||||||
|  | constructor. If you try to save a document that has the same value for a unique | ||||||
|  | field as a document that is already in the database, a | ||||||
|  | :class:`~mongoengine.OperationError` will be raised. You may also specify | ||||||
|  | multi-field uniqueness constraints by using :attr:`unique_with`, which may be | ||||||
|  | either a single field name, or a list or tuple of field names:: | ||||||
|  |  | ||||||
|  |     class User(Document): | ||||||
|  |         username = StringField(unique=True) | ||||||
|  |         first_name = StringField() | ||||||
|  |         last_name = StringField(unique_with='first_name') | ||||||
|  |  | ||||||
|  | Skipping Document validation on save | ||||||
|  | ------------------------------------ | ||||||
|  | You can also skip the whole document validation process by setting | ||||||
|  | ``validate=False`` when caling the :meth:`~mongoengine.document.Document.save` | ||||||
|  | method:: | ||||||
|  |  | ||||||
|  |     class Recipient(Document): | ||||||
|  |         name = StringField() | ||||||
|  |         email = EmailField() | ||||||
|  |  | ||||||
|  |     recipient = Recipient(name='admin', email='root@localhost') | ||||||
|  |     recipient.save()               # will raise a ValidationError while | ||||||
|  |     recipient.save(validate=False) # won't | ||||||
|  |  | ||||||
|  | Document collections | ||||||
|  | ==================== | ||||||
|  | Document classes that inherit **directly** from :class:`~mongoengine.Document` | ||||||
|  | will have their own **collection** in the database. The name of the collection | ||||||
|  | is by default the name of the class, coverted to lowercase (so in the example | ||||||
|  | above, the collection would be called `page`). If you need to change the name | ||||||
|  | of the collection (e.g. to use MongoEngine with an existing database), then | ||||||
|  | create a class dictionary attribute called :attr:`meta` on your document, and | ||||||
|  | set :attr:`collection` to the name of the collection that you want your | ||||||
|  | document class to use:: | ||||||
|  |  | ||||||
|  |     class Page(Document): | ||||||
|  |         title = StringField(max_length=200, required=True) | ||||||
|  |         meta = {'collection': 'cmsPage'} | ||||||
|  |  | ||||||
|  | Capped collections | ||||||
|  | ------------------ | ||||||
|  | A :class:`~mongoengine.Document` may use a **Capped Collection** by specifying | ||||||
|  | :attr:`max_documents` and :attr:`max_size` in the :attr:`meta` dictionary. | ||||||
|  | :attr:`max_documents` is the maximum number of documents that is allowed to be | ||||||
|  | stored in the collection, and :attr:`max_size` is the maximum size of the | ||||||
|  | collection in bytes. If :attr:`max_size` is not specified and | ||||||
|  | :attr:`max_documents` is, :attr:`max_size` defaults to 10000000 bytes (10MB). | ||||||
|  | The following example shows a :class:`Log` document that will be limited to | ||||||
|  | 1000 entries and 2MB of disk space:: | ||||||
|  |  | ||||||
|  |     class Log(Document): | ||||||
|  |         ip_address = StringField() | ||||||
|  |         meta = {'max_documents': 1000, 'max_size': 2000000} | ||||||
|  |  | ||||||
|  | Indexes | ||||||
|  | ======= | ||||||
|  | You can specify indexes on collections to make querying faster. This is done | ||||||
|  | by creating a list of index specifications called :attr:`indexes` in the | ||||||
|  | :attr:`~mongoengine.Document.meta` dictionary, where an index specification may | ||||||
|  | either be a single field name, a tuple containing multiple field names, or a | ||||||
|  | dictionary containing a full index definition. A direction may be specified on | ||||||
|  | fields by prefixing the field name with a **+** or a **-** sign. Note that | ||||||
|  | direction only matters on multi-field indexes. :: | ||||||
|  |  | ||||||
|  |     class Page(Document): | ||||||
|  |         title = StringField() | ||||||
|  |         rating = StringField() | ||||||
|  |         meta = { | ||||||
|  |             'indexes': ['title', ('title', '-rating')] | ||||||
|  |         } | ||||||
|  |  | ||||||
|  | If a dictionary is passed then the following options are available: | ||||||
|  |  | ||||||
|  | :attr:`fields` (Default: None) | ||||||
|  |     The fields to index. Specified in the same format as described above. | ||||||
|  |  | ||||||
|  | :attr:`types` (Default: True) | ||||||
|  |     Whether the index should have the :attr:`_types` field added automatically | ||||||
|  |     to the start of the index. | ||||||
|  |  | ||||||
|  | :attr:`sparse` (Default: False) | ||||||
|  |     Whether the index should be sparse. | ||||||
|  |  | ||||||
|  | :attr:`unique` (Default: False) | ||||||
|  |     Whether the index should be sparse. | ||||||
|  |  | ||||||
|  | .. warning:: | ||||||
|  |  | ||||||
|  |  | ||||||
|  |    Inheritance adds extra indices. | ||||||
|  |    If don't need inheritance for a document turn inheritance off - see :ref:`document-inheritance`. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Geospatial indexes | ||||||
|  | --------------------------- | ||||||
|  | Geospatial indexes will be automatically created for all | ||||||
|  | :class:`~mongoengine.GeoPointField`\ s | ||||||
|  |  | ||||||
|  | It is also possible to explicitly define geospatial indexes. This is | ||||||
|  | useful if you need to define a geospatial index on a subfield of a | ||||||
|  | :class:`~mongoengine.DictField` or a custom field that contains a | ||||||
|  | point. To create a geospatial index you must prefix the field with the | ||||||
|  | ***** sign. :: | ||||||
|  |  | ||||||
|  |     class Place(Document): | ||||||
|  |         location = DictField() | ||||||
|  |         meta = { | ||||||
|  |             'indexes': [ | ||||||
|  |                 '*location.point', | ||||||
|  |             ], | ||||||
|  |         } | ||||||
|  |  | ||||||
|  | Ordering | ||||||
|  | ======== | ||||||
|  | A default ordering can be specified for your | ||||||
|  | :class:`~mongoengine.queryset.QuerySet` using the :attr:`ordering` attribute of | ||||||
|  | :attr:`~mongoengine.Document.meta`.  Ordering will be applied when the | ||||||
|  | :class:`~mongoengine.queryset.QuerySet` is created, and can be overridden by | ||||||
|  | subsequent calls to :meth:`~mongoengine.queryset.QuerySet.order_by`. :: | ||||||
|  |  | ||||||
|  |     from datetime import datetime | ||||||
|  |  | ||||||
|  |     class BlogPost(Document): | ||||||
|  |         title = StringField() | ||||||
|  |         published_date = DateTimeField() | ||||||
|  |  | ||||||
|  |         meta = { | ||||||
|  |             'ordering': ['-published_date'] | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     blog_post_1 = BlogPost(title="Blog Post #1") | ||||||
|  |     blog_post_1.published_date = datetime(2010, 1, 5, 0, 0 ,0) | ||||||
|  |  | ||||||
|  |     blog_post_2 = BlogPost(title="Blog Post #2") | ||||||
|  |     blog_post_2.published_date = datetime(2010, 1, 6, 0, 0 ,0) | ||||||
|  |  | ||||||
|  |     blog_post_3 = BlogPost(title="Blog Post #3") | ||||||
|  |     blog_post_3.published_date = datetime(2010, 1, 7, 0, 0 ,0) | ||||||
|  |  | ||||||
|  |     blog_post_1.save() | ||||||
|  |     blog_post_2.save() | ||||||
|  |     blog_post_3.save() | ||||||
|  |  | ||||||
|  |     # get the "first" BlogPost using default ordering | ||||||
|  |     # from BlogPost.meta.ordering | ||||||
|  |     latest_post = BlogPost.objects.first() | ||||||
|  |     assert latest_post.title == "Blog Post #3" | ||||||
|  |  | ||||||
|  |     # override default ordering, order BlogPosts by "published_date" | ||||||
|  |     first_post = BlogPost.objects.order_by("+published_date").first() | ||||||
|  |     assert first_post.title == "Blog Post #1" | ||||||
|  |  | ||||||
|  | Shard keys | ||||||
|  | ========== | ||||||
|  |  | ||||||
|  | If your collection is sharded, then you need to specify the shard key as a tuple, | ||||||
|  | using the :attr:`shard_key` attribute of :attr:`-mongoengine.Document.meta`. | ||||||
|  | This ensures that the shard key is sent with the query when calling the | ||||||
|  | :meth:`~mongoengine.document.Document.save` or | ||||||
|  | :meth:`~mongoengine.document.Document.update` method on an existing | ||||||
|  | :class:`-mongoengine.Document` instance:: | ||||||
|  |  | ||||||
|  |     class LogEntry(Document): | ||||||
|  |         machine = StringField() | ||||||
|  |         app = StringField() | ||||||
|  |         timestamp = DateTimeField() | ||||||
|  |         data = StringField() | ||||||
|  |  | ||||||
|  |         meta = { | ||||||
|  |             'shard_key': ('machine', 'timestamp',) | ||||||
|  |         } | ||||||
|  |  | ||||||
|  | .. _document-inheritance: | ||||||
|  |  | ||||||
|  | Document inheritance | ||||||
|  | ==================== | ||||||
|  |  | ||||||
|  | To create a specialised type of a :class:`~mongoengine.Document` you have | ||||||
|  | defined, you may subclass it and add any extra fields or methods you may need. | ||||||
|  | As this is new class is not a direct subclass of | ||||||
|  | :class:`~mongoengine.Document`, it will not be stored in its own collection; it | ||||||
|  | will use the same collection as its superclass uses. This allows for more | ||||||
|  | convenient and efficient retrieval of related documents:: | ||||||
|  |  | ||||||
|  |     # Stored in a collection named 'page' | ||||||
|  |     class Page(Document): | ||||||
|  |         title = StringField(max_length=200, required=True) | ||||||
|  |  | ||||||
|  |         meta = {'allow_inheritance': True} | ||||||
|  |  | ||||||
|  |     # Also stored in the collection named 'page' | ||||||
|  |     class DatedPage(Page): | ||||||
|  |         date = DateTimeField() | ||||||
|  |  | ||||||
|  | .. note:: From 0.7 onwards you must declare `allow_inheritance` in the document meta. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Working with existing data | ||||||
|  | -------------------------- | ||||||
|  | To enable correct retrieval of documents involved in this kind of heirarchy, | ||||||
|  | two extra attributes are stored on each document in the database: :attr:`_cls` | ||||||
|  | and :attr:`_types`. These are hidden from the user through the MongoEngine | ||||||
|  | interface, but may not be present if you are trying to use MongoEngine with | ||||||
|  | an existing database. For this reason, you may disable this inheritance | ||||||
|  | mechansim, removing the dependency of :attr:`_cls` and :attr:`_types`, enabling | ||||||
|  | you to work with existing databases. To disable inheritance on a document | ||||||
|  | class, set :attr:`allow_inheritance` to ``False`` in the :attr:`meta` | ||||||
|  | dictionary:: | ||||||
|  |  | ||||||
|  |     # Will work with data in an existing collection named 'cmsPage' | ||||||
|  |     class Page(Document): | ||||||
|  |         title = StringField(max_length=200, required=True) | ||||||
|  |         meta = { | ||||||
|  |             'collection': 'cmsPage', | ||||||
|  |             'allow_inheritance': False, | ||||||
|  |         } | ||||||
							
								
								
									
										95
									
								
								docs/guide/document-instances.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								docs/guide/document-instances.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | |||||||
|  | =================== | ||||||
|  | Documents instances | ||||||
|  | =================== | ||||||
|  | To create a new document object, create an instance of the relevant document | ||||||
|  | class, providing values for its fields as its constructor keyword arguments. | ||||||
|  | You may provide values for any of the fields on the document:: | ||||||
|  |  | ||||||
|  |     >>> page = Page(title="Test Page") | ||||||
|  |     >>> page.title | ||||||
|  |     'Test Page' | ||||||
|  |  | ||||||
|  | You may also assign values to the document's fields using standard object | ||||||
|  | attribute syntax:: | ||||||
|  |  | ||||||
|  |     >>> page.title = "Example Page" | ||||||
|  |     >>> page.title | ||||||
|  |     'Example Page' | ||||||
|  |  | ||||||
|  | Saving and deleting documents | ||||||
|  | ============================= | ||||||
|  | MongoEngine tracks changes to documents to provide efficient saving.  To save | ||||||
|  | the document to the database, call the :meth:`~mongoengine.Document.save` method. | ||||||
|  | If the document does not exist in the database, it will be created. If it does | ||||||
|  | already exist, then any changes will be updated atomically.  For example:: | ||||||
|  |  | ||||||
|  |     >>> page = Page(title="Test Page") | ||||||
|  |     >>> page.save()  # Performs an insert | ||||||
|  |     >>> page.title = "My Page" | ||||||
|  |     >>> page.save()  # Performs an atomic set on the title field. | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |  | ||||||
|  |     Changes to documents are tracked and on the whole perform `set` operations. | ||||||
|  |  | ||||||
|  |     * ``list_field.pop(0)`` - *sets* the resulting list | ||||||
|  |     * ``del(list_field)``   - *unsets* whole list | ||||||
|  |  | ||||||
|  | .. seealso:: | ||||||
|  |     :ref:`guide-atomic-updates` | ||||||
|  |  | ||||||
|  | Cascading Saves | ||||||
|  | --------------- | ||||||
|  | If your document contains :class:`~mongoengine.ReferenceField` or | ||||||
|  | :class:`~mongoengine.GenericReferenceField` objects, then by default the | ||||||
|  | :meth:`~mongoengine.Document.save` method will automatically save any changes to | ||||||
|  | those objects as well.  If this is not desired passing :attr:`cascade` as False | ||||||
|  | to the save method turns this feature off. | ||||||
|  |  | ||||||
|  | Deleting documents | ||||||
|  | ------------------ | ||||||
|  | To delete a document, call the :meth:`~mongoengine.Document.delete` method. | ||||||
|  | Note that this will only work if the document exists in the database and has a | ||||||
|  | valid :attr:`id`. | ||||||
|  |  | ||||||
|  | Document IDs | ||||||
|  | ============ | ||||||
|  | Each document in the database has a unique id. This may be accessed through the | ||||||
|  | :attr:`id` attribute on :class:`~mongoengine.Document` objects. Usually, the id | ||||||
|  | will be generated automatically by the database server when the object is save, | ||||||
|  | meaning that you may only access the :attr:`id` field once a document has been | ||||||
|  | saved:: | ||||||
|  |  | ||||||
|  |     >>> page = Page(title="Test Page") | ||||||
|  |     >>> page.id | ||||||
|  |     >>> page.save() | ||||||
|  |     >>> page.id | ||||||
|  |     ObjectId('123456789abcdef000000000') | ||||||
|  |  | ||||||
|  | Alternatively, you may define one of your own fields to be the document's | ||||||
|  | "primary key" by providing ``primary_key=True`` as a keyword argument to a | ||||||
|  | field's constructor. Under the hood, MongoEngine will use this field as the | ||||||
|  | :attr:`id`; in fact :attr:`id` is actually aliased to your primary key field so | ||||||
|  | you may still use :attr:`id` to access the primary key if you want:: | ||||||
|  |  | ||||||
|  |     >>> class User(Document): | ||||||
|  |     ...     email = StringField(primary_key=True) | ||||||
|  |     ...     name = StringField() | ||||||
|  |     ... | ||||||
|  |     >>> bob = User(email='bob@example.com', name='Bob') | ||||||
|  |     >>> bob.save() | ||||||
|  |     >>> bob.id == bob.email == 'bob@example.com' | ||||||
|  |     True | ||||||
|  |  | ||||||
|  | You can also access the document's "primary key" using the :attr:`pk` field; in | ||||||
|  | is an alias to :attr:`id`:: | ||||||
|  |  | ||||||
|  |     >>> page = Page(title="Another Test Page") | ||||||
|  |     >>> page.save() | ||||||
|  |     >>> page.id == page.pk | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |  | ||||||
|  |    If you define your own primary key field, the field implicitly becomes | ||||||
|  |    required, so a :class:`ValidationError` will be thrown if you don't provide | ||||||
|  |    it. | ||||||
							
								
								
									
										84
									
								
								docs/guide/gridfs.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								docs/guide/gridfs.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,84 @@ | |||||||
|  | ====== | ||||||
|  | GridFS | ||||||
|  | ====== | ||||||
|  |  | ||||||
|  | .. versionadded:: 0.4 | ||||||
|  |  | ||||||
|  | Writing | ||||||
|  | ------- | ||||||
|  |  | ||||||
|  | GridFS support comes in the form of the :class:`~mongoengine.FileField` field | ||||||
|  | object. This field acts as a file-like object and provides a couple of | ||||||
|  | different ways of inserting and retrieving data. Arbitrary metadata such as | ||||||
|  | content type can also be stored alongside the files. In the following example, | ||||||
|  | a document is created to store details about animals, including a photo:: | ||||||
|  |  | ||||||
|  |     class Animal(Document): | ||||||
|  |         genus = StringField() | ||||||
|  |         family = StringField() | ||||||
|  |         photo = FileField() | ||||||
|  |  | ||||||
|  |     marmot = Animal('Marmota', 'Sciuridae') | ||||||
|  |  | ||||||
|  |     marmot_photo = open('marmot.jpg', 'r')      # Retrieve a photo from disk | ||||||
|  |     marmot.photo = marmot_photo                 # Store photo in the document | ||||||
|  |     marmot.photo.content_type = 'image/jpeg'    # Store metadata | ||||||
|  |  | ||||||
|  |     marmot.save() | ||||||
|  |  | ||||||
|  | Another way of writing to a :class:`~mongoengine.FileField` is to use the | ||||||
|  | :func:`put` method. This allows for metadata to be stored in the same call as | ||||||
|  | the file:: | ||||||
|  |  | ||||||
|  |     marmot.photo.put(marmot_photo, content_type='image/jpeg') | ||||||
|  |  | ||||||
|  |     marmot.save() | ||||||
|  |  | ||||||
|  | Retrieval | ||||||
|  | --------- | ||||||
|  |  | ||||||
|  | So using the :class:`~mongoengine.FileField` is just like using any other | ||||||
|  | field. The file can also be retrieved just as easily:: | ||||||
|  |  | ||||||
|  |     marmot = Animal.objects(genus='Marmota').first() | ||||||
|  |     photo = marmot.photo.read() | ||||||
|  |     content_type = marmot.photo.content_type | ||||||
|  |  | ||||||
|  | Streaming | ||||||
|  | --------- | ||||||
|  |  | ||||||
|  | Streaming data into a :class:`~mongoengine.FileField` is achieved in a | ||||||
|  | slightly different manner.  First, a new file must be created by calling the | ||||||
|  | :func:`new_file` method. Data can then be written using :func:`write`:: | ||||||
|  |  | ||||||
|  |     marmot.photo.new_file() | ||||||
|  |     marmot.photo.write('some_image_data') | ||||||
|  |     marmot.photo.write('some_more_image_data') | ||||||
|  |     marmot.photo.close() | ||||||
|  |  | ||||||
|  |     marmot.photo.save() | ||||||
|  |  | ||||||
|  | Deletion | ||||||
|  | -------- | ||||||
|  |  | ||||||
|  | Deleting stored files is achieved with the :func:`delete` method:: | ||||||
|  |  | ||||||
|  |     marmot.photo.delete() | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |  | ||||||
|  |     The FileField in a Document actually only stores the ID of a file in a | ||||||
|  |     separate GridFS collection. This means that deleting a document | ||||||
|  |     with a defined FileField does not actually delete the file. You must be | ||||||
|  |     careful to delete any files in a Document as above before deleting the | ||||||
|  |     Document itself. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Replacing files | ||||||
|  | --------------- | ||||||
|  |  | ||||||
|  | Files can be replaced with the :func:`replace` method. This works just like | ||||||
|  | the :func:`put` method so even metadata can (and should) be replaced:: | ||||||
|  |  | ||||||
|  |     another_marmot = open('another_marmot.png', 'r') | ||||||
|  |     marmot.photo.replace(another_marmot, content_type='image/png') | ||||||
							
								
								
									
										14
									
								
								docs/guide/index.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								docs/guide/index.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | ========== | ||||||
|  | User Guide | ||||||
|  | ========== | ||||||
|  |  | ||||||
|  | .. toctree:: | ||||||
|  |    :maxdepth: 2 | ||||||
|  |  | ||||||
|  |    installing | ||||||
|  |    connecting | ||||||
|  |    defining-documents | ||||||
|  |    document-instances | ||||||
|  |    querying | ||||||
|  |    gridfs | ||||||
|  |    signals | ||||||
							
								
								
									
										31
									
								
								docs/guide/installing.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								docs/guide/installing.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | |||||||
|  | ====================== | ||||||
|  | Installing MongoEngine | ||||||
|  | ====================== | ||||||
|  |  | ||||||
|  | To use MongoEngine, you will need to download `MongoDB <http://mongodb.org/>`_ | ||||||
|  | and ensure it is running in an accessible location. You will also need | ||||||
|  | `PyMongo <http://api.mongodb.org/python>`_ to use MongoEngine, but if you | ||||||
|  | install MongoEngine using setuptools, then the dependencies will be handled for | ||||||
|  | you. | ||||||
|  |  | ||||||
|  | MongoEngine is available on PyPI, so to use it you can use :program:`pip`: | ||||||
|  |  | ||||||
|  | .. code-block:: console | ||||||
|  |  | ||||||
|  |     $ pip install mongoengine | ||||||
|  |  | ||||||
|  | Alternatively, if you don't have setuptools installed, `download it from PyPi | ||||||
|  | <http://pypi.python.org/pypi/mongoengine/>`_ and run | ||||||
|  |  | ||||||
|  | .. code-block:: console | ||||||
|  |  | ||||||
|  |     $ python setup.py install | ||||||
|  |  | ||||||
|  | To use the bleeding-edge version of MongoEngine, you can get the source from | ||||||
|  | `GitHub <http://github.com/hmarr/mongoengine/>`_ and install it as above: | ||||||
|  |  | ||||||
|  | .. code-block:: console | ||||||
|  |  | ||||||
|  |     $ git clone git://github.com/hmarr/mongoengine | ||||||
|  |     $ cd mongoengine | ||||||
|  |     $ python setup.py install | ||||||
							
								
								
									
										545
									
								
								docs/guide/querying.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										545
									
								
								docs/guide/querying.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,545 @@ | |||||||
|  | ===================== | ||||||
|  | Querying the database | ||||||
|  | ===================== | ||||||
|  | :class:`~mongoengine.Document` classes have an :attr:`objects` attribute, which | ||||||
|  | is used for accessing the objects in the database associated with the class. | ||||||
|  | The :attr:`objects` attribute is actually a | ||||||
|  | :class:`~mongoengine.queryset.QuerySetManager`, which creates and returns a new | ||||||
|  | :class:`~mongoengine.queryset.QuerySet` object on access. The | ||||||
|  | :class:`~mongoengine.queryset.QuerySet` object may be iterated over to | ||||||
|  | fetch documents from the database:: | ||||||
|  |  | ||||||
|  |     # Prints out the names of all the users in the database | ||||||
|  |     for user in User.objects: | ||||||
|  |         print user.name | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |  | ||||||
|  |    Once the iteration finishes (when :class:`StopIteration` is raised), | ||||||
|  |    :meth:`~mongoengine.queryset.QuerySet.rewind` will be called so that the | ||||||
|  |    :class:`~mongoengine.queryset.QuerySet` may be iterated over again. The | ||||||
|  |    results of the first iteration are *not* cached, so the database will be hit | ||||||
|  |    each time the :class:`~mongoengine.queryset.QuerySet` is iterated over. | ||||||
|  |  | ||||||
|  | Filtering queries | ||||||
|  | ================= | ||||||
|  | The query may be filtered by calling the | ||||||
|  | :class:`~mongoengine.queryset.QuerySet` object with field lookup keyword | ||||||
|  | arguments. The keys in the keyword arguments correspond to fields on the | ||||||
|  | :class:`~mongoengine.Document` you are querying:: | ||||||
|  |  | ||||||
|  |     # This will return a QuerySet that will only iterate over users whose | ||||||
|  |     # 'country' field is set to 'uk' | ||||||
|  |     uk_users = User.objects(country='uk') | ||||||
|  |  | ||||||
|  | Fields on embedded documents may also be referred to using field lookup syntax | ||||||
|  | by using a double-underscore in place of the dot in object attribute access | ||||||
|  | syntax:: | ||||||
|  |  | ||||||
|  |     # This will return a QuerySet that will only iterate over pages that have | ||||||
|  |     # been written by a user whose 'country' field is set to 'uk' | ||||||
|  |     uk_pages = Page.objects(author__country='uk') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Query operators | ||||||
|  | =============== | ||||||
|  | Operators other than equality may also be used in queries; just attach the | ||||||
|  | operator name to a key with a double-underscore:: | ||||||
|  |  | ||||||
|  |     # Only find users whose age is 18 or less | ||||||
|  |     young_users = Users.objects(age__lte=18) | ||||||
|  |  | ||||||
|  | Available operators are as follows: | ||||||
|  |  | ||||||
|  | * ``ne`` -- not equal to | ||||||
|  | * ``lt`` -- less than | ||||||
|  | * ``lte`` -- less than or equal to | ||||||
|  | * ``gt`` -- greater than | ||||||
|  | * ``gte`` -- greater than or equal to | ||||||
|  | * ``not`` -- negate a standard check, may be used before other operators (e.g. | ||||||
|  |   ``Q(age__not__mod=5)``) | ||||||
|  | * ``in`` -- value is in list (a list of values should be provided) | ||||||
|  | * ``nin`` -- value is not in list (a list of values should be provided) | ||||||
|  | * ``mod`` -- ``value % x == y``, where ``x`` and ``y`` are two provided values | ||||||
|  | * ``all`` -- every item in list of values provided is in array | ||||||
|  | * ``size`` -- the size of the array is | ||||||
|  | * ``exists`` -- value for field exists | ||||||
|  |  | ||||||
|  | The following operators are available as shortcuts to querying with regular | ||||||
|  | expressions: | ||||||
|  |  | ||||||
|  | * ``exact`` -- string field exactly matches value | ||||||
|  | * ``iexact`` -- string field exactly matches value (case insensitive) | ||||||
|  | * ``contains`` -- string field contains value | ||||||
|  | * ``icontains`` -- string field contains value (case insensitive) | ||||||
|  | * ``startswith`` -- string field starts with value | ||||||
|  | * ``istartswith`` -- string field starts with value (case insensitive) | ||||||
|  | * ``endswith`` -- string field ends with value | ||||||
|  | * ``iendswith`` -- string field ends with value (case insensitive) | ||||||
|  | * ``match``  -- performs an $elemMatch so you can match an entire document within an array | ||||||
|  |  | ||||||
|  | There are a few special operators for performing geographical queries, that | ||||||
|  | may used with :class:`~mongoengine.GeoPointField`\ s: | ||||||
|  |  | ||||||
|  | * ``within_distance`` -- provide a list containing a point and a maximum | ||||||
|  |   distance (e.g. [(41.342, -87.653), 5]) | ||||||
|  | * ``within_spherical_distance`` -- Same as above but using the spherical geo model | ||||||
|  |   (e.g. [(41.342, -87.653), 5/earth_radius]) | ||||||
|  | * ``near`` -- order the documents by how close they are to a given point | ||||||
|  | * ``near_sphere`` -- Same as above but using the spherical geo model | ||||||
|  | * ``within_box`` -- filter documents to those within a given bounding box (e.g. | ||||||
|  |   [(35.0, -125.0), (40.0, -100.0)]) | ||||||
|  | * ``within_polygon`` -- filter documents to those within a given polygon (e.g. | ||||||
|  |   [(41.91,-87.69), (41.92,-87.68), (41.91,-87.65), (41.89,-87.65)]). | ||||||
|  |   .. note:: Requires Mongo Server 2.0 | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Querying lists | ||||||
|  | -------------- | ||||||
|  | On most fields, this syntax will look up documents where the field specified | ||||||
|  | matches the given value exactly, but when the field refers to a | ||||||
|  | :class:`~mongoengine.ListField`, a single item may be provided, in which case | ||||||
|  | lists that contain that item will be matched:: | ||||||
|  |  | ||||||
|  |     class Page(Document): | ||||||
|  |         tags = ListField(StringField()) | ||||||
|  |  | ||||||
|  |     # This will match all pages that have the word 'coding' as an item in the | ||||||
|  |     # 'tags' list | ||||||
|  |     Page.objects(tags='coding') | ||||||
|  |  | ||||||
|  | It is possible to query by position in a list by using a numerical value as a | ||||||
|  | query operator. So if you wanted to find all pages whose first tag was ``db``, | ||||||
|  | you could use the following query:: | ||||||
|  |  | ||||||
|  |     Page.objects(tags__0='db') | ||||||
|  |  | ||||||
|  | If you only want to fetch part of a list eg: you want to paginate a list, then | ||||||
|  | the `slice` operator is required:: | ||||||
|  |  | ||||||
|  |     # comments - skip 5, limit 10 | ||||||
|  |     Page.objects.fields(slice__comments=[5, 10]) | ||||||
|  |  | ||||||
|  | For updating documents, if you don't know the position in a list, you can use | ||||||
|  | the $ positional operator :: | ||||||
|  |  | ||||||
|  |     Post.objects(comments__by="joe").update(**{'inc__comments__$__votes': 1}) | ||||||
|  |  | ||||||
|  | However, this doesn't map well to the syntax so you can also use a capital S instead :: | ||||||
|  |  | ||||||
|  |     Post.objects(comments__by="joe").update(inc__comments__S__votes=1) | ||||||
|  |  | ||||||
|  |     .. note:: Due to Mongo currently the $ operator only applies to the first matched item in the query. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Raw queries | ||||||
|  | ----------- | ||||||
|  | It is possible to provide a raw PyMongo query as a query parameter, which will | ||||||
|  | be integrated directly into the query. This is done using the ``__raw__`` | ||||||
|  | keyword argument:: | ||||||
|  |  | ||||||
|  |     Page.objects(__raw__={'tags': 'coding'}) | ||||||
|  |  | ||||||
|  | .. versionadded:: 0.4 | ||||||
|  |  | ||||||
|  | Limiting and skipping results | ||||||
|  | ============================= | ||||||
|  | Just as with traditional ORMs, you may limit the number of results returned, or | ||||||
|  | skip a number or results in you query. | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.limit` and | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.skip` and methods are available on | ||||||
|  | :class:`~mongoengine.queryset.QuerySet` objects, but the prefered syntax for | ||||||
|  | achieving this is using array-slicing syntax:: | ||||||
|  |  | ||||||
|  |     # Only the first 5 people | ||||||
|  |     users = User.objects[:5] | ||||||
|  |  | ||||||
|  |     # All except for the first 5 people | ||||||
|  |     users = User.objects[5:] | ||||||
|  |  | ||||||
|  |     # 5 users, starting from the 10th user found | ||||||
|  |     users = User.objects[10:15] | ||||||
|  |  | ||||||
|  | You may also index the query to retrieve a single result. If an item at that | ||||||
|  | index does not exists, an :class:`IndexError` will be raised. A shortcut for | ||||||
|  | retrieving the first result and returning :attr:`None` if no result exists is | ||||||
|  | provided (:meth:`~mongoengine.queryset.QuerySet.first`):: | ||||||
|  |  | ||||||
|  |     >>> # Make sure there are no users | ||||||
|  |     >>> User.drop_collection() | ||||||
|  |     >>> User.objects[0] | ||||||
|  |     IndexError: list index out of range | ||||||
|  |     >>> User.objects.first() == None | ||||||
|  |     True | ||||||
|  |     >>> User(name='Test User').save() | ||||||
|  |     >>> User.objects[0] == User.objects.first() | ||||||
|  |     True | ||||||
|  |  | ||||||
|  | Retrieving unique results | ||||||
|  | ------------------------- | ||||||
|  | To retrieve a result that should be unique in the collection, use | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.get`. This will raise | ||||||
|  | :class:`~mongoengine.queryset.DoesNotExist` if no document matches the query, | ||||||
|  | and :class:`~mongoengine.queryset.MultipleObjectsReturned` if more than one | ||||||
|  | document matched the query. | ||||||
|  |  | ||||||
|  | A variation of this method exists, | ||||||
|  | :meth:`~mongoengine.queryset.Queryset.get_or_create`, that will create a new | ||||||
|  | document with the query arguments if no documents match the query. An | ||||||
|  | additional keyword argument, :attr:`defaults` may be provided, which will be | ||||||
|  | used as default values for the new document, in the case that it should need | ||||||
|  | to be created:: | ||||||
|  |  | ||||||
|  |     >>> a, created = User.objects.get_or_create(name='User A', defaults={'age': 30}) | ||||||
|  |     >>> b, created = User.objects.get_or_create(name='User A', defaults={'age': 40}) | ||||||
|  |     >>> a.name == b.name and a.age == b.age | ||||||
|  |     True | ||||||
|  |  | ||||||
|  | Default Document queries | ||||||
|  | ======================== | ||||||
|  | By default, the objects :attr:`~mongoengine.Document.objects` attribute on a | ||||||
|  | document returns a :class:`~mongoengine.queryset.QuerySet` that doesn't filter | ||||||
|  | the collection -- it returns all objects. This may be changed by defining a | ||||||
|  | method on a document that modifies a queryset. The method should accept two | ||||||
|  | arguments -- :attr:`doc_cls` and :attr:`queryset`. The first argument is the | ||||||
|  | :class:`~mongoengine.Document` class that the method is defined on (in this | ||||||
|  | sense, the method is more like a :func:`classmethod` than a regular method), | ||||||
|  | and the second argument is the initial queryset. The method needs to be | ||||||
|  | decorated with :func:`~mongoengine.queryset.queryset_manager` in order for it | ||||||
|  | to be recognised. :: | ||||||
|  |  | ||||||
|  |     class BlogPost(Document): | ||||||
|  |         title = StringField() | ||||||
|  |         date = DateTimeField() | ||||||
|  |  | ||||||
|  |         @queryset_manager | ||||||
|  |         def objects(doc_cls, queryset): | ||||||
|  |             # This may actually also be done by defining a default ordering for | ||||||
|  |             # the document, but this illustrates the use of manager methods | ||||||
|  |             return queryset.order_by('-date') | ||||||
|  |  | ||||||
|  | You don't need to call your method :attr:`objects` -- you may define as many | ||||||
|  | custom manager methods as you like:: | ||||||
|  |  | ||||||
|  |     class BlogPost(Document): | ||||||
|  |         title = StringField() | ||||||
|  |         published = BooleanField() | ||||||
|  |  | ||||||
|  |         @queryset_manager | ||||||
|  |         def live_posts(doc_cls, queryset): | ||||||
|  |             return queryset.filter(published=True) | ||||||
|  |  | ||||||
|  |     BlogPost(title='test1', published=False).save() | ||||||
|  |     BlogPost(title='test2', published=True).save() | ||||||
|  |     assert len(BlogPost.objects) == 2 | ||||||
|  |     assert len(BlogPost.live_posts) == 1 | ||||||
|  |  | ||||||
|  | Custom QuerySets | ||||||
|  | ================ | ||||||
|  | Should you want to add custom methods for interacting with or filtering | ||||||
|  | documents, extending the :class:`~mongoengine.queryset.QuerySet` class may be | ||||||
|  | the way to go. To use a custom :class:`~mongoengine.queryset.QuerySet` class on | ||||||
|  | a document, set ``queryset_class`` to the custom class in a | ||||||
|  | :class:`~mongoengine.Document`\ s ``meta`` dictionary:: | ||||||
|  |  | ||||||
|  |     class AwesomerQuerySet(QuerySet): | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     class Page(Document): | ||||||
|  |         meta = {'queryset_class': AwesomerQuerySet} | ||||||
|  |  | ||||||
|  | .. versionadded:: 0.4 | ||||||
|  |  | ||||||
|  | Aggregation | ||||||
|  | =========== | ||||||
|  | MongoDB provides some aggregation methods out of the box, but there are not as | ||||||
|  | many as you typically get with an RDBMS. MongoEngine provides a wrapper around | ||||||
|  | the built-in methods and provides some of its own, which are implemented as | ||||||
|  | Javascript code that is executed on the database server. | ||||||
|  |  | ||||||
|  | Counting results | ||||||
|  | ---------------- | ||||||
|  | Just as with limiting and skipping results, there is a method on | ||||||
|  | :class:`~mongoengine.queryset.QuerySet` objects -- | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.count`, but there is also a more Pythonic | ||||||
|  | way of achieving this:: | ||||||
|  |  | ||||||
|  |     num_users = len(User.objects) | ||||||
|  |  | ||||||
|  | Further aggregation | ||||||
|  | ------------------- | ||||||
|  | You may sum over the values of a specific field on documents using | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.sum`:: | ||||||
|  |  | ||||||
|  |     yearly_expense = Employee.objects.sum('salary') | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |  | ||||||
|  |    If the field isn't present on a document, that document will be ignored from | ||||||
|  |    the sum. | ||||||
|  |  | ||||||
|  | To get the average (mean) of a field on a collection of documents, use | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.average`:: | ||||||
|  |  | ||||||
|  |     mean_age = User.objects.average('age') | ||||||
|  |  | ||||||
|  | As MongoDB provides native lists, MongoEngine provides a helper method to get a | ||||||
|  | dictionary of the frequencies of items in lists across an entire collection -- | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.item_frequencies`. An example of its use | ||||||
|  | would be generating "tag-clouds":: | ||||||
|  |  | ||||||
|  |     class Article(Document): | ||||||
|  |         tag = ListField(StringField()) | ||||||
|  |  | ||||||
|  |     # After adding some tagged articles... | ||||||
|  |     tag_freqs = Article.objects.item_frequencies('tag', normalize=True) | ||||||
|  |  | ||||||
|  |     from operator import itemgetter | ||||||
|  |     top_tags = sorted(tag_freqs.items(), key=itemgetter(1), reverse=True)[:10] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Query efficiency and performance | ||||||
|  | ================================ | ||||||
|  |  | ||||||
|  | There are a couple of methods to improve efficiency when querying, reducing the | ||||||
|  | information returned by the query or efficient dereferencing . | ||||||
|  |  | ||||||
|  | Retrieving a subset of fields | ||||||
|  | ----------------------------- | ||||||
|  |  | ||||||
|  | Sometimes a subset of fields on a :class:`~mongoengine.Document` is required, | ||||||
|  | and for efficiency only these should be retrieved from the database. This issue | ||||||
|  | is especially important for MongoDB, as fields may often be extremely large | ||||||
|  | (e.g. a :class:`~mongoengine.ListField` of | ||||||
|  | :class:`~mongoengine.EmbeddedDocument`\ s, which represent the comments on a | ||||||
|  | blog post. To select only a subset of fields, use | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.only`, specifying the fields you want to | ||||||
|  | retrieve as its arguments. Note that if fields that are not downloaded are | ||||||
|  | accessed, their default value (or :attr:`None` if no default value is provided) | ||||||
|  | will be given:: | ||||||
|  |  | ||||||
|  |     >>> class Film(Document): | ||||||
|  |     ...     title = StringField() | ||||||
|  |     ...     year = IntField() | ||||||
|  |     ...     rating = IntField(default=3) | ||||||
|  |     ... | ||||||
|  |     >>> Film(title='The Shawshank Redemption', year=1994, rating=5).save() | ||||||
|  |     >>> f = Film.objects.only('title').first() | ||||||
|  |     >>> f.title | ||||||
|  |     'The Shawshank Redemption' | ||||||
|  |     >>> f.year   # None | ||||||
|  |     >>> f.rating # default value | ||||||
|  |     3 | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |  | ||||||
|  |     The :meth:`~mongoengine.queryset.QuerySet.exclude` is the opposite of | ||||||
|  |     :meth:`~mongoengine.queryset.QuerySet.only` if you want to exclude a field. | ||||||
|  |  | ||||||
|  | If you later need the missing fields, just call | ||||||
|  | :meth:`~mongoengine.Document.reload` on your document. | ||||||
|  |  | ||||||
|  | Getting related data | ||||||
|  | -------------------- | ||||||
|  |  | ||||||
|  | When iterating the results of :class:`~mongoengine.ListField` or | ||||||
|  | :class:`~mongoengine.DictField` we automatically dereference any | ||||||
|  | :class:`~pymongo.dbref.DBRef` objects as efficiently as possible, reducing the | ||||||
|  | number the queries to mongo. | ||||||
|  |  | ||||||
|  | There are times when that efficiency is not enough, documents that have | ||||||
|  | :class:`~mongoengine.ReferenceField` objects or | ||||||
|  | :class:`~mongoengine.GenericReferenceField` objects at the top level are | ||||||
|  | expensive as the number of queries to MongoDB can quickly rise. | ||||||
|  |  | ||||||
|  | To limit the number of queries use | ||||||
|  | :func:`~mongoengine.queryset.QuerySet.select_related` which converts the | ||||||
|  | QuerySet to a list and dereferences as efficiently as possible.  By default | ||||||
|  | :func:`~mongoengine.queryset.QuerySet.select_related` only dereferences any | ||||||
|  | references to the depth of 1 level.  If you have more complicated documents and | ||||||
|  | want to dereference more of the object at once then increasing the :attr:`max_depth` | ||||||
|  | will dereference more levels of the document. | ||||||
|  |  | ||||||
|  | Advanced queries | ||||||
|  | ================ | ||||||
|  | Sometimes calling a :class:`~mongoengine.queryset.QuerySet` object with keyword | ||||||
|  | arguments can't fully express the query you want to use -- for example if you | ||||||
|  | need to combine a number of constraints using *and* and *or*. This is made | ||||||
|  | possible in MongoEngine through the :class:`~mongoengine.queryset.Q` class. | ||||||
|  | A :class:`~mongoengine.queryset.Q` object represents part of a query, and | ||||||
|  | can be initialised using the same keyword-argument syntax you use to query | ||||||
|  | documents. To build a complex query, you may combine | ||||||
|  | :class:`~mongoengine.queryset.Q` objects using the ``&`` (and) and ``|`` (or) | ||||||
|  | operators. To use a :class:`~mongoengine.queryset.Q` object, pass it in as the | ||||||
|  | first positional argument to :attr:`Document.objects` when you filter it by | ||||||
|  | calling it with keyword arguments:: | ||||||
|  |  | ||||||
|  |     # Get published posts | ||||||
|  |     Post.objects(Q(published=True) | Q(publish_date__lte=datetime.now())) | ||||||
|  |  | ||||||
|  |     # Get top posts | ||||||
|  |     Post.objects((Q(featured=True) & Q(hits__gte=1000)) | Q(hits__gte=5000)) | ||||||
|  |  | ||||||
|  | .. _guide-atomic-updates: | ||||||
|  |  | ||||||
|  | Atomic updates | ||||||
|  | ============== | ||||||
|  | Documents may be updated atomically by using the | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.update_one` and | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.update` methods on a | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet`. There are several different "modifiers" | ||||||
|  | that you may use with these methods: | ||||||
|  |  | ||||||
|  | * ``set`` -- set a particular value | ||||||
|  | * ``unset`` -- delete a particular value (since MongoDB v1.3+) | ||||||
|  | * ``inc`` -- increment a value by a given amount | ||||||
|  | * ``dec`` -- decrement a value by a given amount | ||||||
|  | * ``pop`` -- remove the last item from a list | ||||||
|  | * ``push`` -- append a value to a list | ||||||
|  | * ``push_all`` -- append several values to a list | ||||||
|  | * ``pop`` -- remove the first or last element of a list | ||||||
|  | * ``pull`` -- remove a value from a list | ||||||
|  | * ``pull_all`` -- remove several values from a list | ||||||
|  | * ``add_to_set`` -- add value to a list only if its not in the list already | ||||||
|  |  | ||||||
|  | The syntax for atomic updates is similar to the querying syntax, but the | ||||||
|  | modifier comes before the field, not after it:: | ||||||
|  |  | ||||||
|  |     >>> post = BlogPost(title='Test', page_views=0, tags=['database']) | ||||||
|  |     >>> post.save() | ||||||
|  |     >>> BlogPost.objects(id=post.id).update_one(inc__page_views=1) | ||||||
|  |     >>> post.reload()  # the document has been changed, so we need to reload it | ||||||
|  |     >>> post.page_views | ||||||
|  |     1 | ||||||
|  |     >>> BlogPost.objects(id=post.id).update_one(set__title='Example Post') | ||||||
|  |     >>> post.reload() | ||||||
|  |     >>> post.title | ||||||
|  |     'Example Post' | ||||||
|  |     >>> BlogPost.objects(id=post.id).update_one(push__tags='nosql') | ||||||
|  |     >>> post.reload() | ||||||
|  |     >>> post.tags | ||||||
|  |     ['database', 'nosql'] | ||||||
|  |  | ||||||
|  | .. note :: | ||||||
|  |  | ||||||
|  |     In version 0.5 the :meth:`~mongoengine.Document.save` runs atomic updates | ||||||
|  |     on changed documents by tracking changes to that document. | ||||||
|  |  | ||||||
|  | The positional operator allows you to update list items without knowing the | ||||||
|  | index position, therefore making the update a single atomic operation.  As we | ||||||
|  | cannot use the `$` syntax in keyword arguments it has been mapped to `S`:: | ||||||
|  |  | ||||||
|  |     >>> post = BlogPost(title='Test', page_views=0, tags=['database', 'mongo']) | ||||||
|  |     >>> post.save() | ||||||
|  |     >>> BlogPost.objects(id=post.id, tags='mongo').update(set__tags__S='mongodb') | ||||||
|  |     >>> post.reload() | ||||||
|  |     >>> post.tags | ||||||
|  |     ['database', 'mongodb'] | ||||||
|  |  | ||||||
|  | .. note :: | ||||||
|  |     Currently only top level lists are handled, future versions of mongodb / | ||||||
|  |     pymongo plan to support nested positional operators.  See `The $ positional | ||||||
|  |     operator <http://www.mongodb.org/display/DOCS/Updating#Updating-The%24positionaloperator>`_. | ||||||
|  |  | ||||||
|  | Server-side javascript execution | ||||||
|  | ================================ | ||||||
|  | Javascript functions may be written and sent to the server for execution. The | ||||||
|  | result of this is the return value of the Javascript function. This | ||||||
|  | functionality is accessed through the | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.exec_js` method on | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet` objects. Pass in a string containing a | ||||||
|  | Javascript function as the first argument. | ||||||
|  |  | ||||||
|  | The remaining positional arguments are names of fields that will be passed into | ||||||
|  | you Javascript function as its arguments. This allows functions to be written | ||||||
|  | that may be executed on any field in a collection (e.g. the | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.sum` method, which accepts the name of | ||||||
|  | the field to sum over as its argument). Note that field names passed in in this | ||||||
|  | manner are automatically translated to the names used on the database (set | ||||||
|  | using the :attr:`name` keyword argument to a field constructor). | ||||||
|  |  | ||||||
|  | Keyword arguments to :meth:`~mongoengine.queryset.QuerySet.exec_js` are | ||||||
|  | combined into an object called :attr:`options`, which is available in the | ||||||
|  | Javascript function. This may be used for defining specific parameters for your | ||||||
|  | function. | ||||||
|  |  | ||||||
|  | Some variables are made available in the scope of the Javascript function: | ||||||
|  |  | ||||||
|  | * ``collection`` -- the name of the collection that corresponds to the | ||||||
|  |   :class:`~mongoengine.Document` class that is being used; this should be | ||||||
|  |   used to get the :class:`Collection` object from :attr:`db` in Javascript | ||||||
|  |   code | ||||||
|  | * ``query`` -- the query that has been generated by the | ||||||
|  |   :class:`~mongoengine.queryset.QuerySet` object; this may be passed into | ||||||
|  |   the :meth:`find` method on a :class:`Collection` object in the Javascript | ||||||
|  |   function | ||||||
|  | * ``options`` -- an object containing the keyword arguments passed into | ||||||
|  |   :meth:`~mongoengine.queryset.QuerySet.exec_js` | ||||||
|  |  | ||||||
|  | The following example demonstrates the intended usage of | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.exec_js` by defining a function that sums | ||||||
|  | over a field on a document (this functionality is already available throught | ||||||
|  | :meth:`~mongoengine.queryset.QuerySet.sum` but is shown here for sake of | ||||||
|  | example):: | ||||||
|  |  | ||||||
|  |     def sum_field(document, field_name, include_negatives=True): | ||||||
|  |         code = """ | ||||||
|  |         function(sumField) { | ||||||
|  |             var total = 0.0; | ||||||
|  |             db[collection].find(query).forEach(function(doc) { | ||||||
|  |                 var val = doc[sumField]; | ||||||
|  |                 if (val >= 0.0 || options.includeNegatives) { | ||||||
|  |                     total += val; | ||||||
|  |                 } | ||||||
|  |             }); | ||||||
|  |             return total; | ||||||
|  |         } | ||||||
|  |         """ | ||||||
|  |         options = {'includeNegatives': include_negatives} | ||||||
|  |         return document.objects.exec_js(code, field_name, **options) | ||||||
|  |  | ||||||
|  | As fields in MongoEngine may use different names in the database (set using the | ||||||
|  | :attr:`db_field` keyword argument to a :class:`Field` constructor), a mechanism | ||||||
|  | exists for replacing MongoEngine field names with the database field names in | ||||||
|  | Javascript code. When accessing a field on a collection object, use | ||||||
|  | square-bracket notation, and prefix the MongoEngine field name with a tilde. | ||||||
|  | The field name that follows the tilde will be translated to the name used in | ||||||
|  | the database. Note that when referring to fields on embedded documents, | ||||||
|  | the name of the :class:`~mongoengine.EmbeddedDocumentField`, followed by a dot, | ||||||
|  | should be used before the name of the field on the embedded document. The | ||||||
|  | following example shows how the substitutions are made:: | ||||||
|  |  | ||||||
|  |     class Comment(EmbeddedDocument): | ||||||
|  |         content = StringField(db_field='body') | ||||||
|  |  | ||||||
|  |     class BlogPost(Document): | ||||||
|  |         title = StringField(db_field='doctitle') | ||||||
|  |         comments = ListField(EmbeddedDocumentField(Comment), name='cs') | ||||||
|  |  | ||||||
|  |     # Returns a list of dictionaries. Each dictionary contains a value named | ||||||
|  |     # "document", which corresponds to the "title" field on a BlogPost, and | ||||||
|  |     # "comment", which corresponds to an individual comment. The substitutions | ||||||
|  |     # made are shown in the comments. | ||||||
|  |     BlogPost.objects.exec_js(""" | ||||||
|  |     function() { | ||||||
|  |         var comments = []; | ||||||
|  |         db[collection].find(query).forEach(function(doc) { | ||||||
|  |             // doc[~comments] -> doc["cs"] | ||||||
|  |             var docComments = doc[~comments]; | ||||||
|  |  | ||||||
|  |             for (var i = 0; i < docComments.length; i++) { | ||||||
|  |                 // doc[~comments][i] -> doc["cs"][i] | ||||||
|  |                 var comment = doc[~comments][i]; | ||||||
|  |  | ||||||
|  |                 comments.push({ | ||||||
|  |                     // doc[~title] -> doc["doctitle"] | ||||||
|  |                     'document': doc[~title], | ||||||
|  |  | ||||||
|  |                     // comment[~comments.content] -> comment["body"] | ||||||
|  |                     'comment': comment[~comments.content] | ||||||
|  |                 }); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  |         return comments; | ||||||
|  |     } | ||||||
|  |     """) | ||||||
							
								
								
									
										55
									
								
								docs/guide/signals.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								docs/guide/signals.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | |||||||
|  | .. _signals: | ||||||
|  |  | ||||||
|  | Signals | ||||||
|  | ======= | ||||||
|  |  | ||||||
|  | .. versionadded:: 0.5 | ||||||
|  |  | ||||||
|  | Signal support is provided by the excellent `blinker`_ library and | ||||||
|  | will gracefully fall back if it is not available. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | <<<<<<< HEAD | ||||||
|  | The following document signals exist in MongoEngine and are pretty self explanatory: | ||||||
|  | ======= | ||||||
|  | The following document signals exist in MongoEngine and are pretty self-explanatory: | ||||||
|  | >>>>>>> master | ||||||
|  |  | ||||||
|  |   * `mongoengine.signals.pre_init` | ||||||
|  |   * `mongoengine.signals.post_init` | ||||||
|  |   * `mongoengine.signals.pre_save` | ||||||
|  |   * `mongoengine.signals.post_save` | ||||||
|  |   * `mongoengine.signals.pre_delete` | ||||||
|  |   * `mongoengine.signals.post_delete` | ||||||
|  |   * `mongoengine.signals.pre_bulk_insert` | ||||||
|  |   * `mongoengine.signals.post_bulk_insert` | ||||||
|  |  | ||||||
|  | Example usage:: | ||||||
|  |  | ||||||
|  |     from mongoengine import * | ||||||
|  |     from mongoengine import signals | ||||||
|  |  | ||||||
|  |     class Author(Document): | ||||||
|  |         name = StringField() | ||||||
|  |  | ||||||
|  |         def __unicode__(self): | ||||||
|  |             return self.name | ||||||
|  |  | ||||||
|  |         @classmethod | ||||||
|  |         def pre_save(cls, sender, document, **kwargs): | ||||||
|  |             logging.debug("Pre Save: %s" % document.name) | ||||||
|  |  | ||||||
|  |         @classmethod | ||||||
|  |         def post_save(cls, sender, document, **kwargs): | ||||||
|  |             logging.debug("Post Save: %s" % document.name) | ||||||
|  |             if 'created' in kwargs: | ||||||
|  |                 if kwargs['created']: | ||||||
|  |                     logging.debug("Created") | ||||||
|  |                 else: | ||||||
|  |                     logging.debug("Updated") | ||||||
|  |  | ||||||
|  |     signals.pre_save.connect(Author.pre_save, sender=Author) | ||||||
|  |     signals.post_save.connect(Author.post_save, sender=Author) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | .. _blinker: http://pypi.python.org/pypi/blinker | ||||||
| @@ -1,26 +1,67 @@ | |||||||
|  | ============================== | ||||||
| MongoEngine User Documentation | MongoEngine User Documentation | ||||||
| ======================================= | ============================== | ||||||
|  |  | ||||||
| MongoEngine is an Object-Document Mapper, written in Python for working with  | **MongoEngine** is an Object-Document Mapper, written in Python for working with | ||||||
| MongoDB. To install it, simply run | MongoDB. To install it, simply run | ||||||
|  |  | ||||||
| .. code-block:: console | .. code-block:: console | ||||||
|  |  | ||||||
|     # easy_install mongoengine |     # pip install -U mongoengine | ||||||
|  |  | ||||||
| The source is available on `GitHub <http://github.com/hmarr/mongoengine>`_. | :doc:`tutorial` | ||||||
|  |   Start here for a quick overview. | ||||||
|  |  | ||||||
|  | :doc:`guide/index` | ||||||
|  |   The Full guide to MongoEngine | ||||||
|  |  | ||||||
|  | :doc:`apireference` | ||||||
|  |   The complete API documentation. | ||||||
|  |  | ||||||
|  | :doc:`upgrade` | ||||||
|  |   How to upgrade MongoEngine. | ||||||
|  |  | ||||||
|  | :doc:`django` | ||||||
|  |   Using MongoEngine and Django | ||||||
|  |  | ||||||
|  | Community | ||||||
|  | --------- | ||||||
|  |  | ||||||
|  | To get help with using MongoEngine, use the `MongoEngine Users mailing list | ||||||
|  | <http://groups.google.com/group/mongoengine-users>`_ or come chat on the | ||||||
|  | `#mongoengine IRC channel <irc://irc.freenode.net/mongoengine>`_. | ||||||
|  |  | ||||||
|  | Contributing | ||||||
|  | ------------ | ||||||
|  |  | ||||||
|  | The source is available on `GitHub <http://github.com/hmarr/mongoengine>`_ and | ||||||
|  | contributions are always encouraged. Contributions can be as simple as | ||||||
|  | minor tweaks to this documentation. To contribute, fork the project on | ||||||
|  | `GitHub <http://github.com/hmarr/mongoengine>`_ and send a | ||||||
|  | pull request. | ||||||
|  |  | ||||||
|  | Also, you can join the developers' `mailing list | ||||||
|  | <http://groups.google.com/group/mongoengine-dev>`_. | ||||||
|  |  | ||||||
|  | Changes | ||||||
|  | ------- | ||||||
|  | See the :doc:`changelog` for a full list of changes to MongoEngine and | ||||||
|  | :doc:`upgrade` for upgrade information. | ||||||
|  |  | ||||||
| .. toctree:: | .. toctree:: | ||||||
|    :maxdepth: 2 |    :hidden: | ||||||
|  |  | ||||||
|    tutorial |    tutorial | ||||||
|    userguide |    guide/index | ||||||
|    apireference |    apireference | ||||||
|  |    django | ||||||
|    changelog |    changelog | ||||||
|  |    upgrade | ||||||
|  |  | ||||||
| Indices and tables | Indices and tables | ||||||
| ================== | ------------------ | ||||||
|  |  | ||||||
| * :ref:`genindex` | * :ref:`genindex` | ||||||
|  | * :ref:`modindex` | ||||||
| * :ref:`search` | * :ref:`search` | ||||||
|  |  | ||||||
|   | |||||||
| @@ -152,6 +152,26 @@ We can then store a list of comment documents in our post document:: | |||||||
|         tags = ListField(StringField(max_length=30)) |         tags = ListField(StringField(max_length=30)) | ||||||
|         comments = ListField(EmbeddedDocumentField(Comment)) |         comments = ListField(EmbeddedDocumentField(Comment)) | ||||||
|  |  | ||||||
|  | Handling deletions of references | ||||||
|  | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ | ||||||
|  |  | ||||||
|  | The :class:`~mongoengine.ReferenceField` object takes a keyword | ||||||
|  | `reverse_delete_rule` for handling deletion rules if the reference is deleted. | ||||||
|  | To delete all the posts if a user is deleted set the rule:: | ||||||
|  |  | ||||||
|  |     class Post(Document): | ||||||
|  |         title = StringField(max_length=120, required=True) | ||||||
|  |         author = ReferenceField(User, reverse_delete_rule=CASCADE) | ||||||
|  |         tags = ListField(StringField(max_length=30)) | ||||||
|  |         comments = ListField(EmbeddedDocumentField(Comment)) | ||||||
|  |  | ||||||
|  | See :class:`~mongoengine.ReferenceField` for more information. | ||||||
|  |  | ||||||
|  | ..note:: | ||||||
|  |     MapFields and DictFields currently don't support automatic handling of | ||||||
|  |     deleted references | ||||||
|  |  | ||||||
|  |  | ||||||
| Adding data to our Tumblelog | Adding data to our Tumblelog | ||||||
| ============================ | ============================ | ||||||
| Now that we've defined how our documents will be structured, let's start adding | Now that we've defined how our documents will be structured, let's start adding | ||||||
| @@ -250,5 +270,5 @@ the first matched by the query you provide. Aggregation functions may also be | |||||||
| used on :class:`~mongoengine.queryset.QuerySet` objects:: | used on :class:`~mongoengine.queryset.QuerySet` objects:: | ||||||
|  |  | ||||||
|     num_posts = Post.objects(tags='mongodb').count() |     num_posts = Post.objects(tags='mongodb').count() | ||||||
|     print 'Found % posts with tag "mongodb"' % num_posts |     print 'Found %d posts with tag "mongodb"' % num_posts | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										113
									
								
								docs/upgrade.rst
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								docs/upgrade.rst
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | |||||||
|  | ========= | ||||||
|  | Upgrading | ||||||
|  | ========= | ||||||
|  |  | ||||||
|  | 0.5 to 0.6 | ||||||
|  | ========== | ||||||
|  |  | ||||||
|  | Embedded Documents - if you had a `pk` field you will have to rename it from `_id` | ||||||
|  | to `pk` as pk is no longer a property of Embedded Documents. | ||||||
|  |  | ||||||
|  | Reverse Delete Rules in Embedded Documents, MapFields and DictFields now throw | ||||||
|  | an InvalidDocument error as they aren't currently supported. | ||||||
|  |  | ||||||
|  | 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. | ||||||
|  |  | ||||||
|  | FutureWarning - A future warning has been added to all inherited classes that | ||||||
|  | don't define `allow_inheritance` in their meta. | ||||||
|  |  | ||||||
|  | 0.4 to 0.5 | ||||||
|  | =========== | ||||||
|  |  | ||||||
|  | There have been the following backwards incompatibilities from 0.4 to 0.5.  The | ||||||
|  | main areas of changed are: choices in fields, map_reduce and collection names. | ||||||
|  |  | ||||||
|  | Choice options: | ||||||
|  | --------------- | ||||||
|  |  | ||||||
|  | Are now expected to be an iterable of tuples, with  the first element in each | ||||||
|  | tuple being the actual value to be stored. The second element is the | ||||||
|  | human-readable name for the option. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | PyMongo / MongoDB | ||||||
|  | ----------------- | ||||||
|  |  | ||||||
|  | map reduce now requires pymongo 1.11+- The pymongo merge_output and reduce_output | ||||||
|  | parameters, have been depreciated. | ||||||
|  |  | ||||||
|  | More methods now use map_reduce as db.eval is not supported for sharding as such | ||||||
|  | the following have been changed: | ||||||
|  |  | ||||||
|  |     * :meth:`~mongoengine.queryset.QuerySet.sum` | ||||||
|  |     * :meth:`~mongoengine.queryset.QuerySet.average` | ||||||
|  |     * :meth:`~mongoengine.queryset.QuerySet.item_frequencies` | ||||||
|  |  | ||||||
|  |  | ||||||
|  | Default collection naming | ||||||
|  | ------------------------- | ||||||
|  |  | ||||||
|  | Previously it was just lowercase, its now much more pythonic and readable as its | ||||||
|  | lowercase and underscores, previously :: | ||||||
|  |  | ||||||
|  |     class MyAceDocument(Document): | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     MyAceDocument._meta['collection'] == myacedocument | ||||||
|  |  | ||||||
|  | In 0.5 this will change to :: | ||||||
|  |  | ||||||
|  |     class MyAceDocument(Document): | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     MyAceDocument._get_collection_name() == my_ace_document | ||||||
|  |  | ||||||
|  | To upgrade use a Mixin class to set meta like so :: | ||||||
|  |  | ||||||
|  |     class BaseMixin(object): | ||||||
|  |         meta = { | ||||||
|  |             'collection': lambda c: c.__name__.lower() | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |     class MyAceDocument(Document, BaseMixin): | ||||||
|  |         pass | ||||||
|  |  | ||||||
|  |     MyAceDocument._get_collection_name() == myacedocument | ||||||
|  |  | ||||||
|  | Alternatively, you can rename your collections eg :: | ||||||
|  |  | ||||||
|  |     from mongoengine.connection import _get_db | ||||||
|  |     from mongoengine.base import _document_registry | ||||||
|  |  | ||||||
|  |     def rename_collections(): | ||||||
|  |         db = _get_db() | ||||||
|  |  | ||||||
|  |         failure = False | ||||||
|  |  | ||||||
|  |         collection_names = [d._get_collection_name() for d in _document_registry.values()] | ||||||
|  |  | ||||||
|  |         for new_style_name in collection_names: | ||||||
|  |             if not new_style_name:  # embedded documents don't have collections | ||||||
|  |                 continue | ||||||
|  |             old_style_name = new_style_name.replace('_', '') | ||||||
|  |  | ||||||
|  |             if old_style_name == new_style_name: | ||||||
|  |                 continue  # Nothing to do | ||||||
|  |  | ||||||
|  |             existing = db.collection_names() | ||||||
|  |             if old_style_name in existing: | ||||||
|  |                 if new_style_name in existing: | ||||||
|  |                     failure = True | ||||||
|  |                     print "FAILED to rename: %s to %s (already exists)" % ( | ||||||
|  |                         old_style_name, new_style_name) | ||||||
|  |                 else: | ||||||
|  |                     db[old_style_name].rename(new_style_name) | ||||||
|  |                     print "Renamed:  %s to %s" % (old_style_name, new_style_name) | ||||||
|  |  | ||||||
|  |         if failure: | ||||||
|  |             print "Upgrading  collection names failed" | ||||||
|  |         else: | ||||||
|  |             print "Upgraded collection names" | ||||||
|  |  | ||||||
| @@ -1,385 +0,0 @@ | |||||||
| ========== |  | ||||||
| User Guide |  | ||||||
| ========== |  | ||||||
|  |  | ||||||
| .. _guide-connecting: |  | ||||||
|  |  | ||||||
| Installing |  | ||||||
| ========== |  | ||||||
| MongoEngine is available on PyPI, so to use it you can use  |  | ||||||
| :program:`easy_install` |  | ||||||
|      |  | ||||||
| .. code-block:: console |  | ||||||
|  |  | ||||||
|     # easy_install mongoengine |  | ||||||
|  |  | ||||||
| Alternatively, if you don't have setuptools installed, `download it from PyPi  |  | ||||||
| <http://pypi.python.org/pypi/mongoengine/>`_ and run |  | ||||||
|  |  | ||||||
| .. code-block:: console |  | ||||||
|  |  | ||||||
|     # python setup.py install |  | ||||||
|  |  | ||||||
| Connecting to MongoDB |  | ||||||
| ===================== |  | ||||||
| To connect to a running instance of :program:`mongod`, use the |  | ||||||
| :func:`~mongoengine.connect` function. The first argument is the name of the |  | ||||||
| database to connect to. If the database does not exist, it will be created. If |  | ||||||
| the database requires authentication, :attr:`username` and :attr:`password` |  | ||||||
| arguments may be provided:: |  | ||||||
|  |  | ||||||
|     from mongoengine import connect |  | ||||||
|     connect('project1', username='webapp', password='pwd123') |  | ||||||
|  |  | ||||||
| By default, MongoEngine assumes that the :program:`mongod` instance is running |  | ||||||
| on **localhost** on port **27017**. If MongoDB is running elsewhere, you may |  | ||||||
| provide :attr:`host` and :attr:`port` arguments to |  | ||||||
| :func:`~mongoengine.connect`:: |  | ||||||
|  |  | ||||||
|     connect('project1', host='192.168.1.35', port=12345) |  | ||||||
|  |  | ||||||
| Defining documents |  | ||||||
| ================== |  | ||||||
| In MongoDB, a **document** is roughly equivalent to a **row** in an RDBMS. When |  | ||||||
| working with relational databases, rows are stored in **tables**, which have a |  | ||||||
| strict **schema** that the rows follow. MongoDB stores documents in |  | ||||||
| **collections** rather than tables - the principle difference is that no schema  |  | ||||||
| is enforced at a database level.  |  | ||||||
|  |  | ||||||
| Defining a document's schema |  | ||||||
| ---------------------------- |  | ||||||
| MongoEngine allows you to define schemata for documents as this helps to reduce |  | ||||||
| coding errors, and allows for utility methods to be defined on fields which may |  | ||||||
| be present.  |  | ||||||
|  |  | ||||||
| To define a schema for a document, create a class that inherits from |  | ||||||
| :class:`~mongoengine.Document`. Fields are specified by adding **field |  | ||||||
| objects** as class attributes to the document class:: |  | ||||||
|  |  | ||||||
|     from mongoengine import * |  | ||||||
|     import datetime |  | ||||||
|      |  | ||||||
|     class Page(Document): |  | ||||||
|         title = StringField(max_length=200, required=True) |  | ||||||
|         date_modified = DateTimeField(default=datetime.now) |  | ||||||
|  |  | ||||||
| Fields |  | ||||||
| ------ |  | ||||||
| By default, fields are not required. To make a field mandatory, set the |  | ||||||
| :attr:`required` keyword argument of a field to ``True``. Fields also may have |  | ||||||
| validation constraints available (such as :attr:`max_length` in the example |  | ||||||
| above). Fields may also take default values, which will be used if a value is |  | ||||||
| not provided. Default values may optionally be a callable, which will be called |  | ||||||
| to retrieve the value (such as in the above example). The field types available  |  | ||||||
| are as follows: |  | ||||||
|  |  | ||||||
| * :class:`~mongoengine.StringField` |  | ||||||
| * :class:`~mongoengine.IntField` |  | ||||||
| * :class:`~mongoengine.FloatField` |  | ||||||
| * :class:`~mongoengine.DateTimeField` |  | ||||||
| * :class:`~mongoengine.ListField` |  | ||||||
| * :class:`~mongoengine.ObjectIdField` |  | ||||||
| * :class:`~mongoengine.EmbeddedDocumentField` |  | ||||||
| * :class:`~mongoengine.ReferenceField` |  | ||||||
|  |  | ||||||
| List fields |  | ||||||
| ^^^^^^^^^^^ |  | ||||||
| MongoDB allows the storage of lists of items. To add a list of items to a |  | ||||||
| :class:`~mongoengine.Document`, use the :class:`~mongoengine.ListField` field |  | ||||||
| type. :class:`~mongoengine.ListField` takes another field object as its first |  | ||||||
| argument, which specifies which type elements may be stored within the list:: |  | ||||||
|  |  | ||||||
|     class Page(Document): |  | ||||||
|         tags = ListField(StringField(max_length=50)) |  | ||||||
|  |  | ||||||
| Embedded documents |  | ||||||
| ^^^^^^^^^^^^^^^^^^ |  | ||||||
| MongoDB has the ability to embed documents within other documents. Schemata may |  | ||||||
| be defined for these embedded documents, just as they may be for regular |  | ||||||
| documents. To create an embedded document, just define a document as usual, but |  | ||||||
| inherit from :class:`~mongoengine.EmbeddedDocument` rather than  |  | ||||||
| :class:`~mongoengine.Document`:: |  | ||||||
|  |  | ||||||
|     class Comment(EmbeddedDocument): |  | ||||||
|         content = StringField() |  | ||||||
|  |  | ||||||
| To embed the document within another document, use the |  | ||||||
| :class:`~mongoengine.EmbeddedDocumentField` field type, providing the embedded |  | ||||||
| document class as the first argument:: |  | ||||||
|  |  | ||||||
|     class Page(Document): |  | ||||||
|         comments = ListField(EmbeddedDocumentField(Comment)) |  | ||||||
|  |  | ||||||
|     comment1 = Comment('Good work!') |  | ||||||
|     comment2 = Comment('Nice article!') |  | ||||||
|     page = Page(comments=[comment1, comment2]) |  | ||||||
|  |  | ||||||
| Reference fields |  | ||||||
| ^^^^^^^^^^^^^^^^ |  | ||||||
| References may be stored to other documents in the database using the |  | ||||||
| :class:`~mongoengine.ReferenceField`. Pass in another document class as the |  | ||||||
| first argument to the constructor, then simply assign document objects to the |  | ||||||
| field:: |  | ||||||
|      |  | ||||||
|     class User(Document): |  | ||||||
|         name = StringField() |  | ||||||
|  |  | ||||||
|     class Page(Document): |  | ||||||
|         content = StringField() |  | ||||||
|         author = ReferenceField(User) |  | ||||||
|  |  | ||||||
|     john = User(name="John Smith") |  | ||||||
|     john.save() |  | ||||||
|  |  | ||||||
|     post = Page(content="Test Page") |  | ||||||
|     post.author = john |  | ||||||
|     post.save() |  | ||||||
|  |  | ||||||
| The :class:`User` object is automatically turned into a reference behind the |  | ||||||
| scenes, and dereferenced when the :class:`Page` object is retrieved. |  | ||||||
|  |  | ||||||
| Document collections |  | ||||||
| -------------------- |  | ||||||
| Document classes that inherit **directly** from :class:`~mongoengine.Document` |  | ||||||
| will have their own **collection** in the database. The name of the collection |  | ||||||
| is by default the name of the class, coverted to lowercase (so in the example |  | ||||||
| above, the collection would be called `page`). If you need to change the name |  | ||||||
| of the collection (e.g. to use MongoEngine with an existing database), then |  | ||||||
| create a class dictionary attribute called :attr:`meta` on your document, and |  | ||||||
| set :attr:`collection` to the name of the collection that you want your |  | ||||||
| document class to use:: |  | ||||||
|  |  | ||||||
|     class Page(Document): |  | ||||||
|         title = StringField(max_length=200, required=True) |  | ||||||
|         meta = {'collection': 'cmsPage'} |  | ||||||
|  |  | ||||||
| Capped collections |  | ||||||
| ^^^^^^^^^^^^^^^^^^ |  | ||||||
| A :class:`~mongoengine.Document` may use a **Capped Collection** by specifying |  | ||||||
| :attr:`max_documents` and :attr:`max_size` in the :attr:`meta` dictionary. |  | ||||||
| :attr:`max_documents` is the maximum number of documents that is allowed to be |  | ||||||
| stored in the collection, and :attr:`max_size` is the maximum size of the |  | ||||||
| collection in bytes. If :attr:`max_size` is not specified and |  | ||||||
| :attr:`max_documents` is, :attr:`max_size` defaults to 10000000 bytes (10MB). |  | ||||||
| The following example shows a :class:`Log` document that will be limited to  |  | ||||||
| 1000 entries and 2MB of disk space:: |  | ||||||
|  |  | ||||||
|     class Log(Document): |  | ||||||
|         ip_address = StringField() |  | ||||||
|         meta = {'max_documents': 1000, 'max_size': 2000000} |  | ||||||
|  |  | ||||||
| Document inheritance |  | ||||||
| -------------------- |  | ||||||
| To create a specialised type of a :class:`~mongoengine.Document` you have |  | ||||||
| defined, you may subclass it and add any extra fields or methods you may need. |  | ||||||
| As this is new class is not a direct subclass of |  | ||||||
| :class:`~mongoengine.Document`, it will not be stored in its own collection; it |  | ||||||
| will use the same collection as its superclass uses. This allows for more |  | ||||||
| convenient and efficient retrieval of related documents:: |  | ||||||
|  |  | ||||||
|     # Stored in a collection named 'page' |  | ||||||
|     class Page(Document): |  | ||||||
|         title = StringField(max_length=200, required=True) |  | ||||||
|  |  | ||||||
|     # Also stored in the collection named 'page' |  | ||||||
|     class DatedPage(Page): |  | ||||||
|         date = DateTimeField() |  | ||||||
|  |  | ||||||
| Working with existing data |  | ||||||
| ^^^^^^^^^^^^^^^^^^^^^^^^^^ |  | ||||||
| To enable correct retrieval of documents involved in this kind of heirarchy, |  | ||||||
| two extra attributes are stored on each document in the database: :attr:`_cls` |  | ||||||
| and :attr:`_types`. These are hidden from the user through the MongoEngine |  | ||||||
| interface, but may not be present if you are trying to use MongoEngine with  |  | ||||||
| an existing database. For this reason, you may disable this inheritance |  | ||||||
| mechansim, removing the dependency of :attr:`_cls` and :attr:`_types`, enabling |  | ||||||
| you to work with existing databases. To disable inheritance on a document |  | ||||||
| class, set :attr:`allow_inheritance` to ``False`` in the :attr:`meta` |  | ||||||
| dictionary:: |  | ||||||
|  |  | ||||||
|     # Will work with data in an existing collection named 'cmsPage' |  | ||||||
|     class Page(Document): |  | ||||||
|         title = StringField(max_length=200, required=True) |  | ||||||
|         meta = { |  | ||||||
|             'collection': 'cmsPage', |  | ||||||
|             'allow_inheritance': False, |  | ||||||
|         } |  | ||||||
|  |  | ||||||
| Documents instances |  | ||||||
| =================== |  | ||||||
| To create a new document object, create an instance of the relevant document |  | ||||||
| class, providing values for its fields as its constructor keyword arguments. |  | ||||||
| You may provide values for any of the fields on the document:: |  | ||||||
|      |  | ||||||
|     >>> page = Page(title="Test Page") |  | ||||||
|     >>> page.title |  | ||||||
|     'Test Page' |  | ||||||
|  |  | ||||||
| You may also assign values to the document's fields using standard object  |  | ||||||
| attribute syntax:: |  | ||||||
|  |  | ||||||
|     >>> page.title = "Example Page" |  | ||||||
|     >>> page.title |  | ||||||
|     'Example Page' |  | ||||||
|  |  | ||||||
| Saving and deleting documents |  | ||||||
| ----------------------------- |  | ||||||
| To save the document to the database, call the |  | ||||||
| :meth:`~mongoengine.Document.save` method. If the document does not exist in |  | ||||||
| the database, it will be created. If it does already exist, it will be |  | ||||||
| updated. |  | ||||||
|  |  | ||||||
| To delete a document, call the :meth:`~mongoengine.Document.delete` method. |  | ||||||
| Note that this will only work if the document exists in the database and has a |  | ||||||
| valide :attr:`id`. |  | ||||||
|  |  | ||||||
| Document IDs |  | ||||||
| ------------ |  | ||||||
| Each document in the database has a unique id. This may be accessed through the |  | ||||||
| :attr:`id` attribute on :class:`~mongoengine.Document` objects. Usually, the id |  | ||||||
| will be generated automatically by the database server when the object is save, |  | ||||||
| meaning that you may only access the :attr:`id` field once a document has been |  | ||||||
| saved:: |  | ||||||
|  |  | ||||||
|     >>> page = Page(title="Test Page") |  | ||||||
|     >>> page.id |  | ||||||
|     >>> page.save() |  | ||||||
|     >>> page.id |  | ||||||
|     ObjectId('123456789abcdef000000000') |  | ||||||
|  |  | ||||||
| Alternatively, you may explicitly set the :attr:`id` before you save the |  | ||||||
| document, but the id must be a valid PyMongo :class:`ObjectId`. |  | ||||||
|  |  | ||||||
| Querying the database |  | ||||||
| ===================== |  | ||||||
| :class:`~mongoengine.Document` classes have an :attr:`objects` attribute, which |  | ||||||
| is used for accessing the objects in the database associated with the class. |  | ||||||
| The :attr:`objects` attribute is actually a |  | ||||||
| :class:`~mongoengine.queryset.QuerySetManager`, which creates and returns a new |  | ||||||
| a new :class:`~mongoengine.queryset.QuerySet` object on access. The |  | ||||||
| :class:`~mongoengine.queryset.QuerySet` object may may be iterated over to |  | ||||||
| fetch documents from the database:: |  | ||||||
|  |  | ||||||
|     # Prints out the names of all the users in the database |  | ||||||
|     for user in User.objects: |  | ||||||
|         print user.name |  | ||||||
|  |  | ||||||
| Filtering queries |  | ||||||
| ----------------- |  | ||||||
| The query may be filtered by calling the |  | ||||||
| :class:`~mongoengine.queryset.QuerySet` object with field lookup keyword  |  | ||||||
| arguments. The keys in the keyword arguments correspond to fields on the |  | ||||||
| :class:`~mongoengine.Document` you are querying:: |  | ||||||
|  |  | ||||||
|     # This will return a QuerySet that will only iterate over users whose |  | ||||||
|     # 'country' field is set to 'uk' |  | ||||||
|     uk_users = User.objects(country='uk') |  | ||||||
|  |  | ||||||
| Fields on embedded documents may also be referred to using field lookup syntax |  | ||||||
| by using a double-underscore in place of the dot in object attribute access |  | ||||||
| syntax:: |  | ||||||
|      |  | ||||||
|     # This will return a QuerySet that will only iterate over pages that have |  | ||||||
|     # been written by a user whose 'country' field is set to 'uk' |  | ||||||
|     uk_pages = Page.objects(author__country='uk') |  | ||||||
|  |  | ||||||
| Querying lists |  | ||||||
| ^^^^^^^^^^^^^^ |  | ||||||
| On most fields, this syntax will look up documents where the field specified |  | ||||||
| matches the given value exactly, but when the field refers to a |  | ||||||
| :class:`~mongoengine.ListField`, a single item may be provided, in which case |  | ||||||
| lists that contain that item will be matched:: |  | ||||||
|  |  | ||||||
|     class Page(Document): |  | ||||||
|         tags = ListField(StringField()) |  | ||||||
|  |  | ||||||
|     # This will match all pages that have the word 'coding' as an item in the |  | ||||||
|     # 'tags' list |  | ||||||
|     Page.objects(tags='coding') |  | ||||||
|  |  | ||||||
| Query operators |  | ||||||
| --------------- |  | ||||||
| Operators other than equality may also be used in queries; just attach the |  | ||||||
| operator name to a key with a double-underscore:: |  | ||||||
|      |  | ||||||
|     # Only find users whose age is 18 or less |  | ||||||
|     young_users = Users.objects(age__lte=18) |  | ||||||
|  |  | ||||||
| Available operators are as follows: |  | ||||||
|  |  | ||||||
| * ``neq`` -- not equal to |  | ||||||
| * ``lt`` -- less than |  | ||||||
| * ``lte`` -- less than or equal to |  | ||||||
| * ``gt`` -- greater than |  | ||||||
| * ``gte`` -- greater than or equal to |  | ||||||
| * ``in`` -- value is in list (a list of values should be provided) |  | ||||||
| * ``nin`` -- value is not in list (a list of values should be provided) |  | ||||||
| * ``mod`` -- ``value % x == y``, where ``x`` and ``y`` are two provided values |  | ||||||
| * ``all`` -- every item in array is in list of values provided |  | ||||||
| * ``size`` -- the size of the array is  |  | ||||||
| * ``exists`` -- value for field exists |  | ||||||
|  |  | ||||||
| Limiting and skipping results |  | ||||||
| ----------------------------- |  | ||||||
| Just as with traditional ORMs, you may limit the number of results returned, or |  | ||||||
| skip a number or results in you query. |  | ||||||
| :meth:`~mongoengine.queryset.QuerySet.limit` and |  | ||||||
| :meth:`~mongoengine.queryset.QuerySet.skip` and methods are available on |  | ||||||
| :class:`~mongoengine.queryset.QuerySet` objects, but the prefered syntax for |  | ||||||
| achieving this is using array-slicing syntax:: |  | ||||||
|  |  | ||||||
|     # Only the first 5 people |  | ||||||
|     users = User.objects[:5] |  | ||||||
|  |  | ||||||
|     # All except for the first 5 people |  | ||||||
|     users = User.objects[5:] |  | ||||||
|  |  | ||||||
|     # 5 users, starting from the 10th user found |  | ||||||
|     users = User.objects[10:15] |  | ||||||
|  |  | ||||||
| Aggregation |  | ||||||
| ----------- |  | ||||||
| MongoDB provides some aggregation methods out of the box, but there are not as |  | ||||||
| many as you typically get with an RDBMS. MongoEngine provides a wrapper around |  | ||||||
| the built-in methods and provides some of its own, which are implemented as |  | ||||||
| Javascript code that is executed on the database server. |  | ||||||
|  |  | ||||||
| Counting results |  | ||||||
| ^^^^^^^^^^^^^^^^ |  | ||||||
| Just as with limiting and skipping results, there is a method on |  | ||||||
| :class:`~mongoengine.queryset.QuerySet` objects --  |  | ||||||
| :meth:`~mongoengine.queryset.QuerySet.count`, but there is also a more Pythonic |  | ||||||
| way of achieving this:: |  | ||||||
|  |  | ||||||
|     num_users = len(User.objects) |  | ||||||
|  |  | ||||||
| Further aggregation |  | ||||||
| ^^^^^^^^^^^^^^^^^^^ |  | ||||||
| You may sum over the values of a specific field on documents using |  | ||||||
| :meth:`~mongoengine.queryset.QuerySet.sum`:: |  | ||||||
|  |  | ||||||
|     yearly_expense = Employee.objects.sum('salary') |  | ||||||
|  |  | ||||||
| .. note:: |  | ||||||
|    If the field isn't present on a document, that document will be ignored from |  | ||||||
|    the sum. |  | ||||||
|  |  | ||||||
| To get the average (mean) of a field on a collection of documents, use |  | ||||||
| :meth:`~mongoengine.queryset.QuerySet.average`:: |  | ||||||
|  |  | ||||||
|     mean_age = User.objects.average('age') |  | ||||||
|  |  | ||||||
| As MongoDB provides native lists, MongoEngine provides a helper method to get a |  | ||||||
| dictionary of the frequencies of items in lists across an entire collection -- |  | ||||||
| :meth:`~mongoengine.queryset.QuerySet.item_frequencies`. An example of its use |  | ||||||
| would be generating "tag-clouds":: |  | ||||||
|  |  | ||||||
|     class Article(Document): |  | ||||||
|         tag = ListField(StringField()) |  | ||||||
|  |  | ||||||
|     # After adding some tagged articles... |  | ||||||
|     tag_freqs = Article.objects.item_frequencies('tag', normalize=True) |  | ||||||
|  |  | ||||||
|     from operator import itemgetter |  | ||||||
|     top_tags = sorted(tag_freqs.items(), key=itemgetter(1), reverse=True)[:10] |  | ||||||
|  |  | ||||||
| @@ -6,13 +6,14 @@ import connection | |||||||
| from connection import * | from connection import * | ||||||
| import queryset | import queryset | ||||||
| from queryset import * | from queryset import * | ||||||
|  | import signals | ||||||
|  | from signals import * | ||||||
|  |  | ||||||
| __all__ = (document.__all__ + fields.__all__ + connection.__all__ + | __all__ = (document.__all__ + fields.__all__ + connection.__all__ + | ||||||
|            queryset.__all__) |            queryset.__all__ + signals.__all__) | ||||||
|  |  | ||||||
| __author__ = 'Harry Marr' | VERSION = (0, 6, 1) | ||||||
|  |  | ||||||
| VERSION = (0, 1, 1) |  | ||||||
|  |  | ||||||
| def get_version(): | def get_version(): | ||||||
|     version = '%s.%s' % (VERSION[0], VERSION[1]) |     version = '%s.%s' % (VERSION[0], VERSION[1]) | ||||||
| @@ -21,4 +22,3 @@ def get_version(): | |||||||
|     return version |     return version | ||||||
|  |  | ||||||
| __version__ = get_version() | __version__ = get_version() | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										1256
									
								
								mongoengine/base.py
									
									
									
									
									
								
							
							
						
						
									
										1256
									
								
								mongoengine/base.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,46 +1,157 @@ | |||||||
| from pymongo import Connection | import pymongo | ||||||
|  | from pymongo import Connection, ReplicaSetConnection, uri_parser | ||||||
|  |  | ||||||
|  |  | ||||||
| __all__ = ['ConnectionError', 'connect'] | __all__ = ['ConnectionError', 'connect', 'register_connection', | ||||||
|  |            'DEFAULT_CONNECTION_NAME'] | ||||||
|  |  | ||||||
|  |  | ||||||
| _connection_settings = { | DEFAULT_CONNECTION_NAME = 'default' | ||||||
|     'host': 'localhost', |  | ||||||
|     'port': 27017, |  | ||||||
|     'pool_size': 1, |  | ||||||
| } |  | ||||||
| _connection = None |  | ||||||
| _db = None |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class ConnectionError(Exception): | class ConnectionError(Exception): | ||||||
|     pass |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
| def _get_connection(): | _connection_settings = {} | ||||||
|     global _connection | _connections = {} | ||||||
|     if _connection is None: | _dbs = {} | ||||||
|         _connection = Connection(**_connection_settings) |  | ||||||
|     return _connection |  | ||||||
|  |  | ||||||
| def _get_db(): |  | ||||||
|     global _db |  | ||||||
|     if _db is None: |  | ||||||
|         raise ConnectionError('Not connected to database') |  | ||||||
|     return _db |  | ||||||
|  |  | ||||||
| def connect(db, username=None, password=None, **kwargs): | def register_connection(alias, name, host='localhost', port=27017, | ||||||
|     """Connect to the database specified by the 'db' argument. Connection  |                         is_slave=False, read_preference=False, slaves=None, | ||||||
|     settings may be provided here as well if the database is not running on |                         username=None, password=None, **kwargs): | ||||||
|     the default port on localhost. If authentication is needed, provide |     """Add a connection. | ||||||
|     username and password arguments as well. |  | ||||||
|  |     :param alias: the name that will be used to refer to this connection | ||||||
|  |         throughout MongoEngine | ||||||
|  |     :param name: the name of the specific database to use | ||||||
|  |     :param host: the host name of the :program:`mongod` instance to connect to | ||||||
|  |     :param port: the port that the :program:`mongod` instance is running on | ||||||
|  |     :param is_slave: whether the connection can act as a slave ** Depreciated pymongo 2.0.1+ | ||||||
|  |     :param read_preference: The read preference for the collection ** Added pymongo 2.1 | ||||||
|  |     :param slaves: a list of aliases of slave connections; each of these must | ||||||
|  |         be a registered connection that has :attr:`is_slave` set to ``True`` | ||||||
|  |     :param username: username to authenticate with | ||||||
|  |     :param password: password to authenticate with | ||||||
|  |     :param kwargs: allow ad-hoc parameters to be passed into the pymongo driver | ||||||
|  |  | ||||||
|     """ |     """ | ||||||
|     global _db |     global _connection_settings | ||||||
|  |  | ||||||
|     _connection_settings.update(kwargs) |     # Handle uri style connections | ||||||
|     connection = _get_connection() |     if "://" in host: | ||||||
|     # Get DB from connection and auth if necessary |         uri_dict = uri_parser.parse_uri(host) | ||||||
|     _db = connection[db] |         if uri_dict.get('database') is None: | ||||||
|     if username is not None and password is not None: |             raise ConnectionError("If using URI style connection include "\ | ||||||
|         _db.authenticate(username, password) |                                   "database name in string") | ||||||
|  |         _connection_settings[alias] = { | ||||||
|  |             'host': host, | ||||||
|  |             'name': uri_dict.get('database'), | ||||||
|  |             'username': uri_dict.get('username'), | ||||||
|  |             'password': uri_dict.get('password') | ||||||
|  |         } | ||||||
|  |         _connection_settings[alias].update(kwargs) | ||||||
|  |         return | ||||||
|  |  | ||||||
|  |     _connection_settings[alias] = { | ||||||
|  |         'name': name, | ||||||
|  |         'host': host, | ||||||
|  |         'port': port, | ||||||
|  |         'is_slave': is_slave, | ||||||
|  |         'slaves': slaves or [], | ||||||
|  |         'username': username, | ||||||
|  |         'password': password, | ||||||
|  |         'read_preference': read_preference | ||||||
|  |     } | ||||||
|  |     _connection_settings[alias].update(kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def disconnect(alias=DEFAULT_CONNECTION_NAME): | ||||||
|  |     global _connections | ||||||
|  |     global _dbs | ||||||
|  |  | ||||||
|  |     if alias in _connections: | ||||||
|  |         get_connection(alias=alias).disconnect() | ||||||
|  |         del _connections[alias] | ||||||
|  |     if alias in _dbs: | ||||||
|  |         del _dbs[alias] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_connection(alias=DEFAULT_CONNECTION_NAME, reconnect=False): | ||||||
|  |     global _connections | ||||||
|  |     # Connect to the database if not already connected | ||||||
|  |     if reconnect: | ||||||
|  |         disconnect(alias) | ||||||
|  |  | ||||||
|  |     if alias not in _connections: | ||||||
|  |         if alias not in _connection_settings: | ||||||
|  |             msg = 'Connection with alias "%s" has not been defined' | ||||||
|  |             if alias == DEFAULT_CONNECTION_NAME: | ||||||
|  |                 msg = 'You have not defined a default connection' | ||||||
|  |             raise ConnectionError(msg) | ||||||
|  |         conn_settings = _connection_settings[alias].copy() | ||||||
|  |  | ||||||
|  |         if hasattr(pymongo, 'version_tuple'):  # Support for 2.1+ | ||||||
|  |             conn_settings.pop('name', None) | ||||||
|  |             conn_settings.pop('slaves', None) | ||||||
|  |             conn_settings.pop('is_slave', None) | ||||||
|  |             conn_settings.pop('username', None) | ||||||
|  |             conn_settings.pop('password', None) | ||||||
|  |         else: | ||||||
|  |             # Get all the slave connections | ||||||
|  |             if 'slaves' in conn_settings: | ||||||
|  |                 slaves = [] | ||||||
|  |                 for slave_alias in conn_settings['slaves']: | ||||||
|  |                     slaves.append(get_connection(slave_alias)) | ||||||
|  |                 conn_settings['slaves'] = slaves | ||||||
|  |                 conn_settings.pop('read_preference') | ||||||
|  |  | ||||||
|  |         connection_class = Connection | ||||||
|  |         if 'replicaSet' in conn_settings: | ||||||
|  |             conn_settings['hosts_or_uri'] = conn_settings.pop('host', None) | ||||||
|  |             connection_class = ReplicaSetConnection | ||||||
|  |         try: | ||||||
|  |             _connections[alias] = connection_class(**conn_settings) | ||||||
|  |         except Exception, e: | ||||||
|  |             raise ConnectionError("Cannot connect to database %s :\n%s" % (alias, e)) | ||||||
|  |     return _connections[alias] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_db(alias=DEFAULT_CONNECTION_NAME, reconnect=False): | ||||||
|  |     global _dbs | ||||||
|  |     if reconnect: | ||||||
|  |         disconnect(alias) | ||||||
|  |  | ||||||
|  |     if alias not in _dbs: | ||||||
|  |         conn = get_connection(alias) | ||||||
|  |         conn_settings = _connection_settings[alias] | ||||||
|  |         _dbs[alias] = conn[conn_settings['name']] | ||||||
|  |         # Authenticate if necessary | ||||||
|  |         if conn_settings['username'] and conn_settings['password']: | ||||||
|  |             _dbs[alias].authenticate(conn_settings['username'], | ||||||
|  |                                      conn_settings['password']) | ||||||
|  |     return _dbs[alias] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def connect(db, alias=DEFAULT_CONNECTION_NAME, **kwargs): | ||||||
|  |     """Connect to the database specified by the 'db' argument. | ||||||
|  |  | ||||||
|  |     Connection settings may be provided here as well if the database is not | ||||||
|  |     running on the default port on localhost. If authentication is needed, | ||||||
|  |     provide username and password arguments as well. | ||||||
|  |  | ||||||
|  |     Multiple databases are supported by using aliases.  Provide a separate | ||||||
|  |     `alias` to connect to a different instance of :program:`mongod`. | ||||||
|  |  | ||||||
|  |     .. versionchanged:: 0.6 - added multiple database support. | ||||||
|  |     """ | ||||||
|  |     global _connections | ||||||
|  |     if alias not in _connections: | ||||||
|  |         register_connection(alias, db, **kwargs) | ||||||
|  |  | ||||||
|  |     return get_connection(alias) | ||||||
|  |  | ||||||
|  | # Support old naming convention | ||||||
|  | _get_connection = get_connection | ||||||
|  | _get_db = get_db | ||||||
|   | |||||||
							
								
								
									
										188
									
								
								mongoengine/dereference.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										188
									
								
								mongoengine/dereference.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,188 @@ | |||||||
|  | from bson import DBRef, SON | ||||||
|  |  | ||||||
|  | from base import (BaseDict, BaseList, TopLevelDocumentMetaclass, get_document) | ||||||
|  | from fields import (ReferenceField, ListField, DictField, MapField) | ||||||
|  | from connection import get_db | ||||||
|  | from queryset import QuerySet | ||||||
|  | from document import Document | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DeReference(object): | ||||||
|  |  | ||||||
|  |     def __call__(self, items, max_depth=1, instance=None, name=None): | ||||||
|  |         """ | ||||||
|  |         Cheaply dereferences the items to a set depth. | ||||||
|  |         Also handles the convertion of complex data types. | ||||||
|  |  | ||||||
|  |         :param items: The iterable (dict, list, queryset) to be dereferenced. | ||||||
|  |         :param max_depth: The maximum depth to recurse to | ||||||
|  |         :param instance: The owning instance used for tracking changes by | ||||||
|  |             :class:`~mongoengine.base.ComplexBaseField` | ||||||
|  |         :param name: The name of the field, used for tracking changes by | ||||||
|  |             :class:`~mongoengine.base.ComplexBaseField` | ||||||
|  |         :param get: A boolean determining if being called by __get__ | ||||||
|  |         """ | ||||||
|  |         if items is None or isinstance(items, basestring): | ||||||
|  |             return items | ||||||
|  |  | ||||||
|  |         # cheapest way to convert a queryset to a list | ||||||
|  |         # list(queryset) uses a count() query to determine length | ||||||
|  |         if isinstance(items, QuerySet): | ||||||
|  |             items = [i for i in items] | ||||||
|  |  | ||||||
|  |         self.max_depth = max_depth | ||||||
|  |  | ||||||
|  |         doc_type = None | ||||||
|  |         if instance and instance._fields: | ||||||
|  |             doc_type = instance._fields[name].field | ||||||
|  |  | ||||||
|  |             if isinstance(doc_type, ReferenceField): | ||||||
|  |                 doc_type = doc_type.document_type | ||||||
|  |                 if all([i.__class__ == doc_type for i in items]): | ||||||
|  |                     return items | ||||||
|  |  | ||||||
|  |         self.reference_map = self._find_references(items) | ||||||
|  |         self.object_map = self._fetch_objects(doc_type=doc_type) | ||||||
|  |         return self._attach_objects(items, 0, instance, name) | ||||||
|  |  | ||||||
|  |     def _find_references(self, items, depth=0): | ||||||
|  |         """ | ||||||
|  |         Recursively finds all db references to be dereferenced | ||||||
|  |  | ||||||
|  |         :param items: The iterable (dict, list, queryset) | ||||||
|  |         :param depth: The current depth of recursion | ||||||
|  |         """ | ||||||
|  |         reference_map = {} | ||||||
|  |         if not items or depth >= self.max_depth: | ||||||
|  |             return reference_map | ||||||
|  |  | ||||||
|  |         # Determine the iterator to use | ||||||
|  |         if not hasattr(items, 'items'): | ||||||
|  |             iterator = enumerate(items) | ||||||
|  |         else: | ||||||
|  |             iterator = items.iteritems() | ||||||
|  |  | ||||||
|  |         # Recursively find dbreferences | ||||||
|  |         depth += 1 | ||||||
|  |         for k, item in iterator: | ||||||
|  |             if hasattr(item, '_fields'): | ||||||
|  |                 for field_name, field in item._fields.iteritems(): | ||||||
|  |                     v = item._data.get(field_name, None) | ||||||
|  |                     if isinstance(v, (DBRef)): | ||||||
|  |                         reference_map.setdefault(field.document_type, []).append(v.id) | ||||||
|  |                     elif isinstance(v, (dict, SON)) and '_ref' in v: | ||||||
|  |                         reference_map.setdefault(get_document(v['_cls']), []).append(v['_ref'].id) | ||||||
|  |                     elif isinstance(v, (dict, list, tuple)) and depth <= self.max_depth: | ||||||
|  |                         field_cls = getattr(getattr(field, 'field', None), 'document_type', None) | ||||||
|  |                         references = self._find_references(v, depth) | ||||||
|  |                         for key, refs in references.iteritems(): | ||||||
|  |                             if isinstance(field_cls, (Document, TopLevelDocumentMetaclass)): | ||||||
|  |                                 key = field_cls | ||||||
|  |                             reference_map.setdefault(key, []).extend(refs) | ||||||
|  |             elif isinstance(item, (DBRef)): | ||||||
|  |                 reference_map.setdefault(item.collection, []).append(item.id) | ||||||
|  |             elif isinstance(item, (dict, SON)) and '_ref' in item: | ||||||
|  |                 reference_map.setdefault(get_document(item['_cls']), []).append(item['_ref'].id) | ||||||
|  |             elif isinstance(item, (dict, list, tuple)) and depth - 1 <= self.max_depth: | ||||||
|  |                 references = self._find_references(item, depth - 1) | ||||||
|  |                 for key, refs in references.iteritems(): | ||||||
|  |                     reference_map.setdefault(key, []).extend(refs) | ||||||
|  |  | ||||||
|  |         return reference_map | ||||||
|  |  | ||||||
|  |     def _fetch_objects(self, doc_type=None): | ||||||
|  |         """Fetch all references and convert to their document objects | ||||||
|  |         """ | ||||||
|  |         object_map = {} | ||||||
|  |         for col, dbrefs in self.reference_map.iteritems(): | ||||||
|  |             keys = object_map.keys() | ||||||
|  |             refs = list(set([dbref for dbref in dbrefs if str(dbref) not in keys])) | ||||||
|  |             if hasattr(col, 'objects'):  # We have a document class for the refs | ||||||
|  |                 references = col.objects.in_bulk(refs) | ||||||
|  |                 for key, doc in references.iteritems(): | ||||||
|  |                     object_map[key] = doc | ||||||
|  |             else:  # Generic reference: use the refs data to convert to document | ||||||
|  |                 if doc_type and not isinstance(doc_type, (ListField, DictField, MapField,) ): | ||||||
|  |                     references = doc_type._get_db()[col].find({'_id': {'$in': refs}}) | ||||||
|  |                     for ref in references: | ||||||
|  |                         doc = doc_type._from_son(ref) | ||||||
|  |                         object_map[doc.id] = doc | ||||||
|  |                 else: | ||||||
|  |                     references = get_db()[col].find({'_id': {'$in': refs}}) | ||||||
|  |                     for ref in references: | ||||||
|  |                         if '_cls' in ref: | ||||||
|  |                             doc = get_document(ref["_cls"])._from_son(ref) | ||||||
|  |                         else: | ||||||
|  |                             doc = doc_type._from_son(ref) | ||||||
|  |                         object_map[doc.id] = doc | ||||||
|  |         return object_map | ||||||
|  |  | ||||||
|  |     def _attach_objects(self, items, depth=0, instance=None, name=None): | ||||||
|  |         """ | ||||||
|  |         Recursively finds all db references to be dereferenced | ||||||
|  |  | ||||||
|  |         :param items: The iterable (dict, list, queryset) | ||||||
|  |         :param depth: The current depth of recursion | ||||||
|  |         :param instance: The owning instance used for tracking changes by | ||||||
|  |             :class:`~mongoengine.base.ComplexBaseField` | ||||||
|  |         :param name: The name of the field, used for tracking changes by | ||||||
|  |             :class:`~mongoengine.base.ComplexBaseField` | ||||||
|  |         """ | ||||||
|  |         if not items: | ||||||
|  |             if isinstance(items, (BaseDict, BaseList)): | ||||||
|  |                 return items | ||||||
|  |  | ||||||
|  |             if instance: | ||||||
|  |                 if isinstance(items, dict): | ||||||
|  |                     return BaseDict(items, instance, name) | ||||||
|  |                 else: | ||||||
|  |                     return BaseList(items, instance, name) | ||||||
|  |  | ||||||
|  |         if isinstance(items, (dict, SON)): | ||||||
|  |             if '_ref' in items: | ||||||
|  |                 return self.object_map.get(items['_ref'].id, items) | ||||||
|  |             elif '_types' in items and '_cls' in items: | ||||||
|  |                 doc = get_document(items['_cls'])._from_son(items) | ||||||
|  |                 doc._data = self._attach_objects(doc._data, depth, doc, name) | ||||||
|  |                 return doc | ||||||
|  |  | ||||||
|  |         if not hasattr(items, 'items'): | ||||||
|  |             is_list = True | ||||||
|  |             iterator = enumerate(items) | ||||||
|  |             data = [] | ||||||
|  |         else: | ||||||
|  |             is_list = False | ||||||
|  |             iterator = items.iteritems() | ||||||
|  |             data = {} | ||||||
|  |  | ||||||
|  |         depth += 1 | ||||||
|  |         for k, v in iterator: | ||||||
|  |             if is_list: | ||||||
|  |                 data.append(v) | ||||||
|  |             else: | ||||||
|  |                 data[k] = v | ||||||
|  |  | ||||||
|  |             if k in self.object_map: | ||||||
|  |                 data[k] = self.object_map[k] | ||||||
|  |             elif hasattr(v, '_fields'): | ||||||
|  |                 for field_name, field in v._fields.iteritems(): | ||||||
|  |                     v = data[k]._data.get(field_name, None) | ||||||
|  |                     if isinstance(v, (DBRef)): | ||||||
|  |                         data[k]._data[field_name] = self.object_map.get(v.id, v) | ||||||
|  |                     elif isinstance(v, (dict, SON)) and '_ref' in v: | ||||||
|  |                         data[k]._data[field_name] = self.object_map.get(v['_ref'].id, v) | ||||||
|  |                     elif isinstance(v, dict) and depth <= self.max_depth: | ||||||
|  |                         data[k]._data[field_name] = self._attach_objects(v, depth, instance=instance, name=name) | ||||||
|  |                     elif isinstance(v, (list, tuple)) and depth <= self.max_depth: | ||||||
|  |                         data[k]._data[field_name] = self._attach_objects(v, depth, instance=instance, name=name) | ||||||
|  |             elif isinstance(v, (dict, list, tuple)) and depth <= self.max_depth: | ||||||
|  |                 data[k] = self._attach_objects(v, depth - 1, instance=instance, name=name) | ||||||
|  |             elif hasattr(v, 'id'): | ||||||
|  |                 data[k] = self.object_map.get(v.id, v) | ||||||
|  |  | ||||||
|  |         if instance and name: | ||||||
|  |             if is_list: | ||||||
|  |                 return BaseList(data, instance, name) | ||||||
|  |             return BaseDict(data, instance, name) | ||||||
|  |         depth += 1 | ||||||
|  |         return data | ||||||
							
								
								
									
										0
									
								
								mongoengine/django/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								mongoengine/django/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										147
									
								
								mongoengine/django/auth.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										147
									
								
								mongoengine/django/auth.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,147 @@ | |||||||
|  | from mongoengine import * | ||||||
|  |  | ||||||
|  | from django.utils.hashcompat import md5_constructor, sha_constructor | ||||||
|  | from django.utils.encoding import smart_str | ||||||
|  | from django.contrib.auth.models import AnonymousUser | ||||||
|  | from django.utils.translation import ugettext_lazy as _ | ||||||
|  |  | ||||||
|  | import datetime | ||||||
|  |  | ||||||
|  | REDIRECT_FIELD_NAME = 'next' | ||||||
|  |  | ||||||
|  | def get_hexdigest(algorithm, salt, raw_password): | ||||||
|  |     raw_password, salt = smart_str(raw_password), smart_str(salt) | ||||||
|  |     if algorithm == 'md5': | ||||||
|  |         return md5_constructor(salt + raw_password).hexdigest() | ||||||
|  |     elif algorithm == 'sha1': | ||||||
|  |         return sha_constructor(salt + raw_password).hexdigest() | ||||||
|  |     raise ValueError('Got unknown password algorithm type in password') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class User(Document): | ||||||
|  |     """A User document that aims to mirror most of the API specified by Django | ||||||
|  |     at http://docs.djangoproject.com/en/dev/topics/auth/#users | ||||||
|  |     """ | ||||||
|  |     username = StringField(max_length=30, required=True, | ||||||
|  |                            verbose_name=_('username'), | ||||||
|  |                            help_text=_("Required. 30 characters or fewer. Letters, numbers and @/./+/-/_ characters")) | ||||||
|  |  | ||||||
|  |     first_name = StringField(max_length=30, | ||||||
|  |                              verbose_name=_('first name')) | ||||||
|  |  | ||||||
|  |     last_name = StringField(max_length=30, | ||||||
|  |                             verbose_name=_('last name')) | ||||||
|  |     email = EmailField(verbose_name=_('e-mail address')) | ||||||
|  |     password = StringField(max_length=128, | ||||||
|  |                            verbose_name=_('password'), | ||||||
|  |                            help_text=_("Use '[algo]$[salt]$[hexdigest]' or use the <a href=\"password/\">change password form</a>.")) | ||||||
|  |     is_staff = BooleanField(default=False, | ||||||
|  |                             verbose_name=_('staff status'), | ||||||
|  |                             help_text=_("Designates whether the user can log into this admin site.")) | ||||||
|  |     is_active = BooleanField(default=True, | ||||||
|  |                              verbose_name=_('active'), | ||||||
|  |                              help_text=_("Designates whether this user should be treated as active. Unselect this instead of deleting accounts.")) | ||||||
|  |     is_superuser = BooleanField(default=False, | ||||||
|  |                                 verbose_name=_('superuser status'), | ||||||
|  |                                 help_text=_("Designates that this user has all permissions without explicitly assigning them.")) | ||||||
|  |     last_login = DateTimeField(default=datetime.datetime.now, | ||||||
|  |                                verbose_name=_('last login')) | ||||||
|  |     date_joined = DateTimeField(default=datetime.datetime.now, | ||||||
|  |                                 verbose_name=_('date joined')) | ||||||
|  |  | ||||||
|  |     meta = { | ||||||
|  |         'indexes': [ | ||||||
|  |             {'fields': ['username'], 'unique': True} | ||||||
|  |         ] | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     def __unicode__(self): | ||||||
|  |         return self.username | ||||||
|  |  | ||||||
|  |     def get_full_name(self): | ||||||
|  |         """Returns the users first and last names, separated by a space. | ||||||
|  |         """ | ||||||
|  |         full_name = u'%s %s' % (self.first_name or '', self.last_name or '') | ||||||
|  |         return full_name.strip() | ||||||
|  |  | ||||||
|  |     def is_anonymous(self): | ||||||
|  |         return False | ||||||
|  |  | ||||||
|  |     def is_authenticated(self): | ||||||
|  |         return True | ||||||
|  |  | ||||||
|  |     def set_password(self, raw_password): | ||||||
|  |         """Sets the user's password - always use this rather than directly | ||||||
|  |         assigning to :attr:`~mongoengine.django.auth.User.password` as the | ||||||
|  |         password is hashed before storage. | ||||||
|  |         """ | ||||||
|  |         from random import random | ||||||
|  |         algo = 'sha1' | ||||||
|  |         salt = get_hexdigest(algo, str(random()), str(random()))[:5] | ||||||
|  |         hash = get_hexdigest(algo, salt, raw_password) | ||||||
|  |         self.password = '%s$%s$%s' % (algo, salt, hash) | ||||||
|  |         self.save() | ||||||
|  |         return self | ||||||
|  |  | ||||||
|  |     def check_password(self, raw_password): | ||||||
|  |         """Checks the user's password against a provided password - always use | ||||||
|  |         this rather than directly comparing to | ||||||
|  |         :attr:`~mongoengine.django.auth.User.password` as the password is | ||||||
|  |         hashed before storage. | ||||||
|  |         """ | ||||||
|  |         algo, salt, hash = self.password.split('$') | ||||||
|  |         return hash == get_hexdigest(algo, salt, raw_password) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def create_user(cls, username, password, email=None): | ||||||
|  |         """Create (and save) a new user with the given username, password and | ||||||
|  |         email address. | ||||||
|  |         """ | ||||||
|  |         now = datetime.datetime.now() | ||||||
|  |  | ||||||
|  |         # Normalize the address by lowercasing the domain part of the email | ||||||
|  |         # address. | ||||||
|  |         if email is not None: | ||||||
|  |             try: | ||||||
|  |                 email_name, domain_part = email.strip().split('@', 1) | ||||||
|  |             except ValueError: | ||||||
|  |                 pass | ||||||
|  |             else: | ||||||
|  |                 email = '@'.join([email_name, domain_part.lower()]) | ||||||
|  |  | ||||||
|  |         user = cls(username=username, email=email, date_joined=now) | ||||||
|  |         user.set_password(password) | ||||||
|  |         user.save() | ||||||
|  |         return user | ||||||
|  |  | ||||||
|  |     def get_and_delete_messages(self): | ||||||
|  |         return [] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MongoEngineBackend(object): | ||||||
|  |     """Authenticate using MongoEngine and mongoengine.django.auth.User. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     supports_object_permissions = False | ||||||
|  |     supports_anonymous_user = False | ||||||
|  |     supports_inactive_user = False | ||||||
|  |  | ||||||
|  |     def authenticate(self, username=None, password=None): | ||||||
|  |         user = User.objects(username=username).first() | ||||||
|  |         if user: | ||||||
|  |             if password and user.check_password(password): | ||||||
|  |                 return user | ||||||
|  |         return None | ||||||
|  |  | ||||||
|  |     def get_user(self, user_id): | ||||||
|  |         return User.objects.with_id(user_id) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def get_user(userid): | ||||||
|  |     """Returns a User object from an id (User.id). Django's equivalent takes | ||||||
|  |     request, but taking an id instead leaves it up to the developer to store | ||||||
|  |     the id in any way they want (session, signed cookie, etc.) | ||||||
|  |     """ | ||||||
|  |     if not userid: | ||||||
|  |         return AnonymousUser() | ||||||
|  |     return MongoEngineBackend().get_user(userid) or AnonymousUser() | ||||||
							
								
								
									
										69
									
								
								mongoengine/django/sessions.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								mongoengine/django/sessions.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | |||||||
|  | from django.contrib.sessions.backends.base import SessionBase, CreateError | ||||||
|  | from django.core.exceptions import SuspiciousOperation | ||||||
|  | from django.utils.encoding import force_unicode | ||||||
|  |  | ||||||
|  | from mongoengine.document import Document | ||||||
|  | from mongoengine import fields | ||||||
|  | from mongoengine.queryset import OperationError | ||||||
|  | from mongoengine.connection import DEFAULT_CONNECTION_NAME | ||||||
|  | from django.conf import settings | ||||||
|  | from datetime import datetime | ||||||
|  |  | ||||||
|  | MONGOENGINE_SESSION_DB_ALIAS = getattr( | ||||||
|  |     settings, 'MONGOENGINE_SESSION_DB_ALIAS', | ||||||
|  |     DEFAULT_CONNECTION_NAME) | ||||||
|  |  | ||||||
|  | class MongoSession(Document): | ||||||
|  |     session_key = fields.StringField(primary_key=True, max_length=40) | ||||||
|  |     session_data = fields.StringField() | ||||||
|  |     expire_date = fields.DateTimeField() | ||||||
|  |      | ||||||
|  |     meta = {'collection': 'django_session', | ||||||
|  |             'db_alias': MONGOENGINE_SESSION_DB_ALIAS, | ||||||
|  |             'allow_inheritance': False} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class SessionStore(SessionBase): | ||||||
|  |     """A MongoEngine-based session store for Django. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def load(self): | ||||||
|  |         try: | ||||||
|  |             s = MongoSession.objects(session_key=self.session_key, | ||||||
|  |                                      expire_date__gt=datetime.now())[0] | ||||||
|  |             return self.decode(force_unicode(s.session_data)) | ||||||
|  |         except (IndexError, SuspiciousOperation): | ||||||
|  |             self.create() | ||||||
|  |             return {} | ||||||
|  |  | ||||||
|  |     def exists(self, session_key): | ||||||
|  |         return bool(MongoSession.objects(session_key=session_key).first()) | ||||||
|  |  | ||||||
|  |     def create(self): | ||||||
|  |         while True: | ||||||
|  |             self.session_key = self._get_new_session_key() | ||||||
|  |             try: | ||||||
|  |                 self.save(must_create=True) | ||||||
|  |             except CreateError: | ||||||
|  |                 continue | ||||||
|  |             self.modified = True | ||||||
|  |             self._session_cache = {} | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |     def save(self, must_create=False): | ||||||
|  |         s = MongoSession(session_key=self.session_key) | ||||||
|  |         s.session_data = self.encode(self._get_session(no_load=must_create)) | ||||||
|  |         s.expire_date = self.get_expiry_date() | ||||||
|  |         try: | ||||||
|  |             s.save(force_insert=must_create, safe=True) | ||||||
|  |         except OperationError: | ||||||
|  |             if must_create: | ||||||
|  |                 raise CreateError | ||||||
|  |             raise | ||||||
|  |  | ||||||
|  |     def delete(self, session_key=None): | ||||||
|  |         if session_key is None: | ||||||
|  |             if self.session_key is None: | ||||||
|  |                 return | ||||||
|  |             session_key = self.session_key | ||||||
|  |         MongoSession.objects(session_key=session_key).delete() | ||||||
							
								
								
									
										46
									
								
								mongoengine/django/shortcuts.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								mongoengine/django/shortcuts.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | from django.http import Http404 | ||||||
|  | from mongoengine.queryset import QuerySet | ||||||
|  | from mongoengine.base import BaseDocument | ||||||
|  | from mongoengine.base import ValidationError | ||||||
|  |  | ||||||
|  | def _get_queryset(cls): | ||||||
|  |     """Inspired by django.shortcuts.*""" | ||||||
|  |     if isinstance(cls, QuerySet): | ||||||
|  |         return cls | ||||||
|  |     else: | ||||||
|  |         return cls.objects | ||||||
|  |  | ||||||
|  | def get_document_or_404(cls, *args, **kwargs): | ||||||
|  |     """ | ||||||
|  |     Uses get() to return an document, or raises a Http404 exception if the document | ||||||
|  |     does not exist. | ||||||
|  |  | ||||||
|  |     cls may be a Document or QuerySet object. All other passed | ||||||
|  |     arguments and keyword arguments are used in the get() query. | ||||||
|  |  | ||||||
|  |     Note: Like with get(), an MultipleObjectsReturned will be raised if more than one | ||||||
|  |     object is found. | ||||||
|  |  | ||||||
|  |     Inspired by django.shortcuts.* | ||||||
|  |     """ | ||||||
|  |     queryset = _get_queryset(cls) | ||||||
|  |     try: | ||||||
|  |         return queryset.get(*args, **kwargs) | ||||||
|  |     except (queryset._document.DoesNotExist, ValidationError): | ||||||
|  |         raise Http404('No %s matches the given query.' % queryset._document._class_name) | ||||||
|  |  | ||||||
|  | def get_list_or_404(cls, *args, **kwargs): | ||||||
|  |     """ | ||||||
|  |     Uses filter() to return a list of documents, or raise a Http404 exception if | ||||||
|  |     the list is empty. | ||||||
|  |  | ||||||
|  |     cls may be a Document or QuerySet object. All other passed | ||||||
|  |     arguments and keyword arguments are used in the filter() query. | ||||||
|  |  | ||||||
|  |     Inspired by django.shortcuts.* | ||||||
|  |     """ | ||||||
|  |     queryset = _get_queryset(cls) | ||||||
|  |     obj_list = list(queryset.filter(*args, **kwargs)) | ||||||
|  |     if not obj_list: | ||||||
|  |         raise Http404('No %s matches the given query.' % queryset._document._class_name) | ||||||
|  |     return obj_list | ||||||
							
								
								
									
										112
									
								
								mongoengine/django/storage.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								mongoengine/django/storage.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,112 @@ | |||||||
|  | import os | ||||||
|  | import itertools | ||||||
|  | import urlparse | ||||||
|  |  | ||||||
|  | from mongoengine import * | ||||||
|  | from django.conf import settings | ||||||
|  | from django.core.files.storage import Storage | ||||||
|  | from django.core.exceptions import ImproperlyConfigured | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FileDocument(Document): | ||||||
|  |     """A document used to store a single file in GridFS. | ||||||
|  |     """ | ||||||
|  |     file = FileField() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class GridFSStorage(Storage): | ||||||
|  |     """A custom storage backend to store files in GridFS | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self, base_url=None): | ||||||
|  |  | ||||||
|  |         if base_url is None: | ||||||
|  |             base_url = settings.MEDIA_URL | ||||||
|  |         self.base_url = base_url | ||||||
|  |         self.document = FileDocument | ||||||
|  |         self.field = 'file' | ||||||
|  |  | ||||||
|  |     def delete(self, name): | ||||||
|  |         """Deletes the specified file from the storage system. | ||||||
|  |         """ | ||||||
|  |         if self.exists(name): | ||||||
|  |             doc = self.document.objects.first() | ||||||
|  |             field = getattr(doc, self.field) | ||||||
|  |             self._get_doc_with_name(name).delete()  # Delete the FileField | ||||||
|  |             field.delete()                          # Delete the FileDocument | ||||||
|  |  | ||||||
|  |     def exists(self, name): | ||||||
|  |         """Returns True if a file referened by the given name already exists in the | ||||||
|  |         storage system, or False if the name is available for a new file. | ||||||
|  |         """ | ||||||
|  |         doc = self._get_doc_with_name(name) | ||||||
|  |         if doc: | ||||||
|  |             field = getattr(doc, self.field) | ||||||
|  |             return bool(field.name) | ||||||
|  |         else: | ||||||
|  |             return False | ||||||
|  |  | ||||||
|  |     def listdir(self, path=None): | ||||||
|  |         """Lists the contents of the specified path, returning a 2-tuple of lists; | ||||||
|  |         the first item being directories, the second item being files. | ||||||
|  |         """ | ||||||
|  |         def name(doc): | ||||||
|  |             return getattr(doc, self.field).name | ||||||
|  |         docs = self.document.objects | ||||||
|  |         return [], [name(d) for d in docs if name(d)] | ||||||
|  |  | ||||||
|  |     def size(self, name): | ||||||
|  |         """Returns the total size, in bytes, of the file specified by name. | ||||||
|  |         """ | ||||||
|  |         doc = self._get_doc_with_name(name) | ||||||
|  |         if doc: | ||||||
|  |             return getattr(doc, self.field).length | ||||||
|  |         else: | ||||||
|  |             raise ValueError("No such file or directory: '%s'" % name) | ||||||
|  |  | ||||||
|  |     def url(self, name): | ||||||
|  |         """Returns an absolute URL where the file's contents can be accessed | ||||||
|  |         directly by a web browser. | ||||||
|  |         """ | ||||||
|  |         if self.base_url is None: | ||||||
|  |             raise ValueError("This file is not accessible via a URL.") | ||||||
|  |         return urlparse.urljoin(self.base_url, name).replace('\\', '/') | ||||||
|  |  | ||||||
|  |     def _get_doc_with_name(self, name): | ||||||
|  |         """Find the documents in the store with the given name | ||||||
|  |         """ | ||||||
|  |         docs = self.document.objects | ||||||
|  |         doc = [d for d in docs if getattr(d, self.field).name == name] | ||||||
|  |         if doc: | ||||||
|  |             return doc[0] | ||||||
|  |         else: | ||||||
|  |             return None | ||||||
|  |  | ||||||
|  |     def _open(self, name, mode='rb'): | ||||||
|  |         doc = self._get_doc_with_name(name) | ||||||
|  |         if doc: | ||||||
|  |             return getattr(doc, self.field) | ||||||
|  |         else: | ||||||
|  |             raise ValueError("No file found with the name '%s'." % name) | ||||||
|  |  | ||||||
|  |     def get_available_name(self, name): | ||||||
|  |         """Returns a filename that's free on the target storage system, and | ||||||
|  |         available for new content to be written to. | ||||||
|  |         """ | ||||||
|  |         file_root, file_ext = os.path.splitext(name) | ||||||
|  |         # If the filename already exists, add an underscore and a number (before | ||||||
|  |         # the file extension, if one exists) to the filename until the generated | ||||||
|  |         # filename doesn't exist. | ||||||
|  |         count = itertools.count(1) | ||||||
|  |         while self.exists(name): | ||||||
|  |             # file_ext includes the dot. | ||||||
|  |             name = os.path.join("%s_%s%s" % (file_root, count.next(), file_ext)) | ||||||
|  |  | ||||||
|  |         return name | ||||||
|  |  | ||||||
|  |     def _save(self, name, content): | ||||||
|  |         doc = self.document() | ||||||
|  |         getattr(doc, self.field).put(content, filename=name) | ||||||
|  |         doc.save() | ||||||
|  |  | ||||||
|  |         return name | ||||||
							
								
								
									
										21
									
								
								mongoengine/django/tests.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								mongoengine/django/tests.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | #coding: utf-8 | ||||||
|  | from django.test import TestCase | ||||||
|  | from django.conf import settings | ||||||
|  |  | ||||||
|  | from mongoengine import connect | ||||||
|  |  | ||||||
|  | class MongoTestCase(TestCase): | ||||||
|  |     """ | ||||||
|  |     TestCase class that clear the collection between the tests | ||||||
|  |     """ | ||||||
|  |     db_name = 'test_%s' % settings.MONGO_DATABASE_NAME | ||||||
|  |     def __init__(self, methodName='runtest'): | ||||||
|  |         self.db = connect(self.db_name) | ||||||
|  |         super(MongoTestCase, self).__init__(methodName) | ||||||
|  |  | ||||||
|  |     def _post_teardown(self): | ||||||
|  |         super(MongoTestCase, self)._post_teardown() | ||||||
|  |         for collection in self.db.collection_names(): | ||||||
|  |             if collection == 'system.indexes': | ||||||
|  |                 continue | ||||||
|  |             self.db.drop_collection(collection) | ||||||
| @@ -1,9 +1,18 @@ | |||||||
|  | import pymongo | ||||||
|  | from bson.dbref import DBRef | ||||||
|  |  | ||||||
|  | from mongoengine import signals | ||||||
| from base import (DocumentMetaclass, TopLevelDocumentMetaclass, BaseDocument, | from base import (DocumentMetaclass, TopLevelDocumentMetaclass, BaseDocument, | ||||||
|                   ValidationError) |                   BaseDict, BaseList) | ||||||
| from connection import _get_db | from queryset import OperationError | ||||||
|  | from connection import get_db, DEFAULT_CONNECTION_NAME | ||||||
|  |  | ||||||
|  | __all__ = ['Document', 'EmbeddedDocument', 'DynamicDocument', | ||||||
|  |            'DynamicEmbeddedDocument', 'OperationError', 'InvalidCollectionError'] | ||||||
|  |  | ||||||
|  |  | ||||||
| __all__ = ['Document', 'EmbeddedDocument'] | class InvalidCollectionError(Exception): | ||||||
|  |     pass | ||||||
|  |  | ||||||
|  |  | ||||||
| class EmbeddedDocument(BaseDocument): | class EmbeddedDocument(BaseDocument): | ||||||
| @@ -15,6 +24,21 @@ class EmbeddedDocument(BaseDocument): | |||||||
|  |  | ||||||
|     __metaclass__ = DocumentMetaclass |     __metaclass__ = DocumentMetaclass | ||||||
|  |  | ||||||
|  |     def __init__(self, *args, **kwargs): | ||||||
|  |         super(EmbeddedDocument, self).__init__(*args, **kwargs) | ||||||
|  |         self._changed_fields = [] | ||||||
|  |  | ||||||
|  |     def __delattr__(self, *args, **kwargs): | ||||||
|  |         """Handle deletions of fields""" | ||||||
|  |         field_name = args[0] | ||||||
|  |         if field_name in self._fields: | ||||||
|  |             default = self._fields[field_name].default | ||||||
|  |             if callable(default): | ||||||
|  |                 default = default() | ||||||
|  |             setattr(self, field_name, default) | ||||||
|  |         else: | ||||||
|  |             super(EmbeddedDocument, self).__delattr__(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
| class Document(BaseDocument): | class Document(BaseDocument): | ||||||
|     """The base class used for defining the structure and properties of |     """The base class used for defining the structure and properties of | ||||||
| @@ -44,49 +68,360 @@ class Document(BaseDocument): | |||||||
|     maximum size of the collection in bytes. If :attr:`max_size` is not |     maximum size of the collection in bytes. If :attr:`max_size` is not | ||||||
|     specified and :attr:`max_documents` is, :attr:`max_size` defaults to |     specified and :attr:`max_documents` is, :attr:`max_size` defaults to | ||||||
|     10000000 bytes (10MB). |     10000000 bytes (10MB). | ||||||
|     """ |  | ||||||
|  |  | ||||||
|  |     Indexes may be created by specifying :attr:`indexes` in the :attr:`meta` | ||||||
|  |     dictionary. The value should be a list of field names or tuples of field | ||||||
|  |     names. Index direction may be specified by prefixing the field names with | ||||||
|  |     a **+** or **-** sign. | ||||||
|  |  | ||||||
|  |     By default, _types will be added to the start of every index (that | ||||||
|  |     doesn't contain a list) if allow_inheritence is True. This can be | ||||||
|  |     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. | ||||||
|  |     """ | ||||||
|     __metaclass__ = TopLevelDocumentMetaclass |     __metaclass__ = TopLevelDocumentMetaclass | ||||||
|  |  | ||||||
|     def save(self): |     @apply | ||||||
|  |     def pk(): | ||||||
|  |         """Primary key alias | ||||||
|  |         """ | ||||||
|  |         def fget(self): | ||||||
|  |             return getattr(self, self._meta['id_field']) | ||||||
|  |         def fset(self, value): | ||||||
|  |             return setattr(self, self._meta['id_field'], value) | ||||||
|  |         return property(fget, fset) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _get_db(cls): | ||||||
|  |         """Some Model using other db_alias""" | ||||||
|  |         return get_db(cls._meta.get("db_alias", DEFAULT_CONNECTION_NAME )) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def _get_collection(cls): | ||||||
|  |         """Returns the collection for the document.""" | ||||||
|  |         if not hasattr(cls, '_collection') or cls._collection is None: | ||||||
|  |             db = cls._get_db() | ||||||
|  |             collection_name = cls._get_collection_name() | ||||||
|  |             # Create collection as a capped collection if specified | ||||||
|  |             if cls._meta['max_size'] or cls._meta['max_documents']: | ||||||
|  |                 # Get max document limit and max byte size from meta | ||||||
|  |                 max_size = cls._meta['max_size'] or 10000000  # 10MB default | ||||||
|  |                 max_documents = cls._meta['max_documents'] | ||||||
|  |  | ||||||
|  |                 if collection_name in db.collection_names(): | ||||||
|  |                     cls._collection = db[collection_name] | ||||||
|  |                     # The collection already exists, check if its capped | ||||||
|  |                     # options match the specified capped options | ||||||
|  |                     options = cls._collection.options() | ||||||
|  |                     if options.get('max') != max_documents or \ | ||||||
|  |                        options.get('size') != max_size: | ||||||
|  |                         msg = ('Cannot create collection "%s" as a capped ' | ||||||
|  |                                'collection as it already exists') % cls._collection | ||||||
|  |                         raise InvalidCollectionError(msg) | ||||||
|  |                 else: | ||||||
|  |                     # Create the collection as a capped collection | ||||||
|  |                     opts = {'capped': True, 'size': max_size} | ||||||
|  |                     if max_documents: | ||||||
|  |                         opts['max'] = max_documents | ||||||
|  |                     cls._collection = db.create_collection( | ||||||
|  |                         collection_name, **opts | ||||||
|  |                     ) | ||||||
|  |             else: | ||||||
|  |                 cls._collection = db[collection_name] | ||||||
|  |         return cls._collection | ||||||
|  |  | ||||||
|  |     def save(self, safe=True, force_insert=False, validate=True, write_options=None, | ||||||
|  |             cascade=None, cascade_kwargs=None, _refs=None): | ||||||
|         """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. | ||||||
|         """ |  | ||||||
|         self.validate() |  | ||||||
|         object_id = self.__class__.objects._collection.save(self.to_mongo()) |  | ||||||
|         self.id = self._fields['id'].to_python(object_id) |  | ||||||
|  |  | ||||||
|     def delete(self): |         If ``safe=True`` and the operation is unsuccessful, an | ||||||
|  |         :class:`~mongoengine.OperationError` will be raised. | ||||||
|  |  | ||||||
|  |         :param safe: check if the operation succeeded before returning | ||||||
|  |         :param force_insert: only try to create a new document, don't allow | ||||||
|  |             updates of existing documents | ||||||
|  |         :param validate: validates the document; set to ``False`` to skip. | ||||||
|  |         :param write_options: Extra keyword arguments are passed down to | ||||||
|  |                 :meth:`~pymongo.collection.Collection.save` OR | ||||||
|  |                 :meth:`~pymongo.collection.Collection.insert` | ||||||
|  |                 which will be used as options for the resultant ``getLastError`` command. | ||||||
|  |                 For example, ``save(..., w=2, fsync=True)`` will wait until at least two servers | ||||||
|  |                 have recorded the write and will force an fsync on each server being written to. | ||||||
|  |         :param cascade: Sets the flag for cascading saves.  You can set a 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 | ||||||
|  |  | ||||||
|  |         .. versionchanged:: 0.5 | ||||||
|  |             In existing documents it only saves changed fields using set / unset | ||||||
|  |             Saves are cascaded and any :class:`~bson.dbref.DBRef` objects | ||||||
|  |             that have changes are saved as well. | ||||||
|  |         .. versionchanged:: 0.6 | ||||||
|  |             Cascade saves are optional = defaults to True, if you want fine grain | ||||||
|  |             control then you can turn off using document meta['cascade'] = False | ||||||
|  |             Also you can pass different kwargs to the cascade save using cascade_kwargs | ||||||
|  |             which overwrites the existing kwargs with custom values | ||||||
|  |  | ||||||
|  |         """ | ||||||
|  |         signals.pre_save.send(self.__class__, document=self) | ||||||
|  |  | ||||||
|  |         if validate: | ||||||
|  |             self.validate() | ||||||
|  |  | ||||||
|  |         if not write_options: | ||||||
|  |             write_options = {} | ||||||
|  |  | ||||||
|  |         doc = self.to_mongo() | ||||||
|  |  | ||||||
|  |         created = force_insert or '_id' not in doc | ||||||
|  |  | ||||||
|  |         try: | ||||||
|  |             collection = self.__class__.objects._collection | ||||||
|  |             if created: | ||||||
|  |                 if force_insert: | ||||||
|  |                     object_id = collection.insert(doc, safe=safe, **write_options) | ||||||
|  |                 else: | ||||||
|  |                     object_id = collection.save(doc, safe=safe, **write_options) | ||||||
|  |             else: | ||||||
|  |                 object_id = doc['_id'] | ||||||
|  |                 updates, removals = self._delta() | ||||||
|  |  | ||||||
|  |                 # Need to add shard key to query, or you get an error | ||||||
|  |                 select_dict = {'_id': object_id} | ||||||
|  |                 shard_key = self.__class__._meta.get('shard_key', tuple()) | ||||||
|  |                 for k in shard_key: | ||||||
|  |                     actual_key = self._db_field_map.get(k, k) | ||||||
|  |                     select_dict[actual_key] = doc[actual_key] | ||||||
|  |  | ||||||
|  |                 upsert = self._created | ||||||
|  |                 if updates: | ||||||
|  |                     collection.update(select_dict, {"$set": updates}, upsert=upsert, safe=safe, **write_options) | ||||||
|  |                 if removals: | ||||||
|  |                     collection.update(select_dict, {"$unset": removals}, upsert=upsert, safe=safe, **write_options) | ||||||
|  |  | ||||||
|  |             cascade = self._meta.get('cascade', True) if cascade is None else cascade | ||||||
|  |             if cascade: | ||||||
|  |                 kwargs = { | ||||||
|  |                     "safe": safe, | ||||||
|  |                     "force_insert": force_insert, | ||||||
|  |                     "validate": validate, | ||||||
|  |                     "write_options": write_options, | ||||||
|  |                     "cascade": cascade | ||||||
|  |                 } | ||||||
|  |                 if cascade_kwargs:  # Allow granular control over cascades | ||||||
|  |                     kwargs.update(cascade_kwargs) | ||||||
|  |                 kwargs['_refs'] = _refs | ||||||
|  |                 self.cascade_save(**kwargs) | ||||||
|  |  | ||||||
|  |         except pymongo.errors.OperationFailure, err: | ||||||
|  |             message = 'Could not save document (%s)' | ||||||
|  |             if u'duplicate key' in unicode(err): | ||||||
|  |                 message = u'Tried to save duplicate unique keys (%s)' | ||||||
|  |             raise OperationError(message % unicode(err)) | ||||||
|  |         id_field = self._meta['id_field'] | ||||||
|  |         self[id_field] = self._fields[id_field].to_python(object_id) | ||||||
|  |  | ||||||
|  |         self._changed_fields = [] | ||||||
|  |         self._created = False | ||||||
|  |         signals.post_save.send(self.__class__, document=self, created=created) | ||||||
|  |  | ||||||
|  |     def cascade_save(self, *args, **kwargs): | ||||||
|  |         """Recursively saves any references / generic references on an object""" | ||||||
|  |         from fields import ReferenceField, GenericReferenceField | ||||||
|  |         _refs = kwargs.get('_refs', []) or [] | ||||||
|  |         for name, cls in self._fields.items(): | ||||||
|  |             if not isinstance(cls, (ReferenceField, GenericReferenceField)): | ||||||
|  |                 continue | ||||||
|  |             ref = getattr(self, name) | ||||||
|  |             if not ref: | ||||||
|  |                 continue | ||||||
|  |             ref_id = "%s,%s" % (ref.__class__.__name__, str(ref._data)) | ||||||
|  |             if ref and ref_id not in _refs: | ||||||
|  |                 _refs.append(ref_id) | ||||||
|  |                 kwargs["_refs"] = _refs | ||||||
|  |                 ref.save(**kwargs) | ||||||
|  |                 ref._changed_fields = [] | ||||||
|  |  | ||||||
|  |     def update(self, **kwargs): | ||||||
|  |         """Performs an update on the :class:`~mongoengine.Document` | ||||||
|  |         A convenience wrapper to :meth:`~mongoengine.QuerySet.update`. | ||||||
|  |  | ||||||
|  |         Raises :class:`OperationError` if called on an object that has not yet | ||||||
|  |         been saved. | ||||||
|  |         """ | ||||||
|  |         if not self.pk: | ||||||
|  |             raise OperationError('attempt to update a document not yet saved') | ||||||
|  |  | ||||||
|  |         # Need to add shard key to query, or you get an error | ||||||
|  |         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 self.__class__.objects(**select_dict).update_one(**kwargs) | ||||||
|  |  | ||||||
|  |     def delete(self, safe=False): | ||||||
|         """Delete the :class:`~mongoengine.Document` from the database. This |         """Delete the :class:`~mongoengine.Document` from the database. This | ||||||
|         will only take effect if the document has been previously saved. |         will only take effect if the document has been previously saved. | ||||||
|         """ |  | ||||||
|         object_id = self._fields['id'].to_mongo(self.id) |  | ||||||
|         self.__class__.objects(id=object_id).delete() |  | ||||||
|  |  | ||||||
|     def validate(self): |         :param safe: check if the operation succeeded before returning | ||||||
|         """Ensure that all fields' values are valid and that required fields |  | ||||||
|         are present. |  | ||||||
|         """ |         """ | ||||||
|         # Get a list of tuples of field names and their current values |         signals.pre_delete.send(self.__class__, document=self) | ||||||
|         fields = [(field, getattr(self, name))  |  | ||||||
|                   for name, field in self._fields.items()] |  | ||||||
|  |  | ||||||
|         # Ensure that each field is matched to a valid value |         try: | ||||||
|         for field, value in fields: |             self.__class__.objects(pk=self.pk).delete(safe=safe) | ||||||
|             if value is not None: |         except pymongo.errors.OperationFailure, err: | ||||||
|                 try: |             message = u'Could not delete document (%s)' % err.message | ||||||
|                     field.validate(value) |             raise OperationError(message) | ||||||
|                 except (ValueError, AttributeError, AssertionError), e: |  | ||||||
|                     raise ValidationError('Invalid value for field of type "' + |         signals.post_delete.send(self.__class__, document=self) | ||||||
|                                           field.__class__.__name__ + '"') |  | ||||||
|             elif field.required: |     def select_related(self, max_depth=1): | ||||||
|                 raise ValidationError('Field "%s" is required' % field.name) |         """Handles dereferencing of :class:`~bson.dbref.DBRef` objects to | ||||||
|  |         a maximum depth in order to cut down the number queries to mongodb. | ||||||
|  |  | ||||||
|  |         .. versionadded:: 0.5 | ||||||
|  |         """ | ||||||
|  |         from dereference import DeReference | ||||||
|  |         self._data = DeReference()(self._data, max_depth) | ||||||
|  |         return self | ||||||
|  |  | ||||||
|  |     def reload(self, max_depth=1): | ||||||
|  |         """Reloads all attributes from the database. | ||||||
|  |  | ||||||
|  |         .. versionadded:: 0.1.2 | ||||||
|  |         .. versionchanged:: 0.6  Now chainable | ||||||
|  |         """ | ||||||
|  |         id_field = self._meta['id_field'] | ||||||
|  |         obj = self.__class__.objects( | ||||||
|  |                 **{id_field: self[id_field]} | ||||||
|  |               ).first().select_related(max_depth=max_depth) | ||||||
|  |         for field in self._fields: | ||||||
|  |             setattr(self, field, self._reload(field, obj[field])) | ||||||
|  |         if self._dynamic: | ||||||
|  |             for name in self._dynamic_fields.keys(): | ||||||
|  |                 setattr(self, name, self._reload(name, obj._data[name])) | ||||||
|  |         self._changed_fields = obj._changed_fields | ||||||
|  |         return obj | ||||||
|  |  | ||||||
|  |     def _reload(self, key, value): | ||||||
|  |         """Used by :meth:`~mongoengine.Document.reload` to ensure the | ||||||
|  |         correct instance is linked to self. | ||||||
|  |         """ | ||||||
|  |         if isinstance(value, BaseDict): | ||||||
|  |             value = [(k, self._reload(k, v)) for k, v in value.items()] | ||||||
|  |             value = BaseDict(value, self, key) | ||||||
|  |         elif isinstance(value, BaseList): | ||||||
|  |             value = [self._reload(key, v) for v in value] | ||||||
|  |             value = BaseList(value, self, key) | ||||||
|  |         elif isinstance(value, (EmbeddedDocument, DynamicEmbeddedDocument)): | ||||||
|  |             value._changed_fields = [] | ||||||
|  |         return value | ||||||
|  |  | ||||||
|  |     def to_dbref(self): | ||||||
|  |         """Returns an instance of :class:`~bson.dbref.DBRef` useful in | ||||||
|  |         `__raw__` queries.""" | ||||||
|  |         if not self.pk: | ||||||
|  |             msg = "Only saved documents can have a valid dbref" | ||||||
|  |             raise OperationError(msg) | ||||||
|  |         return DBRef(self.__class__._get_collection_name(), self.pk) | ||||||
|  |  | ||||||
|  |     @classmethod | ||||||
|  |     def register_delete_rule(cls, document_cls, field_name, rule): | ||||||
|  |         """This method registers the delete rules to apply when removing this | ||||||
|  |         object. | ||||||
|  |         """ | ||||||
|  |         cls._meta['delete_rules'][(document_cls, field_name)] = rule | ||||||
|  |  | ||||||
|     @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. | ||||||
|         """ |         """ | ||||||
|         db = _get_db() |         from mongoengine.queryset import QuerySet | ||||||
|         db.drop_collection(cls._meta['collection']) |         db = cls._get_db() | ||||||
|  |         db.drop_collection(cls._get_collection_name()) | ||||||
|  |         QuerySet._reset_already_indexed(cls) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DynamicDocument(Document): | ||||||
|  |     """A Dynamic Document class allowing flexible, expandable and uncontrolled | ||||||
|  |     schemas.  As a :class:`~mongoengine.Document` subclass, acts in the same | ||||||
|  |     way as an ordinary document but has expando style properties.  Any data | ||||||
|  |     passed or set against the :class:`~mongoengine.DynamicDocument` that is | ||||||
|  |     not a field is automatically converted into a | ||||||
|  |     :class:`~mongoengine.BaseDynamicField` and data can be attributed to that | ||||||
|  |     field. | ||||||
|  |  | ||||||
|  |     ..note:: | ||||||
|  |  | ||||||
|  |         There is one caveat on Dynamic Documents: fields cannot start with `_` | ||||||
|  |     """ | ||||||
|  |     __metaclass__ = TopLevelDocumentMetaclass | ||||||
|  |     _dynamic = True | ||||||
|  |  | ||||||
|  |     def __delattr__(self, *args, **kwargs): | ||||||
|  |         """Deletes the attribute by setting to None and allowing _delta to unset | ||||||
|  |         it""" | ||||||
|  |         field_name = args[0] | ||||||
|  |         if field_name in self._dynamic_fields: | ||||||
|  |             setattr(self, field_name, None) | ||||||
|  |         else: | ||||||
|  |             super(DynamicDocument, self).__delattr__(*args, **kwargs) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DynamicEmbeddedDocument(EmbeddedDocument): | ||||||
|  |     """A Dynamic Embedded Document class allowing flexible, expandable and | ||||||
|  |     uncontrolled schemas. See :class:`~mongoengine.DynamicDocument` for more | ||||||
|  |     information about dynamic documents. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     __metaclass__ = DocumentMetaclass | ||||||
|  |     _dynamic = True | ||||||
|  |  | ||||||
|  |     def __delattr__(self, *args, **kwargs): | ||||||
|  |         """Deletes the attribute by setting to None and allowing _delta to unset | ||||||
|  |         it""" | ||||||
|  |         field_name = args[0] | ||||||
|  |         setattr(self, field_name, None) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class MapReduceDocument(object): | ||||||
|  |     """A document returned from a map/reduce query. | ||||||
|  |  | ||||||
|  |     :param collection: An instance of :class:`~pymongo.Collection` | ||||||
|  |     :param key: Document/result key, often an instance of | ||||||
|  |                 :class:`~bson.objectid.ObjectId`. If supplied as | ||||||
|  |                 an ``ObjectId`` found in the given ``collection``, | ||||||
|  |                 the object can be accessed via the ``object`` property. | ||||||
|  |     :param value: The result(s) for this key. | ||||||
|  |  | ||||||
|  |     .. versionadded:: 0.3 | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def __init__(self, document, collection, key, value): | ||||||
|  |         self._document = document | ||||||
|  |         self._collection = collection | ||||||
|  |         self.key = key | ||||||
|  |         self.value = value | ||||||
|  |  | ||||||
|  |     @property | ||||||
|  |     def object(self): | ||||||
|  |         """Lazy-load the object referenced by ``self.key``. ``self.key`` | ||||||
|  |         should be the ``primary_key``. | ||||||
|  |         """ | ||||||
|  |         id_field = self._document()._meta['id_field'] | ||||||
|  |         id_field_type = type(id_field) | ||||||
|  |  | ||||||
|  |         if not isinstance(self.key, id_field_type): | ||||||
|  |             try: | ||||||
|  |                 self.key = id_field_type(self.key) | ||||||
|  |             except: | ||||||
|  |                 raise Exception("Could not cast key as %s" % \ | ||||||
|  |                                 id_field_type.__name__) | ||||||
|  |  | ||||||
|  |         if not hasattr(self, "_key_object"): | ||||||
|  |             self._key_object = self._document.objects.with_id(self.key) | ||||||
|  |             return self._key_object | ||||||
|  |         return self._key_object | ||||||
|   | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										46
									
								
								mongoengine/signals.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										46
									
								
								mongoengine/signals.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,46 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | __all__ = ['pre_init', 'post_init', 'pre_save', 'post_save', | ||||||
|  |            'pre_delete', 'post_delete'] | ||||||
|  |  | ||||||
|  | signals_available = False | ||||||
|  | try: | ||||||
|  |     from blinker import Namespace | ||||||
|  |     signals_available = True | ||||||
|  | except ImportError: | ||||||
|  |     class Namespace(object): | ||||||
|  |         def signal(self, name, doc=None): | ||||||
|  |             return _FakeSignal(name, doc) | ||||||
|  |  | ||||||
|  |     class _FakeSignal(object): | ||||||
|  |         """If blinker is unavailable, create a fake class with the same | ||||||
|  |         interface that allows sending of signals but will fail with an | ||||||
|  |         error on anything else.  Instead of doing anything on send, it | ||||||
|  |         will just ignore the arguments and do nothing instead. | ||||||
|  |         """ | ||||||
|  |  | ||||||
|  |         def __init__(self, name, doc=None): | ||||||
|  |             self.name = name | ||||||
|  |             self.__doc__ = doc | ||||||
|  |  | ||||||
|  |         def _fail(self, *args, **kwargs): | ||||||
|  |             raise RuntimeError('signalling support is unavailable ' | ||||||
|  |                                'because the blinker library is ' | ||||||
|  |                                'not installed.') | ||||||
|  |         send = lambda *a, **kw: None | ||||||
|  |         connect = disconnect = has_receivers_for = receivers_for = \ | ||||||
|  |             temporarily_connected_to = _fail | ||||||
|  |         del _fail | ||||||
|  |  | ||||||
|  | # the namespace for code signals.  If you are not mongoengine code, do | ||||||
|  | # not put signals in here.  Create your own namespace instead. | ||||||
|  | _signals = Namespace() | ||||||
|  |  | ||||||
|  | pre_init = _signals.signal('pre_init') | ||||||
|  | post_init = _signals.signal('post_init') | ||||||
|  | pre_save = _signals.signal('pre_save') | ||||||
|  | post_save = _signals.signal('post_save') | ||||||
|  | pre_delete = _signals.signal('pre_delete') | ||||||
|  | post_delete = _signals.signal('post_delete') | ||||||
|  | pre_bulk_insert = _signals.signal('pre_bulk_insert') | ||||||
|  | post_bulk_insert = _signals.signal('post_bulk_insert') | ||||||
							
								
								
									
										59
									
								
								mongoengine/tests.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										59
									
								
								mongoengine/tests.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,59 @@ | |||||||
|  | from mongoengine.connection import get_db | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class query_counter(object): | ||||||
|  |     """ Query_counter contextmanager to get the number of queries. """ | ||||||
|  |  | ||||||
|  |     def __init__(self): | ||||||
|  |         """ Construct the query_counter. """ | ||||||
|  |         self.counter = 0 | ||||||
|  |         self.db = get_db() | ||||||
|  |  | ||||||
|  |     def __enter__(self): | ||||||
|  |         """ On every with block we need to drop the profile collection. """ | ||||||
|  |         self.db.set_profiling_level(0) | ||||||
|  |         self.db.system.profile.drop() | ||||||
|  |         self.db.set_profiling_level(2) | ||||||
|  |         return self | ||||||
|  |  | ||||||
|  |     def __exit__(self, t, value, traceback): | ||||||
|  |         """ Reset the profiling level. """ | ||||||
|  |         self.db.set_profiling_level(0) | ||||||
|  |  | ||||||
|  |     def __eq__(self, value): | ||||||
|  |         """ == Compare querycounter. """ | ||||||
|  |         return value == self._get_count() | ||||||
|  |  | ||||||
|  |     def __ne__(self, value): | ||||||
|  |         """ != Compare querycounter. """ | ||||||
|  |         return not self.__eq__(value) | ||||||
|  |  | ||||||
|  |     def __lt__(self, value): | ||||||
|  |         """ < Compare querycounter. """ | ||||||
|  |         return self._get_count() < value | ||||||
|  |  | ||||||
|  |     def __le__(self, value): | ||||||
|  |         """ <= Compare querycounter. """ | ||||||
|  |         return self._get_count() <= value | ||||||
|  |  | ||||||
|  |     def __gt__(self, value): | ||||||
|  |         """ > Compare querycounter. """ | ||||||
|  |         return self._get_count() > value | ||||||
|  |  | ||||||
|  |     def __ge__(self, value): | ||||||
|  |         """ >= Compare querycounter. """ | ||||||
|  |         return self._get_count() >= value | ||||||
|  |  | ||||||
|  |     def __int__(self): | ||||||
|  |         """ int representation. """ | ||||||
|  |         return self._get_count() | ||||||
|  |  | ||||||
|  |     def __repr__(self): | ||||||
|  |         """ repr query_counter as the number of queries. """ | ||||||
|  |         return u"%s" % self._get_count() | ||||||
|  |  | ||||||
|  |     def _get_count(self): | ||||||
|  |         """ Get the number of queries. """ | ||||||
|  |         count = self.db.system.profile.find().count() - self.counter | ||||||
|  |         self.counter += 1 | ||||||
|  |         return count | ||||||
							
								
								
									
										62
									
								
								python-mongoengine.spec
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								python-mongoengine.spec
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | |||||||
|  | # sitelib for noarch packages, sitearch for others (remove the unneeded one) | ||||||
|  | %{!?python_sitelib: %global python_sitelib %(%{__python} -c "from distutils.sysconfig import get_python_lib; print(get_python_lib())")} | ||||||
|  | %{!?python_sitearch: %global python_sitearch %(%{__python} -c "from distutils.sysconfig import get_python_lib; print(get_python_lib(1))")} | ||||||
|  |  | ||||||
|  | %define srcname mongoengine | ||||||
|  |  | ||||||
|  | Name:           python-%{srcname} | ||||||
|  | Version:        0.6.1 | ||||||
|  | Release:        1%{?dist} | ||||||
|  | Summary:        A Python Document-Object Mapper for working with MongoDB | ||||||
|  |  | ||||||
|  | Group:          Development/Libraries | ||||||
|  | License:        MIT | ||||||
|  | URL:            https://github.com/MongoEngine/mongoengine | ||||||
|  | Source0:        %{srcname}-%{version}.tar.bz2 | ||||||
|  |  | ||||||
|  | BuildRequires:  python-devel | ||||||
|  | BuildRequires:  python-setuptools | ||||||
|  |  | ||||||
|  | Requires:       mongodb | ||||||
|  | Requires:       pymongo | ||||||
|  | Requires:       python-blinker | ||||||
|  | Requires:       python-imaging | ||||||
|  |  | ||||||
|  |  | ||||||
|  | %description | ||||||
|  | MongoEngine is an ORM-like layer on top of PyMongo. | ||||||
|  |  | ||||||
|  | %prep | ||||||
|  | %setup -q -n %{srcname}-%{version} | ||||||
|  |  | ||||||
|  |  | ||||||
|  | %build | ||||||
|  | # Remove CFLAGS=... for noarch packages (unneeded) | ||||||
|  | CFLAGS="$RPM_OPT_FLAGS" %{__python} setup.py build | ||||||
|  |  | ||||||
|  |  | ||||||
|  | %install | ||||||
|  | rm -rf $RPM_BUILD_ROOT | ||||||
|  | %{__python} setup.py install -O1 --skip-build --root $RPM_BUILD_ROOT | ||||||
|  |  | ||||||
|  | %clean | ||||||
|  | rm -rf $RPM_BUILD_ROOT | ||||||
|  |  | ||||||
|  | %files | ||||||
|  | %defattr(-,root,root,-) | ||||||
|  | %doc docs AUTHORS LICENSE README.rst | ||||||
|  | # For noarch packages: sitelib | ||||||
|  |  %{python_sitelib}/* | ||||||
|  | # For arch-specific packages: sitearch | ||||||
|  | # %{python_sitearch}/* | ||||||
|  |  | ||||||
|  | %changelog | ||||||
|  | * Mon Mar 05 2012 Ross Lawley <ross.lawley@gmail.com> 0.6 | ||||||
|  | - 0.6 released | ||||||
|  | * Thu Oct 27 2011 Pau Aliagas <linuxnow@gmail.com> 0.5.3-1 | ||||||
|  | - Update to latest dev version | ||||||
|  | - Add PIL dependency for ImageField | ||||||
|  | * Wed Oct 12 2011 Pau Aliagas <linuxnow@gmail.com> 0.5.2-1 | ||||||
|  | - Update version | ||||||
|  | * Fri Sep 23 2011 Pau Aliagas <linuxnow@gmail.com> 0.5.0-1 | ||||||
|  | - Initial version | ||||||
							
								
								
									
										27
									
								
								setup.py
									
									
									
									
									
								
							
							
						
						
									
										27
									
								
								setup.py
									
									
									
									
									
								
							| @@ -1,6 +1,5 @@ | |||||||
| from setuptools import setup | from setuptools import setup, find_packages | ||||||
|  | import os | ||||||
| VERSION = '0.1.1' |  | ||||||
|  |  | ||||||
| DESCRIPTION = "A Python Document-Object Mapper for working with MongoDB" | DESCRIPTION = "A Python Document-Object Mapper for working with MongoDB" | ||||||
|  |  | ||||||
| @@ -10,6 +9,20 @@ try: | |||||||
| except: | except: | ||||||
|     pass |     pass | ||||||
|  |  | ||||||
|  | def get_version(version_tuple): | ||||||
|  |     version = '%s.%s' % (version_tuple[0], version_tuple[1]) | ||||||
|  |     if version_tuple[2]: | ||||||
|  |         version = '%s.%s' % (version, version_tuple[2]) | ||||||
|  |     return version | ||||||
|  |  | ||||||
|  | # 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 | ||||||
|  | # file is read | ||||||
|  | init = os.path.join(os.path.dirname(__file__), 'mongoengine', '__init__.py') | ||||||
|  | version_line = filter(lambda l: l.startswith('VERSION'), open(init))[0] | ||||||
|  | VERSION = get_version(eval(version_line.split('=')[-1])) | ||||||
|  | print VERSION | ||||||
|  |  | ||||||
| CLASSIFIERS = [ | CLASSIFIERS = [ | ||||||
|     'Development Status :: 4 - Beta', |     'Development Status :: 4 - Beta', | ||||||
|     'Intended Audience :: Developers', |     'Intended Audience :: Developers', | ||||||
| @@ -22,15 +35,19 @@ CLASSIFIERS = [ | |||||||
|  |  | ||||||
| setup(name='mongoengine', | setup(name='mongoengine', | ||||||
|       version=VERSION, |       version=VERSION, | ||||||
|       packages=['mongoengine'], |       packages=find_packages(), | ||||||
|       author='Harry Marr', |       author='Harry Marr', | ||||||
|       author_email='harry.marr@{nospam}gmail.com', |       author_email='harry.marr@{nospam}gmail.com', | ||||||
|       url='http://hmarr.com/mongoengine/', |       maintainer="Ross Lawley", | ||||||
|  |       maintainer_email="ross.lawley@{nospam}gmail.com", | ||||||
|  |       url='http://mongoengine.org/', | ||||||
|       license='MIT', |       license='MIT', | ||||||
|  |       include_package_data=True, | ||||||
|       description=DESCRIPTION, |       description=DESCRIPTION, | ||||||
|       long_description=LONG_DESCRIPTION, |       long_description=LONG_DESCRIPTION, | ||||||
|       platforms=['any'], |       platforms=['any'], | ||||||
|       classifiers=CLASSIFIERS, |       classifiers=CLASSIFIERS, | ||||||
|       install_requires=['pymongo'], |       install_requires=['pymongo'], | ||||||
|       test_suite='tests', |       test_suite='tests', | ||||||
|  |       tests_require=['blinker', 'django>=1.3', 'PIL'] | ||||||
| ) | ) | ||||||
|   | |||||||
							
								
								
									
										70
									
								
								tests/connection.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										70
									
								
								tests/connection.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,70 @@ | |||||||
|  | import unittest | ||||||
|  | import pymongo | ||||||
|  |  | ||||||
|  | import mongoengine.connection | ||||||
|  |  | ||||||
|  | from mongoengine import * | ||||||
|  | from mongoengine.connection import get_db, get_connection, ConnectionError | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class ConnectionTest(unittest.TestCase): | ||||||
|  |  | ||||||
|  |     def tearDown(self): | ||||||
|  |         mongoengine.connection._connection_settings = {} | ||||||
|  |         mongoengine.connection._connections = {} | ||||||
|  |         mongoengine.connection._dbs = {} | ||||||
|  |  | ||||||
|  |     def test_connect(self): | ||||||
|  |         """Ensure that the connect() method works properly. | ||||||
|  |         """ | ||||||
|  |         connect('mongoenginetest') | ||||||
|  |  | ||||||
|  |         conn = get_connection() | ||||||
|  |         self.assertTrue(isinstance(conn, pymongo.connection.Connection)) | ||||||
|  |  | ||||||
|  |         db = get_db() | ||||||
|  |         self.assertTrue(isinstance(db, pymongo.database.Database)) | ||||||
|  |         self.assertEqual(db.name, 'mongoenginetest') | ||||||
|  |  | ||||||
|  |         connect('mongoenginetest2', alias='testdb') | ||||||
|  |         conn = get_connection('testdb') | ||||||
|  |         self.assertTrue(isinstance(conn, pymongo.connection.Connection)) | ||||||
|  |  | ||||||
|  |     def test_connect_uri(self): | ||||||
|  |         """Ensure that the connect() method works properly with uri's | ||||||
|  |         """ | ||||||
|  |         c = connect(db='mongoenginetest', alias='admin') | ||||||
|  |         c.admin.system.users.remove({}) | ||||||
|  |         c.mongoenginetest.system.users.remove({}) | ||||||
|  |  | ||||||
|  |         c.admin.add_user("admin", "password") | ||||||
|  |         c.admin.authenticate("admin", "password") | ||||||
|  |         c.mongoenginetest.add_user("username", "password") | ||||||
|  |  | ||||||
|  |         self.assertRaises(ConnectionError, connect, "testdb_uri_bad", host='mongodb://test:password@localhost') | ||||||
|  |  | ||||||
|  |         connect("testdb_uri", host='mongodb://username:password@localhost/mongoenginetest') | ||||||
|  |  | ||||||
|  |         conn = get_connection() | ||||||
|  |         self.assertTrue(isinstance(conn, pymongo.connection.Connection)) | ||||||
|  |  | ||||||
|  |         db = get_db() | ||||||
|  |         self.assertTrue(isinstance(db, pymongo.database.Database)) | ||||||
|  |         self.assertEqual(db.name, 'mongoenginetest') | ||||||
|  |  | ||||||
|  |     def test_register_connection(self): | ||||||
|  |         """Ensure that connections with different aliases may be registered. | ||||||
|  |         """ | ||||||
|  |         register_connection('testdb', 'mongoenginetest2') | ||||||
|  |  | ||||||
|  |         self.assertRaises(ConnectionError, get_connection) | ||||||
|  |         conn = get_connection('testdb') | ||||||
|  |         self.assertTrue(isinstance(conn, pymongo.connection.Connection)) | ||||||
|  |  | ||||||
|  |         db = get_db('testdb') | ||||||
|  |         self.assertTrue(isinstance(db, pymongo.database.Database)) | ||||||
|  |         self.assertEqual(db.name, 'mongoenginetest2') | ||||||
|  |  | ||||||
|  |  | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     unittest.main() | ||||||
							
								
								
									
										812
									
								
								tests/dereference.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										812
									
								
								tests/dereference.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,812 @@ | |||||||
|  | import unittest | ||||||
|  |  | ||||||
|  | from mongoengine import * | ||||||
|  | from mongoengine.connection import get_db | ||||||
|  | from mongoengine.tests import query_counter | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class FieldTest(unittest.TestCase): | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         connect(db='mongoenginetest') | ||||||
|  |         self.db = get_db() | ||||||
|  |  | ||||||
|  |     def test_list_item_dereference(self): | ||||||
|  |         """Ensure that DBRef items in ListFields are dereferenced. | ||||||
|  |         """ | ||||||
|  |         class User(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Group(Document): | ||||||
|  |             members = ListField(ReferenceField(User)) | ||||||
|  |  | ||||||
|  |         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 = Group(members=User.objects) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         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_recursive_reference(self): | ||||||
|  |         """Ensure that ReferenceFields can reference their own documents. | ||||||
|  |         """ | ||||||
|  |         class Employee(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             boss = ReferenceField('self') | ||||||
|  |             friends = ListField(ReferenceField('self')) | ||||||
|  |  | ||||||
|  |         Employee.drop_collection() | ||||||
|  |  | ||||||
|  |         bill = Employee(name='Bill Lumbergh') | ||||||
|  |         bill.save() | ||||||
|  |  | ||||||
|  |         michael = Employee(name='Michael Bolton') | ||||||
|  |         michael.save() | ||||||
|  |  | ||||||
|  |         samir = Employee(name='Samir Nagheenanajar') | ||||||
|  |         samir.save() | ||||||
|  |  | ||||||
|  |         friends = [michael, samir] | ||||||
|  |         peter = Employee(name='Peter Gibbons', boss=bill, friends=friends) | ||||||
|  |         peter.save() | ||||||
|  |  | ||||||
|  |         Employee(name='Funky Gibbon', boss=bill, friends=friends).save() | ||||||
|  |         Employee(name='Funky Gibbon', boss=bill, friends=friends).save() | ||||||
|  |         Employee(name='Funky Gibbon', boss=bill, friends=friends).save() | ||||||
|  |  | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             peter = Employee.objects.with_id(peter.id) | ||||||
|  |             self.assertEqual(q, 1) | ||||||
|  |  | ||||||
|  |             peter.boss | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |             peter.friends | ||||||
|  |             self.assertEqual(q, 3) | ||||||
|  |  | ||||||
|  |         # Document select_related | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             peter = Employee.objects.with_id(peter.id).select_related() | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |             self.assertEquals(peter.boss, bill) | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |             self.assertEquals(peter.friends, friends) | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |         # Queryset select_related | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             employees = Employee.objects(boss=bill).select_related() | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |             for employee in employees: | ||||||
|  |                 self.assertEquals(employee.boss, bill) | ||||||
|  |                 self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |                 self.assertEquals(employee.friends, friends) | ||||||
|  |                 self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |     def test_circular_reference(self): | ||||||
|  |         """Ensure you can handle circular references | ||||||
|  |         """ | ||||||
|  |         class Person(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             relations = ListField(EmbeddedDocumentField('Relation')) | ||||||
|  |  | ||||||
|  |             def __repr__(self): | ||||||
|  |                 return "<Person: %s>" % self.name | ||||||
|  |  | ||||||
|  |         class Relation(EmbeddedDocument): | ||||||
|  |             name = StringField() | ||||||
|  |             person = ReferenceField('Person') | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |         mother = Person(name="Mother") | ||||||
|  |         daughter = Person(name="Daughter") | ||||||
|  |  | ||||||
|  |         mother.save() | ||||||
|  |         daughter.save() | ||||||
|  |  | ||||||
|  |         daughter_rel = Relation(name="Daughter", person=daughter) | ||||||
|  |         mother.relations.append(daughter_rel) | ||||||
|  |         mother.save() | ||||||
|  |  | ||||||
|  |         mother_rel = Relation(name="Daughter", person=mother) | ||||||
|  |         self_rel = Relation(name="Self", person=daughter) | ||||||
|  |         daughter.relations.append(mother_rel) | ||||||
|  |         daughter.relations.append(self_rel) | ||||||
|  |         daughter.save() | ||||||
|  |  | ||||||
|  |         self.assertEquals("[<Person: Mother>, <Person: Daughter>]", "%s" % Person.objects()) | ||||||
|  |  | ||||||
|  |     def test_circular_reference_on_self(self): | ||||||
|  |         """Ensure you can handle circular references | ||||||
|  |         """ | ||||||
|  |         class Person(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             relations = ListField(ReferenceField('self')) | ||||||
|  |  | ||||||
|  |             def __repr__(self): | ||||||
|  |                 return "<Person: %s>" % self.name | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |         mother = Person(name="Mother") | ||||||
|  |         daughter = Person(name="Daughter") | ||||||
|  |  | ||||||
|  |         mother.save() | ||||||
|  |         daughter.save() | ||||||
|  |  | ||||||
|  |         mother.relations.append(daughter) | ||||||
|  |         mother.save() | ||||||
|  |  | ||||||
|  |         daughter.relations.append(mother) | ||||||
|  |         daughter.relations.append(daughter) | ||||||
|  |         daughter.save() | ||||||
|  |  | ||||||
|  |         self.assertEquals("[<Person: Mother>, <Person: Daughter>]", "%s" % Person.objects()) | ||||||
|  |  | ||||||
|  |     def test_circular_tree_reference(self): | ||||||
|  |         """Ensure you can handle circular references with more than one level | ||||||
|  |         """ | ||||||
|  |         class Other(EmbeddedDocument): | ||||||
|  |             name = StringField() | ||||||
|  |             friends = ListField(ReferenceField('Person')) | ||||||
|  |  | ||||||
|  |         class Person(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             other = EmbeddedDocumentField(Other, default=lambda: Other()) | ||||||
|  |  | ||||||
|  |             def __repr__(self): | ||||||
|  |                 return "<Person: %s>" % self.name | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |         paul = Person(name="Paul") | ||||||
|  |         paul.save() | ||||||
|  |         maria = Person(name="Maria") | ||||||
|  |         maria.save() | ||||||
|  |         julia = Person(name='Julia') | ||||||
|  |         julia.save() | ||||||
|  |         anna = Person(name='Anna') | ||||||
|  |         anna.save() | ||||||
|  |  | ||||||
|  |         paul.other.friends = [maria, julia, anna] | ||||||
|  |         paul.other.name = "Paul's friends" | ||||||
|  |         paul.save() | ||||||
|  |  | ||||||
|  |         maria.other.friends = [paul, julia, anna] | ||||||
|  |         maria.other.name = "Maria's friends" | ||||||
|  |         maria.save() | ||||||
|  |  | ||||||
|  |         julia.other.friends = [paul, maria, anna] | ||||||
|  |         julia.other.name = "Julia's friends" | ||||||
|  |         julia.save() | ||||||
|  |  | ||||||
|  |         anna.other.friends = [paul, maria, julia] | ||||||
|  |         anna.other.name = "Anna's friends" | ||||||
|  |         anna.save() | ||||||
|  |  | ||||||
|  |         self.assertEquals( | ||||||
|  |             "[<Person: Paul>, <Person: Maria>, <Person: Julia>, <Person: Anna>]", | ||||||
|  |             "%s" % Person.objects() | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |     def test_generic_reference(self): | ||||||
|  |  | ||||||
|  |         class UserA(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class UserB(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class UserC(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Group(Document): | ||||||
|  |             members = ListField(GenericReferenceField()) | ||||||
|  |  | ||||||
|  |         UserA.drop_collection() | ||||||
|  |         UserB.drop_collection() | ||||||
|  |         UserC.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |         members = [] | ||||||
|  |         for i in xrange(1, 51): | ||||||
|  |             a = UserA(name='User A %s' % i) | ||||||
|  |             a.save() | ||||||
|  |  | ||||||
|  |             b = UserB(name='User B %s' % i) | ||||||
|  |             b.save() | ||||||
|  |  | ||||||
|  |             c = UserC(name='User C %s' % i) | ||||||
|  |             c.save() | ||||||
|  |  | ||||||
|  |             members += [a, b, c] | ||||||
|  |  | ||||||
|  |         group = Group(members=members) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         group = Group(members=members) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         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, 4) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             for m in group_obj.members: | ||||||
|  |                 self.assertTrue('User' in m.__class__.__name__) | ||||||
|  |  | ||||||
|  |         # Document select_related | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             group_obj = Group.objects.first().select_related() | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             for m in group_obj.members: | ||||||
|  |                 self.assertTrue('User' in m.__class__.__name__) | ||||||
|  |  | ||||||
|  |         # Queryset select_related | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             group_objs = Group.objects.select_related() | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             for group_obj in group_objs: | ||||||
|  |                 [m for m in group_obj.members] | ||||||
|  |                 self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |                 [m for m in group_obj.members] | ||||||
|  |                 self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |                 for m in group_obj.members: | ||||||
|  |                     self.assertTrue('User' in m.__class__.__name__) | ||||||
|  |  | ||||||
|  |         UserA.drop_collection() | ||||||
|  |         UserB.drop_collection() | ||||||
|  |         UserC.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_list_field_complex(self): | ||||||
|  |  | ||||||
|  |         class UserA(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class UserB(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class UserC(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Group(Document): | ||||||
|  |             members = ListField() | ||||||
|  |  | ||||||
|  |         UserA.drop_collection() | ||||||
|  |         UserB.drop_collection() | ||||||
|  |         UserC.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |         members = [] | ||||||
|  |         for i in xrange(1, 51): | ||||||
|  |             a = UserA(name='User A %s' % i) | ||||||
|  |             a.save() | ||||||
|  |  | ||||||
|  |             b = UserB(name='User B %s' % i) | ||||||
|  |             b.save() | ||||||
|  |  | ||||||
|  |             c = UserC(name='User C %s' % i) | ||||||
|  |             c.save() | ||||||
|  |  | ||||||
|  |             members += [a, b, c] | ||||||
|  |  | ||||||
|  |         group = Group(members=members) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         group = Group(members=members) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         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, 4) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             for m in group_obj.members: | ||||||
|  |                 self.assertTrue('User' in m.__class__.__name__) | ||||||
|  |  | ||||||
|  |         # Document select_related | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             group_obj = Group.objects.first().select_related() | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             for m in group_obj.members: | ||||||
|  |                 self.assertTrue('User' in m.__class__.__name__) | ||||||
|  |  | ||||||
|  |         # Queryset select_related | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             group_objs = Group.objects.select_related() | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             for group_obj in group_objs: | ||||||
|  |                 [m for m in group_obj.members] | ||||||
|  |                 self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |                 [m for m in group_obj.members] | ||||||
|  |                 self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |                 for m in group_obj.members: | ||||||
|  |                     self.assertTrue('User' in m.__class__.__name__) | ||||||
|  |  | ||||||
|  |         UserA.drop_collection() | ||||||
|  |         UserB.drop_collection() | ||||||
|  |         UserC.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_map_field_reference(self): | ||||||
|  |  | ||||||
|  |         class User(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Group(Document): | ||||||
|  |             members = MapField(ReferenceField(User)) | ||||||
|  |  | ||||||
|  |         User.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |         members = [] | ||||||
|  |         for i in xrange(1, 51): | ||||||
|  |             user = User(name='user %s' % i) | ||||||
|  |             user.save() | ||||||
|  |             members.append(user) | ||||||
|  |  | ||||||
|  |         group = Group(members=dict([(str(u.id), u) for u in members])) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         group = Group(members=dict([(str(u.id), u) for u in members])) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         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) | ||||||
|  |  | ||||||
|  |             for k, m in group_obj.members.iteritems(): | ||||||
|  |                 self.assertTrue(isinstance(m, User)) | ||||||
|  |  | ||||||
|  |         # 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) | ||||||
|  |  | ||||||
|  |             for k, m in group_obj.members.iteritems(): | ||||||
|  |                 self.assertTrue(isinstance(m, User)) | ||||||
|  |  | ||||||
|  |        # 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) | ||||||
|  |  | ||||||
|  |                 for k, m in group_obj.members.iteritems(): | ||||||
|  |                     self.assertTrue(isinstance(m, User)) | ||||||
|  |  | ||||||
|  |         User.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_dict_field(self): | ||||||
|  |  | ||||||
|  |         class UserA(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class UserB(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class UserC(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Group(Document): | ||||||
|  |             members = DictField() | ||||||
|  |  | ||||||
|  |         UserA.drop_collection() | ||||||
|  |         UserB.drop_collection() | ||||||
|  |         UserC.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |         members = [] | ||||||
|  |         for i in xrange(1, 51): | ||||||
|  |             a = UserA(name='User A %s' % i) | ||||||
|  |             a.save() | ||||||
|  |  | ||||||
|  |             b = UserB(name='User B %s' % i) | ||||||
|  |             b.save() | ||||||
|  |  | ||||||
|  |             c = UserC(name='User C %s' % i) | ||||||
|  |             c.save() | ||||||
|  |  | ||||||
|  |             members += [a, b, c] | ||||||
|  |  | ||||||
|  |         group = Group(members=dict([(str(u.id), u) for u in members])) | ||||||
|  |         group.save() | ||||||
|  |         group = Group(members=dict([(str(u.id), u) for u in members])) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         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, 4) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             for k, m in group_obj.members.iteritems(): | ||||||
|  |                 self.assertTrue('User' in m.__class__.__name__) | ||||||
|  |  | ||||||
|  |         # Document select_related | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             group_obj = Group.objects.first().select_related() | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             for k, m in group_obj.members.iteritems(): | ||||||
|  |                 self.assertTrue('User' in m.__class__.__name__) | ||||||
|  |  | ||||||
|  |         # Queryset select_related | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             group_objs = Group.objects.select_related() | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             for group_obj in group_objs: | ||||||
|  |                 [m for m in group_obj.members] | ||||||
|  |                 self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |                 [m for m in group_obj.members] | ||||||
|  |                 self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |                 for k, m in group_obj.members.iteritems(): | ||||||
|  |                     self.assertTrue('User' in m.__class__.__name__) | ||||||
|  |  | ||||||
|  |         Group.objects.delete() | ||||||
|  |         Group().save() | ||||||
|  |  | ||||||
|  |         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, 1) | ||||||
|  |             self.assertEqual(group_obj.members, {}) | ||||||
|  |  | ||||||
|  |         UserA.drop_collection() | ||||||
|  |         UserB.drop_collection() | ||||||
|  |         UserC.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_dict_field_no_field_inheritance(self): | ||||||
|  |  | ||||||
|  |         class UserA(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             meta = {'allow_inheritance': False} | ||||||
|  |  | ||||||
|  |         class Group(Document): | ||||||
|  |             members = DictField() | ||||||
|  |  | ||||||
|  |         UserA.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |         members = [] | ||||||
|  |         for i in xrange(1, 51): | ||||||
|  |             a = UserA(name='User A %s' % i) | ||||||
|  |             a.save() | ||||||
|  |  | ||||||
|  |             members += [a] | ||||||
|  |  | ||||||
|  |         group = Group(members=dict([(str(u.id), u) for u in members])) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         group = Group(members=dict([(str(u.id), u) for u in members])) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         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) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |             for k, m in group_obj.members.iteritems(): | ||||||
|  |                 self.assertTrue(isinstance(m, UserA)) | ||||||
|  |  | ||||||
|  |         # 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) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |             for k, m in group_obj.members.iteritems(): | ||||||
|  |                 self.assertTrue(isinstance(m, UserA)) | ||||||
|  |  | ||||||
|  |         # 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) | ||||||
|  |  | ||||||
|  |                 [m for m in group_obj.members] | ||||||
|  |                 self.assertEqual(q, 2) | ||||||
|  |  | ||||||
|  |                 for k, m in group_obj.members.iteritems(): | ||||||
|  |                     self.assertTrue(isinstance(m, UserA)) | ||||||
|  |  | ||||||
|  |         UserA.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_generic_reference_map_field(self): | ||||||
|  |  | ||||||
|  |         class UserA(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class UserB(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class UserC(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         class Group(Document): | ||||||
|  |             members = MapField(GenericReferenceField()) | ||||||
|  |  | ||||||
|  |         UserA.drop_collection() | ||||||
|  |         UserB.drop_collection() | ||||||
|  |         UserC.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |         members = [] | ||||||
|  |         for i in xrange(1, 51): | ||||||
|  |             a = UserA(name='User A %s' % i) | ||||||
|  |             a.save() | ||||||
|  |  | ||||||
|  |             b = UserB(name='User B %s' % i) | ||||||
|  |             b.save() | ||||||
|  |  | ||||||
|  |             c = UserC(name='User C %s' % i) | ||||||
|  |             c.save() | ||||||
|  |  | ||||||
|  |             members += [a, b, c] | ||||||
|  |  | ||||||
|  |         group = Group(members=dict([(str(u.id), u) for u in members])) | ||||||
|  |         group.save() | ||||||
|  |         group = Group(members=dict([(str(u.id), u) for u in members])) | ||||||
|  |         group.save() | ||||||
|  |  | ||||||
|  |         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, 4) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             for k, m in group_obj.members.iteritems(): | ||||||
|  |                 self.assertTrue('User' in m.__class__.__name__) | ||||||
|  |  | ||||||
|  |         # Document select_related | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             group_obj = Group.objects.first().select_related() | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             [m for m in group_obj.members] | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             for k, m in group_obj.members.iteritems(): | ||||||
|  |                 self.assertTrue('User' in m.__class__.__name__) | ||||||
|  |  | ||||||
|  |         # Queryset select_related | ||||||
|  |         with query_counter() as q: | ||||||
|  |             self.assertEqual(q, 0) | ||||||
|  |  | ||||||
|  |             group_objs = Group.objects.select_related() | ||||||
|  |             self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |             for group_obj in group_objs: | ||||||
|  |                 [m for m in group_obj.members] | ||||||
|  |                 self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |                 [m for m in group_obj.members] | ||||||
|  |                 self.assertEqual(q, 4) | ||||||
|  |  | ||||||
|  |                 for k, m in group_obj.members.iteritems(): | ||||||
|  |                     self.assertTrue('User' in m.__class__.__name__) | ||||||
|  |  | ||||||
|  |         Group.objects.delete() | ||||||
|  |         Group().save() | ||||||
|  |  | ||||||
|  |         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, 1) | ||||||
|  |  | ||||||
|  |         UserA.drop_collection() | ||||||
|  |         UserB.drop_collection() | ||||||
|  |         UserC.drop_collection() | ||||||
|  |         Group.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_multidirectional_lists(self): | ||||||
|  |  | ||||||
|  |         class Asset(Document): | ||||||
|  |             name = StringField(max_length=250, required=True) | ||||||
|  |             parent = GenericReferenceField(default=None) | ||||||
|  |             parents = ListField(GenericReferenceField()) | ||||||
|  |             children = ListField(GenericReferenceField()) | ||||||
|  |  | ||||||
|  |         Asset.drop_collection() | ||||||
|  |  | ||||||
|  |         root = Asset(name='', path="/", title="Site Root") | ||||||
|  |         root.save() | ||||||
|  |  | ||||||
|  |         company = Asset(name='company', title='Company', parent=root, parents=[root]) | ||||||
|  |         company.save() | ||||||
|  |  | ||||||
|  |         root.children = [company] | ||||||
|  |         root.save() | ||||||
|  |  | ||||||
|  |         root = root.reload() | ||||||
|  |         self.assertEquals(root.children, [company]) | ||||||
|  |         self.assertEquals(company.parents, [root]) | ||||||
|  |  | ||||||
|  |     def test_dict_in_dbref_instance(self): | ||||||
|  |  | ||||||
|  |         class Person(Document): | ||||||
|  |             name = StringField(max_length=250, required=True) | ||||||
|  |  | ||||||
|  |         class Room(Document): | ||||||
|  |             number = StringField(max_length=250, required=True) | ||||||
|  |             staffs_with_position = ListField(DictField()) | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |         Room.drop_collection() | ||||||
|  |  | ||||||
|  |         bob = Person.objects.create(name='Bob') | ||||||
|  |         bob.save() | ||||||
|  |         sarah = Person.objects.create(name='Sarah') | ||||||
|  |         sarah.save() | ||||||
|  |  | ||||||
|  |         room_101 = Room.objects.create(number="101") | ||||||
|  |         room_101.staffs_with_position = [ | ||||||
|  |             {'position_key': 'window', 'staff': sarah}, | ||||||
|  |             {'position_key': 'door', 'staff': bob.to_dbref()}] | ||||||
|  |         room_101.save() | ||||||
|  |  | ||||||
|  |         room = Room.objects.first().select_related() | ||||||
|  |         self.assertEquals(room.staffs_with_position[0]['staff'], sarah) | ||||||
|  |         self.assertEquals(room.staffs_with_position[1]['staff'], bob) | ||||||
							
								
								
									
										90
									
								
								tests/django_tests.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								tests/django_tests.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,90 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  |  | ||||||
|  | import unittest | ||||||
|  |  | ||||||
|  | from mongoengine import * | ||||||
|  | from mongoengine.django.shortcuts import get_document_or_404 | ||||||
|  |  | ||||||
|  | from django.http import Http404 | ||||||
|  | from django.template import Context, Template | ||||||
|  | from django.conf import settings | ||||||
|  | from django.core.paginator import Paginator | ||||||
|  |  | ||||||
|  | settings.configure() | ||||||
|  |  | ||||||
|  | class QuerySetTest(unittest.TestCase): | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         connect(db='mongoenginetest') | ||||||
|  |  | ||||||
|  |         class Person(Document): | ||||||
|  |             name = StringField() | ||||||
|  |             age = IntField() | ||||||
|  |         self.Person = Person | ||||||
|  |  | ||||||
|  |     def test_order_by_in_django_template(self): | ||||||
|  |         """Ensure that QuerySets are properly ordered in Django template. | ||||||
|  |         """ | ||||||
|  |         self.Person.drop_collection() | ||||||
|  |  | ||||||
|  |         self.Person(name="A", age=20).save() | ||||||
|  |         self.Person(name="D", age=10).save() | ||||||
|  |         self.Person(name="B", age=40).save() | ||||||
|  |         self.Person(name="C", age=30).save() | ||||||
|  |  | ||||||
|  |         t = Template("{% for o in ol %}{{ o.name }}-{{ o.age }}:{% endfor %}") | ||||||
|  |  | ||||||
|  |         d = {"ol": self.Person.objects.order_by('-name')} | ||||||
|  |         self.assertEqual(t.render(Context(d)), u'D-10:C-30:B-40:A-20:') | ||||||
|  |         d = {"ol": self.Person.objects.order_by('+name')} | ||||||
|  |         self.assertEqual(t.render(Context(d)), u'A-20:B-40:C-30:D-10:') | ||||||
|  |         d = {"ol": self.Person.objects.order_by('-age')} | ||||||
|  |         self.assertEqual(t.render(Context(d)), u'B-40:C-30:A-20:D-10:') | ||||||
|  |         d = {"ol": self.Person.objects.order_by('+age')} | ||||||
|  |         self.assertEqual(t.render(Context(d)), u'D-10:A-20:C-30:B-40:') | ||||||
|  |  | ||||||
|  |         self.Person.drop_collection() | ||||||
|  |  | ||||||
|  |     def test_q_object_filter_in_template(self): | ||||||
|  |  | ||||||
|  |         self.Person.drop_collection() | ||||||
|  |  | ||||||
|  |         self.Person(name="A", age=20).save() | ||||||
|  |         self.Person(name="D", age=10).save() | ||||||
|  |         self.Person(name="B", age=40).save() | ||||||
|  |         self.Person(name="C", age=30).save() | ||||||
|  |  | ||||||
|  |         t = Template("{% for o in ol %}{{ o.name }}-{{ o.age }}:{% endfor %}") | ||||||
|  |  | ||||||
|  |         d = {"ol": self.Person.objects.filter(Q(age=10) | Q(name="C"))} | ||||||
|  |         self.assertEqual(t.render(Context(d)), 'D-10:C-30:') | ||||||
|  |  | ||||||
|  |         # Check double rendering doesn't throw an error | ||||||
|  |         self.assertEqual(t.render(Context(d)), 'D-10:C-30:') | ||||||
|  |  | ||||||
|  |     def test_get_document_or_404(self): | ||||||
|  |         p = self.Person(name="G404") | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         self.assertRaises(Http404, get_document_or_404, self.Person, pk='1234') | ||||||
|  |         self.assertEqual(p, get_document_or_404(self.Person, pk=p.pk)) | ||||||
|  |  | ||||||
|  |     def test_pagination(self): | ||||||
|  |         """Ensure that Pagination works as expected | ||||||
|  |         """ | ||||||
|  |         class Page(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         Page.drop_collection() | ||||||
|  |  | ||||||
|  |         for i in xrange(1, 11): | ||||||
|  |             Page(name=str(i)).save() | ||||||
|  |  | ||||||
|  |         paginator = Paginator(Page.objects.all(), 2) | ||||||
|  |  | ||||||
|  |         t = Template("{% for i in page.object_list  %}{{ i.name }}:{% endfor %}") | ||||||
|  |         for p in paginator.page_range: | ||||||
|  |             d = {"page": paginator.page(p)} | ||||||
|  |             end = p * 2 | ||||||
|  |             start = end - 1 | ||||||
|  |             self.assertEqual(t.render(Context(d)), u'%d:%d:' % (start, end)) | ||||||
							
								
								
									
										2533
									
								
								tests/document.py
									
									
									
									
									
								
							
							
						
						
									
										2533
									
								
								tests/document.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										502
									
								
								tests/dynamic_document.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										502
									
								
								tests/dynamic_document.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,502 @@ | |||||||
|  | import unittest | ||||||
|  |  | ||||||
|  | from mongoengine import * | ||||||
|  | from mongoengine.connection import get_db | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class DynamicDocTest(unittest.TestCase): | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         connect(db='mongoenginetest') | ||||||
|  |         self.db = get_db() | ||||||
|  |  | ||||||
|  |         class Person(DynamicDocument): | ||||||
|  |             name = StringField() | ||||||
|  |             meta = {'allow_inheritance': True} | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         self.Person = Person | ||||||
|  |  | ||||||
|  |     def test_simple_dynamic_document(self): | ||||||
|  |         """Ensures simple dynamic documents are saved correctly""" | ||||||
|  |  | ||||||
|  |         p = self.Person() | ||||||
|  |         p.name = "James" | ||||||
|  |         p.age = 34 | ||||||
|  |  | ||||||
|  |         self.assertEquals(p.to_mongo(), | ||||||
|  |             {"_types": ["Person"], "_cls": "Person", | ||||||
|  |              "name": "James", "age": 34} | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         self.assertEquals(self.Person.objects.first().age, 34) | ||||||
|  |  | ||||||
|  |         # Confirm no changes to self.Person | ||||||
|  |         self.assertFalse(hasattr(self.Person, 'age')) | ||||||
|  |  | ||||||
|  |     def test_dynamic_document_delta(self): | ||||||
|  |         """Ensures simple dynamic documents can delta correctly""" | ||||||
|  |         p = self.Person(name="James", age=34) | ||||||
|  |         self.assertEquals(p._delta(), ({'_types': ['Person'], 'age': 34, 'name': 'James', '_cls': 'Person'}, {})) | ||||||
|  |  | ||||||
|  |         p.doc = 123 | ||||||
|  |         del(p.doc) | ||||||
|  |         self.assertEquals(p._delta(), ({'_types': ['Person'], 'age': 34, 'name': 'James', '_cls': 'Person'}, {'doc': 1})) | ||||||
|  |  | ||||||
|  |     def test_change_scope_of_variable(self): | ||||||
|  |         """Test changing the scope of a dynamic field has no adverse effects""" | ||||||
|  |         p = self.Person() | ||||||
|  |         p.name = "Dean" | ||||||
|  |         p.misc = 22 | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         p = self.Person.objects.get() | ||||||
|  |         p.misc = {'hello': 'world'} | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         p = self.Person.objects.get() | ||||||
|  |         self.assertEquals(p.misc, {'hello': 'world'}) | ||||||
|  |  | ||||||
|  |     def test_delete_dynamic_field(self): | ||||||
|  |         """Test deleting a dynamic field works""" | ||||||
|  |         self.Person.drop_collection() | ||||||
|  |         p = self.Person() | ||||||
|  |         p.name = "Dean" | ||||||
|  |         p.misc = 22 | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         p = self.Person.objects.get() | ||||||
|  |         p.misc = {'hello': 'world'} | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         p = self.Person.objects.get() | ||||||
|  |         self.assertEquals(p.misc, {'hello': 'world'}) | ||||||
|  |         collection = self.db[self.Person._get_collection_name()] | ||||||
|  |         obj = collection.find_one() | ||||||
|  |         self.assertEquals(sorted(obj.keys()), ['_cls', '_id', '_types', 'misc', 'name']) | ||||||
|  |  | ||||||
|  |         del(p.misc) | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         p = self.Person.objects.get() | ||||||
|  |         self.assertFalse(hasattr(p, 'misc')) | ||||||
|  |  | ||||||
|  |         obj = collection.find_one() | ||||||
|  |         self.assertEquals(sorted(obj.keys()), ['_cls', '_id', '_types', 'name']) | ||||||
|  |  | ||||||
|  |     def test_dynamic_document_queries(self): | ||||||
|  |         """Ensure we can query dynamic fields""" | ||||||
|  |         p = self.Person() | ||||||
|  |         p.name = "Dean" | ||||||
|  |         p.age = 22 | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         self.assertEquals(1, self.Person.objects(age=22).count()) | ||||||
|  |         p = self.Person.objects(age=22) | ||||||
|  |         p = p.get() | ||||||
|  |         self.assertEquals(22, p.age) | ||||||
|  |  | ||||||
|  |     def test_complex_dynamic_document_queries(self): | ||||||
|  |         class Person(DynamicDocument): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |         Person.drop_collection() | ||||||
|  |  | ||||||
|  |         p = Person(name="test") | ||||||
|  |         p.age = "ten" | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         p1 = Person(name="test1") | ||||||
|  |         p1.age = "less then ten and a half" | ||||||
|  |         p1.save() | ||||||
|  |  | ||||||
|  |         p2 = Person(name="test2") | ||||||
|  |         p2.age = 10 | ||||||
|  |         p2.save() | ||||||
|  |  | ||||||
|  |         self.assertEquals(Person.objects(age__icontains='ten').count(), 2) | ||||||
|  |         self.assertEquals(Person.objects(age__gte=10).count(), 1) | ||||||
|  |  | ||||||
|  |     def test_complex_data_lookups(self): | ||||||
|  |         """Ensure you can query dynamic document dynamic fields""" | ||||||
|  |         p = self.Person() | ||||||
|  |         p.misc = {'hello': 'world'} | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         self.assertEquals(1, self.Person.objects(misc__hello='world').count()) | ||||||
|  |  | ||||||
|  |     def test_inheritance(self): | ||||||
|  |         """Ensure that dynamic document plays nice with inheritance""" | ||||||
|  |         class Employee(self.Person): | ||||||
|  |             salary = IntField() | ||||||
|  |  | ||||||
|  |         Employee.drop_collection() | ||||||
|  |  | ||||||
|  |         self.assertTrue('name' in Employee._fields) | ||||||
|  |         self.assertTrue('salary' in Employee._fields) | ||||||
|  |         self.assertEqual(Employee._get_collection_name(), | ||||||
|  |                          self.Person._get_collection_name()) | ||||||
|  |  | ||||||
|  |         joe_bloggs = Employee() | ||||||
|  |         joe_bloggs.name = "Joe Bloggs" | ||||||
|  |         joe_bloggs.salary = 10 | ||||||
|  |         joe_bloggs.age = 20 | ||||||
|  |         joe_bloggs.save() | ||||||
|  |  | ||||||
|  |         self.assertEquals(1, self.Person.objects(age=20).count()) | ||||||
|  |         self.assertEquals(1, Employee.objects(age=20).count()) | ||||||
|  |  | ||||||
|  |         joe_bloggs = self.Person.objects.first() | ||||||
|  |         self.assertTrue(isinstance(joe_bloggs, Employee)) | ||||||
|  |  | ||||||
|  |     def test_embedded_dynamic_document(self): | ||||||
|  |         """Test dynamic embedded documents""" | ||||||
|  |         class Embedded(DynamicEmbeddedDocument): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         class Doc(DynamicDocument): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         Doc.drop_collection() | ||||||
|  |         doc = Doc() | ||||||
|  |  | ||||||
|  |         embedded_1 = Embedded() | ||||||
|  |         embedded_1.string_field = 'hello' | ||||||
|  |         embedded_1.int_field = 1 | ||||||
|  |         embedded_1.dict_field = {'hello': 'world'} | ||||||
|  |         embedded_1.list_field = ['1', 2, {'hello': 'world'}] | ||||||
|  |         doc.embedded_field = embedded_1 | ||||||
|  |  | ||||||
|  |         self.assertEquals(doc.to_mongo(), {"_types": ['Doc'], "_cls": "Doc", | ||||||
|  |             "embedded_field": { | ||||||
|  |                 "_types": ['Embedded'], "_cls": "Embedded", | ||||||
|  |                 "string_field": "hello", | ||||||
|  |                 "int_field": 1, | ||||||
|  |                 "dict_field": {"hello": "world"}, | ||||||
|  |                 "list_field": ['1', 2, {'hello': 'world'}] | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |         doc.save() | ||||||
|  |  | ||||||
|  |         doc = Doc.objects.first() | ||||||
|  |         self.assertEquals(doc.embedded_field.__class__, Embedded) | ||||||
|  |         self.assertEquals(doc.embedded_field.string_field, "hello") | ||||||
|  |         self.assertEquals(doc.embedded_field.int_field, 1) | ||||||
|  |         self.assertEquals(doc.embedded_field.dict_field, {'hello': 'world'}) | ||||||
|  |         self.assertEquals(doc.embedded_field.list_field, ['1', 2, {'hello': 'world'}]) | ||||||
|  |  | ||||||
|  |     def test_complex_embedded_documents(self): | ||||||
|  |         """Test complex dynamic embedded documents setups""" | ||||||
|  |         class Embedded(DynamicEmbeddedDocument): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         class Doc(DynamicDocument): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         Doc.drop_collection() | ||||||
|  |         doc = Doc() | ||||||
|  |  | ||||||
|  |         embedded_1 = Embedded() | ||||||
|  |         embedded_1.string_field = 'hello' | ||||||
|  |         embedded_1.int_field = 1 | ||||||
|  |         embedded_1.dict_field = {'hello': 'world'} | ||||||
|  |  | ||||||
|  |         embedded_2 = Embedded() | ||||||
|  |         embedded_2.string_field = 'hello' | ||||||
|  |         embedded_2.int_field = 1 | ||||||
|  |         embedded_2.dict_field = {'hello': 'world'} | ||||||
|  |         embedded_2.list_field = ['1', 2, {'hello': 'world'}] | ||||||
|  |  | ||||||
|  |         embedded_1.list_field = ['1', 2, embedded_2] | ||||||
|  |         doc.embedded_field = embedded_1 | ||||||
|  |  | ||||||
|  |         self.assertEquals(doc.to_mongo(), {"_types": ['Doc'], "_cls": "Doc", | ||||||
|  |             "embedded_field": { | ||||||
|  |                 "_types": ['Embedded'], "_cls": "Embedded", | ||||||
|  |                 "string_field": "hello", | ||||||
|  |                 "int_field": 1, | ||||||
|  |                 "dict_field": {"hello": "world"}, | ||||||
|  |                 "list_field": ['1', 2, | ||||||
|  |                     {"_types": ['Embedded'], "_cls": "Embedded", | ||||||
|  |                     "string_field": "hello", | ||||||
|  |                     "int_field": 1, | ||||||
|  |                     "dict_field": {"hello": "world"}, | ||||||
|  |                     "list_field": ['1', 2, {'hello': 'world'}]} | ||||||
|  |                 ] | ||||||
|  |             } | ||||||
|  |         }) | ||||||
|  |         doc.save() | ||||||
|  |         doc = Doc.objects.first() | ||||||
|  |         self.assertEquals(doc.embedded_field.__class__, Embedded) | ||||||
|  |         self.assertEquals(doc.embedded_field.string_field, "hello") | ||||||
|  |         self.assertEquals(doc.embedded_field.int_field, 1) | ||||||
|  |         self.assertEquals(doc.embedded_field.dict_field, {'hello': 'world'}) | ||||||
|  |         self.assertEquals(doc.embedded_field.list_field[0], '1') | ||||||
|  |         self.assertEquals(doc.embedded_field.list_field[1], 2) | ||||||
|  |  | ||||||
|  |         embedded_field = doc.embedded_field.list_field[2] | ||||||
|  |  | ||||||
|  |         self.assertEquals(embedded_field.__class__, Embedded) | ||||||
|  |         self.assertEquals(embedded_field.string_field, "hello") | ||||||
|  |         self.assertEquals(embedded_field.int_field, 1) | ||||||
|  |         self.assertEquals(embedded_field.dict_field, {'hello': 'world'}) | ||||||
|  |         self.assertEquals(embedded_field.list_field, ['1', 2, {'hello': 'world'}]) | ||||||
|  |  | ||||||
|  |     def test_delta_for_dynamic_documents(self): | ||||||
|  |         p = self.Person() | ||||||
|  |         p.name = "Dean" | ||||||
|  |         p.age = 22 | ||||||
|  |         p.save() | ||||||
|  |  | ||||||
|  |         p.age = 24 | ||||||
|  |         self.assertEquals(p.age, 24) | ||||||
|  |         self.assertEquals(p._get_changed_fields(), ['age']) | ||||||
|  |         self.assertEquals(p._delta(), ({'age': 24}, {})) | ||||||
|  |  | ||||||
|  |         p = self.Person.objects(age=22).get() | ||||||
|  |         p.age = 24 | ||||||
|  |         self.assertEquals(p.age, 24) | ||||||
|  |         self.assertEquals(p._get_changed_fields(), ['age']) | ||||||
|  |         self.assertEquals(p._delta(), ({'age': 24}, {})) | ||||||
|  |  | ||||||
|  |         p.save() | ||||||
|  |         self.assertEquals(1, self.Person.objects(age=24).count()) | ||||||
|  |  | ||||||
|  |     def test_delta(self): | ||||||
|  |  | ||||||
|  |         class Doc(DynamicDocument): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         Doc.drop_collection() | ||||||
|  |         doc = Doc() | ||||||
|  |         doc.save() | ||||||
|  |  | ||||||
|  |         doc = Doc.objects.first() | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), []) | ||||||
|  |         self.assertEquals(doc._delta(), ({}, {})) | ||||||
|  |  | ||||||
|  |         doc.string_field = 'hello' | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['string_field']) | ||||||
|  |         self.assertEquals(doc._delta(), ({'string_field': 'hello'}, {})) | ||||||
|  |  | ||||||
|  |         doc._changed_fields = [] | ||||||
|  |         doc.int_field = 1 | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['int_field']) | ||||||
|  |         self.assertEquals(doc._delta(), ({'int_field': 1}, {})) | ||||||
|  |  | ||||||
|  |         doc._changed_fields = [] | ||||||
|  |         dict_value = {'hello': 'world', 'ping': 'pong'} | ||||||
|  |         doc.dict_field = dict_value | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['dict_field']) | ||||||
|  |         self.assertEquals(doc._delta(), ({'dict_field': dict_value}, {})) | ||||||
|  |  | ||||||
|  |         doc._changed_fields = [] | ||||||
|  |         list_value = ['1', 2, {'hello': 'world'}] | ||||||
|  |         doc.list_field = list_value | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['list_field']) | ||||||
|  |         self.assertEquals(doc._delta(), ({'list_field': list_value}, {})) | ||||||
|  |  | ||||||
|  |         # Test unsetting | ||||||
|  |         doc._changed_fields = [] | ||||||
|  |         doc.dict_field = {} | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['dict_field']) | ||||||
|  |         self.assertEquals(doc._delta(), ({}, {'dict_field': 1})) | ||||||
|  |  | ||||||
|  |         doc._changed_fields = [] | ||||||
|  |         doc.list_field = [] | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['list_field']) | ||||||
|  |         self.assertEquals(doc._delta(), ({}, {'list_field': 1})) | ||||||
|  |  | ||||||
|  |     def test_delta_recursive(self): | ||||||
|  |         """Testing deltaing works with dynamic documents""" | ||||||
|  |         class Embedded(DynamicEmbeddedDocument): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         class Doc(DynamicDocument): | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         Doc.drop_collection() | ||||||
|  |         doc = Doc() | ||||||
|  |         doc.save() | ||||||
|  |  | ||||||
|  |         doc = Doc.objects.first() | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), []) | ||||||
|  |         self.assertEquals(doc._delta(), ({}, {})) | ||||||
|  |  | ||||||
|  |         embedded_1 = Embedded() | ||||||
|  |         embedded_1.string_field = 'hello' | ||||||
|  |         embedded_1.int_field = 1 | ||||||
|  |         embedded_1.dict_field = {'hello': 'world'} | ||||||
|  |         embedded_1.list_field = ['1', 2, {'hello': 'world'}] | ||||||
|  |         doc.embedded_field = embedded_1 | ||||||
|  |  | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['embedded_field']) | ||||||
|  |  | ||||||
|  |         embedded_delta = { | ||||||
|  |             'string_field': 'hello', | ||||||
|  |             'int_field': 1, | ||||||
|  |             'dict_field': {'hello': 'world'}, | ||||||
|  |             'list_field': ['1', 2, {'hello': 'world'}] | ||||||
|  |         } | ||||||
|  |         self.assertEquals(doc.embedded_field._delta(), (embedded_delta, {})) | ||||||
|  |         embedded_delta.update({ | ||||||
|  |             '_types': ['Embedded'], | ||||||
|  |             '_cls': 'Embedded', | ||||||
|  |         }) | ||||||
|  |         self.assertEquals(doc._delta(), ({'embedded_field': embedded_delta}, {})) | ||||||
|  |  | ||||||
|  |         doc.save() | ||||||
|  |         doc.reload() | ||||||
|  |  | ||||||
|  |         doc.embedded_field.dict_field = {} | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['embedded_field.dict_field']) | ||||||
|  |         self.assertEquals(doc.embedded_field._delta(), ({}, {'dict_field': 1})) | ||||||
|  |  | ||||||
|  |         self.assertEquals(doc._delta(), ({}, {'embedded_field.dict_field': 1})) | ||||||
|  |         doc.save() | ||||||
|  |         doc.reload() | ||||||
|  |  | ||||||
|  |         doc.embedded_field.list_field = [] | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['embedded_field.list_field']) | ||||||
|  |         self.assertEquals(doc.embedded_field._delta(), ({}, {'list_field': 1})) | ||||||
|  |         self.assertEquals(doc._delta(), ({}, {'embedded_field.list_field': 1})) | ||||||
|  |         doc.save() | ||||||
|  |         doc.reload() | ||||||
|  |  | ||||||
|  |         embedded_2 = Embedded() | ||||||
|  |         embedded_2.string_field = 'hello' | ||||||
|  |         embedded_2.int_field = 1 | ||||||
|  |         embedded_2.dict_field = {'hello': 'world'} | ||||||
|  |         embedded_2.list_field = ['1', 2, {'hello': 'world'}] | ||||||
|  |  | ||||||
|  |         doc.embedded_field.list_field = ['1', 2, embedded_2] | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['embedded_field.list_field']) | ||||||
|  |         self.assertEquals(doc.embedded_field._delta(), ({ | ||||||
|  |             'list_field': ['1', 2, { | ||||||
|  |                 '_cls': 'Embedded', | ||||||
|  |                 '_types': ['Embedded'], | ||||||
|  |                 'string_field': 'hello', | ||||||
|  |                 'dict_field': {'hello': 'world'}, | ||||||
|  |                 'int_field': 1, | ||||||
|  |                 'list_field': ['1', 2, {'hello': 'world'}], | ||||||
|  |             }] | ||||||
|  |         }, {})) | ||||||
|  |  | ||||||
|  |         self.assertEquals(doc._delta(), ({ | ||||||
|  |             'embedded_field.list_field': ['1', 2, { | ||||||
|  |                 '_cls': 'Embedded', | ||||||
|  |                  '_types': ['Embedded'], | ||||||
|  |                  'string_field': 'hello', | ||||||
|  |                  'dict_field': {'hello': 'world'}, | ||||||
|  |                  'int_field': 1, | ||||||
|  |                  'list_field': ['1', 2, {'hello': 'world'}], | ||||||
|  |             }] | ||||||
|  |         }, {})) | ||||||
|  |         doc.save() | ||||||
|  |         doc.reload() | ||||||
|  |  | ||||||
|  |         self.assertEquals(doc.embedded_field.list_field[2]._changed_fields, []) | ||||||
|  |         self.assertEquals(doc.embedded_field.list_field[0], '1') | ||||||
|  |         self.assertEquals(doc.embedded_field.list_field[1], 2) | ||||||
|  |         for k in doc.embedded_field.list_field[2]._fields: | ||||||
|  |             self.assertEquals(doc.embedded_field.list_field[2][k], embedded_2[k]) | ||||||
|  |  | ||||||
|  |         doc.embedded_field.list_field[2].string_field = 'world' | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['embedded_field.list_field.2.string_field']) | ||||||
|  |         self.assertEquals(doc.embedded_field._delta(), ({'list_field.2.string_field': 'world'}, {})) | ||||||
|  |         self.assertEquals(doc._delta(), ({'embedded_field.list_field.2.string_field': 'world'}, {})) | ||||||
|  |         doc.save() | ||||||
|  |         doc.reload() | ||||||
|  |         self.assertEquals(doc.embedded_field.list_field[2].string_field, 'world') | ||||||
|  |  | ||||||
|  |         # Test multiple assignments | ||||||
|  |         doc.embedded_field.list_field[2].string_field = 'hello world' | ||||||
|  |         doc.embedded_field.list_field[2] = doc.embedded_field.list_field[2] | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['embedded_field.list_field']) | ||||||
|  |         self.assertEquals(doc.embedded_field._delta(), ({ | ||||||
|  |             'list_field': ['1', 2, { | ||||||
|  |             '_types': ['Embedded'], | ||||||
|  |             '_cls': 'Embedded', | ||||||
|  |             'string_field': 'hello world', | ||||||
|  |             'int_field': 1, | ||||||
|  |             'list_field': ['1', 2, {'hello': 'world'}], | ||||||
|  |             'dict_field': {'hello': 'world'}}]}, {})) | ||||||
|  |         self.assertEquals(doc._delta(), ({ | ||||||
|  |             'embedded_field.list_field': ['1', 2, { | ||||||
|  |                 '_types': ['Embedded'], | ||||||
|  |                 '_cls': 'Embedded', | ||||||
|  |                 'string_field': 'hello world', | ||||||
|  |                 'int_field': 1, | ||||||
|  |                 'list_field': ['1', 2, {'hello': 'world'}], | ||||||
|  |                 'dict_field': {'hello': 'world'}} | ||||||
|  |             ]}, {})) | ||||||
|  |         doc.save() | ||||||
|  |         doc.reload() | ||||||
|  |         self.assertEquals(doc.embedded_field.list_field[2].string_field, 'hello world') | ||||||
|  |  | ||||||
|  |         # Test list native methods | ||||||
|  |         doc.embedded_field.list_field[2].list_field.pop(0) | ||||||
|  |         self.assertEquals(doc._delta(), ({'embedded_field.list_field.2.list_field': [2, {'hello': 'world'}]}, {})) | ||||||
|  |         doc.save() | ||||||
|  |         doc.reload() | ||||||
|  |  | ||||||
|  |         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]}, {})) | ||||||
|  |         doc.save() | ||||||
|  |         doc.reload() | ||||||
|  |         self.assertEquals(doc.embedded_field.list_field[2].list_field, [2, {'hello': 'world'}, 1]) | ||||||
|  |  | ||||||
|  |         doc.embedded_field.list_field[2].list_field.sort() | ||||||
|  |         doc.save() | ||||||
|  |         doc.reload() | ||||||
|  |         self.assertEquals(doc.embedded_field.list_field[2].list_field, [1, 2, {'hello': 'world'}]) | ||||||
|  |  | ||||||
|  |         del(doc.embedded_field.list_field[2].list_field[2]['hello']) | ||||||
|  |         self.assertEquals(doc._delta(), ({'embedded_field.list_field.2.list_field': [1, 2, {}]}, {})) | ||||||
|  |         doc.save() | ||||||
|  |         doc.reload() | ||||||
|  |  | ||||||
|  |         del(doc.embedded_field.list_field[2].list_field) | ||||||
|  |         self.assertEquals(doc._delta(), ({}, {'embedded_field.list_field.2.list_field': 1})) | ||||||
|  |  | ||||||
|  |         doc.save() | ||||||
|  |         doc.reload() | ||||||
|  |  | ||||||
|  |         doc.dict_field = {'embedded': embedded_1} | ||||||
|  |         doc.save() | ||||||
|  |         doc.reload() | ||||||
|  |  | ||||||
|  |         doc.dict_field['embedded'].string_field = 'Hello World' | ||||||
|  |         self.assertEquals(doc._get_changed_fields(), ['dict_field.embedded.string_field']) | ||||||
|  |         self.assertEquals(doc._delta(), ({'dict_field.embedded.string_field': 'Hello World'}, {})) | ||||||
|  |  | ||||||
|  |     def test_indexes(self): | ||||||
|  |         """Ensure that indexes are used when meta[indexes] is specified. | ||||||
|  |         """ | ||||||
|  |         class BlogPost(DynamicDocument): | ||||||
|  |             meta = { | ||||||
|  |                 'indexes': [ | ||||||
|  |                     '-date', | ||||||
|  |                     ('category', '-date') | ||||||
|  |                 ], | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |         BlogPost.drop_collection() | ||||||
|  |  | ||||||
|  |         info = BlogPost.objects._collection.index_information() | ||||||
|  |         # _id, '-date', ('cat', 'date') | ||||||
|  |         # NB: there is no index on _types by itself, since | ||||||
|  |         # the indices on -date and tags will both contain | ||||||
|  |         # _types as first element in the key | ||||||
|  |         self.assertEqual(len(info), 3) | ||||||
|  |  | ||||||
|  |         # Indexes are lazy so use list() to perform query | ||||||
|  |         list(BlogPost.objects) | ||||||
|  |         info = BlogPost.objects._collection.index_information() | ||||||
|  |         info = [value['key'] for key, value in info.iteritems()] | ||||||
|  |         self.assertTrue([('_types', 1), ('category', 1), ('date', -1)] | ||||||
|  |                         in info) | ||||||
|  |         self.assertTrue([('_types', 1), ('date', -1)] in info) | ||||||
							
								
								
									
										1704
									
								
								tests/fields.py
									
									
									
									
									
								
							
							
						
						
									
										1704
									
								
								tests/fields.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										23
									
								
								tests/fixtures.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								tests/fixtures.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,23 @@ | |||||||
|  | from datetime import datetime | ||||||
|  |  | ||||||
|  | from mongoengine import * | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PickleEmbedded(EmbeddedDocument): | ||||||
|  |     date = DateTimeField(default=datetime.now) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class PickleTest(Document): | ||||||
|  |     number = IntField() | ||||||
|  |     string = StringField(choices=(('One', '1'), ('Two', '2'))) | ||||||
|  |     embedded = EmbeddedDocumentField(PickleEmbedded) | ||||||
|  |     lists = ListField(StringField()) | ||||||
|  |     photo = FileField() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Mixin(object): | ||||||
|  |     name = StringField() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Base(Document): | ||||||
|  |     meta = {'allow_inheritance': True} | ||||||
							
								
								
									
										
											BIN
										
									
								
								tests/mongoengine.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								tests/mongoengine.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 8.1 KiB | 
							
								
								
									
										2952
									
								
								tests/queryset.py
									
									
									
									
									
								
							
							
						
						
									
										2952
									
								
								tests/queryset.py
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										230
									
								
								tests/signals.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										230
									
								
								tests/signals.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,230 @@ | |||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | import unittest | ||||||
|  |  | ||||||
|  | from mongoengine import * | ||||||
|  | from mongoengine import signals | ||||||
|  |  | ||||||
|  | signal_output = [] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class SignalTests(unittest.TestCase): | ||||||
|  |     """ | ||||||
|  |     Testing signals before/after saving and deleting. | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     def get_signal_output(self, fn, *args, **kwargs): | ||||||
|  |         # Flush any existing signal output | ||||||
|  |         global signal_output | ||||||
|  |         signal_output = [] | ||||||
|  |         fn(*args, **kwargs) | ||||||
|  |         return signal_output | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         connect(db='mongoenginetest') | ||||||
|  |         class Author(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |             def __unicode__(self): | ||||||
|  |                 return self.name | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def pre_init(cls, sender, document, *args, **kwargs): | ||||||
|  |                 signal_output.append('pre_init signal, %s' % cls.__name__) | ||||||
|  |                 signal_output.append(str(kwargs['values'])) | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def post_init(cls, sender, document, **kwargs): | ||||||
|  |                 signal_output.append('post_init signal, %s' % document) | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def pre_save(cls, sender, document, **kwargs): | ||||||
|  |                 signal_output.append('pre_save signal, %s' % document) | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def post_save(cls, sender, document, **kwargs): | ||||||
|  |                 signal_output.append('post_save signal, %s' % document) | ||||||
|  |                 if 'created' in kwargs: | ||||||
|  |                     if kwargs['created']: | ||||||
|  |                         signal_output.append('Is created') | ||||||
|  |                     else: | ||||||
|  |                         signal_output.append('Is updated') | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def pre_delete(cls, sender, document, **kwargs): | ||||||
|  |                 signal_output.append('pre_delete signal, %s' % document) | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def post_delete(cls, sender, document, **kwargs): | ||||||
|  |                 signal_output.append('post_delete signal, %s' % document) | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def pre_bulk_insert(cls, sender, documents, **kwargs): | ||||||
|  |                 signal_output.append('pre_bulk_insert signal, %s' % documents) | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def post_bulk_insert(cls, sender, documents, **kwargs): | ||||||
|  |                 signal_output.append('post_bulk_insert signal, %s' % documents) | ||||||
|  |                 if kwargs.get('loaded', False): | ||||||
|  |                     signal_output.append('Is loaded') | ||||||
|  |                 else: | ||||||
|  |                     signal_output.append('Not loaded') | ||||||
|  |         self.Author = Author | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         class Another(Document): | ||||||
|  |             name = StringField() | ||||||
|  |  | ||||||
|  |             def __unicode__(self): | ||||||
|  |                 return self.name | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def pre_init(cls, sender, document, **kwargs): | ||||||
|  |                 signal_output.append('pre_init Another signal, %s' % cls.__name__) | ||||||
|  |                 signal_output.append(str(kwargs['values'])) | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def post_init(cls, sender, document, **kwargs): | ||||||
|  |                 signal_output.append('post_init Another signal, %s' % document) | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def pre_save(cls, sender, document, **kwargs): | ||||||
|  |                 signal_output.append('pre_save Another signal, %s' % document) | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def post_save(cls, sender, document, **kwargs): | ||||||
|  |                 signal_output.append('post_save Another signal, %s' % document) | ||||||
|  |                 if 'created' in kwargs: | ||||||
|  |                     if kwargs['created']: | ||||||
|  |                         signal_output.append('Is created') | ||||||
|  |                     else: | ||||||
|  |                         signal_output.append('Is updated') | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def pre_delete(cls, sender, document, **kwargs): | ||||||
|  |                 signal_output.append('pre_delete Another signal, %s' % document) | ||||||
|  |  | ||||||
|  |             @classmethod | ||||||
|  |             def post_delete(cls, sender, document, **kwargs): | ||||||
|  |                 signal_output.append('post_delete Another signal, %s' % document) | ||||||
|  |  | ||||||
|  |         self.Another = Another | ||||||
|  |         # Save up the number of connected signals so that we can check at the end | ||||||
|  |         # that all the signals we register get properly unregistered | ||||||
|  |         self.pre_signals = ( | ||||||
|  |             len(signals.pre_init.receivers), | ||||||
|  |             len(signals.post_init.receivers), | ||||||
|  |             len(signals.pre_save.receivers), | ||||||
|  |             len(signals.post_save.receivers), | ||||||
|  |             len(signals.pre_delete.receivers), | ||||||
|  |             len(signals.post_delete.receivers), | ||||||
|  |             len(signals.pre_bulk_insert.receivers), | ||||||
|  |             len(signals.post_bulk_insert.receivers), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         signals.pre_init.connect(Author.pre_init, sender=Author) | ||||||
|  |         signals.post_init.connect(Author.post_init, sender=Author) | ||||||
|  |         signals.pre_save.connect(Author.pre_save, sender=Author) | ||||||
|  |         signals.post_save.connect(Author.post_save, sender=Author) | ||||||
|  |         signals.pre_delete.connect(Author.pre_delete, sender=Author) | ||||||
|  |         signals.post_delete.connect(Author.post_delete, sender=Author) | ||||||
|  |         signals.pre_bulk_insert.connect(Author.pre_bulk_insert, sender=Author) | ||||||
|  |         signals.post_bulk_insert.connect(Author.post_bulk_insert, sender=Author) | ||||||
|  |  | ||||||
|  |         signals.pre_init.connect(Another.pre_init, sender=Another) | ||||||
|  |         signals.post_init.connect(Another.post_init, sender=Another) | ||||||
|  |         signals.pre_save.connect(Another.pre_save, sender=Another) | ||||||
|  |         signals.post_save.connect(Another.post_save, sender=Another) | ||||||
|  |         signals.pre_delete.connect(Another.pre_delete, sender=Another) | ||||||
|  |         signals.post_delete.connect(Another.post_delete, sender=Another) | ||||||
|  |  | ||||||
|  |     def tearDown(self): | ||||||
|  |         signals.pre_init.disconnect(self.Author.pre_init) | ||||||
|  |         signals.post_init.disconnect(self.Author.post_init) | ||||||
|  |         signals.post_delete.disconnect(self.Author.post_delete) | ||||||
|  |         signals.pre_delete.disconnect(self.Author.pre_delete) | ||||||
|  |         signals.post_save.disconnect(self.Author.post_save) | ||||||
|  |         signals.pre_save.disconnect(self.Author.pre_save) | ||||||
|  |         signals.pre_bulk_insert.disconnect(self.Author.pre_bulk_insert) | ||||||
|  |         signals.post_bulk_insert.disconnect(self.Author.post_bulk_insert) | ||||||
|  |  | ||||||
|  |         signals.pre_init.disconnect(self.Another.pre_init) | ||||||
|  |         signals.post_init.disconnect(self.Another.post_init) | ||||||
|  |         signals.post_delete.disconnect(self.Another.post_delete) | ||||||
|  |         signals.pre_delete.disconnect(self.Another.pre_delete) | ||||||
|  |         signals.post_save.disconnect(self.Another.post_save) | ||||||
|  |         signals.pre_save.disconnect(self.Another.pre_save) | ||||||
|  |  | ||||||
|  |         # Check that all our signals got disconnected properly. | ||||||
|  |         post_signals = ( | ||||||
|  |             len(signals.pre_init.receivers), | ||||||
|  |             len(signals.post_init.receivers), | ||||||
|  |             len(signals.pre_save.receivers), | ||||||
|  |             len(signals.post_save.receivers), | ||||||
|  |             len(signals.pre_delete.receivers), | ||||||
|  |             len(signals.post_delete.receivers), | ||||||
|  |             len(signals.pre_bulk_insert.receivers), | ||||||
|  |             len(signals.post_bulk_insert.receivers), | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  |         self.assertEqual(self.pre_signals, post_signals) | ||||||
|  |  | ||||||
|  |     def test_model_signals(self): | ||||||
|  |         """ Model saves should throw some signals. """ | ||||||
|  |  | ||||||
|  |         def create_author(): | ||||||
|  |             a1 = self.Author(name='Bill Shakespeare') | ||||||
|  |  | ||||||
|  |         def bulk_create_author_with_load(): | ||||||
|  |             a1 = self.Author(name='Bill Shakespeare') | ||||||
|  |             self.Author.objects.insert([a1], load_bulk=True) | ||||||
|  |  | ||||||
|  |         def bulk_create_author_without_load(): | ||||||
|  |             a1 = self.Author(name='Bill Shakespeare') | ||||||
|  |             self.Author.objects.insert([a1], load_bulk=False) | ||||||
|  |  | ||||||
|  |         self.assertEqual(self.get_signal_output(create_author), [ | ||||||
|  |             "pre_init signal, Author", | ||||||
|  |             "{'name': 'Bill Shakespeare'}", | ||||||
|  |             "post_init signal, Bill Shakespeare", | ||||||
|  |         ]) | ||||||
|  |  | ||||||
|  |         a1 = self.Author(name='Bill Shakespeare') | ||||||
|  |         self.assertEqual(self.get_signal_output(a1.save), [ | ||||||
|  |             "pre_save signal, Bill Shakespeare", | ||||||
|  |             "post_save signal, Bill Shakespeare", | ||||||
|  |             "Is created" | ||||||
|  |         ]) | ||||||
|  |  | ||||||
|  |         a1.reload() | ||||||
|  |         a1.name='William Shakespeare' | ||||||
|  |         self.assertEqual(self.get_signal_output(a1.save), [ | ||||||
|  |             "pre_save signal, William Shakespeare", | ||||||
|  |             "post_save signal, William Shakespeare", | ||||||
|  |             "Is updated" | ||||||
|  |         ]) | ||||||
|  |  | ||||||
|  |         self.assertEqual(self.get_signal_output(a1.delete), [ | ||||||
|  |             'pre_delete signal, William Shakespeare', | ||||||
|  |             'post_delete signal, William Shakespeare', | ||||||
|  |         ]) | ||||||
|  |  | ||||||
|  |         signal_output = self.get_signal_output(bulk_create_author_with_load) | ||||||
|  |  | ||||||
|  |         # 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 | ||||||
|  |         self.assertEquals(signal_output[3], | ||||||
|  |             "pre_bulk_insert signal, [<Author: Bill Shakespeare>]") | ||||||
|  |         self.assertEquals(signal_output[-2:], | ||||||
|  |             ["post_bulk_insert signal, [<Author: Bill Shakespeare>]", | ||||||
|  |              "Is loaded",]) | ||||||
|  |  | ||||||
|  |         self.assertEqual(self.get_signal_output(bulk_create_author_without_load), [ | ||||||
|  |             "pre_init signal, Author", | ||||||
|  |             "{'name': 'Bill Shakespeare'}", | ||||||
|  |             "post_init signal, Bill Shakespeare", | ||||||
|  |             "pre_bulk_insert signal, [<Author: Bill Shakespeare>]", | ||||||
|  |             "post_bulk_insert signal, [<Author: Bill Shakespeare>]", | ||||||
|  |             "Not loaded", | ||||||
|  |         ]) | ||||||
|  |  | ||||||
|  |         self.Author.objects.delete() | ||||||
		Reference in New Issue
	
	Block a user